直接上代码:

# spark/service/sites.py
from django.conf.urls import url
from django.shortcuts import HttpResponse, render
from django.urls import reverse
from django.utils.safestring import mark_safe class ModelStark(object): def __init__(self, model):
self.model = model
self.model_name = self.model._meta.model_name
self.app_label = self.model._meta.app_label
self.list_display += ModelStark.extra_display def edit(self, obj=None, is_header=None):
if is_header:
return '编辑'
name = "{}_{}_change".format(obj._meta.app_label, obj._meta.model_name)
url = reverse(name, args=(obj.pk,))
return mark_safe("<a href='{}'>编辑</a>".format(url)) def delete(self, obj=None, is_header=None):
if is_header:
return '删除'
name = "{}_{}_delete".format(obj._meta.app_label, obj._meta.model_name)
url = reverse(name, args=(obj.pk,))
return mark_safe("<a href='{}'>删除</a>".format(url)) extra_display = [edit, delete]
list_display = ["__str__"] def list_view(self, request):
# 浏览器标题
name = self.model_name
# 获取数据头
header_list = []
for field_or_func in self.list_display:
# 判断如果是默认配置类的list_display,则数据头为模型表名大写
if field_or_func == '__str__':
header_list.append(self.model_name.upper())
elif callable(field_or_func):
header_list.append(field_or_func(self=self, is_header=True))
else:
field_obj = self.model._meta.get_field(field_or_func)
header_list.append(field_obj.verbose_name)
# 分页器
from utils.pagenator import Pagenation
current_page = request.GET.get('page')
if not current_page:
current_page = 1
base_url = request.path_info
total_books_num = self.model.objects.all().count()
my_pagenator = Pagenation(current_page=current_page, base_url=base_url, books_per_page=10, max_show=7,
total_books_num=total_books_num)
page_html = my_pagenator.page_html
start = my_pagenator.start
end = my_pagenator.end
one_page_data_list = self.model.objects.all()[start:end] # 获取数据表所需数据
new_data_list = []
from django.db.models.fields.related import ManyToManyField
for obj in one_page_data_list:
temp = []
for field_or_func in self.list_display:
if callable(field_or_func):
val = field_or_func(self, obj=obj)
elif field_or_func == '__str__':
val = getattr(obj, field_or_func)
else:
field_obj = self.model._meta.get_field(field_or_func)
if isinstance(field_obj, ManyToManyField):
many_queryset = getattr(obj, field_or_func).all()
val = '|'.join([str(item) for item in many_queryset])
else:
val = getattr(obj, field_or_func)
temp.append(val)
new_data_list.append(temp)
return render(request, 'list_view1.html', locals()) def add_view(self, request):
return HttpResponse('add_view') def change_view(self, request, pk):
return HttpResponse('change_view') def del_view(self, request, pk):
return HttpResponse('del_view') def get_urls(self):
tmp = [] tmp.append(url(r'^$', self.list_view, name="{}_{}_list".format(self.app_label, self.model_name)))
tmp.append(url(r'add/$', self.add_view, name="{}_{}_add".format(self.app_label, self.model_name)))
tmp.append(url(r'(\d+)/delete/$', self.del_view, name="{}_{}_delete".format(self.app_label, self.model_name)))
tmp.append(
url('(\d+)/change/$', self.change_view, name="{}_{}_change".format(self.app_label, self.model_name)))
return tmp, None, None @property
def urls(self):
return self.get_urls() class AdminSite(object):
def __init__(self):
self._registy = {} def register(self, model, stark_class=None):
if not stark_class:
stark_class = ModelStark self._registy[model] = stark_class(model) def get_urls(self):
tmp = []
for model, config_obj in self._registy.items():
tmp.append(url(r"{}/{}/".format(model._meta.app_label, model._meta.model_name), config_obj.urls))
return tmp, None, None @property
def urls(self):
return self.get_urls() site = AdminSite()

# spark/service/sites.py

from stark.service.sites import site, ModelStark
from app01 import models
from django.utils.safestring import mark_safe
from django.urls import reverse print('app01................') class BookStark(ModelStark):
list_display = ['title', 'price', 'publisher', 'authors', ] site.register(models.Book, BookStark) site.register(models.Publish)

# app01/stark.py

