mirror of
				https://github.com/django/django.git
				synced 2025-10-25 14:46:09 +00:00 
			
		
		
		
	Fixed #14655 -- Made formsets iterable. This allows a slightly more natural iteration API (for form in formsets), and allows you to easily override the form rendering order. Thanks to Kent Hauser for the suggestion and patch.
				
					
				
			git-svn-id: http://code.djangoproject.com/svn/django/trunk@14986 bcc190cf-cafb-0310-a4f2-bffc1f526a37
This commit is contained in:
		
							
								
								
									
										1
									
								
								AUTHORS
									
									
									
									
									
								
							
							
						
						
									
										1
									
								
								AUTHORS
									
									
									
									
									
								
							| @@ -216,6 +216,7 @@ answer newbie questions, and generally made Django that much better: | |||||||
|     Brant Harris |     Brant Harris | ||||||
|     Ronny Haryanto <http://ronny.haryan.to/> |     Ronny Haryanto <http://ronny.haryan.to/> | ||||||
|     Hawkeye |     Hawkeye | ||||||
|  |     Kent Hauser <kent@khauser.net> | ||||||
|     Joe Heck <http://www.rhonabwy.com/wp/> |     Joe Heck <http://www.rhonabwy.com/wp/> | ||||||
|     Joel Heenan <joelh-django@planetjoel.com> |     Joel Heenan <joelh-django@planetjoel.com> | ||||||
|     Mikko Hellsing <mikko@sorl.net> |     Mikko Hellsing <mikko@sorl.net> | ||||||
|   | |||||||
| @@ -49,6 +49,17 @@ class BaseFormSet(StrAndUnicode): | |||||||
|     def __unicode__(self): |     def __unicode__(self): | ||||||
|         return self.as_table() |         return self.as_table() | ||||||
|  |  | ||||||
|  |     def __iter__(self): | ||||||
|  |         """Yields the forms in the order they should be rendered""" | ||||||
|  |         return iter(self.forms) | ||||||
|  |  | ||||||
|  |     def __getitem__(self, index): | ||||||
|  |         """Returns the form at the given index, based on the rendering order""" | ||||||
|  |         return list(self)[index] | ||||||
|  |  | ||||||
|  |     def __len__(self): | ||||||
|  |         return len(self.forms) | ||||||
|  |  | ||||||
|     def _management_form(self): |     def _management_form(self): | ||||||
|         """Returns the ManagementForm instance for this FormSet.""" |         """Returns the ManagementForm instance for this FormSet.""" | ||||||
|         if self.is_bound: |         if self.is_bound: | ||||||
| @@ -323,17 +334,17 @@ class BaseFormSet(StrAndUnicode): | |||||||
|         # XXX: there is no semantic division between forms here, there |         # XXX: there is no semantic division between forms here, there | ||||||
|         # probably should be. It might make sense to render each form as a |         # probably should be. It might make sense to render each form as a | ||||||
|         # table row with each field as a td. |         # table row with each field as a td. | ||||||
|         forms = u' '.join([form.as_table() for form in self.forms]) |         forms = u' '.join([form.as_table() for form in self]) | ||||||
|         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) |         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) | ||||||
|  |  | ||||||
|     def as_p(self): |     def as_p(self): | ||||||
|         "Returns this formset rendered as HTML <p>s." |         "Returns this formset rendered as HTML <p>s." | ||||||
|         forms = u' '.join([form.as_p() for form in self.forms]) |         forms = u' '.join([form.as_p() for form in self]) | ||||||
|         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) |         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) | ||||||
|  |  | ||||||
|     def as_ul(self): |     def as_ul(self): | ||||||
|         "Returns this formset rendered as HTML <li>s." |         "Returns this formset rendered as HTML <li>s." | ||||||
|         forms = u' '.join([form.as_ul() for form in self.forms]) |         forms = u' '.join([form.as_ul() for form in self]) | ||||||
|         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) |         return mark_safe(u'\n'.join([unicode(self.management_form), forms])) | ||||||
|  |  | ||||||
| def formset_factory(form, formset=BaseFormSet, extra=1, can_order=False, | def formset_factory(form, formset=BaseFormSet, extra=1, can_order=False, | ||||||
|   | |||||||
| @@ -23,7 +23,7 @@ the ability to iterate over the forms in the formset and display them as you | |||||||
| would with a regular form:: | would with a regular form:: | ||||||
|  |  | ||||||
|     >>> formset = ArticleFormSet() |     >>> formset = ArticleFormSet() | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> | ||||||
|     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> |     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> | ||||||
| @@ -35,6 +35,20 @@ display two blank forms:: | |||||||
|  |  | ||||||
|     >>> ArticleFormSet = formset_factory(ArticleForm, extra=2) |     >>> ArticleFormSet = formset_factory(ArticleForm, extra=2) | ||||||
|  |  | ||||||
|  | .. versionchanged:: 1.3 | ||||||
|  |  | ||||||
|  | Prior to Django 1.3, formset instances were not iterable. To render | ||||||
|  | the formset you iterated over the ``forms`` attribute:: | ||||||
|  |  | ||||||
|  |     >>> formset = ArticleFormSet() | ||||||
|  |     >>> for form in formset.forms: | ||||||
|  |     ...     print form.as_table() | ||||||
|  |  | ||||||
|  | Iterating over ``formset.forms`` will render the forms in the order | ||||||
|  | they were created. The default formset iterator also renders the forms | ||||||
|  | in this order, but you can change this order by providing an alternate | ||||||
|  | implementation for the :method:`__iter__()` method. | ||||||
|  |  | ||||||
| Using initial data with a formset | Using initial data with a formset | ||||||
| --------------------------------- | --------------------------------- | ||||||
|  |  | ||||||
| @@ -50,7 +64,7 @@ example:: | |||||||
|     ...      'pub_date': datetime.date.today()}, |     ...      'pub_date': datetime.date.today()}, | ||||||
|     ... ]) |     ... ]) | ||||||
|  |  | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Django is now open source" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Django is now open source" id="id_form-0-title" /></td></tr> | ||||||
|     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" value="2008-05-12" id="id_form-0-pub_date" /></td></tr> |     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" value="2008-05-12" id="id_form-0-pub_date" /></td></tr> | ||||||
| @@ -77,7 +91,7 @@ limit the maximum number of empty forms the formset will display:: | |||||||
|  |  | ||||||
|     >>> ArticleFormSet = formset_factory(ArticleForm, extra=2, max_num=1) |     >>> ArticleFormSet = formset_factory(ArticleForm, extra=2, max_num=1) | ||||||
|     >>> formset = ArticleFormset() |     >>> formset = ArticleFormset() | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> | ||||||
|     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> |     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> | ||||||
| @@ -250,7 +264,7 @@ Lets create a formset with the ability to order:: | |||||||
|     ...     {'title': u'Article #1', 'pub_date': datetime.date(2008, 5, 10)}, |     ...     {'title': u'Article #1', 'pub_date': datetime.date(2008, 5, 10)}, | ||||||
|     ...     {'title': u'Article #2', 'pub_date': datetime.date(2008, 5, 11)}, |     ...     {'title': u'Article #2', 'pub_date': datetime.date(2008, 5, 11)}, | ||||||
|     ... ]) |     ... ]) | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Article #1" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Article #1" id="id_form-0-title" /></td></tr> | ||||||
|     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" value="2008-05-10" id="id_form-0-pub_date" /></td></tr> |     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" value="2008-05-10" id="id_form-0-pub_date" /></td></tr> | ||||||
| @@ -306,7 +320,7 @@ Lets create a formset with the ability to delete:: | |||||||
|     ...     {'title': u'Article #1', 'pub_date': datetime.date(2008, 5, 10)}, |     ...     {'title': u'Article #1', 'pub_date': datetime.date(2008, 5, 10)}, | ||||||
|     ...     {'title': u'Article #2', 'pub_date': datetime.date(2008, 5, 11)}, |     ...     {'title': u'Article #2', 'pub_date': datetime.date(2008, 5, 11)}, | ||||||
|     ... ]) |     ... ]) | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ....    print form.as_table() |     ....    print form.as_table() | ||||||
|     <input type="hidden" name="form-TOTAL_FORMS" value="3" id="id_form-TOTAL_FORMS" /><input type="hidden" name="form-INITIAL_FORMS" value="2" id="id_form-INITIAL_FORMS" /><input type="hidden" name="form-MAX_NUM_FORMS" id="id_form-MAX_NUM_FORMS" /> |     <input type="hidden" name="form-TOTAL_FORMS" value="3" id="id_form-TOTAL_FORMS" /><input type="hidden" name="form-INITIAL_FORMS" value="2" id="id_form-INITIAL_FORMS" /><input type="hidden" name="form-MAX_NUM_FORMS" id="id_form-MAX_NUM_FORMS" /> | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Article #1" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" value="Article #1" id="id_form-0-title" /></td></tr> | ||||||
| @@ -360,7 +374,7 @@ default fields/attributes of the order and deletion fields:: | |||||||
|  |  | ||||||
|     >>> ArticleFormSet = formset_factory(ArticleForm, formset=BaseArticleFormSet) |     >>> ArticleFormSet = formset_factory(ArticleForm, formset=BaseArticleFormSet) | ||||||
|     >>> formset = ArticleFormSet() |     >>> formset = ArticleFormSet() | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> |     <tr><th><label for="id_form-0-title">Title:</label></th><td><input type="text" name="form-0-title" id="id_form-0-title" /></td></tr> | ||||||
|     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> |     <tr><th><label for="id_form-0-pub_date">Pub date:</label></th><td><input type="text" name="form-0-pub_date" id="id_form-0-pub_date" /></td></tr> | ||||||
| @@ -393,7 +407,7 @@ The ``manage_articles.html`` template might look like this: | |||||||
|     <form method="post" action=""> |     <form method="post" action=""> | ||||||
|         {{ formset.management_form }} |         {{ formset.management_form }} | ||||||
|         <table> |         <table> | ||||||
|             {% for form in formset.forms %} |             {% for form in formset %} | ||||||
|             {{ form }} |             {{ form }} | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
|         </table> |         </table> | ||||||
|   | |||||||
| @@ -684,7 +684,7 @@ so long as the total number of forms does not exceed ``max_num``:: | |||||||
|  |  | ||||||
|     >>> AuthorFormSet = modelformset_factory(Author, max_num=4, extra=2) |     >>> AuthorFormSet = modelformset_factory(Author, max_num=4, extra=2) | ||||||
|     >>> formset = AuthorFormSet(queryset=Author.objects.order_by('name')) |     >>> formset = AuthorFormSet(queryset=Author.objects.order_by('name')) | ||||||
|     >>> for form in formset.forms: |     >>> for form in formset: | ||||||
|     ...     print form.as_table() |     ...     print form.as_table() | ||||||
|     <tr><th><label for="id_form-0-name">Name:</label></th><td><input id="id_form-0-name" type="text" name="form-0-name" value="Charles Baudelaire" maxlength="100" /><input type="hidden" name="form-0-id" value="1" id="id_form-0-id" /></td></tr> |     <tr><th><label for="id_form-0-name">Name:</label></th><td><input id="id_form-0-name" type="text" name="form-0-name" value="Charles Baudelaire" maxlength="100" /><input type="hidden" name="form-0-id" value="1" id="id_form-0-id" /></td></tr> | ||||||
|     <tr><th><label for="id_form-1-name">Name:</label></th><td><input id="id_form-1-name" type="text" name="form-1-name" value="Paul Verlaine" maxlength="100" /><input type="hidden" name="form-1-id" value="3" id="id_form-1-id" /></td></tr> |     <tr><th><label for="id_form-1-name">Name:</label></th><td><input id="id_form-1-name" type="text" name="form-1-name" value="Paul Verlaine" maxlength="100" /><input type="hidden" name="form-1-id" value="3" id="id_form-1-id" /></td></tr> | ||||||
| @@ -778,7 +778,7 @@ itself:: | |||||||
|  |  | ||||||
|     <form method="post" action=""> |     <form method="post" action=""> | ||||||
|         {{ formset.management_form }} |         {{ formset.management_form }} | ||||||
|         {% for form in formset.forms %} |         {% for form in formset %} | ||||||
|             {{ form }} |             {{ form }} | ||||||
|         {% endfor %} |         {% endfor %} | ||||||
|     </form> |     </form> | ||||||
| @@ -791,7 +791,7 @@ Third, you can manually render each field:: | |||||||
|  |  | ||||||
|     <form method="post" action=""> |     <form method="post" action=""> | ||||||
|         {{ formset.management_form }} |         {{ formset.management_form }} | ||||||
|         {% for form in formset.forms %} |         {% for form in formset %} | ||||||
|             {% for field in form %} |             {% for field in form %} | ||||||
|                 {{ field.label_tag }}: {{ field }} |                 {{ field.label_tag }}: {{ field }} | ||||||
|             {% endfor %} |             {% endfor %} | ||||||
| @@ -804,7 +804,7 @@ if you were rendering the ``name`` and ``age`` fields of a model:: | |||||||
|  |  | ||||||
|     <form method="post" action=""> |     <form method="post" action=""> | ||||||
|         {{ formset.management_form }} |         {{ formset.management_form }} | ||||||
|         {% for form in formset.forms %} |         {% for form in formset %} | ||||||
|             {{ form.id }} |             {{ form.id }} | ||||||
|             <ul> |             <ul> | ||||||
|                 <li>{{ form.name }}</li> |                 <li>{{ form.name }}</li> | ||||||
|   | |||||||
| @@ -767,6 +767,38 @@ class FormsFormsetTestCase(TestCase): | |||||||
|         self.assertFalse(formset.is_valid()) |         self.assertFalse(formset.is_valid()) | ||||||
|         self.assertEqual(formset.non_form_errors(), [u'You may only specify a drink once.']) |         self.assertEqual(formset.non_form_errors(), [u'You may only specify a drink once.']) | ||||||
|  |  | ||||||
|  |     def test_formset_iteration(self): | ||||||
|  |         # Regression tests for #16455 -- formset instances are iterable | ||||||
|  |         ChoiceFormset = formset_factory(Choice, extra=3) | ||||||
|  |         formset = ChoiceFormset() | ||||||
|  |  | ||||||
|  |         # confirm iterated formset yields formset.forms | ||||||
|  |         forms = list(formset) | ||||||
|  |         self.assertEqual(forms, formset.forms) | ||||||
|  |         self.assertEqual(len(formset), len(forms)) | ||||||
|  |  | ||||||
|  |         # confirm indexing of formset | ||||||
|  |         self.assertEqual(formset[0], forms[0]) | ||||||
|  |         try: | ||||||
|  |             formset[3] | ||||||
|  |             self.fail('Requesting an invalid formset index should raise an exception') | ||||||
|  |         except IndexError: | ||||||
|  |             pass | ||||||
|  |  | ||||||
|  |         # Formets can override the default iteration order | ||||||
|  |         class BaseReverseFormSet(BaseFormSet): | ||||||
|  |             def __iter__(self): | ||||||
|  |                 for form in reversed(self.forms): | ||||||
|  |                     yield form | ||||||
|  |  | ||||||
|  |         ReverseChoiceFormset = formset_factory(Choice, BaseReverseFormSet, extra=3) | ||||||
|  |         reverse_formset = ReverseChoiceFormset() | ||||||
|  |  | ||||||
|  |         # confirm that __iter__ modifies rendering order | ||||||
|  |         # compare forms from "reverse" formset with forms from original formset | ||||||
|  |         self.assertEqual(str(reverse_formset[0]), str(forms[-1])) | ||||||
|  |         self.assertEqual(str(reverse_formset[1]), str(forms[-2])) | ||||||
|  |         self.assertEqual(len(reverse_formset), len(forms)) | ||||||
|  |  | ||||||
| data = { | data = { | ||||||
|     'choices-TOTAL_FORMS': '1', # the number of forms rendered |     'choices-TOTAL_FORMS': '1', # the number of forms rendered | ||||||
|   | |||||||
		Reference in New Issue
	
	Block a user