django modeladmin에서 외국 키 필드의 디스플레이 속성에서 "list_display"가있을 수 있습니까?
-
03-07-2019 - |
문제
나는있다 Person
외국의 핵심 관계가있는 모델 Book
, 많은 필드가 있지만 가장 걱정하는데 author
(표준 샤 필드).
그 말로 내 PersonAdmin
모델, 표시하고 싶습니다 book.author
사용 list_display
:
class PersonAdmin(admin.ModelAdmin):
list_display = ['book.author',]
나는 그렇게하기위한 모든 명백한 방법을 시도했지만 아무것도 효과가없는 것 같습니다.
제안이 있습니까?
해결책
다른 옵션으로, 당신은 다음과 같은 조회를 할 수 있습니다.
class UserAdmin(admin.ModelAdmin):
list_display = (..., 'get_author')
def get_author(self, obj):
return obj.book.author
get_author.short_description = 'Author'
get_author.admin_order_field = 'book__author'
다른 팁
위의 모든 위대한 답변에도 불구하고 나는 장고를 처음 접했기 때문에 여전히 갇혀있었습니다. 다음은 매우 초보자 관점에서 내 설명입니다.
Models.py
class Author(models.Model):
name = models.CharField(max_length=255)
class Book(models.Model):
author = models.ForeignKey(Author)
title = models.CharField(max_length=255)
admin.py (잘못된 방법) - 'Model__field'를 참조하여 작동한다고 생각하지만 그렇지 않습니다.
class BookAdmin(admin.ModelAdmin):
model = Book
list_display = ['title', 'author__name', ]
admin.site.register(Book, BookAdmin)
admin.py (올바른 방법) - 이것은 외국 키 이름을 Django Way를 참조하는 방법입니다.
class BookAdmin(admin.ModelAdmin):
model = Book
list_display = ['title', 'get_name', ]
def get_name(self, obj):
return obj.author.name
get_name.admin_order_field = 'author' #Allows column order sorting
get_name.short_description = 'Author Name' #Renames column head
#Filtering on side - for some reason, this works
#list_filter = ['title', 'author__name']
admin.site.register(Book, BookAdmin)
추가 참조는 Django 모델 링크를 참조하십시오 여기
나머지처럼, 나도 콜블 즈와 함께 갔다. 그러나 그들은 하나의 단점을 가지고 있습니다. 기본적으로 주문할 수 없습니다. 다행히도 다음에 대한 해결책이 있습니다.
def author(self):
return self.book.author
author.admin_order_field = 'book__author'
추가하십시오 get_author
함수는 각 사람을 보여 주면 SQL 쿼리를 만들기 때문에 관리자의 List_Display가 느려집니다.
이를 피하려면 수정해야합니다 get_queryset
Personadmin의 방법 : 예를 들어 :
def get_queryset(self, request):
return super(PersonAdmin,self).get_queryset(request).select_related('book')
전 : 36.02ms의 쿼리 73 개 (관리자의 67 개의 복제 쿼리)
후 : 10.81ms의 6 쿼리
문서에 따르면 __unicode__
외국인의 표현 :
http://docs.djangoproject.com/en/dev/ref/contrib/admin/#list-display
그것이 지원하지 않는다는 것이 이상해 보입니다 'book__author'
DB API의 다른 곳에서 사용되는 스타일 형식.
거기에 나왔습니다 이 기능을위한 티켓, 고정되지 않은 것으로 표시되어 있습니다.
호출 가능을 사용하여 목록 디스플레이에서 원하는 것을 보여줄 수 있습니다. 다음과 같습니다.
def book_author(object): return object.book.author class PersonAdmin(admin.ModelAdmin): list_display = [book_author,]
방금 admin.modeladmin support '__'구문을 만드는 스 니펫을 게시했습니다.
http://djangosnippets.org/snippets/2887/
그래서 당신은 할 수 있습니다 :
class PersonAdmin(RelatedFieldAdmin):
list_display = ['book__author',]
이것은 기본적으로 다른 답변에 설명 된 것과 동일한 작업을 수행하지만 (1) admin_order_field 설정 (2) Short_Description 설정 및 (3) 각 행의 데이터베이스에 적중을 피하기 위해 쿼리 세트를 수정하는 것을 자동으로 처리합니다.
이것은 이미 받아 들여졌지만, 나와 같은 다른 인형이 있다면 즉시 그것을 얻지 못했습니다. 현재 받아 들여진 답변, 여기에 좀 더 자세한 내용이 있습니다.
모델 클래스 ForeignKey
필요합니다 __unicode__
여기서와 같이 내부의 방법 :
class Category(models.Model):
name = models.CharField(max_length=50)
def __unicode__(self):
return self.name
그것은 저에게 차이를 만들었고 위의 시나리오에 적용해야합니다. 이것은 Django 1.0.2에서 작동합니다.
인라인으로 시도하면 다음과 같이하지 않으면 성공하지 못할 것입니다.
인라인으로 :
class AddInline(admin.TabularInline):
readonly_fields = ['localname',]
model = MyModel
fields = ('localname',)
모델 (MyModel)에서 :
class MyModel(models.Model):
localization = models.ForeignKey(Localizations)
def localname(self):
return self.localization.name
사용할 관계 속성 필드가 많은 경우 list_display
그리고 각각에 대한 함수를 만들고 싶지 않지만 흙이지만 간단한 솔루션은 ModelAdmin
Instace __getattr__
메소드, 즉시 콜블 가능성 생성 :
class DynamicLookupMixin(object):
'''
a mixin to add dynamic callable attributes like 'book__author' which
return a function that return the instance.book.author value
'''
def __getattr__(self, attr):
if ('__' in attr
and not attr.startswith('_')
and not attr.endswith('_boolean')
and not attr.endswith('_short_description')):
def dyn_lookup(instance):
# traverse all __ lookups
return reduce(lambda parent, child: getattr(parent, child),
attr.split('__'),
instance)
# get admin_order_field, boolean and short_description
dyn_lookup.admin_order_field = attr
dyn_lookup.boolean = getattr(self, '{}_boolean'.format(attr), False)
dyn_lookup.short_description = getattr(
self, '{}_short_description'.format(attr),
attr.replace('_', ' ').capitalize())
return dyn_lookup
# not dynamic lookup, default behaviour
return self.__getattribute__(attr)
# use examples
@admin.register(models.Person)
class PersonAdmin(admin.ModelAdmin, DynamicLookupMixin):
list_display = ['book__author', 'book__publisher__name',
'book__publisher__country']
# custom short description
book__publisher__country_short_description = 'Publisher Country'
@admin.register(models.Product)
class ProductAdmin(admin.ModelAdmin, DynamicLookupMixin):
list_display = ('name', 'category__is_new')
# to show as boolean field
category__is_new_boolean = True
처럼 여기서 요점
호출 가능한 특별한 속성과 같은 boolean
그리고 short_description
정의해야합니다 ModelAdmin
속성, 예를 들어 book__author_verbose_name = 'Author name'
그리고 category__is_new_boolean = True
.
전화 가능 admin_order_field
속성은 자동으로 정의됩니다.
사용하는 것을 잊지 마십시오 list_select_related 당신의 속성 ModelAdmin
django를 만들려면 탁월한 쿼리를 피하십시오.
PYPI에는 사용하기 쉬운 패키지가 정확히 처리됩니다. Django 관련 -admin. 당신은 또한 수 Github의 코드를 참조하십시오.
이것을 사용하여 달성하고자하는 것은 다음과 같이 간단합니다.
class PersonAdmin(RelatedFieldAdmin):
list_display = ['book__author',]
두 링크 모두 설치 및 사용에 대한 자세한 내용이 포함되어 있으므로 변경할 경우 여기에 붙여 넣지 않습니다.
부수적 노트처럼, 이미 다른 것을 사용하고 있다면 model.Admin
(예 : 내가 사용하고 있었다 SimpleHistoryAdmin
대신), 당신은 이것을 할 수 있습니다 : class MyAdmin(SimpleHistoryAdmin, RelatedFieldAdmin)
.
Alexrobbins의 답변은 처음 두 줄이 모델에 있어야한다는 점을 제외하고는 저에게 효과가 있었고 (아마도 이것은 가정되었을 것입니까?) 자기를 참조해야합니다.
def book_author(self):
return self.book.author
그런 다음 관리자 부분이 잘 작동합니다.
나는 이것을 선호한다 :
class CoolAdmin(admin.ModelAdmin):
list_display = ('pk', 'submodel__field')
@staticmethod
def submodel__field(obj):
return obj.submodel.field