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

mli*_*ner 6 django validation date

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

dob_year
dob_month
dob_day
Run Code Online (Sandbox Code Playgroud)

我想尽DateFields我所能获得许多好处.我现在能想到的最重要的是验证.有效:

2010
2010,02
2010,02,28
Run Code Online (Sandbox Code Playgroud)

无效:

2010,02,30
Run Code Online (Sandbox Code Playgroud)

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

<p>{{my_date|date:"%Y"}}</p>
Run Code Online (Sandbox Code Playgroud)

但有了这个,我将不得不做一些非常奇怪的事情,因为我需要它来支持部分日期和常规日期.我想我可能用一种@property方法来完成这个,但我还没有对此进行排序.

我确信还有其他便利我也放弃了(比如admin中的日期小部件).这里的想法也很受欢迎,因为我知道部分日期是一个常见的问题,但我现在最关心的是验证.

更新:Twitter上的一位朋友指出,使用三个字段会产生可怕的日期查询.不要将三个字段用于部分日期,除非您想要考虑如何处理"2011年7月到2012年6月之间"等查询.

dan*_*era 11

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

您可以将所有日期存储在数据库中,即使不是所有日期都需要.例如,即使您只需要日期,Oracle也会将日期和时间存储在日期中.

然后,您可以使用DateField+" ChoiceField"来存储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, )
Run Code Online (Sandbox Code Playgroud)

验证两个字段将由每个自己的窗口小部件验证.您可以添加表单清理(清除和验证彼此依赖的字段)或模型清理验证级别.

Quering使用Q对象轻松进行条件查询:

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 )
Run Code Online (Sandbox Code Playgroud)

渲染显示为了在模板中渲染:

<p>{{item.partial_date_date|date:item.partial_date_part}}</p>
Run Code Online (Sandbox Code Playgroud)

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

  date type: (*) Year    ( ) Month     ( ) Day
  date:      [ change form widget dynamically  ]
Run Code Online (Sandbox Code Playgroud)

您可以发送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)    
Run Code Online (Sandbox Code Playgroud)

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( )
Run Code Online (Sandbox Code Playgroud)

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})
Run Code Online (Sandbox Code Playgroud)

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