Python 自定义小部件的验证
当小部件输入本身可能不一致时,应该如何传播自定义小部件的验证错误?举个例子,我正在为一个Python 自定义小部件的验证,python,django,Python,Django,当小部件输入本身可能不一致时,应该如何传播自定义小部件的验证错误?举个例子,我正在为一个date字段创建一个自定义日期输入小部件,允许用户根据日本帝国日历选择日期。这需要一个era下拉列表和一个年份输入,完全可以选择一个本身无效的era-年份组合。小部件使用来自多小部件的MultiWidget.value\u将此输入转换为Pythondate对象,或将其转换为Pythondate对象。解压缩方法: def value_from_datadict(self, data, files, name):
date
字段创建一个自定义日期输入小部件,允许用户根据日本帝国日历选择日期。这需要一个era下拉列表和一个年份输入,完全可以选择一个本身无效的era-年份组合。小部件使用来自多小部件的MultiWidget.value\u将此输入转换为Pythondate
对象,或将其转换为Pythondate
对象。解压缩方法:
def value_from_datadict(self, data, files, name):
era, imperial_year, month, day = [widget.value_from_datadict(data, files, f'{name}_{i}')
for i, widget in enumerate(self.widgets)]
try:
return date(self._j2w.convert(f'{era}{imperial_year}年'), int(month), int(day))
except ValueError:
# selected era/year combination was invalid
return ''
在这个方法中,我所能做的就是捕获任何ValueError
,并返回一个空值,这意味着字段的验证器会抱怨缺少数据,而不是错误的值。如果我只是引发ValueError
或ValidationError
,则会导致未捕获的异常错误
这种验证应该在哪里以及如何进行?我希望将日文选择器的抽象完全保留在UI层中,并将backing字段保留为一个简单的Date
字段。现在,我已经这样解决了它:
widgets.py
fields.py
models.py
换句话说,小部件的值\u from_datadict
将自定义输入元素转换为常规的日期,或者返回验证错误的实例。这是由一个特殊的表单字段支持的,该字段在其to_python
方法中识别这一点,从该方法可以引发该异常。为了将其直接烘焙到模型中,我还添加了一个自定义模型字段,但这是可选的
不确定这是否是解决这个问题的最优雅的方法;它确实有点恶心。现在我已经这样解决了:
widgets.py
fields.py
models.py
换句话说,小部件的值\u from_datadict
将自定义输入元素转换为常规的日期,或者返回验证错误的实例。这是由一个特殊的表单字段支持的,该字段在其to_python
方法中识别这一点,从该方法可以引发该异常。为了将其直接烘焙到模型中,我还添加了一个自定义模型字段,但这是可选的
不确定这是否是解决这个问题的最优雅的方法;确实有点恶心
from django.core.exceptions import ValidationError
from django.forms import MultiWidget
class ImperialDateInput(MultiWidget):
def value_from_datadict(self, data, files, name):
try:
...
except ValueError:
return ValidationError(
_('Invalid date: %(era)s%(year)s/%(month)s/%(day)s'),
code='invalid',
params=dict(era=era, year=imperial_year, month=month, day=day)
)
from django.core.exceptions import ValidationError
from django.forms import DateField
from .widgets import ImperialDateInput
class ImperialDateField(DateField):
widget = ImperialDateInput
def to_python(self, value):
if isinstance(value, ValidationError):
raise value
return super().to_python(value)
from django.db import models
from . import fields
class ImperialDateField(models.DateField):
def formfield(self, **kwargs):
return super().formfield(form_class=fields.ImperialDateField, **kwargs)
class MyModel(models.Model):
date = ImperialDateField()