对于小部件输入本身可能不连贯的自定义小部件,应该如何传播验证错误?例如,我正在为Date字段创建一个自定义日期输入小部件,允许用户根据日本帝国日历选择日期。这需要一个纪元下拉列表和一个年份输入,而且完全可以选择一个本身无效的纪元年组合。小部件使用date /MultiWidget.decompress方法将此输入转换为/从Python MultiWidget.decompress对象:
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并返回一个空值,这意味着字段的验证程序抱怨数据丢失,而不是一个不正确的值。如果我只是简单地raise ValueError或ValidationError,它就会导致一个不明显的异常错误。
这种验证应该在哪里以及如何进行?我希望将日语选择器的抽象完全保留在UI层中,并将后台字段保留为一个简单的Date字段。
发布于 2019-09-17 13:26:02
现在我已经用这种方法解决了这个问题:
widgets.py
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)
)fields.py
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)models.py
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()换句话说,小部件的value_from_datadict执行从自定义输入元素到常规date的转换,或者返回ValidationError的一个实例。这是由一个特殊的表单字段支持的,该字段在其to_python方法中识别这一点,并允许从该方法中对该异常进行raise。为了在模型中使用这个权限,我还添加了一个自定义模型字段,但这是可选的。
不确定这是否是解决这个问题的最优雅的方法;它确实感觉有点不舒服。
https://stackoverflow.com/questions/57969987
复制相似问题