带有MultiValueField的FormWizard中的另一个ForeignKey的ForeignKey字段

时间:2011-10-24 15:02:14

标签: python django django-formwizard django-forms

我有一个带有ForeignKey的模型到另一个也有ForeignKey的模型。我使用django表单向导(尝试支持尽可能旧的django版本)渲染它,而不是 inlineformset friendly 。我希望用户在表单向导中输入两个模型和额外信息,并将其转换为正确的pk值(这样,额外的信息可以依赖于组合)我想知道最好的方法来解决这个问题。< / p>

进一步澄清代码。我有三种模式:

class Subject(models.Model):
   title = models.CharField(...)
   extra_info = models.CharField(...)

class Topic(models.Model):
   title = models.CharField(...)
   extra_info = models.CharField(...)
   subject = models.ForeignKey(Subject)

class AwesomeThing(models.Model):
   title = models.CharField(...)
   topic = models.ForeignKey(Topic)

现在,我想在表单向导中向用户显示 AwesomThing 主题字段,作为四个字段:

  • 主题
  • 主题额外信息
  • 主题
  • 主题额外信息

我正在使用带有 MultiWidget MultiValueField 来完成此操作,但我不确定保留值并在表单向导中的步骤之间传递它的最佳方法。我能够做到这一点,但我担心我的方法过于频繁地访问数据库。以下是我目前的做法:

class SubjectTopicField(MultiValueField):
  widget = SubjectTopicInput # Multiwidget to present four input fields
  hidden_widget = HiddenInput

  def __init__(self, *args, **kwargs):
    fields = (
      CharField(label='Subject'),
      CharField(label='Subject extra information'),
      CharField(label='Topics'),
      CharField(label='Topic extra information'),
        )

    super(SubjectTopicField, self).__init__(fields, *args, **kwargs)

  def compress(self, data_list):
    # If all four fields are present ...
    if data_list and len(data_list) == 4:
      # ... call and return the topic 'pk' value from a custom method that 
      # creates and/or gets the topic based on the subject and topic info
      return get_or_create_topic(data_list[0], data_list[1], # Subject
                                 data_list[2], data_list[3]) # Topic

    return None

我的get_or_create_topic方法基本上只查找主题和主题的组合是否存在,如果是,则返回它,如果不存在则创建它。问题是这意味着它必须在每个步骤中访问数据库。我看到该字段的隐藏字段表示只包含值列表(data_list)而不是主题的pk值。在我看来,这不是最优的。有一个更好的方法吗?我可能只是专注于让它与 MultiValueField 一起工作,我没有看到正确的方法来做到这一点。

1 个答案:

答案 0 :(得分:1)

我认为我想出了这个。答案是使用 MultiWidget 子类,但不要使用MultiValueField的子类。我只是将我的主题字段指向继承 MultiWidget 的SubjectTopicInput,然后除了解压缩I子类value_from_datadict以返回pk(我知道我可以让它更漂亮,但这正是我为了得到它而做的工作):

def value_from_datadict(self, data, files, name):
  # Is there just a single value available which we can return?
  single_value = data.get(name, None)
  if single_value: return single_value

  # No single value, let's try to find our topic (or create it)
  super_value = super(SubjectTopicInput, self).value_from_datadict(data, files, name)
  tc = get_or_create_topic(super_value[0], super_value[1],
                           super_value[2], super_value[3])

  if tc: return tc.pk 

  return None

当你停止挖掘并质疑这个洞是否真的有必要时,真的很容易。