我有一个带有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 一起工作,我没有看到正确的方法来做到这一点。
答案 0 :(得分:1)
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
当你停止挖掘并质疑这个洞是否真的有必要时,真的很容易。