利用Django模版生成树状结构实例代码


Posted in Python onMay 19, 2019

前言

我们经常会有这样的需求,比如评论功能,每个评论都有可能会有自己的子评论,如果在界面只展示成一列的话非常不美观,也不能体现出他们的层级关系。那么我们今天就来看看如何使用Django的模版来生成树状结构,以本站为例,效果如下图所示:

利用Django模版生成树状结构实例代码

那么我们要怎么实现呢?首先先看看评论实体的定义,如下所示:

class Comment(models.Model):
 body = models.TextField('正文', max_length=300)
 author = models.ForeignKey(settings.AUTH_USER_MODEL, verbose_name='作者', on_delete=models.CASCADE)
 article = models.ForeignKey(Article, verbose_name='文章', on_delete=models.CASCADE)
 parent_comment = models.ForeignKey('self', verbose_name="上级评论", blank=True, null=True, on_delete=models.CASCADE)

可以看到,有一个parent_comment字段,关联自己。这样就可以根据这个字段来生成层级关系。 为了方便我们使用,我们自定义了一个叫query的tag,也可以叫修饰器。定义tag的代码如下,tag的定义应该定义在app/templatetags目录下,这里py文件命名为blog_tags.py:

@register.simple_tag
def query(qs, **kwargs):
 """ template tag which allows queryset filtering. Usage:
   {% query books author=author as mybooks %}
   {% for book in mybooks %}
   ...
   {% endfor %}
 """
 return qs.filter(**kwargs)

接下来下面这段代码是树节点的模版代码。

{% load blog_tags %}
{% load comments_tags %}
 <div id="commentlist-container" class="comment-tab" style="display: block;">
    <ol class="commentlist">
     {% query article_comments parent_comment=None as parent_comments %}
     {% for comment_item in parent_comments %}
      {% with 0 as depth %}
       {% include "comments/tags/comment_item_tree.html" %}
      {% endwith %}
     {% endfor %}
    </ol>
   </div>

其中的{% query article_comments parent_comment=None as parent_comments %}的功能就是从评论中筛选出来是父级的评论。 comment_item_tree.html的实现也很简单:

{% load blog_tags %}
<li class="comment even thread-even depth-{{ depth }} parent" id="comment-{{ comment_item.pk }}"
 style="margin-left: {% widthratio depth 1 3 %}rem">
 <div id="div-comment-{{ comment_item.pk }}" class="comment-body">
  <div class="comment-meta commentmetadata">
   {{ comment_item.created_time }}
  </div>
  <p>{{ comment_item.body |escape|custom_markdown }}</p>
  <div class="reply"><a class="comment-reply-link"
        href="javascript:void(0)" rel="external nofollow" 
        onclick="do_reply({{ comment_item.pk }})"
        aria-label="回复给{{ comment_item.author.username }}">回复</a></div>
 </div>

</li><!-- #comment-## -->
{% query article_comments parent_comment=comment_item as cc_comments %}
{% for cc in cc_comments %}
 {% with comment_item=cc template_name="comments/tags/comment_item_tree.html" %}
  {% with depth=depth|add:1 %}
   {% include template_name %}
  {% endwith %}
 {% endwith %}
{% endfor %}

其中最主要的部分就是</li>标签后面那段。这里使用with和include配合来在每一次循环里面重复的引入comment_item_tree.html,并且每次引入时赋予当前的评论变量和depth(每层循环depth会+1)。然后在每个评论处使用style="margin-left: {% widthratio depth 1 3 %}rem"来实现缩进,这样就实现了树状显示。

总结

以上就是这篇文章的全部内容了,希望本文的内容对大家的学习或者工作具有一定的参考学习价值,谢谢大家对三水点靠木的支持。

