我已经更改了模型表单中的 ForeignKey,改为使用 TextBox。 然后我重写 clean 方法以根据名称字段(而不是 id 字段)返回对象
类 SongForm(forms.ModelForm):
艺术家 = forms.CharField(widget=forms.TextInput())
def clean_artist(self):
data = self.cleaned_data['artist']
artist = Artist.objects.get(name=data)
self.cleaned_data['artist_id'] = artist.id
return artist
class Meta:
model = Song
它正确地保存了表单,但是当它再次呈现时出现的是 id 值而不是 name 值。如何更改 Django 表单的显示值?我认为重写 init 会做到这一点,但找不到 value 属性在哪里
最佳答案
我刚刚编写了 Field 和 Widget 子类,它们解决了这个特殊问题并且可以与 JS 自动完成一起使用,例如 - 并且是可重用的。不过,它比您的解决方案需要更多的工作,我不确定您是否愿意使用我的解决方案。不管怎样——我希望我能得到很少的赞成票——我花了很多时间和精力写这篇文章……
与其像您那样定义您的 ModelForm 并弄乱 clean_
,不如我建议这样:
class SongForm(forms.ModelForm):
artist = CustomModelChoiceField( queryset = Artist.objects.all(), query_field = "name" )
class Meta:
model = Song
现在,CustomModelChoiceField(我想不出更好的类名)是 ModelChoiceField 的子类,这很好,因为我们可以使用 queryset
参数来缩小可接受的选择范围。如果 widget
参数不存在,如上,则使用该字段的默认参数(稍后详细介绍)。 query_field
是可选的,默认为 "pk"
。所以,这是域代码:
class CustomModelChoiceField( forms.ModelChoiceField ):
def __init__( self, queryset, query_field = "pk", **kwargs ):
if "widget" not in kwargs:
kwargs["widget"] = ModelTextInput( model_class = queryset.model, query_field = query_field )
super( CustomModelChoiceField, self ).__init__( queryset, **kwargs )
def to_python( self, value ):
try:
int(value)
except:
from django.core.exceptions import ValidationError
raise ValidationError(self.error_messages['invalid_choice'])
return super( CustomModelChoiceField, self ).to_python( value )
__init__
的意思是在创建 CustomModelChoiceField
期间设置 widget = None
给我们简单的 ModelChoiceField
(这在调试时非常有帮助......)。现在,实际工作在 ModelTextInput
小部件中完成:
class ModelTextInput( forms.TextInput ):
def __init__( self, model_class, query_field, attrs = None ):
self.model_class = model_class
self.query_field = query_field
super( ModelTextInput, self ).__init__( attrs )
def render(self, name, value, attrs = None ):
try:
obj = self.model_class.objects.get( pk = value )
value = getattr( obj, self.query_field )
except:
pass
return super(ModelTextInput, self).render( name, value, attrs )
def value_from_datadict( self, data, files, name ):
try:
return self.model_class.objects.get( **{ self.query_field : data[name] } ).id
except:
return data[name]
它本质上是 TextInput,它知道另外两件事 - 它代表哪个模型的哪个属性。 ( model_class
应该替换为 queryset
以缩小实际工作的可能选择范围,我稍后会修复它)。查看 value_from_datadict
的实现很容易发现为什么必须覆盖字段中的 to_python
- 它需要 int
值,但不检查是否这是真的 - 只是将值传递给关联的模型,该模型因丑陋的异常而失败。
我对此进行了一段时间的测试并且它有效——您可以指定不同的模型字段,表单字段将通过这些字段尝试找到您的 artist
,表单错误处理由基类自动完成,而您不需要每次您想使用类似的功能时,都不需要编写自定义的 clean_
方法。
我现在太累了,但明天我会尝试编辑这篇文章(和代码)。
关于django - 更改 Django 表单值,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/3321399/