如何实现vue的tree组件


Posted in Vue.js onDecember 03, 2020

前言

Tree一直是大家熟知的组件,做一些大型的后台管理系统都会用到。使用树组件可以完整的展现其中的层级关系,并具有展开收起选择等交互功能。

效果

如何实现vue的tree组件

节点可以无限的递归延伸
可以展开和收起子节点
如果子节点全部选择对应的父节点也应该选中,反之父节点取消选中对应子节点也需要取消选中

API

prop传递data属性,来描述所有的节点的信息

如何实现vue的tree组件

每个节点的配置描述如下

  • title: 展示的标题
  • expand 是否展开节点
  • checked 是否选中节点
  • children 子节点

以及还有两个event

  • on-toggle-expand 展开和收起子列表时触发的
  • on-check-change 点击checkbox触发

我们来 init tree主组件

首先需要考虑一个问题因为tree是递归遍历的,因为我们需要创建一个入口组件还有一个递归children的组件。

首先创建我们的tree组件

如何实现vue的tree组件

如何实现vue的tree组件

我们在初始化以及watch监听的时候重新深拷贝了一下prop传来的data并赋值给了cloneData

然后在template里来引入node.vue,然后循环cloneData来循环node.vue。node.vue接受两个prop

showCheckbox 就是tree组件接收的 showCheckbox 这里传给node组件来判断展示
data 为item 是一个object 负责渲染当前节点,如果当前节点有children 那就递归调用本身来递归渲染
这是使用了一个deepCopy的方法,这个是深拷贝的简单实现,递归的去重新重新赋值data数组,开辟新的堆内存与传入的数据无关联。不会破坏原有的数据

我们来 init node递归组件

node组件为主要组件,主要功能是展示当前项的title 以及 如果有children时递归本身。

  • 展开关闭按钮
  • checkbox
  • 节点的title
  • 递归

node的基本构造

如何实现vue的tree组件

prop中的data就是当前节点的所有信息,比如说是否展开和关闭当前的节点,是否选中,title标题以及children的子节点数组。

  • expand 判断条件为 data.children &&  data.children.length 才会展示 + 或者 - 按钮
  • checkbox就是当前的节点是否需要默认勾选

点击 + 号时会展开当前的子节点,点击 - 号会关闭,这一步只需要在handleExpand 中修改data的expand数据即可同时我们还需要触发一个emit来提示用户展示或者收起了节点

如何实现vue的tree组件

如何实现vue的tree组件

这里有一点需要注意 修改data.expand我们通过 VUE的 $set 并没有像下面这样

this.data.expand = !this.data.expand;

这里有什么区别呢?如果直接用上面的代码进行修改,就会发现数据虽然被修改了,但是视图没有被更新,这是因为这里的this.data 时props通过上一级传递出来的,也有可能时node递归传递的,无论如何咱们需要的cloneData里的节点数据,此时不一定初始化定义时就含有expand或者checked字段 如果不含有直接通过this.data.expand修改,这个expand时不可响应式的数据,所以视图不会被更新,干脆就直接用$set来改变

接下来我们就需要处理响应状态了,大家可能觉得不就是选中和取消吗 的确这样可以,但是树组件时有上下级关系,他们分为两种逻辑,当选中(或取消选中)一个节点时

  • 它下面的所有子节点都会被选中
  • 如果同一级所有子节点选中时,它的父级也自动选中,一直递归判断到根节点。

第 1 个逻辑相对简单,当选中一个节点时,只要递归地遍历它下面所属的所有子节点数据,修改所有的 checked 字段即可

如何实现vue的tree组件

再来看第2个逻辑 一个节点,除了手动选中(或反选),还有就是第 2 种逻辑的被动选中(或反选),也就是说,如果这个节点的所有直属子节点(就是它的第一级子节点)都选中(或反选)时,这个节点就自动被选中(或反选),递归地,可以一级一级响应上去。有了这个思路,我们就可以通过 watch 来监听当前节点的子节点是否都选中,进而修改当前的 checked 字段:

如何实现vue的tree组件

在 watch 中,监听了 data.children 的改变,并且是深度监听的。这段代码的意思是,当 data.children 中的数据的某个字段发生变化时(这里当然是指 checked 字段),也就是说它的某个子节点被选中(或反选)了,这时执行绑定的句柄 handler 中的逻辑。const checkedAll = !data.some(item => !item.checked); 也是一个巧妙的缩写,checkedAll 最终返回结果就是当前子节点是否都被选中了。

这里非常巧妙地利用了递归的特性,因为 node.vue 是一个递归组件,那每一个组件里都会有 watch 监听 data.children,要知道,当前的节点有两个”身份“,它既是下属节点的父节点,同时也是上级节点的子节点,它作为下属节点的父节点被修改的同时,也会触发上级节点中的 watch 监听函数。这就是递归。

