Django管理界面中同一模型的多个ModelAdmin/视图
我想知道如何为同一个模型创建多个不同的ModelAdmin,每个都可以自定义,并且链接到不同的URL。
假设我有一个叫做Posts的Django模型。默认情况下,这个模型的管理界面会列出所有的Post对象。
我知道可以通过设置一些变量,比如list_display,或者在我的ModelAdmin中重写queryset
方法,来定制页面上显示的对象列表,像这样:
class MyPostAdmin(admin.ModelAdmin):
list_display = ('title', 'pub_date')
def queryset(self, request):
request_user = request.user
return Post.objects.filter(author=request_user)
admin.site.register(MyPostAdmin, Post)
默认情况下,这个页面可以通过URL /admin/myapp/post
访问。不过,我想要有多个视图/ModelAdmin来处理同一个模型。例如,/admin/myapp/post
会列出所有的帖子,而/admin/myapp/myposts
会列出属于当前用户的所有帖子,/admin/myapp/draftpost
可能会列出所有尚未发布的帖子。(这些只是例子,我的实际需求更复杂)
你不能为同一个模型注册多个ModelAdmin(这样会导致AlreadyRegistered
的错误)。理想情况下,我希望在不把所有内容放到一个ModelAdmin类里,也不写自己的'urls'函数来根据URL返回不同的查询集的情况下实现这个目标。
我查看了Django的源代码,发现像ModelAdmin.changelist_view
这样的函数,可能可以在我的urls.py中使用,但我不太确定具体该怎么做。
更新:我找到了一种实现我想要的方式(见下文),但我仍然想听听其他的实现方法。
3 个回答
根据正确的答案,我对AdminSite
这个类进行了修改,添加了一个叫register_via_proxy
的方法,这样可以让这个任务变得更简单。
import re
from django.contrib import admin
def _register_proxy(self, model, admin_class):
proxy_model = type(
admin_class.__name__, (model,), {
"__module__": re.sub(
r'(^.*?)(\.[^\.]+)$', r'\1.proxy', model.__module__
),
"Meta": type("Meta", tuple(), {
"proxy": True,
"app_label": model._meta.app_label
})
}
)
return self.register(proxy_model, admin_class)
admin.sites.AdminSite.register_via_proxy = _register_proxy
使用这个方法的方式如下:
site = admin.sites.AdminSite()
site.register_via_proxy(models.ModelType, AdminClass)
保罗·斯通的回答真是太棒了!我想补充一下,对于Django 1.4.5版本,我需要让我的自定义类继承自 admin.ModelAdmin
。
class MyPostAdmin(admin.ModelAdmin):
def queryset(self, request):
return self.model.objects.filter(id=1)
我找到了一种方法,可以实现我想要的功能,那就是使用代理模型来解决每个模型只能注册一次的问题。
class PostAdmin(admin.ModelAdmin):
list_display = ('title', 'pubdate','user')
class MyPost(Post):
class Meta:
proxy = True
class MyPostAdmin(PostAdmin):
def get_queryset(self, request):
return self.model.objects.filter(user = request.user)
admin.site.register(Post, PostAdmin)
admin.site.register(MyPost, MyPostAdmin)
这样,默认的 PostAdmin
就可以在 /admin/myapp/post
这个地址访问,而用户自己发布的帖子列表可以在 /admin/myapp/myposts
找到。
在查看了 http://code.djangoproject.com/wiki/DynamicModels 后,我想出了以下这个实用函数来实现同样的功能:
def create_modeladmin(modeladmin, model, name = None):
class Meta:
proxy = True
app_label = model._meta.app_label
attrs = {'__module__': '', 'Meta': Meta}
newmodel = type(name, (model,), attrs)
admin.site.register(newmodel, modeladmin)
return modeladmin
这个函数可以这样使用:
class MyPostAdmin(PostAdmin):
def get_queryset(self, request):
return self.model.objects.filter(user = request.user)
create_modeladmin(MyPostAdmin, name='my-posts', model=Post)