问题 Django ModelForm上的ForeignKey字段的自由格式输入


我有两个与外键相关的模型:

# models.py    
class TestSource(models.Model):
  name        = models.CharField(max_length=100)

class TestModel(models.Model):
  name        = models.CharField(max_length=100)
  attribution = models.ForeignKey(TestSource, null=True)

默认情况下,django ModelForm会将其显示为 <select> 同 <option>S;但我更喜欢这个函数作为自由格式输入, <input type="text"/>,并在幕后获取或创建必要的TestSource对象,然后将其与TestModel对象相关联。

我试图定义一个自定义ModelForm和Field来完成这个:

# forms.py
class TestField(forms.TextInput):
  def to_python(self, value):
    return TestSource.objects.get_or_create(name=value)

class TestForm(ModelForm):
  class Meta:
    model=TestModel
    widgets = {
      'attribution' : TestField(attrs={'maxlength':'100'}),
    }

不幸的是,我得到了: invalid literal for int() with base 10: 'test3' 当试图检查 is_valid 在提交的表格上。我哪里错了?是他们更容易实现这一目标的方法吗?


4564
2017-11-09 18:34


起源



答案:


像这样的东西应该工作:

class TestForm(ModelForm):
  attribution = forms.CharField(max_length=100)

  def save(self, commit=True):
      attribution_name = self.cleaned_data['attribution']
      attribution = TestSource.objects.get_or_create(name=attribution_name)[0]  # returns (instance, <created?-boolean>)
      self.instance.attribution = attribution

      return super(TestForm, self).save(commit)

  class Meta:
    model=TestModel
    exclude = ('attribution')

11
2017-11-09 19:23



这有很大帮助。但是我仍然遇到一些问题,视图中是否需要任何非标准代码?我的is_valid()检查仍然爆炸:无法分配“u'myString”:“TestModel.attribution”必须是“TestSource”实例。 - brown145
@ brown145:我认为你必须排除归属,否则它会尝试验证它。看到我更新的答案。 - Sam Dolan


答案:


像这样的东西应该工作:

class TestForm(ModelForm):
  attribution = forms.CharField(max_length=100)

  def save(self, commit=True):
      attribution_name = self.cleaned_data['attribution']
      attribution = TestSource.objects.get_or_create(name=attribution_name)[0]  # returns (instance, <created?-boolean>)
      self.instance.attribution = attribution

      return super(TestForm, self).save(commit)

  class Meta:
    model=TestModel
    exclude = ('attribution')

11
2017-11-09 19:23



这有很大帮助。但是我仍然遇到一些问题,视图中是否需要任何非标准代码?我的is_valid()检查仍然爆炸:无法分配“u'myString”:“TestModel.attribution”必须是“TestSource”实例。 - brown145
@ brown145:我认为你必须排除归属,否则它会尝试验证它。看到我更新的答案。 - Sam Dolan


这里有一些问题。

首先,您已经定义了一个字段,而不是一个小部件,因此您无法在其中使用它 widgets 字典。您需要覆盖表单顶层的字段声明。

其次 get_or_create 回报  values:检索或创建的对象,以及显示是否已创建的布尔值。你真的只想从你的第一个返回这些值 to_python 方法。

我不确定其中任何一个是否会导致您的实际错误。您需要为我们发布实际回溯以确定。


1
2017-11-09 19:23





TestForm.attribution需要int值 - TestSource模型的关键。

也许这个版本的模型对您来说会更方便:

class TestSource(models.Model):
    name = models.CharField(max_length=100, primary_key=True)

0
2017-11-19 03:50