Django Admin, как изменить текст в поле отношений

У меня есть следующий код:

models.py

from django.db import models
from parler.models import TranslatableModel, TranslatedFields

class Federation(TranslatableModel):
    translations = TranslatedFields(
        name = models.CharField('name', max_length=50)
    )
    created_at = models.DateTimeField(auto_now_add=True)
    updated_at = models.DateTimeField(auto_now=True)

class Athlete(models.Model):
    first_name = models.CharField(max_length=50)
    last_name = models.CharField(max_length=50)
    federation = models.ForeignKey('Federation', on_delete=models.SET_NULL, null=True)
    height = models.IntegerField();
    weight = models.IntegerField();
    created_at = models.DateTimeField(auto_now_add=True)
    updated_at = models.DateTimeField(auto_now=True)

admin.py

from django.contrib import admin
from parler.admin import TranslatableAdmin

from .models import Athlete, Federation

class AthleteAdmin(admin.ModelAdmin):
    list_display = ['first_name', 'last_name', 'height', 'weight', 'get_federation_name']
    fields = ['first_name', 'last_name', 'height', 'weight', 'federation']

    def get_federation_name(self, obj):
        obj.federation.set_current_language('en')
        return obj.federation.name
    get_federation_name.short_description = 'Federation'

class FederationAdmin(TranslatableAdmin):
    search_fields = ['translations__name']
    list_display = ['name']
    fields = ['name']

admin.site.register(Federation, FederationAdmin)
admin.site.register(Athlete, AthleteAdmin)

Поле федерации отображается в виде списка, но текст в меню выбора отображается как "Объект федерации". Для списка я создал функцию для извлечения данных из отношения перевода связанной модели федерации. Я хочу сделать то же самое с полями формы. Если я заставлю это работать в полях формы без функции, я также изменю отображение списка, чтобы работать таким же образом.

Я новичок в Python и Django (впервые), и я не могу найти решение этой проблемы.

Спасибо!

1 ответ

Решение

По умолчанию он использует __str__() метод объекта. Поэтому самый простой способ изменить это - установить этот метод. Например:

class Federation(models.Model):
    ...

    def __str__(self):
        return "{0} ({1})".format(self.translation, self.created_at)

Другой способ - если вы не хотите переопределять __str__ Метод - будет переопределить label_from_instance самого поля формы. Но это более сложно.

def _federation_label_from_instance(self, obj):
    return "{0} ({1})".format(obj.translation, obj.created_at)

class AthleteAdmin(admin.ModelAdmin):
    ...

    def formfield_for_foreignkey(self, db_field, request, **kwargs):
        formfield = super().form_field(db_field, request, **kwargs)
        if db_field.name == 'federation':
            formfield.label_from_instance = _federation_label_from_instance
        return formfield
Другие вопросы по тегам