添加:

  需求: 根据用户的权限, 决定是否,有添加按钮。  通过配置进行定制,预留钩子进行权限的判断。

class StartHandler(object):
  ..................... has_add_btn = True # 指定配置,默认显示。 用户在子类中,自定制是否显示
def get_add_btn(self):
'''预留钩子,子类中重写该方法。 根据权限的判断是否显示添加按钮'''
if self.has_add_btn:
return "<a class='btn btn-primary'>添加</a>"
return None   def check_list_view(self, request, *args, **kwargs):
    # ################6. 处理添加按钮######################
    add_btn = self.get_add_btn()
    return render(request, "stark/changelist.html",
{"header_list": header_list, "data_list": data_list,
"body_list": body_list,
"pager": pager,
"add_btn": add_btn,
"search_list": search_list,
"search_value": search_value,
"action_dict": action_dict})   ..................
class UserInfoHandler(StartHandler):
list_display = ["name", "age", "depart", get_choice_txt("性别", "gender"), StartHandler.display_edit,
StartHandler.display_del]
has_add_btn = True # False就表示, 不显示添加按钮

get_add_btn(self):  #  可以进行重写。 权限的判断,或者,重写样式

      pass

如果感觉,默认的样式不好看!
  可以在, 子类中, 重写get_add_btn() 方法。 返回一个,自己喜欢的样式的 a 标签。 就行了!

然后,就是这个 a 标签的, 路有问题了!  并且还要,携带上,本次请求的  GET 的数据。
老套路: 自定义一个反向解析的,函数:

    def reverse_url(self):
'''用于反向生成url, 并且携带,get请求的参数,跳转到下一个网页'''
name = "%s:%s" % (self.site.namespace, self.get_add_url_name)
base_url = reverse(name)
# 记录原搜索条件
if not self.request.GET:
add_url = base_url
else:
param = self.request.GET.urlencode() # 获取到GET请求的,所有的参数。 ?page=1&age=20
new_query_dict = QueryDict(mutable=True)
new_query_dict["_filter"] = param
add_url = "%s?%s" % (base_url, new_query_dict.urlencode())
return add_url

这里最重要的一个就是, self.request  这个东西, 是从前端传过来的, 我在初始化函数里面,定义了一下。 但是 他从哪里传给我的初始化函数呢?  就是在,获取 URL 的时候。

    def wrapper(self, func):
@functools.wraps(func) # 保留原函数的 原信息
def inner(request, *args, **kwargs): # 这个inner 就是,我的每一个视图函数了!
self.request = request
return func(request, *args, **kwargs)
return inner def get_urls(self):
partterns = [
re_path(r"list/$", self.wrapper(self.check_list_view), name=self.get_list_url_name),
re_path(r"add/$", self.wrapper(self.add_view), name=self.get_add_url_name),
re_path(r"change/(\d+)/$", self.wrapper(self.change_view), name=self.get_edit_url_name),
re_path(r"del/(\d+)/$", self.wrapper(self.delete_view), name=self.get_del_url_name),
]

为了避免麻烦, 使用了 闭包的方式。来做这件事!
解释一下:


    def wrapper(self, func):
@functools.wraps(func) # 保留原函数的 原信息
def inner(request, *args, **kwargs): # 这个inner 就是,我的每一个视图函数了!
self.request = request
return func(request, *args, **kwargs)
return inner
self.wrapper(self.check_list_view) 这个函数的工作其实就是: 重新赋值。 装饰器的套路
self.check_list_view = self.wrapper(self.check_list_view)  # self.check_list_view == inner 可以这么理解
self.wrapper的返回值是  inner 函数的内存地址。所以其实执行  self.check_list_view  就相当于执行的是 inner。
re_path() 这个django内部的函数。 认为 self.wrapper(self.check_list_view) 是一个视图函数。 所以会给他带上一个 request 的参数。

注:self.wrapper(self.check_list_view) 这个一位加了 括号, 所以会先执行。并返回 inner。 而re_path 认为inner 就是视图函数。

所以我才能够在 inner 这个函数这里, 直接收到这个 request 。 然后我就将他 赋值给了, __init__ 初始化函数中的。 self.request = None

