私は次のモデルを持っています:
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)
次に、これを標準ビューで処理します。
クラスベースのビュー(管理インターフェイスではなく)を使用して、同じことを実現する方法があるかどうか(つまり、請求書に属するアイテムを追加/編集するためにインラインを使用する)かどうか疑問に思いました。
キーポイントは次のとおりです。
forms.py
を使用してinlineformset_factory
内でFormSet
sを生成しました:
BookImageFormSet = inlineformset_factory(BookForm, BookImage, extra=2)
BookPageFormSet = inlineformset_factory(BookForm, BookPage, extra=5)
views.py
のFormSet
クラス内のCreateView
sを返しました:
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))
これらの既製の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()
_
Django-extra-views を試してください。 CreateWithInlinesView
およびUpdateWithInlinesView
を探します。
ヨルダンの答えのコードは私にはうまくいきませんでした。 自分の質問 について投稿しましたが、今は解決したと思います。 inlineformset_factoryの最初の引数はBook、notBookFormである必要があります。
JordanのビューとSpeqのビューのget_context_data()
をもう1つ変更して、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.3-beta-1の一般的なソースコードを赤にします。
コードは絶対にリスト編集の準備ができていないか、ここにブラックマジックがあります。しかし、すぐに実装できると思います。
Django.view.generic.edit(詳細なオブジェクト編集をサポートする)モジュールを見ると、Django.view.generic.detailモジュールの使用方法がわかります。
Django.view.generic.list_editモジュールは、Django.view.generic.listとDjango.view.generic.editの一部を使用して実装できると思います。
Formset.is_valid()が機能するように、元のソリューションにいくつかの変更を加えました。
if self.request.POST:
context['fs'] = MyInlineFS(self.request.POST, instance=self.object)
else:
context['fs'] = MyInlineFS(instance=self.object)