element-ui Tree 树形控件 过滤保留子级并获取过滤后的数据 多选改单选

本文主要是介绍element-ui Tree 树形控件 过滤保留子级并获取过滤后的数据 多选改单选,希望对大家解决编程问题提供一定的参考价值,需要的开发者们随着小编来一起学习吧!

本示例基于vue2 + element-ui

element-ui 的官网demo是只保留到过滤值一级的,并不会保留其子级

目标

1、Tree 树形控件 保留过滤值的子级

2、在第一次过滤数据的基础上进行第二次过滤

3、Tree 树形控件 多选改为单选,且只有最末端子级可以选择

不足之处

过滤时无法指定过滤哪一级的数据(保留其子级数据)

实际场景需求

  1. 第一次过滤时只过滤第三级数据
  2. 第二次过滤时在第一次过滤的基础上过滤最后一级

1 暂时无法实现(如果有大佬用前端方法实现,记得call我!)

2 过滤最后一级可以实现(代码随后附上)

先看效果

1、2

3  多选改单选效果

Tree 树形控件 保留过滤值的子级

<el-treeclass="filter-tree"node-key="id":data="treeData":props="defaultProps"default-expand-allshow-checkbox@check-change="handleCheckChange":filter-node-method="filterNode"ref="tree">
</el-tree>// 对树节点进行筛选时执行的方法
filterNode (value, data, node) {let parentNode = node.parent; // 父级nodelet labels = [node.label]; // 当前node的名字let level = 1; // 层级while (level < node.level) {labels = [...labels, parentNode.label]; // 当前node名字,父级node的名字parentNode = parentNode.parent;level++;}return labels.some((d) => d.indexOf(value) !== -1);
},

在第一次过滤数据的基础上进行第二次过滤

1、下载插件库

npm install -S circular-json

2、在使用页面引入

import CircularJSON from 'circular-json'

3、代码应用

<template><div class="about-container"><h1>获取过滤后的tree数据</h1><el-inputclearableplaceholder="第一次过滤"@keyup.enter.native="handleFilter1"v-model="firstText"></el-input><el-inputstyle="margin-top: 10px;"clearableplaceholder="第二次过滤"@keyup.enter.native="handleFilter2"v-model="secondText"></el-input><el-button @click="handleFilter1(),handleFilter2()" type="primary" icon="el-icon-search">搜索</el-button><div><el-treeclass="filter-tree"node-key="id":data="treeData":props="defaultProps"default-expand-allshow-checkbox@check-change="handleCheckChange":filter-node-method="filterNode"ref="tree"></el-tree></div></div>
</template><script>// 在使用的组件内引入import CircularJSON from 'circular-json'export default {data() {return {// 第一次过滤firstText: '',// 第二次过滤secondText:'',// tree控件的渲染值treeData: [{id: 1,label: '一级1',children: [{id: 4,label: '二级1-1',children: [{id: 9,label: '三级1-1-1',children: [{id:91,label: '四级1'},{id:92,label: '四级2'}]}, {id: 10,label: '三级1-1-2'}]}]}, {id: 2,label: '一级2',children: [{id: 5,label: '二级2-1'}, {id: 6,label: '二级2-2'}]}, {id: 3,label: '一级',children: [{id: 7,label: '二级3-1'}, {id: 8,label: '二级3-2'},{id: 82,label: '四级3'}]},{id: 31,label: '奇迹',children: [{id: 71,label: '奇迹1'}, {id: 81,label: '奇迹2'}]}],defaultProps: {children: 'children',label: 'label'},// tree的原版备份数据deepCloneTreeData:[],// 第一次过滤后tree控件渲染数据firstFilterdata:[],}},mounted() {this.deepClone(this.treeData).then((res)=>{this.deepCloneTreeData = res;})},methods: {// 第一次过滤handleFilter1() {// 第一次过滤时没有输入值,即用tree控件的原始值if(!this.firstText){this.treeData = this.deepCloneTreeData;this.firstFilterdata = this.deepCloneTreeData;return}this.$refs.tree.filter(this.firstText)this.firstFilterdata = this.getFilterData();},// 第二次过滤handleFilter2() {// 基于第一次过滤出来的数据if(this.firstFilterdata.length > 0) this.treeData = this.firstFilterdata;this.$nextTick(()=>{this.$refs.tree.filter(this.secondText);})},// 对树节点进行筛选时执行的方法filterNode (value, data, node) {let parentNode = node.parent; // 父级nodelet labels = [node.label]; // 当前node的名字let level = 1; // 层级while (level < node.level) {labels = [...labels, parentNode.label]; // 当前node名字,父级node的名字parentNode = parentNode.parent;level++;}return labels.some((d) => d.indexOf(value) !== -1);},// tree 的选择事件handleCheckChange(data, checked, indeterminate) {const arr = this.$refs.tree.getCheckedKeys()},// 需要获取过滤后的 Tree组件数据getFilterData() {const rootData = this.$refs.tree.root;if (rootData.visible) {const childNodesStr = CircularJSON.stringify(rootData.childNodes);const childNodes = CircularJSON.parse(childNodesStr);const filterData = this.recursionNodes(childNodes);return filterData;}},/*** 递归遍历数据* 这里解释一下为什么要用CircularJSON这个插件,因为element tree* node数据存在一个对象里的子项存在循环引用,存在循环引用的对象*/recursionNodes(childNodes) {const nodes = childNodes;const result = [];for (const item of nodes) {if (item.visible) {result.push(item.data);}if (item.childNodes && item.childNodes.length) {const tempResult = this.recursionNodes(item.childNodes);item.data.children = tempResult;}}return result;},/*** 深拷贝*/deepClone(obj){return new Promise((resolve) => {const { port1,port2 } = new MessageChannel();port1.postMessage(obj);port2.onmessage = (msg) => {resolve(msg.data)}})}},}
</script>

