Подтвердить что ты не робот

Создание настраиваемых фильтров для list_filter в Django Admin

Я хотел бы создать собственные фильтры для django admin вместо обычных 'is_staff' и 'is_superuser'. Я прочитал этот list_filter в документах Django. Пользовательские фильтры работают следующим образом:

from datetime import date

from django.utils.translation import ugettext_lazy as _
from django.contrib.admin import SimpleListFilter

class DecadeBornListFilter(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(ModelAdmin):
    list_filter = (DecadeBornListFilter,)

Но я уже сделал пользовательские функции для list_display следующим образом:

def Student_Country(self, obj):
    return '%s' % obj.country
Student_Country.short_description = 'Student-Country'

Возможно ли использовать пользовательские функции для list_display в list_filter вместо того, чтобы писать новую настраиваемую функцию для list_filter? Любые предложения или улучшения приветствуются. Вам нужно руководствоваться этим... Спасибо...

4b9b3361

Ответ 1

Вы действительно можете добавить настраиваемые фильтры в админ-фильтры, расширив SimpleListFilter. Например, если вы хотите добавить фильтр континентов для "Африка" в фильтр страны-администратора, использованный выше, вы можете сделать следующее:

В admin.py:

from django.contrib.admin import SimpleListFilter

class CountryFilter(SimpleListFilter):
    title = 'country' # or use _('country') for translated title
    parameter_name = 'country'

    def lookups(self, request, model_admin):
        countries = set([c.country for c in model_admin.model.objects.all()])
        return [(c.id, c.name) for c in countries] + [
          ('AFRICA', 'AFRICA - ALL')]

    def queryset(self, request, queryset):
        if self.value() == 'AFRICA':
            return queryset.filter(country__continent='Africa')
        if self.value():
            return queryset.filter(country__id__exact=self.value())

class CityAdmin(ModelAdmin):
    list_filter = (CountryFilter,)

Ответ 2

Ваш метод list_display возвращает строку, но если я правильно понимаю, что вы хотите сделать, это добавить фильтр, который позволяет выбирать страны студентов, правильно?

Для этого фильтра простых отношений и, фактически, для столбца отображения списка "Студент-страна", вам не нужно создавать собственный класс фильтра или метод отображения пользовательского списка; этого было бы достаточно:

class MyAdmin(admin.ModelAdmin):
    list_display = ('country', )
    list_filter = ('country', )

Способ django делает list_filter, как объяснено в документах, сначала автоматически сопоставляя поля, которые вы предоставляете, к предварительно построенным классам фильтров; эти фильтры включают CharField и ForeignKey.

list_display аналогично автоматизирует совокупность столбца списка изменений, используя поле, переданное путем извлечения связанных объектов и возвращающее значение unicode (такое же, как и в приведенном выше методе).