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将此输入转换为Python
date
对象,或将其转换为Python
date
对象。解压缩方法:

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()