{% load static %}
<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<title>{{ name }}</title>
<link rel="stylesheet" href="{% static 'css/bootstrap.css' %}"> {# <link href="https://cdn.bootcss.com/bootstrap/3.3.7/css/bootstrap.css" rel="stylesheet">#}
</head>
<body>
<div class="container"> <div class="row">
<div class="col-md-8"> <table class="table table-bordered">
<thead>
<tr>
{% for list in header_list %}
<th>
{{ list }}
</th>
{% endfor %}
</tr>
</thead>
<tbody>
{% for data in new_data_list %}
<tr>
{% for field in data %}
<td>{{ field }}</td>
{% endfor %}
</tr> {% endfor %}
</tbody>
</table> <div>
{{ page_html|safe }}
</div>
</div>
</div>
</div>
</body>
</html>

list_view1.html

from django.shortcuts import render, redirect

class Pagenation():
def __init__(self, current_page, base_url, books_per_page, max_show, total_books_num):
""" :param current_page: 当前访问页面,从request.GET.get('page',1)中获取
:param base_url: 需要访问的url地址
:param books_per_page: 每页显示几条数据
:param max_show: #最多显示几个分页
:param total_books_num: # 传入总计多少条数据记录
"""
self.current_page = current_page
self.base_url = base_url
self.books_per_page = books_per_page
self.max_show = max_show
self.total_books_num = total_books_num
self.__data_start = 0
self.__data_end = 0
self.page_html @property
def start(self):
""" :return: 返回切片数据的开始位置
"""
return self.__data_start @property
def end(self):
"""
返回切片数据的结束位置
:return:
"""
return self.__data_end @property
def page_html(self): """ :return: 返回分页的HTML代码,如果访问不在1~最大页码范围则返回redirect对象,引用的时候要注意。+
"""
page_start = 0
page_end = 0
if self.total_books_num <= 0:
return '没有数据!' total_pages_num, mod = divmod(self.total_books_num, self.books_per_page)
if mod:
total_pages_num = total_pages_num + 1
try:
current_page = int(self.current_page)
except:
current_page = 1 half_num = self.max_show // 2
if current_page <= 0:
current_page = 1
# return redirect('{}?page={}'.format(self.base_url, 1))
elif current_page > total_pages_num:
current_page = total_pages_num
# return redirect('{}?page={}'.format(self.base_url, total_pages_num))
if current_page <= half_num:
page_start = 1
if total_pages_num >= self.max_show:
page_end = self.max_show
else:
page_end = total_pages_num
else:
if current_page + half_num > total_pages_num:
page_end = total_pages_num
if total_pages_num > self.max_show:
page_start = total_pages_num - self.max_show + 1
else:
page_start = 1
else:
page_start = current_page - half_num
page_end = current_page + half_num
self.__data_start = (current_page - 1) * self.books_per_page
self.__data_end = current_page * self.books_per_page
page_html_list = []
ul_start = '<ul class="pagination">'
page_html_list.append(ul_start)
previous_page = '<li class ="enable"><a href="{}?page={}" aria-label="Previous">' \
'<span aria-hidden="true"> « </span> </a> </li>'.format(self.base_url, current_page - 1)
page_html_list.append(previous_page)
first_page = '<li> <a href = "{}?page=1"> 首页 </a> </li>'.format(self.base_url)
page_html_list.append(first_page)
for i in range(page_start, page_end + 1):
if i == current_page:
li = '<li class="active"> <a href="{}?page={}">{}</a> </li>'.format(self.base_url, i, i)
else:
li = '<li> <a href="{}?page={}">{}</a> </li>'.format(self.base_url, i, i)
page_html_list.append(li)
end_page = '<li> <a href="{}?page={}">尾页</a> </li>'.format(self.base_url, total_pages_num)
page_html_list.append(end_page)
last_page = '<li class ="enable"><a href="{}?page={}" aria-label="Next">' \
'<span aria-hidden="true"> » </span> </a> </li>'.format(self.base_url, current_page + 1)
page_html_list.append(last_page)
ul_end = '</ul>'
page_html_list.append(ul_end)
page_html = ''.join(page_html_list)
return page_html

分页器

python全栈开发day85-查:数据表 数据头 增加列 展示多对多字段 反向解析编辑和删除按钮的url的更多相关文章

  1. python 全栈开发,Day114(装饰器,排序规则,显示列,添加按钮,定制ModelForm,自定义列表页面,自定制URL)

    一.装饰器 装饰器本质上就是一个python函数,他可以让其他函数在不需要做任何代码变动的前提下,增加额外的功能,装饰器的返回值也是一个函数对象. 装饰器的应用场景:比如插入日志,性能测试,事务处理, ...

  2. python 全栈开发,Day6(is,小数据池,编码转换)

    一.is a = 100 b = 100 print(a == b) print(a is b) 执行输出: TrueTrue 查看内存地址,使用id函数 print(id(a)) print(id( ...

  3. 巨蟒python全栈开发django11:ajax&&form表单上传文件contentType

    回顾: 什么是异步? 可以开出一个线程,我发出请求,不用等待返回,可以做其他事情. 什么是同步? 同步就是,我发送出了一个请求,需要等待返回给我信息,我才可以操作其他事情. 局部刷新是什么? 通过jq ...

  4. python全栈开发-json和pickle模块(数据的序列化)

    一.什么是序列化? 我们把对象(变量)从内存中变成可存储或传输的过程称之为序列化,在Python中叫pickling,在其他语言中也被称之为serialization,marshalling,flat ...

  5. python 全栈开发,Day80(博客系统分析,博客主页展示)

    一.博客系统分析 数据库的构建 首先,我们分析一个博客系统的功能: 一个博客可以有多个标签(多对多) 一个博客可以有多条评论(一对多) 一个博客只可以有一个类别(多对一) 接下来,我们分析关系的属性: ...

  6. python全栈开发中级班全程笔记(第二模块、第三章)(员工信息增删改查作业讲解)

    python全栈开发中级班全程笔记 第三章:员工信息增删改查作业代码 作业要求: 员工增删改查表用代码实现一个简单的员工信息增删改查表需求: 1.支持模糊查询,(1.find name ,age fo ...

  7. Python全栈开发【面向对象】

    Python全栈开发[面向对象] 本节内容: 三大编程范式 面向对象设计与面向对象编程 类和对象 静态属性.类方法.静态方法 类组合 继承 多态 封装 三大编程范式 三大编程范式: 1.面向过程编程 ...

  8. Python全栈开发【模块】

    Python全栈开发[模块] 本节内容: 模块介绍 time random os sys json & picle shelve XML hashlib ConfigParser loggin ...

  9. python 全栈开发之路 day1

    python 全栈开发之路 day1   本节内容 计算机发展介绍 计算机硬件组成 计算机基本原理 计算机 计算机(computer)俗称电脑,是一种用于高速计算的电子计算机器,可以进行数值计算,又可 ...

随机推荐

  1. 视觉SLAM之词袋(bag of words) 模型与K-means聚类算法浅析

    原文地址:http://www.cnblogs.com/zjiaxing/p/5548265.html 在目前实际的视觉SLAM中,闭环检测多采用DBOW2模型https://github.com/d ...

  2. Mudo C++网络库第七章学习笔记

    muduo编程示例 muduo库是设计来开发内网的网络程序, 它没有做任何安全方面的加强措施, 如果在公网上可能会受到攻击; muduo库把主动关闭连接这件事分成两步来做: 如果主动关闭连接, 会先关 ...

  3. new-delete-malloc-free关系总结

    new-delete-malloc-free关系总结 写在前面的话 这个系列的笔记总结是根据网上的两篇基础拓展而来的 C++经典面试题(最全,面中率最高) C++面试集锦( 面试被问到的问题 ) 面试 ...

  4. 数据库join union 区别

    join 是两张表做交连后里面条件相同的部分记录产生一个记录集,union是产生的两个记录集(字段要一样的)并在一起,成为一个新的记录集. 1.JOIN和UNION区别  join 是两张表做交连后里 ...

  5. ubuntu安装 opencv-3.4.3

    1 .官网(https://opencv.org/releases.html)下载下源码 opencv-3.4.3.zip 2.解压 unzip  opencv-3.4.3.zip 3.cmake c ...

  6. Android来电拦截及来电转移

    1. 电话拦截这个功能大家可能都知道了,就是利用反射原理调用ITelephony的隐藏方法来实现.这个就不说了,在附件的代码里有.2.拦截后提示忙音/空号/已关机/已停机这个功能其实是要用到MMI指令 ...

  7. web中绝对路径换虚拟路径

    最近在做一个web项目,将图片上传到服务器后,再访问时拿到的是绝对路劲,而需要的是虚拟路劲.经过一番折腾找到了下列方法可以直接转换. /// <summary>        /// 将W ...

  8. Hibernatede 一对多映射配置

    Hibernatede 一对多映射配置 以公司和员工为例:公司是一,员工是多   第一步 创建两个实体类,公司和员工        写核心配置文件hibernate.cfg.xml        写映 ...

  9. java 中int与integer的区别

    int与integer的区别从大的方面来说就是基本数据类型与其包装类的区别: int 是基本类型,直接存数值,而integer是对象,用一个引用指向这个对象 1.Java 中的数据类型分为基本数据类型 ...

  10. LeetCode(124):二叉树中的最大路径和

    Hard! 题目描述: 给定一个非空二叉树,返回其最大路径和. 本题中,路径被定义为一条从树中任意节点出发,达到任意节点的序列.该路径至少包含一个节点,且不需要经过根节点. 示例 1: 输入: [1, ...