如何在Django管理界面中过滤filter_horizontal字段?

19

我正在寻找一种在筛选过的查询集基础上使用filter_horizontal的方法。

我尝试使用自定义管理器来实现:

在models.py中:

class AvailEquipManager(models.Manager):
    def get_query_set(self):
        return super(AvailEquipManager, self).get_query_set().filter(id=3)

class Equipment(models.Model):
    description = models.CharField(max_length=50)
    manufacturer = models.ForeignKey(Manufacturer)
    [...]
    objects = models.Manager()
    avail = AvailEquipManager()

    def __unicode__(self):
        return u"%s" % (self.description)

在 admin.py 文件中:

class SystemAdmin(admin.ModelAdmin):
    filter_horizontal = ('equipment',) # this works but obviously shows all entries
    #filter_horizontal = ('avail',)     # this does not work

那么问题是,我如何减少filter_horizontal左侧显示的内容,只显示特定的项目?


3
https://dev59.com/Bpvga4cB1Zd3GeqPzkzj#39835254 - Michael Kalika
3个回答

29

我通过调整在Google Groups中找到的解答来解决了问题。

使用自定义的ModelForm可以解决:

新建一个forms.py文件:

from django import forms
from models import Equipment

class EquipmentModelForm(forms.ModelForm):
    class Meta:
        model = Equipment

    def __init__(self, *args, **kwargs):
        forms.ModelForm.__init__(self, *args, **kwargs)
        self.fields['equipment'].queryset = Equipment.avail.all()

然后在 admin.py 文件中:

class SystemAdmin(admin.ModelAdmin):
    form = EquipmentModelForm
    filter_horizontal = ('equipment',) 

希望这能在某个时候帮助到其他人。


如果您需要访问模型实例(使用表单self.instance),那么这将非常有用。如果您需要访问请求(例如会话用户),则可能更喜欢在管理员中覆盖formfield_for_manytomany,如此答案所示。 - vlz

5

虽然这是一个旧问题,但根据您的具体要求,设置equipment关系字段上的limit_choices_to属性可能更容易。此方法适用于ForeignKey字段和ManyToManyField

根据Django文档,此属性:

在使用ModelFormadmin渲染此字段时,为此字段设置可用选项的限制...

下面是一个最简示例,假设您有一个带有equipment多对多字段的System模型:

class Equipment(models.Model):
    available = models.BooleanField(default=True)


class System(models.Model):
    equipment = models.ManyToManyField(to=Equipment,
                                       limit_choices_to={'available': True})


class SystemAdmin(admin.ModelAdmin):
    filter_horizontal = ['equipment']

这里使用了一个available标志,但也可以使用更复杂的查询。


1
在ModelAdmin中,formfield_for_foreignkey方法可以处理此类情况:Django 文档 下面是使用System和外键关联到 Equipment的示例:
def formfield_for_foreignkey(self, db_field, request, **kwargs):
    if db_field.name == 'equipment':
        system = System.objects.get(id=request.resolver_match.args[0])
        kwargs["queryset"] = system.equipment_set.all()
    return super().formfield_for_foreignkey(db_field, request, **kwargs)

或者另一种选择是 formfield_for_manytomany() - vlz

网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接