Фильтры списка ModelAdmin¶
Классы ModelAdmin могут определять фильтры списков, которые появляются на правой боковой панели страницы списка изменений администратора, как показано на следующем снимке экрана:
Чтобы активировать фильтрацию по полю, установите ModelAdmin.list_filter в список или кортеж элементов, где каждый элемент относится к одному из следующих типов:
Имя поля.
Подкласс django.contrib.admin.SimpleListFilter.
Двухкортеж, содержащий имя поля и подкласс django.contrib.admin.FieldListFilter.
См. примеры ниже для обсуждения каждого из этих вариантов определения list_filter.
Использование имени поля¶
Самый простой вариант — указать нужные имена полей из вашей модели.
Каждое указанное поле должно быть либо BooleanField, CharField, DateField, DateTimeField, IntegerField, ForeignKey или ManyToManyField, например:
class PersonAdmin(admin.ModelAdmin):
list_filter = ["is_staff", "company"]
Поле в list_filter может указывать и на связанный объект используя __, например:
class PersonAdmin(admin.UserAdmin):
list_filter = ["company__name"]
Использование SimpleListFilter¶
Для пользовательской фильтрации вы можете определить свой собственный фильтр списка, создав подкласс django.contrib.admin.SimpleListFilter. Вам необходимо предоставить атрибуты title и parameter_name, а также переопределить методы lookups и queryset, например:
from datetime import date
from django.contrib import admin
from django.utils.translation import gettext_lazy as _
class DecadeBornListFilter(admin.SimpleListFilter):
# Human-readable title which will be displayed in the
# right admin sidebar just above the filter options.
title = _("decade born")
# Parameter for the filter that will be used in the URL query.
parameter_name = "decade"
def lookups(self, request, model_admin):
"""
Returns a list of tuples. The first element in each
tuple is the coded value for the option that will
appear in the URL query. The second element is the
human-readable name for the option that will appear
in the right sidebar.
"""
return [
("80s", _("in the eighties")),
("90s", _("in the nineties")),
]
def queryset(self, request, queryset):
"""
Returns the filtered queryset based on the value
provided in the query string and retrievable via
`self.value()`.
"""
# Compare the requested value (either '80s' or '90s')
# to decide how to filter the queryset.
if self.value() == "80s":
return queryset.filter(
birthday__gte=date(1980, 1, 1),
birthday__lte=date(1989, 12, 31),
)
if self.value() == "90s":
return queryset.filter(
birthday__gte=date(1990, 1, 1),
birthday__lte=date(1999, 12, 31),
)
class PersonAdmin(admin.ModelAdmin):
list_filter = [DecadeBornListFilter]
Примечание
Для удобства объект HttpRequest передается в методы lookups и queryset, например:
class AuthDecadeBornListFilter(DecadeBornListFilter):
def lookups(self, request, model_admin):
if request.user.is_superuser:
return super().lookups(request, model_admin)
def queryset(self, request, queryset):
if request.user.is_superuser:
return super().queryset(request, queryset)
Также объект ModelAdmin передается в метод lookups. Например, вы можете использовать существующие данные при создании фильтра:
class AdvancedDecadeBornListFilter(DecadeBornListFilter):
def lookups(self, request, model_admin):
"""
Only show the lookups if there actually is
anyone born in the corresponding decades.
"""
qs = model_admin.get_queryset(request)
if qs.filter(
birthday__gte=date(1980, 1, 1),
birthday__lte=date(1989, 12, 31),
).exists():
yield ("80s", _("in the eighties"))
if qs.filter(
birthday__gte=date(1990, 1, 1),
birthday__lte=date(1999, 12, 31),
).exists():
yield ("90s", _("in the nineties"))
Использование имени поля и явного FieldListFilter¶
Наконец, если вы хотите указать явный тип фильтра для использования с полем, вы можете предоставить элемент list_filter в виде двухкортежа, где первый элемент — это имя поля, а второй элемент — это класс, унаследованный от django.contrib.admin.FieldListFilter, например:
class PersonAdmin(admin.ModelAdmin):
list_filter = [
("is_staff", admin.BooleanFieldListFilter),
]
Здесь поле is_staff будет использовать BooleanFieldListFilter. Указав только имя поля, в большинстве случаев поля будут автоматически использовать соответствующий фильтр, но этот формат позволяет вам контролировать используемый фильтр.
В следующих примерах показаны доступные классы фильтров, для использования которых необходимо дать согласие.
Теперь вы можете ограничить фильтр значениями, которые встречаются в связи, используя RelatedOnlyFieldListFilter:
class BookAdmin(admin.ModelAdmin):
list_filter = [
("author", admin.RelatedOnlyFieldListFilter),
]
Предполагая, что author является ForeignKey для модели User, это ограничит выбор list_filter пользователями, написавшими книгу, вместо перечисления всех пользователей.
Вы можете фильтровать пустые значения, используя EmptyFieldListFilter, который может фильтровать как пустые строки, так и значения NULL, в зависимости от того, что поле позволяет хранить:
class BookAdmin(admin.ModelAdmin):
list_filter = [
("title", admin.EmptyFieldListFilter),
]
Определив фильтр с помощью поиска __in, можно фильтровать любую группу значений. Вам необходимо переопределить метод «expected_parameters» и указать атрибут «lookup_kwargs» с соответствующим именем поля. По умолчанию несколько значений в строке запроса будут разделены запятыми, но это можно настроить с помощью атрибута list_separator. В следующем примере показан такой фильтр, использующий в качестве разделителя символ вертикальной черты:
class FilterWithCustomSeparator(admin.FieldListFilter):
# custom list separator that should be used to separate values.
list_separator = "|"
def __init__(self, field, request, params, model, model_admin, field_path):
self.lookup_kwarg = "%s__in" % field_path
super().__init__(field, request, params, model, model_admin, field_path)
def expected_parameters(self):
return [self.lookup_kwarg]
Примечание
Поле GenericForeignKey не поддерживается.
Фильтры списков обычно отображаются только в том случае, если фильтр имеет более одного варианта выбора. Метод has_output() фильтра определяет, будет ли он отображаться.
Вы можете указать собственный шаблон для отображения фильтра:
class FilterWithCustomTemplate(admin.SimpleListFilter):
template = "custom_template.html"
Конкретный пример можно найти в шаблоне приложения интерфейса администратора, который использует Django (admin/filter.html).
Фасеты¶
По умолчанию счетчики для каждого фильтра, известные как фасеты, можно отобразить, включив их через интерфейс администратора. Эти счетчики будут обновляться в соответствии с примененными в данный момент фильтрами. Дополнительную информацию см. в ModelAdmin.show_facets.