我正在尝试为ModelForm创建自定义字段。我从ModelMultipleChoiceField扩展然后覆盖render和render_options,但是,当我只是尝试导入我的表单时,我一直得到这个异常:
AttributeError: 'ModelMultipleChoiceField' object has no attribute 'to_field_name'
我不确定我错过了什么。我甚至尝试过为我的新类添加一个to_field_name属性,但这没有用。这是我的代码:
class MultiSelect(ModelMultipleChoiceField):
def __init__(self, queryset, cache_choices=False, required=True,
widget=None, label=None, initial=None, help_text=None, *args, **kwargs):
super(MultiSelect, self).__init__(queryset, cache_choices, required, widget,
label, initial, help_text, *args, **kwargs)
def render_options(self, name, choices, selected_choices):
output = []
i = 0
for option_value, option_label in chain(self.choices, choices):
checked_html = (option_value in selected_choices) and u' checked="checked"' or ''
class_html = (i % 2 == 0) and u'even' or u'odd'
output.append('<li class="{0}"><input type="checkbox" name="{1}" value="{2}"{3}/>{4}</li>'
.format(class_html, name, escape(option_value), checked_html, escape(option_label)))
i += 1
def render(self, name, value, attrs=None, choices=()):
if value is None: value = []
final_attrs = self.build_attrs(attrs, name=name)
output = [u'<ul class="multiSelect">']
options = self.render_options(name, choices, value)
if options:
output.append(options)
output.append('</ul>')
return mark_safe(u'\n'.join(output))
class RoleForm(ModelForm):
class Meta:
model = Role
exclude = ('user_id',)
widgets = {
'permissions': MultiSelect(queryset=Permission.objects.all())
}
每当我只做from myapp.forms import RoleForm
时,我就会收到上述错误。
我应该在课堂上添加一些我缺少的东西吗?
答案 0 :(得分:16)
你似乎在字段和小部件之间感到困惑。您继承自ModelMultipleChoiceField
,(顾名思义)是一个字段,而不是一个小部件。但是render
和render_options
是小部件上的方法,而不是字段。你已经在widgets
字典中使用了你的课程。
我怀疑你的意思是创建一个小部件。您应该从widget类继承,可能是forms.CheckboxSelectMultiple
。
答案 1 :(得分:2)
根据您发布的代码不确定为什么会出现问题,但to_field_name
是ModelChoiceField
上的属性:
class ModelChoiceField(ChoiceField):
...
def __init__(self, queryset, empty_label=u"---------", cache_choices=False,
required=True, widget=None, label=None, initial=None,
help_text=None, to_field_name=None, *args, **kwargs):
...
self.to_field_name = to_field_name
但是,当ModelMultipleChoiceField
子类ModelChoiceField
时,它的__init__
方法不接受to_field_name
作为关键字参数。它显然依赖ModelChoiceField
设置self.to_field_name
默认值None
的默认行为。
你的子类应该做同样的事情,因此这部分令人困惑。