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以在选定的零件更改时隐藏/显示日期零件字段.
| 归档时间: |
|
| 查看次数: |
615 次 |
| 最近记录: |