当前位置: 首页 > article >正文

cocos3.X的oops框架oops-plugin-excel-to-json改进兼容多表单导出功能

        在使用oops框架的过程中,它的导出数据并生成数据结构的插件oops-plugin-excel-to-json有些小的坑点,为满足我个人习惯,对此部分进行了一个小的修改,有需要的拿去用,记录下供大家参考;

一、配置:其他基本配置请自行搜索,首先能导出其例子中的xlsx表格,基于此来看这篇文章。

我的项目环境配置,如下图:

  • 表的配置小坑:

核心文件

extensions\oops-plugin-excel-to-json\dist\ExcelToJson.js

extensions\oops-plugin-excel-to-json\src\ExcelToJson.ts

就是表格的关键字必须在表格的名字中标明:

"【KEY】"

否则只能导出结构,不能导出数据,结构的主键还不对;所以根据项目自己增加的表格,必须增加这个关键字,才能正确导出结构和数据;

  • 一个表个内的多个表单同时导出:

1,修改调用处的输出文件的绝对文件名,为输出路径,这里的输出是项目配置中的输出路径+原表格名称;现在不需要,只要路径,名称由内部的表单决定;

表单名决定数据文件名json和数据结构名ts;

核心是将获得表单的数量,然后循环处理下即可:

整体源码如下:复制过去覆盖,重新开启编辑器即可。

