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
, который может фильтровать как пустые строки, так и нули, в зависимости от того, что поле позволяет хранить:
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
).