mirror of
				https://github.com/django/django.git
				synced 2025-10-25 06:36:07 +00:00 
			
		
		
		
	Fixed #3082 -- newforms: Changed Form as_table() and as_ul() not to display verbose names for hidden fields, and to add field-name prefix to error messages to avoid user confusion. Also added unit tests.
git-svn-id: http://code.djangoproject.com/svn/django/trunk@4146 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		| @@ -80,6 +80,12 @@ class Form(object): | |||||||
|             output.append(u'<tr><td colspan="2">%s</td></tr>' % self.non_field_errors()) |             output.append(u'<tr><td colspan="2">%s</td></tr>' % self.non_field_errors()) | ||||||
|         for name, field in self.fields.items(): |         for name, field in self.fields.items(): | ||||||
|             bf = BoundField(self, field, name) |             bf = BoundField(self, field, name) | ||||||
|  |             if bf.is_hidden: | ||||||
|  |                 if bf.errors: | ||||||
|  |                     new_errors = ErrorList(['(Hidden field %s) %s' % (name, e) for e in bf.errors]) | ||||||
|  |                     output.append(u'<tr><td colspan="2">%s</td></tr>' % new_errors) | ||||||
|  |                 output.append(str(bf)) | ||||||
|  |             else: | ||||||
|                 if bf.errors: |                 if bf.errors: | ||||||
|                     output.append(u'<tr><td colspan="2">%s</td></tr>' % bf.errors) |                     output.append(u'<tr><td colspan="2">%s</td></tr>' % bf.errors) | ||||||
|                 output.append(u'<tr><td>%s</td><td>%s</td></tr>' % (bf.label_tag(escape(bf.verbose_name+':')), bf)) |                 output.append(u'<tr><td>%s</td><td>%s</td></tr>' % (bf.label_tag(escape(bf.verbose_name+':')), bf)) | ||||||
| @@ -93,11 +99,13 @@ class Form(object): | |||||||
|             output.append(u'<li>%s</li>' % self.non_field_errors()) |             output.append(u'<li>%s</li>' % self.non_field_errors()) | ||||||
|         for name, field in self.fields.items(): |         for name, field in self.fields.items(): | ||||||
|             bf = BoundField(self, field, name) |             bf = BoundField(self, field, name) | ||||||
|             line = u'<li>' |             if bf.is_hidden: | ||||||
|                 if bf.errors: |                 if bf.errors: | ||||||
|                 line += str(bf.errors) |                     new_errors = ErrorList(['(Hidden field %s) %s' % (name, e) for e in bf.errors]) | ||||||
|             line += u'%s %s</li>' % (bf.label_tag(escape(bf.verbose_name+':')), bf) |                     output.append(u'<li>%s</li>' % new_errors) | ||||||
|             output.append(line) |                 output.append(str(bf)) | ||||||
|  |             else: | ||||||
|  |                 output.append(u'<li>%s%s %s</li>' % (bf.errors, bf.label_tag(escape(bf.verbose_name+':')), bf)) | ||||||
|         return u'\n'.join(output) |         return u'\n'.join(output) | ||||||
|  |  | ||||||
|     def non_field_errors(self): |     def non_field_errors(self): | ||||||
| @@ -222,6 +230,11 @@ class BoundField(object): | |||||||
|             contents = '<label for="%s">%s</label>' % (widget.id_for_label(id_), contents) |             contents = '<label for="%s">%s</label>' % (widget.id_for_label(id_), contents) | ||||||
|         return contents |         return contents | ||||||
|  |  | ||||||
|  |     def _is_hidden(self): | ||||||
|  |         "Returns True if this BoundField's widget is hidden." | ||||||
|  |         return self._field.widget.is_hidden | ||||||
|  |     is_hidden = property(_is_hidden) | ||||||
|  |  | ||||||
|     def _auto_id(self): |     def _auto_id(self): | ||||||
|         """ |         """ | ||||||
|         Calculates and returns the ID attribute for this BoundField, if the |         Calculates and returns the ID attribute for this BoundField, if the | ||||||
|   | |||||||
| @@ -23,6 +23,8 @@ flatatt = lambda attrs: u''.join([u' %s="%s"' % (k, escape(v)) for k, v in attrs | |||||||
|  |  | ||||||
| class Widget(object): | class Widget(object): | ||||||
|     requires_data_list = False # Determines whether render()'s 'value' argument should be a list. |     requires_data_list = False # Determines whether render()'s 'value' argument should be a list. | ||||||
|  |     is_hidden = False          # Determines whether this corresponds to an <input type="hidden">. | ||||||
|  |  | ||||||
|     def __init__(self, attrs=None): |     def __init__(self, attrs=None): | ||||||
|         self.attrs = attrs or {} |         self.attrs = attrs or {} | ||||||
|  |  | ||||||
| @@ -76,6 +78,7 @@ class PasswordInput(Input): | |||||||
|  |  | ||||||
| class HiddenInput(Input): | class HiddenInput(Input): | ||||||
|     input_type = 'hidden' |     input_type = 'hidden' | ||||||
|  |     is_hidden = True | ||||||
|  |  | ||||||
| class FileInput(Input): | class FileInput(Input): | ||||||
|     input_type = 'file' |     input_type = 'file' | ||||||
|   | |||||||
| @@ -1689,6 +1689,56 @@ subclass' __init__(). | |||||||
| <tr><td>Last name:</td><td><input type="text" name="last_name" /></td></tr> | <tr><td>Last name:</td><td><input type="text" name="last_name" /></td></tr> | ||||||
| <tr><td>Birthday:</td><td><input type="text" name="birthday" /></td></tr> | <tr><td>Birthday:</td><td><input type="text" name="birthday" /></td></tr> | ||||||
|  |  | ||||||
|  | HiddenInput widgets are displayed differently in the as_table() and as_ul() | ||||||
|  | output of a Form -- their verbose names are not displayed, and a separate | ||||||
|  | <tr>/<li> is not displayed. | ||||||
|  | >>> class Person(Form): | ||||||
|  | ...     first_name = CharField() | ||||||
|  | ...     last_name = CharField() | ||||||
|  | ...     hidden_text = CharField(widget=HiddenInput) | ||||||
|  | ...     birthday = DateField() | ||||||
|  | >>> p = Person() | ||||||
|  | >>> print p | ||||||
|  | <tr><td>First name:</td><td><input type="text" name="first_name" /></td></tr> | ||||||
|  | <tr><td>Last name:</td><td><input type="text" name="last_name" /></td></tr> | ||||||
|  | <input type="hidden" name="hidden_text" /> | ||||||
|  | <tr><td>Birthday:</td><td><input type="text" name="birthday" /></td></tr> | ||||||
|  | >>> print p.as_ul() | ||||||
|  | <li>First name: <input type="text" name="first_name" /></li> | ||||||
|  | <li>Last name: <input type="text" name="last_name" /></li> | ||||||
|  | <input type="hidden" name="hidden_text" /> | ||||||
|  | <li>Birthday: <input type="text" name="birthday" /></li> | ||||||
|  |  | ||||||
|  | With auto_id set, a HiddenInput still gets an ID, but it doesn't get a label. | ||||||
|  | >>> p = Person(auto_id='id_%s') | ||||||
|  | >>> print p | ||||||
|  | <tr><td><label for="id_first_name">First name:</label></td><td><input type="text" name="first_name" id="id_first_name" /></td></tr> | ||||||
|  | <tr><td><label for="id_last_name">Last name:</label></td><td><input type="text" name="last_name" id="id_last_name" /></td></tr> | ||||||
|  | <input type="hidden" name="hidden_text" id="id_hidden_text" /> | ||||||
|  | <tr><td><label for="id_birthday">Birthday:</label></td><td><input type="text" name="birthday" id="id_birthday" /></td></tr> | ||||||
|  | >>> print p.as_ul() | ||||||
|  | <li><label for="id_first_name">First name:</label> <input type="text" name="first_name" id="id_first_name" /></li> | ||||||
|  | <li><label for="id_last_name">Last name:</label> <input type="text" name="last_name" id="id_last_name" /></li> | ||||||
|  | <input type="hidden" name="hidden_text" id="id_hidden_text" /> | ||||||
|  | <li><label for="id_birthday">Birthday:</label> <input type="text" name="birthday" id="id_birthday" /></li> | ||||||
|  |  | ||||||
|  | If a field with a HiddenInput has errors, the as_table() and as_ul() output | ||||||
|  | will include the error message(s) with the text "(Hidden field [fieldname]) " | ||||||
|  | prepended. | ||||||
|  | >>> p = Person({'first_name': 'John', 'last_name': 'Lennon', 'birthday': '1940-10-9'}) | ||||||
|  | >>> print p | ||||||
|  | <tr><td>First name:</td><td><input type="text" name="first_name" value="John" /></td></tr> | ||||||
|  | <tr><td>Last name:</td><td><input type="text" name="last_name" value="Lennon" /></td></tr> | ||||||
|  | <tr><td colspan="2"><ul class="errorlist"><li>(Hidden field hidden_text) This field is required.</li></ul></td></tr> | ||||||
|  | <input type="hidden" name="hidden_text" /> | ||||||
|  | <tr><td>Birthday:</td><td><input type="text" name="birthday" value="1940-10-9" /></td></tr> | ||||||
|  | >>> print p.as_ul() | ||||||
|  | <li>First name: <input type="text" name="first_name" value="John" /></li> | ||||||
|  | <li>Last name: <input type="text" name="last_name" value="Lennon" /></li> | ||||||
|  | <li><ul class="errorlist"><li>(Hidden field hidden_text) This field is required.</li></ul></li> | ||||||
|  | <input type="hidden" name="hidden_text" /> | ||||||
|  | <li>Birthday: <input type="text" name="birthday" value="1940-10-9" /></li> | ||||||
|  |  | ||||||
| A Form's fields are displayed in the same order in which they were defined. | A Form's fields are displayed in the same order in which they were defined. | ||||||
| >>> class TestForm(Form): | >>> class TestForm(Form): | ||||||
| ...     field1 = CharField() | ...     field1 = CharField() | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user