vue使用树形结构展示文件和文件夹
1. 树形结构显示
- 显示文件夹和文件:使用
el-tree组件展示树形结构,文件夹和文件的图标通过el-icon进行动态显示。文件夹使用Folder图标,文件使用Files图标。 - 节点点击:点击树形节点后,会将选中的节点保存到
selectedNode中,方便后续操作(如重命名、删除)。
2. 创建文件夹
- 在根目录或者当前选中的文件夹下,点击
新建文件夹按钮,触发createFolder方法创建一个新的文件夹。 - 新文件夹将被添加到当前选中节点的
children数组中,或者如果没有选中任何节点,则将文件夹添加到根目录。
3. 创建文件
- 类似于文件夹,点击
新建文件按钮,触发createFile方法创建一个新的文件,并将其添加到当前选中的文件夹下或者根目录。
4. 删除文件/文件夹
- 通过点击
删除文件/文件夹按钮,触发removeFile方法,删除当前选中的文件或文件夹。 - 删除操作会检查所选文件夹或文件的父级节点,如果有父节点,直接从父节点的
children中移除该项;如果是根节点,则从根目录数组中删除。
5. 重命名文件/文件夹
- 通过右键点击某个节点,弹出自定义的上下文菜单。选择
重命名选项后,触发renameNode方法,通过prompt弹窗输入新的名称,并更新节点的label。
6. 删除文件/文件夹(右键菜单)
- 右键点击文件夹或文件,弹出自定义右键菜单,选择
删除后会弹出确认对话框,确认删除后,调用deleteNode删除当前选中的文件/文件夹。
7. 上下文菜单
- 在树形结构中,右键点击文件夹或文件时,会弹出自定义的上下文菜单,显示
重命名和删除选项。 - 上下文菜单会根据鼠标点击的位置动态显示,通过
contextMenuPosition控制菜单的位置。
8. 事件监听
- 点击外部区域隐藏上下文菜单:使用
document.addEventListener("click", handleOutsideClick)监听点击事件,点击树形组件外部区域时,隐藏上下文菜单并清空选中的节点。 - 组件卸载时移除事件监听:在组件卸载时,通过
onBeforeUnmount钩子移除点击外部区域的事件监听,避免内存泄漏。

