27

我有一个想要内联的 Django 模型字段。字段是多对多的关系。所以有“项目”和“用户配置文件”。每个用户配置文件都可以选择任意数量的项目。

目前,我已经让“表格”内联视图正常工作。有没有办法拥有“水平过滤器”,以便我可以轻松地从用户配置文件中添加和删除项目?

请参阅附图中的示例。在此处输入图像描述

这是用户配置文件的模型代码:

class UserProfile(models.Model):
    user = models.OneToOneField(User, unique=True)
    projects = models.ManyToManyField(Project, blank=True, help_text="Select the projects that this user is currently working on.")

以及项目的模型代码:

class Project(models.Model):
    name = models.CharField(max_length=100, unique=True)
    application_identifier = models.CharField(max_length=100)
    type = models.IntegerField(choices=ProjectType)
    account = models.ForeignKey(Account)
    principle_investigator = models.ForeignKey(User)
    active = models.BooleanField()

以及视图的管理代码:

class UserProfileInline(admin.TabularInline):
    model = UserProfile.projects.through
    extra = 0
    verbose_name = 'user'
    verbose_name_plural = 'users'

class ProjectAdmin(admin.ModelAdmin):
    list_display = ('name', 'application_identifier', 'type', 'account', 'active')
    search_fields = ('name', 'application_identifier', 'account__name')
    list_filter = ('type', 'active')
    inlines = [UserProfileInline,]
admin.site.register(Project, ProjectAdmin)
4

3 回答 3

52

问题不在于内联;一般来说,它来自ModelForm工作方式。他们只为模型上的实际字段构建表单字段,而不是相关的经理属性。但是,您可以将此功能添加到表单中:

from django.contrib.admin.widgets import FilteredSelectMultiple

class ProjectAdminForm(forms.ModelForm):
    class Meta:
        model = Project

    userprofiles = forms.ModelMultipleChoiceField(
        queryset=UserProfile.objects.all(),
        required=False,
        widget=FilteredSelectMultiple(
            verbose_name='User Profiles',
            is_stacked=False
        )
    )

    def __init__(self, *args, **kwargs):
        super(ProjectAdminForm, self).__init__(*args, **kwargs)
            if self.instance.pk:
                self.fields['userprofiles'].initial = self.instance.userprofile_set.all()

    def save(self, commit=True):
        project = super(ProjectAdminForm, self).save(commit=False)  
        if commit:
            project.save()

        if project.pk:
            project.userprofile_set = self.cleaned_data['userprofiles']
            self.save_m2m()

        return project

class ProjectAdmin(admin.ModelAdmin):
    form = ProjectAdminForm
    ...

一个小演练可能是为了。首先,我们定义一个userprofiles表单域。它将使用 a ModelMultipleChoiceField,默认情况下会产生一个多选框。由于这不是模型上的实际字段,我们不能只将其添加到 中filter_horizontal,因此我们改为告诉它简单地使用相同的小部件,FilteredSelectMultiple如果它在 中列出,它将使用filter_horizontal

我们最初将查询集设置为整个UserProfile集合,您不能在这里过滤它,但是,因为在类定义的这个阶段,表单还没有被实例化,因此还没有instance设置它。因此,我们重写__init__,以便我们可以将过滤后的查询集设置为字段的初始值。

最后,我们重写该save方法,以便我们可以将相关管理器的内容设置为与表单的 POST 数据中的内容相同,您就完成了。

于 2012-07-25T20:36:40.290 回答
3

在处理与其自身的多对多关系时的一个小补充。人们可能希望将自己排除在选择之外:

if self.instance.pk:
        self.fields['field_being_added'].queryset = self.fields['field_being_added'].queryset.exclude(pk=self.instance.pk)
        self.fields['field_being_added'].initial = """Corresponding result queryset"""
于 2014-04-30T22:07:22.473 回答
2

有一个更简单的解决方案,只需添加filter_horizontal如下所述:

class YourAdmin(ModelAdmin)
    filter_horizontal = ('your_many_to_many_field',)

前:在此处输入图像描述

后:在此处输入图像描述

于 2020-09-10T12:25:47.487 回答