3

我有一些看起来像这样的 Django 模型(这不是我的确切代码,而是一个具有相同结构的更简单示例):

class Player(models.Model):
    # Some fields here.
    pass

class Team(models.Model):
    players = models.ManyToManyField(Player, through='TeamPlayer')

class TeamPlayer(models.Model):
    team = models.ForeignKey(Team)
    player = models.ForeignKey(Player)
    some_other_field = models.BooleanField()

我正在使用该through机制,因为我的链接表上有额外的列。

我的管理类看起来像这样(请注意,我使用的是内联管理员来添加玩家):

class TeamPlayerInline(admin.TabularInline):
    model = TeamPlayer
    max_num = 11
    extra = 11

class TeamAdmin(admin.ModelAdmin):
    inlines = [TeamPlayerInline]

admin.site.register(Team, TeamAdmin)

问题:我的问题是,在我的管理员中,我想验证一个团队是否有 11 名球员。任何更少都应该导致错误。我怎样才能做到这一点?

这些是我尝试过的事情以及它们不起作用的原因:

  1. clean验证模型的方法中的玩家数量Team。这不起作用,因为玩家还没有被保存,所以对于一个新对象,总是有零个玩家。

  2. 验证a使用的clean_players方法中的数字。这个方法永远不会被调用。确实会调用其他非多对多字段的类似方法。ModelFormTeamAdmin

  3. 验证上述clean方法中的数字ModelForm。此方法被调用,但self.cleaned_data字典中没有'players'.

有什么想法可以实现这种类型的验证吗?我远不是 Django 专家,所以不要以为我必须做所有应该显而易见的事情。

4

1 回答 1

5

您需要formset在 TeamPlayerInline 上设置。并覆盖该表单集中的 clean 方法。例如:

from django.forms.models import BaseInlineFormSet

class TeamPlayerFormset(BaseInlineFormSet):
    def clean(self):
        """Check that exactly 11 players are entered."""
        super(TeamPlayerFormset, self).clean()

        if any(self.errors):
            return

        count = 0
        for cleaned_data in self.cleaned_data:
            if cleaned_data and not cleaned_data.get('DELETE', False):
                count += 1
        if count != 11:
            raise forms.ValidationError('You must enter 11 team players.')


class TeamPlayerInline(admin.TabularInline):
    model = TeamPlayer
    max_num = 11
    extra = 11
    formset = TeamPlayerFormset


class TeamAdmin(admin.ModelAdmin):
    inlines = [TeamPlayerInline]
于 2013-09-20T21:52:25.040 回答