gpt4 book ai didi

django - 更改 Django 表单值

转载 作者:行者123 更新时间:2023-12-04 16:34:37 25 4
gpt4 key购买 nike

我已经更改了模型表单中的 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/

25 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com