0

DurationFieldDjango で、期間を秒数で表す整数を格納するカスタムを実装しました。また、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) 
4

1 に答える 1

0

うーん、それはタイプミスでした:formfield()メソッドの下で、誤って の**kwargs代わりに を渡しました**defaults

余談ですが、誰かがまったく同じことをしようとしているというまれな例ではkwargs.pop('min_value')DurationMultiValueField.__init__(PositiveIntegerField であるため) で a を実行する必要があります。

于 2014-03-06T16:14:28.533 に答える