python django和日期时间感知对象

时间:2014-05-06 12:44:46

标签: python django datetime django-timezone

我有一个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不会使用系统时区吗?

1 个答案:

答案 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的问题的答案 - 您可能需要重新启动浏览器才能获得系统时区的更改吗?