自定义Django Admin的action

在Django Admin中,可以通过action来自定义一些操作,其中默认的action的功能是选中多条数据来进行删除操作

我们在king_admin中也可以通过开发类似的功能性组件来实现自定义action

首先在king_admin中添加actions字段

actions = ['delete_selected_objs',] #和django admin一样,添加默认删除选中的多条记录的方法

在views视图中,获取前端页面传来的选中的记录id和action的函数名

  #post请求执行action操作
if request.method == 'POST':
selected_ids = request.POST.get('selected_ids')#获取选中记录的id
action_name = request.POST.get('action')#获取函数名
if selected_ids:
       #根据选中记录的id去数据库中获取记录
selected_objs = admin_class.model.objects.filter(id__in=selected_ids.split(','))
else:
raise KeyError('没有数据')
     #如果在admin_class中找到了该函数,则获取该函数,然后将admin_class,request和查询的记录传递到king_admin.py中的函数
if hasattr(admin_class,action_name):
action_func = getattr(admin_class,action_name)
request._action = action_name
return action_func(admin_class,request,selected_objs)

在king_admin.py中,获取这些参数,第一次调用时候,返回预删除页面,将值回填到预删除页面,当点击确认删除,此时delete_confrim值可获取到,

并且执行删除操作,执行完毕跳转到展示页

 def delete_selected_objs(self,request,query_set):
app_name = self.model._meta.app_label #获取app的名称
model_name = self.model._meta.model_name #获取表名
selected_ids = ','.join([str(i.id) for i in query_set]) #获取选中记录的id
if request.POST.get('delete_confirm') == 'yes':
       #当在待删除页面,点击确认删除的时候,delete_confirm才会有值,第一次进入该方法时,这里的值为空 
query_set.delete()
return redirect('/king_admin//%s/%s/' % (app_name,model_name))
#返回到待删除页面,传递的参数分别是(要删除的记录,admin_class,app名称,表名,选中的id,和要调用的方法)
#主要是为了点击确认删除时候,页面跳转到views中的方法,此时selected_id和action_name的值要回填,否则报错
return render(request,'king_admin/table_delete.html',{'model_obj':query_set,
'admin_class':self,
'app_name':app_name,
'table_name':model_name,
'selected_ids':selected_ids,
'action':request._action})

前端页面

作用: 1.在下拉列表中展示可调用的action

2.checkbox的多条数据删除和数据校验

