Element Ui Tree组件实现增、删、改、查、拖拽节点 的树形结构

news2025/1/22 23:01:05

介绍:首先组件 | Element官网某些功能都具备了,这里我就把这些功能结合在一起更完美的使用,其次编辑节点官网是没有实例,所以这里搞了一套较完整的功能,其次编辑和添加,这里直接使用了弹窗(顾及到多个参数设置),接下来效果图展示!

效果图如下:

1,其中点击展开/折叠看个人需求;

2,组件中 :default-checked-keys="List_id" (如: 获取所有权限,匹配改个节点(角色)所包含的权限,修改时默认勾选,List_id 该角色拥有权限的id的集合!)讲述的可能稍有复杂,代码有注释可以看看!

(自己引入相关插件)

    src: url('../css/element-icons.woff') format("woff");这个是图标不展示问题

 这里就直接展示代码了!

<!DOCTYPE html>
<html>
	<head>
		<meta charset="utf-8">
		<title></title>
		<link rel="stylesheet" href="../css/index.css" />
		<link rel="stylesheet" href="../css/element-ui.css" />
		<script src="../js/vue.js"></script>
		<script src="../js/element-ui.js"></script>
		<style type="text/css">
			@font-face {
				font-family: element-icons;
				src: url('../css/element-icons.woff') format("woff");
				font-weight: 400;
				font-display: "auto";
				font-style: normal
			}

			.fade-enter-active,
			.fade-leave-active {
				transition: opacity 5s
			}

			.fade-enter,
			.fade-leave-active {
				opacity: 0
			}

			.custom-tree-node {
				flex: 1;
				/* display: flex; */
				align-items: center;
				justify-content: space-between;
				font-size: 14px;
				padding-right: 8px;
			}

			.show-hide:hover :nth-child(2) {
				display: inline-block !important;
			}

			.el-tree-node__content {
				height: 30px !important;
			}

			.el-input__inner {
				height: 28px !important;
			}

			em {
				color: red;
				font-style: inherit !important;
			}

			.el-input__inner {
				height: 20px;
			}

			.dialog_input>input {
				height: 34px !important;
			}

			input.el-input__inner {
				height: 34px !important;
			}

			.btn_is {
				padding: 7px !important;
				font-size: 15px !important;
				background: rgb(39, 99, 220);
				color: white;
			}

			.el-tree {
				overflow-y: auto;
				max-height: calc(100vh - 180px);
				border-bottom: 1px solid #c8c8c8;
				padding: 5px 0px;
			}

			.dialog_input {
				width: 400px !important;
			}

			input.el-input__inner {
				width: 80%;
			}
		</style>
	</head>
	<body>
		<div id="app">
			<template>
				<el-button class="btn_is" size="medium" @click="toggleRowExpansion()">全部展开/折叠</el-button>
				<el-input placeholder="输入关键字" v-model="filterText" highlight-current></el-input>
				<!-- default-expand-all:默认打开全部
				 :expand-on-click-node="true" 打开点击当前行节点-->
				<el-tree v-if="refreshTable" :default-expand-all="isExpandAll" :data="data" class="filter-tree"
					:highlight-current="true" :filter-node-method="filterNode" show-checkbox node-key="id" ref="tree"
					highlight-current :props="defaultProps" @node-drag-start="handleDragStart"
					@node-drag-leave="handleDragLeave" @node-drag-over="handleDragOver" @node-drag-end="handleDragEnd"
					@node-drop="handleDrop" :allow-drop="allowDrop" :allow-drag="allowDrag" draggable
					:expand-on-click-node="false" :default-checked-keys="List_id">
					<span class="custom-tree-node show-hide keywords_em" slot-scope="{ node, data }">
						<span class="ssss" v-if="filterText != ''" v-html="ruleTitle(node.label)"></span>

						<span class="ddd" v-if="filterText == ''" v-text="data.label"></span>
						<span style="display:none">
							<el-button v-if="data.id !== 1" v-if="data.id!=1" type="text" size="mini"
								@click="() => edit(node,data)">
								<i class="el-icon-edit"></i>
							</el-button>
							<el-button type="text" size="mini" @click="() => append(node,data)">
								<i class="el-icon-plus"></i>
							</el-button>
							<el-button type="text" size="mini" @click="() => remove(node, data)">
								<i class="el-icon-delete"></i>
							</el-button>
						</span>
					</span>
				</el-tree>
				<!-- 编辑 -->
				<el-dialog style="width: 100%;height: 100%;" title="角色权限修改" :visible.sync="upd_dialog">
					<div>
						<el-form :model="upd_data">
							<el-form-item label="名称:" width="80px">
								<el-input class="dialog_input" v-model="upd_data.label"></el-input>
							</el-form-item>
							<el-form-item label="id:" width="80px">
								<el-input class="dialog_input" v-model="upd_data.id"></el-input>
							</el-form-item>
						</el-form>
					</div>
					<div slot="footer" class="dialog-footer">
						<button class="submitbtn" @click="upd_submit">提 交</button>
						<button class="censelbtn" @click="cancel">取 消</button>
					</div>
				</el-dialog>
				<!-- 添加 -->
				<el-dialog style="width: 100%;height: 100%;" title="角色权限添加" :visible.sync="zdydialog">
					<div>
						<el-form :model="add_data">
							<el-form-item label="名称:" width="80px">
								<el-input class="dialog_input" v-model="add_data.label"></el-input>
							</el-form-item>
							<el-form-item label="序号:" width="80px">
								<el-input class="dialog_input" v-model="add_data.id"></el-input>
							</el-form-item>
						</el-form>
					</div>
					<div slot="footer" class="dialog-footer">
						<button class="submitbtn" @click="add_submit">提 交</button>
						<button class="censelbtn" @click="cancel">取 消</button>
					</div>
				</el-dialog>
			</template>
		</div>
	</body>
	<script>
		new Vue({
			el: "#app",
			data() {
				return {
					data: [{
						id: 1,
						label: '一级',
						children: [{
							id: 4,
							label: '二级',
							children: [{
								id: 9,
								label: '三级'
							}, {
								id: 10,
								label: '三级'
							}]
						}]
					}, {
						id: 2,
						label: '一级',
						children: [{
							id: 5,
							label: '二级'
						}, {
							id: 6,
							label: '二级'
						}]
					}, {
						id: 3,
						label: '一级',
						children: [{
							id: 7,
							label: '二级'
						}, {
							id: 8,
							label: '二级'
						}]
					}],
					newApiGroupName: '',
					defaultProps: {
						children: 'children',
						label: 'label'
					},
					updateApiGroup: {},
					id: 1000,
					resetNodeForm: {},
					filterText: '',
					newname: '',
					isExpandAll: true,
					refreshTable: true,
					List_id: [5, 7], //匹配到的id默认勾选,
					zdydialog: false,
					upd_dialog: false,
					add_data: {},
					upd_data: {},
				};
			},
			watch: {
				filterText(val) {
					console.log(val)
					this.$refs.tree.filter(val);
				}
			},
			created() {},
			methods: {
				edit(node, row) {
					console.log(row)
					this.upd_dialog = true
					this.upd_data = row
				},
				upd_submit() {
					//调取相关接口
					this.upd_dialog = false
				},

				append(node, row) {
					this.add_data = {
						id: '',
						label: '',
					}
					if (!row.children) {
						this.$set(row, 'children', [])
					}
					row.children.push(this.add_data)
					console.log(this.add_data)
					this.zdydialog = true
				},
				add_submit() {
					//调取相关接口
					this.zdydialog = false
				},
				cancel() {
					this.upd_dialog = false;
					this.zdydialog = false;
				},
				// 全部展开/折叠
				toggleRowExpansion() {
					this.refreshTable = false;
					this.isExpandAll = !this.isExpandAll;
					this.$nextTick(() => {
						this.refreshTable = true;
					});
				},
				// 高亮搜索的关键字
				ruleTitle(items) {
					const title = items
					const rep = new RegExp(this.filterText, 'g')
					const resDtring = `<span style="color: #145afe">${this.filterText}</span>`
					return title.replace(rep, resDtring)
				},
				//筛选结点
				filterNode(value, data, node) {
					if (!value) return true;
					return this.findSearKey(node, value);
				},
				// 拖拽
				handleDragStart(node, ev) {
					console.log('drag start', node.data.label)
				},
				handleDragEnter(draggingNode, dropNode, ev) {
					console.log('tree drag enter: ', dropNode.data.label)
				},
				handleDragLeave(draggingNode, dropNode, ev) {
					console.log('tree drag leave: ', dropNode.data.label)
				},
				handleDragOver(draggingNode, dropNode, ev) {
					console.log('tree drag over: ', dropNode.data.label)
				},
				handleDragEnd(draggingNode, dropNode, dropType, ev) {
					console.log(
						'tree drag end: ',
						dropNode && dropNode.data.label,
						dropType
					)
					// 调后端更新
					// this.updateApiGroup(this.data)
				},
				handleDrop(draggingNode, dropNode, dropType, ev) {
					console.log('tree drop: ', dropNode.data.label, dropType)
				},
				allowDrop(draggingNode, dropNode, type) {
					if (dropNode.data.id === 1) {
						return false
					} else {
						return true
					}
				},
				allowDrag(draggingNode) {
					// 顶层默认分组不允许拖拽
					if (draggingNode.data.id === 1) {
						return false
					} else {
						return true
					}
					// return draggingNode.data.apiGroupName.indexOf('三级 3-2-2') === -1
				},
				remove(node, data) {
					const parent = node.parent
					const children = parent.data.children || parent.data
					const index = children.findIndex(d => d.id === data.id)
					children.splice(index, 1)
					// this.updateApiGroup(this.data)
				},
			}
		})
	</script>
