Django:从Form子类中删除一个字段

时间:2013-03-21 20:19:53

标签: python django forms inheritance field

class LoginForm(forms.Form):
    nickname = forms.CharField(max_length=100)
    username = forms.CharField(max_length=100)
    password = forms.CharField(widget=forms.PasswordInput)


class LoginFormWithoutNickname(LoginForm):
    # i don't want the field nickname here
    nickname = None #??

有没有办法实现这一目标?

注意:我没有ModelForm,因此带有Meta的{​​{1}}课程不起作用。

4 个答案:

答案 0 :(得分:63)

您可以通过覆盖 init 方法来更改子类中的字段:

class LoginFormWithoutNickname(LoginForm):
    def __init__(self, *args, **kwargs):
        super(LoginFormWithoutNickname, self).__init__(*args, **kwargs)
        self.fields.pop('nickname')

答案 1 :(得分:9)

Django 1.7在提交b16dd1fe019中为票证#8620解决了这个问题。在Django 1.7中,OP可以在子类中执行nickname = None。从提交中的文档更改:

  

可以通过遮蔽从父类继承的Field退出。虽然任何非Field值都可用于此目的,但建议使用None明确表示字段无效。

答案 2 :(得分:4)

我发现了,如果有兴趣请发表评论。

(在Django 1.7.4中) 扩展表单,使用以下代码:

class MyForm(forms.ModelForm):

    def __init__(self, *args, **kwargs):
        super(MyForm, self).__init__(*args, **kwargs)

        for key, field in self.fields.iteritems():
            self.fields[key].required = False

    class Meta:
        model = MyModel
        exclude = []

    field_1 = forms.CharField(label="field_1_label")
    field_2 = forms.CharField(label="field_2_label", widget=forms.Textarea(attrs={'class': 'width100 h4em'}),)
    field_3 = forms.CharField(label="field_3_label", widget=forms.TextInput(attrs={'class': 'width100'}),)
    field_4 = forms.ModelChoiceField(label='field_4_label', queryset=AnotherModel.objects.all().order_by("order") )

class MyForm_Extended_1(MyForm):
    field_1 = None


class MyForm_Extended_2(MyForm):
    class Meta:
        model = MyModel
        exclude =[
                    'field_1',
                ]

MyForm_Extended_1将field_1设置为None,(db中的列更新为Null)

MyForm_Extended_2忽略该字段(在保存期间忽略db中的列)

所以,为了我的目的,我使用第二种方法。

答案 3 :(得分:2)

我不喜欢事实(或者我理解)使用“类Meta:”排除第二类中的字段仍会导致未使用的字段位于数据库中。

也许最简单的方法是定义一个抽象类,它具有两个类共享的字段。然后上面的两个原始类成为这个新类的子类。因此,在此线程开头给出的示例可能如下所示。这是更多的代码,但这样你扩展了一个子类,而不是从超类中进行(不完整的)排除。

class LoginForm_Common(forms.Form):
    username = forms.CharField(max_length=100)
    password = forms.CharField(widget=forms.PasswordInput)
    class Meta:
        abstract = True

class LoginForm(LoginForm_Common):
    nickname = forms.CharField(max_length=100)

class LoginFormWithoutNickname(LoginForm_Common):
    pass