Jolly

Django中xadmin的学习与记录总结
xadmin是Django中admin后台的进阶插件,它包含了admin的基本功能,并在此基础上进行进一步的升级,...
扫描右侧二维码阅读全文
16
2019/03

Django中xadmin的学习与记录总结

xadmin是Django中admin后台的进阶插件,它包含了admin的基本功能,并在此基础上进行进一步的升级,使得成为一个功能完善的后台管理系统,像瑞士军刀一样拥有各种功能。

一.xadmin的安装与简单使用

1.安装

xadmin有两种方式,一种是通过pip的方式安装,如下:

pip install xadmin

但也可以直接进入github下载,xadmin的下载地址是:
点我进入下载链接
并将其直接放置在项目目录下。
采用直接下载方式的原因是直接下载的源码包含更多没有发布至pypi上面的新功能,然后有了源码,后面可以直接更改源码,增加自己的功能,这样其实更方便扩展。

2.注册xadmin

在settings.py的INSTALLED_APPS中增加如下:

'xadmin',
'crispy_forms'

然后把urls中默认admin指向xadmin:

#导入xadmin,替换admin
import xadmin
urlpatterns = [
    url(r'^xadmin/', xadmin.site.urls),
]

3.生成xadmin数据表

点击Tools 菜单下 Run manage.py Task:

makemigrations
migrate

4.注册model

简单的注册一个model看一下效果,如下:


from .models import UserInfo
import xadmin
 
 
class UserInfoAdmin(object):
    list_display = ['user_name', 'user_email', 'user_mobile']  # 默认显示
    search_fields = ['user_name', 'user_email', 'user_mobile']  # 查询
    list_filter = ['user_name', 'user_email', 'user_mobile']  # 过滤筛选
 
xadmin.site.register(UserInfo, UserInfoAdmin)

如代码中所示,这里是将model和自定义的admin显示规则绑定注册至xadmin中,list_display、search_fields和list_filter分别表示默认显示的字段、可以查询的字段和可以过滤筛选的字段。

实际显示效果如下:
20180901214106_476.png


二.xadmin的进阶功能

1.一般外键关联数据在xadmin只能通过在列表选择对应外键数据跳转再去编辑,但是想在详情编辑中,可以直接编辑(只能一层嵌套):

class LessonInline(object):
    model = Lesson
    extra = 0

class CourseAdmin(object):
    list_display = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num','add_time']
    search_fields = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num']
    list_filter = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num','add_time']
    inlines = [LessonInline]

2.自定义图标、自定义字段排序、自定义字段不可编辑、编辑页面自定义影藏某字段、下拉选涉及到外检选择的改为搜索选择

class CourseAdmin(object):
    list_display = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num','add_time']
    search_fields = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num']
    list_filter = ['name','desc','detail','degree','learn_times','students','fav_nums','image','click_num','add_time']
    model_icon = 'fa fa-book' #自定义图标
    ordering = ['-click_num'] #显示排序
    readonly_fields = ['click_num','fav_nums']#后台不可编辑
    exclude= ['add_time']  #详情不显示add_time 此字段与readonly_fields互斥
    relfield_style='fk-ajax' #下拉筛选改为搜索

3.一张表在后台注册成两个管理器,以课程为例,课程一张表,后台分为轮播课程、非轮播课程。
model代码如下:

class Course(models.Model):
    name = models.CharField(verbose_name=u'课程名',max_length=50)
    is_banner = models.BooleanField(verbose_name=u'是否轮播',default=False) #是否为轮播课程
    add_time = models.DateTimeField(verbose_name=u'添加时间',default=datetime.now)

    class Meta:
        verbose_name = u'课程'
        verbose_name_plural = verbose_name

    def __unicode__(self):
        return self.name

class BannerCourse(Course):
    '''banner课程'''
    class Meta:
        verbose_name = u'轮播课程'
        verbose_name_plural = verbose_name
        proxy = True #不会生成表

这样定义其实原理是,其中自定义一个model,继承自另一个model,然后关键的是自定义的马哥model,需要在Meta中声明proxy = True,这样就不会生成另一个表,实现两个model公共一个表,而两个model可以在xadmin中注册两个管理器。

下面是adminx的代码:

class CourseAdmin(object):
    list_display = ['name','is_banner','add_time']
    search_fields = ['name','is_banner']
    list_filter = ['name','is_banner', 'add_time']

    def queryset(self):
        qs = super().queryset()
        qs = qs.filter(is_banner=False)
        return qs

class BannerCourseAdmin(object):
    list_display = ['name', 'is_banner', 'add_time']
    search_fields = ['name', 'is_banner']
    list_filter = ['name', 'is_banner', 'add_time']

    def queryset(self):
        qs = super().queryset()
        qs = qs.filter(is_banner=True)
        return qs

xadmin.site.register(BannerCourse, BannerCourseAdmin)
xadmin.site.register(Course, CourseAdmin)

其中重要的是queryset方法,该方法的重写,意义在于重写父类的queryset方法,然后过滤出自定义管理器,如直接定义is_banner=True,这样,两个管理器就实现了分别管理不同的数据了。

4.将model中的方法获取的值,在后台列表字段显示:

