Джанго: Очистка и проверка форм, которые зависят друг от друга - PullRequest
5 голосов
/ 14 августа 2011

Документы django охватывают очистку и проверку ПОЛЕЙ, которые зависят друг от друга , но я не могу найти ничего, что охватывает формы, которые зависят друг от друга.

У меня есть одна HTML-форма, которая содержит как стандартную форму django, так и набор форм django. Надлежащая проверка каждой формы в наборе форм полностью условна на основе значения из основной формы (например, установите флажок в главной форме и определенное поле каждой формы в наборе форм вдруг становится обязательным).

Моя интуиция заключается в том, чтобы "просто" передать всю основную форму в вызов проверки формы, например:

def my_view(request):
    MyFormSet = formset_factory(MyForm, extra=2, can_order=True)

    if request.method == 'POST':
        form = MainForm(request.POST)
        formset = MyFormSet(request.POST)

        if form.is_valid() and formset.is_valid(form): # <-- ?!?!
            # The formset is now validated based on the form

Однако, чтобы это работало, я считаю, что мне придется переопределить оба набора форм is_valid() вместе с базовым методом is_valid() и clean(). Так что все становится довольно грязно и быстро.

Есть ли лучший способ сделать это?

Ответы [ 2 ]

8 голосов
/ 14 августа 2011

Я однажды исследовал, как делать что-то подобное, и этот урок http://yergler.net/blog/2009/09/27/nested-formsets-with-django/ оказался довольно полезным.

Еще один способ сделать это:

def my_view(request):
MyFormSet = formset_factory(MyForm, extra=2, can_order=True)

if request.method == 'POST':
    form = MainForm(request.POST)
    formset = MyFormSet(request.POST, other_form = form)

    if form.is_valid() and formset.is_valid(): # <-- ?!?!
        # The formset is now validated based on the form

Тогда

class MyFormSet(...):

   def __init__(self, *args, **kwargs):
       if kwargs.has_key('other_form'):
           self.myformforlater = kwargs.pop('other_form')
       Super(MyFormSet, self).__init__(*args, **kwargs)

Таким образом, вам нужно только переопределить метод init, и у вас есть доступ к внешней форме с любого шага проверки.

6 голосов
/ 15 августа 2011

Вот код, которым я закончил, используя ответ Теда (django 1.3):

class BaseMyFormSet(BaseFormSet):
    main_form = None

    def __init__(self, *args, **kwargs):
        # Save the main form until validation
        if kwargs.has_key('main_form'):
            self.main_form = kwargs.pop('main_form')

        super(BaseMyFormSet, self).__init__(*args, **kwargs)

    def clean(self):
        if any(self.errors):
            # Don't bother validating the formset unless each 
            # form is valid on its own
            return

        checkbox = self.main_form.cleaned_data['my_checkbox']

        if checkbox:
            for form in self.forms:
                # Do some extra validation


def my_view(request):
    MyFormSet = formset_factory(MyForm, extra=2, can_order=True,
        formset=BaseMyFormSet)

    if request.method == 'POST':
        form = MainForm(request.POST)
        formset = MyFormSet(request.POST, main_form=form)

        if form.is_valid() and formset.is_valid():
            # The formset is now validated based on the form
...