我正在尝试在我的数据库中存储通常缺少月和日的日期。我目前的方法(似乎有很多不同的方法)是使用三个字段:
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;。
答案 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,以便在选定的部件更改时隐藏/显示日期部件字段。