最近对django真是大爱呀,哈哈。公司有个需求是为一些素材动态生成属性字段,所以需要django来动态生成表单。
一开始的写法如下:
class AddObjForm(forms.Form):
id = forms.CharField()
def __init__(self, *args, **kwargs):
mdfields = copy.deepcopy(kwargs['mdfields'])
del kwargs['mdfields']
super(AddObjForm, self).__init__(*args, **kwargs)
if mdfields is not None:
for f in mdfields:
exestr = ""
if f.showtype == 0:
exestr = 'self.'+f.name+"=forms.CharField()"
else:
exestr = 'self.'+f.name+"=forms.TextField()"
exec(exestr)
但是发现mdfields里面的字段并没有成功生成表单项,而是只有一个ID项。
于是研究了一下form的源码,其构造函数的源码如下:
class BaseForm(StrAndUnicode):
def __init__(self, data=None, files=None, auto_id='id_%s', prefix=None,
initial=None, error_class=ErrorList, label_suffix=':',
empty_permitted=False):
....
self.fields = deepcopy(self.base_fields)
而其在生成html表单的时候,也都是从fields变量生成,所以这样试了一下:
class AddObjForm(forms.Form):
id = forms.CharField()
def __init__(self, *args, **kwargs):
mdfields = copy.deepcopy(kwargs['mdfields'])
del kwargs['mdfields']
super(AddObjForm, self).__init__(*args, **kwargs)
if mdfields is not None:
for f in mdfields:
exestr = ""
if f.showtype == 0:
self.fields['name'] = forms.CharField()
else:
self.fields['name'] = forms.TextField()
果然就OK了,注意super一定要在我们后面的赋值之前,因为否则fields还没有生成……
OK,成果图如下:
PS:
一直使用C++编程,python带给我的是另一种完全不同的体验。
1.你要随时的利用python的动态性,要记住连代码也是可以动态生成
2.库的代码是可见的,你可以认为他封装的已经足够好,你也可以随时去看它的代码实现。(而不是去用man或者msdn)
### on #
<img src="http://img.t.sinajs.cn/t35/style/images/common/face/ext/normal/ec/zs_org.gif" />
Reply