I have a page that displays multiple Formsets, each of which has a prefix. The formsets are created using formset_factory the default options, including extra=1. Rows can be added or deleted with JavaScript.
If the user is adding new data, one blank row shows up. Perfect.
If the user has added data but form validation failed, in which case the formset is populated with POST data using MyFormset(data, prefix='o1-formsetname') etc., only the data that they have entered shows up. Again, perfect. (the o1 etc. are dynamically generated, each o corresponds to an “option”, and each “option” may have multiple formsets).
However if the user is editing existing data, in which case the view populates the formset using MyFormset(initial=somedata, prefix='o1-formsetname') where somedata is a list of dicts of data that came from a model in the database, an extra blank row is inserted after this data. I don’t want a blank row to appear unless the user explicitly adds one using the JavaScript.
Is there any simple way to prevent the formset from showing an extra row if the initial data is set? The reason I’m using initial in the third example is that if I just passed the data in using MyFormset(somedata, prefix='o1-formsetname') I’d have to do an extra step of reformatting all the data into a POSTdata style dict including prefixes for each field, for example o1-formsetname-1-price: x etc., as well as calculating the management form data, which adds a whole load of complication.
One solution could be to intercept the formset before it’s sent to the template and manually remove the row, but the extra_forms attribute doesn’t seem to be writeable and setting extra to 0 doesn’t make any difference. I could also have the JavaScript detect this case and remove the row. However I can’t help but think I’m missing something obvious since the behaviour I want is what would seem to be sensible expected behaviour to me.
Thanks.
I’ve come up with a solution that works with Django 1.1. I created a subclass of
BaseFormSetthat overrides thetotal_form_countmethod such that, if initial forms exist, the total does not include extra forms. Bit of a hack perhaps, and maybe there’s a better solution that I couldn’t find, but it works.