Django管理界面中同一模型的多个ModelAdmin/视图

195 投票
3 回答
49952 浏览
提问于 2025-04-15 19:02

我想知道如何为同一个模型创建多个不同的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 个回答

2

根据正确的答案,我对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)

3

保罗·斯通的回答真是太棒了!我想补充一下,对于Django 1.4.5版本,我需要让我的自定义类继承自 admin.ModelAdmin

class MyPostAdmin(admin.ModelAdmin):
    def queryset(self, request):
        return self.model.objects.filter(id=1)
359

我找到了一种方法,可以实现我想要的功能,那就是使用代理模型来解决每个模型只能注册一次的问题。

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)

撰写回答