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实现的校园周边美食探索及分享平台
🥂(❁◡❁)您的点赞👍➕评论📝➕收藏⭐是作者创作的最大动力🤞 💖📕🎉🔥 支持我:点赞👍收藏⭐️留言📝欢迎留言讨论 🔥🔥&…...
基于大模型的 UI 自动化系统
基于大模型的 UI 自动化系统 下面是一个完整的 Python 系统,利用大模型实现智能 UI 自动化,结合计算机视觉和自然语言处理技术,实现"看屏操作"的能力。 系统架构设计 #mermaid-svg-2gn2GRvh5WCP2ktF {font-family:"trebuchet ms",verdana,arial,sans-…...

遍历 Map 类型集合的方法汇总
1 方法一 先用方法 keySet() 获取集合中的所有键。再通过 gey(key) 方法用对应键获取值 import java.util.HashMap; import java.util.Set;public class Test {public static void main(String[] args) {HashMap hashMap new HashMap();hashMap.put("语文",99);has…...

安宝特方案丨XRSOP人员作业标准化管理平台:AR智慧点检验收套件
在选煤厂、化工厂、钢铁厂等过程生产型企业,其生产设备的运行效率和非计划停机对工业制造效益有较大影响。 随着企业自动化和智能化建设的推进,需提前预防假检、错检、漏检,推动智慧生产运维系统数据的流动和现场赋能应用。同时,…...

蓝牙 BLE 扫描面试题大全(2):进阶面试题与实战演练
前文覆盖了 BLE 扫描的基础概念与经典问题蓝牙 BLE 扫描面试题大全(1):从基础到实战的深度解析-CSDN博客,但实际面试中,企业更关注候选人对复杂场景的应对能力(如多设备并发扫描、低功耗与高发现率的平衡)和前沿技术的…...
C++.OpenGL (10/64)基础光照(Basic Lighting)
基础光照(Basic Lighting) 冯氏光照模型(Phong Lighting Model) #mermaid-svg-GLdskXwWINxNGHso {font-family:"trebuchet ms",verdana,arial,sans-serif;font-size:16px;fill:#333;}#mermaid-svg-GLdskXwWINxNGHso .error-icon{fill:#552222;}#mermaid-svg-GLd…...
安卓基础(aar)
重新设置java21的环境,临时设置 $env:JAVA_HOME "D:\Android Studio\jbr" 查看当前环境变量 JAVA_HOME 的值 echo $env:JAVA_HOME 构建ARR文件 ./gradlew :private-lib:assembleRelease 目录是这样的: MyApp/ ├── app/ …...

回溯算法学习
一、电话号码的字母组合 import java.util.ArrayList; import java.util.List;import javax.management.loading.PrivateClassLoader;public class letterCombinations {private static final String[] KEYPAD {"", //0"", //1"abc", //2"…...

【JVM面试篇】高频八股汇总——类加载和类加载器
目录 1. 讲一下类加载过程? 2. Java创建对象的过程? 3. 对象的生命周期? 4. 类加载器有哪些? 5. 双亲委派模型的作用(好处)? 6. 讲一下类的加载和双亲委派原则? 7. 双亲委派模…...
在 Spring Boot 项目里,MYSQL中json类型字段使用
前言: 因为程序特殊需求导致,需要mysql数据库存储json类型数据,因此记录一下使用流程 1.java实体中新增字段 private List<User> users 2.增加mybatis-plus注解 TableField(typeHandler FastjsonTypeHandler.class) private Lis…...
【Elasticsearch】Elasticsearch 在大数据生态圈的地位 实践经验
Elasticsearch 在大数据生态圈的地位 & 实践经验 1.Elasticsearch 的优势1.1 Elasticsearch 解决的核心问题1.1.1 传统方案的短板1.1.2 Elasticsearch 的解决方案 1.2 与大数据组件的对比优势1.3 关键优势技术支撑1.4 Elasticsearch 的竞品1.4.1 全文搜索领域1.4.2 日志分析…...