DurationField
Django で、期間を秒数で表す整数を格納するカスタムを実装しました。また、5 つの異なる数値入力 (週、日、時間、分、および秒を表す) を受け取るカスタム マルチウィジェットも定義しました。
私はそれを管理者で機能させることがdecompress
でき、期待どおりに機能しているようです。インタープリターで値を変更して保存すると、5 つのフィールドに正しく表示されます。ただし、compress
呼び出されていないようです。値を変更しようとすると (または値を変更せずにフォームを保存しようとすると)Enter a whole number.
からエラーが発生しIntegerField
ます。[u'0', u'1', u'2', u'0', u'0']
値のリスト全体 (例: ) が に渡されているように見えますto_python
。
何か小さなものを見落としている/見落としているように感じますが、コードを長時間見つめすぎて、それが何であるかを理解できなかったと思います。
これが私のものmodel_fields.py
です:
from datetime import timedelta
from django.db.models import PositiveIntegerField
from django.forms import IntegerField, MultiWidget
from django.forms.fields import MultiValueField
from django.forms.widgets import TextInput # NumberInput in Django 1.6
DURATION_FORM_FIELDS = ('weeks', 'days', 'hours', 'minutes', 'seconds')
class DurationMultiWidget(MultiWidget):
def __init__(self, attrs=None):
_widgets = tuple(
[TextInput(attrs=attrs) for field in DURATION_FORM_FIELDS]
)
super(DurationMultiWidget, self).__init__(_widgets, attrs)
def decompress(self, value):
if value:
td = timedelta(seconds=value)
return [
getattr(td, label, 0) for label in DURATION_FORM_FIELDS
]
return [None for field in DURATION_FORM_FIELDS]
class DurationMultiValueField(MultiValueField):
def __init__(self, *args, **kwargs):
fields = tuple(
[IntegerField(label=field_label) for field_label in DURATION_FORM_FIELDS]
)
super(DurationMultiValueField, self).__init__(fields=fields, *args, **kwargs)
def compress(self, data_list):
duration_dict = dict(zip(DURATION_FORM_FIELDS, data_list))
timedelta_object = timedelta(**duration_dict)
return int(timedelta_object.total_seconds())
class DurationField(PositiveIntegerField):
def get_internal_type(self):
return 'DurationField'
def formfield(self, **kwargs):
defaults = {'form_class': DurationMultiValueField}
defaults.update(kwargs)
return super(DurationField, self).formfield(**kwargs)
from south.modelsinspector import add_introspection_rules
add_introspection_rules([], ["^polls\.model_fields\.DurationField"])
ここにありadmin.py
ます:
from django import forms
from django.contrib import admin
from .models import Poll
from .model_fields import DurationMultiWidget
class PollAdminForm(forms.ModelForm):
class Meta:
model = Poll
widgets = {
'limit': DurationMultiWidget(),
}
...
class PollAdmin(admin.ModelAdmin):
form = PollAdminForm
admin.site.register(Poll, PollAdmin)
models.py
念のため、ここに を示します。
from django.db import models
from django.utils.translation import ugettext_lazy as _
class Poll(models.Model):
...
limit = DurationField(_('limit'), default=0)