我有一个django模型,有些model.DateTimeFields作为属性。因为用户必须能够单独添加时间和日期,并且因为我希望能够完全自定义在模板中呈现的日期和时间输入,所以我执行了以下操作
class Entry(models.Model):
title = models.CharField(max_length=40)
body = models.CharField(max_length=10000, blank=True)
created = models.DateTimeField(auto_now_add=True)
start_date = models.DateField(default=datetime.now().date())
start_hour = models.TimeField(default=datetime.now().time())
start = models.DateTimeField()
end_hour = models.TimeField(default=(datetime.now()+ timedelta(minutes=45)).time())
end = models.DateTimeField()
表格
class EntryAddForm(ModelForm):
def __init__(self, *args, **kwargs):
super(EntryAddForm, self).__init__(*args, **kwargs)
self.fields['doctor'].queryset = Doctor.objects.all()
class Meta:
model = Entry
exclude = ('canceled','creator', 'reason','start', 'end')
widgets....#setting widgets here
def save(self, commit=True):
model = super(EntryAddForm, self).save(commit=False)
model.start = datetime.combine(self.cleaned_data['start_date'], self.cleaned_data['start_hour'])
model.end = datetime.combine(self.cleaned_data['start_date'], self.cleaned_data['end_hour'])
if commit:
model.save()
return model
因此,在save方法中,我创建了一个新的日期时间对象,其中包含开始日期和开始时间
的组合问题是当您将它们呈现在表单上进行编辑时。因为输入日期存储在utc start_hour中,而end_hour不是从我从表单中保存它们但落后3小时的确切时间(因为我住在希腊,我的设置文件中有TIME_ZONE = Athens)。
是否真的有一种简单的方法可以解决这个问题,或者解决方案是不使用感知对象?
还有一个小题目。 Javascript如何知道时区偏移量。我做了
x = new Date();
x.getTimeZoneOffset() / 60;
我得到-3这是正确的。如果我在我的Ubuntu系统中更改我的时区,我的ubuntu时钟会更改以匹配所选时区的时间,但上面的javascript代码会返回相同的结果。那么javascript不会使用系统时区吗?
答案 0 :(得分:2)
它不会直接解决您的时区问题,但您不应该在模型中执行此操作:
start_date = models.DateField(default=datetime.now().date())
start_hour = models.TimeField(default=datetime.now().time())
start = models.DateTimeField()
为了更改用户界面,您已使用冗余字段污染了数据模型 - 进行此更改的正确位置位于关联的ModelForm(和模板)中
Django正是为了这个目的而提供https://docs.djangoproject.com/en/dev/ref/forms/widgets/#splitdatetimewidget。
对于你的时区问题...如果你在设置中有USE_TZ = True
,那么Django将假设来自用户输入的所有日期和时间都在TIME_ZONE = 'Athens'
的每个设置中...然后它将转换这些在保存到数据库之前到UTC(并在检索时将它们转换回雅典时间)
https://docs.djangoproject.com/en/dev/ref/settings/#std:setting-USE_TZ
如果您需要处理用户在各种时区提交的数据时间信息,可能更容易设置USE_TZ = False
...您的日期时间将不再有时区信息,因此您应将所有内容规范化为在Django应用程序之外的UTC(即在javascript中)
我不确定您对getTimeZoneOffset
的问题的答案 - 您可能需要重新启动浏览器才能获得系统时区的更改吗?