2

请参阅以下Django 文档中的示例:

from django.db import models

class Author(models.Model):
    name = models.CharField(max_length=100)

class Book(models.Model):
    author = models.ForeignKey(Author)
    title = models.CharField(max_length=100)

如果您想创建一个允许您编辑属于特定作者的书籍的表单集,您可以这样做:

>>> from django.forms.models import inlineformset_factory
>>> BookFormSet = inlineformset_factory(Author, Book)
>>> author = Author.objects.get(name=u'Mike Royko')
>>> formset = BookFormSet(instance=author)

这是作者与书籍的一对多关系,并提供了一种为单个作者编辑书籍的简单方法。

现在我想做相反的事情:一个用于编辑/创建作者的表单,其中包含要编辑/创建的内联书籍表单集。我怎么做?我真的更喜欢用 ModelForms 做到这一点的方法,而无需创建所涉及模型之一的实例。

例如:

添加作者表格

  • 作者姓名:[________]
  • 图书:
    • 标题:[________]
    • 标题:[________]
    • 标题:[________]
    • (添加更多)
  • [save]

注意:此时我需要这个没有任何作者实例。

我可能在这里错过了一些简单的东西,而且我在这里的文档上花了太多时间。欢迎提供一点帮助!

4

1 回答 1

5

我会在同一视图中使用ModelFormforAuthorModelFormsetfor来执行此操作。Book像这样的东西:

# in forms.py:
AuthorForm = modelform_factory(Author)
BooksFormset = modelformset_factory(Book, extra=3, fields=('title',))

# in views.py:
if request.method == POST:
    author_form = AuthorForm(request.POST)
    books_formset = BooksFormset(request.POST)
    if author_form.is_valid() and books_formset.is_valid():
        author = author_form.save()
        new_books = books_formset.save(commit=False)
        for new_book in new_books:
            new_book.author = author
            new_book.save()
        # not actually needed with these models, but a good habit to include nonetheless
        books_formset.save_m2m()
        return redirect('some-success-view')
else:
    author_form = AuthorForm()
    books_formset = BooksFormset(queryset=Book.objects.none()) # or give a different initial queryset if you want some preselected choice
extra_context = {'author_form': author_form, 'books_formset': books_formset}
return render(request, 'some_template', extra_context)

内联表单集基本上为您完成作者实例和书籍实例之间的绑定。在内部,内联表单集使用基于您给它的父模型实例的查询集 - 我认为在instance=None代码get路径中传递并且如果作者表单未在其中验证post将起作用,但我从未测试过。

此外,我不会在这里处理“添加更多”行为 - 管理员从 Jquery 插件中获取该行为,并且应该很容易将其复制到您自己的表单中。

于 2013-07-07T21:17:25.527 回答