основанные на классах django представления со встроенной моделью-формой или формой - PullRequest
54 голосов
/ 21 декабря 2010

У меня есть следующие модели:

class Bill(models.Model):
    date = models.DateTimeField(_("Date of bill"),null=True,blank=True)

class Item(models.Model):
    name = models.CharField(_("Name"),max_length=100)
    price = models.FloatField(_("Price"))
    quantity = models.IntegerField(_("Quantity"))
    bill = models.ForeignKey("Bill",verbose_name=_("Bill"),
                             related_name="billitem")

Я знаю, что это возможно:

from django.forms.models import inlineformset_factory
inlineformset_factory(Bill, Item)

, а затем обработаю это в стандартном представлении.

Теперь яИнтересно, есть ли способ добиться того же (имеется в виду: использование встроенного для добавления / редактирования элементов, принадлежащих счету) с использованием представлений на основе классов (не для интерфейса администратора).

Ответы [ 7 ]

61 голосов
/ 25 июля 2011

Ключевые моменты:

  1. сгенерировано FormSet с в forms.py с использованием inlineformset_factory:

    BookImageFormSet = inlineformset_factory(BookForm, BookImage, extra=2)
    BookPageFormSet = inlineformset_factory(BookForm, BookPage, extra=5)
    
  2. вернул FormSet в пределах класса CreateView в views.py:

    def get_context_data(self, **kwargs):
        context = super(BookCreateView, self).get_context_data(**kwargs)
        if self.request.POST:
            context['bookimage_form'] = BookImageFormSet(self.request.POST)
            context['bookpage_form'] = BookPageFormSet(self.request.POST)
        else:
            context['bookimage_form'] = BookImageFormSet()
            context['bookpage_form'] = BookPageFormSet()
        return context
    
  3. Используется form_valid для сохранения формы и набора форм:

     def form_valid(self, form):
         context = self.get_context_data()
         bookimage_form = context['bookimage_formset']
         bookpage_form = context['bookpage_formset']
         if bookimage_form.is_valid() and bookpage_form.is_valid():
             self.object = form.save()
             bookimage_form.instance = self.object
             bookimage_form.save()
             bookpage_form.instance = self.object
             bookpage_form.save()
             return HttpResponseRedirect('thanks/')
         else:
             return self.render_to_response(self.get_context_data(form=form))
    
14 голосов
/ 11 августа 2012

Я только что добавил свою собственную версию после проверки некоторых из этих готовых CBV. Мне особенно нужен был контроль над multiple formsets -> one parent в одном представлении, каждый с отдельными функциями сохранения.

Я в основном вставил привязку данных FormSet в функцию get_named_formsets, которая вызывается get_context_data и form_valid.

Там я проверяю, все ли наборы форм действительны, а также ищу метод, который переопределяет старый formset.save() для каждого набора форм для индивидуального сохранения.

Шаблон отображает формы через

{% with named_formsets.my_specific_formset as formset %}
 {{ formset }}
 {{ formset.management_form }}
{% endwith %}

Думаю, я буду регулярно пользоваться этой системой.

class MyView(UpdateView): # FormView, CreateView, etc
  def get_context_data(self, **kwargs):
        ctx = super(MyView, self).get_context_data(**kwargs)
        ctx['named_formsets'] = self.get_named_formsets()
        return ctx

    def get_named_formsets(self):
        return {
            'followup': FollowUpFormSet(self.request.POST or None, prefix='followup'),
            'action': ActionFormSet(self.request.POST or None, prefix='action'),
        }

    def form_valid(self, form):
        named_formsets = self.get_named_formsets()
        if not all((x.is_valid() for x in named_formsets.values())):
            return self.render_to_response(self.get_context_data(form=form))

        self.object = form.save()

        # for every formset, attempt to find a specific formset save function
        # otherwise, just save.
        for name, formset in named_formsets.items():
            formset_save_func = getattr(self, 'formset_{0}_valid'.format(name), None)
            if formset_save_func is not None:
                formset_save_func(formset)
            else:
                formset.save()
        return http.HttpResponseRedirect('')

    def formset_followup_valid(self, formset):
        """
        Hook for custom formset saving.. useful if you have multiple formsets
        """
        followups = formset.save(commit=False) # self.save_formset(formset, contact)
        for followup in followups:
            followup.who = self.request.user
            followup.contact = self.object
            followup.save()
8 голосов
/ 03 ноября 2013

Вы должны попробовать django-extra-views . Ищите CreateWithInlinesView и UpdateWithInlinesView.

1 голос
/ 30 января 2013

Мне нужно было сделать еще одну модификацию представления Иордана и Спека get_context_data(), чтобы formset.non_form_errors существовало в контексте шаблона.

...
if self.request.POST:
    context['fs'] = MyInlineFS(self.request.POST, instance=self.object)
    context['fs'].full_clean()  # <-- new
else:
    context['fs'] = MyInlineFS(instance=self.object)
return context
1 голос
/ 26 июня 2012

Код в ответе Джордана не сработал для меня. Я отправил мой собственный вопрос об этом, который, я думаю, я уже решил. Первый аргумент inlineformset_factory должен быть Book, , а не BookForm.

1 голос
/ 22 октября 2011

Я внес некоторые изменения в исходное решение, чтобы позволить formset.is_valid () работать:

    if self.request.POST:
        context['fs'] = MyInlineFS(self.request.POST, instance=self.object)
    else:
        context['fs'] = MyInlineFS(instance=self.object)
1 голос
/ 24 января 2011

Я переделал общий исходный код 1.3-бета-1:

Код абсолютно не готов к редактированию списка, или здесь есть какая-то черная магия.Но я думаю, что это может быть реализовано быстро.

Если вы посмотрите на модуль django.view.generic.edit (который поддерживает детальное редактирование объектов), как он использует модуль django.view.generic.detail.

Я думаю, что модуль django.view.generic.list_edit можно реализовать с использованием django.view.generic.list и некоторой части из django.view.generic.edit.

...