我有以下型号:
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)
然后通过标准视图进行处理。
现在我想知道,是否有一种方法可以使用基于类的视图(不适用于管理界面)来实现相同的目的(意思是:使用内联来添加/编辑属于帐单的项目)。
要点是:
FormSet
在
forms.py
内生成 inlineformset_factory
:BookImageFormSet = inlineformset_factory(BookForm, BookImage, extra=2)
BookPageFormSet = inlineformset_factory(BookForm, BookPage, extra=5)
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
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))
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()
if self.request.POST:
context['fs'] = MyInlineFS(self.request.POST, instance=self.object)
else:
context['fs'] = MyInlineFS(instance=self.object)
代码绝对没有准备好进行列表编辑,或者这里有一些黑魔法。 但我认为可以很快实施。
如果您查看 django.view.generic.edit (支持详细对象编辑)模块,它是如何使用 django.view.generic.detail 模块的。
我认为 django.view.generic.list_edit 模块可以使用 django.view.generic.list 和 django.view.generic.edit 的某些部分来实现。
,我相信我现在已经解决了。 inlineformset_factory 的第一个参数应该是 Book,不是 BookForm。
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
具有多个内联表单集的 Django 基于类的视图。
将所有内容复制到这里是没有意义的,但示例带有解释以及完整的模型、视图和模板源代码。