Django - Xadmin 组件(二)
Django 自带的 admin 组件可以自定义配置,本文实现 Xadmin 对自定义显示数据列 (list_display) 的配置。
构建表单数据
模板层
从视图函数传来的数据变量是双层列表,第一层是数据库当中的每一条数据,第二层是每一条数据的各个字段值。
类似于
[
["1","King","益达出版社",12],
["2","Fight","西湖出版社",22],
]
所以在模板层通过双循环来取数据。
<tbody>
{% for new_data in new_data_list %}
<tr>
{% for data in new_data %}
<td>{{ data }}</td>
{% endfor %}
</tr>
{% endfor %}
</tbody>
ModelXadmin 类
首先,设置一个列表变量 list_display 存储要显示数据表列信息。该变量不能再视图函数中设置,应该作为类的一个成员变量。该变量需要设置默认值 list_display = ["__str__", ] ,使在没有 Model 配置类的 model 中数据也能正常显示。
显示的列字段分为两种,一种是数据表自带的,一种是个人在配置类中设置的函数。
自带的列字段的处理
判断 list_display 中的元素是否为 str 类型,如果是代表为自带列字段,对每个数据对象用 getattr 函数获取相应的字段值。
# 取数据对象相应的字段
if isinstance(field, str):
val = getattr(data_obj, field)
配置类中函数的处理
如果是函数,就执行该函数,添加的是该函数的返回值。
val = field(self, data_obj)
完整代码:
# 处理表单数据的列表
new_data_list = []
for data_obj in data_list:
temp = []
for field in self.list_display:
# 取数据对象相应的字段
if isinstance(field, str):
val = getattr(data_obj, field)
else:
val = field(self, data_obj)
temp.append(val)
new_data_list.append(temp)
Model 配置类
在每个 app 下面的 Xadmin.py 文件中对数据表进行注册和配置。
list_play 变量
更改自己的 list_display 变量,将想显示的数据表中的字段加入进去。也可以加入函数字段。
函数作为列字段
当作为表单内容时,函数返回的是字符串,字符串可以是 HTML 标签。如果是 HTML 标签,则返回值需要经过 mark_safe 处理。
如果返回的是 a 标签,其中的 href 属性所用链接应当设置为相对路径,并且此处会用到 url 反向解析的内容。
def edit(self, obj=None):
url = reverse('change', args=(obj.pk, ))
return mark_safe('<a href="'+url+'">编辑</a>')
构建表头
模板层
传表头的数据时使用的是一维列表,即每个列字段的名称,所以只需要一个循环。
<thead>
<tr>
{% for item in header_list %}
<td>{{ item }}</td>
{% endfor %}
</tr>
</thead>
ModelXadmin 类
为 list_display 的每个字段都添加相应的表头。分为三种情况,一是默认 list_display 列表,二是自定义的 list_display 列表,三是为函数列字段添加表头。
默认 list_display
若判断 list_display 中的元素为 __str__ 则认为是为默认的 list_display 添加表头,这里我们将 model_name 全部大写作为表头。
if isinstance(field, str):
# 默认list_display
if field == '__str__':
val = self.model._meta.model_name.upper()
自定义 list_display
通过数据表类中的 meta 对象获取相应的字段值,若想显示中文,则应在数据表类中添加 verbose_name 属性。
# 字段为数据表自带
if isinstance(field, str):
# 默认list_display
if field == '__str__':
val = self.model._meta.model_name.upper()
else:
field_obj = self.model._meta.get_field(field)
val = field_obj.verbose_name
函数列字段
传回作为表头的确认信息,由 Model 配置类进行处理。
# is_header判断是否在传表头时调用
val = field(self, is_header=True)
完整添加表头代码:
# 处理表头
header_list = []
# 为list_display的每个字段都添加相应的表头
for field in self.list_display:
# 字段为数据表自带
if isinstance(field, str):
# 默认list_display
if field == '__str__':
val = self.model._meta.model_name.upper()
else:
field_obj = self.model._meta.get_field(field)
val = field_obj.verbose_name
# 字段为自添加函数
else:
# is_header判断是否在传表头时调用
val = field(self, is_header=True)
header_list.append(val)
Model 配置类
别的地方不需要动,在判断传回的是表头时返回自定义表头。
# 用在传表头的时候
if is_header is True:
return "操作"
完整 ModelXadmin 代码:
class ModelXadmin(object):
# 存放要显示哪些字段的列表
list_display = ["__str__", ]
def __init__(self, model, site):
self.model = model
self.site = site
def list_view(self, request):
# 所有的数据
data_list = self.model.objects.all()
# 数据表的名称
model_name = self.model._meta.model_name
# 处理表头
header_list = []
# 为list_display的每个字段都添加相应的表头
for field in self.list_display:
# 字段为数据表自带
if isinstance(field, str):
# 默认list_display
if field == '__str__':
val = self.model._meta.model_name.upper()
else:
field_obj = self.model._meta.get_field(field)
val = field_obj.verbose_name
# 字段为自添加函数
else:
# is_header判断是否在传表头时调用
val = field(self, is_header=True)
header_list.append(val)
# 处理表单数据的列表
new_data_list = []
for data_obj in data_list:
temp = []
for field in self.list_display:
# 取数据对象相应的字段
if isinstance(field, str):
val = getattr(data_obj, field)
else:
val = field(self, data_obj)
temp.append(val)
new_data_list.append(temp)
return render(request,
"list_view.html",
{
"data_list": data_list,
"model_name": model_name,
"new_data_list": new_data_list,
"header_list": header_list,
}
)
完整 Model 配置类 BookConfig 代码:
class BookConfig(ModelXadmin):
def edit(self, obj=None, is_header=False):
# 用在传表头的时候
if is_header is True:
return "操作"
url = reverse('change', args=(obj.pk, ))
return mark_safe('<a href="'+url+'">编辑</a>')
def delete(self, obj=None, is_header=False):
# 用在传表头的时候
if is_header is True:
return "操作"
url = reverse('delete', args=(obj.pk,))
return mark_safe('<a href="'+url+'">删除</a>')
def check(self, obj=None, is_header=False):
# 用在传表头的时候
if is_header is True:
return "操作"
return mark_safe('<input type="checkbox">')
list_display = [check, "nid", "title", "price", "publish", edit, delete]
完整模板代码:
<h3>查看{{ model_name }}数据</h3>
<div class="container">
<div class="row">
<div class="col-md-9">
<table class="table table-bordered table-striped">
<thead>
<tr>
{% for item in header_list %}
<td>{{ item }}</td>
{% endfor %}
</tr>
</thead>
<tbody>
{% for new_data in new_data_list %}
<tr>
{% for data in new_data %}
<td>{{ data }}</td>
{% endfor %}
</tr>
{% endfor %}
</tbody>
</table>
</div>
</div>
</div>
完成效果

