1. 问题描述
在模型(Model)中定义了字段为DecimalField,并且按照规范设置了decimal_places和max_digits属性。
class TeachWorkload(Model):
teach_type = models.IntegerField(
verbose_name='工作类型',
choices=TeachType.choices)
quantity = models.DecimalField(
verbose_name='数量',
decimal_places=2,
max_digits=5)
在表单(Form)中,直接使用模型表单(ModelForm)生成字段。但因为HTML使用了BS4的一个后台管理模板,所以要为每个生成的input添加css的类名。为了方便,重写了初始化方法,直接往widget里面设值。
class TeachworkloadForm(ModelForm):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) # 初始化父类方法
for field in self.fields.values():
field.widget.attrs = {'class': 'form-control'}
class Meta:
model = TeachWorkload
fields = '__all__'
在页面实际的运行中,数据验证的时候把该字段作为整型来处理了。
?翻遍了全部的资料,都没有找出原因。网上基本都是说设置decimal_places和max_digits属就可以了。
2.问题的原因
找一个原生使用{{ form.as_p }}呈现的页面查看,发现渲染出来完整的input标记是带有step等额外属性(attribute)的。而自己重写设置的就丢失了这些内容。
<input type="number" name="quantity" step="0.01" min="0.00" max="999.99" required id="id_quantity">
3.问题的解决方案
用来一个比较笨的办法,直接在初始化函数中,判断字段类型是否DecimalField,如果是就补充加上相关的数据校验的内容。
class TeachworkloadForm(ModelForm):
def __init__(self, *args, **kwargs):
super().__init__(*args, **kwargs) # 初始化父类方法
for field in self.fields.values():
# 判断字段类型属于DecimalField时,增加step等attribute
if type(field) is django.forms.fields.DecimalField:
field.widget.attrs = {'class': 'form-control', 'step': '0.01', 'min': '0.00', 'max': '999.99'}
else:
field.widget.attrs = {'class': 'form-control'}
class Meta:
model = TeachWorkload
fields = '__all__'
4.结语
一种头疼医头的急救解决方案,不知有没有其他的一些方法能更准确快捷解决这个问题。有空再去研究Django的数据校验机制吧。
|