On Sat, Aug 8, 2009 at 2:09 PM, Margie<margierogin...@yahoo.com> wrote:
>
> Ok, still slightly confused.  First - a high level description.  I
> have a field that contains choices, but when I display the select, I
> want to display some extra html below the select box, so I am creating
> a custom widget that displays the standard select followed by my html.
>
> Question: If want to use a special widget for a ChoiceField, is it
> true that I need to instantiate the ChoiceField (or TypedChoiceField),
> rather than just setting the .widget attribute on the one that is by
> default created for me (due to it being a modelForm)?
>
> I find that if I just do this:
>
>            self.fields["status"].widget = StatusWidget(task=instance)
>
> then my widget's select does not contain any choices.  My widget
> doesn't do anything special to the select, it looks like this:
>
> class StatusWidget(widgets.Select):
>    def __init__(self, task, attrs={}):
>        self.task = task
>        super(StatusWidget, self).__init__(attrs)
>
>    def render(self, name, value, attrs=None):
>        rendered = super(StatusWidget, self).render(name, value,
> attrs)
>        rendered = rendered + .... add a bunch of stuff to the end
>        return rendered
>
> Because I seem unable to display the choices correctly in the select
> box when I just set the field's widget attribute to my StatusWidget, I
> am instantiating the field itself.  That now looks like this:
>
>            self.fields["status"] = forms.TypedChoiceField
> (choices=Task.STATUS_CHOICES, widget=StatusWidget(task=instance),
> required=False, coerce=IntegerField.to_python)
>
> However, I see when debugging that IntegerField.to_python is an
> unbound method:
>
> (Pdb) self.coerce
> <unbound method IntegerField.to_python>
>
> What is the right thing to set coerce to if I just want it to do
> whatever it would "normally" do for the corresponding model field if I
> wasn't trying to override the widget?  In my case I have verified that
> if I set coerce=int that does work, but that doesn't seem very
> general.  I'd much rather use whatever the standard coerce method
> would have been if I hadn't overridden the widget.
>
> Margie
>
>
>
>
> On Aug 8, 12:11 am, Malcolm Tredinnick <malc...@pointy-stick.com>
> wrote:
>> Hi Margie,
>>
>>
>>
>> On Fri, 2009-08-07 at 23:17 -0700, Margie wrote:
>>
>> > Hmmm, ok, after digging around I realize that full_clean was not
>> > setting cleaned_data for the status field to an integer value;
>> > cleaned_data['status'] was just getting set to something like u'1'.
>>
>> > I am in fact using sqllite, and yes, my status fields are just
>> > integers:
>>
>> > OPEN_STATUS = 1
>> > CLOSED_STATUS = 2
>> > STALLED_STATUS = 3
>>
>> > I think the problem has to do with the way I created the status field,
>> > which is like this:
>>
>> >             self.fields["status"] = forms.ChoiceField
>> > (choices=Task.STATUS_CHOICES, widget=StatusWidget(task=instance),
>> > required=False)
>>
>> Right, that won't do what you want. ChoiceField normalizes to a unicode
>> object.
>>
>>
>>
>> > I tried moving to TypedChoiceField(), but that didn't help.  I
>> > debugged into it in the case where I used TypedChoiceField() and I can
>> > see that when coerce is called it isn't doing anything, it is just
>> > returning the unicode value.
>>
>> > I find that if I do this instead, that it does do the coerce
>> > correctly:
>>
>> >             self.fields["status"].widget = StatusWidget(task=instance)
>>
>> > In looking at the doc it looks like the purpose of TypedChoiceField()
>> > is to allow me to create my own coerce function, is that right?
>>
>> Correct.
>>
>> >   And
>> > of course I wasn't doing that so it was behaving the same as
>> > ChoiceField, and it looks like the default there is to just return the
>> > unicode.
>>
>> Also correct. The documentation says "Defaults to an identity function"
>> and all the data coming from a form submission are strings (Python
>> unicode objects), so if you don't supply the coerce parameter, it does
>> nothing.
>>
>> It's probably a slight API wart that TypedChoiceField doesn't just raise
>> an exception if you don't supply coerce(). The default is slightly
>> dangerous, as it almost always means you're misusing the field. Not a
>> fatal flaw in the design, however.
>>
>>
>>
>> > When I don't declare the status field at all (ie, just let django do
>> > it's default thing), my guess is that it is choosing a coerce function
>> > based on the integer type of my choices, is that true?
>>
>> Yes. The django.db.models.fields.Field.formfield() method detects if you
>> have specified "choices" in the field and uses the Field subclass's
>> to_python() function as the coerce method.
>>
>> >   I have never
>> > used anything but sqlite3 so far, so I guess that was masking the
>> > error and I would have run into this in a more serious way when I
>> > moved to a different db?
>>
>> Actually, my SQLite observation was entirely bogus. I suspect what
>> you're seeing is the difference between these two lines:
>>
>>         t1 = Task.objects.create(status=u'3')
>>         t2 = Task.objects.get(id=t1.id)
>>
>> In this case, t1.status will be a unicode string, but t2.status will
>> always be an integer, no matter what DB backend is involved. So you are
>> seemingly stumbling over a case like the t1 situation -- where you're
>> using the object after it has been populated and saved, but still using
>> the raw data.
>>
>> I've long argued that allowing the wrong types to be assigned to
>> attributes in Django models is a flaw (Python got rid of automatic type
>> coercion for operator arguments years ago for the same reason), but it's
>> been there forever in Django-years, so isn't going to go away now.
>>
>> Regards,
>> Malcolm
> >
>

You can just set coerce to `IntegerField().to_python`, here you can
instantiate a dummy IntegerField instance to use.

Alex

-- 
"I disapprove of what you say, but I will defend to the death your
right to say it." -- Voltaire
"The people's good is the highest law." -- Cicero
"Code can always be simpler than you think, but never as simple as you
want" -- Me

--~--~---------~--~----~------------~-------~--~----~
You received this message because you are subscribed to the Google Groups 
"Django users" group.
To post to this group, send email to django-users@googlegroups.com
To unsubscribe from this group, send email to 
django-users+unsubscr...@googlegroups.com
For more options, visit this group at 
http://groups.google.com/group/django-users?hl=en
-~----------~----~----~----~------~----~------~--~---

Reply via email to