gpt4 book ai didi

django - Django 中部分日期的日期便利(验证、显示等)

转载 作者:行者123 更新时间:2023-12-04 13:49:20 28 4
gpt4 key购买 nike

我试图在我的数据库中存储通常缺少月份和日期的日期。我目前的方法(似乎有很多不同的方法可以做到这一点)为此使用三个字段:

dob_year
dob_month
dob_day

我想获得 DateFields 的所有好处我可以。我现在能想到的最重要的一个是验证。有效的:
2010
2010,02
2010,02,28

无效的:
2010,02,30

还有在模板中将整数转换为人类可读形式的问题。很高兴能够说:
<p>{{my_date|date:"%Y"}}</p>

但是有了这个,我将不得不做一些非常奇怪的事情,因为我需要它来支持部分日期和常规日期。我想我可以用 @property 来完成这个任务。方法,但我也还没有解决这个问题。

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

更新 : Twitter 上的一位 friend 指出,为此使用三个字段会创建 horrible date queries .不要对部分日期使用三个字段,除非您想考虑如何处理诸如“2011 年 7 月和 2012 年 6 月之间”之类的查询。

最佳答案

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

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

然后你可以使用 DateField + "ChoiceField "存储部分日期和相关部分信息。例如: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验证级别。

查询 使用 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 个小部件控件以形成并一次只显示一个。在更改 radio 日期类型选择时更改可见性。我用 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 以在选定部分更改时隐藏/显示日期部分字段。

关于django - Django 中部分日期的日期便利(验证、显示等),我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/30134526/

28 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com