stark——快速过滤list_filter
一、获取过滤字段
1、给自定义配置类配置list_filter
app01/stark.py:
class BookConfig(ModelStark): list_display = ["title", "price", "publishDate"] modelform_class = BookModelForm search_fields = ['title', "price"] def patch_init(self, request, queryset): print(queryset) queryset.update(price=123) patch_init.short_description = "批量初始化" actions = [patch_init] list_filter = ["publish", "authors", ] # 一对多、多对多 site.register(Book, BookConfig)
2、构建实例方法获取过滤字段
class ShowList(object): """展示页面类""" def __init__(self, config, data_list, request):... def get_filter_linktags(self): """获取过滤字段""" link_list = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: print(filter_field) # 'publish' 'authors' # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) print(filter_field_obj) # app01.Book.publish app01.Book.authors print(type(filter_field_obj)) """ <class 'django.db.models.fields.related.ForeignKey'> <class 'django.db.models.fields.related.ManyToManyField'> from django.db.models.fields.related import ForeignKey from django.db.models.fields.related import ManyToManyField """ # 拿到关联表下的所有数据 # print("rel...", filter_field_obj.rel.to.objects.all()) # 版本问题失效 print("rel...", filter_field_obj.related_model.objects.all()) """ rel... <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> rel... <QuerySet [<Author: alex>, <Author: egon>]> """ return link_list class ModelStark(object): """默认类,定制配置类""" list_display = ["__str__",] list_display_links = [] modelform_class = [] search_fields = [] actions = [] # 调用self.actions拿到的是函数 list_filter = []
注意:
(1)获取自定义配置类定义的list_filter列表
ShowList类对象,通过self.config.list_filter可以拿到当前访问页面对象自定义配置类配置的list_filter列表。
(2)根据字段字符串获取模型字段对象
filter_field_obj = self.config.model._meta.get_field(filter_field) model_name = self.config.model._meta.model_name # 模型名 book app_label = self.config.model._meta.app_label # app名 app01
(3)根据一对多,多对多对象关联关系,得到关联模型表和数据
# 拿到关联表下的所有数据 # print("rel...", filter_field_obj.rel.to.objects.all()) # 版本问题失效(filter_field_obj.rel.to是关联模型表) print("rel...", filter_field_obj.related_model.objects.all()) # 拿到对象下的关联数据 """ rel... <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> rel... <QuerySet [<Author: alex>, <Author: egon>]> """
二、根据拿到的对象关联数据完成数据组织
1、get_filter_linktags方法组织返回链接字典
class ShowList(object): """展示页面类""" def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: print(filter_field) # 'publish' 'authors' # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) print(filter_field_obj) # app01.Book.publish app01.Book.authors # 拿到关联表下的所有数据 # print("rel...", filter_field_obj.rel.to.objects.all()) # 版本问题失效 # print("rel...", filter_field_obj.related_model.objects.all()) # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> temp = [] for obj in data_list: # obj是每一个对象 # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href=>%s</a>" % str(obj) # print(link_tag) # <a href=>苹果出版社</a> temp.append(link_tag) link_dic[filter_field] = temp # print(link_dic) # {'publish': ['<a href=>苹果出版社</a>', '<a href=>香蕉出版社</a>'], 'authors': ['<a href=>alex</a>', '<a href=>egon</a>']} return link_dic
这里最重要就是理清楚每个变量的类型和含义:
self.config.list_filter——['publish', 'authors'] filter_field_obj—— app01.Book.publish、app01.Book.authors filter_field_obj.related_model.objects.all()——<QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]>、<QuerySet [<Author: alex>, <Author: egon>]> obj——苹果出版社 香蕉出版社 alex egon 数据类型:<class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag——<a href=>苹果出版社</a> link_dic——{'publish': ['<a href=>苹果出版社</a>', '<a href=>香蕉出版社</a>'], 'authors': ['<a href=>alex</a>', '<a href=>egon</a>']}
2、list_view.html构建显示
<h4>数据列表</h4> <div class="container"> <div class="row"> <div class="col-md-9".....> <div class="col-md-3"> <div class="filter"> <h4>Filter</h4> {% for filter_field, linktags in show_list.get_filter_linktags.items %} <div> <p>{{ filter_field }}</p> {% for link in linktags %} <p>{{ link|safe }}</p> {% endfor %} </div> {% endfor %} </div> </div> </div> </div>
注意这里使用{{link|safe}}来实现取消转义。显示效果如下:
三、标签href处理
class ShowList(object): """展示页面类""" def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: """循环每一个过滤字段""" import copy # self.request.GET # GET请求的所有数据 params = copy.deepcopy(self.request.GET) print(filter_field) # 'publish' 'authors' # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) print(filter_field_obj) # app01.Book.publish app01.Book.authors # 拿到关联表下的所有数据 # print("rel...", filter_field_obj.rel.to.objects.all()) # 版本问题失效 # print("rel...", filter_field_obj.related_model.objects.all()) # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> temp = [] for obj in data_list: # obj是每一个对象 """循环每一个过滤字段关联的数据""" # 构成一个新字典 过滤字段:当前对象主键值 params[filter_field + "__id"] = obj.pk # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href='?%s'>%s</a>" % (_url, str(obj)) # print(link_tag) # <a href=>苹果出版社</a> temp.append(link_tag) link_dic[filter_field] = temp # print(link_dic) # {'publish': ['<a href=>苹果出版社</a>', '<a href=>香蕉出版社</a>'], 'authors': ['<a href=>alex</a>', '<a href=>egon</a>']} return link_dic
注意:
1、copy.deepcopy()使用
-
直接赋值:其实就是对象的引用(别名)。
-
浅拷贝(copy):拷贝父对象,不会拷贝对象的内部的子对象。
-
深拷贝(deepcopy): copy 模块的 deepcopy 方法,完全拷贝了父对象及其子对象。
这里每循环一次过滤字段都会重新创建一个params。保证按钮对应路径的唯一性。
2、利用urlencode将键值对转化为a=1&b=2的格式
这里主要是应为发送的是get请求,请求数据必须是a=1&b=2的格式。
3、params = copy.deepcopy(self.request.GET)意义
self.request.GET获取的是GET请求的所有数据,多次点击可以实现get请求数据的拼接。打印params,在页面点击访问,控制台输出如下:
params <QueryDict: {}> ——没有点击a标签 params <QueryDict: {'authors__id': ['1']}> ——第一次点击 params <QueryDict: {'authors__id': ['1'], 'publish__id': ['2']}> ——第二次点击
四、a标签点击后颜色变化
1、前置准备
(1)将过滤字段显示为大写
{% for filter_field, linktags in show_list.get_filter_linktags.items %} <div class="well"> {# upper方法改为大写 #} <p>{{ filter_field.upper }}</p> {% for link in linktags %} <p>{{ link|safe }}</p> {% endfor %} </div> {% endfor %}
(2)取消a标签颜色
<style> .filter a { text-decoration: none; /* 取消a标签颜色 */ color: grey; } </style>
2、对当前get请求数据进行判断
class ShowList(object): def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: """循环每一个过滤字段""" import copy # self.request.GET # GET请求的所有数据 params = copy.deepcopy(self.request.GET) print("params", params) # cid是当前字段传过来的值 cid = self.request.GET.get(filter_field + "__id", 0) # 没有值的时候默认为None,None是不能进行int()转换的,因此在这里给它设置默认值为0 # print(filter_field) # 'publish' 'authors' # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> temp = [] for obj in data_list: # obj是每一个对象 """循环每一个过滤字段关联的数据""" # 构成一个新字典 过滤字段:当前对象主键值 params[filter_field + "__id"] = obj.pk # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() if int(cid) == obj.pk: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, str(obj)) else: link_tag = "<a href='?%s'>%s</a>" % (_url, str(obj)) temp.append(link_tag) link_dic[filter_field] = temp return link_dic
给模板添加样式:
<style> .filter a { text-decoration: none; /* 取消a标签颜色 */ color: grey; } .active { color: red!important; /* 提升优先级 */ } </style>
注意:
(1)cid是当前get请求传递的值
cid = self.request.GET.get(filter_field + "__id", 0)
需要注意的是在get请求没有值的时候,默认值是None,但是None是不能进行int()转换的,因此在这里给它设置默认值0.
(2)根据get请求的值和对象主键比对,给a标签添加avtice类
if int(cid) == obj.pk: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, str(obj)) else: link_tag = "<a href='?%s'>%s</a>" % (_url, str(obj))
(3)显示效果
五、过滤器添加all按钮取消过滤
class ShowList(object): def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: """循环每一个过滤字段""" import copy # self.request.GET # GET请求的所有数据 params = copy.deepcopy(self.request.GET) print("params", params) # <QueryDict: {'publish__id': ['1']}> # cid是当前字段传过来的值 cid = self.request.GET.get(filter_field + "__id", 0) # 没有值的时候默认为None,None是不能进行int()转换的,因此在这里给它设置默认值为0 # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> temp = [] # 处理all标签 if params.get(filter_field + "__id"): del params[filter_field + "__id"] temp.append("<a href='?%s'>all</a>" % params.urlencode()) else: temp.append("<a class='active' href='#'>all</a>") # 默认是all的状态 # 处理数据标签 for obj in data_list: # obj是每一个对象 """循环每一个过滤字段关联的数据""" # 构成一个新字典 过滤字段:当前对象主键值 params[filter_field + "__id"] = obj.pk # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() if int(cid) == obj.pk: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, str(obj)) else: link_tag = "<a href='?%s'>%s</a>" % (_url, str(obj)) temp.append(link_tag) link_dic[filter_field] = temp return link_dic
注意:
1、在处理数据标签前,在temp临时数组中添加all的<a>标签
# 处理all标签 if params.get(filter_field + "__id"): del params[filter_field + "__id"] temp.append("<a href='?%s'>all</a>" % params.urlencode()) else: temp.append("<a class='active' href='#'>all</a>") # 默认是all的状态
点击a标签由于href没有在?前填任何值,默认是将get请求发送给原函数处理。
params是深度复制了get请求的数据,因此每次点击a标签都在添加params的值:
params <QueryDict: {}> ——没有点击a标签 params <QueryDict: {'authors__id': ['1']}> ——第一次点击 params <QueryDict: {'authors__id': ['1'], 'publish__id': ['2']}> ——第二次点击
params.get(filter_field + "__id") 就可以拿到对应的authors__id和publish__id.如果if判断拿不到值,说明还没有进行过滤,添加带有active类的a标签:<a class='active' href='#'>all</a>。all标签显示为激活状态。
如果if判断有值,通过del方法清除对应的params中的值,添加不带有active类的a标签:
temp.append("<a href='?%s'>all</a>" % params.urlencode())
2、params.urlencode解析
# 访问http://127.0.0.1:8000/stark/app01/book/?authors__id=1&publish__id=1 print("urlencode", params.urlencode) print("_url", params.urlencode()) print("params", params) """ urlencode <bound method QueryDict.urlencode of <QueryDict: {'authors__id': ['1'], 'publish__id': ['1']}>> _url authors__id=1&publish__id=1 params <QueryDict: {'authors__id': ['1'], 'publish__id': ['1']}> """
3、删改params不是修改get请求数据,而是修改a标签href值(本质)
通过点击按钮修改href值,修改每次发送的get请求数据。
此时查看PUBLISH下的all按钮:
此时查看PUBLISH下的香蕉出版社:
六、过滤实现
1、删除filter_field后面拼接的"__id"
class ShowList(object): """展示页面类""" def __init__(self, config, data_list, request): self.config = config # 接收传递过来的配置类对象 ModelStark的实例对象 self.data_list = data_list # 接收传递过来的当前表的所有对象 self.request = request # <WSGIRequest: GET '/stark/app01/book/?page=2'> # 分页 data_count = self.data_list.count() current_page = int(self.request.GET.get("page", 1)) # 默认是第一页 base_path = self.request.path # /stark/app01/book/ self.pagination = Pagination(current_page, data_count, base_path, self.request.GET, per_page_num=3, pager_count=11,) # print("data_list", self.data_list) # data_list <QuerySet [<Book: python葵花宝典>, <Book: go>, <Book: java>]> self.page_data = self.data_list[self.pagination.start:self.pagination.end] # print("page_data", self.page_data) # page_data <QuerySet [<Book: python葵花宝典>]> # actions # self.actions = self.config.actions # 拿到配置好的函数对象列表 [patch_init,] self.actions = self.config.new_actions() # 拿到方法运行的返回结果 def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: """循环每一个过滤字段""" import copy # self.request.GET # GET请求的所有数据 params = copy.deepcopy(self.request.GET) print("params", params) # <QueryDict: {'publish__id': ['1']}> # cid是当前字段传过来的值 cid = self.request.GET.get(filter_field, 0) # 没有值的时候默认为None,None是不能进行int()转换的,因此在这里给它设置默认值为0 # print(filter_field) # 'publish' 'authors' # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) # print(filter_field_obj) # app01.Book.publish app01.Book.authors # 拿到关联表下的所有数据 # print("rel...", filter_field_obj.rel.to.objects.all()) # 版本问题失效 # print("rel...", filter_field_obj.related_model.objects.all()) # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> temp = [] # 处理all标签 if params.get(filter_field): print("_url", params.urlencode) del params[filter_field] temp.append("<a href='?%s'>all</a>" % params.urlencode()) else: temp.append("<a class='active' href='#'>all</a>") # 默认是all的状态 # 处理数据标签 for obj in data_list: # obj是每一个对象 """循环每一个过滤字段关联的数据""" # 构成一个新字典 过滤字段:当前对象主键值 params[filter_field] = obj.pk # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() if int(cid) == obj.pk: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, str(obj)) else: # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href='?%s'>%s</a>" % (_url, str(obj)) # print(link_tag) # <a href=>苹果出版社</a> temp.append(link_tag) link_dic[filter_field] = temp # print(link_dic) # {'publish': ['<a href=>苹果出版社</a>', '<a href=>香蕉出版社</a>'], 'authors': ['<a href=>alex</a>', '<a href=>egon</a>']} return link_dic
2、构建filter的Q对象(过滤条件)
class ModelStark(object): def get_filter_condition(self, request): """拿到过滤条件""" filter_condition = Q() # 默认查询条件为且 and for filter_field, val in request.GET.items(): # 过滤字段、查询的值 去除fitler_field拼接的__id if filter_field in self.list_filter: # 只处理filter过滤列表的键值(分页等排除) filter_condition.children.append((filter_field, val)) return filter_condition
注意get_filter_condition只处理filter过滤列表键值,需要将分页等请求数据排除。
3、在list_view方法中获取filter的Q对象完成过滤
class ModelStark(object): def list_view(self, request): if request.method == "POST": # action print("POST:", request.POST) action = request.POST.get("action") selected_pk = request.POST.getlist("selected_pk") # 拿到列表 # 反射 # self这里是配置类BookConfig,要在类中找到对应的函数 action_func = getattr(self, action) # patch_init # 拿到选中状态的pk值对象 queryset = self.model.objects.filter(pk__in=selected_pk) # <QuerySet [<Book: go>]> action_func(request, queryset) # 获取search的Q对象 search_condition = self.get_search_condition(request) # 获取filter构建Q对象 filter_condition = self.get_filter_condition(request) # 筛选当前表获取的数据 data_list = self.model.objects.all().filter(search_condition).filter(filter_condition) # 链式操作,二次过滤 # 获取showlist展示页面 show_list = ShowList(self, data_list, request) header_list = show_list.get_header() new_data_list = show_list.get_body() # 构建一个查看url add_url = self.get_add_url() print("add_url", add_url) return render(request, "list_view.html", locals())
注意这里是运用了链式操作,二次过滤。过滤效果显示如下:
七、一对多、多对多字段渲染处理
1、添加一对多、多对多字段 到list_display
app01/stark.py:
class BookConfig(ModelStark): list_display = ["title", "price", "publishDate", "publish", "authors"] list_display_links = ["title"] modelform_class = BookModelForm search_fields = ['title', "price"] def patch_init(self, request, queryset): print(queryset) queryset.update(price=123) patch_init.short_description = "批量初始化" actions = [patch_init] list_filter = ["publish", "authors", ] # 一对多、多对多 site.register(Book, BookConfig)
publish是一对多字段、authors是多对多字段。页面显示如下:
可以看到多对多字段无法正常显示,这个因为在service/stark.py中
class ShowList(object): """展示页面类""" def get_body(self): """构建表单数据""" new_data_list = [] # for obj in self.data_list: for obj in self.page_data: # 当前页面的数据 temp = [] for field in self.config.new_list_display(): # ["__str__", ] ["pk","name","age",edit] if callable(field): val = field(self.config, obj) else: val = getattr(obj, field) # 拿到的关联对象 处理不了多对多 if field in self.config.list_display_links: # _url = reverse("%s_%s_change" % (app_label, model_name), args=(obj.pk,)) _url = self.config.get_change_url(obj) val = mark_safe("<a href='%s'>%s</a>" % (_url, val)) temp.append(val) new_data_list.append(temp) return new_data_list
get_body方法,val = getattr(obj, field)拿到的是关联对象,在一对一、一对多情况下,利用模型定了__str__可以正常显示名称,但是却无法处理多对多的情况。
2、多对多字段处理
class ShowList(object): """展示页面类""" def get_body(self): """构建表单数据""" new_data_list = [] # for obj in self.data_list: for obj in self.page_data: # 当前页面的数据 temp = [] for field in self.config.new_list_display(): # ["__str__", ] ["pk","name","age",edit] if callable(field): val = field(self.config, obj) else: from django.db.models.fields.related import ManyToManyField field_obj = self.config.model._meta.get_field(field) # 拿到字段对象 if isinstance(field_obj, ManyToManyField): # 判断是否是多对多 # 反射处理 增加.all # 多对多的情况 obj.field.all() ret = getattr(obj, field).all() # <QuerySet [<Author: alex>, <Author: egon>]> t = [] for obj in ret: t.append(str(obj)) val = ",".join(t) # 用join方法实现拼接 alex,egon else: # 非多对多的情况 val = getattr(obj, field) # 拿到的关联对象 处理不了多对多 if field in self.config.list_display_links: # _url = reverse("%s_%s_change" % (app_label, model_name), args=(obj.pk,)) _url = self.config.get_change_url(obj) val = mark_safe("<a href='%s'>%s</a>" % (_url, val)) temp.append(val) new_data_list.append(temp) return new_data_list
显示效果:
注意:
(1)引入多对多类,利用isinstance判断对象是否是多对多对象
from django.db.models.fields.related import ManyToManyField field_obj = self.config.model._meta.get_field(field) # 拿到字段对象 if isinstance(field_obj, ManyToManyField): # 判断是否是多对多
(2)利用反射处理多对多的情况
# 反射处理 增加.all # 多对多的情况 obj.field.all() ret = getattr(obj, field).all() # <QuerySet [<Author: alex>, <Author: egon>]> t = [] for obj in ret: t.append(str(obj)) val = ",".join(t) # 用join方法实现拼接 alex,egon
(3)注意getattr(obj, field) 和getattr(obj, field).all()的区别
print("ret",getattr(obj, field)) # ret app01.Author.None print("ret", getattr(obj, field).all()) # ret <QuerySet [<Author: alex>, <Author: egon>]>
(4)join()方法
用于将序列中的元素以指定的字符连接生成一个新的字符串。
str = "-" seq = ("a", "b", "c") # 字符串序列 print str.join( seq ) # a-b-c
八、普通字段筛选
1、给list_filter添加普通字段"title"
app01/stark.py:
class BookConfig(ModelStark): list_display = ["title", "price", "publishDate", "publish", "authors"] list_display_links = ["title"] modelform_class = BookModelForm search_fields = ['title', "price"] def patch_init(self, request, queryset): print(queryset) queryset.update(price=123) patch_init.short_description = "批量初始化" actions = [patch_init] list_filter = ["title", "publish", "authors", ] # 普通字段、一对多、多对多 site.register(Book, BookConfig)
添加后访问页面直接报错:
这是由于在ShowList类get_filter_linktags方法中:
# 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) # 关联表下所有数据 data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社>
data_list这种取法只适用于一对一和一对多的情况。
2、处理过滤字段对象
from django.db.models.fields.related import ManyToManyField, ForeignKey class ShowList(object): """展示页面类""" def get_filter_linktags(self): """获取过滤字段""" link_dic = {} print("list_filter", self.config.list_filter) # list_filter ['publish', 'authors'] for filter_field in self.config.list_filter: """循环每一个过滤字段""" import copy params = copy.deepcopy(self.request.GET) cid = self.request.GET.get(filter_field, 0) # 获取字段对象 filter_field_obj = self.config.model._meta.get_field(filter_field) if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): data_list = filter_field_obj.related_model.objects.all() # <QuerySet [<Publish: 苹果出版社> else: # 普通字段直接查询 data_list = self.config.model.objects.all().values("pk", filter_field) # 主键值 字段对象值
引入ForeignKey和ManyToManyField类,利用isinstance判断是否是一对多、多对多对象。如果不是就是普通字段,直接查询处理。
3、处理数据标签对data_list做对应处理
# 处理数据标签 for obj in data_list: # obj是每一个对象 """循环每一个过滤字段关联的数据""" if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> pk = obj.pk text = str(obj) else: # 列表里面套着字典 data_list=[{"pk":1, "title":"go"},....] pk = obj.get("pk") text = obj.get(filter_field) # 构成一个新字典 过滤字段:当前对象主键值 params[filter_field] = pk # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() if int(cid) == pk: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, text) else: # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href='?%s'>%s</a>" % (_url, text)
两种data_list,一种是QuerySet,一种是数组套字典。两种数据类型的处理方式略有不同。
显示效果:
这样做完后点击TITLE下的过滤项是查不到任何对应数据的。这是因为默认传递的过滤字段都是PK值,但是针对普通字段过滤需要传递过滤字段值。
4、处理数据标签时过滤字段按情况拆分
# 处理数据标签 for obj in data_list: # obj是每一个对象(或者是数组) """循环每一个过滤字段关联的数据""" if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> pk = obj.pk text = str(obj) params[filter_field] = pk # 过滤字段:当前对象主键值 else: # 列表里面套着字典 data_list=[{"pk":1, "title":"go"},....] pk = obj.get("pk") text = obj.get(filter_field) params[filter_field] = text # 过滤字段:当前对象字段值 # 利用urlencode将键值对转化为a=1&b=2的格式 _url = params.urlencode() if cid == str(pk) or cid == text: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, text) else: # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href='?%s'>%s</a>" % (_url, text) # print(link_tag) # <a href=>苹果出版社</a> temp.append(link_tag)
注意:
(1)params分拆为两种情况
一开始统一用params[filter_field] = pk 来设置过滤字段,但是设置普通过滤字段后,如果title=7这样是无法进行过滤的,必须让过滤字段等于"go"、"python"等字段值。因此将params也分拆为两种情况:
if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): # <QuerySet [<Publish: 苹果出版社>, <Publish: 香蕉出版社>]> pk = obj.pk text = str(obj) params[filter_field] = pk # 过滤字段:当前对象主键值 else: # 列表里面套着字典 data_list=[{"pk":1, "title":"go"},....] pk = obj.get("pk") text = obj.get(filter_field) params[filter_field] = text # 过滤字段:当前对象字段值
(2)cid(get请求数据)判断的调整
cid = self.request.GET.get(filter_field, 0) 由此可见cid是get请求传递的值,之前默认都是pk值,现在有可能是pk值也可能是"python"等普通字段。因此需要调整cid判断:
if cid == str(pk) or cid == text: # get请求数据int转换后与对象主键值匹配,匹配成功添加active类 link_tag = "<a class='active' href='?%s'>%s</a>" % (_url, text) else: # print(obj) # 苹果出版社 香蕉出版社 alex egon # print(type(obj)) # <class 'app01.models.Publish'> <class 'app01.models.Author'> link_tag = "<a href='?%s'>%s</a>" % (_url, text) # print(link_tag) # <a href=>苹果出版社</a> temp.append(link_tag)
(3)最终展示效果
九、filter_list配置与否,决定是否显示FILTER
list_view.html:
<div class="col-md-3"> {% if showlist.config.list_filter %} {# list_filter有值才显示FILTER #} <div class="filter"> <h4>Filter</h4> {% for filter_field, linktags in show_list.get_filter_linktags.items %} <div class="well"> {# upper方法改为大写 #} <p>{{ filter_field.upper }}</p> {% for link in linktags %} <p>{{ link|safe }}</p> {% endfor %} </div> {% endfor %} </div> {% endif %} </div>