这样这参数,就不再是 None  而是一个,带着从前端返回的,带有参数的 request 对象。
我就可以,在程序的, 其他地方。 使用这个参数。

整体结构,就是这样:

    def get_add_btn(self):
'''预留钩子,子类中重写该方法。 根据权限的判断是否显示添加按钮'''
if self.has_add_btn:
# 根据别名反向生成, URL
add_url = self.reverse_url()
return "<a class='btn btn-primary' href='%s'>添加</a>" % add_url
return None def reverse_url(self):
'''用于反向生成url, 并且携带,get请求的参数,跳转到下一个网页'''
name = "%s:%s" % (self.site.namespace, self.get_add_url_name)
base_url = reverse(name)
# 记录原搜索条件
if not self.request.GET:
add_url = base_url
else:
param = self.request.GET.urlencode() # 获取到GET请求的,所有的参数。 ?page=1&age=20
new_query_dict = QueryDict(mutable=True)
new_query_dict["_filter"] = param
add_url = "%s?%s" % (base_url, new_query_dict.urlencode())
return add_url per_page = 10 # 默认每页显示,多少数据。 也可在子类中,自行定制 def check_list_view(self, request):
'''
列表查看页面
:param request:
:return:
'''
# self.request = request # 进入查看页面,为request赋值! 使其他地方可以用到!
list_display = self.get_list_display()
# 页面要显示的列 self.list_display 示例:['name', 'age', 'depart'] # 1. 制作表头, 就是每张表中,每个字段写的 verbose_name.。 如何获取到这个值呢?
# self.model_class._meta.get_field('name').verbose_name
header_list = [] # 表头
if list_display:
for key_or_func in list_display:
if isinstance(key_or_func, FunctionType): # 判断当前参数, 是一个字符串还是一个函数。
verbose_name = key_or_func(self, obj=None, is_header=True)
else:
verbose_name = self.model_class._meta.get_field(key_or_func).verbose_name
header_list.append(verbose_name)
else:
header_list.append(self.model_class._meta.model_name) # 2. 处理 从数据库 取到的数据 # 用户访问的表 self.model_class
# 2.1 ###############处理分页#################
'''1.根据用户访问页面,计算出索引的位置, 比如 page=3
2. 生成html页码
'''
all_count = self.model_class.objects.all().count()
query_params = request.GET.copy() # page=1&level=2
query_params._mutable = True # request.get中的值默认是不能被修改的。加上这句代码就可以修改了 pager = Pagination(
current_page=request.GET.get("page"), # 用户访问的当前叶
all_count=all_count, # 数据库一共有多少数据
base_url=request.path_info, # 所在的url 就是 ?page=1 之前的URL
# 用于保留,用户的请求信息,比如 level=2 被用户先选中。 那么分页后。因为查询的东西少了,分页也应该想要的减少,
# 但是level=2这个, 请求的信息!不能因为。分页的原因。而减少。
query_params=query_params,
per_page=self.per_page, # 每页显示多少数据。
) # 2.1 ###############处理表格#################
data_list = self.model_class.objects.all()[pager.start:pager.end] body_list = []
for row in data_list:
row_list = []
if list_display:
for key_or_func in list_display:
if isinstance(key_or_func, FunctionType):
# 这里is_header=False obj=row(数据库中循环的每一行的对象)
row_list.append(key_or_func(self, obj=row, is_header=False))
else:
row_list.append(getattr(row, key_or_func))
else:
row_list.append(row)
body_list.append(row_list)
# 3 ############# 处理添加按钮####################
add_btn = self.get_add_btn() # 在这里调用了!此方法! return render(request, "stark/changelist.html",
{"header_list": header_list, "data_list": data_list,
"body_list": body_list,
"pager": pager,
"add_btn": add_btn})