<script setup>
import { ref, onMounted, onBeforeUnmount } from "vue";
import { Folder, Files } from "@element-plus/icons-vue";// 在组件挂载后添加事件监听
onMounted(() => {document.addEventListener("click", handleOutsideClick);
});
// 在组件卸载前移除事件监听
onBeforeUnmount(() => {document.removeEventListener("click", handleOutsideClick);
});
// 定义树形数据
const data = ref([{label: "文件夹1",type: "folder",children: [{label: " 1-1",type: "folder",children: [{label: "1-1-1",type: "file",},],},],},{label: "文件夹2",type: "folder",children: [{label: "2-1",type: "folder",children: [{label: "2-1-1",type: "file",},],},{label: "2-2",type: "file",},],},{label: "文件夹3",type: "folder",children: [],},
]);// 配置树形数据的显示属性,指定文件夹使用 Folder 图标
const defaultProps = {children: "children",label: "label",icon: "icon", // 图标属性
};
// 当前选中的节点
const selectedNode = ref(null);
const handleNodeClick = (data) => {selectedNode.value = data;
};
// 监听点击事件,判断点击是否在树形组件外
const handleOutsideClick = (event) => {const treeElement = document.querySelector(".el-tree"); // 获取树形组件的 DOM 元素const buttonContainer = document.querySelector(".header"); // 获取按钮区域// 判断点击区域是否在树形组件或按钮区域外if (!treeElement.contains(event.target) &&!buttonContainer.contains(event.target)) {selectedNode.value = null; // 如果点击区域不是树或按钮区域,则清除选中的节点showContextMenu.value = false; // 隐藏菜单}
};
// 创建文件夹
const createFolder = () => {const newFolder = {label: "新文件夹",type: "folder",children: [],};if (selectedNode.value && selectedNode.value.type === "folder") {// 在选中的文件夹下创建selectedNode.value.children.push(newFolder);} else {// 根目录下创建data.value.push(newFolder);}
};// 创建文件
const createFile = () => {const newFile = {label: "新文件",type: "file",};if (selectedNode.value && selectedNode.value.type === "folder") {// 在选中的文件夹下创建selectedNode.value.children.push(newFile);} else {// 根目录下创建data.value.push(newFile);}
};// 删除文件或文件夹
const removeFile = () => {if (!selectedNode.value) {alert("请先选择一个文件或文件夹");return;}const parentNode = findParentNode(data.value, selectedNode.value);if (parentNode) {const index = parentNode.children.findIndex((item) => item.label === selectedNode.value.label);if (index !== -1) {parentNode.children.splice(index, 1);selectedNode.value = null; // 删除后清空选中的节点}} else {// 如果没有父节点,表示是根节点,直接从根节点删除const index = data.value.findIndex((item) => item.label === selectedNode.value.label);if (index !== -1) {data.value.splice(index, 1);selectedNode.value = null; // 删除后清空选中的节点}}
};
// 查找父节点
const findParentNode = (nodes, targetNode) => {for (let node of nodes) {if (node.children && node.children.includes(targetNode)) {return node;} else if (node.children) {const result = findParentNode(node.children, targetNode);if (result) return result;}}return null;
};
// 控制上下文菜单显示
const showContextMenu = ref(false);
// 上下文菜单位置
const contextMenuPosition = ref({ top: 0, left: 0 });
//右键显示菜单
const handleContextMenu = (event, data, node) => {event.preventDefault(); // 阻止默认右键菜单selectedNode.value = node.data; // 保存当前节点showContextMenu.value = true; // 显示自定义右键菜单contextMenuPosition.value = { top: event.clientY, left: event.clientX }; // 设置菜单显示位置
};
//重命名
const renameNode = () => {if (selectedNode.value) {const newName = prompt("请输入新名称", selectedNode.value.label);if (newName) {selectedNode.value.label = newName; // 修改节点名称}}showContextMenu.value = false; // 隐藏菜单
};
//删除
const deleteNode = () => {if (selectedNode.value) {const confirmDelete = confirm(`确定删除 ${selectedNode.value.label} 吗?`);if (confirmDelete) {removeFile()}}showContextMenu.value = false; // 隐藏菜单
};
</script><template><div class="container"><header class="header"><!-- 新建文件和文件夹的按钮 --><el-button type="primary" @click="createFolder">新建文件夹</el-button><el-button type="primary" @click="createFile">新建文件</el-button><el-button type="primary" @click="removeFile">删除文件/文件夹</el-button></header><div class="main"><aside class="aside"><el-treestyle="max-width: 600px":data="data":props="defaultProps"@node-click="handleNodeClick"@node-contextmenu="handleContextMenu"><!-- 使用 render-content 插槽自定义节点显示 --><template #default="{ node, data }"><span><!-- 判断节点类型来动态显示图标 --><el-icon v-if="data.type === 'folder'"><Folder /></el-icon><el-icon v-if="data.type === 'file'"><Files /></el-icon>{{ data.label }}</span></template></el-tree><divclass="dropdown"v-if="showContextMenu":style="{position: 'absolute',top: contextMenuPosition.top + 'px',left: contextMenuPosition.left + 'px',}"><ul slot="dropdown"><li @click="renameNode">重命名</li><li @click="deleteNode">删除</li></ul></div></aside><div class="content"><!-- 内容区域 --></div></div></div>
</template><style scoped>
.container {width: 100%;height: 100%;display: flex;flex-direction: column;
}.header {height: 60px;border:1px solid #ccc;
}.main {flex: 1;display: flex;
}.aside {width: 200px;border-right: 1px solid #ccc;
}.content {flex: 1;padding: 10px;
}
.dropdown {position: absolute;background-color: #fff;padding:20px;border-radius: 5px;border:1px solid #ccc;
}
</style>
相关文章:
vue使用树形结构展示文件和文件夹
1. 树形结构显示 显示文件夹和文件:使用 el-tree 组件展示树形结构,文件夹和文件的图标通过 el-icon 进行动态显示。文件夹使用 Folder 图标,文件使用 Files 图标。节点点击:点击树形节点后,会将选中的节点保存到 sel…...
PHP框架+gatewayworker实现在线1对1聊天--聊天界面布局+创建websocket连接(5)
文章目录 聊天界面布局html代码 创建websocket连接为什么要绑定? 聊天界面布局 在View/Index目录下创建index.html html代码 <div id"chat"><div id"nbar"><div class"pull-left">与牛德胜正在聊天...</div…...
LinuxUbuntu打开VSCode白屏解决方案
解决方法是 以root权限打开VSCode sudo /usr/share/code/code --no-sandbox --unity-launch...
在 ESP 上运行 AWTK
AWTK 基于 esp 的移植。 测试硬件平台为 ESP32-S3-Touch-LCD-4.3,其它平台请根据实际平台自行调整。 安装下载工具 建议下载离线版本 ESP IDF v5.3.2 下载代码 git clone https://github.com/zlgopen/awtk-esp.git cd awtk-esp git clone https://github.com/zlg…...
硬件工程师面试题 21-30
把常见的硬件面试题进行总结,方便及时巩固复习。其中包括网络上的资源、大佬们的大厂面试题,其中可能会题目类似,加强印象即可。 更多硬件面试题:硬件工程师面试题 1-10硬件工程师面试题 11-20 21、单片机最小系统需要什么&#x…...
开源架构的容器化部署优化版
上三篇文章推荐: 开源架构的微服务架构实践优化版(New) 开源架构中的数据库选择优化版(New) 开源架构学习指南:文档与资源的智慧锦囊(New) 我管理的社区推荐:【青云交社区…...
Qt使用CMake编译项目时报错:#undefined reference to `vtable for MainView‘
博主将.h文件和.cpp文件放到了不同的文件目录下面,如下图所示: 于是构建项目的时候就报错了#undefined reference to vtable for MainView,这个是由于src/view目录下的CMake无法自动moc头文件导致的,需要手动moc include/view目录…...
python学习笔记—12—
1. 布尔类型 (1) 定义 (2) 比较运算符 (3) 代码演示 1. 手动定义 bool_1 True bool_2 False print(f"bool_1的内容是:{bool_1}, 类型是:{type(bool_1)}") print(f"bool_2的内容是:{bool_2}, 类型是:{type(bool…...
==和===的区别,被坑的一天
在 JavaScript 中, 和 都用于比较两个值,但它们有一个重要的区别: 1. (宽松相等运算符) 进行比较时,会 自动类型转换(也叫做强制类型转换),即如果比较的两个值的类型不同,JavaScr…...
基于 GPUTasker 的 GPU 使用情况钉钉推送机器人实现
引言 https://github.com/cnstark/gputasker 随着 AI 模型的广泛应用,GPU 成为团队中最重要的资源之一。然而,如何实时监控 GPU 的使用情况并及时通知团队是一个值得关注的问题。为了更好地管理显卡资源,本文基于 GPUTasker,实现了…...
Python自学 - 函数初步(内置函数、模块函数、自定义函数)
1 Python自学 - 函数初步(内置函数、模块函数、自定义函数) 1.1 内置函数 几乎所有的编程都会提供一些内置函数,以便完成一些最基本的任务,Python提供了丰富的内置函数,熟悉内置函数可以给工作带来极大便利。 Python官方的内置函数介绍网…...
【生活】冬天如何选口罩(医用口罩,N95, KN95还是KP95?带不带呼吸阀门?带不带活性炭?)
💡总结一下就是: 日常防护的话,医用口罩就可以啦。要是想长时间佩戴N95(KN95)口罩的话也可以. 在高风险环境(像医院、疫情防控期间),一定要选不带呼吸阀门的N95口罩KN95)…...
HTML5新特性|01 音频视频
音频 1、Audio (音频) HTML5提供了播放音频文件的标准 2、control(控制器) control 属性供添加播放、暂停和音量控件 3、标签: <audio> 定义声音 <source> 规定多媒体资源,可以是多个<!DOCTYPE html> <html lang"en"> <head><…...
迅为RK3568开发板编译Android12源码包-设置屏幕配置
在源码编译之前首先要确定自己想要使用的屏幕并修改源码,在编译镜像,烧写镜像。如下图所示: 第一步:确定要使用的屏幕种类,屏幕种类选择如下所示: iTOP-3568 开发板支持以下种类屏幕: 迅为 LV…...
力扣hot100——图论
200. 岛屿数量 class Solution { public:int numIslands(vector<vector<char>>& grid) {int ans 0;vector<int> dx { 0, 1, 0, -1 };vector<int> dy { 1, 0, -1, 0 };int n grid.size(), m grid[0].size();vector<vector<int>> …...
Docker- Unable to find image “hello-world“locally
Docker- Unable to find image “hello-world“locally 文章目录 Docker- Unable to find image “hello-world“locally问题描述一. 切换镜像1. 编辑镜像源2. 切换镜像内容 二、 检查设置1、 重启dockers2、 检查配置是否生效3. Docker镜像源检查4. Dokcer执行测试 三、自定义…...
spring-boot启动源码分析(二)之SpringApplicationRunListener
在上一篇《spring-boot启动源码分析(一)之SpringApplication实例构造》后,继续看了一个月的Spring boot启动源码,初步把流程看完了,接下来会不断输出总结,以巩固这段时间的学习。同时也希望能帮到同样感兴趣…...
ELK入门教程(超详细)
什么是ELK? ELK是Elasticsearch、Logstash、Kibana三大开源框架首字母大写简称(后来出现的filebeat属于beats家族中的一员,可以用来替代logstash的数据收集功能,比较轻量级),也被称为Elastic Stack。 Filebeat Filebeat是用于转…...
人工智能知识分享第六天-机器学习_逻辑回归(Logistic Regression)
简介 在机器学习中,分类问题是一种常见的任务,目标是根据输入特征将数据点分配到不同的类别中。为了实现分类,我们需要训练一个分类器,该分类器能够根据输入数据的特征进行预测。 逻辑回归(Logistic Regression&…...
基于Springboot + vue实现的校园周边美食探索及分享平台
🥂(❁◡❁)您的点赞👍➕评论📝➕收藏⭐是作者创作的最大动力🤞 💖📕🎉🔥 支持我:点赞👍收藏⭐️留言📝欢迎留言讨论 🔥🔥&…...
【力扣数据库知识手册笔记】索引
索引 索引的优缺点 优点1. 通过创建唯一性索引,可以保证数据库表中每一行数据的唯一性。2. 可以加快数据的检索速度(创建索引的主要原因)。3. 可以加速表和表之间的连接,实现数据的参考完整性。4. 可以在查询过程中,…...
java 实现excel文件转pdf | 无水印 | 无限制
文章目录 目录 文章目录 前言 1.项目远程仓库配置 2.pom文件引入相关依赖 3.代码破解 二、Excel转PDF 1.代码实现 2.Aspose.License.xml 授权文件 总结 前言 java处理excel转pdf一直没找到什么好用的免费jar包工具,自己手写的难度,恐怕高级程序员花费一年的事件,也…...
Java多线程实现之Callable接口深度解析
Java多线程实现之Callable接口深度解析 一、Callable接口概述1.1 接口定义1.2 与Runnable接口的对比1.3 Future接口与FutureTask类 二、Callable接口的基本使用方法2.1 传统方式实现Callable接口2.2 使用Lambda表达式简化Callable实现2.3 使用FutureTask类执行Callable任务 三、…...
Nuxt.js 中的路由配置详解
Nuxt.js 通过其内置的路由系统简化了应用的路由配置,使得开发者可以轻松地管理页面导航和 URL 结构。路由配置主要涉及页面组件的组织、动态路由的设置以及路由元信息的配置。 自动路由生成 Nuxt.js 会根据 pages 目录下的文件结构自动生成路由配置。每个文件都会对…...
selenium学习实战【Python爬虫】
selenium学习实战【Python爬虫】 文章目录 selenium学习实战【Python爬虫】一、声明二、学习目标三、安装依赖3.1 安装selenium库3.2 安装浏览器驱动3.2.1 查看Edge版本3.2.2 驱动安装 四、代码讲解4.1 配置浏览器4.2 加载更多4.3 寻找内容4.4 完整代码 五、报告文件爬取5.1 提…...
中医有效性探讨
文章目录 西医是如何发展到以生物化学为药理基础的现代医学?传统医学奠基期(远古 - 17 世纪)近代医学转型期(17 世纪 - 19 世纪末)现代医学成熟期(20世纪至今) 中医的源远流长和一脉相承远古至…...
Java编程之桥接模式
定义 桥接模式(Bridge Pattern)属于结构型设计模式,它的核心意图是将抽象部分与实现部分分离,使它们可以独立地变化。这种模式通过组合关系来替代继承关系,从而降低了抽象和实现这两个可变维度之间的耦合度。 用例子…...
2025年渗透测试面试题总结-腾讯[实习]科恩实验室-安全工程师(题目+回答)
安全领域各种资源,学习文档,以及工具分享、前沿信息分享、POC、EXP分享。不定期分享各种好玩的项目及好用的工具,欢迎关注。 目录 腾讯[实习]科恩实验室-安全工程师 一、网络与协议 1. TCP三次握手 2. SYN扫描原理 3. HTTPS证书机制 二…...
破解路内监管盲区:免布线低位视频桩重塑停车管理新标准
城市路内停车管理常因行道树遮挡、高位设备盲区等问题,导致车牌识别率低、逃费率高,传统模式在复杂路段束手无策。免布线低位视频桩凭借超低视角部署与智能算法,正成为破局关键。该设备安装于车位侧方0.5-0.7米高度,直接规避树枝遮…...
零知开源——STM32F103RBT6驱动 ICM20948 九轴传感器及 vofa + 上位机可视化教程
STM32F1 本教程使用零知标准板(STM32F103RBT6)通过I2C驱动ICM20948九轴传感器,实现姿态解算,并通过串口将数据实时发送至VOFA上位机进行3D可视化。代码基于开源库修改优化,适合嵌入式及物联网开发者。在基础驱动上新增…...
