根据外键中的字段过滤Django admin中的list_filter

Mat*_*ore 4 django django-admin

我想通过外键指向的表中的字段过滤我的list_filters之一.

我的模特:

class Organisation(models.Model):
    name = models.CharField()
    COMPANY = 'COMPANY'
    CHARITY = 'CHARITY'
    ORG_CHOICES = (
        (COMPANY, 'COMPANY'),
        (CHARITY, 'CHARITY'),
        )
    type = models.CharField(choices = ORG_CHOICES)

class Issue(models.Model):
   name = models.CharField
   charity = models.ForeignKey(Organisation)
Run Code Online (Sandbox Code Playgroud)

我想放入IssueAdmin:

list_filter = (charity)
Run Code Online (Sandbox Code Playgroud)

并为此提供慈善机构清单.目前,它只列出了组织模型中的所有内容,包括慈善机构和公司.例如,我现在在过滤器中获取此列表:

oxfam
yamaha
greenpeace
microsoft
Run Code Online (Sandbox Code Playgroud)

当我想要一个列出的过滤器:

oxfam
greenpeace
Run Code Online (Sandbox Code Playgroud)

我可以通过将组织表分成两个表(慈善机构和公司)来解决这个问题,但这感觉不对.

似乎SimpleListFilter应该可以工作,但到目前为止我还没有运气.基本上我想要使用以下过滤器并返回过滤器的慈善机构列表:

Organisation.objects.filter(type = 'CHARITY')
Run Code Online (Sandbox Code Playgroud)

我(差)尝试过滤器:

class CharityFilter(SimpleListFilter):
    title = _('Charity')
    parameter = _('charity__type')
    def lookups(self, request, model_admin):
        return Organisation.objects.filter(type = 'CHARITY')

    def queryset(self, request, queryset):
        if not self.value() is not None:
            return queryset.filter(type = 'CHARITY')
        else:
            return queryset
Run Code Online (Sandbox Code Playgroud)

谁能帮我?

bru*_*ers 5

为什么不直接过滤类型,即:

class OrganisationAdmin(admin.ModelAdmin):
    list_filter = ("type", )
    # etc
Run Code Online (Sandbox Code Playgroud)

它允许您拥有所有组织,仅限公司或慈善机构.

否则,如果您真的想要将"慈善机构"和"公司"视为具有不同管理员的不同实体,但仍将它们保存在同一个表中,您可以使用代理模型,但我不明白这一点.

编辑:根据组织类型过滤问题,它与在中继模型上查询的工作方式相同(不是一个惊喜,因为它正是发生的事情)

class IssueAdmin(admin.ModelAdmin):
    list_filter =("organisation__type", )
Run Code Online (Sandbox Code Playgroud)

  • 与我编辑的答案相同,效果相同. (2认同)

cai*_*nka 5

如果我的理解正确,您将尝试不是按问题实例所属的组织类型,而是按组织实例本身来过滤问题实例,因此您将在右侧看到一个可能很长的组织列表来过滤问题开,但您只希望慈善机构出现在此列表中。

如果组织有很多,这可能不是最好的解决方案,但前提是组织很少......

无论如何,根据 Django 文档:https://docs.djangoproject.com/en/dev/ref/contrib/admin/ Filter 的 Lookups() 方法必须返回元组列表(coded_value,displayed_value)。

以下未经测试的代码应该适用于您的情况:

class CharityFilter(SimpleListFilter):
    title = _('Charity')
    parameter_name = 'charity'
    def lookups(self, request, model_admin):
        queryset = model_admin.queryset(request).filter(organisation__type='CHARITY')
        return queryset.values_list('organisation__pk','organisation__name').order_by('organisation__name')

    def queryset(self, request, queryset):
        if self.value():
          return queryset.filter(organisation=self.value())
Run Code Online (Sandbox Code Playgroud)

  • 效果还是很好,谢谢!但对于使用现代版本 django 的任何人来说,将 model_admin.queryset 更改为 model_admin.get_queryset (2认同)