vue3.2 + vxe-table4.x 实现多层级结构的 合并、 展开、收起 功能

<template><div style="padding: 20px"><vxe-table border :data="list" :height="800" :span-method="rowspanMethod"><vxe-column title="一级类目" field="category1"><template #default="{ row }"><a-space><span>{{ row.category1 }}</span><a-button @click="toggleCategoryExpand(row, '1')">{{ showText(row, '1') ? '收起' : '展开' }}</a-button></a-space></template></vxe-column><vxe-column title="二级类目" field="category2"><template #default="{ row }"><a-space><span>{{ row.category2 }}</span><a-buttonv-if="showText(row, '1')"@click="toggleCategoryExpand(row, '2')">{{ showText(row, '2') ? '收起' : '展开' }}</a-button></a-space></template></vxe-column><vxe-column title="三级类目" field="category3"></vxe-column><vxe-column title="报告金额" field="amount"></vxe-column><vxe-column title="合计" field="total"></vxe-column></vxe-table></div>
</template><script>
import { defineComponent, ref } from 'vue'
import XEUtils from 'xe-utils'export default defineComponent({setup() {const flattenedData = [{id: '0',category: '人工成本',parentId: null},{id: '0-0',parentId: '0',category: '人工成本0-0'},{id: '0-0-0',parentId: '0-0',category: '人工成本0-0-0'},{id: '0-0-1',parentId: '0-0',category: '人工成本0-0-1'},{id: '0-1',parentId: '0',category: '人工成本0-1'},{id: '0-1-0',parentId: '0-1',category: '人工成本0-1-0'},{id: '0-1-1',parentId: '0-1',category: '人工成本0-1-1'}]const treeData = XEUtils.toArrayTree(flattenedData)const treeDataTemp = JSON.parse(JSON.stringify(treeData))const renderTreeData = ref(treeDataTemp)const list = ref([])// 工具方法const toColTreeData = (treeData) => {const options = { children: 'children' }const list = []const keyMap = {}XEUtils.eachTree(treeData,(item, index, result, paths, parent) => {keyMap[item.id] = itemitem.keys = parent ? parent.keys.concat([item.id]) : [item.id]if (!item.children || !item.children.length) {const row = {}item.keys.forEach((key, index) => {const level = index + 1const obj = keyMap[key]row[`category${level}`] = obj.categoryrow[`id${level}`] = obj.id})list.push(row)}},options)return list}const rowspanMethod = ({ row, _rowIndex, column, visibleData }) => {const fields = ['category1', 'category2']const cellValue = row[column.field]if (cellValue && fields.includes(column.field)) {const prevRow = visibleData[_rowIndex - 1]let nextRow = visibleData[_rowIndex + 1]if (prevRow && prevRow[column.field] === cellValue) {return { rowspan: 0, colspan: 0 }} else {let countRowspan = 1while (nextRow && nextRow[column.field] === cellValue) {nextRow = visibleData[++countRowspan + _rowIndex]}if (countRowspan > 1) {return { rowspan: countRowspan, colspan: 1 }}}}}renderTreeData.value.forEach((item) => {if (item.children.length) {item.children = []}})list.value = toColTreeData(renderTreeData.value)const toggleCategoryExpand = (row, level) => {if (level === '1') {toggleLevelOne(row)}if (level === '2') {toggleLevelTwo(row)}}const toggleLevelOne = (row) => {const { id1 } = rowconst item = renderTreeData.value.find((item) => item.id === id1)if (item.children.length) {item.children = []} else {const a = treeData.find((item) => item.id === id1)item.children = JSON.parse(JSON.stringify(a.children))item.children.forEach((child) => {child.children = []})}list.value = toColTreeData(renderTreeData.value)}const toggleLevelTwo = (row) => {const { id1, id2 } = rowconst item1 = renderTreeData.value.find((item) => item.id === id1)const item2 = item1.children.find((item) => item.id === id2)if (item2.children.length) {item2.children = []} else {const a1 = treeData.find((item) => item.id === id1)const a2 = a1.children.find((item) => item.id === id2)item2.children = a2.children}list.value = toColTreeData(renderTreeData.value)}const showText = (row, level) => {const { id1, id2 } = rowif (level === '1') {const item = renderTreeData.value.find((item) => item.id === id1)if (item) {return item.children.length}return false}if (level === '2') {const item1 = renderTreeData.value.find((item) => item.id === id1)const item2 = item1.children.find((item) => item.id === id2)if (item2) {return item2.children.length}return false}}return {rowspanMethod,toggleCategoryExpand,list,showText}}
})
</script>

在这里插入图片描述

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

如若内容造成侵权/违法违规/事实不符,请联系长河编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

C++ Primer 成员访问运算符

欢迎阅读我的 【CPrimer】专栏 专栏简介&#xff1a;本专栏主要面向C初学者&#xff0c;解释C的一些基本概念和基础语言特性&#xff0c;涉及C标准库的用法&#xff0c;面向对象特性&#xff0c;泛型特性高级用法。通过使用标准库中定义的抽象设施&#xff0c;使你更加适应高级…

Linux:Shell环境变量与命令行参数

目录 Shell的变量功能 什么是变量 变数的可变性与方便性 影响bash环境操作的变量 脚本程序设计&#xff08;shell script&#xff09;的好帮手 变量的使用&#xff1a;echo 变量的使用&#xff1a;HOME 环境变量相关命令 获取环境变量 环境变量和本地变量 命令行…

ollama和open-webui部署ds

博客地址&#xff1a; ollama和open-webui部署ds 引言 最近&#xff0c;deepseek是越来越火&#xff0c;我也趁着这个机会做了下私有化部署&#xff0c;我这边使用的ollama和 open-webui实现的web版本 ollama 简介 Ollama 是一个开源的工具&#xff0c;专门用于简化机器学…

SpringBoot接口自动化测试实战:从OpenAPI到压力测试全解析

引言&#xff1a;接口测试的必要性 在微服务架构盛行的今天&#xff0c;SpringBoot项目的接口质量直接影响着系统稳定性。本文将分享如何通过自动化工具链实现接口的功能验证与性能压测&#xff0c;使用OpenAPI规范打通测试全流程&#xff0c;让您的接口质量保障体系更加完备。…

Spring Boot 项目开发流程全解析

目录 引言 一、开发环境准备 二、创建项目 三、项目结构 四、开发业务逻辑 1.创建实体类&#xff1a; 2.创建数据访问层&#xff08;DAO&#xff09;&#xff1a; 3.创建服务层&#xff08;Service&#xff09;&#xff1a; 4.创建控制器层&#xff08;Controller&…

RabbitMQ 集群部署方案

RabbitMQ 一、安装 RabbitMQ 二、更改配置文件 三、配置集群 四、测试 环境准备&#xff1a;三台服务器&#xff0c;系统是 CentOS7 IP地址分别是&#xff1a; rabbitmq1&#xff1a;192.168.152.71rabbitmq2&#xff1a;192.168.152.72rabbitmq3&#xff1a;192.168.152.…

SocketTool、串口调试助手、MQTT中间件基础

目录 一、SocketTool 二、串口通信 三、MQTT中间件 一、SocketTool 1、TCP 通信测试&#xff1a; 1&#xff09;创建 TCP Server 2&#xff09;创建 TCP Client 连接 Socket 4&#xff09;数据收发 在TCP Server发送数据12345 在 TCP Client 端的 Socket 即可收到数据12…

LSTM长短期记忆网络-原理分析

1 简介 概念 LSTM&#xff08;Long Short-Term Memory&#xff09;也称为长短期记忆网络&#xff0c;是一种改进的循环神经网络&#xff08;RNN&#xff09;&#xff0c;专门设计用于解决传统RNN的梯度消失问题和长程依赖问题。LSTM通过引入门机制和细胞状态&#xff0c;能够更…

一文了解:部署 Deepseek 各版本的硬件要求

很多朋友在咨询关于 DeepSeek 模型部署所需硬件资源的需求&#xff0c;最近自己实践了一部分&#xff0c;部分信息是通过各渠道收集整理&#xff0c;so 仅供参考。 言归正转&#xff0c;大家都知道&#xff0c;DeepSeek 模型的性能在很大程度上取决于它运行的硬件。我们先看一下…

IP-----动态路由OSPF

这只是IP的其中一块内容&#xff0c;IP还有更多内容可以查看IP专栏&#xff0c;前一章内容为GRE和MGRE &#xff0c;可通过以下路径查看IP-------GRE和MGRE-CSDN博客,欢迎指正 注意&#xff01;&#xff01;&#xff01;本部分内容较多所以分成了两部分在下一章 5.动态路由OS…

ClkLog里程碑:荣获2024上海开源技术应用创新竞赛三等奖

2024年10月&#xff0c;ClkLog团队参加了由上海计算机软件技术开发中心、上海开源信息技术协会联合承办的2024上海数智融合“智慧工匠”选树、“领军先锋”评选活动——开源技术应用创新竞赛。我们不仅成功晋级决赛&#xff0c;还荣获了三等奖&#xff01;这一成就不仅是对ClkL…

计算机毕业设计Python+DeepSeek-R1大模型考研院校推荐系统 考研分数线预测 考研推荐系统 考研(源码+文档+PPT+讲解)

温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 温馨提示&#xff1a;文末有 CSDN 平台官方提供的学长联系方式的名片&#xff01; 作者简介&#xff1a;Java领…

NFC拉起微信小程序申请URL scheme 汇总

NFC拉起微信小程序&#xff0c;需要在微信小程序开发里边申请 URL scheme &#xff0c;审核通过后才可以使用NFC标签碰一碰拉起微信小程序 有不少人被难住了&#xff0c;从微信小程序开发社区汇总了以下信息&#xff0c;供大家参考 第一&#xff0c;NFC标签打开小程序 https://…

DeepSeek推出DeepEP:首个开源EP通信库,让MoE模型训练与推理起飞!

今天&#xff0c;DeepSeek 在继 FlashMLA 之后&#xff0c;推出了第二个 OpenSourceWeek 开源项目——DeepEP。 作为首个专为MoE&#xff08;Mixture-of-Experts&#xff09;训练与推理设计的开源 EP 通信库&#xff0c;DeepEP 在EP&#xff08;Expert Parallelism&#xff09…

【数据结构】 最大最小堆实现优先队列 python

堆的定义 堆&#xff08;Heap&#xff09;是一种特殊的完全二叉树结构&#xff0c;通常分为最大堆和最小堆两种类型。 在最大堆中&#xff0c;父节点的值总是大于或等于其子节点的值&#xff1b; 而在最小堆中&#xff0c;父节点的值总是小于或等于其子节点的值。 堆常用于实…

重新审视 ChatGPT 和 Elasticsearch:第 2 部分 - UI 保持不变

作者&#xff1a;来自 Elastic Jeff Vestal 本博客在第 1 部分的基础上进行了扩展&#xff0c;介绍了基于 RAG 的搜索系统的功能齐全的 Web UI。最后&#xff0c;你将拥有一个将检索、搜索和生成过程结合在一起的工作界面&#xff0c;同时使事情易于调整和探索。 不想读完整个内…

【开源】低代码 C++程序框架,Linux多线程程序

大家好&#xff0c;欢迎来到停止重构的频道。 本期介绍我们新的C低代码框架&#xff1a;Bees&#xff0c;用于编写Linux/Unix的多线程程序。 低代码框架一般是不会对C程序下手的&#xff0c;因为C程序一般是比较复杂的程序&#xff0c;光是多线程同步就够头疼的了。 但是我们…

数据库的sql语句

本篇文章主要用来收集项目开发中&#xff0c;遇到的各种sql语句的编写。 1、根据user表的role_id字段&#xff0c;查询role表。 sql语句&#xff1a;使用JOIN连接两个表 SELECT u.*,r.rolename FROM user u JOIN role r ON u.role_id r.id WHERE u.id 1; 查询结果&#xff1a…

从零开始用react + tailwindcs + express + mongodb实现一个聊天程序(二)

1.安装mogondb数据库 参考MongoDB安装配置教程&#xff08;详细版&#xff09;_mongodb安装详细步骤-CSDN博客 安装mondbcompass数据库连接工具 参考https://www.mongodb.com/zh-cn/docs/compass/current/connect/ 2.后端服务 1.创建src文件夹 并在src文件夹下创建 index…