</html>

本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.coloradmin.cn/o/727952.html

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈,一经查实,立即删除!

相关文章

单位列表单列出来,假(封装)组件

效果图&#xff1a; 因为每个页面都用到这个单位&#xff0c;所以把单位列表单列出来&#xff0c;假装是个封装的组件&#xff0c;在其他页面直接用。 源码&#xff1a; <template><div style"height: 48rem;overflow-y: scroll"><h4>单位列表<…

阿里云国际站代理商:阿里云是干什么的?阿里云app和建网站有什么关系?

标题&#xff1a;阿里云是干什么的&#xff1f;阿里云app和建网站有什么关系&#xff1f;   一、解析阿里云的业务范围   阿里云&#xff0c;作为阿里巴巴集团的关键业务板块&#xff0c;主要提供云计算、大数据、人工智能及其他信息化服务。通过其全球网络&#xff0c;阿里…

8 spring-boot访问静态资源

8.1 静态资源存放的位置 在资源目录下分别创建public和resources两个文件夹&#xff0c;static是一开始就存在的&#xff0c;静态资源可以存放在这三个文件夹中。当这三个文件夹同时出现相同的静态资源&#xff0c;如每个文件夹都有一个1.js时&#xff0c;则优先访问resources里…

vue 目录

vue学习资源 vue.js中文官网&#xff1a; http://cn.vuejs.org/ vue.js源码&#xff1a; https://github.com/vuejs/vue vue.js官方工具&#xff1a; https://github.com/vuejs vue.js英文官网&#xff1a; https://vuejs.org/ vue全家桶 介绍 介绍 【 Vue全家桶 Vue&#xff…

