Django中部分日期的日期便利性(验证,显示等)

时间:2015-05-08 23:58:39

标签: django validation date

我正在尝试在我的数据库中存储通常缺少月和日的日期。我目前的方法(似乎有很多不同的方法)是使用三个字段:

dob_year
dob_month
dob_day

我希望尽可能多地获得DateFields的好处。我现在能想到的最重要的是验证。有效的:

2010
2010,02
2010,02,28

无效:

2010,02,30

还有在模板中将int转换为人类可读形式的问题。能够说:

非常棒
<p>{{my_date|date:"%Y"}}</p>

但有了这个,我必须做一些非常奇怪的事情,因为我需要它来支持部分日期和常规日期。我想我可以使用@property方法完成此操作,但我还没有对此进行整理。

我确信还有其他便利我也放弃了(比如管理员中的日期小部件)。这里的想法也很受欢迎,因为我知道部分日期是一个常见的问题,但我目前主要关注验证。

更新:Twitter上的一位朋友指出,使用三个字段会创建horrible date queries。不要将三个字段用于部分日期,除非您想要考虑如何处理2011年7月到2012年6月之间的问题&#34;。

1 个答案:

答案 0 :(得分:11)

将日期作为文本存储在数据库中是一种不好的做法。它不是可移植的,不是django查询api友好,而不是索引友好。

您可以将所有日期存储在数据库中,即使不是所有日期都需要。例如,Oracle stores date and time in dates即使您只需要日期。

然后你可以使用DateField +&#34; ChoiceField&#34;存储partialdate和相关的零件信息。例如:2015-02-01在月份截断。 2015-02-28完整日期。 2015-01-01在年级截断。代码:

class Item(models.Model):
    PARTIAL_YEAR='%Y'
    PARTIAL_MONTH='%Y-%m'
    PARTIAL_DAY='%Y-%m-%d'
    PARTIAL_CHOICES = (
      (PARTIAL_YEAR, 'Year'),
      (PARTIAL_MONTH, 'Month'),
      (PARTIAL_DAY, 'Day'),
    )
    partial_date_date = models.DateField()
    partial_date_part = models.CharField('Date part', 
                                          choices=PARTIAL_CHOICES,  
                                          max_length=10, )

验证两个字段都将由每个自己的窗口小部件验证。您可以添加表单干净(Cleaning and validating fields that depend on each other)或model clean验证级别。

Quering 使用Q objects

轻松进行条件查询
q_is_year   = q( partial_date_part = Item.PARTIAL_YEAR )
q_by_year   = q( partial_date_date__year = 2015 )

q_is_month  = q( partial_date_part = Item.PARTIAL_MONTH )
q_by_month  = q( partial_date_date__year = 2105 ) 
q_by_month &= q( partial_date_date__month = 2 ) 

qs = Item.objects.filter( q_is_year&q_by_year | q_is_month&q_by_month )

渲染显示要在模板中渲染:

<p>{{item.partial_date_date|date:item.partial_date_part}}</p>

呈现表单要呈现表单控件,您可以使用JavaScript更改UI并帮助用户输入数据:

  date type: (*) Year    ( ) Month     ( ) Day
  date:      [ change form widget dynamically  ]

您可以发送3个小部件控件来形成并一次只显示一个。更改无线电日期类型选择的可见性。我使用MultiWidget

已编辑 2015年8月13日,所有示例代码:

  

models.py

from django.db import models
from datetime import date

class Item(models.Model):
    PARTIAL_YEAR='%Y'
    PARTIAL_MONTH='%Y-%m'
    PARTIAL_DAY='%Y-%m-%d'
    PARTIAL_CHOICES = (
      (PARTIAL_YEAR, 'Year'),
      (PARTIAL_MONTH, 'Month'),
      (PARTIAL_DAY, 'Day'),
    )
    partial_date_part = models.CharField('Date part', 
                                          choices=PARTIAL_CHOICES,  
                                          max_length=10, )
    partial_date_date = models.DateField()
    some_comment = models.CharField('Comment', max_length=100, )

    def save(self, *args, **kwargs):
        if self.partial_date_part==self.PARTIAL_YEAR:
            self.partial_date_date = date( self.partial_date_date.year, 1, 1 )
        elif self.partial_date_part==self.PARTIAL_MONTH:
            self.partial_date_date = date( self.partial_date_date.year, 
                                          self.partial_date_date.month, 1 )  
        super(Item, self).save(*args, **kwargs)    
  

forms.py

from django import forms
from django.forms import widgets
from datetime import date

class DateSelectorWidget(widgets.MultiWidget):
    def __init__(self, attrs=None):
        days = [(d, d) for d in range(1,32)]
        months = [(m, m) for m in range(1,13)]
        years = [(year, year) for year in (2011, 2012, 2013)]
        _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 ''.join(rendered_widgets)

    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)]
        D = date(
                day=int(datelist[0]),month=int(datelist[1]),year=int(datelist[2]),
            )
        return D

class ItemForm(forms.Form):
    partial_date_part = forms.CharField(widget=forms.RadioSelect)
    partial_date_date = DateSelectorWidget( )
  

view.py

from django.http import HttpResponseRedirect
from django.views.generic import View
from models import Item
from django.forms.models import modelform_factory
from .forms import DateSelectorWidget
from django import forms
from django.forms import widgets    

class MyDatAppView(View):
    form_class = modelform_factory(Item ,
                                   exclude=[], 
                                   widgets={ 'partial_date_date': 
                                              DateSelectorWidget() ,})
    initial = {'some_comment': '-*-', }
    template_name = 'form.html'

    def get(self, request, *args, **kwargs):
        form = self.form_class(initial=self.initial)
        return render(request, self.template_name, {'form': form})

    def post(self, request, *args, **kwargs):
        form = self.form_class(request.POST)
        if form.is_valid():
            m=form.save()
            return HttpResponseRedirect('/')

        return render(request, self.template_name, {'form': form})

您应该在模板上添加javascript,以便在选定的部件更改时隐藏/显示日期部件字段。