"use strict";
var __importDefault = (this && this.__importDefault) || function (mod) {return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
exports.run = run;
const path_1 = __importDefault(require("path"));
const JsonToTs_1 = require("./JsonToTs");
const main_1 = require("./main");
const fs = require('fs');
const excel = require('exceljs');
/*** Excel转Json数据* @param {*} src           读取的excel文件目录* @param {*} dst           导出的json文件目录* @param {*} name          excel文件名* @param {*} isClient      是否为客户端数据*/
async function convert(src, dst, name, isClient) {console.warn("src = ", src, "  dst = ", dst, "  name = ", name);const workbook = new excel.Workbook();// 读取excelawait workbook.xlsx.readFile(src);console.warn("本次 xlsx的 文件路径 : src = ", src, " 包含>>>  ", workbook.worksheets.length , " <<<< 个分表 sheet", "  workbook.worksheets = ", workbook.worksheets);for(let sheet_id = 1; sheet_id <= workbook.worksheets.length; sheet_id++){let r = {};let names = []; // 文名字段名let keys = []; // 字段名let types = []; // 通用字段数据类型let types_client = {}; // 客户端数据类型let servers = []; // 是否输出服务器字段数据let clients = []; // 是否输出客户端字段数据let primary = []; // 多主键配置let primary_index = [];const worksheet = workbook.getWorksheet(sheet_id); // 获取第一个worksheet console.log("src = ", src, " tablename = ", worksheet.name);worksheet.eachRow((row, rowNumber) => {let data = {};row.eachCell((cell, colNumber) => {const value = cell.text;// console.warn(cell.text, cell.string, cell.number, cell.result, cell.formula)if (rowNumber === 1) { // 字段中文名names.push(value);if (value.indexOf("【KEY】") > -1)primary_index.push(colNumber);}else if (rowNumber === 2) { // 字段英文名keys.push(value);if (primary_index.indexOf(colNumber) > -1)primary.push(value);}else if (rowNumber === 3) { // 通用字段数据类型types.push(value);}else if (isClient == false && rowNumber === 4) { // 是否输出服务器字段数据servers.push(value);}else if (isClient == true && rowNumber === 5) { // 客户端数据类型 clients.push(value);}else if (rowNumber > 5) {let index = colNumber - 1;let type = types[index];let server = servers[index];let client = clients[index];// 验证是否输出这个字段let isWrite = isClient && client === "client" || isClient == false && server === "server";if (isWrite) {let key = keys[index];switch (type) {case "int":// console.warn(`${index}int`, key, value, cell.string, cell.number, cell.result)if (cell.formula) {data[key] = parseInt(cell.result);}else {data[key] = parseInt(value);}types_client[key] = {en: "number",zh: names[index]};break;case "float":// console.warn(`${index}int`, key, value, cell.string, cell.number, cell.result)if (cell.formula) {data[key] = parseFloat(cell.result);}else {data[key] = parseFloat(value);}types_client[key] = {en: "number",zh: names[index]};break;case "string":// console.warn(`${index}int`, key, value, cell.string, cell.number, cell.result)data[key] = value;types_client[key] = {en: "string",zh: names[index]};break;case "any":// console.warn(`${index}int`, key, value, cell.string, cell.number, cell.result)try {data[key] = JSON.parse(value);types_client[key] = {en: "any",zh: names[index]};}catch (_a) {console.log('Cell ' + cell.address + ' has value ' + cell.text);console.warn(`文件【${src}】的【${key}】字段【${data[key]}】类型数据【${value}】JSON转字段串错误【${client}】`);}break;}}}});// 生成数据(多主键)if (rowNumber > 5) {let temp = null;for (var i = 0; i < primary.length; i++) {let k = primary[i];let id = data[k];delete data[k]; // 主键数据删除if (primary.length == 1) {r[id] = data;}else {if (i == primary.length - 1) {temp[id] = data;}else if (i == 0) {if (r[id] == undefined) {r[id] = {};}temp = r[id];}else {temp[id] = {};temp = temp[id];}}}}});// 写入流if (r["undefined"] == null) {await fs.writeFileSync(dst+ worksheet.name+ ".json", JSON.stringify(r));// 生成客户端脚本if (isClient) {(0, JsonToTs_1.createTsClient)( worksheet.name, types_client, r, primary);}else {(0, JsonToTs_1.createTsServer)( worksheet.name, types_client, r, primary);}console.log(isClient ? "客户端数据" : "服务器数据", "生成成功", dst);}else {console.log(isClient ? "客户端数据" : "服务器数据", "无数据2", dst);}}}
function run() {var inputExcelPath = path_1.default.join(__dirname, main_1.config.PathExcel.replace("project://", "../../../") + "/");var outJsonPathClient = path_1.default.join(__dirname, main_1.config.PathJsonClient.replace("project://", "../../../") + "/");var outJsonPathServer = null;if (main_1.config.PathJsonServer != null && main_1.config.PathJsonServer.length > 0) {outJsonPathServer = path_1.default.join(__dirname, main_1.config.PathJsonServer.replace("project://", "../../../") + "/");}const files = fs.readdirSync(inputExcelPath);files.forEach((f) => {let name = f.substring(0, f.indexOf("."));let ext = f.toString().substring(f.lastIndexOf(".") + 1);if (ext == "xlsx") {if (outJsonPathServer)convert(inputExcelPath + f, outJsonPathServer , name, false); // 服务器数据convert(inputExcelPath + f, outJsonPathClient, name, true); // 客户端数据}});
}

最后说明下:这里面还有一个不完善的地方就是:关于excel表中的分表的编号问题:

就是说必须是连续的表单顺序,如果不连续就会有报错,要新建一张表,把各个分表拷贝过去,保证它的表单ID顺序是连续的。也就是说策划可以改分表,但是轻易不要删除分表,重新建一张分表。如果必须要删除分表,要重新做一个新表文件,把分表逐一拷贝一份进去即可。保证分表顺序是从1开始连续的即可。

祝各位用餐快乐!

相关文章:

cocos3.X的oops框架oops-plugin-excel-to-json改进兼容多表单导出功能

在使用oops框架的过程中&#xff0c;它的导出数据并生成数据结构的插件oops-plugin-excel-to-json有些小的坑点&#xff0c;为满足我个人习惯&#xff0c;对此部分进行了一个小的修改&#xff0c;有需要的拿去用&#xff0c;记录下供大家参考&#xff1b; 一、配置&#xff1a;…...

Spring Boot + OpenAI 构建基于RAG的智能问答系统

一、技术架构设计 1.1 系统架构图 [前端]│▼ (HTTP/REST) [Spring Boot Controller]│▼ (Service Call) [问答处理服务层]├─▶ [知识库检索模块] ──▶ [向量数据库]└─▶ [OpenAI集成模块] ──▶ [OpenAI API]│▼ [结果组装与返回] 1.2 技术选型 组件技术栈版本要求…...

开源量子模拟引擎:Quantum ESPRESSO本地部署教程,第一性原理计算轻松入门!

一、介绍 Quantum ESPRESSO 是一个用于电子结构计算和纳米尺度材料建模的开源计算机代码集成套件&#xff0c;专门用于进行第一性原理&#xff08;第一性原理&#xff09;计算&#xff0c;涵盖了电子结构、晶体学和材料性能的模拟。 Quantum ESPRESSO GPU 版本支持GPU加速&am…...

算法blog合集

https://zhuanlan.zhihu.com/p/600245782 https://zhuanlan.zhihu.com/p/696212679 https://zhuanlan.zhihu.com/p/291406172 【推荐系统】DSSM双塔召回2_pair-wise训练和推理-CSDN博客 精通推荐算法1&#xff1a;为什么需要推荐系统&#xff08;系列文章&#xff0c;建议收…...

每日八股文6.3

每日八股-6.3 Mysql1.COUNT 作用于主键列和非主键列时&#xff0c;结果会有不同吗&#xff1f;2.MySQL 中的内连接&#xff08;INNER JOIN&#xff09;和外连接&#xff08;OUTER JOIN&#xff09;有什么主要的区别&#xff1f;3.能详细描述一下 MySQL 执行一条查询 SQL 语句的…...

Kubernetes (k8s)版本发布情况

Kubernetes (k8s)版本发布情况 代码放在 GitHub - kubernetes/kubernetes: Production-Grade Container Scheduling and Management https://github.com/kubernetes/kubernetes/releases 文档放在 kubernetes.io各个版本变更等: https://github.com/kubernetes/kubernet…...

QT 5.9.2+VTK8.0实现等高线绘制

项目下载链接&#xff1a;QT5.9.2VTK8.0实现等高线绘制资源-CSDN文库 示例如下&#xff1a; 主要代码如下&#xff1a; #include "vtkRenderer.h" #include "vtkRenderWindow.h" #include "vtkRenderWindowInteractor.h" #include "vtkPo…...

CppCon 2015 学习:3D Face Tracking and Reconstruction using Modern C++

1. 3D面部追踪和重建是什么&#xff1f; 3D面部追踪&#xff08;3D Face Tracking&#xff09;&#xff1a; 实时检测并追踪人脸在三维空间中的位置和姿态&#xff08;如转头、点头、表情变化等&#xff09;&#xff0c;通常基于摄像头捕获的视频帧。3D面部重建&#xff08;3D…...

Three.js进阶之音频处理与展示

引擎在对音频处理提供了丰富的接口&#xff0c;本文展示两个音频处理示例。 一、声音可视化 Three.js中的声音可视化是以视觉为核心&#xff0c;以音乐为载体&#xff0c;为音乐提供直观的视觉呈现。通过对音乐数据的分析并结合开发需求&#xff0c;能实现酷炫的视觉效果。在…...

4.2 HarmonyOS NEXT分布式AI应用实践:联邦学习、跨设备协作与个性化推荐实战

HarmonyOS NEXT分布式AI应用实践&#xff1a;联邦学习、跨设备协作与个性化推荐实战 在HarmonyOS NEXT的全场景分布式架构下&#xff0c;AI能力突破设备边界&#xff0c;通过联邦学习保护数据隐私、跨设备任务协作释放算力潜能、个性化推荐实现服务主动化。本文结合华为分布式…...

兼容老设备!EtherNet/IP转DeviceNet网关解决储能产线通讯难题

在新能源行业飞速发展的当下&#xff0c;工业自动化水平的高低直接影响着企业的生产效率与产品质量。JH-EIP-DVN疆鸿智能ETHERNET/IP和DEVICENET作为工业领域常用的通信协议&#xff0c;它们之间的转换应用在新能源生产线上发挥着关键作用。本文重点探讨ETHERNETIP从站转DEVICE…...

健康检查:在 .NET 微服务模板中优雅配置 Health Checks

&#x1f680; 健康检查&#xff1a;在 .NET 微服务模板中优雅配置 Health Checks &#x1f4da; 目录 &#x1f680; 健康检查&#xff1a;在 .NET 微服务模板中优雅配置 Health Checks一、背景与意义 &#x1f50d;二、核心配置 &#x1f527;2.1 引入必要的 NuGet 依赖 &…...

【Pytorch学习笔记】模型模块08——AlexNet模型详解

AlexNet模型详解&#xff1a;结构、算法与PyTorch实现 一、AlexNet模型结构 AlexNet是2012年ImageNet竞赛冠军模型&#xff0c;由Alex Krizhevsky等人提出&#xff0c;标志着深度学习在计算机视觉领域的突破。 网络结构&#xff08;5卷积层 3全连接层&#xff09;&#xff…...

LabVIEW自感现象远程实验平台

LabVIEW开发自感现象远程实验平台&#xff0c;通过整合 NI数据采集设备、菲尼克斯&#xff08;Phoenix Contact&#xff09;继电器模块及罗技&#xff08;Logitech&#xff09;高清摄像头&#xff0c;实现远程数据采集、仪器控制与实时监控三大核心功能。平台突破传统实验装置局…...

AppTrace 视角下 App 一键拉起:提升应用转化率的高效方案​

官网地址&#xff1a;AppTrace - 专业的移动应用推广追踪平台 在大规模开展 App 推广、用户召回、广告投放、邀请传播等活动时&#xff0c;高效的深度链接方案至关重要。它不仅能缩短用户路径&#xff0c;带来无缝、流畅的跳转体验&#xff0c;更核心的是通过参数传递打通 web…...

梯度下降:机器学习优化的核心算法

梯度下降算法原理及其在机器学习中的实践应用 引言 在机器学习领域,优化算法扮演着核心角色。其中梯度下降法作为最基础的优化方法,为神经网络、支持向量机等模型提供了参数优化解决方案。本文将深入解析梯度下降的数学原理,探讨其多种变体实现,并通过Python代码演示具体…...

Vue-6-前端框架Vue之基于Plotly.js绘制曲线

文章目录 1 安装Plotly.js2 折线图2.1 创建一个Vue组件来绘制图表2.1.1 Vue模板部分template2.1.2 Vue脚本部分script2.1.3 Vue样式部分style2.2 使用这个组件APP.vue3 动态更新图表3.1 创建一个Vue组件来绘制图表3.1.1 Vue模板部分template3.1.2 Vue脚本部分script3.1.3 Vue样…...

Python----目标检测(《YOLOv3:AnIncrementalImprovement》和YOLO-V3的原理与网络结构)

一、《YOLOv3:AnIncrementalImprovement》 1.1、基本信息 标题&#xff1a;YOLOv3: An Incremental Improvement 作者&#xff1a;Joseph Redmon, Ali Farhadi 机构&#xff1a;华盛顿大学&#xff08;University of Washington&#xff09; 发表时间&#xff1a;2018年 代…...

Redux:不可变数据与纯函数的艺术

Redux&#xff1a;不可变数据与纯函数的艺术 状态管理的困境 随着现代 Web 应用功能的不断扩展&#xff0c;前端开发者面临着日益复杂的状态管理挑战。当应用从简单的表单交互发展到复杂的单页应用时&#xff0c;组件间共享状态的问题变得尤为突出。想象一个电商平台&#xf…...

算法篇 八大排序(冒泡 插入 选择 堆 希尔 快排 归并 计数)

目录 引言 1.冒泡排序 思路 代码实现 2.选择排序 思路 代码实现&#xff08;存在易错点&#xff09; 3.插入排序 思路 代码实现 4.希尔排序 思路 代码实现 5.堆排序 思路 代码实现 6.快速排序&#xff08;快排&#xff09; 一.三路划分 思路 代码实现 二.自…...

技术文档写作全攻略

一、引言 在快速迭代的软件开发中&#xff0c;技术文档早已不只是附属品&#xff0c;而是与代码同等重要的交付物&#xff1a; 帮助新成员 T0 → T1 学习曲线指数下降&#xff1b;降低支持成本&#xff0c;将重复性问答前移到自助文档&#xff1b;为合规审计、知识传承及商业…...

网络安全全景解析

引言 在数字化时代&#xff0c;网络已深度融入社会生产生活的各个领域&#xff0c;成为推动经济发展和社会进步的关键力量。然而&#xff0c;随着网络应用的日益复杂&#xff0c;网络安全问题也呈现出多样化、复杂化的趋势。从个人隐私泄露到企业核心数据被盗&#xff0c;从基础…...

音视频之视频压缩编码的基本原理

系列文章&#xff1a; 1、音视频之视频压缩技术及数字视频综述 2、音视频之视频压缩编码的基本原理 一、预测编码&#xff1a; 1、预测编码的基本概念&#xff1a; 预测法是最简单、实用的视频压缩编码方法&#xff0c;经过压缩编码后传输的并不是像素本身的取样值&#xff0…...

IDEA 包分层显示设置

方法一&#xff08;用的IntelliJ IDEA 2024.1.4版本&#xff09;&#xff1a; 找到项目视图设置入口&#xff1a;在左侧Project&#xff08;项目&#xff09;面板的顶部&#xff0c;有个三个点...的按钮 &#xff0c;点击它。 进入树形外观配置&#xff1a;在弹出的菜单中&…...

书籍将正方形矩阵顺时针转动90°(8)0605

题目 给定一个N x N的矩阵matrix,把这个矩阵调整成顺时针转动90后的形式。 1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 顺时针转动90后为&#xff1a; 13 9 5 1 14 …...

【docker】容器技术如何改变软件开发与部署格局

在当今数字化时代&#xff0c;软件开发与部署的效率和灵活性至关重要。就像古人云&#xff1a;“工欲善其事&#xff0c;必先利其器。”Docker 作为一款强大的容器技术&#xff0c;正如同软件开发领域的一把利器&#xff0c;极大地改变了应用的开发、交付和运行方式。本文将深入…...

C#抽象类深度解析 _ 核心特性与实战指南

—— 面向对象设计的基石 &#x1f50d;抽象类核心定义 abstract class AbClass { ... } // abstract修饰符声明 不可实例化&#xff1a;new AbClass() 将触发编译错误继承专用&#xff1a;仅能作为其他类的基类存在混合成员组合&#xff1a;可同时包含抽象方法和已实现方法…...

时序数据库IoTDB的UDF Sample算法在数据监控、故障预防的应用

一、数据监控在工业物联网中的重要性 设备数据监控是工业物联网&#xff08;IoT&#xff09;中最为广泛应用的领域之一。通过实时监控工厂机械设备的运行状态&#xff0c;企业能够提前发现设备的潜在故障&#xff0c;从而实现预防性维护与可预测性维护。这一做法不仅能有效提升…...

Flask-SQLAlchemy使用小结

链表查询 join方法允许你指定两个或多个表之间的连接条件&#xff0c;并返回一个新的查询对象&#xff0c;该对象包含了连接后的结果。 内连接 from sqlalchemy import join # 使用join函数 query db.session.query(User, Order).join(Order, User.id Order.user_id) res…...

深度学习和神经网络 卷积神经网络CNN

1.什么是卷积神经网络 一种前馈神经网络&#xff1b;受生物学感受野的机制提出专门处理网格结构数据的深度学习模型 核心特点&#xff1a;通过卷积操作自动提取空间局部特征&#xff08;如纹理、边缘&#xff09;&#xff0c;显著降低参数量 2.CNN的三个结构特征 局部连接&a…...