7

我正在练习 Django 的 FormViews。

在这个应用程序中,我正在创建一个用于创建博客文章的 PostCreate 视图。

这是我的代码:

模型.py

class Post(models.Model):
    user = models.ForeignKey(User)
    post_title = models.CharField(max_length=200)
    post_content = models.CharField(max_length=500)

class Tag(models.Model):
    name = models.CharField(max_length=64, unique=True)
    posts = models.ManyToManyField(Post)

表格.py

class PostForm(forms.ModelForm):
    post_title = forms.CharField(
        label=u'Title',
        widget=forms.TextInput(attrs={'size':64})
    )
    post_content = forms.CharField(
        label=u'Content',
        widget=forms.TextInput(attrs={'size':128})
    )
    tags = forms.CharField(
        label=u'Tags',
        required=True,
        widget=forms.TextInput(attrs={'size':64})
    )
    class Meta:
        model = Post
        exclude = ('user', 'post_date')

视图.py

class PostCreate(CreateView):
    template_name = 'app_blog/post_save_form.html'
    model = Post
    form_class = PostForm

    def form_valid(self, form):
        self.object = form.save(commit=False)
        self.object.user = self.request.user

        tag_names = form.cleaned_data['tags'].split()
        for tag_name in tag_names:
            tag, dummy = Tag.objects.get_or_create(name=tag_name)
            self.object.tag_set.add(tag)

        return super(PostCreate, self).form_valid(form)

当我尝试将帖子(测试帖子 1)保存到数据库中时,它给了我一个错误:

在可以使用这种多对多关系之前,需要为字段“post”设置一个值。

看起来问题是由views.py中的行引起的,我在其中拆分了“tag”字段并将标签添加到post对象的tag_set字段。我想这个逻辑不应该包含在 form_valid 方法中......但我不知道应该在哪里实现......

任何帮助将不胜感激:)谢谢

4

2 回答 2

9

不,这不是问题所在。问题是您还没有保存 Post 对象,因此您无法与它创建多对多关系(因为在后台,m2m 只是一个链接表,其中包含两个实体的外键)。解决方法是先保存self.object

于 2013-07-26T03:42:57.737 回答
7

您应该在分配用户后保存您的对象,然后使用 m2m 关系执行操作:

self.object = form.save(commit=False)  # Not saved to database.
self.object.user = self.request.user
self.object.save()  # Saved.

tag_names = form.cleaned_data['tags'].split()
for tag_name in tag_names:
    tag, dummy = Tag.objects.get_or_create(name=tag_name)
    self.object.tag_set.add(tag)
于 2013-07-26T13:45:14.543 回答