如何实现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+Element实现增删改查的示例源码
Nov 23 Vue.js
vue祖孙组件之间的数据传递案例
Dec 07 Vue.js
vue实现登录功能
Dec 31 Vue.js
vue动态设置路由权限的主要思路
Jan 13 Vue.js
Vue单页面应用中实现Markdown渲染
Feb 14 Vue.js
如何用vue实现网页截图你知道吗
Nov 17 Vue.js
vue+echarts实现多条折线图
Mar 21 Vue.js
Axios代理配置及封装响应拦截处理方式
Apr 07 Vue.js
vue组件vue-esign实现电子签名
Apr 21 Vue.js
Vue组件化(ref,props, mixin,.插件)详解
May 15 Vue.js
Vue ECharts实现机舱座位选择展示功能
May 15 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小教程之实现双向链表
2014/06/12 PHP
PHP实现搜索相似图片
2015/09/22 PHP
php处理json格式数据经典案例总结
2016/05/19 PHP
jQuery图片预加载 等比缩放实现代码
2011/10/04 Javascript
浅析js中2个等号与3个等号的区别
2013/08/06 Javascript
JavaScript中join()方法的使用简介
2015/06/09 Javascript
Angularjs material 实现搜索框功能
2016/03/08 Javascript
JavaScript编写Chrome扩展实现与浏览器的交互及时间通知
2016/05/16 Javascript
jQuery中的each()详细介绍(推荐)
2016/05/25 Javascript
解析浏览器端的AJAX缓存机制
2016/06/21 Javascript
javascript类型系统——日期Date对象全面了解
2016/07/13 Javascript
JS 实现计算器详解及实例代码(一)
2017/01/08 Javascript
JS实现图片放大缩小的方法
2017/02/15 Javascript
Vue表单demo v-model双向绑定问题
2018/06/29 Javascript
JavaScript实现异步图像上传功能
2018/07/12 Javascript
Nodejs中怎么实现函数的串行执行
2019/03/02 NodeJs
vue全局自定义指令-元素拖拽的实现代码
2019/04/14 Javascript
vue使用i18n实现国际化的方法详解
2019/09/05 Javascript
Vue.js实现大转盘抽奖总结及实现思路
2019/10/09 Javascript
react+antd 递归实现树状目录操作
2020/11/02 Javascript
Python实现115网盘自动下载的方法
2014/09/30 Python
python图片二值化提高识别率代码实例
2019/08/24 Python
Python实现数值积分方式
2019/11/20 Python
opencv之颜色过滤只留下图片中的红色区域操作
2020/06/05 Python
利用python实现后端写网页(flask框架)
2021/02/28 Python
canvas粒子动画背景的实现示例
2018/09/03 HTML / CSS
英国最大线上综合鞋类商城:Office
2017/12/08 全球购物
毕业生大学生活自我总结
2014/01/31 职场文书
小班下学期评语
2014/05/04 职场文书
巾帼建功标兵事迹材料
2014/05/11 职场文书
园林专业毕业生自荐信
2014/07/04 职场文书
领导干部对照检查材料
2014/08/24 职场文书
团党委领导干部党的群众路线教育实践活动个人对照检查材料思想汇
2014/10/05 职场文书
工作失职检讨书(精华篇)
2014/10/15 职场文书
2020年基层司法所建设情况调研报告
2019/11/30 职场文书
低版本Druid连接池+MySQL驱动8.0导致线程阻塞、性能受限
2021/07/01 MySQL