Python 相关文章推荐
使用python实现递归版汉诺塔示例(汉诺塔递归算法)
Apr 08 Python
python中合并两个文本文件并按照姓名首字母排序的例子
Apr 25 Python
python机器学习理论与实战(一)K近邻法
Jan 28 Python
Python针对给定字符串求解所有子序列是否为回文序列的方法
Apr 21 Python
python使用udp实现聊天器功能
Dec 10 Python
python 计算平均平方误差(MSE)的实例
Jun 29 Python
Python 使用list和tuple+条件判断详解
Jul 30 Python
python SocketServer源码深入解读
Sep 17 Python
python 变量初始化空列表的例子
Nov 28 Python
python 常见的排序算法实现汇总
Aug 21 Python
一篇文章带你搞定Ubuntu中打开Pycharm总是卡顿崩溃
Nov 02 Python
使用Pytorch实现two-head(多输出)模型的操作
May 28 Python
使用Python3内置文档高效学习以及官方中文文档
May 19 #Python
python反编译学习之字节码详解
May 19 #Python
python从入门到精通 windows安装python图文教程
May 18 #Python
详解用Python实现自动化监控远程服务器
May 18 #Python
Python实现打砖块小游戏代码实例
May 18 #Python
如何在Python中实现goto语句的方法
May 18 #Python
OpenCV搞定腾讯滑块验证码的实现代码
May 18 #Python
You might like
收藏的一个php小偷的核心程序
2007/04/09 PHP
php mysql Errcode: 28 终极解决方法
2009/07/01 PHP
PHP小技巧之JS和CSS优化工具Minify的使用方法
2014/05/19 PHP
基于php的CMS中展示文章类实例分析
2015/06/18 PHP
PHP面向对象程序设计之对象生成方法详解
2016/12/02 PHP
yii2利用自带UploadedFile实现上传图片的示例
2017/02/16 PHP
PHP自动生成缩略图函数的源码示例
2019/03/18 PHP
使用FlexiGrid实现Extjs表格效果方法分享
2014/12/16 Javascript
JavaScript学习笔记之Cookie对象
2015/01/22 Javascript
使用jQuery的toggle()方法对HTML标签进行显示、隐藏的方法(示例)
2016/09/01 Javascript
jQuery基于xml格式数据实现模糊查询及分页功能的方法
2016/12/25 Javascript
js实现带缓动动画的导航栏效果
2017/01/16 Javascript
js实现二级导航功能
2017/03/03 Javascript
knockoutjs模板实现树形结构列表
2017/07/31 Javascript
javascript 中事件冒泡和事件捕获机制的详解
2017/09/01 Javascript
js+html5生成自动排列对话框实例
2017/10/09 Javascript
关于js的三种使用方式(行内js、内部js、外部js)的程序代码
2018/05/05 Javascript
vue自动化表单实例分析
2018/05/06 Javascript
vue填坑之webpack run build 静态资源找不到的解决方法
2018/09/03 Javascript
vue数据操作之点击事件实现num加减功能示例
2019/01/19 Javascript
JS实现的字符串数组去重功能小结
2019/06/17 Javascript
在vue项目实现一个ctrl+f的搜索功能
2020/02/28 Javascript
[48:54]VGJ.T vs infamous Supermajor小组赛D组败者组第一轮 BO3 第二场 6.3
2018/06/04 DOTA
Python3 能振兴 Python的原因分析
2014/11/28 Python
Python简单定义与使用字典dict的方法示例
2017/07/25 Python
python异常处理之try finally不报错的原因
2020/05/18 Python
python报错: 'list' object has no attribute 'shape'的解决
2020/07/15 Python
CSS3实现歌词进度文字颜色填充变化动态效果的思路详解
2020/06/02 HTML / CSS
澳大利亚玩具剧场:Toy Playhouse
2019/03/03 全球购物
师范生实习个人的自我评价
2013/09/28 职场文书
银行介绍信范文
2014/01/10 职场文书
硕士生工作推荐信
2014/03/07 职场文书
环保项目建议书
2014/08/26 职场文书
会计实训报告范文
2014/11/04 职场文书
失职检讨书大全
2015/01/26 职场文书
自荐信大全
2019/03/21 职场文书