首页 文章

Django ModelForm外键过滤

提问于
浏览
0

我正在尝试过滤模型表单中的外键字段选择,但表单不起作用 . 我的脚本:

forms.py

from django import forms
from .models import Album, Song

class SongCreateForm(forms.ModelForm):

    class Meta:
        model = Song
        fields = [
            'album',
            'name',
            'is_favorite'
        ]
        widgets = {
            'album': forms.Select(attrs={'class': 'form-control'}),
            'name': forms.TextInput(attrs={'class': 'form-control'}),
            'is_favorite': forms.CheckboxInput(attrs={'class': 'form-check-input'}),
        }

    def __init__(self, user, *args, **kwargs):
        super(SongCreateForm, self).__init__(*args, **kwargs)
        self.fields['album'].queryset = Album.objects.filter(owner=user)

views.py

from django.views.generic import CreateView

class SongCreateView(CreateView):
    template_name = 'music/song_create.html'
    success_url = '/songs/'

    def get_form(self, form_class=None):
        form_class = SongCreateForm(user=self.request.user)
        return form_class
        print(form_class.errors.as_data())
        print(form_class.is_valid())

song_create.html

{% extends 'base.html' %}

{% block content %}
<form method="post">{% csrf_token %}
  {% if form.errors %}
    <p>{{ form.errors }}</p>
  {% endif %}
  <div class="form-group">
    <label for="{{ form.album.id_for_label }}">Album</label>
    {{ form.album }}
  </div>
  <div class="form-group">
    <label for="{{ form.name.id_for_label }}">Name</label>
    {{ form.name }}
  </div>
  <div class="form-check">
    <label for="{{ form.is_favorite.id_for_label }}" class="form-check-label">
    {{ form.is_favorite }}Favorite
    </label>
  </div>
  <button type="submit" class="btn btn-primary" value="Save">Add</button>
</form>
{% endblock %}

过滤'album'字段的查询集正常工作 . 它只显示与经过身份验证的用户关联的相册,但是当我在表单中单击“提交”时,不会将数据添加到数据库中 . 我在views.py的末尾添加了两个print语句来检查表单是否有效并且有任何错误 . 当form.errors返回 empty dictionary 时,form.is_valid()等于 False . 如果Django没有任何错误,为什么将此表单视为无效?

Note: 当我在views.py中的forms.py和 get_form 函数中注释掉 init 函数时,SongCreateView成功运行 .

2 回答

  • 0

    我的猜测是歌曲的ID丢失了 . 当你使用魔术代码时

    {{ form }}
    

    尽管输入是隐藏的,但此字段会自动添加 . 当您手动定义所有字段时,您必须确保添加ID,这是您需要添加的

    {{ form.pk }}
    

    或者你的歌的id是什么名字(这里是pk) .

  • 1

    首先,在 return 语句之后删除那些 print 语句,因为在方法返回值之后它们永远不会被执行 . 其次,将 form_class 属性添加到您的视图类,如下所示:

    class SongCreateView(CreateView):
        template_name = 'music/song_create.html'
        success_url = '/songs/'
        form_class = SongCreateForm
    

    然后在 get_form 方法:

    def get_form(self, form_class=None):
        if form_class is None:
            form_class = self.get_form_class()
        return form_class(user=self.request.user, **self.get_form_kwargs())
    

    然后重写 form_valid 方法以将登录用户与 Song 实例关联,如下所示:

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

    从顶部的 django.shortcuts 导入 redirect . 希望能帮助到你!

相关问题