结语

递归的可以把一个大问题通过不断调用自身的方式,使代码简洁的实现功能,但是个别问题像算法中斐波那契数列如果使用递归就会使得时间复杂度以及空间复杂度会飙升。总的来说要合理运用,活学活用。

以上就是如何实现vue的tree组件的详细内容,更多关于vue tree组件的资料请关注三水点靠木其它相关文章!

Vue.js 相关文章推荐
vue 表单输入框不支持focus及blur事件的解决方案
Nov 17 Vue.js
浅谈Vue使用Elementui修改默认的最快方法
Dec 05 Vue.js
vue实现购物车的小练习
Dec 21 Vue.js
基于vue+echarts数据可视化大屏展示的实现
Dec 25 Vue.js
Vue页面渲染中key的应用实例教程
Jan 12 Vue.js
vue 实现click同时传入事件对象和自定义参数
Jan 29 Vue.js
vue中h5端打开app(判断是安卓还是苹果)
Feb 26 Vue.js
Vue-router编程式导航的两种实现代码
Mar 04 Vue.js
vue中三级导航的菜单权限控制
Mar 31 Vue.js
Vue Element UI自定义描述列表组件
May 18 Vue.js
VUE中的v-if与v-show区别介绍
Mar 13 Vue.js
Vue组件更新数据v-model不生效的解决
Apr 02 Vue.js
Vue实现图书管理小案例
Dec 03 #Vue.js
Vue router安装及使用方法解析
Dec 02 #Vue.js
vue3.0中setup使用(两种用法)
Dec 02 #Vue.js
vue3.0+vue-router+element-plus初实践
Dec 02 #Vue.js
Vue router传递参数并解决刷新页面参数丢失问题
Dec 02 #Vue.js
详解Vue3 Teleport 的实践及原理
Dec 02 #Vue.js
vue $router和$route的区别详解
Dec 02 #Vue.js
You might like
php设计模式 Singleton(单例模式)
2011/06/26 PHP
PHP输出数组中重名的元素的几种处理方法
2012/09/05 PHP
深入for,while,foreach遍历时间比较的详解
2013/06/08 PHP
如何用C语言编写PHP扩展的详解
2013/06/13 PHP
ThinkPHP后台首页index使用frameset时的注意事项分析
2014/08/22 PHP
php生成高清缩略图实例详解
2015/12/07 PHP
Zend Framework实现留言本分页功能(附demo源码下载)
2016/03/22 PHP
php生成图片缩略图功能示例
2017/02/22 PHP
visual studio code 调试php方法(图文详解)
2017/09/15 PHP
PHP使用 Imagick 扩展实现图片合成,圆角处理功能示例
2019/09/09 PHP
js单词形式的运算符
2014/05/06 Javascript
Javascript毫秒数用法实例
2015/02/05 Javascript
Bootstrap富文本组件wysiwyg数据保存到mysql的方法
2016/05/09 Javascript
jQuery实现简单倒计时功能的方法
2016/07/04 Javascript
js实现随机抽选效果、随机抽选红色球效果
2017/01/13 Javascript
提高Node.js性能的应用技巧分享
2017/08/10 Javascript
jfinal与bootstrap的登出实战详解
2017/11/27 Javascript
vue项目强制清除页面缓存的例子
2019/11/06 Javascript
vue 图片裁剪上传组件的实现
2020/11/12 Javascript
使用Python的Tornado框架实现一个简单的WebQQ机器人
2015/04/24 Python
对Python 窗体(tkinter)文本编辑器(Text)详解
2018/10/11 Python
Flask-WTF表单的使用方法
2019/07/12 Python
给Python学习者的文件读写指南(含基础与进阶)
2020/01/29 Python
使用Tkinter制作信息提示框
2020/02/18 Python
浅谈pytorch torch.backends.cudnn设置作用
2020/02/20 Python
pytorch查看模型weight与grad方式
2020/06/24 Python
利用Python实现斐波那契数列的方法实例
2020/07/26 Python
PyTorch中的拷贝与就地操作详解
2020/12/09 Python
美国运动鞋类和服装零售连锁店:Shoe Palace
2019/08/13 全球购物
有机婴儿毛毯和衣服:Monica + Andy
2020/03/01 全球购物
一套软件测试笔试题
2014/07/25 面试题
食品科学与工程专业毕业生求职信范文
2014/07/21 职场文书
王金山在党的群众路线教育实践活动总结大会上的讲话稿
2014/10/25 职场文书
兴趣班停课通知
2015/04/24 职场文书
配置nginx负载均衡
2022/05/06 Servers
java实现面板之间切换功能
2022/06/10 Java/Android