26

次のモデルが与えられます

class AnotherModel(models.Model):
    n = models.IntegerField()

class MyModel(models.Model):
    somefield = models.ForeignKey(AnotherModel)

および管理者

class MyModelAdmin(admin.ModelAdmin):        
    list_filter = ('somefield',)

のインスタンスをフィルタリングして、管理者フィルターに特定の値をAnotherModel持つインスタンスのみを表示するにはどうすればよいですか?n

私は次のようなものが必要です:

フィルター

どこかで

すべて

[指定されたnを持つAnotherModelインスタンスのリスト]

4

4 に答える 4

34

ModelAdmin.querysetおよびModelAdmin.formfield_for_foreignkeyを参照してください。ドキュメントから:

ModelAdminのquerysetメソッドは、管理サイトで編集できるすべてのモデルインスタンスのQuerySetを返します。このメソッドをオーバーライドする1つの使用例は、ログインしたユーザーが所有するオブジェクトを表示することです。

class MyModelAdmin(admin.ModelAdmin):
    def queryset(self, request):
        qs = super(MyModelAdmin, self).queryset(request)
        if request.user.is_superuser:
            return qs
        return qs.filter(author=request.user)

ModelAdminのformfield_for_foreignkeyメソッドを使用すると、外部キーフィールドのデフォルトのフォームフィールドをオーバーライドできます。たとえば、ユーザーに基づいてこの外部キーフィールドのオブジェクトのサブセットを返すには、次のようにします。

class MyModelAdmin(admin.ModelAdmin):
    def formfield_for_foreignkey(self, db_field, request, **kwargs):
        if db_field.name == "car":
            kwargs["queryset"] = Car.objects.filter(owner=request.user)
        return super(MyModelAdmin, self).formfield_for_foreignkey(db_field, request, **kwargs)

これは、HttpRequestインスタンスを使用してCar外部キーフィールドをフィルタリングし、Userインスタンスが所有する車のみを表示します。

[アップデート]

申し訳ありませんが、「フィルター」の部分を読むことができませんでした。Django> = 1.4 django.contrib.admin.SimpleListFilterでは、list_filter引数リストでのサブクラスを渡すことができます。これを使用して、ルックアップメソッドとquerysetメソッドをオーバーライドできます。

from datetime import date

from django.contrib import admin
from django.utils.translation import ugettext_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,)
于 2012-09-21T00:14:35.770 に答える
7

編集-この方法には問題があることが指摘されています。以下を参照してください

あなたはこのようにそれを行うことができます:

SpeciesというモデルへのForeignKeyフィールドを持つAnimalというモデルがあるとします。特定の管理者リストで、動物フィルターの選択肢に特定の種のみを表示できるようにする必要があります。

まず、AnimalのModelAdminでSpeciesFilterというカスタムListFilterを指定します。

class AnimalAdmin(ModelAdmin):
    list_filter = (('species', SpeciesFilter),)

次に、SpeciesFilterを定義します。

from django.contrib.admin.filters import RelatedFieldListFilter

class SpeciesFilter(RelatedFieldListFilter):
    def __init__(self, field, request, *args, **kwargs):
        """Get the species you want to limit it to.
        This could be determined by the request,
        But in this example we'll just specify an
        arbitrary species"""
        species = Species.objects.get(name='Tarantula')

        #Limit the choices on the field
        field.rel.limit_choices_to = {'species': species}

        #Let the RelatedFieldListFilter do its magic 
        super(SpeciesFilter, self).__init__(field, request, *args, **kwargs)

それはそれをする必要があります。

于 2013-02-18T15:58:47.317 に答える
2

@seddonymに似た別のメソッドを見つけましたが、キャッシングを台無しにしません。これはこのDjangoコードに基づいていますが、文書化されていない方法field_choicesを使用しており、将来のDjangoリリースで変更される可能性があります。@seddonymの場合のコードは次のようになります。

from django.contrib.admin.filters import RelatedFieldListFilter

class SpeciesFilter(RelatedFieldListFilter):
    def field_choices(self, field, request, model_admin):
        return field.get_choices(include_blank=False, limit_choices_to={'name': 'Tarantula'})

または私の場合、動作するコードは次のとおりです。

from django.contrib.admin.filters import RelatedFieldListFilter

class UnitFilter(RelatedFieldListFilter):
    def field_choices(self, field, request, model_admin):
        return field.get_choices(include_blank=False, limit_choices_to={'pk__in': request.user.administrated_units.all()})

于 2019-07-15T09:58:23.863 に答える
1

dbテーブルからルックアップフィールドを作成する必要がありました。以下のようにカスタムフィルタークラスを作成し、ログインしたユーザーに関連する値のみを表示し、それに応じてフィルター処理します。

class ShiftFilter_Org(admin.SimpleListFilter):
title = 'Organisation'
parameter_name = 'org'

def lookups(self, request, model_admin):
"""Get the organisations you want to limit"""
    qs_org = Organisation.objects.filter(users=request.user)
    list_org = []
    for og in qs_org:
        list_org.append(
            (og.id, og.name)
        )
    return (
        sorted(list_org, key=lambda tp:tp[1])
    )

def queryset(self, request, queryset):
    if self.value():
        return queryset.filter(org=self.value())

詳細については、Django管理フィルターを最大限に活用するをご覧ください

于 2018-05-31T13:29:03.380 に答える