只过滤最后一级

/** * 数据处理* * 需要对每级数据进行level标记,方便后续过滤*/
treeData: [{id: 1,label: '一级1',level:1,children: [{id: 4,label: '二级1-1',level:2,children: [{id: 9,label: '三级1-1-1',level:3,children: [{id:91,label: '四级1',level:4,},{id:92,label: '四级2',level:4,}]}, {id: 10,label: '三级1-1-2',level:3,}]}]
}, {id: 2,label: '一级2',level:1,children: [{id: 5,label: '二级2-1',level:2,}, {id: 6,label: '二级2-2',level:2,}]
}, {id: 3,label: '一级',level:1,children: [{id: 7,label: '二级3-1',level:2,}, {id: 8,label: '二级3-2',level:2,},{id: 82,label: '四级3',level:2,}]
},{id: 31,label: '奇迹',level:1,children: [{id: 71,label: '奇迹1',level:2,}, {id: 81,label: '奇迹2',level:2,}]
}]// 对树节点进行筛选时执行的方法
filterNode(value, data, node) {/*** 筛选最后一级可用* * [4].includes(data.level* [] 中写入最后一级的level值*/if ([4].includes(data.level) && data.label.indexOf(value) !== -1) {console.log(value,data,node,'4-------',data.label.indexOf(value));let parentNode = node.parent; // 父级nodelet labels = [node.label]; // 当前node的名字let level = 1; // 层级while (level < node.level) {// console.log(labels,'labels');labels = [...labels, parentNode.label]; // 当前node名字,父级node的名字parentNode = parentNode.parent;level++;}return labels.some((d) => d.indexOf(value) !== -1);}
},

Element tree组件 多选改为单选

<el-treeclass="treeRadio"check-strictly@check-change="deptCheck"deptCheck(data, check, childCheck) {if(check){this.$refs.tree.setCheckedNodes([data]);}
}// css 让父级不显示checkbox
::v-deep .el-tree .el-tree-node .is-leaf + .el-checkbox .el-checkbox__inner{display: inline-block;
}
::v-deep .el-tree .el-tree-node .el-checkbox .el-checkbox__inner{display: none;
}
::v-deep .treeRadio .el-checkbox__inner{border-radius: 50%;transform: scale(1.2);
}
// 样式改为单选框的样式
::v-deep .treeRadio .el-checkbox__inner{border-radius: 50%;transform: scale(1.2);
}

这篇关于element-ui Tree 树形控件 过滤保留子级并获取过滤后的数据 多选改单选的文章就介绍到这儿,希望我们推荐的文章对编程师们有所帮助!



http://www.chinasem.cn/article/570448

相关文章

SQL中如何添加数据(常见方法及示例)

《SQL中如何添加数据(常见方法及示例)》SQL全称为StructuredQueryLanguage,是一种用于管理关系数据库的标准编程语言,下面给大家介绍SQL中如何添加数据,感兴趣的朋友一起看看吧... 目录在mysql中,有多种方法可以添加数据。以下是一些常见的方法及其示例。1. 使用INSERT I

Python使用vllm处理多模态数据的预处理技巧

《Python使用vllm处理多模态数据的预处理技巧》本文深入探讨了在Python环境下使用vLLM处理多模态数据的预处理技巧,我们将从基础概念出发,详细讲解文本、图像、音频等多模态数据的预处理方法,... 目录1. 背景介绍1.1 目的和范围1.2 预期读者1.3 文档结构概述1.4 术语表1.4.1 核

Python使用OpenCV实现获取视频时长的小工具

《Python使用OpenCV实现获取视频时长的小工具》在处理视频数据时,获取视频的时长是一项常见且基础的需求,本文将详细介绍如何使用Python和OpenCV获取视频时长,并对每一行代码进行深入解析... 目录一、代码实现二、代码解析1. 导入 OpenCV 库2. 定义获取视频时长的函数3. 打开视频文

MySQL 删除数据详解(最新整理)

《MySQL删除数据详解(最新整理)》:本文主要介绍MySQL删除数据的相关知识,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录一、前言二、mysql 中的三种删除方式1.DELETE语句✅ 基本语法: 示例:2.TRUNCATE语句✅ 基本语

MyBatisPlus如何优化千万级数据的CRUD

《MyBatisPlus如何优化千万级数据的CRUD》最近负责的一个项目,数据库表量级破千万,每次执行CRUD都像走钢丝,稍有不慎就引起数据库报警,本文就结合这个项目的实战经验,聊聊MyBatisPl... 目录背景一、MyBATis Plus 简介二、千万级数据的挑战三、优化 CRUD 的关键策略1. 查

python实现对数据公钥加密与私钥解密

《python实现对数据公钥加密与私钥解密》这篇文章主要为大家详细介绍了如何使用python实现对数据公钥加密与私钥解密,文中的示例代码讲解详细,感兴趣的小伙伴可以跟随小编一起学习一下... 目录公钥私钥的生成使用公钥加密使用私钥解密公钥私钥的生成这一部分,使用python生成公钥与私钥,然后保存在两个文

mysql中的数据目录用法及说明

《mysql中的数据目录用法及说明》:本文主要介绍mysql中的数据目录用法及说明,具有很好的参考价值,希望对大家有所帮助,如有错误或未考虑完全的地方,望不吝赐教... 目录1、背景2、版本3、数据目录4、总结1、背景安装mysql之后,在安装目录下会有一个data目录,我们创建的数据库、创建的表、插入的

MySQL 获取字符串长度及注意事项

《MySQL获取字符串长度及注意事项》本文通过实例代码给大家介绍MySQL获取字符串长度及注意事项,本文通过实例代码给大家介绍的非常详细,对大家的学习或工作具有一定的参考借鉴价值,需要的朋友参考下吧... 目录mysql 获取字符串长度详解 核心长度函数对比⚠️ 六大关键注意事项1. 字符编码决定字节长度2

Navicat数据表的数据添加,删除及使用sql完成数据的添加过程

《Navicat数据表的数据添加,删除及使用sql完成数据的添加过程》:本文主要介绍Navicat数据表的数据添加,删除及使用sql完成数据的添加过程,具有很好的参考价值,希望对大家有所帮助,如有... 目录Navicat数据表数据添加,删除及使用sql完成数据添加选中操作的表则出现如下界面,查看左下角从左

SpringBoot中4种数据水平分片策略

《SpringBoot中4种数据水平分片策略》数据水平分片作为一种水平扩展策略,通过将数据分散到多个物理节点上,有效解决了存储容量和性能瓶颈问题,下面小编就来和大家分享4种数据分片策略吧... 目录一、前言二、哈希分片2.1 原理2.2 SpringBoot实现2.3 优缺点分析2.4 适用场景三、范围分片