16

所以我的 django 书又回到了大学,我正在努力解决这个问题。

我已经django.forms.widgets.MultiWidget像这样子类化了:

class DateSelectorWidget(widgets.MultiWidget):
    def __init__(self, attrs=None, dt=None, mode=0):  
        if dt is not None:
            self.datepos = dt
        else:
            self.datepos = date.today()    

        # bits of python to create days, months, years
        # example below, the rest snipped for neatness.

        years = [(year, year) for year in year_digits]

        _widgets = (
            widgets.Select(attrs=attrs, choices=days), 
            widgets.Select(attrs=attrs, choices=months),
            widgets.Select(attrs=attrs, choices=years),
            )
        super(DateSelectorWidget, self).__init__(_widgets, attrs)

    def decompress(self, value):
        if value:
            return [value.day, value.month, value.year]
        return [None, None, None]

    def format_output(self, rendered_widgets):
        return u''.join(rendered_widgets)

这给了我一个漂亮的日期选择字段,如下所示: 漂亮的日期选择

我的问题真的很简单。当我将所述表单提交给它的处理方法时(它使用这样的过程:

forminstance = ModelNameForm(request.POST, instance=modelinstance)
    if forminstance.is_valid():
        forminstance.save()

这失败了,因为 Django 不知道如何获取我的 multi-widget 并将其转换回基础字段类型,该字段类型设置models.pyDateField(),显然。

现在,django 源代码中围绕 MultiWidget 的注释给了我这个有用的提示:

您可能希望将此类与 MultiValueField 一起使用。

但问题是——我可能不会。我想保留我的DateField(),因为它非常有用,而且没有必要复制它。然后我需要做的是以某种方式将这些多个字段转换回单个有效的日期字符串 ( yyyy-mm-dd) 以插入到数据库中。

我的问题是:

如何?实现这一目标的最佳方法是什么?

4

1 回答 1

12

回答了我自己的问题!

我实现了这个方法:

def value_from_datadict(self, data, files, name):
    datelist = [widget.value_from_datadict(data, files, name + '_%s' % i) \ 
                                      for i, widget in enumerate(self.widgets)]
    try:
        D = date(day=int(datelist[0]), month=int(datelist[1]), \
             year=int(datelist[2]))
        return str(D)
    except ValueError:
        return ""

value_from_datadict 从整个 post datadict 中提取所有子小部件的数据。我所做的是提取出各种日期对应项,并使用日期构造函数来验证日期。如果有效,我们以正确的格式打印字符串,否则,我们返回一个空字符串

forminstance.is_valid()

会抓住的。

我喜欢这样做!

于 2011-01-11T23:38:36.497 回答