Джанго на основе классов с встроенной моделью или формированием
-
12-10-2019 - |
Вопрос
У меня есть следующие модели:
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
s внутриforms.py
с использованиемinlineformset_factory
:BookImageFormSet = inlineformset_factory(BookForm, BookImage, extra=2) BookPageFormSet = inlineformset_factory(BookForm, BookPage, extra=5)
вернул
FormSet
S в пределах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))
Другие советы
Я только что добавил свою собственную версию после проверки некоторых из этих готовых CBV. Мне особенно нужен контроль над multiple formsets -> one parent
В одном представлении каждый с отдельными функциями сохранения.
Я в основном заполнил привязку данных с помощью данных в 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
.
Я краснею общий исходный код 1,3-бета-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)
Код в ответе Иордании не сработал для меня. Я отправил Мой собственный вопрос об этом, что, я полагаю, я сейчас решил. Первым аргументом в inlineformset_factory должен быть книга, нет Книжная форма.
Мне нужно было внести еще одну модификацию для представления Джордана и Спека 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