美文网首页
Django 的 Admin

Django 的 Admin

作者: vckah | 来源:发表于2018-08-01 15:49 被阅读0次

Django 非常好用的一个功能就是后台界面的自动管理。

from django.contrib import admin

from django.utils.safestring import mark_safe
class xxxConfig(admin.ModelAdmin):
    
    def deletes(self):
        return mark_safe('<a href=''>删除</a>')
    
    list_display = [, deletes] # 里面的字段不能是 ManyToMany 字段
    list_filter = ()
    list_display_links = ()
    search_fields = ()
    
    def patch_init(self, request, queryset):
        queryset.update(xxx=xxx)
    patch_init.short_description = "批量处理"
    actions = [patch_init, ]
所有字段都在 admin.ModelAdmin 中可见
admin.site.register(模型, '你自定义的类')

Admin 的实现流程:启动,注册,设计url

  • 启动
    INSTALLED_APPS 中的注册 app 中,扫描 admin 文件,并执行
def autodiscover():
    autodiscover_modules('admin', register_to=site)
  • 注册,在每个安装的 app 下面的 admin 文件中注册 site,即admin.site.register(模型, '你自定义的类')
  • 设计 url
    在项目的根目录下 url 文件中会有 path('admin/', admin.site.urls), 这么一行注册 url 的函数
    综上,其实这是很明显的一个单例模式,因为后两步中都用到了一个特殊的对象 admin.siteadmin 是从 django.contrib 中导入的,而 admin 中又导入了 sitefrom django.contrib.admin.sites import AdminSite, site,之后进入到 site 中,发现了也就这么一个东西:
class AdminSite:
    pass

site = AdminSite()

这其实用到了 Python 中对于单例模式的一个经典的用法,模块就是一个特殊的单例,因为模块无论声明多少次,最终导入到 Python 解释器中只会有一次,就是刚开始那次。

接下来我们看一看源码做了什么?

# admin 的 __init__.py
from django.contrib.admin.sites import AdminSite, site
def autodiscover():
    autodiscover_modules('admin', register_to=site)

这里随即进入 AdminSite:

# sites.py
pass
class AdminSite:
    def __init__(self, name='admin'):
        self._registry = {}  # model_class class -> admin_class instance
        self.name = name
        pass

    def register(self, model_or_iterable, admin_class=None, **options):
        if not admin_class:
            admin_class = ModelAdmin
        pass

site = AdminSite()

注意这里的 site = AdminSite(),在 Django 的其它地方导入它以后,它就是一个单例了,它的 id 就不变了。在 AdminSite 中定义了一个字典 _registry,里面存放的是注册的 model 和 其配置类,配置类默认是 ModelAdmin,也就是 Django 自己的那个后台。这里完成了 url 的注册。
接下来看一看它里面是如何构造 url 的。还记得在项目根目录下的 url(r'admin', admin.site.urls),吗,进入它:

# 跟上面的一样是 sites.py
def get_urls(self):
    pass
    for model, model_admin in self._registry.items():
     urlpatterns += [
        path('%s/%s/' % (model._meta.app_label, model._meta.model_name), include(model_admin.urls)),
            ]
    if model._meta.app_label not in valid_app_labels:
        valid_app_labels.append(model._meta.app_label)
    return urlpatterns

@property
def urls(self):
    return self.get_urls(), 'admin', self.name

这里面 include(model_admin.urls) 及其重要,Django 默认将自己的配置类注册到 _register 了,它是 ModelAdminurls

# 注:这是 Django 2 的有些部分可能不一样,不过不影响
def get_urls(self):
    pass
    info = self.model._meta.app_label, self.model._meta.model_name

    urlpatterns = [
        path('', wrap(self.changelist_view), name='%s_%s_changelist' % info),
        path('add/', wrap(self.add_view), name='%s_%s_add' % info),
        path('autocomplete/', wrap(self.autocomplete_view), name='%s_%s_autocomplete' % info),
        path('<path:object_id>/history/', wrap(self.history_view), name='%s_%s_history' % info),
        path('<path:object_id>/delete/', wrap(self.delete_view), name='%s_%s_delete' % info),
        path('<path:object_id>/change/', wrap(self.change_view), name='%s_%s_change' % info),
        # For backwards compatibility (was the change url before 1.9)
        path('<path:object_id>/', wrap(RedirectView.as_view(
            pattern_name='%s:%s_%s_change' % ((self.admin_site.name,) + info)
        ))),
    ]
    return urlpatterns

@property
def urls(self):
    return self.get_urls()

这里重要的是 Django 对 url 做了两级分发,而且这两级 url 不在同一个类中,这样做的目的就是为了方便构造模板。最后那一级 url 放在了 ModelAdmin 中,就是增删改查那四个功能。

说明:这里二级分发 url 主要是为了方便后面模板构造数据,因为构造数据要访问 model,如果两级 url 全部放在一个类中,那么需要找到 model 便需要根据 url 来匹配,但是将二级 url 放在默认的类中,那么它便可以找到对应的 model 了,因为 _register 中存放的是 model 和 配置类形成的字典。而默认配置类里面会注册进来 对应的 model,所以在配置类中可以找到 model。

先暂时到这,可能有诸多不清楚的,后面再补充。
顺便说一句,AdminSite 的源码连注释,加空行总共 500 行代码,有空了解一下。😋

Django 是如何找到每个 app 下的 admin.py 文件呢,在每个 app 下都有一个配置类,如 admin 的 AdminConfig,它里面会定义一个 ready 方法,django 在加载 INSTALLED_APPS 时,会调用里面的方法。

class AdminConfig(SimpleAdminConfig):
    def ready(self):
        super().ready()
        self.module.autodiscover()

在我们自己构建组件时,可以这样写:

from django.apps improt AppConfig
from django.uitls.moudle_loading improt autodiscover_moudles

class xxxConfig(AppConfig):
    name = xxx
    def ready(self):
        autodiscover_moudles('里面放需要加载文件的名称')

设计 url

首先来了解一下 url 注册的另一种方式:

def test01(request):
    return HttpResponse('test01')
def test02(request):
    pass
def test03(request):
    pass

urlpatterns = [
    url(r'^test/', ([
        url(r't1/', test01),
        url(r't2/', test02),
        url(r't3/', test03),
    ], None, None))
]

这种相当于是将 include 中的 url 放到当前 url 中了。
构造 url:

get_urls2():
    temp = []
    temp.append(url(r'^$'), view_name)
    temp.append(url(r'^add$'), view_name)
    temp.append(url(r'^(\d+)/change/$'), view_name)
     temp.append(url(r'^(\d+)/delete/$'), view_name)
    return temp

def get_urls():
    res = []
    for model, admin_class_obj in admin.site_register.items():
        app_name = model._meta.app_label
        model_name = model._meta.model_name
        temp.append(url(r'{0}/{1}'.format(app_name, model_name), (get_urls2(), None, None)), )
    return res

urlpatterns = [
    url(r'^xadmin/', (get_urls(), None, None)),
]

以上有几个函数很有趣:
model._meta.app_label --> 获取模型所属 app 的名称字符串
model._meta.model_name --> 获取模型名称字符串
obj = model._meta.get_field("field") --> 获取模型内某个属性,别忘记,其实模型内的属性也是一个类
所以:obj.verbose_name 即可取到其对应的属性。

相关文章

网友评论

      本文标题:Django 的 Admin

      本文链接:https://www.haomeiwen.com/subject/eotkvftx.html