Django基于类的视图与内联模型表单或表单集

60

我有以下模型:

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个回答

68

重点如下:

  1. 使用 inlineformset_factoryforms.py 中生成了 FormSet

    BookImageFormSet = inlineformset_factory(BookForm, BookImage, extra=2)
    BookPageFormSet = inlineformset_factory(BookForm, BookPage, extra=5)
    
  2. views.py中的CreateView类中返回了FormSets:

  3. 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
    
  4. 使用form_valid保存表单和表单集:

  5.  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))
    

3
我认为应该是 self.object,而不是 self.instance - orokusaki
5
“form_valid”似乎不是最合适的位置,因为它通常在表单验证完成后调用。我认为覆盖“post”会是更清晰的解决方案。 - cvk
3
虽然略有些晚,但这对我很起作用,@orokusaki关于使用self.object而不是self.instance是正确的。 - edhedges
4
尽管我认为这个解决方案是可行的,但在get_context_data()中检查请求方法好像没有太多意义。根据Django源代码中的ProcessFormView,你应该在get()post()方法中实例化表单并将它们添加到get_context_data(form=form)中。 - Soviut
5
在例子中,当表单无效时(参见 self.get_context_data(form=form)),会重新创建表单集。这并不是问题,直到您的表单集中出现非表单错误,这些错误会在调用 formset.is_valid()(访问 formset.errors 导致实例化错误列表)时出现。解决方法是重用表单集(将其传递给 get_context_data 并在 get_context_data 中检查是否提供了它,只有在没有提供时才添加它...或者像之前的评论建议的那样使用 post,自然而然地导致重复使用表单集)。 - orokusaki
显示剩余8条评论

19

我在查看了一些预先制作的类视图后,添加了自己的版本。 我特别需要在单个视图中对多个表单集合 -> 一个父级进行控制,每个表单集合都有独立的保存功能。

我基本上将FormSet数据绑定到get_named_formsets函数中,该函数由get_context_dataform_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()

3
非常干净,我喜欢它 :) - Natim

12

你应该尝试使用django-extra-views。寻找CreateWithInlinesViewUpdateWithInlinesView


似乎在 Django 2.2 中出现了故障。 - alias51

2

我对原始解决方案进行了一些修改,以使formset.is_valid()能够正常工作:

    if self.request.POST:
        context['fs'] = MyInlineFS(self.request.POST, instance=self.object)
    else:
        context['fs'] = MyInlineFS(instance=self.object)

是的,你是英雄。没有这个,删除和更新都无法正常工作。 - tread

1

我需要对Jordan和Speq的视图中的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

我阅读了1.3-beta-1的通用源代码:

这个代码绝对没有准备好进行列表编辑,或者这里有一些黑魔法。但我认为它可以很快实现。

如果您查看django.view.generic.edit(支持详细对象编辑)模块如何使用django.view.generic.detail模块。

我认为可以使用django.view.generic.list和从django.view.generic.edit中获取的一些部分来实现django.view.generic.list_edit模块。


是的,我读了那部分。虽然它并不是“自动”内联的答案,因为它在那里声明:“可以用于显示对象列表的混合”。我想添加/编辑/删除内联,而不是显示它们 ;) - Hixi
Hixi在想DJANGO-1.3和使用基于类的视图是否有可用的功能。我阅读了django源代码,这个功能显然没有被实现(在django 1.3 beta 1中)。我提供了一些提示来帮助实现,但并非完整的开发。 - VGE

1
Jordan的答案中的代码对我没有用。我发了自己的问题,我相信现在已经解决了。inlineformset_factory的第一个参数应该是Book,而不是BookForm。<\p>

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接