mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #3787, #3788 -- Corrected check for IndexError on MultiValueField, and fixed the value_from_datadict method for MultiWidgets to handle Multiwidgets containing Multiwidgets. Also added a testcase walking through the use of MultiWidget/MultiValueField. Thanks to Max Derkachev for reporting these issues and providing fixes.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@5088 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -457,7 +457,7 @@ class MultiValueField(Field): | |||||||
|         for i, field in enumerate(self.fields): |         for i, field in enumerate(self.fields): | ||||||
|             try: |             try: | ||||||
|                 field_value = value[i] |                 field_value = value[i] | ||||||
|             except KeyError: |             except IndexError: | ||||||
|                 field_value = None |                 field_value = None | ||||||
|             if self.required and field_value in EMPTY_VALUES: |             if self.required and field_value in EMPTY_VALUES: | ||||||
|                 raise ValidationError(gettext(u'This field is required.')) |                 raise ValidationError(gettext(u'This field is required.')) | ||||||
|   | |||||||
| @@ -347,7 +347,7 @@ class MultiWidget(Widget): | |||||||
|     id_for_label = classmethod(id_for_label) |     id_for_label = classmethod(id_for_label) | ||||||
|  |  | ||||||
|     def value_from_datadict(self, data, name): |     def value_from_datadict(self, data, name): | ||||||
|         return [data.get(name + '_%s' % i) for i in range(len(self.widgets))] |         return [widget.value_from_datadict(data, name + '_%s' % i) for i, widget in enumerate(self.widgets)] | ||||||
|  |  | ||||||
|     def format_output(self, rendered_widgets): |     def format_output(self, rendered_widgets): | ||||||
|         return u''.join(rendered_widgets) |         return u''.join(rendered_widgets) | ||||||
|   | |||||||
| @@ -5,6 +5,7 @@ from regressions import regression_tests | |||||||
| form_tests = r""" | form_tests = r""" | ||||||
| >>> from django.newforms import * | >>> from django.newforms import * | ||||||
| >>> import datetime | >>> import datetime | ||||||
|  | >>> import time | ||||||
| >>> import re | >>> import re | ||||||
|  |  | ||||||
| ########### | ########### | ||||||
| @@ -3297,6 +3298,94 @@ True | |||||||
| <option value="2016">2016</option> | <option value="2016">2016</option> | ||||||
| </select> | </select> | ||||||
|  |  | ||||||
|  | # MultiWidget and MultiValueField ############################################# | ||||||
|  | # MultiWidgets are widgets composed of other widgets. They are usually  | ||||||
|  | # combined with MultiValueFields - a field that is composed of other fields. | ||||||
|  | # MulitWidgets can themselved be composed of other MultiWidgets. | ||||||
|  | # SplitDateTimeWidget is one example of a MultiWidget. | ||||||
|  |  | ||||||
|  | >>> class ComplexMultiWidget(MultiWidget): | ||||||
|  | ...     def __init__(self, attrs=None): | ||||||
|  | ...         widgets = ( | ||||||
|  | ...             TextInput(),  | ||||||
|  | ...             SelectMultiple(choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), | ||||||
|  | ...             SplitDateTimeWidget(), | ||||||
|  | ...         ) | ||||||
|  | ...         super(ComplexMultiWidget, self).__init__(widgets, attrs) | ||||||
|  | ... | ||||||
|  | ...     def decompress(self, value): | ||||||
|  | ...         if value: | ||||||
|  | ...             data = value.split(',') | ||||||
|  | ...             return [data[0], data[1], datetime.datetime(*time.strptime(data[2], "%Y-%m-%d %H:%M:%S")[0:6])] | ||||||
|  | ...         return [None, None, None] | ||||||
|  | ...     def format_output(self, rendered_widgets): | ||||||
|  | ...         return u'\n'.join(rendered_widgets) | ||||||
|  | >>> w = ComplexMultiWidget() | ||||||
|  | >>> print w.render('name', 'some text,JP,2007-04-25 06:24:00') | ||||||
|  | <input type="text" name="name_0" value="some text" /> | ||||||
|  | <select multiple="multiple" name="name_1"> | ||||||
|  | <option value="J" selected="selected">John</option> | ||||||
|  | <option value="P" selected="selected">Paul</option> | ||||||
|  | <option value="G">George</option> | ||||||
|  | <option value="R">Ringo</option> | ||||||
|  | </select> | ||||||
|  | <input type="text" name="name_2_0" value="2007-04-25" /><input type="text" name="name_2_1" value="06:24:00" /> | ||||||
|  |  | ||||||
|  | >>> class ComplexField(MultiValueField): | ||||||
|  | ...     def __init__(self, required=True, widget=None, label=None, initial=None):  | ||||||
|  | ...         fields = ( | ||||||
|  | ...             CharField(),  | ||||||
|  | ...             MultipleChoiceField(choices=(('J', 'John'), ('P', 'Paul'), ('G', 'George'), ('R', 'Ringo'))), | ||||||
|  | ...             SplitDateTimeField() | ||||||
|  | ...         ) | ||||||
|  | ...         super(ComplexField, self).__init__(fields, required, widget, label, initial)  | ||||||
|  | ... | ||||||
|  | ...     def compress(self, data_list): | ||||||
|  | ...         if data_list: | ||||||
|  | ...             return '%s,%s,%s' % (data_list[0],''.join(data_list[1]),data_list[2]) | ||||||
|  | ...         return None | ||||||
|  |  | ||||||
|  | >>> f = ComplexField(widget=w) | ||||||
|  | >>> f.clean(['some text', ['J','P'], ['2007-04-25','6:24:00']]) | ||||||
|  | u'some text,JP,2007-04-25 06:24:00' | ||||||
|  | >>> f.clean(['some text',['X'], ['2007-04-25','6:24:00']]) | ||||||
|  | Traceback (most recent call last): | ||||||
|  | ... | ||||||
|  | ValidationError: [u'Select a valid choice. X is not one of the available choices.'] | ||||||
|  |  | ||||||
|  | # If insufficient data is provided, None is substituted | ||||||
|  | >>> f.clean(['some text',['JP']]) | ||||||
|  | Traceback (most recent call last): | ||||||
|  | ... | ||||||
|  | ValidationError: [u'This field is required.'] | ||||||
|  |  | ||||||
|  | >>> class ComplexFieldForm(Form): | ||||||
|  | ...     field1 = ComplexField(widget=w) | ||||||
|  | >>> f = ComplexFieldForm() | ||||||
|  | >>> print f | ||||||
|  | <tr><th><label for="id_field1_0">Field1:</label></th><td><input type="text" name="field1_0" id="id_field1_0" /> | ||||||
|  | <select multiple="multiple" name="field1_1" id="id_field1_1"> | ||||||
|  | <option value="J">John</option> | ||||||
|  | <option value="P">Paul</option> | ||||||
|  | <option value="G">George</option> | ||||||
|  | <option value="R">Ringo</option> | ||||||
|  | </select> | ||||||
|  | <input type="text" name="field1_2_0" id="id_field1_2_0" /><input type="text" name="field1_2_1" id="id_field1_2_1" /></td></tr> | ||||||
|  |  | ||||||
|  | >>> f = ComplexFieldForm({'field1_0':'some text','field1_1':['J','P'], 'field1_2_0':'2007-04-25', 'field1_2_1':'06:24:00'}) | ||||||
|  | >>> print f | ||||||
|  | <tr><th><label for="id_field1_0">Field1:</label></th><td><input type="text" name="field1_0" value="some text" id="id_field1_0" /> | ||||||
|  | <select multiple="multiple" name="field1_1" id="id_field1_1"> | ||||||
|  | <option value="J" selected="selected">John</option> | ||||||
|  | <option value="P" selected="selected">Paul</option> | ||||||
|  | <option value="G">George</option> | ||||||
|  | <option value="R">Ringo</option> | ||||||
|  | </select> | ||||||
|  | <input type="text" name="field1_2_0" value="2007-04-25" id="id_field1_2_0" /><input type="text" name="field1_2_1" value="06:24:00" id="id_field1_2_1" /></td></tr> | ||||||
|  |  | ||||||
|  | >>> f.clean_data | ||||||
|  | {'field1': u'some text,JP,2007-04-25 06:24:00'} | ||||||
|  |  | ||||||
| ################################# | ################################# | ||||||
| # Tests of underlying functions # | # Tests of underlying functions # | ||||||
| ################################# | ################################# | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user