GitHub 地址:https://github.com/protea-ban/oldboy/tree/master/s9day84/Xadmindemo

Django - Xadmin 组件(二)的更多相关文章
- Django - Xadmin 组件(一)
Django - Xadmin 组件(一) Web 应用中离不开的就是后台管理, Django 自带的 admin 组件提供了一部分内容,但往往现实项目中会有更多的需求,所以自定义自己的后台管理就十分 ...
- Django+xadmin打造在线教育平台(二)
三.xadmin后台管理 3.1.xadmin的安装 django2.0的安装(源码安装方式): https://github.com/sshwsfc/xadmin/tree/django2 把zip ...
- 第四百零二节,Django+Xadmin打造上线标准的在线教育平台—生产环境部署,uwsgi安装和启动,nginx的安装与启动,uwsgi与nginx的配置文件+虚拟主机配置
第四百零二节,Django+Xadmin打造上线标准的在线教育平台—生产环境部署,uwsgi安装和启动,nginx的安装与启动,uwsgi与nginx的配置文件+虚拟主机配置 软件版本 uwsgi- ...
- Django框架(二十)—— Django rest_framework-认证组件
目录 Django rest_framework-认证组件 一.什么是认证 二.利用token记录认证过的用户 1.什么是token 2.token的原理 3.cookie.session.token ...
- Django+xadmin打造在线教育平台(三)
五.完成注册.找回密码和激活验证码功能 5.1.用户注册 register.html拷贝到templates目录 (1)users/views.py class RegisterView(View): ...
- django Form组件
django Form组件 Django的Form主要具有一下几大功能: 生成HTML标签 验证用户数据(显示错误信息) HTML Form提交保留上次提交数据 初始化页面显示内容 小试牛刀 1.创建 ...
- Django+xadmin打造在线教育平台(一)
目录 在线教育平台(一) 在线教育平台(二) 在线教育平台(三) 在线教育平台(四) 在线教育平台(五) 在线教育平台(六) 在线教育平台(七) 在线教育平台( ...
- Django admin组件源码流程
admin 组件 Django 自带的用户后台组件 用于用户便携的操作 admin 组件核心 启动 注册 设计url 启动核心代码 每个app 通过 apps.py 扫描 admin.py 文件 并执 ...
- xadmin 组件拓展自定义使用
xadmin 组件相关可选自定义字段 list_display 功能 设置默认的显示字段(列) 配置 list_display = ['name', 'desc', 'detail', 'degree ...
随机推荐
- C++11中lock_guard和unique_lock的区别
c++11中有一个区域锁lock_guard,还有第二个区域锁unique_lock. 区域锁lock_guard使用起来比较简单,除了构造函数外没有其他member function,在整个区域都有 ...
- web Servise(服务)
服务层:对项目的业务层(业务层调用数据层)的一个包装成对外的服务,是的UI展示可以从单一的WEB扩展为移动端可WINDFROM端等,这叫做面向服务的编程思想. 发布:和网站发布也是一样的. //web ...
- 383. Ransom Note 在字典数组中查找笔记数组
[抄题]: Given an arbitrary ransom note string and another string containing letters from all the magaz ...
- SpringMVC 课程第一天
SpringMVC第一天 框架课程 1. 课程计划 第一天 1.SpringMVC介绍 2.入门程序 3.SpringMVC架构讲解 a) 框架结构 b) 组件说明 4.SpringMVC整合My ...
- 5.WHERE 子句
WHERE 子句用于规定选择的标准. WHERE 子句 如需有条件地从表中选取数据,可将 WHERE 子句添加到 SELECT 语句. 语法 SELECT 列名称 FROM 表名称 WHERE 列 运 ...
- 第十二课 Actionlib(1)
一\Actionlib概念 在ROS系统中,有时需发送请求给某个节点完成相应的任务,同时获得一个一个响应,这种情况下可以通过ROS服务来 完成;然而,在某些情况下,服务需要很长时间才能执行完,如让机器 ...
- CSS3动画设置后台登录页背景切换图片
CSS3的动画很实用很好用,简单几句话就可以做出一个好看而且过渡平滑的body背景图片,不多说,来来来,上代码 body{ animation:mybg 7s; -webkit-animation:m ...
- 【Azure Active Directory】单一登录 (SAML 协议)
Azure Active Directory 支持 SAML 2.0 Web 浏览器单一登录 (SSO) 配置文件. 若要请求 Azure Active Directory 对用户进行身份验证时,云服 ...
- 请教一个Jquery ligerui 框架的小问题
关闭子窗体时,要刷新父窗体,百度了很多像使用“window.opener.location.reload();”都不行,和easyui框架是有区别的 在子窗体里写Response.Write(&quo ...
- C# 中请使用Contains判断字符串是否包含另一段字符串
∵ :使用Contains 比 IndexOf 的性能要高很多. 因为 Contains 是判断某个字符串是否在该字符串里面,而IndexOf是返回对应下标值 但是在使用contains的时候,注意转 ...