Wiele do wielu na liście wyświetla django

91
class PurchaseOrder(models.Model):
    product = models.ManyToManyField('Product')
    vendor = models.ForeignKey('VendorProfile')
    dollar_amount = models.FloatField(verbose_name='Price')


class Product(models.Model):
   products = models.CharField(max_length=256)

   def __unicode__(self):
       return self.products

Mam ten kod. Niestety błąd pojawia się w admin.py z rozszerzeniemManyToManyField

class PurchaseOrderAdmin(admin.ModelAdmin):
    fields = ['product', 'dollar_amount']
    list_display = ('product', 'vendor')

Błąd mówi:

„PurchaseOrderAdmin.list_display [0]”, „produkt” to ManyToManyField, który nie jest obsługiwany.

Jednak kiedy kompiluje się 'product'z list_display. Jak więc mogę wyświetlić 'product'w programie list_displaybez podawania błędów?

edycja : Może lepszym pytaniem byłoby, jak wyświetlić ManyToManyFieldw list_display?

Mdjon26
źródło

Odpowiedzi:

171

Możesz nie być w stanie zrobić tego bezpośrednio. Z dokumentacjilist_display

Pola ManyToManyField nie są obsługiwane, ponieważ wymagałoby to wykonania oddzielnej instrukcji SQL dla każdego wiersza w tabeli. Jeśli jednak chcesz to zrobić, nadaj modelowi niestandardową metodę i dodaj nazwę tej metody do list_display. (Zobacz poniżej, aby uzyskać więcej informacji na temat metod niestandardowych w list_display).

Możesz zrobić coś takiego:

class PurchaseOrderAdmin(admin.ModelAdmin):
    fields = ['product', 'dollar_amount']
    list_display = ('get_products', 'vendor')

    def get_products(self, obj):
        return "\n".join([p.products for p in obj.product.all()])

LUB zdefiniuj metodę modelową i użyj jej

class PurchaseOrder(models.Model):
    product = models.ManyToManyField('Product')
    vendor = models.ForeignKey('VendorProfile')
    dollar_amount = models.FloatField(verbose_name='Price')

    def get_products(self):
        return "\n".join([p.products for p in self.product.all()])

i w admin list_display

list_display = ('get_products', 'vendor')
karthikr
źródło
To wygląda na naprawdę dobre rozwiązanie. Dziękuję Ci. Chociaż teraz pojawia się błąd informujący, że „wartość zerowa w kolumnie„ identyfikator_produktu ”narusza ograniczenie niezerowe” Masz pojęcie, co to oznacza?
Mdjon26
3
Skoro to sprowadza bazę danych na kolana, jak byś to zrobił z select_related () lub prefetch_related (), aby poprawić wydajność?
Cloud Artisans
3
Na wypadek, gdyby kwestia optymalizacji jest nadal interesująca, właśnie miałem ten sam problem i dowiedziałem się, że możesz po prostu zaimplementować zoptymalizowaną get_queryset()metodę dla swojej ModelAdmin, patrz stackoverflow.com/questions/12354099/ ...
goetz
1
@ SebastiánVansteenkiste Dobry pomysł, może cached_propertypomogłoby. Ale myślę, że prawdopodobnie nie. Kiedy używasz zoptymalizowanego get_queryset, możesz na przykład dodać adnotacje / wstępnie przetworzyć dane tam, na przykład wykonując konkatenację produktów w SQL zamiast w Django, i przechowywać niestandardowe dane w swoim zestawie zapytań. Wtedy wystarczyłoby wykonać tę logikę tylko raz w SQL, a nie dla każdego wiersza, gdy uzyskiwany jest dostęp do właściwości.
goetz
1
Słuszna uwaga. Powinienem zająć się zrobieniem własnej optymalizacji get_queryset, po prostu nie przeczytałem pełnej dokumentacji (ani nie znalazłem wystarczająco prostego przykładu tego, co powinienem robić)
Sebastián Vansteenkiste
16

W ten sposób możesz to zrobić, uprzejmie sprawdź następujący fragment:

class Categories(models.Model):
    """ Base category model class """

    title       = models.CharField(max_length=100)
    description = models.TextField()
    parent      = models.ManyToManyField('self', default=None, blank=True)
    when        = models.DateTimeField('date created', auto_now_add=True)

    def get_parents(self):
        return ",".join([str(p) for p in self.parent.all()])

    def __unicode__(self):
        return "{0}".format(self.title)

A w metodzie wywołania modułu admin.py w następujący sposób:

class categories(admin.ModelAdmin):
    list_display    = ('title', 'get_parents', 'when')
JKV
źródło