过滤项

在配置类中定义,title字段没什么意义,这次仅作演示用

list_filter = ["title", "publish", "authors"]调用和取数据

调用:

filter_condition = self.get_filter_condition(request)

取出数据:
连续两个filter之间是且的关系,等于取出符合这两个条件的数据

data_list = self.model.objects.all().filter(search_condition).filter(filter_condition)处理逻辑

def get_filter_linktag(self): link_dic = {} #--------------------数据处理逻辑------------------------- #循环过滤项列表 for filter_field in self.config.list_filter: #["title", "publish", "authors"] #获取request.GET中相应是哪个字段 #举例:http://127.0.0.1:8000/stark/app01/book/?publish=2 #当循环到publish时获取这个2 cid = self.request.GET.get(filter_field, 0) #根据字符串名称拿字段对象,self.config是当前这张表的配置类实例对象 filter_field_obj = self.config.model._meta.get_field(filter_field) #print("filter_field_obj==>", filter_field_obj) #得到app01.Book.publish和app01.Book.authors #拿对象下面的数据 #.rel .to 来自于ForeignKey和ManyToManyField这两个类 #.rel可以看到对象的数据类型,加上.to可以拿到对象下面的数据,.to只对一对一,一对多,多对多有效 #print(filter_field_obj.rel.to) #结果: #< class 'app01.models.Publish'> #< class 'app01.models.Author'> #判断当前字段对象是不是一对多或多对多类型(需要提前引入这两个类),两种情况取得值也有所不懂 if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): #如果是的话就取这个字段关联的内容 data_list = filter_field_obj.rel.to.objects.all() #[publish2, publish3....] # print("data_list==>", data_list) # < QuerySet[ < Publish: 图灵新知 >, < Publish: dzm >, < Publish: 老男孩出版社 >] > # < QuerySet[ < Author: egon >, < Author: alex >] > else:#如果不是的话直接取字段对象的内容,这里只取主键和当前filter_field的值,比如当前正好是"title" data_list = self.config.model.objects.all().values("pk", filter_field) #print("data_list==>", data_list) #data_list==> <QuerySet [{'pk': 7, 'title': 'go语言第三版'}, {'pk': 8, 'title': 'linux'} #----------------页面渲染--------------- temp = [] #拷贝一份request.GET 用于修改 params = copy.deepcopy(self.request.GET) #处理ALL标签 #如果params(request.GET)中有值 if params.get(filter_field): #删除字典中原有的filter_field(例如是"publish"),但保留其它字段的内容 del params[filter_field] #添加html到列表,此处需要把params序列化 temp.append(mark_safe("<a href='?%s'>ALL</a>" %params.urlencode())) else: #如果没有值 temp.append(mark_safe("<a href=''>ALL</a>")) #处理数据标签 #将data_list循环起来 < QuerySet[ < Publish: 图灵新知 >, < Publish: dzm >...] > for obj in data_list: #判断当前字段对象是不是一对多或多对多 if isinstance(filter_field_obj, ForeignKey) or isinstance(filter_field_obj, ManyToManyField): #此时的obj是一个个对象,取主键值和文本内容 pk = obj.pk text = str(obj) #图灵新知 params[filter_field] = pk #结果<QueryDict: {'publish': [2]}> else: #如果不是 #此时的obj是QuerySet{'pk': 10, 'title': 'PHP'} pk = obj.get("pk") text = obj.get(filter_field) params[filter_field] = text #params里面是QueryDict(<QueryDict: {'publish': [2]}>),所以要做序列化 _url = params.urlencode() #给选中内容加高亮 if cid == str(pk) or cid == text: link_tag = "<a href='?%s' class='action'>%s</a>" % (_url, text) else: link_tag = "<a href='?%s'>%s</a>" % (_url, text) #添加到temp列表 temp.append(mark_safe(link_tag)) #构建字典用作返回 link_dic[filter_field] = temp #{'publish': ["<a href=''>ALL</a>", "<a href='?title=go%E8%AF%AD%E8%A8%80%E7%AC%AC%E4%B8%89%E7%89%88&publish=2'>图灵新知</a>"... return link_dic