我正在尝试过滤模型表单中的外键字段选择,但表单不起作用 . 我的脚本:
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 回答
我的猜测是歌曲的ID丢失了 . 当你使用魔术代码时
尽管输入是隐藏的,但此字段会自动添加 . 当您手动定义所有字段时,您必须确保添加ID,这是您需要添加的
或者你的歌的id是什么名字(这里是pk) .
首先,在
return
语句之后删除那些print
语句,因为在方法返回值之后它们永远不会被执行 . 其次,将form_class
属性添加到您的视图类,如下所示:然后在
get_form
方法:然后重写
form_valid
方法以将登录用户与Song
实例关联,如下所示:从顶部的
django.shortcuts
导入redirect
. 希望能帮助到你!