From 9ac4dbd7b53d187ca54f28e247d3a120660938ca Mon Sep 17 00:00:00 2001 From: Baptiste Mispelon Date: Sat, 13 Apr 2013 02:02:28 +0200 Subject: [PATCH] Fixed #4592: Made CheckboxSelectMultiple more like RadioSelect I refactored RadioSelect and CheckboxSelectMultiple to make them inherit from a base class, allowing them to share the behavior of being able to iterate over their subwidgets. Thanks to Matt McClanahan for the initial patch and to Claude Paroz for the review. --- django/forms/widgets.py | 130 ++++++++++++++---------- docs/internals/deprecation.txt | 3 + docs/ref/forms/widgets.txt | 3 + tests/forms_tests/tests/test_widgets.py | 28 +++-- 4 files changed, 106 insertions(+), 58 deletions(-) diff --git a/django/forms/widgets.py b/django/forms/widgets.py index b0e355eeb6..e72ab014b8 100644 --- a/django/forms/widgets.py +++ b/django/forms/widgets.py @@ -11,6 +11,7 @@ try: from urllib.parse import urljoin except ImportError: # Python 2 from urlparse import urljoin +import warnings from django.conf import settings from django.forms.util import flatatt, to_current_timezone @@ -585,14 +586,16 @@ class SelectMultiple(Select): @python_2_unicode_compatible -class RadioInput(SubWidget): +class ChoiceInput(SubWidget): """ - An object used by RadioFieldRenderer that represents a single - . + An object used by ChoiceFieldRenderer that represents a single + . """ + input_type = None # Subclasses must define this def __init__(self, name, value, attrs, choice, index): - self.name, self.value = name, value + self.name = name + self.value = value self.attrs = attrs self.choice_value = force_text(choice[0]) self.choice_label = force_text(choice[1]) @@ -609,8 +612,7 @@ class RadioInput(SubWidget): label_for = format_html(' for="{0}_{1}"', self.attrs['id'], self.index) else: label_for = '' - choice_label = force_text(self.choice_label) - return format_html('{1} {2}', label_for, self.tag(), choice_label) + return format_html('{1} {2}', label_for, self.tag(), self.choice_label) def is_checked(self): return self.value == self.choice_value @@ -618,34 +620,69 @@ class RadioInput(SubWidget): def tag(self): if 'id' in self.attrs: self.attrs['id'] = '%s_%s' % (self.attrs['id'], self.index) - final_attrs = dict(self.attrs, type='radio', name=self.name, value=self.choice_value) + final_attrs = dict(self.attrs, type=self.input_type, name=self.name, value=self.choice_value) if self.is_checked(): final_attrs['checked'] = 'checked' return format_html('', flatatt(final_attrs)) + +class RadioChoiceInput(ChoiceInput): + input_type = 'radio' + + def __init__(self, *args, **kwargs): + super(RadioChoiceInput, self).__init__(*args, **kwargs) + self.value = force_text(self.value) + + +class RadioInput(RadioChoiceInput): + def __init__(self, *args, **kwargs): + msg = "RadioInput has been deprecated. Use RadioChoiceInput instead." + warnings.warn(msg, PendingDeprecationWarning, stacklevel=2) + super(RadioInput, self).__init__(*args, **kwargs) + + +class CheckboxChoiceInput(ChoiceInput): + input_type = 'checkbox' + + def __init__(self, *args, **kwargs): + super(CheckboxChoiceInput, self).__init__(*args, **kwargs) + self.value = set(force_text(v) for v in self.value) + + def is_checked(self): + return self.choice_value in self.value + + @python_2_unicode_compatible -class RadioFieldRenderer(object): +class ChoiceFieldRenderer(object): """ An object used by RadioSelect to enable customization of radio widgets. """ + choice_input_class = None + def __init__(self, name, value, attrs, choices): - self.name, self.value, self.attrs = name, value, attrs + self.name = name + self.value = value + self.attrs = attrs self.choices = choices def __iter__(self): for i, choice in enumerate(self.choices): - yield RadioInput(self.name, self.value, self.attrs.copy(), choice, i) + yield self.choice_input_class(self.name, self.value, self.attrs.copy(), choice, i) def __getitem__(self, idx): choice = self.choices[idx] # Let the IndexError propogate - return RadioInput(self.name, self.value, self.attrs.copy(), choice, idx) + return self.choice_input_class(self.name, self.value, self.attrs.copy(), choice, idx) def __str__(self): return self.render() def render(self): - """Outputs a
    for this set of radio fields.""" + """ + Outputs a
      for this set of choice fields. + If an id was given to the field, it is applied to the
        (each + item in the list will get an id of `$id_$i`). + """ id_ = self.attrs.get('id', None) start_tag = format_html('
          ', id_) if id_ else '
            ' output = [start_tag] @@ -654,15 +691,25 @@ class RadioFieldRenderer(object): output.append('
          ') return mark_safe('\n'.join(output)) -class RadioSelect(Select): - renderer = RadioFieldRenderer + +class RadioFieldRenderer(ChoiceFieldRenderer): + choice_input_class = RadioChoiceInput + + +class CheckboxFieldRenderer(ChoiceFieldRenderer): + choice_input_class = CheckboxChoiceInput + + +class RendererMixin(object): + renderer = None # subclasses must define this + _empty_value = None def __init__(self, *args, **kwargs): # Override the default renderer if we were passed one. renderer = kwargs.pop('renderer', None) if renderer: self.renderer = renderer - super(RadioSelect, self).__init__(*args, **kwargs) + super(RendererMixin, self).__init__(*args, **kwargs) def subwidgets(self, name, value, attrs=None, choices=()): for widget in self.get_renderer(name, value, attrs, choices): @@ -670,56 +717,35 @@ class RadioSelect(Select): def get_renderer(self, name, value, attrs=None, choices=()): """Returns an instance of the renderer.""" - if value is None: value = '' - str_value = force_text(value) # Normalize to string. + if value is None: + value = self._empty_value final_attrs = self.build_attrs(attrs) choices = list(chain(self.choices, choices)) - return self.renderer(name, str_value, final_attrs, choices) + return self.renderer(name, value, final_attrs, choices) def render(self, name, value, attrs=None, choices=()): return self.get_renderer(name, value, attrs, choices).render() def id_for_label(self, id_): - # RadioSelect is represented by multiple fields, - # each of which has a distinct ID. The IDs are made distinct by a "_X" - # suffix, where X is the zero-based index of the radio field. Thus, - # the label for a RadioSelect should reference the first one ('_0'). + # Widgets using this RendererMixin are made of a collection of + # subwidgets, each with their own