<div class="row">
<form method="POST" onsubmit="return submitAction(this)">{% csrf_token %}
<div class="col-lg-2">
<select name="action" id="action_list" class="form-control" style="margin-top:20px">
<option value="">-----</option>
{% for action in admin_class.actions %}
<option value="{{ action }}">{% build_action_name admin_class action %}</option>
{% endfor %}
</select>
</div>
<div class="col-lg-2">
<button type="submit" class="btn" style="margin-top:20px;margin-left:-10px">执行</button>
</div>
</form>
</div>
<table class="table table-hover">
<thead>
<tr>
<th><input type="checkbox" onclick="checkAllToggle(this)"></th>
{% for column in admin_class.list_display %}
{# 动态展示表格中的数据,排序关键字,和筛选条件#}
{% build_table_header_column column orderby_key filter_conditions %}
{% endfor %}
</tr>
</thead>
<tbody>
{# <!--动态展示后端的表格中的数据--!>#}
{% for obj in query_sets %} <tr>
<td><input tag="obj_check" type="checkbox" value="{{ obj.id }}"/></td>
{% bulid_table_row request obj admin_class %}
</tr>
{% endfor %}
</tbody>
</table>

js部分

function checkAllToggle(ths){
//复选框全选,反选
if($(ths).prop('checked')){
$("input[tag='obj_check']").prop('checked',true);
}
else {
$("input[tag='obj_check']").prop('checked', false);
}
} function submitAction(ths){
//后台提交action所要的数据
selected_ids = [];
//将复选框选中的值放到列表中
$("input[tag='obj_check']:checked").each(function(){
selected_ids.push($(this).val());
});
//获取action的名字
var select_action = $('#action_list').val();
console.log(select_action)
if(selected_ids.length == 0){
alert('没有选择数据');
return;
}
if(!select_action){
alert('没有选择方法');
return;
} //添加已经选择的id,放到隐藏域中,一起提交到后台
var selected_id_ele = "<input name='selected_ids' type='hidden' value='"+selected_ids.toString()+"'/>";
$(ths).append(selected_id_ele);
return true;
}

预删除页面展示

在预删除页面,加入回填的参数,在点击确认删除时,获取king_admin中传递到前端的值,在把这些值第二次传递给后端的views中,此时views中会第二次调用

king_admin中的action的方法,同时获取到delete_confirm的值,执行删除操作,删除完毕后跳转首页

{% block container %}

    {% display_obj_related model_obj %}

    <form method="post">{% csrf_token %}
<input type="submit" class="btn btn-danger" value="确认删除">
<input type="hidden" value="yes" name="delete_confirm">
<input type="hidden" value="{{ selected_ids }}" name="selected_ids">
<input type="hidden" value="{{ action }}" name="action">
<a class="btn btn-info" href="{% url 'table_objs' app_name table_name %}">返回</a>
</form> {% endblock %}

最后一步,可以通过在action中自定义显示的函数名

class CustomerAdmin(BaseAdmin):
list_display = ['qq','name','source','consultant','consult_course','date','status']
list_filters = ['source','consultant','consult_course','status','date']
search_fields = ['qq','name','consultant__name']
list_per_page = 5
ordering = 'id'
filter_horizontal = ['tags',]
#继承父类中的action,每个类自定义的action
actions = ['delete_selected_objs','test',] def test(self,request,query_set):
print('test...')
#加上这条属性表示在action下拉框中展示此字段
test.display_name = '测试'

在tags中渲染

@register.simple_tag
def build_action_name(admin_class,action):
#获取admin_class的action对象
action_func = getattr(admin_class,action)
#如果有display_name就返回display_name的名称,否则就展示action字段
return action_func.display_name if hasattr(action_func,'display_name') else action

前端页面展示

<select name="action" id="action_list" class="form-control" style="margin-top:20px">
<option value="">-----</option>
{% for action in admin_class.actions %}
{# <!--自定义标签中去显示下拉框的名称--!>#}
<option value="{{ action }}">{% build_action_name admin_class action %}</option>
{% endfor %}
</select>

Python CRM项目六的更多相关文章

  1. Python CRM项目二

    一.准备工作 如果没有配置基本的项目,请参考 http://www.cnblogs.com/luhuajun/p/7771196.html 当我们配置完成后首先准备我们的app 创建2个app分别对应 ...

  2. Python CRM项目八

    自定义用户认证 目的:实现Django自定义的认证系统,在生产环境都是根据此代码进行定制的 步骤: 1.在settings文件中配置要使用的类 #命名规则 app名称.类名 AUTH_USER_MOD ...

  3. Python CRM项目一

    开发环境: 语言Python3.X以上 MTV WEB框架 Django 前端框架 jQuery+bootstrap 数据库 MySQL 运行环境 安装Python3.x 安装Django 除IE8以 ...

  4. Python CRM项目三

    1.分页: 分页使用Django内置的分页模块来实现 官方的分页案例 from django.core.paginator import Paginator, EmptyPage, PageNotAn ...

  5. python实践项目六:正则表达式-强口令

    描述:写一个函数,它使用正则表达式,确保传入的口令字符串是强口令.强口令的定义是:长度不少于8 个字符,  同时包含大写和小写字符, 至少有一位数字. 代码: #!/usr/bin/python # ...

  6. Python CRM项目七

    仿照Django Admin实现对readonly的字段进行设置 功能点: 1.页面不可进行更改 2.如果改变html代码中的值,则需要进行后端的数据库数据校验 3.可以对某些字段进行自定制校验规则 ...

  7. Python CRM项目四

    实现Django Admin的多对多的复选框效果 效果:左边显示的是未选中的字段,右边显示的是已选中的字段,两边点击的标签可以互相更换 首先在king_admin.py中增加filter_horizo ...

  8. crm项目开发之架构设计

    CRM customer relationship management 客户管理系统 1. 干什么用的? 管理客户 维护客户关系 2. 谁去使用? 销售 班主任 项目经理 3. 需求: 1. 登录 ...

  9. CRM项目总结

                CRM项目总结      一:开发背景 在公司日益扩大的过程中,不可避免的会伴随着更多问题出现. 对外 : 如何更好的管理客户与公司的关系?如何更及时的了解客户日益发展的需求变 ...

随机推荐

  1. 把玩爬虫框架Gecco

    如果你现在接到一个任务,获取某某行业下的分类. 作为一个非该领域专家,没有深厚的运营经验功底,要提供一套摆的上台面且让人信服的行业分类,恐怕不那么简单. 找不到专家没有关系,我们可以爬虫.把那些专家的 ...

  2. JDBC(二)之JDBC处理CLOB和BLOB及事务与数据库元数据获取

    前面大概介绍了JDBC连接数据库的过程,以及怎么操作数据库,今天给大家分享JDBC怎么处理CLOB和BLOB存储图片的事情,以及JDBC怎么去处理事务.怎么在插入数据的时候生成主键返回值 一.JDBC ...

  3. js面向对象学习笔记(一):创建空对象,理解this指向

    var obj = new Object();//创建一个空对象 obj.name = '小王';//属性 obj.sayName = function () { //对象方法 对象最重要的是this ...

  4. HDU 1847 Good Luck in CET-4 Everybody!(规律,博弈)

    Good Luck in CET-4 Everybody! Time Limit: 1000/1000 MS (Java/Others)    Memory Limit: 32768/32768 K ...

  5. 2016 USP-ICMC-Codeforces-Gym101063C-Sleep Buddies Gym101063F-Bandejao Gym101063J-The Keys

    Gym101063C-Sleep Buddies It is nighttime in the Earth Colony on Mars and everyone is getting ready t ...

  6. 2017多校第一套&&hdu6038 思维 数学

    链接  http://acm.hdu.edu.cn/showproblem.php?pid=6038 题意: 给你一个a序列,代表0到n-1的排列:一个b序列代表0到m-1的排列.问你可以找出多少种函 ...

  7. 数位dp初探

    我这种蒟蒻就一直不会写数位dp.. 于是开了个坑.. 1833: [ZJOI2010]count 数字计数 这道被KPM大爷说是入门题..嗯似乎找找规律然后减掉0的情况后乱搞就可以了..(但是还是写了 ...

  8. malloc函数用法

    malloc函数用法 函数声明(函数原型): void *malloc(int size); 说明:malloc 向系统申请分配指定size个字节的内存空间.返回类型是 void* 类型.void* ...

  9. Android开发——BroadcastReceiver广播的使用

    想要了解广播定义及相关原理的可以看下这一篇BroadcastReceiver史上最全面解析 简单地对广播进行分类吧,广播有两个角色,一个是广播发送者,另外一个是广播接收者 广播按照类型分为两种,一种是 ...

  10. Spark算子--coalesce和repartition

    coalesce和repartition--Transformation类算子 代码示例