Thursday, March 31, 2011

Custom django widget - decompress() arg not populated.

As an exercise I am trying to create a custom django widget for a 24 hour clock. The widget will is a MultiWidget - a select box for each field.

I am trying to follow docs online (kinda sparse) and looking at the Pro Django book, but I can't seem to figure it out. Am I on the right track? I can save my data from the form, but when I prepopulate the form, the form doesn't have the previous values.

It seems the issue is that the decompress() methods 'value' argument is always empty, so I have nothing to interpret.

from django.forms import widgets

import datetime

class MilitaryTimeWidget(widgets.MultiWidget):
    """
    A widget that displays 24 hours time selection.
    """
    def __init__(self, attrs=None):
        hours = [ (i, "%02d" %(i)) for i in range(0, 24) ]
        minutes = [ (i, "%02d" %(i)) for i in range(0, 60) ]
        _widgets = (
            widgets.Select(attrs=attrs, choices=hours), 
            widgets.Select(attrs=attrs, choices=minutes),
            )
        super(MilitaryTimeWidget, self).__init__(_widgets, attrs)

    def decompress(self, value):
        print "******** %s" %value
        if value:
            return [int(value.hour), int(value.minute)]
        return [None, None]

    def value_from_datadict(self, data, files, name):
        hour = data.get("%s_0" %name, None)
        minute = data.get("%s_1" %name, None)
        if hour and minute:
            hour = int(hour)
            minute = int(minute)
            return datetime.time(hour=hour, minute=minute)
        return None

In my form, I am calling the widget like:

arrival_time = forms.TimeField(label="Arrival Time", required=False, widget=MilitaryTimeWidget())
From stackoverflow
  • I can't reproduce the problem:

    >>> class MyForm(forms.Form):
    ...     t = forms.TimeField(widget=MilitaryTimeWidget())
    ...
    >>> print MyForm(data={'t_0': '13', 't_1': '34'})
    ******** 13:34:00
    <tr><th><label for="id_t_0">T:</label></th><td><select name="t_0" id="id_t_0">
    <option value="0">00</option>
    [...]
    <option value="13" selected="selected">13</option>
    [...]
    <option value="23">23</option>
    </select><select name="t_1" id="id_t_1">
    <option value="0">00</option>
    [...]
    <option value="34" selected="selected">34</option>
    [...]
    <option value="59">59</option>
    </select></td></tr>
    

    Check that your request.POST is correct.

    As a sidenote, are you sure this widget gives good usability? Four mouse clicks and possible scrolling of the minutes combobox...

    ashchristopher : I think my problem is that I am not trying to populate the form using the POST data but instead trying to populate using one of my models. Since I am missing the arrival_time_0 and arrival_time_1 (the names given to each select widget) in the data I pass in, it isnt being populated.
    ashchristopher : It seems rather heavy-handed and non-pythonic to purposfully split that data in my view before passing into the form. Thoughts?
  • Note this line in the docstring for MultiWidget:

    You'll probably want to use this class with MultiValueField.

    That's the root of your problem. You might be able to get the single-widget-only approach working (Marty says it's possible in Pro Django, but I've never tried it, and I think it's likely to be more work), but in that case your widget shouldn't be a subclass of MultiWidget.

    What you need to do (if you want to follow the MultiWidget/MultiValueField path) is:

    • remove your value_from_datadict method
    • define a subclass of MultiValueField with a definition of the compress() method which does the task you're currently doing in value_from_datadict() (transforming a list of numbers into a datetime.time object)
    • set your Widget as the default one for your custom form Field (using the widget class attribute)
    • either create a custom model Field which returns your custom form Field from its formfield() method, or use your custom form field manually as a field override in a ModelForm.

    Then everything will Just Work.

    akaihola : Looks like django.forms.fields.SplitDateTimeField and django.forms.widgets.SplitDateTimeWidget are good models to base on.

0 comments:

Post a Comment