django - 更改 Django 表单值

标签 django django-forms

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

相关文章:

python - 显示基于另一个表数据的表单元素

Django 更新 m2m 字段

javascript - 如何将 Django 表单与 Ajax 集成?

python - 如何从 Django 网站启动 Python 脚本?

Django 管理.py : Is it possible to pass command line argument (for unit testing)

javascript - 在 django 和 javascript 之间传递日期时间时保持微秒精度

Django 表格 : Validation message not showing

ios - Django 休息框架日期时间字段格式

python - 在 Django 模板中的元组列表的第一个元素中访问元组的第三项

django - 如何在 Django 中获取表单字段的 ID