stark组件开发之添加按钮显示和URL的更多相关文章

  1. stark组件开发之添加功能实现

    添加功能,还是使用, form 组件来完成!  并且 完成添加之后,需要保留原搜索条件. def memory_url(self): '''用于反向生成url, 并且携带,get请求的参数,跳转到下一 ...

  2. stark组件开发之列表页面定制列

    先看一张页面展示的效果图: 看一看我的  model 表!是什么样子: 看一看数据库是什么样子: 看 页面展示图,有表头. 有数据.模型表中,每一个字段, 都指定了 verbose_name. 如何解 ...

  3. 7 stark组件介绍、配置、2层url

    1.django的admin配置 model.py from django.db import models # Create your models here. class UserInfo(mod ...

  4. 12 stark组件之pop,按钮,url,页面

    1.Window open() 方法 http://www.runoob.com/jsref/met-win-open.html 效果图   2.admin的pop添加按钮 3.stark之pop功能 ...

  5. stark组件开发之批量操作

    class UserInfoHandler(StartHandler): ....... # 批量操作功能的列表,添加则显示, 使用此功能.需要将StartHandler.display_checkb ...

  6. stark组件开发之列表页面应用示例

    已经解决的,自定义的扩展函数,功能.但是 不可能返回. 一个 固定的页面把!  应该是,点击那条 记录之后的编辑, 就会跳转到相应的,编辑页面.所以 这个标签的  <a href="/ ...

  7. stark组件(1):动态生成URL

    项目启动时自动生成URL 效果图: 知识点: Django启动前通过apps下的ready方法执行一个可以生成URL的py文件 include函数主要返回有三个元素的一个元组.第一个是url配置(ur ...

  8. stark组件开发之组合搜索高级显示和扩展

    上一篇,我只是做了. 默认的显示. def __iter__(self): '''默认显示. 用户可以自定制''' if isinstance(self.queryset_or_tuple, list ...

  9. stark组件开发之列表页面预留钩子方法。 可根据用户的不同,显示不同的列

    要实现,这个方法.子类中 list_diplay 这个列表, 就不能够写死.他应该是 可以根据.用户的不同,返回不同的值. 所以 就需要一个函数, 可以进行判断当前用户是谁. 并且往这个列表中添加,他 ...

随机推荐

  1. Mysql环境搭建(及中文乱码解决)

    卸载MySQL 电脑已经安装过mysql的 卸载电脑上的mysql方法: 我的电脑-->右键-->属性-->高级系统设置-->环境变量-->系统变量里面-->找到环 ...

  2. c 链表和动态内存分配

    兜兜转转又用到了c.c的一些基本却忘记的差不多了(笑哭)!! 动态内存分配 当malloc完将返回的指针类型强制转换成想要的类型后,指针中存有该指针的数据结构,而分配的内存恰好可用于该数据结构. 链表 ...

  3. 关于缓冲的认识---Frame Buffer

    关于缓冲的认识---Frame Buffer 重点来了:

  4. SpringSecurity-FilterSecurityInterceptor的作用

    FilterSecurityInterceptor也是很重要的一个interceptor,它的作用是对request进行权限判断,允许访问或者抛出accessDenied异常. 这个类继承Abstra ...

  5. NIOS II With uCOSII

    1.如果使用uCOS,那么Qsys中Nios II核就不能使用外部中断控制器(EIC). 2.遇到很迷惑的问题,运行uCOSII的实例代码,总是在第二个OSTimeDlyHMSM(0, 0, 3, 0 ...

  6. 在consul上注册web服务

    1. 创建web服务 IDEA->File->New->Project->Empty Project,project name取名provider,点击finish 2. 创建 ...

  7. Python全栈开发记录_第二篇(文件操作及三级菜单栏增删改查)

    python3文件读写操作(本篇代码大约100行) f = open(xxx.txt, "r", encoding="utf-8") 不写“r”(只读)默认是只 ...

  8. BZOJ 4584 luogu P3643: [Apio2016]赛艇

    4584: [Apio2016]赛艇 Time Limit: 70 Sec  Memory Limit: 256 MB[Submit][Status][Discuss] Description 在首尔 ...

  9. JRebel 代理激活

    1.生成GUID   https://www.guidgen.com/ 例:04cfff79-8f45-481c-a858-a5b9590422e7 2.License Server 例: http: ...

  10. Navicat for MySQL使用

    Navicat for MySQL使用 导出数据库表与结构 新数据库导入