Vue3使用echarts仪表盘(gauge)

Documentation - Apache ECharts 可自定义设置以下属性 仪表盘数据源&#xff08;gaugeData&#xff09;&#xff0c;类型&#xff1a;Gauge[]&#xff0c;必传&#xff0c;默认 []容器宽度&#xff08;width&#xff09;&#xff0c;类型&#xff1a;number | string&#x…

web前端(二)

表格标签&#xff1a; <table> </table>按照这个顺序&#xff1a;一个可选的 <caption> 元素零个或多个的 <colgroup> 元素一个可选的 <thead> 元素下列任意一个&#xff1a;零个或多个 <tbody>零个或多个 <tr>一个可选的 <t…

轻量应用服务器5m支持多少人访问?

​  轻量应用服务器5m支持多少人访问?对于网站而言&#xff0c;服务器的带宽肯定是越大越好&#xff0c;但对于用户的钱包则相反&#xff0c;服务器的价格高低与带宽大小、类型也有很大的关系&#xff0c;我们只有选择到合适的带宽才能将轻量应用服务器显得更有性价比&#…

复习V2+V3之——01 前言回顾

前言 Vue的特点 采用组件化的模式&#xff0c;提高代码复用率&#xff0c;让代码更好维护 声明式编码&#xff0c;开发者无需直接操作DOM&#xff0c;提高开发效率 使用虚拟DOM Diff算法&#xff0c;尽量复用DOM节点 虚拟DOM&#xff08;Virtual DOM&#xff09;&#xff1…

JAVA 牛客网 NC32求平方根

题目如图&#xff1a; 先展示代码&#xff1a; import java.util.*;public class Solution {/*** 代码中的类名、方法名、参数名已经指定&#xff0c;请勿修改&#xff0c;直接返回方法规定的值即可*** param x int整型* return int整型*/public int sqrt (int x) {if(x<2)…

递归回溯两个例题:1.数组组合 2.在矩阵中搜索单词