#model中代码如下:
class Course(models.Model):
    name = models.CharField(verbose_name=u'课程名',max_length=50)

    class Meta:
        verbose_name = u'课程'
        verbose_name_plural = verbose_name

    def get_zj_nums(self):
        '''获取该课程章节'''
        return  self.lesson_set.all().count()
    get_zj_nums.short_description = u'章节数'   #指定后台显示列表字段名

    def __unicode__(self):
        return self.name
#admix中代码如下:
class CourseAdmin(object):
    list_display = ['name','get_zj_nums']
    search_fields = ['name']
    list_filter = ['name']

直接将在model中定义的方法,写在list_display中展示,xadmin会自动判断这是一个方法,并将方法得到的值进行展示。
注意:如果需要改变该方法的字段名显示,直接方法名.short_description即可,如get_zj_nums.short_description = u'章节数'

5.在model中定义方法,返回html,在后台以html代码形式显示:

#model中代码:
class Course(models.Model):
    name = models.CharField(verbose_name=u'课程名',max_length=50)

    class Meta:
        verbose_name = u'课程'
        verbose_name_plural = verbose_name

    def go_to(self):
        from django.utils.safestring import mark_safe
        return mark_safe('<a href="http://www.baidu.com">百度</a>')百度</a>' #如果不使用mark_safe,在后台显示的就是一段文本了
     go_to.short_description = u'跳转'  # 指定后台显示列表字段名

    def __unicode__(self):
        return self.name

# admix中代码如下:
class CourseAdmin(object):
    list_display = ['name', 'go_to']
    search_fields = ['name']
    list_filter = ['name']

这里关键处在于,对返回的链接的处理,这里需要使用django.utils.safestring中的mark_safe方法,因为Django有自己的保护机制,如果不使用mark_safe,在后台显示的就是一段文本了,这里需要使用mark_safe将其正确解读成一个链接。

6.xadmin集成百度uditor编辑器
基本步骤:

github上搜素django ueditor,找到zhangfisher/DjangoUeditor,下载并解压,
切换命令行到解压后的目录执行python setup.py install(如果是虚拟环境开发,进入虚拟环境在进入对应目录)
将 DjangoUeditor 放入到settings.py中的INSTALLED_APPS

配置urls.py如下:

urlpatterns = [
    ...
    url(r'^ueditor/', include('DjangoUeditor.urls')),
]

比如给课程加富文本编辑器,去Course的model中引入代码:

from DjangoUeditor.models import UEditorField

对需要添加ueditor字段的修改:

detail = UEditorField(u'课程详情', width=600, height=300, imagePath="course/ueditor/", filePath="course/ueditor/", upload_settings={"imageMaxSize": 1204000},default='')
#imagePath:图片上传路径,跟平时写的model中的路径是一样的
#filePath:富文本中文件的路径,跟平时写的model中的路径是一样的

给xadmin写插件集成ueditor,在xadmin/plugin下新建ueditor.py(名字随意.py),代码如下:

import xadmin
from xadmin.views import BaseAdminPlugin,CreateAdminView,UpdateAdminView
from DjangoUeditor.models import UEditorField
from DjangoUeditor.widgets import UEditorWidget
from django.conf import settings

class XadminUEditorWidget(UEditorWidget):

def __init__(self,**kwargs):
    self.ueditor_options = kwargs
    self.Media.js = None
    super(XadminUEditorWidget,self).__init__(kwargs)

class UeditorPlugin(BaseAdminPlugin):

def get_field_style(self,attrs,db_field,style,**kwargs):
    if style == 'ueditor': #这个字段与adminx中的style_fields = {'detail':'ueditor'} 字段对应
        if isinstance(db_field,UEditorField):
            widget = db_field.formfield().widget
            param = {}
            param.update(widget.ueditor_settings)
            param.update(widget.attrs)
            return {'widget':XadminUEditorWidget(**param)}
        return attrs

def block_extrahead(self,context,nodes):
    js = '<script type="text/javascript" src="%s"></script>' % (settings.STATIC_URL + "ueditor/ueditor.config.js")
    js += '<script type="text/javascript" src="%s"></script>' % (settings.STATIC_URL + "ueditor/ueditor.all.min.js")
        nodes.append(js)

xadmin.site.register_plugin(UeditorPlugin,UpdateAdminView)
xadmin.site.register_plugin(UeditorPlugin,CreateAdminView)

在课程的adminx中CourseAdmin加入代码如下:

class CourseAdmin(object):
    ....
    style_fields = {'detail':'ueditor'}

修改urls中的上传路由:

#富文本相关url(用于上传文件等)
    url(r'^ueditor/', include('DjangoUeditor.urls')),

前端展示,全是转义后的html,关闭django模板转义:

{% autoescape off%}
   {{ course.detail }}
{% endautoescape %}
Last modification:July 1st, 2020 at 08:52 am
如果觉得我的文章对你有用,请随意赞赏

3 comments

  1. Bai Shaofan

    想问下楼主,博客页面也是用什么搭建的?真的非常好看呢!!!

    1. Jolly
      @Bai Shaofan

      哈哈,用的是handsome主题,然后自己美化了下

      1. Bai Shaofan
        @Jolly

        谢谢୧(๑•̀⌄•́๑)૭

Leave a Comment Cancel reply

🌓