使用ModelForm和ModelMultipleChoiceField保存ManyToManyFields

2 投票
2 回答
949 浏览
提问于 2025-04-17 04:14

我创建了一个基本的Django应用,里面有书籍、作者和出版社,都是按照《Django书籍》的内容做的。我想用ModelForm来修改已有的书籍,但遇到了问题:'authors'这个字段是一个多对多的关系,当我在ModelForm中选择一个选项时,它会把原来的选择清空,而不保存新的选择。

models.py

class Book(models.Model):
    title = models.CharField(max_length=100)
    authors = models.ManyToManyField2(Author)
    publisher = models.ForeignKey(Publisher)
    publication_date = models.DateField()

    def __unicode__(self):
        return self.title

class BookForm(ModelForm):
    class Meta:
        model = Book

    authors = forms.ModelMultipleChoiceField(queryset=Author.objects.all(), required=False)

views.py

def editBook(request, b=None):
    instance = None
    if b is not None:
        instance = Book.objects.get(title=b)
    if request.method == 'POST':
        form = BookForm(request.POST, instance=instance)
        if form.is_valid():
            form.save()
            return HttpResponseRedirect('/contact/thanks/')
    else:
        form = BookForm(instance=instance)
    return render_to_response('book_form.html', {'form':form})

谢谢!

补充 我刚刚找到几篇文章,建议在views.py中这样做。

authors = form.save(commit=False)
authors.user = request.user
authors.save()
form.save_m2m()

但还是没有成功,真是太难了!

2 个回答

0

因为你覆盖了 authors 这个字段。要么就别覆盖,让Django自己处理,要么就手动处理这个字段。

2

解决办法是重写ModelForm中的保存方法:

def save(self, commit=True):
    authors =[]
    for a in self.cleaned_data['authors']:
         authors.append(Author.objects.get(first_name=t.first_name))

    b = super(BookForm, self).save(commit=commit)

    for a in authors:
        b.authors.add(a)
    b.save()

撰写回答