题目1&#xff1a;组合 给定两个整数 n 和 k&#xff0c;返回范围 [1, n] 中所有可能的 k 个数的组合。 你可以按 任何顺序 返回答案。 输入&#xff1a;n 4, k 2 输出&#xff1a; [ [2,4], [3,4], [2,3], [1,2], [1,3], [1,4], ] 解题思路&#xff1a; 1.定…

Mac苹果电脑也可以玩原神了,运行流程,nice~

最近发现了一个很棒的工具&#xff0c;他可以让你的 Mac 苹果电脑运行原神&#xff0c;而且画质和流畅度都是在线的&#xff0c;今天分享给大家 软件名字叫 playCover &#xff0c;根据作者的介绍这款软件最初就是国外的一位博主想在 Mac 上玩原神特意开发的一款软件&#xff…

代码随想录二刷 day45 | 动态规划 之 70. 爬楼梯 (进阶) 322. 零钱兑换 279.完全平方数

day45 70. 爬楼梯 &#xff08;进阶&#xff09;1. 确定dp数组以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例来推导dp数组 322. 零钱兑换1. 确定dp数组以及下标的含义2.确定递推公式3.dp数组如何初始化4.确定遍历顺序5.举例推导dp数组 279.完全平方数1.…

作为一名研究生/博士生,我应该知道什么

今天分享的是一位深度学习领域的大佬在2020年7月份写的一篇博客&#xff0c;看完或许对有缘的你带来些启发。 英文原址&#xff1a;大佬博客地址 https://theorangeduck.com/page/reproduce-their-results 1&#xff1a;我希望作为一名研究生我知道什么 在我攻读博士学位之初…

vue 多级导航菜单

目录&#xff0c;三个文件 tree-node.vue <template><div class"tree-node-container"><node-content></node-content><divclass"tree-node-children":style"{paddingLeft: indent}"v-if"nextShow">&l…

力扣 47. 全排列 II

题目来源&#xff1a;https://leetcode.cn/problems/permutations-ii/description/ C题解1&#xff1a;这道题需要有两个不同的数组来标记使用过的元素&#xff0c;一个存放同层使用过的元素&#xff0c;根据元素数值来判断&#xff0c;范围为[-10,10]&#xff0c;在同层更新即…

95、基于STM32单片机烟雾温度报警器设计火灾防火检测可燃气体监测系统设计(程序+原理图+任务书+参考论文+开题报告+元器件清单等)

摘 要 本设计采用使用STM32F103作为控制器件&#xff0c;MQ-2型半导体可燃气体敏感元件烟雾传感器作为检测烟雾器件&#xff0c;DS18B20温度传感器作为温度检测器件&#xff0c;蜂鸣器和电源指示灯作为报警器件&#xff0c;继电器作为水泵的控制器件。烟雾报警器主要由烟雾信…

动态多目标优化算法:基于自适应启动策略的混合交叉动态约束多目标优化算法(MC-DCMOEA)求解CEC2015(提供MATLAB代码)

一、动态多目标优化问题 1.1问题定义 1.2 动态支配关系定义 二、 基于自适应启动策略的混合交叉动态多目标优化算法 基于自适应启动策略的混合交叉动态多目标优化算法&#xff08;Mixture Crossover Dynamic Constrained Multi-objective Evolutionary Algorithm Based on S…

mysql sysdate()不走索引问题解析

1.常规比较 结论: SYSDATE()返回执行时间,而NOW()返回时间常量 2.索引比较 结论: 因为SYSDATE()是非确定性的,索引不能用于评估求值引用它的表达式,所以不走索引

隐语1.0正式发布|MVP部署体验包、资源调度框架Kuscia全新亮相!

2023 年 7 月 7 日&#xff0c;在世界人工智能大会组委会办公室指导下&#xff0c;隐语开源社区携手蚂蚁集团和机器之心共同主办的数据要素与隐私计算论坛在上海世博会议中心举行。论坛上&#xff0c;蚂蚁集团隐私计算部总经理、隐语社区负责人王磊发布了隐语 1.0 版本&#xf…

java项目之美食推荐管理系统(ssm+mysql+jsp)

风定落花生&#xff0c;歌声逐流水&#xff0c;大家好我是风歌&#xff0c;混迹在java圈的辛苦码农。今天要和大家聊的是一款基于ssm的美食推荐管理系统。 开发环境&#xff1a; 后端&#xff1a; 开发语言&#xff1a;Java 框架&#xff1a;ssm&#xff0c;mybatis JDK版…