2012-11-30 18 views
7

mam modele A i B, gdzie B ma FK do A.Admin Django queryset filtrowania przez klucz obcy wstecznej relacji

Używam Django 1.3 i muszę dwa filtry django admin:

1) a.b_set.exists() # (True/False) 

2) not a.b_set.filter(some_condition=False).exists() # (True/False) 

Jak mogę to osiągnąć? Niestety, nie mogłem znaleźć żadnych rozwiązań przez szukanie w Google.

+0

czy 'A.objects.filter (b__isnull = false) 'i' A.objects.filter (b__isnull = False, some_condition = False) 'help? – mkoistinen

+0

Może, ale pytam, jak mogę napisać niestandardowy display_filter, a nie jak zrobić zapytanie. – aemdy

+0

Czy jest możliwe przejście na Django 1.4? Filtry listy stały się tam bardziej elastyczne. Możesz zasadniczo określić własną klasę filtru listy: https://docs.djangoproject.com/en/1.4/ref/contrib/admin/#django.contrib.admin.ModelAdmin.list_filter – ptrck

Odpowiedz

5

Musisz to przeczytać: Custom Filter in Django Admin on Django 1.3 or below

jest to moja pierwsza próba bez testów, ale trzeba zobaczyć mniej więcej jak jej zrobić -

from django.db import models 
from django.contrib.admin.filterspecs import FilterSpec, ChoicesFilterSpec 
from django.utils.encoding import smart_unicode 
from django.utils.translation import ugettext as _ 

class BNullSetFilterSpec(FilterSpec): 

    def __init__(self, f, request, params, model, model_admin): 
     super(BSetFilterSpec, self).__init__(f, request, params, model, model_admin) 

     self.links = (
      ('Yes', {'b__isnull': False}), 
      ('No', {})) 

    def title(self): 
     return _('B Set') 

# registering the filter 
FilterSpec.filter_specs.insert(0, (lambda f: getattr(f, 'empty_bset', False), BNullSetFilterSpec)) 
1

Co wspomniano w @petrku jest prawdą. Nie mogłeś przełączyć się na 1.4?

W tym przypadku, podobnie jak w the doc opisane, bez konieczności zmiany wyszukiwań, zrobić coś podobnego (bo tu pierwszy, a następnie odpowiednio drugi):

from django.contrib.admin import BooleanFieldListFilter 

class BInA(BooleanFieldListFilter): 
    def queryset(self, request, queryset): 
     if self.value() is True: 
      return queryset.filter(a.b_set.exists()) 
     else: 
      return queryset.filter(a.b_set.exists() is False)