Django ModelForm外键过滤

时间:2018-01-06 09:50:22

标签: python django django-forms

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

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 %}

过滤&#39;专辑&#39;的查询集字段工作正常。它仅显示与经过身份验证的用户关联的相册,但是当我在表单中单击提交时,浏览器不会将我重定向到成功URL并且不会将歌曲添加到数据库中。我在views.py的末尾添加了两个print语句来检查表单是否有效并且有任何错误。当form.errors返回空字典时,form.is_valid()等于 False 。为什么Django将此表格视为无效,如果它没有任何错误?

注意:当我在views.py中注释掉 init 功能并在views.py中注释 get_form 功能时,SongCreateView可以正常运行。

2 个答案:

答案 0 :(得分:1)

首先,在print语句之后删除那些return语句,因为在方法返回值之后它们永远不会被执行。其次,将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())

从顶部的redirect导入django.shortcuts。希望它有所帮助!

答案 1 :(得分:0)

我猜是宋的ID丢失了。当你使用魔术代码

{{ form }}

虽然隐藏了输入,但会自动添加此字段。当您手动定义所有字段时,您必须确保添加ID,即需要添加

{{ form.pk }}

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