在 Django 模板中隐藏来自 ModelForms 的外键字段

Hiding foreign key fields from ModelForms in Django Template

精华在下面。 我有几本不同的杂志。模板中的 AutoForm 显示所有文章和所有作者以供选择(不仅来自当前杂志)。如何限制作者 a 的选择仅限于当前杂志的文章?我可以仅使用模板来执行此操作吗?如果没有,那又如何?

models.py

class Magazine(models.Model):
    Name = models.CharField(max_length=200)

class Article(models.Model):
    Name = models.CharField(max_length=200)
    Magazine = models.ForeignKey(Magazine)

class Author(models.Model):
    Name = models.CharField(max_length=200)
    Magazine = models.ForeignKey(Magazine)

class Sets(models.Model):
    Name = models.CharField(max_length=200)
    Magazine = models.ForeignKey(Magazine)
    Aut = models.ManyToManyField(Author, null=True, blank=True)
    Set = models.ForeignKey(Article, null=True, blank=True)

forms.py:

class MagazineForm(ModelForm):
    class Meta:
        model = Magazine
        fields = {'Name'}

class ArticleForm(ModelForm):
    class Meta:
        model = Article
        fields = {'Name'}

class AuthorForm(ModelForm):
    class Meta:
        model = Author
        fields = {'Name'}

class SetsForm(ModelForm):
    class Meta:
        model = Sets
        fields = {'Name', 'Set', 'Aut'}

views.py

def building_details(request, magazine_id):
    sets_form = SetsForm
    args = {}
    args.update(csrf(request))
    args['magazine'] = Magazine.objects.get(id=magazine_id)
    args['article'] = Article.objects.filter(Magazine=magazine_id)
    args['author'] = Author.objects.filter(Magazine=magazine_id)
    args['sets'] = Sets.objects.filter(Magazine=magazine_id)
    args['setform'] = sets_form
    return render_to_response('building.html', args)

模板

<form action='/add/{{ magazine.id }}/' method='post'>
{% csrf_token %}
{{ setform }}
<input type='submit' class='button' value="Добавить">
</form>
</div>

也许这个词条的某处有错误(我简单编辑了一下,删除了这里没有发挥作用,本质上)。总的来说,我一切正常,除了显示所有对象,而不是只有当前对象。

听起来您正试图根据某些条件限制外键选择器中的选项。

这意味着您需要覆盖表单 __init__ 方法中的一些细节:

class SetsForm(ModelForm):
    class Meta:
        model = Sets
        fields = {'Name', 'Set', 'Aut'}

    def __init__(self, *args, **kwargs):
        articles = Article.objects.filter(Magazine=kwargs.pop('magazine_id'))
        super(SetsForm, self).__init__(*args, **kwargs)
        self.fields['articles'] = forms.ModelChoiceField(
            queryset=articles ,
            label="Articles",
        )

然后像这样调用您的表单:

sets_form = SetsForm(magazine_id=magazine_id)

仅获取当前杂志文章的作者:

articles = Article.objects.filter(Magazine=magazine_id)
sets = Sets.objects.filter(Magazine=magazine_id, Set__in=articles)

import itertools
# build a flat list of all Authors in these sets (and call `set` to avoid duplicates)
authors = set(itertools.chain(*[s.Aut.all() for s in sets]))

如果 Author to Article 有一个直接的外键,这会更容易。

此外,您应该遵循 Django 约定,使用小写字段名称。这将避免使用大写字母的字段和模型之间的混淆。