738 Commits

Author SHA1 Message Date
sjl
97d1f08bbe 正式检测闪变 2025-10-13 13:56:37 +08:00
sjl
7d0053eb71 被检设备监测点台账线路号占满不可新增 2025-10-11 14:12:15 +08:00
贾同学
666fb22c49 UPDATE: 优化客户端桌面窗口内容自适应 2025-10-10 15:29:54 +08:00
贾同学
b319a89501 UPDATE: 处理控制台警告问题 2025-10-10 13:23:40 +08:00
贾同学
8e0b3be438 Merge remote-tracking branch 'origin/master' 2025-10-10 08:24:55 +08:00
sjl
b0d92de738 微调 2025-10-09 15:51:45 +08:00
sjl
92b3e25989 冲突 2025-10-09 15:49:29 +08:00
sjl
43c75c96a7 录波数据 2025-10-09 15:47:00 +08:00
贾同学
da0aa0cd0f UPDATE: 调整超时时间到1200秒 2025-10-09 14:34:45 +08:00
贾同学
b0c88b9df2 UPDATE: 优化; 2025-09-30 08:42:45 +08:00
caozehui
7c0ec5844a 微调 2025-09-29 14:49:29 +08:00
贾同学
a4a64ef0f9 UPDATE: 优化; 2025-09-29 09:47:53 +08:00
贾同学
58bb25500e UPDATE: 优化; 2025-09-25 14:40:45 +08:00
贾同学
3d73c34343 Merge remote-tracking branch 'origin/master' 2025-09-25 11:17:15 +08:00
贾同学
f74fedc213 UPDATE: 优化; 2025-09-25 11:16:57 +08:00
sjl
8ea49f9609 预检测实时数据对齐失败展示 2025-09-25 10:11:15 +08:00
caozehui
9ee71d29d4 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	frontend/src/views/home/components/table.vue
2025-09-25 08:54:05 +08:00
caozehui
039a67c35a 微调 2025-09-25 08:53:32 +08:00
sjl
e17749d47e Merge branch 'master' of http://192.168.1.125:3000/root/pqs-9100_client 2025-09-25 08:53:28 +08:00
sjl
21c859c8f1 微调 2025-09-25 08:51:40 +08:00
贾同学
4fe239c86f UPDATE: 1、子计划管理,筛选条件改成搜索、设备厂家、是否分配;
2、重复导入子计划时,增量被检设备并删除未检设备;
        3、优化删除子计划后,刷新主计划信息;
2025-09-25 08:49:15 +08:00
ab62e56bbb 微调 2025-09-25 08:39:06 +08:00
5730b9c5cf 比对模式的检测报告生成和下载 2025-09-23 16:14:03 +08:00
贾同学
d4992db198 UPDATE: 优化批量导入提示; 2025-09-23 15:15:20 +08:00
sjl
5ccd1709a5 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-23 11:14:44 +08:00
sjl
b48c1e0d78 微调 2025-09-23 11:14:34 +08:00
贾同学
fcdbbce7a9 UPDATE: 1、测试项改成检测项;
2、检测项勾选闪变添加提示;
        3、检测项由误差体系反推为下拉框让用户多选,默认全选;
        4、修改检测计划有子计划后部分字段不可编辑;
2025-09-23 10:55:07 +08:00
sjl
d08194bfd8 相序校验拦截错误 2025-09-23 08:39:15 +08:00
贾同学
55f579ef64 UPDATE: 优化 2025-09-22 16:16:45 +08:00
sjl
783e1c080b 预检测,正式检测,数据查询能处理只有录波的情况 2025-09-22 15:51:58 +08:00
guanj
44cdb3273c 微调 2025-09-22 15:37:28 +08:00
guanj
dbc21cdbfa 修改通道配对布局 2025-09-22 15:26:13 +08:00
sjl
24d83cfd39 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-22 09:17:17 +08:00
sjl
b213b721bb 微调 2025-09-22 09:17:10 +08:00
caozehui
4ae42408c3 微调 2025-09-22 09:16:35 +08:00
贾同学
a9156f0954 UPDATE: 异步导出检测数据逻辑。 2025-09-19 16:18:10 +08:00
sjl
3e7509cd44 预检测重复初始化拦截 2025-09-19 11:20:06 +08:00
sjl
24becb82e1 比对式误差切换和删除零时表 2025-09-19 09:31:02 +08:00
sjl
6608587edd 预检测处理错误 2025-09-18 14:11:50 +08:00
sjl
5ad8cdecba 重新计算 2025-09-18 10:06:24 +08:00
贾同学
6b4cca1ef7 Merge remote-tracking branch 'origin/master' 2025-09-17 16:28:17 +08:00
贾同学
dea0844829 UPDATE: 根据计划的状态调按钮显示还是隐藏。 2025-09-17 16:27:46 +08:00
sjl
bbd438d23f 修复比对式一键检测 2025-09-17 15:42:05 +08:00
sjl
c88128b63b 1.被检设备监测点新增是否参与检测
2.调整通道配对只显示绑定和参与检测的通道数
2025-09-17 14:08:58 +08:00
sjl
95c68942ed 数据查询树替换接口联调 2025-09-17 08:46:26 +08:00
贾同学
5db685baca UPDATE: 修改计划数据源选择逻辑。 2025-09-16 14:42:02 +08:00
贾同学
fa710efea4 Merge remote-tracking branch 'origin/master' 2025-09-16 14:37:53 +08:00
贾同学
d0c3e1d9bd UPDATE: 修改计划数据源选择逻辑。 2025-09-16 14:37:17 +08:00
caozehui
589ddd38f3 微调 2025-09-16 13:47:40 +08:00
sjl
47d1500296 模拟式报告生成不弹框 2025-09-16 13:34:22 +08:00
sjl
4a8f8bff6a 预检测新增录波 2025-09-15 14:59:54 +08:00
贾同学
2b9b87a3db UPDATE: 布局样式优化。 2025-09-15 14:58:12 +08:00
贾同学
b241128105 Merge remote-tracking branch 'origin/master' 2025-09-15 10:56:13 +08:00
贾同学
226e3271ee UPDATE: 及格改成符合,添加无法比较。 2025-09-15 10:55:38 +08:00
sjl
1c253fd713 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-15 10:43:53 +08:00
sjl
ed81d3d398 比对式报告生成显示检测中设备 2025-09-15 10:43:42 +08:00
贾同学
09b54a29ab ADD: 报告生成选择检测数据源并更新监测点结果。 2025-09-15 10:38:14 +08:00
贾同学
b27615baaf ADD: 1、报告生成选择界面流程。 2025-09-12 16:34:27 +08:00
sjl
c735e7a5bb 数据查询检测次数默认最后一次 2025-09-12 16:09:41 +08:00
sjl
c78f591baf 实时数据过了录波状态处理和进度条 2025-09-12 13:35:29 +08:00
贾同学
cfd8b072dd ADD: 修改数据合并逻辑,新增 event-source-polyfill包,实现长连接通信,展示合并进度 2025-09-11 11:03:14 +08:00
sjl
d18e34d2c9 比对模式没有检测次数,数据操作可查看检测中设备 2025-09-10 14:20:42 +08:00
sjl
53813795db 调整正式检测录波状态 2025-09-10 09:28:04 +08:00
sjl
8c3098e19a Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-09 20:54:32 +08:00
sjl
780a446aed 正式检测录波数据查询 2025-09-09 20:54:22 +08:00
贾同学
375f01a6ab UPDATE: 完善检修计划项目负责人和项目成员逻辑。 2025-09-09 10:25:53 +08:00
sjl
48aab7c1e9 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-09 09:14:52 +08:00
sjl
d7cfe665e2 录波数据查询 2025-09-09 09:14:42 +08:00
贾同学
237c23bb70 UPDATE: 完善检修计划项目负责人和项目成员逻辑。 2025-09-08 16:23:19 +08:00
guanj
4cd6302ee0 修改 通道配对样式 2025-09-08 16:01:04 +08:00
caozehui
6a75709774 微调 2025-09-08 14:55:25 +08:00
贾同学
629dff1256 ADD: 检修计划添加项目负责人和项目成员 2025-09-08 10:21:54 +08:00
贾同学
6d6d03c03c UPDATE: 优化高度 2025-09-05 09:09:18 +08:00
贾同学
6122f53c8e UPDATE: 检测计划选择数据源逻辑改成主计划默认全选,子计划勾选校验 2025-09-05 08:57:39 +08:00
贾同学
5a7eea1052 UPDATE: 优化选择被检设备组件 2025-09-04 19:39:20 +08:00
贾同学
25f3570c18 UPDATE: 替换新增比对检测计划时,可选择的标准设备接口;优化选择被检设备组件 2025-09-04 11:12:29 +08:00
贾同学
74e015bd12 UPDATE: 重构检修计划选择被检设备组件,并处理对应数据逻辑 2025-09-03 20:44:32 +08:00
caozehui
da6a72807b 微调 2025-09-03 08:38:13 +08:00
贾同学
bb7ebaea45 UPDATE: 比对被检设备提交校验监测点信息 2025-09-02 11:03:39 +08:00
sjl
ae51b590af Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-01 14:53:44 +08:00
sjl
2ec3102eff 新建检测计划时,检测项下拉框要和误差体系里配置的要一致 2025-09-01 14:53:30 +08:00
贾同学
f5f7d259a9 ADD: 是否下载检测报告选择 2025-09-01 09:36:32 +08:00
guanj
4364f88526 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-09-01 08:36:25 +08:00
guanj
0f5e21a06c 微调 2025-09-01 08:36:18 +08:00
贾同学
ddbaf5651a UPDATE: 勾选被检设备下载检测结果数据等 2025-08-29 15:09:05 +08:00
贾同学
a847419ab5 Merge remote-tracking branch 'origin/master' 2025-08-29 11:35:05 +08:00
贾同学
d5fb41cbab ADD: 检测计划添加导入标识字段以及逻辑 2025-08-29 11:34:55 +08:00
guanj
25e7b754b7 修改 正式检测结果 2025-08-29 11:12:57 +08:00
贾同学
a32ca3c849 ADD:数据合并按钮操作 2025-08-29 09:54:19 +08:00
caozehui
6e979c5dcb 微调 2025-08-28 16:29:00 +08:00
caozehui
8b578d4d8b 微调 2025-08-28 10:37:50 +08:00
贾同学
52fcdbfe1e ADD:导入子计划元数据二次确认提示 2025-08-27 20:21:55 +08:00
guanj
4559a7b5e2 修改检测查询页面空白问题 2025-08-27 16:32:29 +08:00
guanj
567201563d 修改预检测实时数据详情展示 2025-08-27 14:55:00 +08:00
guanj
772707ac42 修改比对式检测页面 2025-08-27 11:17:13 +08:00
guanj
4a6db824ba 修改通道配置页面 2025-08-26 20:52:24 +08:00
guanj
8b4c22e959 微调 2025-08-26 18:29:14 +08:00
caozehui
d7f1224df4 微调 2025-08-26 15:39:58 +08:00
贾同学
ac4e0e2077 UPDATE:优化子计划增删tab展示数据刷新问题 2025-08-26 15:28:01 +08:00
guanj
56a6f199c0 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-26 15:16:25 +08:00
guanj
0abb765b32 修改查看 模拟式 页面展示 2025-08-26 15:16:18 +08:00
贾同学
4f8fdb83d1 ADD:检测计划添加检测配置相关 2025-08-26 11:13:23 +08:00
caozehui
300b220de2 微调 2025-08-26 10:58:29 +08:00
guanj
825d2cc46a Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-26 10:37:24 +08:00
guanj
95b602e6d4 修改检测数据查询展示页面 2025-08-26 10:37:13 +08:00
贾同学
a7b5bbf0bf Merge remote-tracking branch 'origin/master' 2025-08-25 14:53:06 +08:00
贾同学
dfbba11aae ADD:ICD添加检测是否两个开关字段 2025-08-25 14:39:59 +08:00
guanj
5cf39e8aa8 联调检测结果查询弹框 2025-08-25 11:35:38 +08:00
贾同学
a19a20ddd8 ADD:导入检测计划检测结果; 2025-08-25 11:02:07 +08:00
guanj
0985cc5d7c 修改正式监测返回结果 2025-08-22 16:21:57 +08:00
guanj
2be0be681e Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-22 15:34:07 +08:00
guanj
dd9ca8f956 联调 正式检测结果页面 2025-08-22 15:33:57 +08:00
贾同学
5cd60d9a32 ADD:导出检测计划检测结果; 2025-08-22 09:04:09 +08:00
guanj
959ae1dee9 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-21 13:15:14 +08:00
guanj
d2d1490e9b 修改正式检测推送逻辑 2025-08-21 13:14:59 +08:00
caozehui
7bcd88c3a7 微调 2025-08-21 10:42:09 +08:00
guanj
8e3368bd29 修改正式检测逻辑 2025-08-21 09:33:13 +08:00
guanj
bc03ba88f0 修改样式 2025-08-20 20:05:07 +08:00
guanj
2aee4b281d 联调正式检测 2025-08-20 20:02:22 +08:00
贾同学
26647222e2 UPDATE:优化v-auth指令逻辑 2025-08-20 10:15:09 +08:00
贾同学
a2db45cace ADD:添加按钮权限 2025-08-20 08:51:01 +08:00
caozehui
d761c0449b 微调 2025-08-19 19:16:20 +08:00
caozehui
dc6a346fd4 微调 2025-08-19 09:37:42 +08:00
贾同学
e938c6b3d9 UPDATE:修改编辑计划按钮判断是否为主计划逻辑 2025-08-18 16:28:29 +08:00
caozehui
c9fef2a9d7 微调 2025-08-18 08:30:03 +08:00
sjl
9319dd06c5 首页判断被检设备下绑定监测点 2025-08-15 16:22:58 +08:00
贾同学
7b96ce84fc ADD:添加导出子计划元信息和导入检修计划按钮逻辑 2025-08-15 16:03:42 +08:00
sjl
b105ff890c 标准 2025-08-15 08:43:42 +08:00
sjl
61b87304e6 数模没有标准设备 2025-08-15 08:37:35 +08:00
sjl
83c8dc5f19 调整开始检测接口 2025-08-14 15:02:58 +08:00
贾同学
b1ddf540ca UPDATE:1.完善主计划导入被检设备逻辑;2.完善新增编辑子计划逻辑。 2025-08-13 20:34:08 +08:00
sjl
0025895696 预检测描述 2025-08-13 15:52:48 +08:00
sjl
1ec8cce63e 实时数据对齐,修改成标准对应被检 2025-08-13 10:06:55 +08:00
贾同学
865d52c135 Merge remote-tracking branch 'origin/master' 2025-08-12 20:43:11 +08:00
贾同学
ce8607af36 UPDATE:优化完善新增编辑检测计划绑定被检设备 2025-08-12 20:42:48 +08:00
sjl
4e8a6300dd 预检测 2025-08-12 20:17:37 +08:00
919e81da8b 前端优化调整 2025-08-12 10:26:30 +08:00
caozehui
18cb6dbde8 微调 2025-08-12 08:34:09 +08:00
6d405d16ed 前端优化调整 2025-08-11 16:30:53 +08:00
sjl
77d2176812 接口调整 2025-08-11 15:59:29 +08:00
c85eac3888 前端优化调整 2025-08-11 11:14:20 +08:00
sjl
27d2d82fcd 去除协议校验 2025-08-08 15:27:17 +08:00
sjl
ecbc3c30c8 1 2025-08-08 13:24:19 +08:00
sjl
0a65efd235 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client
# Conflicts:
#	frontend/src/views/home/components/compareTestPopup.vue
#	frontend/src/views/home/components/testPopup.vue
2025-08-08 13:21:42 +08:00
sjl
5cd8fea60c 预见测 2025-08-08 13:18:01 +08:00
caozehui
3d1b4eb7c6 微调 2025-08-07 16:08:36 +08:00
ec1330bdb8 前端优化调整 2025-08-07 15:44:17 +08:00
caozehui
e66bcdb293 正式检测userPageId参数 2025-08-07 15:29:49 +08:00
sjl
88f1876ef0 通道配对 2025-08-07 15:17:04 +08:00
sjl
fdc1fd6fbd Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-07 14:44:04 +08:00
sjl
022d80f30e 实时数据对齐 2025-08-07 14:43:56 +08:00
f59f287b63 前端优化调整 2025-08-07 08:55:28 +08:00
6e573cc597 前端优化调整 2025-08-07 08:47:56 +08:00
d2f92ecde4 前端优化调整 2025-08-06 19:53:22 +08:00
sjl
b2a6a1de4e websocket 2025-08-06 15:28:17 +08:00
sjl
f374df79a6 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client
# Conflicts:
#	frontend/src/utils/webSocketClient.ts
2025-08-06 15:21:17 +08:00
sjl
154eb9f79c 1 2025-08-06 15:19:27 +08:00
sjl
d0724cb7f6 预检测 2025-08-06 15:18:27 +08:00
d6d63523a3 websocket优化 2025-08-06 13:33:09 +08:00
sjl
83998f88ac Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-08-05 10:37:48 +08:00
sjl
9c5e54507b 通道配对,预检测界面 2025-08-05 10:37:40 +08:00
caozehui
15689b5284 Merge branch 'qr_branch'
# Conflicts:
#	frontend/src/views/machine/device/index.vue
2025-07-31 09:47:40 +08:00
caozehui
00893d2d1f 微调 2025-07-31 09:40:12 +08:00
sjl
0079f7415e 检测计划列表编辑 2025-07-29 18:35:46 +08:00
327801d040 出厂检测报告改造 2025-07-24 16:30:19 +08:00
sjl
6e22c01dd8 通道配对 2025-07-24 08:29:03 +08:00
sjl
bccb4b1f17 被检设备监测点 2025-07-22 16:12:08 +08:00
sjl
1f37cc567c 首页隐藏子计划,计划列表子计划设备管理 2025-07-22 14:56:58 +08:00
sjl
f81503091d Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-07-21 13:48:07 +08:00
sjl
e29f25653e 比对检测计划 2025-07-21 13:47:56 +08:00
caozehui
9e8e44b886 二楼检测项为空问题 2025-07-18 14:33:12 +08:00
caozehui
6e10b0c645 二楼检测项为空问题 2025-07-14 15:02:07 +08:00
sjl
c8f3b4eddc tab监测点框高度 2025-07-09 20:58:52 +08:00
sjl
cc848b1ffb 标准设备,比对模式被检设备 2025-07-09 20:12:40 +08:00
sjl
d2f0382bd9 新增场景切换 2025-07-01 14:58:11 +08:00
sjl
9fecf0ce3f 用户管理去除时间查询 2025-07-01 10:47:59 +08:00
sjl
290586d0ff 全局误差小数位4-6位 2025-06-30 10:06:19 +08:00
sjl
ddc45af223 优化计划弹框 2025-06-27 10:57:57 +08:00
sjl
0e6f123306 新增计划初始化 2025-06-27 10:53:56 +08:00
sjl
7b93363b23 通用配置误差保留小数位 2025-06-27 09:01:06 +08:00
caozehui
2fd0dcb8a3 微调 2025-06-27 08:40:27 +08:00
caozehui
86ee05f8af 微调 2025-06-26 14:01:04 +08:00
caozehui
38814b9f44 手动检测根据检测内容不同动态决定是否进行初始化操作 2025-06-26 13:17:11 +08:00
caozehui
fb5d13671d Merge remote-tracking branch 'origin/master' 2025-06-25 09:22:14 +08:00
caozehui
310a769092 微调 2025-06-25 09:21:58 +08:00
sjl
8c63edabdc 楼下场景不用报告模版 2025-06-24 11:03:55 +08:00
caozehui
5363625a2f 微调 2025-06-24 09:22:36 +08:00
efcd6e1cfe 临时提交(自动检测步骤) 2025-06-23 13:19:38 +08:00
acc4d0ca67 被检设备表格增加IP字段 2025-06-20 14:58:28 +08:00
772f38feca 代码提交 2025-06-20 12:11:57 +08:00
sjl
1eb141e559 新增检测计划默认显示,为空判断 2025-06-11 16:01:23 +08:00
caozehui
16b446bf20 微调 2025-06-04 22:15:37 +08:00
sjl
bc0de34c15 预检测不显示温度框 2025-06-04 16:38:02 +08:00
sjl
15f2c1ee41 用户,角色超级管理员不可编辑 2025-05-29 10:49:26 +08:00
sjl
81a944062a Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-05-29 10:15:10 +08:00
sjl
3c19b05f4b 间谐波次数,IP地址 2025-05-29 10:15:03 +08:00
caozehui
c1f53cdc69 微调 2025-05-27 09:07:59 +08:00
sjl
6ef2a6049b 系数校准二次确认,输入框长度限值 2025-05-26 15:36:06 +08:00
caozehui
9989cc98cb 微调 2025-05-26 14:41:06 +08:00
caozehui
e192158deb 微调 2025-05-23 10:15:05 +08:00
caozehui
2e5d551e5d 添加脚本与ICD检验时出现10500错误 2025-05-22 18:43:33 +08:00
caozehui
4622eb36d9 ABC三项可以分开单独展示部分相的数据 2025-05-22 08:45:55 +08:00
caozehui
9c53b7c18e 微调 2025-05-21 15:29:39 +08:00
caozehui
7b3805060f 微调 2025-05-19 09:15:48 +08:00
caozehui
629600bc00 归档前如果设备没有生成报告则自动生成报告 2025-05-16 10:33:43 +08:00
caozehui
8b144b63fc 批量生成报告 2025-05-15 16:08:35 +08:00
caozehui
098ab3a41d 省级平台在正式检测时增加温度、相对湿度参数 2025-05-15 08:57:24 +08:00
caozehui
2a53f577aa 检测计划新增数据处理原则字段、全局配置中移除数据处理原则字段 2025-05-14 08:58:21 +08:00
caozehui
f8b7c224b7 检测计划数据源字段显示bug、脚本大项启用/禁用功能 2025-05-12 15:57:19 +08:00
caozehui
f0b3bdd37c 预检测出现连接超时错误提示 2025-05-12 14:38:25 +08:00
caozehui
bb35eb749b 不合格项复检bug 2025-05-09 08:51:54 +08:00
caozehui
b3750d6a7f 微调 2025-05-06 15:22:26 +08:00
caozehui
33ebb91ab6 微调 2025-05-06 15:14:58 +08:00
sjl
3e00f2fee7 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-05-06 11:00:23 +08:00
sjl
68e9856641 省级平台不展示系数校准 2025-05-06 11:00:12 +08:00
caozehui
51b2e80493 微调 2025-05-06 10:19:41 +08:00
sjl
cf77572f41 设备类型报告模版只在楼下厂家展示 2025-05-06 10:08:57 +08:00
GGJ
fe6524404c 微调 2025-04-29 10:47:25 +08:00
caozehui
ccda7218f4 检测计划数据源字段调整、绑定设备样式调整 2025-04-28 09:18:52 +08:00
caozehui
ddd0e6a22b (间)谐波含有率可以为小数 2025-04-27 16:11:48 +08:00
caozehui
ed85453de1 误差体系切换时-生成报告 2025-04-25 14:32:17 +08:00
caozehui
04939606fc 间谐波电流、间谐波电流复制bug,tooltip位置调整 2025-04-24 16:16:04 +08:00
caozehui
2df455b667 暂态设定幅值最大为180% 2025-04-24 10:18:27 +08:00
caozehui
8fb214d7a7 微调 2025-04-23 15:28:35 +08:00
caozehui
4f2c8a8016 微调 2025-04-23 10:45:46 +08:00
caozehui
c2974dbe0d 微调 2025-04-22 15:52:03 +08:00
caozehui
3891e5459a 误差体系修改 2025-04-22 14:21:24 +08:00
caozehui
9ed47ac5bc 检测结果数据显示问题 2025-04-22 14:12:07 +08:00
caozehui
114fcd140d 微调 2025-04-22 13:49:59 +08:00
caozehui
19a30894f8 微调 2025-04-21 09:11:31 +08:00
caozehui
2134fea518 微调 2025-04-21 09:10:24 +08:00
caozehui
82f495ce3b 调整窗口大小 2025-04-18 13:42:48 +08:00
caozehui
f2ecff54a3 微调 2025-04-18 11:07:04 +08:00
caozehui
f56004ac15 微调 2025-04-17 14:47:32 +08:00
caozehui
5303c47248 微调 2025-04-17 10:21:38 +08:00
GGJ
37731592f7 微调 2025-04-16 10:17:37 +08:00
caozehui
c9e41e0c82 日志导出csv、报告模板编辑框样式调整 2025-04-15 09:45:00 +08:00
caozehui
4a80a88e6e 微调 2025-04-14 19:51:06 +08:00
caozehui
30b219e14d 微调 2025-04-14 14:43:29 +08:00
caozehui
ecec9adeea icd校验 2025-04-14 13:22:46 +08:00
caozehui
1853152138 日志分页查询及展示 2025-04-11 14:34:56 +08:00
GGJ
5619413f37 修改点击使能默认够选通道使能,修改暂态使能持续时间保持一致 2025-04-10 11:11:23 +08:00
caozehui
5b736bc475 微调 2025-04-10 10:50:53 +08:00
caozehui
2aa37c3fe3 微调 2025-04-09 20:42:17 +08:00
caozehui
4e0cb72f32 微调 2025-04-09 20:06:08 +08:00
caozehui
ce74ec4db7 bug修复 2025-04-09 19:30:00 +08:00
caozehui
9938306884 微调 2025-04-09 14:30:00 +08:00
caozehui
ef63fcf807 bug修复 2025-04-08 09:51:35 +08:00
caozehui
08cffa4256 bug修复 2025-04-08 09:43:12 +08:00
caozehui
e1a740f8bf 文件大小提示 2025-03-31 18:54:13 +08:00
caozehui
56d035253b 微调 2025-03-31 13:22:57 +08:00
caozehui
e7348107d0 解决forge库在桌面端无法使用的问题 2025-03-28 09:38:59 +08:00
caozehui
ea38b4eb21 微调 2025-03-28 09:31:46 +08:00
GGJ
67ef976739 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-27 18:34:51 +08:00
GGJ
60145ec4f7 修改参考设定值添加单位 2025-03-27 18:34:40 +08:00
caozehui
81600ddc69 引入获取是否自动生成报告接口 2025-03-27 16:20:16 +08:00
caozehui
e0941d628b 引入获取是否自动生成报告接口 2025-03-27 15:54:27 +08:00
caozehui
760dcbf723 微调 2025-03-27 14:59:01 +08:00
caozehui
b75ecb8c66 微调 2025-03-26 15:42:40 +08:00
caozehui
39f4cfa670 微调 2025-03-25 15:49:32 +08:00
caozehui
862251b83c 微调 2025-03-25 15:08:25 +08:00
caozehui
312490ce59 微调 2025-03-24 19:56:11 +08:00
caozehui
d7df999cf2 登录接口加密传输登录信息 2025-03-24 19:22:56 +08:00
caozehui
e0b1b5907b 微调 2025-03-24 15:41:53 +08:00
sjl
f06ca03881 微调 2025-03-24 14:59:45 +08:00
caozehui
a9b80ed33c 微调 2025-03-24 10:46:16 +08:00
sjl
834230b26a 报告模版 2025-03-24 08:56:08 +08:00
caozehui
403ddbfb6e 微调 2025-03-20 19:42:46 +08:00
caozehui
ffb44ea909 微调 2025-03-20 16:31:15 +08:00
sjl
b729e4efa1 微调 2025-03-19 13:30:11 +08:00
sjl
962c286fd5 主题切换 2025-03-19 10:26:41 +08:00
caozehui
c215f51554 微调 2025-03-18 19:38:27 +08:00
GGJ
859c85b427 检测脚本添加 额定电压 额定电流字段 2025-03-18 16:28:22 +08:00
sjl
5fe637e84f 微调 2025-03-17 16:03:53 +08:00
sjl
5f68071c77 微调颜色 2025-03-17 15:55:30 +08:00
caozehui
1675a5af31 微调 2025-03-17 14:33:19 +08:00
caozehui
ec1d09dbd0 微调 2025-03-17 13:30:00 +08:00
caozehui
eda7b516f6 微调 2025-03-17 13:24:35 +08:00
sjl
630eb48d28 微调 2025-03-17 13:17:12 +08:00
caozehui
0d2f8a7788 微调 2025-03-14 16:30:45 +08:00
GGJ
230c68f454 修改他不删除后编辑检测监本还存在问题 2025-03-14 16:23:27 +08:00
GGJ
68934060e6 修改参考设定值0不显示 2025-03-14 16:10:00 +08:00
GGJ
846e7514b7 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-14 15:03:45 +08:00
GGJ
410cfb0f7a 检测脚本添加额定电压、额定电流 修改bug 2025-03-14 15:03:36 +08:00
caozehui
c892410b07 微调 2025-03-14 14:45:39 +08:00
sjl
7caa4a5303 微调 2025-03-14 12:27:04 +08:00
sjl
c17efbf9f6 微调 2025-03-14 12:13:34 +08:00
c46abeecf5 websocket代码提交 2025-03-14 11:35:55 +08:00
caozehui
0947097932 微调 2025-03-14 11:23:49 +08:00
caozehui
50fc02d4eb 微调 2025-03-13 14:40:33 +08:00
caozehui
a122e9e1b3 微调 2025-03-12 15:01:58 +08:00
sjl
d7cfcc7855 微调 2025-03-12 11:17:12 +08:00
sjl
05948aebf5 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-12 09:53:01 +08:00
sjl
cca214aaf7 程控源 2025-03-12 09:52:52 +08:00
caozehui
0111954c5e 微调 2025-03-11 19:07:47 +08:00
sjl
ce31499964 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-11 18:58:39 +08:00
sjl
059864d193 微调 2025-03-11 18:58:26 +08:00
caozehui
6d2fb4dde8 微调 2025-03-11 18:29:09 +08:00
caozehui
49059d9c59 微调 2025-03-11 18:19:18 +08:00
caozehui
d2f7041988 微调 2025-03-11 18:10:10 +08:00
caozehui
f94372d20b 微调 2025-03-11 16:13:03 +08:00
sjl
9af98b034f 微调 2025-03-11 14:16:21 +08:00
sjl
341b8df5a9 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-11 11:31:02 +08:00
sjl
7470f697bd 微调 2025-03-11 11:30:52 +08:00
caozehui
c058b99a0d 微调 2025-03-11 11:30:10 +08:00
caozehui
d5eed13f38 测试项为1条时也展示出来 2025-03-11 11:04:02 +08:00
caozehui
c27ee88182 微调 2025-03-10 20:14:35 +08:00
sjl
e2559ffa65 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-10 20:00:03 +08:00
sjl
a0bd32453c 微调 2025-03-10 19:59:55 +08:00
sjl
54e7a91841 微调 2025-03-10 19:58:53 +08:00
caozehui
042f6eef51 脚本切换 2025-03-10 19:53:52 +08:00
GGJ
c9857326c6 修改 检测脚本参考值 2025-03-10 16:11:08 +08:00
GGJ
9c4e0094f9 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-10 16:00:15 +08:00
GGJ
781b27af51 修改间谐波值 2025-03-10 16:00:07 +08:00
caozehui
37e257afc8 Loading加载效果 2025-03-10 11:39:10 +08:00
GGJ
4d4fdd6190 修改测试bug 2025-03-10 11:18:32 +08:00
caozehui
0fd2d8252d 微调 2025-03-10 10:50:53 +08:00
caozehui
3cf5019880 源程控-时间展示 2025-03-10 10:10:32 +08:00
caozehui
258d3d692f 源参数默认展开 2025-03-10 10:09:15 +08:00
caozehui
667ed3bcc0 微调 2025-03-08 18:27:10 +08:00
caozehui
a882189022 微调 2025-03-08 18:13:16 +08:00
caozehui
686e009243 微调 2025-03-08 18:07:47 +08:00
caozehui
30d5f3390c Merge branch 'release' 2025-03-08 18:06:50 +08:00
caozehui
3398b414ca 微调 2025-03-08 16:56:57 +08:00
caozehui
b6a2c42feb Merge remote-tracking branch 'origin/release' into release
# Conflicts:
#	frontend/src/views/machine/controlSource/index.vue
2025-03-08 16:05:56 +08:00
caozehui
cb5a9050f2 微调 2025-03-08 16:04:43 +08:00
caozehui
661b27abd0 微调 2025-03-08 16:04:08 +08:00
caozehui
59a24c9ea2 Merge branch 'release' 2025-03-08 16:02:09 +08:00
caozehui
6cf5da5d48 微调 2025-03-08 16:01:41 +08:00
caozehui
537f5e2f4c Merge branch 'release'
# Conflicts:
#	frontend/src/views/machine/controlSource/index.vue
2025-03-08 14:39:20 +08:00
caozehui
70531a0729 微调 2025-03-08 14:36:48 +08:00
caozehui
ded15e5c8b 微调 2025-03-08 10:50:39 +08:00
caozehui
3021f7e4d3 微调 2025-03-08 10:49:42 +08:00
caozehui
876d9cfa00 微调 2025-03-08 10:48:25 +08:00
caozehui
8f28825162 微调 2025-03-08 10:47:19 +08:00
caozehui
0e6612f6bd 微调 2025-03-08 10:15:11 +08:00
caozehui
39183fde5d 微调 2025-03-07 16:26:14 +08:00
caozehui
b967419cb1 微调 2025-03-07 15:43:36 +08:00
sjl
0712698638 微调 2025-03-07 14:44:12 +08:00
sjl
d3a74e830f 微调 2025-03-07 14:00:20 +08:00
sjl
f68da773ab 微调 2025-03-07 13:17:11 +08:00
sjl
8db814dba4 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-03-07 10:17:14 +08:00
sjl
7dcff99de5 微调 2025-03-07 10:17:06 +08:00
caozehui
97100dfa63 复制-二次弹窗确认 2025-03-07 09:02:19 +08:00
caozehui
d41707fcf9 微调 2025-03-06 15:50:45 +08:00
sjl
2377916f29 微调 2025-03-05 11:19:16 +08:00
caozehui
be2f40715a 微调 2025-03-05 10:42:25 +08:00
sjl
489c0882ac 微调 2025-03-05 09:32:13 +08:00
sjl
492db88b45 检测脚本查看隐藏显示 2025-03-04 16:01:34 +08:00
caozehui
3b198eb4ed 微调 2025-03-04 14:12:50 +08:00
caozehui
5efc27d4b6 微调 2025-03-04 10:28:09 +08:00
sjl
ec9f270d04 检测脚本详情查看 2025-03-04 09:35:41 +08:00
caozehui
afd2d467f3 微调 2025-03-03 13:33:17 +08:00
caozehui
ff0798450d 微调 2025-03-03 13:29:48 +08:00
caozehui
fbefc70368 微调 2025-03-03 13:25:19 +08:00
caozehui
faa0bc54a6 不合格项复检 2025-03-03 11:35:12 +08:00
caozehui
bf5f4eab77 Merge branch 'master' into dev 2025-02-28 15:38:08 +08:00
GGJ
188456c970 添加查看页面 2025-02-28 13:58:15 +08:00
caozehui
c06334e96f Merge branch 'master' into dev
# Conflicts:
#	frontend/src/views/home/components/dataCheckSingleChannelSingleTestPopup.vue
2025-02-28 09:03:05 +08:00
GGJ
82e8bf2e0b 优化检测脚本样式 2025-02-28 09:00:15 +08:00
GGJ
30382fb34d 添加一键重算功能 2025-02-27 16:24:13 +08:00
GGJ
c7e5ee0862 优化检测脚本页面 2025-02-27 15:09:09 +08:00
caozehui
d6ac63586b 合并master分支 2025-02-27 14:45:22 +08:00
caozehui
e0fd42199f 微调 2025-02-27 14:34:47 +08:00
caozehui
d9e3168484 Merge branch 'master' into dev
# Conflicts:
#	frontend/src/views/home/components/test.vue
#	frontend/src/views/home/components/testPopup.vue
2025-02-27 13:52:17 +08:00
caozehui
dc7e91ca66 微调 2025-02-27 10:07:41 +08:00
caozehui
692358d278 微调 2025-02-27 08:48:13 +08:00
GGJ
3750031d6b Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-02-27 08:42:37 +08:00
GGJ
0bae200241 优化检测脚本页面
录入检测脚本
2025-02-27 08:41:33 +08:00
sjl
61bd1a0d6a 微调 2025-02-26 19:49:23 +08:00
sjl
37cf3dfdd7 测试 2025-02-26 19:44:27 +08:00
caozehui
8be96ca64e 微调 2025-02-26 19:31:34 +08:00
sjl
d225391248 日志测试 2025-02-26 19:20:02 +08:00
sjl
f2fd0b078e 日志测试 2025-02-26 19:05:56 +08:00
caozehui
f71963ea3b 微调 2025-02-26 19:02:43 +08:00
sjl
6a635246bb 日志测试 2025-02-26 19:00:37 +08:00
caozehui
d338ca6b25 微调 2025-02-26 18:55:50 +08:00
caozehui
193c3b963c 微调 2025-02-26 18:07:57 +08:00
sjl
cad4965b1b 微调 2025-02-26 16:23:27 +08:00
caozehui
79306d846f 微调 2025-02-26 15:58:16 +08:00
caozehui
f4ea510c5d 测试1 2025-02-26 15:56:03 +08:00
sjl
07cc26fa8d 微调 2025-02-26 13:58:56 +08:00
sjl
f87e268242 微调 2025-02-26 13:57:42 +08:00
caozehui
1aeceffe0b 测试1 2025-02-26 13:57:07 +08:00
caozehui
fc97087767 测试1 2025-02-26 13:22:01 +08:00
sjl
61be97b04b Merge branch 'dev' of http://192.168.1.22:3000/frontend/pqs-9100_client into dev 2025-02-26 11:40:41 +08:00
sjl
934162875c 微调 2025-02-26 11:40:31 +08:00
caozehui
4b3991099a 微调 2025-02-26 11:25:20 +08:00
caozehui
b14a961c1a 微调 2025-02-26 11:13:28 +08:00
sjl
cbe41d80ce 合并frontend/src/views/home/components/testPopup.vue 2025-02-26 10:54:43 +08:00
sjl
28203f2621 微调 2025-02-26 10:52:39 +08:00
caozehui
b8492dfbe6 微调 2025-02-26 10:42:01 +08:00
caozehui
15faac8079 正式检测-移除重新检测按钮 2025-02-26 10:05:43 +08:00
caozehui
ad850ca4ee 微调 2025-02-26 10:00:44 +08:00
caozehui
6c05c6e06e 微调 2025-02-26 09:14:27 +08:00
caozehui
8b2cda80b1 微调 2025-02-25 16:37:59 +08:00
caozehui
3f7756b417 微调 2025-02-25 16:18:22 +08:00
caozehui
5004e319b6 微调 2025-02-25 15:11:23 +08:00
caozehui
19035f2dc9 微调 2025-02-25 15:03:45 +08:00
caozehui
9d8a5c76b2 微调 2025-02-25 14:52:06 +08:00
caozehui
6300eb2f43 微调 2025-02-25 14:13:59 +08:00
sjl
da58ffb621 微调 2025-02-25 14:01:07 +08:00
sjl
1500fafb92 微调 2025-02-25 14:00:40 +08:00
sjl
df53eea432 微调 2025-02-25 13:53:47 +08:00
sjl
cc1771c208 微调 2025-02-25 13:52:57 +08:00
caozehui
5e881c0804 微调 2025-02-25 13:48:44 +08:00
caozehui
5666e01ae0 微调 2025-02-25 13:47:33 +08:00
caozehui
4acec8a2d6 微调 2025-02-25 13:46:01 +08:00
sjl
426a511099 微调 2025-02-25 11:20:19 +08:00
caozehui
89303b44ae 微调 2025-02-25 11:00:04 +08:00
caozehui
ffb82066bb 正式检测设备连接超时,停止检测 2025-02-25 10:46:12 +08:00
caozehui
f46b8c0a56 微调 2025-02-25 10:29:29 +08:00
caozehui
280289af37 手动检测功能修改、一键检测功能修改(均未完成) 2025-02-25 10:17:33 +08:00
GGJ
a0c7f79302 修改检测脚本闪变编辑逻辑 2025-02-25 08:58:03 +08:00
GGJ
bd1eecc106 优化检测脚本页面 2025-02-24 16:45:39 +08:00
sjl
1a23bcb510 检测计划未检下拉框不可编辑 2025-02-24 16:30:52 +08:00
sjl
942c28d3bd 微调 2025-02-24 08:54:22 +08:00
GGJ
034b31ba47 联调检测脚本 2025-02-24 08:38:54 +08:00
sjl
7a76c2da8a 微调 2025-02-21 14:57:52 +08:00
caozehui
1a9d8e8606 检测通道数量微调 2025-02-21 09:05:06 +08:00
GGJ
14583d919d 联调检测脚本修改功能 2025-02-20 16:39:15 +08:00
caozehui
629fd174e4 微调 2025-02-20 15:01:15 +08:00
sjl
9080fe06a4 角标对齐 2025-02-20 10:21:36 +08:00
sjl
0e183bb5c1 检测源删除 2025-02-20 10:10:28 +08:00
GGJ
8caf856d2d 联调检测脚本页面 2025-02-19 16:54:54 +08:00
caozehui
d28669e6b0 正式检测-时间显示调整 2025-02-19 13:17:19 +08:00
700884d01a websocket代码提交 2025-02-19 11:28:52 +08:00
c06bd9fa24 websocket代码提交 2025-02-19 11:00:09 +08:00
GGJ
ce92a1d645 联调 新增检测脚本 2025-02-18 16:36:54 +08:00
caozehui
4f622da52c 调整检测计划导入、导出功能适应二楼装备 2025-02-18 15:58:38 +08:00
7d3e86a5fc websocket代码提交 2025-02-18 14:59:13 +08:00
sjl
c66d181b98 检测计划取消勾选 2025-02-18 09:00:23 +08:00
GGJ
f860630bfb 修改检测脚本新增页面路由权限 2025-02-18 08:51:20 +08:00
caozehui
6db4d79839 调整 2025-02-18 08:48:59 +08:00
GGJ
e2924d5e57 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client
# Conflicts:
#	frontend/src/views/machine/testScript/components/testScriptPopup.vue
#	frontend/src/views/machine/testScript/index.vue
2025-02-17 16:46:20 +08:00
GGJ
92b9a82f21 修改检测脚本页面 2025-02-17 16:44:02 +08:00
sjl
c3f89ca1f8 微调 2025-02-17 16:32:02 +08:00
sjl
8c435c91dd Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-02-17 16:29:06 +08:00
sjl
6e6cfa5be0 微调 2025-02-17 16:28:59 +08:00
caozehui
ea203d0ccc 调整统计分析功能 2025-02-17 16:26:23 +08:00
caozehui
370cd52774 调整静态路由 2025-02-17 13:37:27 +08:00
caozehui
942bae3045 微调 2025-02-17 11:30:56 +08:00
caozehui
356548c8fb 显示时间 2025-02-17 10:16:48 +08:00
GGJ
5c012c2bc9 修改 检测脚本新增页面 2025-02-17 09:00:27 +08:00
GGJ
9c9b8015a0 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-02-17 08:40:19 +08:00
GGJ
bba0ced7f9 修改 测试脚本页面 2025-02-17 08:39:18 +08:00
caozehui
e8de7b56d9 微调 2025-02-14 16:22:45 +08:00
caozehui
95022f62a8 数据分析 2025-02-14 13:30:50 +08:00
sjl
1d918f3335 微调 2025-02-14 11:44:19 +08:00
caozehui
f992637ad2 微调 2025-02-14 10:59:21 +08:00
caozehui
40f18bbdf8 微调 2025-02-14 10:39:00 +08:00
caozehui
3bb9cec08a 微调 2025-02-14 10:31:16 +08:00
caozehui
6842260f59 微调 2025-02-14 09:57:26 +08:00
sjl
305c30d725 微调 2025-02-14 09:16:07 +08:00
caozehui
e799dd2f08 移除headers中的Refresh-Token,添加Is-Refresh-Token 2025-02-13 21:02:34 +08:00
GGJ
1e83172e9a 修改检测脚本页面 2025-02-13 16:15:26 +08:00
sjl
475d236b7e 微调 2025-02-13 15:40:13 +08:00
sjl
ddeb84fcfe 微调 2025-02-11 14:25:12 +08:00
sjl
b3aeebb7c7 icd管理 2025-02-11 11:26:50 +08:00
sjl
b11f4839f4 微调 2025-02-11 10:59:41 +08:00
sjl
4adc28a055 设备类型 2025-02-11 10:40:34 +08:00
sjl
e46e9e1d3a 设备类型CRUD 2025-02-11 10:32:09 +08:00
sjl
4fcb26bbde Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-02-11 10:31:57 +08:00
caozehui
6bd0e891fa 改为12通道 2025-02-11 09:10:50 +08:00
sjl
15f1308557 设备类型 2025-02-10 16:35:14 +08:00
caozehui
9689197db9 微调 2025-02-10 15:41:59 +08:00
caozehui
4ad490657f 微调 2025-02-10 15:16:13 +08:00
caozehui
9f8f8244e3 微调 2025-02-08 15:36:56 +08:00
sjl
9206be2562 登录失效 2025-02-07 14:28:15 +08:00
caozehui
6065fd1f67 双token(未做完) 2025-02-07 10:39:30 +08:00
sjl
c16560a0ca 微调 2025-01-23 10:05:47 +08:00
caozehui
440cfe1ac5 源未开启外联机错误弹窗出现2次 2025-01-23 09:33:00 +08:00
sjl
1ad1a0198e 微调 2025-01-22 16:28:09 +08:00
caozehui
b603ac9bdb 源未开启外联机错误 2025-01-22 14:31:53 +08:00
caozehui
68aa4c409b 三项不平衡原始数据表头修改 2025-01-22 14:03:50 +08:00
sjl
f22b4e7fb8 微调 2025-01-22 12:09:32 +08:00
caozehui
400328d5fa 引入导出原始数据接口 2025-01-21 14:52:44 +08:00
caozehui
cce3e6861a 查看结果弹窗-通道下拉框 2025-01-21 11:06:22 +08:00
sjl
55368ce9b8 微调 2025-01-21 09:05:35 +08:00
caozehui
c6ef54f7d3 正式检测-预检测阶段错误码处理、频率、三项电压不平衡表头修改 2025-01-20 15:24:29 +08:00
GGJ
6f97c48226 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-17 15:06:58 +08:00
GGJ
5f45bc98fb 修改 WebSocket 关闭浏览器心跳问题 2025-01-17 15:06:44 +08:00
sjl
5738594be2 微调 2025-01-17 14:57:17 +08:00
sjl
ce47831992 socket连接不上处理 2025-01-17 09:59:23 +08:00
634c2d84fc Merge remote-tracking branch 'origin/master' 2025-01-17 09:35:58 +08:00
70855c5a21 websocket代码提交 2025-01-17 09:35:44 +08:00
sjl
b1e5e70411 微调 2025-01-17 09:25:52 +08:00
sjl
bf2e3219fc Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-17 09:15:58 +08:00
sjl
2f5aeb9ffc 微调 2025-01-17 09:15:52 +08:00
9e8ed310fc websocket代码提交 2025-01-17 09:14:19 +08:00
07a5e32579 批量导入被检设备 2025-01-16 19:45:48 +08:00
sjl
fe1bb04922 微调 2025-01-16 16:32:06 +08:00
a75e964adc 正式检测代码提交 2025-01-16 15:57:29 +08:00
sjl
87d22032e6 微调 2025-01-16 15:43:58 +08:00
sjl
55ae64b3e1 微调 2025-01-16 14:24:55 +08:00
caozehui
891c6d8e93 检测结果查看,相角类的放在最后展示 2025-01-16 10:00:49 +08:00
caozehui
ba7747dfe9 暂停时间过长页面效果 2025-01-16 08:51:47 +08:00
b148c04059 正式检测代码提交 2025-01-15 21:08:07 +08:00
sjl
4e179d61b5 微调 2025-01-15 20:37:58 +08:00
sjl
b23ee84543 微调 2025-01-15 20:32:14 +08:00
sjl
d02f85cea5 微调 2025-01-15 20:23:05 +08:00
ab02f8f1b3 正式检测代码提交 2025-01-15 19:27:33 +08:00
8fc1d1ffc9 微调报告 2025-01-15 18:01:12 +08:00
caozehui
44b1da94c1 二楼使用设备导入导出 2025-01-15 16:02:04 +08:00
sjl
c2b63a7a3f 微调 2025-01-15 13:46:11 +08:00
sjl
56619c9140 微调 2025-01-15 13:26:05 +08:00
caozehui
99bbb5434d 切换通道左侧树问题、点击左侧树测试项类别清空、点击左侧树赋值ScriptType 2025-01-15 10:49:28 +08:00
GGJ
40c9c5fdad 修改打包后svg无法显示问题 2025-01-15 10:29:19 +08:00
caozehui
3ea2c93e43 T相显示为检测名称 2025-01-15 10:14:33 +08:00
sjl
6369c8ee63 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-15 09:36:12 +08:00
sjl
da4bbf1e2c 微调 2025-01-15 09:36:04 +08:00
sjl
14e5535fd8 微调 2025-01-15 09:35:36 +08:00
caozehui
4e53bb3b27 10552错误重新检测、相序校验25003错误 2025-01-15 09:33:07 +08:00
GGJ
e793fdbe42 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-15 09:07:23 +08:00
GGJ
0e41a5b9f1 修改tab刷新位置问题 2025-01-15 09:07:12 +08:00
caozehui
ac9dfd1669 测试小项日志展示 2025-01-15 08:59:47 +08:00
caozehui
72eb3e189b 暂停页面效果调整 2025-01-15 08:53:45 +08:00
GGJ
0cace11975 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-15 08:42:59 +08:00
GGJ
b8e502b168 修改页面缓存 2025-01-15 08:42:41 +08:00
caozehui
15502d0107 暂停页面效果调整 2025-01-14 21:07:52 +08:00
GGJ
cb4e4646bd Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-14 19:26:29 +08:00
GGJ
93ce1b4994 修改登录页样式 2025-01-14 19:26:22 +08:00
sjl
1c452b5574 微调 2025-01-14 19:05:08 +08:00
caozehui
876c06ef54 暂停页面效果调整 2025-01-14 18:58:13 +08:00
sjl
37d65a944d 微调 2025-01-14 16:24:29 +08:00
caozehui
7bc9321c86 Merge remote-tracking branch 'origin/master' 2025-01-14 16:22:05 +08:00
caozehui
ee564dccb6 暂停正式检测 2025-01-14 16:21:50 +08:00
GGJ
cdbb386c7c Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-14 16:19:51 +08:00
GGJ
325ade7cdc 修改样式 2025-01-14 16:19:15 +08:00
caozehui
fcd8c1a0dd 修复数据查询弹窗,左侧切换,检测项目切换时,数据错误 2025-01-14 16:16:30 +08:00
sjl
fa1cb02a51 微调 2025-01-14 16:03:37 +08:00
caozehui
40fe12cd96 相别名称修改,数据查询弹窗,数据为空时处理 2025-01-14 15:34:35 +08:00
sjl
d83c04dc8b Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-14 15:14:44 +08:00
sjl
23b2974597 微调 2025-01-14 15:14:37 +08:00
caozehui
f5e78316ef 被检测设备-模式字段隐藏 2025-01-14 14:40:35 +08:00
caozehui
cc2020f3d6 暂停调整 2025-01-14 14:17:05 +08:00
sjl
26cda335c5 微调 2025-01-14 11:43:35 +08:00
caozehui
171533cef0 调整优先级 2025-01-14 10:36:42 +08:00
sjl
69de6dd7a3 设备名称 2025-01-14 10:03:44 +08:00
sjl
9039ed4bdc Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-13 23:17:31 +08:00
sjl
131d3b7018 微调 2025-01-13 23:17:24 +08:00
caozehui
454aabdc97 右上角错误提示 2025-01-13 22:59:03 +08:00
caozehui
089c97d859 微调 2025-01-13 22:53:38 +08:00
caozehui
9ff5423368 完善归档接口 2025-01-13 22:50:55 +08:00
sjl
f29d6e8d88 归档 2025-01-13 22:47:15 +08:00
sjl
5199ee2db0 微调 2025-01-13 22:42:14 +08:00
caozehui
6de9e22b4c 谐波次数切换时,原始数据显示为T项NaN问题 2025-01-13 22:05:46 +08:00
caozehui
52968d9aaa 调整检测结果,可筛选测试项 2025-01-13 21:50:24 +08:00
sjl
b584142f5e 微调 2025-01-13 21:38:52 +08:00
sjl
edea30a406 刷新 2025-01-13 21:26:12 +08:00
sjl
1f3024eb91 刷新饼图 2025-01-13 21:06:24 +08:00
8b9abba74b 微调 2025-01-13 20:19:04 +08:00
sjl
4dec1392f7 设备类型接口 2025-01-13 19:28:51 +08:00
sjl
e54cbfbc0a 设备icd 2025-01-13 18:32:42 +08:00
caozehui
505cd93446 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	frontend/src/api/device/device/index.ts
2025-01-13 18:21:24 +08:00
caozehui
f3ee11ebab 引入归档接口 2025-01-13 18:20:35 +08:00
sjl
45295ef898 微调 2025-01-13 18:13:55 +08:00
sjl
0635dda6a0 微调 2025-01-13 18:12:36 +08:00
caozehui
f58f7b97e5 展示单位调整 2025-01-13 15:03:24 +08:00
caozehui
7556f0e5bc 原始数据导出按钮、返回左侧树形列表接口调整 2025-01-13 14:22:54 +08:00
caozehui
17fe29cb1b 谐波次数、弹窗位置 2025-01-13 13:57:24 +08:00
caozehui
4bb4eb80d9 报告生成 2025-01-13 09:59:05 +08:00
ae1507e775 添加下载 2025-01-13 09:35:41 +08:00
sjl
85d8a57bf7 微调 2025-01-13 08:56:13 +08:00
caozehui
42cf23a516 微调 2025-01-10 14:19:29 +08:00
caozehui
858cf75b46 处理右侧原始数据表格行数错误问题 2025-01-10 14:15:07 +08:00
caozehui
de8a392d2c 处理源未知异常、检测结果最大误差回显 2025-01-10 13:49:09 +08:00
caozehui
b614767b2d 处理装置配置异常 2025-01-10 11:03:29 +08:00
caozehui
5ebb8549bf 调整优先级 2025-01-10 10:47:08 +08:00
caozehui
ead1feb379 Merge remote-tracking branch 'origin/master'
# Conflicts:
#	frontend/src/views/home/components/test.vue
2025-01-10 09:36:00 +08:00
caozehui
1474c15bd9 放开数据异常按钮弹窗 2025-01-10 09:34:22 +08:00
caozehui
f24f87dcc8 放开数据异常按钮弹窗 2025-01-10 09:20:32 +08:00
caozehui
d416f46b3a 微调 2025-01-10 09:15:59 +08:00
352df35205 正式检测代码提交 2025-01-09 20:02:34 +08:00
caozehui
e5f201515e 引入暂停、继续检测接口 2025-01-09 16:27:05 +08:00
caozehui
40520a7d30 修改正式检测结果展示错误 2025-01-09 15:49:32 +08:00
caozehui
cffbfe1380 预检测失败处理、正式检测-只检测某一个大项,日志bug 2025-01-09 14:45:43 +08:00
caozehui
839777ea4d 微调 2025-01-09 14:18:50 +08:00
sjl
5be1d22115 微调 2025-01-09 13:56:56 +08:00
caozehui
80d1912fd5 微调 2025-01-09 11:13:34 +08:00
caozehui
b94e0323d9 微调 2025-01-08 21:02:36 +08:00
caozehui
90f4e53e9f 微调 2025-01-08 20:57:11 +08:00
caozehui
7ebad7d6b9 微调 2025-01-08 19:11:34 +08:00
caozehui
34534a1dbf 微调 2025-01-08 18:52:17 +08:00
645fac0681 正式检测代码提交 2025-01-08 15:15:49 +08:00
0a23700403 正式检测代码提交 2025-01-08 15:15:25 +08:00
caozehui
c00f8843c0 微调 2025-01-08 14:48:19 +08:00
caozehui
8869974471 Merge remote-tracking branch 'origin/master' 2025-01-08 14:34:52 +08:00
caozehui
8125d60105 微调 2025-01-08 14:33:47 +08:00
sjl
5a32d5b7a8 系数校准多装置按行展示 2025-01-08 11:40:12 +08:00
caozehui
a1d113562b 微调 2025-01-08 11:02:16 +08:00
caozehui
4a7535e4f5 微调 2025-01-08 10:01:56 +08:00
caozehui
de5548bbbd 微调 2025-01-08 09:46:47 +08:00
caozehui
87c46e8477 微调 2025-01-07 20:24:44 +08:00
caozehui
edda224847 微调 2025-01-07 19:47:38 +08:00
caozehui
5ac6bbf04b 微调 2025-01-07 19:03:00 +08:00
caozehui
b0a5329937 微调 2025-01-07 17:44:25 +08:00
caozehui
7a0626413d 微调 2025-01-07 17:34:15 +08:00
fe04ad1573 正式检测代码提交 2025-01-07 16:27:51 +08:00
caozehui
689cc39bbc 返回结果 2025-01-07 14:54:44 +08:00
caozehui
ba0c577514 修改检测结果弹窗框 2025-01-07 11:19:33 +08:00
caozehui
a629907196 微调 2025-01-07 09:46:35 +08:00
caozehui
bf60bcfe12 微调 2025-01-07 09:01:10 +08:00
caozehui
27eb9a7d93 右侧表格1 2025-01-06 19:20:36 +08:00
caozehui
336ee843cc 微调 2025-01-06 14:50:36 +08:00
sjl
9eac101794 系数校准不合格提示 2025-01-06 14:27:57 +08:00
sjl
085fcc62aa Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-06 14:07:28 +08:00
sjl
99e6abbcfd 微调 2025-01-06 14:07:21 +08:00
a4b96066e0 样式调整 2025-01-06 14:02:22 +08:00
sjl
c4e2da8d67 主界面,系数校准界面调整 2025-01-06 09:21:24 +08:00
caozehui
1be15ee849 修改右侧表格结构 2025-01-06 08:51:14 +08:00
caozehui
99bc54349b 微调 2025-01-05 14:52:37 +08:00
caozehui
bf823ddff0 微调 2025-01-04 16:33:23 +08:00
caozehui
23e72f38a6 微调 2025-01-04 14:09:01 +08:00
caozehui
2d993b2f73 存在已经初始化步骤,重新检测 2025-01-04 10:49:05 +08:00
9703d35a3a 正式检测代码提交 2025-01-04 09:58:39 +08:00
caozehui
d10a7a4884 微调 2025-01-03 18:57:35 +08:00
caozehui
938b9054d3 微调 2025-01-03 11:27:36 +08:00
caozehui
8a5bcca901 微调 2025-01-02 20:27:36 +08:00
caozehui
410cd53e51 正式检测-预检测提示及日志功能 2025-01-02 18:00:58 +08:00
sjl
4031724fa9 tab页面自动切换 2025-01-02 14:17:41 +08:00
sjl
e95552e6b9 微调 2025-01-02 13:12:13 +08:00
sjl
012f7e3346 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2025-01-02 09:02:54 +08:00
sjl
cbe6c4a98d 微调 2025-01-02 09:02:46 +08:00
caozehui
aa52778e99 微调 2024-12-31 19:54:27 +08:00
caozehui
5e8dda0b75 微调 2024-12-31 19:03:52 +08:00
caozehui
081aeacff7 微调 2024-12-31 14:27:36 +08:00
caozehui
65e54ee2b5 正式检测-左侧检测大项动态获取 2024-12-30 20:55:11 +08:00
sjl
0fd6ebf810 系数校准调整 2024-12-30 19:59:15 +08:00
sjl
fa4b5d26e5 系数校准 2024-12-30 19:17:39 +08:00
sjl
23f45b38cb Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-30 14:43:20 +08:00
sjl
d76b055730 系数校准 2024-12-30 14:43:13 +08:00
caozehui
c334c07b39 微调 2024-12-30 14:41:34 +08:00
caozehui
45a5dcb068 微调 2024-12-28 16:50:32 +08:00
caozehui
3d9ba8b1cb 微调 2024-12-27 14:29:21 +08:00
caozehui
c6836cad06 微调 2024-12-27 13:36:20 +08:00
caozehui
e70d1cb3f2 处理正式检测bug 2024-12-27 10:31:15 +08:00
caozehui
af454e1a56 修改正式检测样式 2024-12-26 10:49:55 +08:00
sjl
6ce58e86ad 微调 2024-12-26 09:28:19 +08:00
2e17531c52 预见测代码 2024-12-25 20:29:27 +08:00
caozehui
6b76d89507 微调 2024-12-25 18:04:16 +08:00
caozehui
e4025e39cb 微调 2024-12-25 13:22:54 +08:00
sjl
af12dc8d0a frontend/src/views/machine/errorSystem/components/errorSystemDetailTable.vue 2024-12-24 20:25:10 +08:00
sjl
f0edeaee85 误差体系禁用 2024-12-24 20:22:36 +08:00
caozehui
5cbdadecc4 微调 2024-12-24 19:21:43 +08:00
caozehui
30c1f90513 微调 2024-12-24 11:29:31 +08:00
c241d8b819 预见测代码 2024-12-24 08:55:04 +08:00
24f12a3590 预见测代码 2024-12-23 21:02:00 +08:00
sjl
f17f5c9925 误差体系 2024-12-23 16:12:46 +08:00
sjl
99877b1c98 系数校准动态通道数 2024-12-23 15:08:14 +08:00
sjl
fdf91965d9 微调 2024-12-23 14:05:53 +08:00
caozehui
fd21d32f4a 微调 2024-12-23 13:23:28 +08:00
caozehui
0a4385f29b 微调 2024-12-21 12:41:06 +08:00
caozehui
63c6693d04 微调 2024-12-20 20:34:39 +08:00
caozehui
ff7084a332 微调 2024-12-20 20:20:08 +08:00
caozehui
b63e8897f5 微调 2024-12-20 20:14:52 +08:00
caozehui
dc3976f525 微调 2024-12-20 19:14:07 +08:00
caozehui
9b8765f001 引入store存放检测设备的相关信息 2024-12-20 18:19:18 +08:00
bbe23e1b14 预见测代码 2024-12-20 16:32:03 +08:00
sjl
5f0b36b3bb Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-20 10:43:58 +08:00
sjl
f0b75eae2f frontend/src/views/home/components/test.vue
frontend/src/views/home/components/timeTest.vue
2024-12-20 10:43:43 +08:00
caozehui
45e62b2750 微调 2024-12-20 10:43:17 +08:00
sjl
d11e4822ed 系数校准 2024-12-20 10:21:36 +08:00
caozehui
93b67fe845 修改正式检测,使其能按照通道进行页面展示 2024-12-20 10:19:58 +08:00
caozehui
ebc2ac1a41 微调 2024-12-18 20:26:18 +08:00
caozehui
4fd2f10ec6 正式检测页面修改 2024-12-18 18:53:55 +08:00
caozehui
2715871017 正式检测页面修改 2024-12-18 15:56:59 +08:00
sjl
4cfca89895 样式调整 2024-12-18 15:49:29 +08:00
53ead0e62f 下拉框添加模糊匹配 2024-12-18 14:55:14 +08:00
caozehui
b00967159a Merge remote-tracking branch 'origin/master' 2024-12-17 14:08:35 +08:00
caozehui
1b37684e3c 微调 2024-12-17 14:08:13 +08:00
sjl
48f393be25 主页模式切换 2024-12-17 14:08:01 +08:00
sjl
645fb83bce 微调 2024-12-17 13:57:45 +08:00
caozehui
fe8bc407d3 饼图数据默认,饼图数据计算方式调整 2024-12-17 13:57:18 +08:00
sjl
d20cab57be 微调 2024-12-17 13:38:59 +08:00
sjl
68df3d5e23 # frontend/src/views/home/components/table.vue
#	frontend/src/views/home/tabs/dashboard.vue
2024-12-17 11:36:02 +08:00
sjl
8984bc577e 微调 2024-12-17 11:34:54 +08:00
caozehui
07d02307e2 默认选中第一条检测计划 2024-12-17 11:12:21 +08:00
sjl
a027dc75c3 左侧树点击刷新饼图和表格 2024-12-16 16:33:18 +08:00
sjl
5da07d8631 微调 2024-12-16 15:25:30 +08:00
sjl
d061fc9b32 检测计划列表 2024-12-16 14:09:21 +08:00
caozehui
ae47d429ba 微调 2024-12-16 11:34:12 +08:00
caozehui
396d6a7336 微调 2024-12-16 10:54:40 +08:00
sjl
fdf826d5f8 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-13 16:35:55 +08:00
sjl
0db5fa650f 微调 2024-12-13 16:35:27 +08:00
GYYM
53b4616a87 微调 2024-12-13 08:56:42 +08:00
GYYM
b923b7c35d 饼图样式调整 2024-12-12 13:23:09 +08:00
sjl
8bd5351d29 检测计划表格误差体系,检测源跳转 2024-12-12 11:38:42 +08:00
GYYM
726f7841e8 日志显示内容微调 2024-12-11 19:12:04 +08:00
sjl
47883ce426 微调 2024-12-11 20:24:00 +08:00
GYYM
2459fc1848 正式检测日志修改 2024-12-11 18:48:54 +08:00
sjl
cbc11f636a Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-11 19:39:16 +08:00
sjl
5dbd6eb8be 检测计划 2024-12-11 19:39:09 +08:00
GYYM
167716e61b 首页、守时检测页样式调整 2024-12-11 17:46:59 +08:00
GYYM
6d7f045bce 归档页调整 2024-12-09 16:17:04 +08:00
GYYM
a73417ea02 修改未检状态的描述 2024-12-06 15:06:38 +08:00
GYYM
9c3850db98 文字修改 2024-12-06 09:21:55 +08:00
393ad5fa0b 单元格样式调整 2024-12-06 09:18:13 +08:00
sjl
9e8fe2c91f Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-06 09:11:55 +08:00
sjl
b57a7edf28 微调弹出框 2024-12-06 09:11:44 +08:00
GYYM
85cc5ebb27 修改不合格项表头的提示方式 2024-12-06 09:10:39 +08:00
GGJ
0cd32fbb5f 修改表格样式 2024-12-06 09:07:16 +08:00
GYYM
2833e65f57 修改正式检测首列的显示样式 2024-12-05 22:15:12 +08:00
GYYM
5a29403636 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-05 21:34:10 +08:00
GYYM
9c0b9a7c93 样式调整 2024-12-05 21:33:52 +08:00
sjl
69a4cbbbc2 微调 2024-12-05 20:29:28 +08:00
sjl
e2b2382cc4 误差体系 2024-12-05 20:11:26 +08:00
sjl
f8a4d05f57 误差体系 2024-12-05 20:06:59 +08:00
caozehui
d996c43f9f Merge remote-tracking branch 'origin/master' 2024-12-05 19:31:56 +08:00
caozehui
c195a58eb4 微调 2024-12-05 19:31:43 +08:00
GGJ
b1c9375873 修改样式 2024-12-05 18:14:43 +08:00
GYYM
6010cbf20a 微调 2024-12-05 17:30:25 +08:00
sjl
11c32ed360 3列弹出框 2024-12-05 16:23:20 +08:00
GYYM
0494822ff4 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-05 15:26:06 +08:00
GYYM
2611f43d1f 微调 2024-12-05 15:25:49 +08:00
GGJ
ea4edfadd0 修改样式 2024-12-05 15:22:46 +08:00
caozehui
8e2a22cf61 微调 2024-12-05 14:37:50 +08:00
sjl
3a1cfe0f31 微调 2024-12-05 13:59:15 +08:00
sjl
5bdc44b301 微调 2024-12-05 13:51:47 +08:00
GYYM
92a1736667 守时微调 2024-12-05 13:46:53 +08:00
caozehui
57f9238033 Merge remote-tracking branch 'origin/master' 2024-12-05 13:43:19 +08:00
caozehui
1b171d4b8d 微调 2024-12-05 13:43:05 +08:00
GYYM
c5acb84d97 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-05 11:21:28 +08:00
GYYM
63a1e9ec56 微调 2024-12-05 11:21:10 +08:00
sjl
f06a8d6b15 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-05 11:07:57 +08:00
GGJ
92df927453 修改样式问题 2024-12-05 11:07:54 +08:00
sjl
2878dce69d 模式切换 2024-12-05 11:07:45 +08:00
caozehui
7d5d128b54 微调 2024-12-05 09:55:35 +08:00
GYYM
6e9e3db358 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-04 21:37:25 +08:00
GYYM
a415ffa12a 样式调整 2024-12-04 21:36:12 +08:00
sjl
9e8cdaead6 Merge branch 'master' of http://192.168.1.22:3000/frontend/pqs-9100_client 2024-12-04 20:00:15 +08:00
sjl
e5d4aae846 微调 2024-12-04 20:00:04 +08:00
caozehui
d6af35828d 样式调整 2024-12-04 19:54:56 +08:00
GYYM
aa86c5e662 微调 2024-12-04 10:20:55 +08:00
GYYM
f99aa94b8f 数据内容微调 2024-12-04 09:44:26 +08:00
sjl
f5c226ba28 微调通用配置 2024-12-04 09:08:48 +08:00
sjl
741020fe79 微调 2024-12-04 08:49:42 +08:00
GYYM
33004e081d 样式调整 2024-12-03 17:13:47 +08:00
228 changed files with 35221 additions and 15670 deletions

View File

@@ -1,112 +0,0 @@
import axios, { AxiosInstance, AxiosError, AxiosRequestConfig, InternalAxiosRequestConfig, AxiosResponse } from "axios";
import { showFullScreenLoading, tryHideFullScreenLoading } from "@/components/Loading/fullScreen";
import { LOGIN_URL } from "@/config";
import { ElMessage } from "element-plus";
import { ResultData } from "@/api/interface";
import { ResultEnum } from "@/enums/httpEnum";
import { checkStatus } from "./helper/checkStatus";
import { useUserStore } from "@/stores/modules/user";
import router from "@/routers";
export interface CustomAxiosRequestConfig extends InternalAxiosRequestConfig {
loading?: boolean;
}
const config = {
// 默认地址请求地址,可在 .env.** 文件中修改
baseURL: import.meta.env.VITE_API_URL as string,
// 设置超时时间
timeout: ResultEnum.TIMEOUT as number,
// 跨域时候允许携带凭证
withCredentials: true,
// post请求指定数据类型以及编码
headers: { 'Content-Type': 'application/json;charset=utf-8' }
};
class RequestHttp {
service: AxiosInstance;
public constructor(config: AxiosRequestConfig) {
// 创建实例
this.service = axios.create(config);
/**
* @description 请求拦截器
* 客户端发送请求 -> [请求拦截器] -> 服务器
* token校验(JWT) : 接受服务器返回的 token,存储到 vuex/pinia/本地储存当中
*/
this.service.interceptors.request.use(
(config: CustomAxiosRequestConfig) => {
const userStore = useUserStore();
// 当前请求不需要显示 loading在 api 服务中通过指定的第三个参数: { loading: false } 来控制
config.loading ?? (config.loading = true);
config.loading && showFullScreenLoading();
if (config.headers && typeof config.headers.set === "function") {
config.headers.set("x-access-token", userStore.token);
}
return config;
},
(error: AxiosError) => {
return Promise.reject(error);
}
);
/**
* @description 响应拦截器
* 服务器换返回信息 -> [拦截统一处理] -> 客户端JS获取到信息
*/
this.service.interceptors.response.use(
(response: AxiosResponse) => {
const { data } = response;
const userStore = useUserStore();
tryHideFullScreenLoading();
// 登陆失效
if (data.code == ResultEnum.OVERDUE) {
userStore.setToken("");
router.replace(LOGIN_URL);
ElMessage.error(data.message);
return Promise.reject(data);
}
// 全局错误信息拦截(防止下载文件的时候返回数据流,没有 code 直接报错)
if (data.code && data.code !== ResultEnum.SUCCESS) {
ElMessage.error(data.message);
return Promise.reject(data);
}
// 成功请求(在页面上除非特殊情况,否则不用处理失败逻辑)
return data;
},
async (error: AxiosError) => {
const { response } = error;
tryHideFullScreenLoading();
// 请求超时 && 网络错误单独判断,没有 response
if (error.message.indexOf("timeout") !== -1) ElMessage.error("请求超时!请您稍后重试");
if (error.message.indexOf("Network Error") !== -1) ElMessage.error("网络错误!请您稍后重试");
// 根据服务器响应的错误状态码,做不同的处理
if (response) checkStatus(response.status);
// 服务器结果都没有返回(可能服务器错误可能客户端断网),断网处理:可以跳转到断网页面
if (!window.navigator.onLine) router.replace("/500");
return Promise.reject(error);
}
);
}
/**
* @description 常用请求方法封装
*/
get<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> {
return this.service.get(url, { params, ..._object });
}
post<T>(url: string, params?: object | string, _object = {}): Promise<ResultData<T>> {
return this.service.post(url, params, _object);
}
put<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> {
return this.service.put(url, params, _object);
}
delete<T>(url: string, params?: any, _object = {}): Promise<ResultData<T>> {
return this.service.delete(url, { params, ..._object });
}
download(url: string, params?: object, _object = {}): Promise<BlobPart> {
return this.service.post(url, params, { ..._object, responseType: "blob" });
}
}
export default new RequestHttp(config);

12
.vscode/settings.json vendored
View File

@@ -1,18 +1,28 @@
{ {
"cSpell.words": [ "cSpell.words": [
"Analyse",
"CHNFACTOR",
"CHNPACTOR",
"Chns", "Chns",
"Combox", "Combox",
"Datasource",
"daterange", "daterange",
"devicedata", "devicedata",
"errordata", "errordata",
"errordetail", "errordetail",
"Interharmonic",
"IRMS",
"logdata", "logdata",
"MSQI",
"Pids", "Pids",
"plandata", "plandata",
"PQDIF", "PQDIF",
"Preinvestment",
"resourcedata", "resourcedata",
"resourcename", "resourcename",
"rmark", "rmark",
"showtest" "showtest",
"UNBAN",
"VRMS"
] ]
} }

View File

@@ -25,10 +25,10 @@ module.exports = (appInfo) => {
*/ */
config.windowsOption = { config.windowsOption = {
title: '自动检测平台', title: '自动检测平台',
width: 1280, width: 1920 /1.5,
height: 850, height: 1080 /1.2,
minWidth: 1280, minWidth: 1920 /1.5,
minHeight: 850, minHeight: 1080 /1.2,
webPreferences: { webPreferences: {
//webSecurity: false, //webSecurity: false,
contextIsolation: false, // false -> 可在渲染进程中使用electron的apitrue->需要bridge.js(contextBridge) contextIsolation: false, // false -> 可在渲染进程中使用electron的apitrue->需要bridge.js(contextBridge)

View File

@@ -9,7 +9,7 @@ module.exports = (appInfo) => {
/** /**
* 开发者工具 * 开发者工具
*/ */
config.openDevTools = true; config.openDevTools = false;
/** /**
* 应用程序顶部菜单 * 应用程序顶部菜单

View File

@@ -19,6 +19,7 @@ VITE_API_URL=/api
# 开发环境跨域代理,支持配置多个 # 开发环境跨域代理,支持配置多个
#VITE_PROXY=[["/api","http://127.0.0.1:18092/"]]
VITE_PROXY=[["/api","http://192.168.1.124:18092/"]] VITE_PROXY=[["/api","http://192.168.1.124:18092/"]]
# VITE_PROXY=[["/api","http://192.168.1.125:18092/"]] #VITE_PROXY=[["/api","http://192.168.1.125:18092/"]]
# VITE_PROXY=[["/api","http://192.168.1.138:8080/"]]张文 # VITE_PROXY=[["/api","http://192.168.1.138:8080/"]]张文

View File

@@ -22,4 +22,5 @@ VITE_DROP_CONSOLE=true
VITE_PWA=true VITE_PWA=true
# 线上环境接口地址 # 线上环境接口地址
#VITE_API_URL="/api" # 打包时用
VITE_API_URL="http://192.168.1.125:18092/" VITE_API_URL="http://192.168.1.125:18092/"

11
frontend/.prettierrc Normal file
View File

@@ -0,0 +1,11 @@
{
"singleQuote": true,
"trailingComma": "none",
"tabWidth": 4,
"printWidth": 120,
"useTabs": false,
"semi": false,
"arrowParens": "avoid",
"endOfLine": "lf",
"htmlWhitespaceSensitivity": "ignore"
}

View File

@@ -14,7 +14,10 @@
}, },
"dependencies": { "dependencies": {
"@element-plus/icons-vue": "^2.3.1", "@element-plus/icons-vue": "^2.3.1",
"@types/event-source-polyfill": "^1.0.5",
"@vue-flow/core": "^1.45.0",
"@vueuse/core": "^10.4.1", "@vueuse/core": "^10.4.1",
"autofit.js": "^3.2.8",
"axios": "^1.7.3", "axios": "^1.7.3",
"crypto-js": "^4.2.0", "crypto-js": "^4.2.0",
"dayjs": "^1.11.9", "dayjs": "^1.11.9",
@@ -22,9 +25,12 @@
"echarts": "^5.4.3", "echarts": "^5.4.3",
"echarts-liquidfill": "^3.1.0", "echarts-liquidfill": "^3.1.0",
"element-plus": "^2.7.8", "element-plus": "^2.7.8",
"event-source-polyfill": "^1.0.31",
"html2canvas": "^1.4.1",
"md5": "^2.3.0", "md5": "^2.3.0",
"mitt": "^3.0.1", "mitt": "^3.0.1",
"mkdirp": "^3.0.1", "mkdirp": "^3.0.1",
"node-forge": "^1.3.1",
"nprogress": "^0.2.0", "nprogress": "^0.2.0",
"pinia": "^2.2.1", "pinia": "^2.2.1",
"pinia-plugin-persistedstate": "^3.2.1", "pinia-plugin-persistedstate": "^3.2.1",
@@ -41,11 +47,14 @@
"devDependencies": { "devDependencies": {
"@rushstack/eslint-patch": "^1.8.0", "@rushstack/eslint-patch": "^1.8.0",
"@tsconfig/node20": "^20.1.4", "@tsconfig/node20": "^20.1.4",
"@types/html2canvas": "^1.0.0",
"@types/md5": "^2.3.2", "@types/md5": "^2.3.2",
"@types/node": "^20.14.14", "@types/node": "^20.14.14",
"@types/node-forge": "^1.3.11",
"@types/nprogress": "^0.2.0", "@types/nprogress": "^0.2.0",
"@types/qs": "^6.9.8", "@types/qs": "^6.9.8",
"@types/sortablejs": "^1.15.2", "@types/sortablejs": "^1.15.2",
"@types/xlsx": "^0.0.36",
"@typescript-eslint/eslint-plugin": "^6.7.0", "@typescript-eslint/eslint-plugin": "^6.7.0",
"@typescript-eslint/parser": "^6.7.0", "@typescript-eslint/parser": "^6.7.0",
"@vitejs/plugin-vue": "^5.0.5", "@vitejs/plugin-vue": "^5.0.5",
@@ -67,11 +76,11 @@
"unplugin-auto-import": "^0.18.3", "unplugin-auto-import": "^0.18.3",
"unplugin-vue-components": "^0.27.4", "unplugin-vue-components": "^0.27.4",
"unplugin-vue-setup-extend-plus": "^1.0.0", "unplugin-vue-setup-extend-plus": "^1.0.0",
"vite": "^5.3.1", "vite": "^5.4.19",
"vite-plugin-compression": "^0.5.1", "vite-plugin-compression": "^0.5.1",
"vite-plugin-eslint": "^1.8.1", "vite-plugin-eslint": "^1.8.1",
"vite-plugin-html": "^3.2.0", "vite-plugin-html": "^3.2.0",
"vite-plugin-node-polyfills": "^0.22.0", "vite-plugin-node-polyfills": "^0.24.0",
"vite-plugin-pwa": "^0.16.5", "vite-plugin-pwa": "^0.16.5",
"vite-plugin-svg-icons": "^2.0.1", "vite-plugin-svg-icons": "^2.0.1",
"vue-tsc": "^2.0.21" "vue-tsc": "^2.0.21"

View File

@@ -1,27 +1,25 @@
<template> <template>
<!--element-plus语言国际化全局修改为中文--> <!--element-plus语言国际化全局修改为中文-->
<el-config-provider <el-config-provider :locale="locale" :size="assemblySize" :button="buttonConfig">
:locale='locale' <router-view />
:size='assemblySize' </el-config-provider>
:button='buttonConfig'
>
<router-view :key='$route.fullPath' />
</el-config-provider>
</template> </template>
<script lang='ts' setup> <script lang="ts" setup>
defineOptions({ import autofit from 'autofit.js'
name: 'App',
})
import { useI18n } from 'vue-i18n' import { useI18n } from 'vue-i18n'
import { getBrowserLang } from '@/utils' import { getBrowserLang } from '@/utils'
import { useTheme } from '@/hooks/useTheme' import { useTheme } from '@/hooks/useTheme'
import { ElConfigProvider } from 'element-plus' import { ElConfigProvider } from 'element-plus'
import { LanguageType } from './stores/interface' import { type LanguageType } from './stores/interface'
import { useGlobalStore } from '@/stores/modules/global' import { useGlobalStore } from '@/stores/modules/global'
import en from 'element-plus/es/locale/lang/en' import en from 'element-plus/es/locale/lang/en'
import zhCn from 'element-plus/es/locale/lang/zh-cn' import zhCn from 'element-plus/es/locale/lang/zh-cn'
defineOptions({
name: 'App'
})
const globalStore = useGlobalStore() const globalStore = useGlobalStore()
// init theme // init theme
@@ -31,16 +29,26 @@ initTheme()
// init language // init language
const i18n = useI18n() const i18n = useI18n()
onMounted(() => { onMounted(() => {
const language = globalStore.language ?? getBrowserLang() const language = globalStore.language ?? getBrowserLang()
i18n.locale.value = language i18n.locale.value = language
globalStore.setGlobalState('language', language as LanguageType) globalStore.setGlobalState('language', language as LanguageType)
// 自动适配
autofit.init({
el: '#app',
//dh: 720 * 1,
//dw: 1280 * 1.2,
dw: 1920 / 1.5,
dh: 1080 / 1.2,
resize: true,
limit: 0.1
})
}) })
// element language // element language
const locale = computed(() => { const locale = computed(() => {
if (globalStore.language == 'zh') return zhCn if (globalStore.language == 'zh') return zhCn
if (globalStore.language == 'en') return en if (globalStore.language == 'en') return en
return getBrowserLang() == 'zh' ? zhCn : en return getBrowserLang() == 'zh' ? zhCn : en
}) })
// element assemblySize // element assemblySize

View File

@@ -0,0 +1,240 @@
export namespace CheckData {
export interface DataCheck {
scriptName: string
errorSysId: string
dataRule: string
deviceName: string
chnNum: string
deviceId: string
num?: string | number
}
export interface PhaseCheckResult {
// 检测源定值-标准值
resultData: number
// 装置原始数据-被检值
data: number
// 误差值
errorData: number
// 第几次谐波
num?: number
//符合、不符合
isData?: number
//最大误差值
radius?: string
unit?: string
}
export interface DataItem {
num: number
isData: number
data: number
resultData: number
radius: string
errorData: number
unit: string
}
export interface TableRow {
isData: number
harmNum: number
radius: string
dataA: DataItem
dataB: DataItem
dataC: DataItem
dataT: DataItem | null
unit: string
timeDev?: string
uaDev?: string | number
ubDev?: string | number
ucDev?: string | number
utDev?: string | number
uaStdDev?: string | number
ubStdDev?: string | number
ucStdDev?: string | number
utStdDev?: string | number
}
/**
* 用于定义 查看(设备)通道检测结果响应数据 类型
*/
export interface ResCheckResult {
dataA?: PhaseCheckResult | null
dataB?: PhaseCheckResult | null
dataC?: PhaseCheckResult | null
dataT?: PhaseCheckResult | null
// 第几次谐波
//num: number | null,
//符合、不符合
isData?: number
//最大误差值
radius?: string
//单位
unit?: string
}
/**
* 用于定义 查看(设备)通道检测结果表格展示数据 类型
*/
export interface CheckResult {
stdA?: string
dataA?: string
errorA?: string
maxErrorA?: string
isDataA?: number
unitA?: string
stdB?: string
dataB?: string
errorB?: string
maxErrorB?: string
isDataB?: number
unitB?: string
stdC?: string
dataC?: string
errorC?: string
maxErrorC?: string
isDataC?: number
unitC?: string
stdT?: string
dataT?: string
errorT?: string
maxErrorT?: string
isDataT?: number
unitT?: string
//最大误差值
maxError?: string
unit?: string
//符合、不符合
result?: number
}
/**
* 用于定义 具体通道的原始数据类型
*/
export interface RawDataItem {
time?: string
harmNum?: number | null
dataA?: string
dataB?: string
dataC?: string
dataT?: string
unit?: string | null
}
export interface Device {
deviceId: string //装置序号Id
deviceName: string //设备名称
chnNum: number //设备通道数
planId: string //计划Id
devType: string //设备类型
devVolt: number //设备电压
devCurr: number //设备电流
factorFlag: number //是否支持系数校准
checkResult: number //检测结果
chnNumList: string[] //连线存储数据
}
// 用来描述检测脚本类型
export interface ScriptItem {
id: string
code: string
scriptName: string
}
// 用来描述 检测数据-左侧树结构
export interface TreeItem {
id: string | null
scriptTypeName: string | null
sourceDesc: string | null
harmNum: number | null
index: number | null
fly: number | null
children: TreeItem[] | null
}
// 用来描述 通道检测结果
export enum ChnCheckResultEnum {
UNKNOWN = -1,
LOADING = 0,
SUCCESS = 1,
FAIL = 2,
TIMEOUT = 3,
ERRORDATA = 4,
NOT_PART_IN_ERROR = 5
}
export interface DeviceCheckResult {
deviceId: string
deviceName: string
code?: string
chnResult: ChnCheckResultEnum[] //通道检测结果
}
//用来描述 某个脚本测试项对所有通道的检测结果
export interface ScriptChnItem {
scriptType: string
scriptName?: string //可以不要该属性,有点多余
code?: string
// 设备
devices: Array<DeviceCheckResult>
}
export enum ButtonColorEnum {
INFO = '#909399',
LOADING = 'var(--el-color-primary)',
SUCCESS = '#91cc75',
WARNING = '#e6a23c',
DANGER = '#f56c6c'
}
/**
* 用于描述 (设备)通道检测结果展示的按钮类型
*/
export interface ButtonResult {
color: ButtonColorEnum
icon: 'More' | 'Loading' | 'Close' | 'CircleCheckFilled' | 'Link' | 'WarnTriangleFilled' | 'Minus'
}
/**
* 用于描述 脚本检测结果展示的按钮类型
*/
export interface ScriptChnViewItem {
scriptType: string
scriptName?: string //脚本项名称,可以不要该属性,有点多余
// 设备
devices: Array<{
deviceId: string
deviceName: string
chnResult: ButtonResult[]
}>
}
/**
* 定义检测日志类型
*/
export interface LogItem {
type: 'info' | 'warning' | 'error'
log: string
}
/**
* 定义手动检测时,勾选的测试项
*/
export interface SelectTestItem {
preTest: boolean
timeTest: boolean
channelsTest: boolean
test: boolean
}
//描述比对式检测项描述
export interface CompareTestItem {
id: string
code: string
name: string
}
}

View File

@@ -0,0 +1,143 @@
import { pa } from 'element-plus/es/locale/index.mjs';
import http from '@/api'
import {CheckData} from '@/api/check/interface'
export const getBigTestItem = (params: {
reCheckType: number
planId: string
devIds: string[]
patternId: string
}) => {
return http.post(`/adPlan/getBigTestItem`, params, {loading: false})
}
export const getScriptList = (params: {
devId:string,
chnNum:number,
num:number
}) => {
return http.post('/result/getCheckItem', params, {loading: false})
}
/**
* 获取弹出框表单数据
* @param params 当为scriptType为null时表示查询所有脚本类型否则只查询指定脚本类型。当为chnNum为-1时表示查询所有通道否则只查询指定通道。
*/
export const getFormData = (params: {
planId: string
deviceId: string
chnNum: string
scriptType: string | null
}) => {
return http.post('/result/formContent/', params, {loading: false})
}
/**
* 获取树形结构数据
* @param params
*/
export const getTreeData = (params: {
scriptId?: string
devId?: string
devNum?: string
scriptType?: string | null
code?: string
}) => {
return http.post<CheckData.TreeItem[]>('/result/treeData/', params, {loading: false})
}
/**
* 获取检查数据
* @param params
*/
export const getTableData = (params: {
scriptType: string | null
scriptId: string
devId: string
devNum: string
code: string
index: number
}) => {
return http.post('/result/resultData/', params, {loading: false})
}
export const exportRawData = (params: {
scriptType: string | null
scriptId: string
devId: string
devNum: string
code: string
index: number
}) => {
return http.download('/result/exportRawData', params, {loading: false})
}
/**
* 重新计算
* @param params
*/
export const reCalculate = (params: {
planId: string
scriptId: string
errorSysId: string
deviceId: string
code: string
patternId: string
}) => {
return http.post('/result/reCalculate', params, {loading: true})
}
/**
* 获取数据获取基本信息
* @param params
*/
export const getContrastFormContent = (params: {
planId: string
scriptType: string
deviceId: string
chnNum: string
num: number | null
patternId: string
}) => {
return http.post('/result/getContrastFormContent', params, {loading: false})
}
/**
* 获取检测结果
* @param params
*/
export const getContrastResult = (params: {
planId: string
scriptType: string
deviceId: string
chnNum: string | number
num: number | string | null
waveNum: number | null
isWave: boolean
patternId: string
}) => {
return http.post('/result/getContrastResult', params, {loading: true})
}
/**
* 更换误差体系
* @param params
*/
export const changeErrorSystem = (params: {
planId: string
scriptId: string
errorSysId: string
deviceId: string
code: string
patternId: string
}) => {
return http.post('/result/changeErrorSystem', params, {loading: true})
}
/**
* 删除(误差体系切换时产生的)临时表
* @param code
*/
export const deleteTempTable = (code: string) => {
return http.get(`/result/deleteTempTable?code=${code}`, null, {loading: false})
}

View File

@@ -0,0 +1,24 @@
import type { controlSource } from '@/api/device/interface/controlSource'
import http from '@/api'
/**
* @name 程控源管理模块
*/
//通讯校验
export const checkSimulate = (params: controlSource.ResControl) => {
return http.post(`/prepare/ytxCheckSimulate`,params)
}
//启动
export const startSimulateTest = (params: controlSource.ResControl) => {
return http.post(`prepare/startSimulateTest`,params,{loading:false})
}
//停止
export const closeSimulateTest = (params: controlSource.ResControl) => {
return http.post(`/prepare/closeSimulateTest`,params,{loading:false})
}

View File

@@ -0,0 +1,29 @@
import type { DevType } from '@/api/device/interface/devType'
import http from '@/api'
/**
* @name 设备类型管理模块
*/
//获取设备类型
export const getDevTypeList = (params: DevType.ReqPqDevTypeParams) => {
return http.post(`/devType/list`, params)
}
//添加设备类型
export const addDevType = (params: DevType.ResPqDevType) => {
return http.post(`/devType/add`, params)
}
//编辑设备类型
export const updateDevType = (params: DevType.ResPqDevType) => {
return http.post(`/devType/update`, params)
}
//删除设备类型
export const deleteDevType = (params: string[]) => {
return http.post(`/devType/delete`, params)
}

View File

@@ -48,64 +48,64 @@ const data = [
{ {
id: '1', //装置序号ID id: '1', //装置序号ID
name: '240001', //设备名称 name: '240001', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '2', //装置序号ID id: '2', //装置序号ID
name: '240002', //设备名称 name: '240002', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '3', //装置序号ID id: '3', //装置序号ID
name: '240003', //设备名称 name: '240003', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '4', //装置序号ID id: '4', //装置序号ID
name: '240004', //设备名称 name: '240004', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '5', //装置序号ID id: '5', //装置序号ID
name: '240005', //设备名称 name: '240005', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '有不合格项', //检测结果 check_Result: '不符合', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 1, //复检次数
}, },
{ {
id: '6', //装置序号ID id: '6', //装置序号ID
name: '240006', //设备名称 name: '240006', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '有不合格项', //检测结果 check_Result: '不符合', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -114,9 +114,9 @@ const data = [
{ {
id: '7', //装置序号ID id: '7', //装置序号ID
name: '240007', //设备名称 name: '240007', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '全部合格', //检测结果 check_Result: '符合', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -125,20 +125,20 @@ const data = [
{ {
id: '8', //装置序号ID id: '8', //装置序号ID
name: '240008', //设备名称 name: '240008', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '全部合格', //检测结果 check_Result: '符合', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 1, //复检次数
}, },
{ {
id: '9', //装置序号ID id: '9', //装置序号ID
name: '240009', //设备名称 name: '240009', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '有不合格项', //检测结果 check_Result: '不符合', //检测结果
report_State: '已生成', //报告状态 report_State: '已生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -147,9 +147,9 @@ const data = [
{ {
id: '10', //装置序号ID id: '10', //装置序号ID
name: '240010', //设备名称 name: '240010', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '全部合格', //检测结果 check_Result: '符合', //检测结果
report_State: '已生成', //报告状态 report_State: '已生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -158,9 +158,9 @@ const data = [
{ {
id: '11', //装置序号ID id: '11', //装置序号ID
name: '240011', //设备名称 name: '240011', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '全部合格', //检测结果 check_Result: '符合', //检测结果
report_State: '已生成', //报告状态 report_State: '已生成', //报告状态
document_State: '已归档', //归档状态 document_State: '已归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -169,9 +169,9 @@ const data = [
{ {
id: '12', //装置序号ID id: '12', //装置序号ID
name: '240012', //设备名称 name: '240012', //设备名称
dev_Type: 'PQS882B4',//设备类型 dev_Type: 'PQS-882B4',//设备类型
dev_Chns: 4, //设备通道数 dev_Chns: 4, //设备通道数
check_Result: '全部合格', //检测结果 check_Result: '符合', //检测结果
report_State: '已生成', //报告状态 report_State: '已生成', //报告状态
document_State: '已归档', //归档状态 document_State: '已归档', //归档状态
check_State:'检测完成',//检测状态 check_State:'检测完成',//检测状态
@@ -180,258 +180,90 @@ const data = [
{ {
id: '13', //装置序号ID id: '13', //装置序号ID
name: '240013', //设备名称 name: '240013', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '14', //装置序号ID id: '14', //装置序号ID
name: '240014', //设备名称 name: '240014', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '15', //装置序号ID id: '15', //装置序号ID
name: '240013', //设备名称 name: '240015', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '16', //装置序号ID id: '16', //装置序号ID
name: '240014', //设备名称 name: '240016', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '17', //装置序号ID id: '17', //装置序号ID
name: '240017', //设备名称 name: '240017', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '18', //装置序号ID id: '18', //装置序号ID
name: '240018', //设备名称 name: '240018', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '19', //装置序号ID id: '19', //装置序号ID
name: '240019', //设备名称 name: '240019', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
{ {
id: '20', //装置序号ID id: '20', //装置序号ID
name: '240020', //设备名称 name: '240020', //设备名称
dev_Type: 'PQS882A',//设备类型 dev_Type: 'PQS-882A',//设备类型
dev_Chns: 1, //设备通道数 dev_Chns: 1, //设备通道数
check_Result: '未检', //检测结果 check_Result: '未检', //检测结果
report_State: '未生成', //报告状态 report_State: '未生成', //报告状态
document_State: '未归档', //归档状态 document_State: '未归档', //归档状态
check_State:'未检',//检测状态 check_State:'未检',//检测状态
reCheck_Num: 0, //复检次数 reCheck_Num: 0, //复检次数
}, },
] ]
// const plan_devicedata = [
// {
// id: '1', //装置序号ID
// name: '模拟装置1', //设备名称
// dev_Type: 'PQS882A',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 0, //复检次数
// },
// {
// id: '2', //装置序号ID
// name: '模拟装置2', //设备名称
// dev_Type: 'PQS882B4',//设备类型
// dev_Chns: 4, //设备通道数
// check_Result: '/', //检测结果
// report_State: '未生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'未检',//检测状态
// reCheck_Num: 0, //复检次数
// },
// {
// id: '3', //装置序号ID
// name: '模拟装置3', //设备名称
// dev_Type: 'PQS882B4',//设备类型
// dev_Chns: 4, //设备通道数
// check_Result: '/', //检测结果
// report_State: '未生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测中',//检测状态
// reCheck_Num: 0, //复检次数
// },
// {
// id: '4', //装置序号ID
// name: '模拟装置4', //设备名称
// dev_Type: 'PQS882B4',//设备类型
// dev_Chns: 4, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '未生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 1, //复检次数
// },
// {
// id: '5', //装置序号ID
// name: '中电测试装置', //设备名称
// dev_Type: 'PMC-680M-22-22-00-115ANBC',//设备类型
// dev_Chns: 4, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '未生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 1, //复检次数
// },
// {
// id: '6', //装置序号ID
// name: '易司拓测试装置1', //设备名称
// dev_Type: 'E703A',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 1, //复检次数
// },
// {
// id: '7', //装置序号ID
// name: '易司拓测试装置2', //设备名称
// dev_Type: 'E703A',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 1, //复检次数
// },
// {
// id: '8', //装置序号ID
// name: '山大电力测试装置1', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 1, //复检次数
// },
// {
// id: '9', //装置序号ID
// name: '山大电力测试装置2', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '10', //装置序号ID
// name: '山大电力测试装置2', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '11', //装置序号ID
// name: '山大电力测试装置2', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '12', //装置序号ID
// name: '山大电力测试装置2', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '13', //装置序号ID
// name: '山大电力测试装置2', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '14', //装置序号ID
// name: '山大电力测试装置3', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// {
// id: '15', //装置序号ID
// name: '山大电力测试装置4', //设备名称
// dev_Type: 'SDL-3002C',//设备类型
// dev_Chns: 1, //设备通道数
// check_Result: '不合格', //检测结果
// report_State: '已生成', //报告状态
// document_State: '未归档', //归档状态
// check_State:'检测完成',//检测状态
// reCheck_Num: 2, //复检次数
// },
// ]
export default {data,plan_devicedata} export default {data,plan_devicedata}

View File

@@ -1,4 +1,5 @@
import type { Device } from '@/api/device/interface/device' import { pa } from 'element-plus/es/locale/index.mjs';
import type {Device} from '@/api/device/interface/device'
import http from '@/api' import http from '@/api'
/** /**
@@ -7,33 +8,67 @@ import http from '@/api'
//获取被检设备 //获取被检设备
export const getPqDevList = (params: Device.ReqPqDevParams) => { export const getPqDevList = (params: Device.ReqPqDevParams) => {
return http.post(`/pqDev/list`, params) return http.post(`/pqDev/list`, params)
} }
//添加被检设备 //添加被检设备
export const addPqDev = (params: Device.ResPqDev) => { export const addPqDev = (params: Device.ResPqDev) => {
return http.post(`/pqDev/add`, params) return http.post(`/pqDev/add`, params)
} }
//编辑被检设备 //编辑被检设备
export const updatePqDev = (params: Device.ResPqDev) => { export const updatePqDev = (params: Device.ResPqDev) => {
return http.post(`/pqDev/update`, params) return http.post(`/pqDev/update`, params)
} }
//删除被检设备 //删除被检设备
export const deletePqDev = (params: string[]) => { export const deletePqDev = (params: string[]) => {
return http.post(`/pqDev/delete`, params) return http.post(`/pqDev/delete`, params)
} }
//导出被检设备 //导出被检设备
export const exportPqDev=(params: Device.ReqPqDevParams)=>{ export const exportPqDev = (params: Device.ReqPqDevParams) => {
return http.download(`/pqDev/export`, params) return http.download(`/pqDev/export`, params)
} }
export const downloadTemplate = () => { // 下载导入文件模板
return http.download(`/pqDev/downloadTemplate`) export const downloadTemplate = (params: { pattern: string }) => {
} return http.download(`/pqDev/downloadTemplate`,params)
//导入被检设备
export const importPqDev=(params: Device.ReqPqDevParams)=>{
return http.upload(`/pqDev/import`, params)
} }
//导入被检设备
export const importPqDev = (params: Device.ReqPqDevParams) => {
return http.uploadExcel(`/pqDev/import`, params)
}
//导入比对式被检设备
export const importContrastPqDev = (params: Device.ReqPqDevParams) => {
return http.upload(`/pqDev/importContrast`, params)
}
// //导出灿能二楼设备
// export const exportCNDev = (params: Device.ReqPqDevParams) => {
// return http.download(`/pqDev/exportCNDev`, params)
// }
//
// // 下载灿能二楼设备导入文件模板
// export const downloadCNDevTemplate = () => {
// return http.download(`/pqDev/downloadCNDevTemplate`)
// }
//
// //导入灿能二楼设备
// export const importCNDev = (params: Device.ReqPqDevParams) => {
// return http.uploadExcel(`/pqDev/importCNDev`, params)
// }
export const getPqDevById = (params: Device.ReqPqDevParams) => {
return http.get(`/pqDev/getById?id=${params.id}`)
}
//根据设备类型决定电源、icd、模板、通道数、额定电压、额定电流
export const getPqDev = () => {
return http.get(`/devType/listAll`)
}
export const getSelectOptions = (params:{ pattern: string }) => {
return http.get(`/pqDev/getSelectOptions`, params)
}

View File

@@ -30,183 +30,394 @@ const errordata = ref<ErrorSystem.ErrorSystemList[]>([
}, },
]) ])
const errordetail = ref<ErrorSystem.Error_detail[]>([ const errorADetail = ref<ErrorSystem.Error_detail[]>([
{ {
measured: '电压偏差', col1: '电压偏差',
col2: '',
deviceLevel: 'A', deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: '10%~150%标称电压', condition: '10%~150%标称电压',
maxErrorValue: '±0.1%Un' maxErrorValue: '±0.1%Un'
}, },
{ {
measured: '频率偏差', col1:'频率偏差',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'频率', measurementType:'频率',
condition: '42.5~57.5 Hz', condition: '42.5~57.5 Hz',
maxErrorValue: '±0.01Hz' maxErrorValue: '±0.01Hz'
}, },
{ {
measured: '三相不平衡度', col1:'三相不平衡度',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: '0.5%~5%', condition: '0.5%~5%',
maxErrorValue: '±0.15%' maxErrorValue: '±0.15%'
}, },
{ {
measured: '三相不平衡度', col1:'三相不平衡度',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: '5%(不包含)~ 40%', condition: '5%(不包含)~ 40%',
maxErrorValue: '±0.3%' maxErrorValue: '±0.3%'
}, },
{ {
measured: '三相不平衡度', col1:'三相不平衡度',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: '', condition: '',
maxErrorValue: '±1%' maxErrorValue: '±1%'
}, },
{ {
measured: '电压波动幅值', col1:'电压波动',
deviceLevel: 'A', col2: '电压幅值',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: '', condition: '',
maxErrorValue: '±0.2%Un' maxErrorValue: '±0.2%Un'
}, },
{ {
measured: '电压波动持续时间', col1:'电压波动',
deviceLevel: 'A', col2: '持续时间',
deviceLevel: 'A',
measurementType:'时间', measurementType:'时间',
condition: '', condition: '',
maxErrorValue: '±1周波' maxErrorValue: '±1周波'
}, },
{ {
measured: '闪变', col1:'闪变',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'短时间闪变', measurementType:'短时间闪变',
condition: 'Pst0.2~10', condition: 'Pst0.2~10',
maxErrorValue: '±5%' maxErrorValue: '±5%'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: 'Uh≥1%Un', condition: 'Uh≥1%Un',
maxErrorValue: '±5%Uh' maxErrorValue: '±5%Uh'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: 'Uh1%Un', condition: 'Uh1%Un',
maxErrorValue: '±0.05%Un' maxErrorValue: '±0.05%Un'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: 'In≥3%In', condition: 'In≥3%In',
maxErrorValue: '±5%Ih' maxErrorValue: '±5%Ih'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: 'In3%In', condition: 'In3%In',
maxErrorValue: '±0.15%In' maxErrorValue: '±0.15%In'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'相角', measurementType:'相角',
condition: '', condition: '',
maxErrorValue: 'h≤5,±1°*h' maxErrorValue: 'h≤5,±1°*h'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'相角', measurementType:'相角',
condition: '', condition: '',
maxErrorValue: 'h>5,±5°' maxErrorValue: 'h>5,±5°'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'功率', measurementType:'功率',
condition: 'Ph≥150W', condition: 'Ph≥150W',
maxErrorValue: '±1%Ph' maxErrorValue: '±1%Ph'
}, },
{ {
measured: '谐波和间谐波', col1:'谐波和间谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'功率', measurementType:'功率',
condition: 'Ph150W', condition: 'Ph150W',
maxErrorValue: '±1.5W' maxErrorValue: '±1.5W'
}, },
{ {
measured: '高频次谐波', col1:'高频次谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: 'Uh≥1%Un', condition: 'Uh≥1%Un',
maxErrorValue: '±5%Uh' maxErrorValue: '±5%Uh'
}, },
{ {
measured: '高频次谐波', col1:'高频次谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: 'Uh1%Un', condition: 'Uh1%Un',
maxErrorValue: '±0.05%Un' maxErrorValue: '±0.05%Un'
}, },
{ {
measured: '高频次谐波', col1:'高频次谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: 'In≥3%In', condition: 'In≥3%In',
maxErrorValue: '±5%Ih' maxErrorValue: '±5%Ih'
}, },
{ {
measured: '高频次谐波', col1:'高频次谐波',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: 'In3%In', condition: 'In3%In',
maxErrorValue: '±0.15%In' maxErrorValue: '±0.15%In'
}, },
{ {
measured: '功率', col1:'功率',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'功率', measurementType:'功率',
condition: '', condition: '',
maxErrorValue: '±0.5%' maxErrorValue: '±0.5%'
}, },
{ {
measured: '电流', col1:'电流',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: 'I≥0.05In', condition: 'I≥0.05In',
maxErrorValue: '±0.5%' maxErrorValue: '±0.5%'
}, },
{ {
measured: '电流', col1:'电流',
deviceLevel: 'A', col2: '',
deviceLevel: 'A',
measurementType:'电流', measurementType:'电流',
condition: '0.01In≤I0.05In', condition: '0.01In≤I0.05In',
maxErrorValue: '±1%' maxErrorValue: '±1%'
}, },
{ {
measured: '暂态电压幅值', col1: '电压暂降、电压暂升和短时中断',
deviceLevel: 'A', col2: '电压幅值',
deviceLevel: 'A',
measurementType:'电压', measurementType:'电压',
condition: '', condition: '',
maxErrorValue: '±0.2%Un' maxErrorValue: '±0.2%Un'
}, },
{ {
measured: '暂态持续时间', col1: '电压暂降、电压暂升和短时中断',
deviceLevel: 'A', col2: '持续时间',
deviceLevel: 'A',
measurementType:'时间', measurementType:'时间',
condition: '', condition: '',
maxErrorValue: '±1周波' maxErrorValue: '±1周波'
}, },
]) ])
const errorSDetail = ref<ErrorSystem.Error_detail[]>([
{
col1: '电压偏差',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: '20%~120%标称电压',
maxErrorValue: '±0.5%'
},
{
col1:'频率偏差',
col2: '',
deviceLevel: 'S',
measurementType:'频率',
condition: '42.5~57.5 Hz',
maxErrorValue: '±0.05Hz'
},
{
col1:'三相不平衡度',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: '1%~5%',
maxErrorValue: '±0.2%'
},
{
col1:'三相不平衡度',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: '5%(不包含)~ 40%',
maxErrorValue: '±0.6%'
},
{
col1:'三相不平衡度',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: '',
maxErrorValue: '±1%'
},
{
col1:'电压波动',
col2: '电压幅值',
deviceLevel: 'S',
measurementType:'电压',
condition: '',
maxErrorValue: '±1.0%Un'
},
{
col1:'电压波动',
col2: '持续时间',
deviceLevel: 'S',
measurementType:'时间',
condition: '使用半波刷新方均根值',
maxErrorValue: '±1周波'
},
{
col1:'电压波动',
col2: '持续时间',
deviceLevel: 'S',
measurementType:'时间',
condition: '使用全波刷新方均根值',
maxErrorValue: '±2周波'
},
{
col1:'闪变',
col2: '',
deviceLevel: 'S',
measurementType:'短时间闪变',
condition: 'Pst0.4~4',
maxErrorValue: '±10%'
},
{
col1:'谐波和间谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: 'Uh≥3%Un',
maxErrorValue: '±5%Uh'
},
{
col1:'谐波和间谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: 'Uh3%Un',
maxErrorValue: '±0.15%Un'
},
{
col1:'谐波和间谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: 'In≥10%In',
maxErrorValue: '±5%Ih'
},
{
col1:'谐波和间谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: 'In10%In',
maxErrorValue: '±0.5%In'
},
{
col1:'高频次谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: 'Uh≥3%Un',
maxErrorValue: '±5%Uh'
},
{
col1:'高频次谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电压',
condition: 'Uh3%Un',
maxErrorValue: '±0.15%Un'
},
{
col1:'高频次谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: 'In≥10%In',
maxErrorValue: '±5%Ih'
},
{
col1:'高频次谐波',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: 'In10%In',
maxErrorValue: '±0.5%In'
},
{
col1:'功率',
col2: '',
deviceLevel: 'S',
measurementType:'功率',
condition: '',
maxErrorValue: '±0.5%'
},
{
col1:'电流',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: 'I≥0.05In',
maxErrorValue: '±0.5%'
},
{
col1:'电流',
col2: '',
deviceLevel: 'S',
measurementType:'电流',
condition: '0.01In≤I0.05In',
maxErrorValue: '±1%'
},
{
col1: '电压暂降、电压暂升和短时中断',
col2: '电压幅值',
deviceLevel: 'S',
measurementType:'电压',
condition: '',
maxErrorValue: '±1.0%Un'
},
{
col1: '电压暂降、电压暂升和短时中断',
col2: '持续时间',
deviceLevel: 'S',
measurementType:'时间',
condition: '使用半波刷新方均根值',
maxErrorValue: '±1周波'
},
{
col1: '电压暂降、电压暂升和短时中断',
col2: '持续时间',
deviceLevel: 'S',
measurementType:'时间',
condition: '使用全波刷新方均根值',
maxErrorValue: '±2周波'
},
])
export default {errordata,errordetail} export default {errordata,errorADetail,errorSDetail}

View File

@@ -30,5 +30,7 @@ export const deletePqErrSys = (params: string[]) => {
return http.post(`/pqErrSys/delete`, params) return http.post(`/pqErrSys/delete`, params)
} }
//复制误差体系
export const copyPqErrSys = (params: ErrorSystem.ErrorSystemList) => {
return http.get(`/pqErrSys/copy?id=${params.id}`)
}

View File

@@ -0,0 +1,38 @@
import type { ICD } from '@/api/device/interface/icd'
import http from '@/api'
import { pa } from 'element-plus/es/locale/index.mjs'
/**
* @name ICD管理模块
*/
//获取ICD分页
export const getICDList = (params: ICD.ReqICDParams) => {
return http.post(`/icd/list`,params)
}
//获取ICD
export const getICDAllList = (params: ICD.ResICD) => {
return http.get(`/icd/listAll`,params)
}
//添加ICD
export const addICD = (params: ICD.ResICD) => {
return http.post(`/icd/add`, params)
}
//编辑ICD
export const updateICD = (params: ICD.ResICD) => {
return http.post(`/icd/update`, params)
}
//删除ICD
export const deleteICD = (params: string[]) => {
return http.post(`/icd/delete`, params)
}

View File

@@ -0,0 +1,15 @@
import type { ReqPage,ResPage } from '@/api/interface'
// 被检设备模块
export namespace controlSource {
/**
* 被检设备新增、修改、根据id查询返回的对象
*/
export interface ResControl {
userPageId: string;
scriptId: string;
scriptIndex: number;
sourceId: string;
}
}

View File

@@ -0,0 +1,41 @@
import type { ReqPage, ResPage } from '@/api/interface'
// 设备类型模块
export namespace DevType {
/**
* 设备类型数据表格分页查询参数
*/
export interface ReqPqDevTypeParams extends ReqPage {
id: string; // 装置序号id 必填
devType?: string; // 设备名称
createTime?: string; //创建时间
}
/**
* 设备类型新增、修改、根据id查询返回的对象
*/
export interface ResPqDevType {
id: string; //设备类型ID
name: string;//设备类型名称
icd: string| null;//设备关联的ICD
power: string| null;//工作电源
devVolt: number; //额定电压V
devCurr: number; //额定电流A
devChns: number; //设备通道数
reportName: string| null;//报告模版名称
state: number;
waveCmd:string| null;//录波指令
createBy?: string| null; //创建用户
createTime?: string| null; //创建时间
updateBy?: string| null; //更新用户
updateTime?: string| null; //更新时间
}
/**
* 设备类型表格查询分页返回的对象;
*/
export interface ResPqDevTypePage extends ResPage<ResPqDevType> {
}
}

View File

@@ -1,63 +1,119 @@
import type { ReqPage,ResPage } from '@/api/interface' import type { ReqPage, ResPage } from '@/api/interface'
import type { Monitor } from './monitor'
// 被检设备模块 // 被检设备模块
export namespace Device { export namespace Device {
/**
* 电能质量指标字典数据表格分页查询参数
*/
export interface ReqPqDevParams extends ReqPage{
id: string; // 装置序号id 必填
devType?: string; // 设备名称
createTime?:string; //创建时间
}
/** /**
* 被检设备新增、修改、根据id查询返回的对象 * 被检设备表格分页查询参数
*/ */
export interface ResPqDev { export interface ReqPqDevParams extends ReqPage {
id: string; //装置序号ID id: string // 装置序号id 必填
name: string; //设备名称 name: string //设备名称
pattern: string; //设备模式 模拟 数字 比对 devType?: string // 设备名称
devType: string;//设备类型 createTime?: string //创建时间
devChns: number; //设备通道数 pattern: string
devVolt: number; //额定电压V
devCurr: number; //额定电流A
manufacturer: string;//生产厂家
createDate: string; //生产日期
createId: string; //出厂编号
hardwareVersion: string; //固件版本
softwareVersion: string; //软件版本
protocol:string; //通讯协议
ip: string; //IP地址
port: number; //端口号
encryptionFlag: number; //装置是否为加密版本
series?: string| null; //装置识别码3ds加密
devKey?: string| null; //装置秘钥3ds加密
sampleID?: string| null; //样品编号
arrivedDate?: string; //送样日期
cityName?: string| null; //所属地市名称
gDName?: string| null; //所属供电公司名称
subName?: string| null; //所属电站名称
checkState?: number| null; //检测状态
checkResult?: number| null; //检测结果
reportState?: number| null; //报告状态
documentState?: number| null; //归档状态
reportPath?: string| null; //报告路径
qRCode?: string| null; //设备关键信息二维码
reCheckNum: number; //复检次数
state: number; //状态
createBy?: string| null; //创建用户
createTime?: string| null; //创建时间
updateBy?: string| null; //更新用户
updateTime?: string| null; //更新时间
} }
/** /**
* 被检设备表格查询分页返回的对象; * 被检设备表格分页查询参数
*/ */
export interface ResPqDevPage extends ResPage<ResPqDev> { export interface ReqDevReportParams extends ReqPage {
planId?: string // 计划id
devId?: string // 装置id
scriptId?: string // 脚本id
planCode?: string
devIdList?: string[] // 装置id列表
}
} /**
} * 被检设备新增、修改、根据id查询返回的对象
*/
export interface ResPqDev {
id: string //装置序号ID
name: string //设备名称
pattern: string //设备模式 模拟 数字 比对
devType: string //设备类型
manufacturer?: string | null //生产厂家
createDate: string //生产日期
createId: string //出厂编号
hardwareVersion: string //固件版本
softwareVersion: string //软件版本
protocol: string //通讯协议
ip: string //IP地址
port: number //端口号
encryptionFlag: number //装置是否为加密版本
series?: string | null //装置识别码3ds加密
devKey?: string | null //装置秘钥3ds加密
sampleId?: string | null //样品编号
arrivedDate?: string //送样日期
cityName?: string | null //所属地市名称
gdName?: string | null //所属供电公司名称
subName?: string | null //所属电站名称
reportPath?: string | null //报告路径
planId?: string //检测计划Id
factorFlag?: number //是否支持系数校准(0:不支持,1:支持)
preinvestmentPlan: string | null //预投计划
delegate: string | null //委托方
inspectChannel?: string[] | string //被检通道
inspectDate?: string | null //定检日期
harmSysId?: string | null //谐波系统设备id
importFlag?: number //是否为导入设备 0否 1是
state: number //状态
createBy?: string | null //创建用户
createTime?: string | null //创建时间
updateBy?: string | null //更新用户
updateTime?: string | null //更新时间
devChns: number //设备通道数
devVolt: number //额定电压V
devCurr: number //额定电流A
icdId: string | null
power: string | null //工作电源
devId?: number
checkState?: number | null //检测状态(0:未检1:检测中2:检测完成 3:归档)
checkResult?: number | null //检测结果(0:不符合1:符合2:未检)
reportState?: number | null //报告状态(0:未生成1:已生成2:未检)
recheckNum: number //复检次数
timeCheckResult?: number //守时检测结果(0:不符合1:符合)
factorCheckResult?: number //系数校准结果(0:不合格1:合格2:未检)
realtimeResult?: number //实时数据结论(0:不符合1:符合2:未检)
statisticsResult?: number //统计数据结论(0:不符合1:符合2:未检)
recordedResult?: number //录波数据结论(0:不符合1:符合2:未检)
checkBy?: string | null //检测人
checkTime?: string | null //检测时间
preDetectTime?: number //预检测耗时
coefficientTime?: number //系数校准耗时
formalCheckTime?: number //正式检测耗时
boundPlanName?: string | null
assign?: number ////是否分配给检测人员 0否 1是
monitorList: Monitor.ResPqMon[]
checked: boolean // 是否已选择
disabled: boolean // 是否禁用
}
export interface SelectOption {
label: string
value: string | number
}
export interface ResDev {
id: string
name: string
icd: string
power: string
devVolt: number
devCurr: number
devChns: number
}
export interface ResTH {
temperature: number | null //温度
humidity: number | null //湿度
}
/**
* 被检设备表格查询分页返回的对象;
*/
export interface ResPqDevPage extends ResPage<ResPqDev> {}
}

View File

@@ -22,25 +22,29 @@ export namespace ErrorSystem {
// 查看详细误差体系 // 查看详细误差体系
export interface ErrorSystemDetail { export interface ErrorSystemDetail {
nextId: number; sort: number;
id:string;//误差体系子表ID id:string;//误差体系子表ID
errorSysId:string;//所属误差体系ID errorSysId:string;//所属误差体系ID
type: string;//检测脚本类型,树形字典表(没有树形表则需要拆分字段) errorType: string;//误差类型,树形字典表(没有树形表则需要拆分字段)
startValue?:number;//误差判断起始值(误差范围) scriptType: string;//脚本类型
startValue?:number | null;//误差判断起始值(误差范围)
startFlag?:number;//是否包含起始值 startFlag?:number;//是否包含起始值
endValue?:number;//;误差判断结束值(误差范围) endValue?:number | null;//;误差判断结束值(误差范围)
endFlag?:number;//是否包含结束值 endFlag?:number;//是否包含结束值
conditionType?:string;//判断条件值类型(包括值类型,绝对值、相对值) conditionType?:string;//判断条件值类型(包括值类型,绝对值、相对值)
maxErrorValue:number;//误差最大值 maxErrorValue:number;//误差最大值
errorValueType:string;//误差值类型(包括值类型绝对值、相对值1、相对值2 errorValueType:any;//误差值类型(0标称值、1标准值、2值类型
valueType:number;//值类型1绝对值、2相对值
errorUnit:string;//误差单位
} }
// 查看详细误差体系 //查看详细误差体系
// export interface Error_detail { export interface Error_detail {
// measured: string; col1: string;
// deviceLevel: string; col2: string;
// measurementType:string; deviceLevel: string;
// condition: string; measurementType:string;
// maxErrorValue: string; condition: string;
// } maxErrorValue: string;
}
} }

View File

@@ -0,0 +1,37 @@
import type { ReqPage, ResPage } from '@/api/interface'
// ICD模块
export namespace ICD {
/**
* ICD表格分页查询参数
*/
export interface ReqICDParams extends ReqPage {
id: string; // 装置序号id 必填
devType?: string; // 设备名称
createTime?: string; //创建时间
}
/**
* ICD新增、修改、根据id查询返回的对象
*/
export interface ResICD {
id: string; //icdID
name: string;//icd名称
path: string;//icd存储地址
state: number;
createBy?: string| null; //创建用户
createTime?: string| null; //创建时间
updateBy?: string| null; //更新用户
updateTime?: string| null; //更新时间
angle: number; // 是否支持电压相角、电流相角指标
usePhaseIndex: number; // 角型接线时是否使用相别的指标来进行检测
}
/**
* ICD表格查询分页返回的对象
*/
export interface ResICDPage extends ResPage<ResICD> {
}
}

View File

@@ -1,35 +1,38 @@
import type { ReqPage,ResPage } from '@/api/interface' import type { ReqPage, ResPage } from '@/api/interface'
// 被检设备模块 // 监测点模块
export namespace Monitor { export namespace Monitor {
/** /**
* 电能质量指标字典数据表格分页查询参数 * 监测点表格分页查询参数
*/ */
export interface ReqPqMonParams extends ReqPage{ export interface ReqPqMonParams extends ReqPage {
id: string; // 装置序号id 必填 id: string; // 装置序号id 必填
devType?: string; // 设备名称 name?: string; // 设备名称
createTime?:string; //创建时间
} }
/** /**
* 被检设备新增、修改、根据id查询返回的对象 * 监测点新增、修改、根据id查询返回的对象
*/ */
export interface ResPqMon { export interface ResPqMon {
id: string; //监测点ID id: string; //监测点ID
code: string; //默认与谐波系统监测点ID相同 devId: string; //所属设备ID
devId: string; //所属设备ID busbar: string;//所属母线
name: string; //所属母线 name: string; //监测点名称
num: number; //监测点序号 num: number; //线路号,监测点序号
pt: number; //PT变比 pt: string; //PT变比
ct: number; //CT变比 ct: string; //CT变比
ptType:string; //接线方式,字典表 connection: string; //接线方式,字典表
} statInterval: number; //统计间隔
harmSysId: string; //默认与谐波系统监测点ID相同
checkFlag: number;//是否参与检测0否1是
}
/** /**
* 被检设备表格查询分页返回的对象; * 监测点表格查询分页返回的对象;
*/ */
export interface ResPqMonPage extends ResPage<ResPqMon> { export interface ResPqMonPage extends ResPage<ResPqMon> {
} }
} }

View File

@@ -0,0 +1,39 @@
import type { ReqPage, ResPage } from '@/api/interface'
import type { UploadFile } from 'element-plus';
// 报告模版接口
export namespace PqReport {
/**
* 报告模版表格分页查询参数
*/
export interface ReqReportParams extends ReqPage {
id: string; // 装置序号id 必填
name?: string; // 设备名称
createTime?: string; //创建时间
}
/**
* 报告模版新增、修改、根据id查询返回的对象
*/
export interface ResReport {
id: string; //报告模板id
name: string;//报告模板名称
version:string;//版本号
baseFile?:string;//基础模板文件路径
detailFile?:string;//检测项模版文件路径
description:string;//描述信息
state:number;//状态:8-删除 1-正常
createBy?: string| null; //创建用户
createTime?: string| null; //创建时间
updateBy?: string| null; //更新用户
updateTime?: string| null; //更新时间
}
/**
* 报告模版表格查询分页返回的对象;
*/
export interface ResReportPage extends ResPage<ResReport> {
}
}

View File

@@ -0,0 +1,47 @@
import type { ReqPage, ResPage } from '@/api/interface'
// 标准设备模块
export namespace StandardDevice {
/**
* 标准设备表格分页查询参数
*/
export interface ReqPqStandardDeviceParams extends ReqPage {
id: string; // 装置序号id 必填
name: string; //设备名称
devType?: string; // 设备名称
createTime?: string; //创建时间
}
/**
* 标准设备新增、修改、根据id查询返回的对象
*/
export interface ResPqStandardDevice {
id: string; //装置序号ID
name: string; //设备名称
devType: string;//设备类型
manufacturer?: string | null;//生产厂家
protocol: string;//通讯协议
ip: string; //IP地址
port: number; //端口号
inspectChannel:string[] |string;//可检通道数
encryptionFlag: number; //装置是否为加密版本
series?: string | null; //装置识别码3ds加密
devKey?: string | null; //装置秘钥3ds加密
state: number; //状态
createBy?: string | null; //创建用户
createTime?: string | null; //创建时间
updateBy?: string | null; //更新用户
updateTime?: string | null; //更新时间
disabled?: boolean;
}
/**
* 标准设备表格查询分页返回的对象;
*/
export interface ResPqStandardDevicePage extends ResPage<ResPqStandardDevice> {
}
}

View File

@@ -2,37 +2,37 @@ import type { ReqPage, ResPage } from '@/api/interface'
// 检测脚本模块 // 检测脚本模块
export namespace TestScript { export namespace TestScript {
/**
* 检测脚本表格分页查询参数
/** */
* 检测脚本表格分页查询参数 export interface ReqTestScriptParams extends ReqPage {
*/ id: string // 装置序号id 必填
export interface ReqTestScriptParams extends ReqPage{ name: string
id: string; // 装置序号id 必填 type: string
name: string; pattern: string
type: string; }
}
// 检测脚本接口 // 检测脚本接口
export interface ResTestScript { export interface ResTestScript {
id?: string; //检测脚本ID id?: string //检测脚本ID
name: string; //检测脚本名称 name: string //检测脚本名称
type: string; //检测脚本类型(字典表Code字段脚本还是模板) type: string //设定0为脚本1为模板
pattern: string;//检测脚本模式(字典表Code字段数字、模拟、比对) pattern: string //检测脚本模式(字典表Code字段数字、模拟、比对)
valueType?: string;//脚本值类型(字典表Code字段相对值脚本、绝对值脚本、无) valueType?: string //脚本值类型(字典表Code字段相对值脚本、绝对值脚本、无)
standardName: string;//参照标准名称 standardName: string //参照标准名称
standardTime: string;//标准推行时间 standardTime: string //标准推行时间
state:number;// state?: number //
createBy?: string; createBy?: string
createTime?: string; createTime?: string
updateBy?: string; updateBy?: string
updateTime?: string; updateTime?: string
selectedValue?: string
ratedCurr?: number
ratedVolt?: number
} }
/** /**
* 检测脚本查询分页返回的对象; * 检测脚本查询分页返回的对象;
*/ */
export interface ResTestScriptPage extends ResPage<ResTestScript> { export interface ResTestScriptPage extends ResPage<ResTestScript> {}
}
}
}

View File

@@ -15,7 +15,7 @@ export namespace TestSource {
// 检测源接口 // 检测源接口
export interface ResTestSource { export interface ResTestSource {
id: string; //检测源ID id: string; //检测源ID
name: string; //检测源名称(检测源类型 + 设备类型 + 数字自动生成) name?: string; //检测源名称(检测源类型 + 设备类型 + 数字自动生成)
pattern: string;//检测源模式(字典表Code字段数字、模拟、比对) pattern: string;//检测源模式(字典表Code字段数字、模拟、比对)
type: string; //检测源类型(字典表Code字段标准源、高精度设备) type: string; //检测源类型(字典表Code字段标准源、高精度设备)
devType: string;//检测源设备类型(字典表Code字段) devType: string;//检测源设备类型(字典表Code字段)
@@ -34,9 +34,9 @@ export namespace TestSource {
} }
export interface ParameterType{ export interface ParameterType{
id:string; id:string;
sourceParamType:string; type:string;
sourceParamDesc:string; desc:string;
sourceParamValue:string; value:string|null;
sort:number; sort:number;
pId:string; pId:string;
children?:ParameterType[]; children?:ParameterType[];

View File

@@ -1,4 +1,3 @@
import type { Monitor } from '@/api/device/interface/monitor'
import http from '@/api' import http from '@/api'
/** /**
@@ -6,23 +5,9 @@ import http from '@/api'
*/ */
//获取监测点 //获取监测点
export const getPqMonList = (params: Monitor.ReqPqMonParams) => { export const getPqMonList = (param:any) => {
//return http.post(`/pqMon/list`, params) return http.post(`/pqMonitor/list`, param)
}
//添加监测点
export const addPqMon = (params: Monitor.ResPqMon) => {
//return http.post(`/pqMon/add`, params)
}
//编辑监测点
export const updatePqMon = (params: Monitor.ResPqMon) => {
//return http.post(`/pqMon/update`, params)
}
//删除监测点
export const deletePqMon = (params: string[]) => {
//return http.post(`/pqMon/delete`, params)
} }

View File

@@ -0,0 +1,51 @@
import type {PqReport} from '@/api/device/interface/report'
import http from '@/api'
/**
* @name 报告模版模块
*/
//获取报告模版
export const getPqReportList = (params: PqReport.ReqReportParams) => {
return http.post(`/report/list`, params)
}
//新增报告模版
export const addPqReport = (params: PqReport.ResReport) => {
return http.upload(`/report/add`, params)
}
//删除报告模版
export const deletePqReport = (params: string[]) => {
return http.post(`/report/delete`, params)
}
//查询报告模板详情
export const getPqReportById = (params: PqReport.ResReport) => {
return http.get(`/report/getById?id=${params.id}`)
}
//修改报告模板
export const updatePqReport = (params: PqReport.ResReport) => {
return http.upload(`/report/update`, params)
}
//查询所有报告模板名称
export const getPqReportAllName = () => {
return http.get(`/report/listAllName`)
}
//根据名称查询指定报告模板的所有版本
export const getPqReportAllVersion = (params:any) => {
return http.get(`/report/listAllVersion?name=${params.name}`)
}
//被检设备归档
export const documentedPqDev = (ids: string[]) => {
return http.post(`/report/documented`, ids)
}
//上传报告到云端
export const uploadReportToCloud = (deviceIds: string[]) => {
return http.post(`/report/uploadReportToCloud`, deviceIds)
}

View File

@@ -0,0 +1,55 @@
import type { StandardDevice } from '@/api/device/interface/standardDevice'
import http from '@/api'
/**
* @name 标准设备管理模块
*/
//获取标准设备
export const getPqStandardDevList = (params: StandardDevice.ReqPqStandardDeviceParams) => {
return http.post(`/pqStandardDev/list`, params)
}
//根据id查询标准设备
export const getPqStandardDevById = (params: StandardDevice.ReqPqStandardDeviceParams) => {
return http.get(`/pqStandardDev/getById?id=${params.id}`)
}
//添加标准设备
export const addPqStandardDev = (params: StandardDevice.ResPqStandardDevice) => {
return http.post(`/pqStandardDev/add`, params)
}
//编辑标准设备
export const updatePqStandardDev = (params: StandardDevice.ResPqStandardDevice) => {
return http.post(`/pqStandardDev/update`, params)
}
//删除标准设备
export const deletePqStandardDev = (params: string[]) => {
return http.post(`/pqStandardDev/delete`, params)
}
//导出标准设备
export const exportPqStandardDev = (params: StandardDevice.ReqPqStandardDeviceParams) => {
return http.download(`/pqStandardDev/export`, params)
}
// 下载导入文件模板
export const downloadTemplate = () => {
return http.download(`/pqStandardDev/downloadTemplate`)
}
//导入标准设备
export const importPqStandardDev = (params: StandardDevice.ReqPqStandardDeviceParams) => {
return http.uploadExcel(`/pqStandardDev/import`, params)
}
//获取所有标准设备
export const getAllPqStandardDev = () => {
return http.get(`/pqStandardDev/getAll`)
}
//获取可以绑定的标准设备
export const canBindingList = () => {
return http.get(`/pqStandardDev/canBindingList`)
}

View File

@@ -1,21 +1,53 @@
import type { TestScript } from '@/api/device/interface/testScript' import type { TestScript } from '@/api/device/interface/testScript'
import http from '@/api' import http from '@/api'
import { Loading } from '@element-plus/icons-vue'
/** /**
* @name 检测脚本管理模块 * @name 检测脚本管理模块
*/ */
// 新增脚本
export const pqScriptAdd = (params: TestScript.ResTestScript) => {
return http.post(`/pqScript/add`, params)
}
// 修改脚本
export const pqScriptUpdate = (params: TestScript.ResTestScript) => {
return http.post(`/pqScript/update`, params)
}
// 获取检测脚本列表 // 获取检测脚本列表
export const getPqScriptList = (params: TestScript.ReqTestScriptParams) => { export const getPqScriptList = (params: TestScript.ReqTestScriptParams) => {
return http.post(`/pqScript/list`, params) return http.post(`/pqScript/list`, params)
} }
//升级为模版 //升级为模版
export const updatePqScript = (params: TestScript.ResTestScript) => { export const updatePqScript = (params: TestScript.ResTestScript) => {
return http.get(`/pqScript/upgradeToTemplate?id=${params.id}`) return http.get(`/pqScript/upgradeToTemplate?id=${params.id}`)
} }
//删除检测脚本 //删除检测脚本
export const deletePqScript = (params: string[]) => { export const deletePqScript = (params: string[]) => {
return http.post(`/pqScript/delete`, params) return http.post(`/pqScript/delete`, params)
}
//添加检测脚本
export const addScriptDtls = (params: any) => {
return http.post(`/pqScript/addScriptDtls`, params)
}
//根据脚本id查询检测脚本详情
export const dlsDetails = (params: any) => {
return http.post(`/pqScript/dlsDetails`, params)
}
// 删除脚本
export const deleteDtls = (params: any) => {
return http.post(`/pqScript/deleteDtls`, params)
}
// 启用关闭脚本
export const updateDtls = (params: any) => {
return http.post(`/pqScript/updateDtls`, params)
}
// 根据通讯参数生成装置下发原始数据公式
export const scriptDtlsCheckDataList = (params: any) => {
return http.post(`/pqScript/scriptDtlsCheckDataList`, params,{loading: false})
}
// 通讯脚本回显
export const checkDataList = (params: any) => {
return http.post(`/pqScript/checkDataList`, params, { loading: true })
} }

View File

@@ -10,9 +10,9 @@ export const checkStatus = (status: number) => {
case 400: case 400:
ElMessage.error("请求失败!请您稍后重试"); ElMessage.error("请求失败!请您稍后重试");
break; break;
case 401: // case 401:
ElMessage.error("登录失效!请您重新登录"); // ElMessage.error("登录失效!请您重新登录");
break; // break;
case 403: case 403:
ElMessage.error("当前账号无权限访问!"); ElMessage.error("当前账号无权限访问!");
break; break;

View File

@@ -0,0 +1,6 @@
import http from "@/api";
//系数校准发送基本信息
export const getCoefficientCheck = (params: any) => {
return http.post(`/prepare/coefficientCheck`, params,{loading: false})
}

View File

@@ -0,0 +1,31 @@
// 系数校准模块
export namespace ChannelsTest {
// 系数校准列表
export interface CoefficientVO {
devName?: string;//设备名称
type?:string;//区分大小电压
monitorNum: string;//监测点序号
desc: string;//描述
aVuData:string;//电压通道A数据
aVuXi:string;//电压通道A系数
bVuData:string;//电压通道B数据
bVuXi:string;//电压通道B系数
cVuData:string;//电压通道C数据
cVuXi:string;//电压通道C系数
aIeData:string;//电流通道A数据
aIeXi:string;//电流通道A系数
bIeData:string;//电流通道B数据
bIeXi:string;//电流通道B系数
cIeData:string;//电流通道C数据
cIeXi:string;//电流通道C系数
loading: boolean;
aV:string;
bV:string;
cV:string;
aI:string;
bI:string;
cI:string;
}
}

View File

@@ -1,123 +1,242 @@
import axios, { AxiosInstance, AxiosError, AxiosRequestConfig, InternalAxiosRequestConfig, AxiosResponse } from 'axios' import { ElMessage } from 'element-plus'
import axios, {
AxiosError,
type AxiosInstance,
type AxiosRequestConfig,
type AxiosResponse,
type InternalAxiosRequestConfig
} from 'axios'
import { showFullScreenLoading, tryHideFullScreenLoading } from '@/components/Loading/fullScreen' import { showFullScreenLoading, tryHideFullScreenLoading } from '@/components/Loading/fullScreen'
import { LOGIN_URL } from '@/config' import { LOGIN_URL } from '@/config'
import { ElMessage } from 'element-plus' import { type ResultData } from '@/api/interface'
import { ResultData } from '@/api/interface'
import { ResultEnum } from '@/enums/httpEnum' import { ResultEnum } from '@/enums/httpEnum'
import { checkStatus } from './helper/checkStatus' import { checkStatus } from './helper/checkStatus'
import { useUserStore } from '@/stores/modules/user' import { useUserStore } from '@/stores/modules/user'
import router from '@/routers' import router from '@/routers'
import { refreshToken } from '@/api/user/login'
import { EventSourcePolyfill } from 'event-source-polyfill'
export interface CustomAxiosRequestConfig extends InternalAxiosRequestConfig { export interface CustomAxiosRequestConfig extends InternalAxiosRequestConfig {
loading?: boolean; loading?: boolean
} }
const config = { const config = {
// 默认地址请求地址,可在 .env.** 文件中修改 // 默认地址请求地址,可在 .env.** 文件中修改
baseURL: import.meta.env.VITE_API_URL as string, baseURL: import.meta.env.VITE_API_URL as string,
// 设置超时时间 // 设置超时时间
timeout: ResultEnum.TIMEOUT as number, timeout: ResultEnum.TIMEOUT as number,
// 跨域时候允许携带凭证 // 跨域时候允许携带凭证
withCredentials: true, withCredentials: true,
// post请求指定数据类型以及编码 // post请求指定数据类型以及编码
headers: { 'Content-Type': 'application/json;charset=utf-8' }, headers: { 'Content-Type': 'application/json;charset=utf-8' }
} }
class RequestHttp { class RequestHttp {
service: AxiosInstance service: AxiosInstance
public constructor(config: AxiosRequestConfig) { public constructor(config: AxiosRequestConfig) {
// 创建实例 // 创建实例
this.service = axios.create(config) this.service = axios.create(config)
/**
* @description 请求拦截器
* 客户端发送请求 -> [请求拦截器] -> 服务器
* token校验(JWT) : 接受服务器返回的 token,存储到 vuex/pinia/本地储存当中
*/
this.service.interceptors.request.use(
(config: CustomAxiosRequestConfig) => {
isFirst = true
const userStore = useUserStore()
// 当前请求不需要显示 loading在 api 服务中通过指定的第三个参数: { loading: false } 来控制
config.loading ?? (config.loading = true)
config.loading && showFullScreenLoading()
if (config.headers && typeof config.headers.set === 'function') {
config.headers.set('Authorization', 'Bearer ' + userStore.accessToken)
config.headers.set('Is-Refresh-Token', userStore.isRefreshToken + '')
}
return config
},
(error: AxiosError) => {
return Promise.reject(error)
}
)
let isFirst = true
/**
* @description 响应拦截器
* 服务器换返回信息 -> [拦截统一处理] -> 客户端JS获取到信息
*/
this.service.interceptors.response.use(
async (response: AxiosResponse) => {
const { data } = response
const userStore = useUserStore()
tryHideFullScreenLoading()
if (data.code === ResultEnum.ACCESSTOKEN_EXPIRED) {
// 用长token去换短token
userStore.setAccessToken(userStore.refreshToken)
userStore.setIsRefreshToken(true)
const result = await refreshToken()
if (result) {
//获取新token成功的话
// 有新的token后重新请求
userStore.setAccessToken(result.data.accessToken)
userStore.setRefreshToken(result.data.refreshToken)
userStore.setIsRefreshToken(false)
userStore.setExp(1000 * 60 * 60 * 24 * 30)
response.config.headers.Authorization = `Bearer ${result.data.accessToken}` //重新请求前需要将更新后的新token更换掉之前无效的token,不然会死循环
const resp = await this.service.request(response.config)
return resp
} else {
// 刷新失效,跳转登录页
}
}
// 登陆失效
if (data.code === ResultEnum.OVERDUE) {
console.log('登陆失效')
userStore.setAccessToken('')
userStore.setRefreshToken('')
userStore.setIsRefreshToken(false)
userStore.setUserInfo({ id: '', name: '' })
userStore.setExp(0)
await router.replace(LOGIN_URL)
if (isFirst) {
//临时处理token失效弹窗多次
ElMessage.error(data.message)
isFirst = false
}
return Promise.reject(data)
}
// 全局错误信息拦截(防止下载文件的时候返回数据流,没有 code 直接报错)
if (data.code && data.code !== ResultEnum.SUCCESS) {
if (data.message.includes('&')) {
let formattedMessage = data.message.split('&').join('<br>')
if (data.message.includes(':')) {
formattedMessage = formattedMessage.replace(':', '')
}
ElMessage.error({ message: formattedMessage, dangerouslyUseHTMLString: true })
return Promise.reject(data)
}
ElMessage.error(data.message)
return Promise.reject(data)
}
// 成功请求(在页面上除非特殊情况,否则不用处理失败逻辑)
if (userStore.exp <= Date.now() && userStore.exp !== 0) {
userStore.setAccessToken('')
userStore.setRefreshToken('')
userStore.setIsRefreshToken(false)
userStore.setUserInfo({ id: '', name: '' })
userStore.setExp(0)
ElMessage.error('登录已过期,请重新登录!')
await router.replace(LOGIN_URL)
return Promise.reject(data)
}
// 对于blob类型的响应返回完整的response对象以保留响应头
if (response.config.responseType === 'blob') {
return response
}
return data
},
async (error: AxiosError) => {
const { response } = error
tryHideFullScreenLoading()
console.log('error', error.message)
// 请求超时 && 网络错误单独判断,没有 response
if (error.message.indexOf('timeout') !== -1) ElMessage.error('请求超时!请您稍后重试')
if (error.message.indexOf('Network Error') !== -1) ElMessage.error('网络错误!请您稍后重试')
// 根据服务器响应的错误状态码,做不同的处理
if (response) checkStatus(response.status)
// 服务器结果都没有返回(可能服务器错误可能客户端断网),断网处理:可以跳转到断网页面
if (!window.navigator.onLine) router.replace('/500')
return Promise.reject(error)
}
)
}
/** /**
* @description 请求拦截器 * @description 常用请求方法封装
* 客户端发送请求 -> [请求拦截器] -> 服务器
* token校验(JWT) : 接受服务器返回的 token,存储到 vuex/pinia/本地储存当中
*/ */
this.service.interceptors.request.use( get<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> {
(config: CustomAxiosRequestConfig) => { return this.service.get(url, { params, ..._object })
const userStore = useUserStore() }
// 当前请求不需要显示 loading在 api 服务中通过指定的第三个参数: { loading: false } 来控制
config.loading ?? (config.loading = true) post<T>(url: string, params?: object | string, _object = {}): Promise<ResultData<T>> {
config.loading && showFullScreenLoading() return this.service.post(url, params, _object)
if (config.headers && typeof config.headers.set === 'function') { }
config.headers.set('Authorization', 'Bearer '+userStore.token)
} put<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> {
return config return this.service.put(url, params, _object)
}, }
(error: AxiosError) => {
return Promise.reject(error) delete<T>(url: string, params?: any, _object = {}): Promise<ResultData<T>> {
}, return this.service.delete(url, { params, ..._object })
) }
download(url: string, params?: object, _object = {}): Promise<BlobPart> {
return this.service.post(url, params, { ..._object, responseType: 'blob' }).then(res => res.data)
}
downloadWithHeaders(url: string, params?: object, _object = {}): Promise<AxiosResponse<Blob>> {
return this.service.post(url, params, { ..._object, responseType: 'blob' })
}
upload(url: string, params?: object, _object = {}): Promise<BlobPart> {
return this.service.post(url, params, {
..._object,
headers: { 'Content-Type': 'multipart/form-data' }
})
}
/** /**
* @description 响应拦截器 * 针对excel的上传默认返回的是blob类型Excel没问题时返回json特殊处理
* 服务器换返回信息 -> [拦截统一处理] -> 客户端JS获取到信息
*/ */
this.service.interceptors.response.use( uploadExcel(url: string, params?: object, _object = {}): Promise<BlobPart> {
(response: AxiosResponse) => { return this.service
const { data } = response .post(url, params, {
..._object,
headers: { 'Content-Type': 'multipart/form-data' },
responseType: 'blob'
})
.then(res => res.data)
}
// 添加SSE连接方法
sse(url: string, params?: any): EventSource {
const userStore = useUserStore() const userStore = useUserStore()
tryHideFullScreenLoading() // 构造带参数的URL
// 登陆失效 let requestUrl = config.baseURL + url
if (data.code == ResultEnum.OVERDUE) { if (params) {
userStore.setToken('') const searchParams = new URLSearchParams()
userStore.setUserInfo({name: ''}) for (const key in params) {
router.replace(LOGIN_URL) if (Object.prototype.hasOwnProperty.call(params, key)) {
ElMessage.error(data.message) searchParams.append(key, String(params[key]))
return Promise.reject(data) }
}
requestUrl += '?' + searchParams.toString()
} }
// 全局错误信息拦截(防止下载文件的时候返回数据流,没有 code 直接报错)
if (data.code && data.code !== ResultEnum.SUCCESS) {
ElMessage.error(data.message)
return Promise.reject(data)
}
// 成功请求(在页面上除非特殊情况,否则不用处理失败逻辑)
return data
},
async (error: AxiosError) => {
const { response } = error
tryHideFullScreenLoading()
// 请求超时 && 网络错误单独判断,没有 response
if (error.message.indexOf('timeout') !== -1) ElMessage.error('请求超时!请您稍后重试')
if (error.message.indexOf('Network Error') !== -1) ElMessage.error('网络错误!请您稍后重试')
// 根据服务器响应的错误状态码,做不同的处理
if (response) checkStatus(response.status)
// 服务器结果都没有返回(可能服务器错误可能客户端断网),断网处理:可以跳转到断网页面
if (!window.navigator.onLine) router.replace('/500')
return Promise.reject(error)
},
)
}
/** // 创建EventSource连接
* @description 常用请求方法封装 const eventSource = new EventSourcePolyfill(requestUrl, {
*/ headers: {
get<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> { Authorization: 'Bearer ' + userStore.accessToken
return this.service.get(url, { params, ..._object }) },
} // 增加超时时间到1200秒
heartbeatTimeout: 1200000
})
post<T>(url: string, params?: object | string, _object = {}): Promise<ResultData<T>> { // 设置默认的Authorization头部
return this.service.post(url, params, _object) eventSource.addEventListener('open', function () {
} console.log('SSE连接已建立')
})
put<T>(url: string, params?: object, _object = {}): Promise<ResultData<T>> { // 添加错误处理
return this.service.put(url, params, _object) eventSource.addEventListener('error', function (err) {
} console.error('SSE连接错误:', err)
})
delete<T>(url: string, params?: any, _object = {}): Promise<ResultData<T>> {
return this.service.delete(url, { params, ..._object })
}
download(url: string, params?: object, _object = {}): Promise<BlobPart> {
return this.service.post(url, params, { ..._object, responseType: 'blob' })
}
upload(url: string, params?: object, _object = {}): Promise<BlobPart> {
return this.service.post(url, params, { ..._object, headers: { 'Content-Type': 'multipart/form-data' } })
}
return eventSource
}
} }
export default new RequestHttp(config) export default new RequestHttp(config)

View File

@@ -15,6 +15,7 @@ export interface Result {
* 请求响应参数包含data * 请求响应参数包含data
*/ */
export interface ResultData<T = any> extends Result { export interface ResultData<T = any> extends Result {
map(arg0: (item: any) => { label: any; value: any; }): { label: string; value: string; }[] | { label: string; value: string; }[];
data: T; data: T;
} }

View File

@@ -1,227 +1,194 @@
{ {
"code": "A0000", "code": "A0000",
"data": [ "data": [
{
"id": "1",
"pid": "0",
"name": "频率准确度检测",
"children": [
{ {
"id": "0", "id": "1-1",
"name": "频率准确度检测", "pid": "1",
"children": [ "name": "额定工作条件下的检测",
{ "children": [
"pid": 0, {
"id": "0-1", "id": "1-1-1",
"name": "额定工作条件下的检测", "pid": "1-1",
"children": [ "name": "输入:频率 42.5Hz..."
{ },
"scriptIdx":1, {
"isChildNode":true, "id": "1-1-2",
"pid": "0-1", "pid": "1-1",
"id": "0-1-1", "name": "输入:频率 50.0Hz..."
"name": "输入:频率 42.5Hz..." },
}, {
{ "id": "1-1-3",
"scriptIdx":2, "pid": "1-1",
"isChildNode":true, "name": "输入:频率 50.05Hz...."
"pid": "0-1", }
"id": "0-1-2", ]
"name": "输入:频率 50.0Hz..."
},
{
"scriptIdx":3,
"isChildNode":true,
"pid": "0-1",
"id": "0-1-3",
"name": "输入:频率 50.05Hz..."
}
]
},
{
"pid": 0,
"id": "0-2",
"name": "电压对频率测量的影响",
"children": [
{
"scriptIdx":4,
"isChildNode":true,
"pid": "0-2",
"id": "0-2-1",
"name": "输入:频率 50.05Hz Ua =10%Un..."
},
{
"scriptIdx":5,
"isChildNode":true,
"pid": "0-2",
"id": "0-2-1",
"name": "输入:频率 51.05Hz Ua =10%Un..."
},
{
"scriptIdx":6,
"isChildNode":true,
"pid": "0-2",
"id": "0-2-2",
"name": "输入:频率 52.05Hz Ua =10%Un..."
}
]
},
{
"pid": 0,
"id": "0-3",
"name": "谐波对频率测量的影响",
"children": [
{
"scriptIdx":7,
"isChildNode":true,
"pid": "0-3",
"id": "0-3-1",
"name": "输入:频率 50.05Hz Ua =100%Un..."
}
]
}
]
}, },
{ {
"id": "1", "id": "1-2",
"name": "电压准确度检测", "pid": "1",
"children": [ "name": "电压对频率测量的影响",
{ "children": [
"pid": 1, {
"id": "1-1", "id": "1-2-1",
"name": "额定工作条件下的检测", "pid": "1-2",
"children": [ "name": "输入:频率 50.05Hz Ua =10%Un..."
{ },
"scriptIdx":8, {
"isChildNode":true, "id": "1-2-1",
"pid": "1-1", "pid": "1-2",
"id": "1-1-1", "name": "输入:频率 51.05Hz Ua =10%Un..."
"name": "输入:频率 42.5Hz..." },
}, {
{ "id": "1-2-2",
"scriptIdx":9, "pid": "1-2",
"isChildNode":true, "name": "输入:频率 52.05Hz Ua =10%Un..."
"pid": "1-1", }
"id": "1-1-2", ]
"name": "输入:频率 50.0Hz..."
},
{
"scriptIdx":10,
"isChildNode":true,
"pid": "1-1",
"id": "1-1-3",
"name": "输入:频率 50.05Hz..."
}
]
},
{
"pid": 0,
"id": "1-2",
"name": "电压对频率测量的影响",
"children": [
{
"scriptIdx":11,
"isChildNode":true,
"pid": "1-2",
"id": "1-2-1",
"name": "输入:频率 50.05Hz Ua =10%Un..."
},
{
"scriptIdx":12,
"isChildNode":true,
"pid": "1-2",
"id": "1-2-1",
"name": "输入:频率 51.05Hz Ua =10%Un..."
},
{
"scriptIdx":13,
"isChildNode":true,
"pid": "1-2",
"id": "1-2-2",
"name": "输入:频率 52.05Hz Ua =10%Un..."
}
]
},
{
"pid": 0,
"id": "0-3",
"name": "谐波对频率测量的影响",
"children": [
{
"scriptIdx":14,
"isChildNode":true,
"pid": "0-3",
"id": "0-3-1",
"name": "输入:频率 50.05Hz Ua =100%Un..."
}
]
}
]
}, },
{ {
"id": "2", "id": "1-3",
"name": "谐波准确度检测", "pid": "1",
"children": [ "name": "谐波对频率测量的影响",
{ "children": [
"pid": 2, {
"id": "2-1", "id": "1-3-1",
"name": "额定工作条件下的检测", "pid": "1-3",
"children": [ "name": "输入:频率 50.05Hz Ua =100%Un..."
{ }
"scriptIdx":15, ]
"pid": "2-1",
"id": "2-1-1",
"name": "输入:频率 42.5Hz..."
},
{
"scriptIdx":16,
"pid": "2-1",
"id": "2-1-2",
"name": "输入:频率 50.0Hz..."
},
{
"scriptIdx":17,
"pid": "2-1",
"id": "2-1-3",
"name": "输入:频率 50.05Hz..."
}
]
},
{
"pid": 2,
"id": "2-2",
"name": "电压对频率测量的影响",
"children": [
{
"scriptIdx":18,
"pid": "2-2",
"id": "2-2-1",
"name": "输入:频率 50.05Hz Ua =10%Un..."
},
{
"scriptIdx":19,
"pid": "2-2",
"id": "2-2-1",
"name": "输入:频率 51.05Hz Ua =10%Un..."
},
{
"scriptIdx":20,
"pid": "2-2",
"id": "2-2-2",
"name": "输入:频率 52.05Hz Ua =10%Un..."
}
]
},
{
"pid": 2,
"id": "2-3",
"name": "谐波对频率测量的影响",
"children": [
{
"scriptIdx":21,
"pid": "2-3",
"id": "2-3-1",
"name": "输入:频率 50.05Hz Ua =100%Un..."
}
]
}
]
} }
] ]
} },
{
"id": "2",
"pid": "0",
"name": "电压准确度检测",
"children": [
{
"pid": "2",
"id": "2-1",
"name": "额定工作条件下的检测",
"children": [
{
"id": "2-1-1",
"pid": "2-1",
"name": "输入:频率 42.5Hz..."
},
{
"id": "2-1-2",
"pid": "2-1",
"name": "输入:频率 50.0Hz..."
},
{
"id": "2-1-3",
"pid": "2-1",
"name": "输入:频率 50.05Hz..."
}
]
},
{
"pid": "2",
"id": "2-2",
"name": "电压对频率测量的影响",
"children": [
{
"id": "2-2-1",
"pid": "2-2",
"name": "输入:频率 50.05Hz Ua =10%Un..."
},
{
"id": "2-2-1",
"pid": "2-2",
"name": "输入:频率 51.05Hz Ua =10%Un..."
},
{
"id": "2-2-2",
"pid": "2-2",
"name": "输入:频率 52.05Hz Ua =10%Un..."
}
]
},
{
"pid": "2",
"id": "2-3",
"name": "谐波对频率测量的影响",
"children": [
{
"id": "2-3-1",
"pid": "2-3",
"name": "输入:频率 50.05Hz Ua =100%Un..."
}
]
}
]
},
{
"id": "3",
"pid": "0",
"name": "谐波准确度检测",
"children": [
{
"id": "3-1",
"pid": "3",
"name": "额定工作条件下的检测",
"children": [
{
"id": "3-1-1",
"pid": "3-1",
"name": "输入:频率 42.5Hz..."
},
{
"id": "3-1-2",
"pid": "3-1",
"name": "输入:频率 50.0Hz..."
},
{
"id": "3-1-3",
"pid": "3-1",
"name": "输入:频率 50.05Hz..."
}
]
},
{
"id": "3-2",
"pid": "3",
"name": "电压对频率测量的影响",
"children": [
{
"id": "3-2-1",
"pid": "3-2",
"name": "输入:频率 50.05Hz Ua =10%Un..."
},
{
"id": "3-2-1",
"pid": "3-2",
"name": "输入:频率 51.05Hz Ua =10%Un..."
},
{
"id": "3-2-2",
"pid": "3-2",
"name": "输入:频率 52.05Hz Ua =10%Un..."
}
]
},
{
"id": "3-3",
"pid": "3",
"name": "谐波对频率测量的影响",
"children": [
{
"id": "3-3-1",
"pid": "3-3",
"name": "输入:频率 50.05Hz Ua =100%Un..."
}
]
}
]
}
]
}

View File

@@ -1,72 +1,73 @@
import type { ReqPage } from '@/api/interface' import type { ReqPage } from '@/api/interface'
import type { DatetimeFormatProps } from 'vue-i18n';
// 检测计划模块 // 检测计划模块
export namespace Plan { export namespace Plan {
// 检测计划接口 // 检测计划接口
export interface PlanBO { export interface ResPlan {
id?: string; //检测计划ID id: string; //检测计划ID
name: string; //检测计划名称 name: string; //检测计划名称
pattern: string; //模式,字典表(数字、模拟、比对) pattern?: string; //模式,字典表(数字、模拟、比对)
father_Plan_Id?: string; //父计划ID fatherPlanId?: string; //父计划ID
dataSource_Id: string[]; //数据源ID dataSourceId: string; //数据源ID
script_Id: string; //检测脚本ID scriptId: string; //检测脚本ID
error_Sys_Id: string;//误差体系ID errorSysId: string;//误差体系ID
test_State: string; //检测状态 timeCheck:number;//守时检测:0否1。是
report_State: string; //报告生成状态 testState: number; //检测状态:0:未检、1检测中、2检测完成,默认为 0
result: string;//检测结果 reportState: number; //报告生成状态:0:未生成、1部分生成、2全部生成,默认为 0
create_Time?: string;//创建时间 result: number;//检测结果:0:不符合、1:符合、2/,默认为 2
code:number; //自动生成,用于生成数据表后缀
state: number;//;状态0-删除 1-正常
createBy?:string; //创建用户
createTime?:string; //创建时间
updateBy?:string; //更新用户
updateTime?:string; //更新时间
associateReport:number;//是否关联报告模板 0否 1是
reportTemplateName:string;
reportTemplateVersion:string;
dataRule:string;//数据处理原则
standardDevIds:string[];
standardDevMap:Map<string,number>;//标准设备
testItems:string[];//测试项
Check_By?:string;//计划检测人
progress?: number; // 进度百分比,例如 75
children?: ResPlan[];
testConfig?: PlanTestConfig;
importFlag?: number; // 导入标识0-否1-是
leader?: string; // 负责人
memberIds?: string | string[]; //成员
members?: string; //成员字符串
} }
// 检测计划 + 分页 // 检测计划 + 分页
export interface ReqPlanParams extends ReqPage,PlanBO { export interface ReqPlanParams extends ReqPage,ResPlan {
} }
// 检测计划 + 检测源
export interface PlanAndSourceBO extends PlanBO {
testSourceName: string;//计划所属检测源 export interface ReqPlan extends ResPlan {
source_Id: string[]; datasourceIds:string | string[];
device_Id?: string[]; sourceIds: string | null;
testSourceList?: string[];//临时测试 planId:string;
dataSource_Ids:string; scriptName: string ;
errorSysName: string;
sourceName: string ;
standardDevNameStr: string;
testItemNameStr:string;
devIds: string[];
} }
// // 检测计划列表
// export interface PlanList { export interface PlanTestConfig {
// id: string; //检测计划ID planId: string;
// name: string; //检测计划名称 waveRecord: number;
// pattern: string; //模式,字典表(数字、模拟、比对) realTime: number;
// father_Plan_Id: string; //父计划ID statistics: number;
// dataSource_Id: string; //数据源ID flicker: number;
// script_Id: string; //检测脚本ID maxTime: number;
// error_Sys_Id: string;//误差体系ID }
// test_State: string; //检测状态
// report_State: string; //报告生成状态
// result: string;//检测结果
// state: number; //状态
// create_By?: string; //创建用户
// create_Time?: string;//创建时间
// update_By?: string; //更新用户
// update_Time?: string; //更新时间
// }
// // 被检设备参数
// export interface ReqPlanParams extends ReqPage {
// id: string; //检测计划ID
// name: string; //检测计划名称
// pattern: string; //模式,字典表(数字、模拟、比对)
// father_Plan_Id: string; //父计划ID
// dataSource_Id: string; //数据源ID
// script_Id: string; //检测脚本ID
// error_Sys_Id: string;//误差体系ID
// test_State: string; //检测状态
// report_State: string; //报告生成状态
// result: string;//检测结果
// state: number; //状态
// create_By?: string; //创建用户
// create_Time?: string;//创建时间
// update_By?: string; //更新用户
// update_Time?: string; //更新时间
// }
} }

View File

@@ -1,27 +1,162 @@
import { ResPage } from '@/api/interface' import type { Plan } from './interface'
import { Plan } from './interface'
import { ADMIN as rePrefix } from '@/api/config/serviceName'
import http from '@/api' import http from '@/api'
import type { ErrorSystem } from '../device/interface/error'
import type { Device } from '../device/interface/device'
/** /**
* @name 检测计划管理模块 * @name 检测计划管理模块
*/ */
// 获取检测计划列表 // 获取检测计划列表
export const getPlanList = (params: Plan.ReqPlanParams) => { export const getPlanList = (params: Plan.ReqPlanParams) => {
return http.post<ResPage<Plan.PlanBO>>(`/plan/list`, params) return http.post(`/adPlan/list`, params)
} }
// 新增检测计划 // 新增检测计划
export const addPlan = (params: Plan.PlanBO) => { export const addPlan = (params: any) => {
return http.post(`/role/add`, params) return http.post(`/adPlan/add`, params)
} }
// 编辑检测计划 // 编辑检测计划
export const editPlan = (params: Plan.PlanBO) => { export const updatePlan = (params: any) => {
return http.post(`/role/edit`, params) return http.post(`/adPlan/update`, params)
} }
// 删除检测计划 // 删除检测计划
export const deletePlan = (params: { id: string[] }) => { export const deletePlan = (params: { id: string[]; pattern: string }) => {
return http.post(`/role/del`, params) return http.post(`/adPlan/delete?pattern=${params.pattern}`, params.id)
} }
// 获取指定模式下所有检测源
export const getTestSourceList = (params: Plan.ReqPlan) => {
return http.get(`/pqSource/getAll?patternId=${params.pattern}`)
}
// 获取指定模式下所有检测脚本
export const getPqScriptList = (params: Plan.ReqPlan) => {
return http.get(`/pqScript/getAll?patternId=${params.pattern}`)
}
//获取所有误差体系
export const getPqErrSysList = () => {
return http.get<ErrorSystem.ErrorSystemList>(`/pqErrSys/getAll`)
}
//获取指定模式下所有未绑定的设备
export const getUnboundPqDevList = (params: { pattern: string}) => {
return http.get(`/pqDev/listUnbound?pattern=${params.pattern}`)
}
//根据检测计划id查询出所有已绑定的设备
export const getBoundPqDevList = (params: any) => {
return http.post(`/adPlan/listByPlanId`, params)
}
//检测计划绑定设备
// export const BindPqDevList = (params: any) => {
// return http.post(`/pqDev/bindDev`,params)
// }
// 按照模式查询检测计划(用于首页展示)
export const getPlanListByPattern = (params: Plan.ReqPlan) => {
return http.get(`/adPlan/listByPattern?pattern=${params.pattern}`)
}
// 导出检测计划
export const exportPlan = (params: Device.ReqPqDevParams) => {
return http.download(`/adPlan/export`, params)
}
// 下载模板
export const downloadTemplate = (params: { patternId: string }) => {
return http.download(`/adPlan/downloadTemplate`, params)
}
// 导入检测计划
export const importPlan = (params: Device.ReqPqDevParams) => {
return http.uploadExcel(`/adPlan/import`, params)
}
// 装置检测报告生成
export const generateDevReport = (params: Device.ReqDevReportParams) => {
return http.post(`/report/generateReport`, params)
}
// 装置检测报告下载
export const downloadDevData = (params: Device.ReqDevReportParams) => {
return http.download(`/report/downloadReport`, params)
}
// 装置检测报告下载(带响应头)
export const downloadDevDataWithHeaders = (params: Device.ReqDevReportParams) => {
return http.downloadWithHeaders(`/report/downloadReport`, params)
}
export const staticsAnalyse = (params: { id: string[] }) => {
return http.download('/adPlan/analyse', params)
}
//根据计划id分页查询被检设
export const getDevListByPlanId = (params: any) => {
return http.post(`/adPlan/listDevByPlanId`, params)
}
//修改子计划名称
export const updateSubPlanName = (params: Plan.ReqPlan) => {
return http.get(`/adPlan/updateSubPlanName?planId=${params.id}&name=${params.name}`)
}
//子计划绑定/解绑标准设备
export const subPlanBindStandardDevList = (params: Plan.ReqPlan) => {
return http.post(`/adPlan/updateBindStandardDev`, params)
}
//子计划绑定/解绑被检设备
export const subPlanBindDev = (params: Plan.ReqPlan) => {
return http.post(`/adPlan/updateBindDev`, params)
}
//根据父计划ID获取未被子计划绑定的标准设备
export const getUnboundStandardDevList = (params: Plan.ResPlan) => {
return http.get(`/adPlan/getUnBoundStandardDev?fatherPlanId=${params.fatherPlanId}`)
}
//根据计划ID获取已绑定的标准设备
export const getBoundStandardDevList = (params: Plan.ResPlan) => {
return http.get(`/adPlan/getBoundStandardDev?planId=${params.id}`)
}
//根据计划ID获取已绑定的所有标准设备
export const getBoundStandardDevAllList = (params: { id: string }) => {
return http.get(`/adPlan/getBoundStandardDev?planId=${params.id}&all=1`)
}
// 导出子计划
export const exportSubPlan = (params: Plan.ResPlan) => {
return http.download(`/adPlan/exportSubPlan?planId=${params.id}`)
}
// 导入子检测计划
export const importSubPlan = (params: Plan.ResPlan) => {
return http.upload(`/adPlan/importSubPlan`, params)
}
// 导出计划检测结果数据
export const exportPlanCheckData = (params: any) => {
return http.post(
`/adPlan/exportPlanCheckData?planId=${params.id}&devIds=${params.devIds}&report=${params.report}`
)
}
//根据误差体系id获取测试项
export const getPqErrSysTestItemList = (params: {errorSysId : string}) => {
return http.get(`/pqErrSys/getTestItems?id=${params.errorSysId}`)
}
// 获取计划项目成员
export const getMemberList = (params: {id : string}) => {
return http.get(`/adPlan/getMemberList?planId=${params.id}`)
}
// 导入并合并子检测计划检测结果数据
export const importAndMergePlanCheckData = (params: Plan.ResPlan) => {
return http.upload(`/adPlan/importAndMergePlanCheckData`, params)
}

View File

@@ -58,11 +58,15 @@ const dictReportState: Dict[] = [
const dictResult: Dict[] = [ const dictResult: Dict[] = [
{ {
id: "0", id: "0",
label: '符合', label: '符合',
}, },
{ {
id: "1", id: "1",
label: '符合', label: '符合',
},
{
id: "2",
label: '/',
}, },
] ]
@@ -170,35 +174,35 @@ const testFatherPlanList: Dict[] = [
const sourceDataList: Dict[] = [ const sourceDataList: Dict[] = [
{ {
id: "1", id: "1",
label: '标准源-福禄克-6100A', label: '标准源-FLUKE.6100A电能功率标准源-1',
}, },
{ {
id: "2", id: "2",
label: '标准源-昂立-PF2', label: '标准源-ANGLI-FP2高性能数字信号源-1',
}, },
{ {
id: "3", id: "3",
label: '标准源-丹迪克-DKLN1', label: '标准源-DKLN-1电能质量测试分析仪智能检定装置-1',
}, },
{ {
id: "4", id: "4",
label: '标准源-博电源-PQC600A', label: '标准源-PQC600A高精度电能功率标准源-1',
}, },
{ {
id: "5", id: "5",
label: '高精度设备-PQV520-1', label: '高精度设备-PQV-520便携式电能质量监测装置-1',
}, },
{ {
id: "6", id: "6",
label: '高精度设备-PQV520-2', label: '高精度设备-PQV-520便携式电能质量监测装置-2',
}, },
{ {
id: "7", id: "7",
label: '高精度设备-PQV520-3', label: '高精度设备-PQV-520便携式电能质量监测装置-3',
}, },
{ {
id: "8", id: "8",
label: '高精度设备-PQV520-4', label: '高精度设备-PQV-520便携式电能质量监测装置-4',
}, },
] ]
@@ -206,46 +210,80 @@ const sourceDataList: Dict[] = [
const deviceDataList: Dict[] = [ const deviceDataList: Dict[] = [
{ {
id: "1", id: "1",
label: '模拟装置1', label: '240001',
}, },
{ {
id: "2", id: "2",
label: '模拟装置2', label: '240002',
}, },
{ {
id: "3", id: "3",
label: '模拟装置3', label: '240003',
}, },
{ {
id: "4", id: "4",
label: '模拟装置4', label: '240004',
}, },
{ {
id: "5", id: "5",
label: '中电送检装置', label: '240005',
}, },
{ {
id: "6", id: "6",
label: '易司拓测试装置', label: '240006',
}, },
{ {
id: "7", id: "7",
label: '山大电力测试装置1', label: '240007',
}, },
{ {
id: "8", id: "8",
label: '山大电力测试装置2', label: '240008',
}, },
] ]
// const deviceDataList: Dict[] = [
// {
// id: "1",
// label: '模拟装置1',
// },
// {
// id: "2",
// label: '模拟装置2',
// },
// {
// id: "3",
// label: '模拟装置3',
// },
// {
// id: "4",
// label: '模拟装置4',
// },
// {
// id: "5",
// label: '中电送检装置',
// },
// {
// id: "6",
// label: '易司拓测试装置',
// },
// {
// id: "7",
// label: '山大电力测试装置1',
// },
// {
// id: "8",
// label: '山大电力测试装置2',
// },
// ]
const planData = ref<Plan.PlanAndSourceBO[]>([ const planData = ref<Plan.PlanAndSourceBO[]>([
{ {
'id': '1', 'id': '1',
'name': '检测计划1', 'name': '沧州110kV东光站装置送检',
'pattern':'1', 'pattern':'1',
"source_Id" : ['1'], "source_Id" : ['1'],
"testSourceName":'标准源-福禄克-6100A', "testSourceName":'标准源-FLUKE.6100A电能功率标准源-1',
'dataSource_Id':['1'], 'dataSource_Id':['1'],
'dataSource_Ids':'实时数据', 'dataSource_Ids':'实时数据',
'script_Id':'1', 'script_Id':'1',
@@ -256,32 +294,32 @@ const planData = ref<Plan.PlanAndSourceBO[]>([
}, },
{ {
'id': '2', 'id': '2',
'name': '检测计划2', 'name': '邯郸2台安徽振兴终端送检',
'pattern':'1', 'pattern':'1',
"source_Id" : ['5','6','7','8'], "source_Id" : ['5','6','7','8'],
"testSourceName":'高精度设备-PQV520-1', "testSourceName":'高精度设备-PQV-520便携式电能质量监测装置-1',
'father_Plan_Id':'1', 'father_Plan_Id':'1',
'dataSource_Id':['5'], 'dataSource_Id':['1'],
'dataSource_Ids':'分钟统计数据CP95值', 'dataSource_Ids':'实时数据',
'script_Id':'2', 'script_Id':'2',
'error_Sys_Id':'2', 'error_Sys_Id':'2',
'test_State':'2', 'test_State':'2',
'report_State':'2', 'report_State':'2',
'result':'0', 'result':'0',
"testSourceList":[ "testSourceList":[
'高精度设备-PQV520-2','高精度设备-PQV520-3','高精度设备-PQV520-4' '高精度设备-PQV-520便携式电能质量监测装置-2','高精度设备-PQV-520便携式电能质量监测装置-3','高精度设备-PQV-520便携式电能质量监测装置-4'
] ]
}, },
{ {
'id': '3', 'id': '3',
'name': '检测子计划3', 'name': '浙江分布式光伏电能质量试点',
'pattern':'1', 'pattern':'1',
"source_Id" : ['3'], "source_Id" : ['3'],
"testSourceName":'标准源-丹迪克-DKLN1', "testSourceName":'标准源-DKLN-1电能质量测试分析仪智能检定装置-1',
'father_Plan_Id':'1', 'father_Plan_Id':'1',
'dataSource_Id':['1'], 'dataSource_Id':['1'],
'dataSource_Ids':'实时数据', 'dataSource_Ids':'实时数据',
'script_Id':'3', 'script_Id':'4',
'error_Sys_Id':'3', 'error_Sys_Id':'3',
'test_State':'1', 'test_State':'1',
'report_State':'1', 'report_State':'1',

View File

@@ -2,12 +2,12 @@
"code": "200", "code": "200",
"data": [ "data": [
{ {
"name": "未检", "name": "未检",
"children": [ "children": [
{ {
"id": 4, "id": 4,
"pid": 1, "pid": 1,
"name": "邯郸220kV团城站等4座站电能质量检测" "name": "邯郸220kV团城站电能质量检测"
}, },
{ {
"id": 5, "id": 5,
@@ -17,7 +17,7 @@
{ {
"id": 6, "id": 6,
"pid": 1, "pid": 1,
"name": "深圳市中电软件有限公司委托送检" "name": "深圳市中电公司委托送检"
} }
], ],
"id": 1 "id": 1
@@ -28,17 +28,17 @@
{ {
"id": 7, "id": 7,
"pid": 2, "pid": 2,
"name": "沧州220kV留古等4座变电站电能质量检测" "name": "沧州220kV留古站电能质量检测"
}, },
{ {
"id": 8, "id": 8,
"pid": 2, "pid": 2,
"name": "沧州供电公司110kV东光站等10台装置送检" "name": "沧州110kV东光站装置送检"
}, },
{ {
"id": 9, "id": 9,
"pid": 2, "pid": 2,
"name": "保定分布式光伏电能质量监测试点" "name": "浙江分布式光伏电能质量试点"
} }
], ],
"id": 2 "id": 2
@@ -54,7 +54,7 @@
{ {
"id": 11, "id": 11,
"pid": 3, "pid": 3,
"name": "北京2024现场检测" "name": "深圳2台中电终端委托送检"
} }
], ],
"id": 3 "id": 3

View File

@@ -0,0 +1,51 @@
export interface MonitorResult {
/**
* 监测点id
*/
monitorId: string;
/**
* 监测点序号
*/
monitorNum: number;
/**
* 总检测次数
*/
totalNum: number;
/**
* 合格检测次数
*/
qualifiedNum: number;
/**
* 不合格检测次数
*/
unQualifiedNum: number;
/**
* 误差体系名称
*/
errorSysName: string;
/**
* 检测结果
*/
checkResult: number;
/**
* 哪次
*/
whichTime: string;
/**
* 结论来源
*/
resultOrigin: string;
/**
* 来源类型
*/
resultType: string;
}

View File

@@ -0,0 +1,7 @@
import http from '@/api'
export const getMonitorResult = (devId: string) => http.post(`/result/getMonitorResult?devId=${devId}`)
export const getMonitorDataSourceResult = (monitorId: string) =>
http.get(`/result/getMonitorDataSourceResult?monitorId=${monitorId}`)
export const updateMonitorResult = (data: any) => http.post('/result/updateMonitorResult', data)

View File

@@ -0,0 +1,46 @@
import http from '@/api'
export const startPreTest = (params) => {
return http.post(`/prepare/startPreTest`, params, {loading: false})
}
export const closePreTest = (params) => {
return http.post(`/prepare/closePreTest`, params,{ loading: false })
}
/**
* 开始正式检测
* @param params
*/
// export const startTest = (params) => {
// return http.post(`/prepare/startTest`, params, {loading: false})
// }
/**
* 暂停正式检测
* @param params
*/
export const pauseTest = () => {
return http.get(`/prepare/closePreTest`, {loading: false})
}
/**
* 继续正式检测
* @param params
*/
export const resumeTest = (params) => {
return http.post(`/prepare/restartTemTest/`, params, {loading: false})
}
/**
* 比对式通道配对
* @param params
*/
export const contrastTest = (params: any) => {
return http.post(`/prepare/startContrastTest`,params)
}
export const exportAlignData= () => {
return http.download(`/prepare/exportAlignData`)
}

View File

@@ -1,3 +1,4 @@
import http from '@/api' import http from '@/api'
import {type Base} from '@/api/system/base/interface' import {type Base} from '@/api/system/base/interface'
@@ -12,6 +13,10 @@ export const updateTestConfig = (params: Base.ResTestConfig) => {
return http.post(`/sysTestConfig/update`, params) return http.post(`/sysTestConfig/update`, params)
} }
//场景切换
export const updateScene = (params: any) => {
return http.post(`/sysTestConfig/update`,params)
}

View File

@@ -8,8 +8,8 @@ export namespace Base {
id: string; //系统配置表Id id: string; //系统配置表Id
autoGenerate:number;//检测报告是否自动生成0 否1是 autoGenerate:number;//检测报告是否自动生成0 否1是
maxTime:number;//最大复检次数默认3次 maxTime:number;//最大复检次数默认3次
dataRule:string;//数据处理原则,关联字典所有值、部分值、cp95值、平均值、任意值默认任意值
state: number; //状态 state: number; //状态
scale:number;//小数位
createBy?: string| null; //创建用户 createBy?: string| null; //创建用户
createTime?: string| null; //创建时间 createTime?: string| null; //创建时间
updateBy?: string| null; //更新用户 updateBy?: string| null; //更新用户

View File

@@ -216,7 +216,7 @@ const dictData: Dict[] = [
children: [ children: [
{ {
id: "0", id: "0",
label: 'FLUKE6100A电能功率标准源', label: 'FLUKE.6100A电能功率标准源',
code: 0, code: 0,
}, },
{ {
@@ -236,7 +236,7 @@ const dictData: Dict[] = [
}, },
{ {
id: "4", id: "4",
label: 'PQV-520电能质量测装置', label: 'PQV-520便携式电能质量测装置',
code: 4, code: 4,
}, },
], ],

View File

@@ -1,10 +1,16 @@
import http from '@/api' import http from '@/api'
import { type Dict } from '@/api/system/dictionary/interface' import { type Dict } from '@/api/system/dictionary/interface'
import { c } from 'vite/dist/node/types.d-aGj9QkWt'
//获取字典类型 //获取字典类型
export const getDictTreeList = (params: Dict.ResDictTree) => { export const getDictTreeByCode = (params: Dict.ResDictTree) => {
const name = params.name || ''; const code = params.code || ''
return http.get(`/dictTree/getTree?keyword=${name}`, params) return http.get(`/dictTree/getTreeByCode?code=${code}`, { loading: true })
}
export const getDictTreeByName = (params: Dict.ResDictTree) => {
const name = params.name || ''
return http.get(`/dictTree/getTreeByName?name=${name}`)
} }
//添加字典类型 //添加字典类型
@@ -21,4 +27,3 @@ export const updateDictTree = (params: Dict.ResDictTree) => {
export const deleteDictTree = (params: Dict.ResDictTree) => { export const deleteDictTree = (params: Dict.ResDictTree) => {
return http.post(`/dictTree/delete?id=${params.id}`) return http.post(`/dictTree/delete?id=${params.id}`)
} }

View File

@@ -62,6 +62,7 @@ export namespace Dict {
name: string; // 名称 name: string; // 名称
code: string; // 编码 code: string; // 编码
sort: number; // 排序 sort: number; // 排序
openValue?: number | null;
level?: number | null; // 事件等级0-普通1-中等2-严重 (默认为0) level?: number | null; // 事件等级0-普通1-中等2-严重 (默认为0)
algoDescribe?: number | null; // 与高级算法内部Id描述对应 algoDescribe?: number | null; // 与高级算法内部Id描述对应
value?: string | null; // 字典针对电压等级 value?: string | null; // 字典针对电压等级
@@ -116,10 +117,10 @@ export namespace Dict {
storeFlag?:string ;//sts、di的是否存储 1:存储 0:不存 储; storeFlag?:string ;//sts、di的是否存储 1:存储 0:不存 储;
curSts?:number | null;//sts、do的当前值 curSts?:number | null;//sts、do的当前值
ctlSts?:number;//do的是否可远程控制 1:是 0:否; ctlSts?:number;//do的是否可远程控制 1:是 0:否;
maxNum?:number ;//设置最大值 maxNum?:number | null;//设置最大值
minNum?: number;//设置最小值 minNum?: number| null;//设置最小值
setValue?:string | null;//参数为enum可设置的所有值序列 setValue?:string | null;//参数为enum可设置的所有值序列
strlen?:number ;//参数string可设置字符串的长度上 限 strlen?:number | null;//参数string可设置字符串的长度上 限
defaultValue?:string | null; //参数缺省值、告警code值 defaultValue?:string | null; //参数缺省值、告警code值
resourcesId?:string ; //报表数据来源(统计表表名) resourcesId?:string ; //报表数据来源(统计表表名)
limitName?:string | null; //限值字段名称 limitName?:string | null; //限值字段名称

View File

@@ -0,0 +1,16 @@
import type {AuditLog } from '@/api/system/log/interface/log.ts'
import http from '@/api'
/**
* @name 审计日志管理模块
*/
//获取审计日志
export const getAuditLog = (params: AuditLog.ReqAuditLogParams) => {
return http.post(`/sysLog/list`, params)
}
export const exportCsv = (params: AuditLog.ReqAuditLogParams) => {
return http.download(`/sysLog/exportCSV`, params)
}

View File

@@ -1,15 +1,36 @@
// 审计日志管理模块 import type { ReqPage, ResPage } from '@/api/interface'
export namespace Sys_Log_Audit {
// 日志列表
export interface Audit_LogList {
id: string;//日志表Id
operate_Type:string;//日志类型
ip:string;//操作IP
result: string;//事件结果
remark: string;//事件描述
warn:number;//告警标志
create_By:string;//创建用户
create_Time:string;//创建时间
}
} // 审计日志管理模块
export namespace AuditLog {
/**
* 设备类型数据表格分页查询参数
*/
export interface ReqAuditLogParams extends ReqPage {
id: string; //审计日志Id 必填
createTime?: string; //创建时间
}
/**
* 设备类型新增、修改、根据id查询返回的对象
*/
export interface ResAuditLog {
id: string;//审计日志Id
operate_Type:string;//操作类型
ip:string;//操作IP
result: string;//事件结果
remark: string;//事件描述
level:number;//告警等级
warn:number;//告警标志
create_By:string;//创建用户
create_Time:string;//创建时间
sort:number;//排序
}
/**
* 设备类型表格查询分页返回的对象;
*/
export interface ResAuditLogPage extends ResPage<ResAuditLog> {
}
}

View File

@@ -1,12 +1,11 @@
import http from '@/api' import http from '@/api'
import {type VersionRegister} from '@/api/system/versionRegister/interface' import { type VersionRegister } from '@/api/system/versionRegister/interface'
//获取有效数据配置 //获取有效数据配置
export const getRegRes = (params: VersionRegister.ResSys_Reg_Res) => { export const getRegRes = (params: { type: string }) => {
return http.get(`/sysRegRes/getRegResByType?id=${params.type}`) return http.get(`/sysRegRes/getRegResByType?id=${params.type}`)
} }
//编辑有效数据配置 //编辑有效数据配置
export const updateRegRes = (params: VersionRegister.Sys_Reg_Res) => { export const updateRegRes = (params: VersionRegister.Sys_Reg_Res) => {
return http.post(`/sysRegRes/update`, params) return http.post(`/sysRegRes/update`, params)

View File

@@ -1,20 +1,23 @@
// 登录模块 // 登录模块
import type { ReqPage,ResPage } from '@/api/interface' import type { ReqPage, ResPage } from '@/api/interface'
export namespace Login { export namespace Login {
export interface ReqLoginForm { export interface ReqLoginForm {
username: string; username: string
password: string; password: string
} checked: boolean
export interface ResLogin { }
accessToken: string; export interface ResLogin {
userInfo:{ accessToken: string
name: string; refreshToken: string
userInfo: {
id: string
name: string
}
}
export interface ResAuthButtons {
[key: string]: string[]
} }
}
export interface ResAuthButtons {
[key: string]: string[];
}
} }
@@ -49,6 +52,8 @@ export namespace User {
updateTime?: string;//更新时间 updateTime?: string;//更新时间
roleIds?: string[]; // roleIds?: string[]; //
roleNames?:string[]; // roleNames?:string[]; //
roleCodes?:string[]; //
disabled?: boolean;
} }
// 用户接口 // 用户接口

View File

@@ -1,32 +1,60 @@
import type { Login } from '@/api/user/interface/user' import { pa } from 'element-plus/es/locale/index.mjs';
import { ADMIN as rePrefix } from '@/api/system/config/serviceName' import type {Login} from '@/api/user/interface/user'
import {ADMIN as rePrefix} from '@/api/system/config/serviceName'
import http from '@/api' import http from '@/api'
import type { Dict } from '@/api/interface' import type {Dict} from '@/api/interface'
/** /**
* @name 登录模块 * @name 登录模块
*/ */
// 用户登录 // 用户登录
export const loginApi = (params: Login.ReqLoginForm) => { export const loginApi = (params: { username: string; password: string}) => {
return http.post<Login.ResLogin>(`${rePrefix}/login`, params, { loading: false }) return http.post<Login.ResLogin>(`${rePrefix}/login`, params, {loading: false})
// return http.post<Login.ResLogin>(`/Register1`, params, { loading: false }) // return http.post<Login.ResLogin>(`/Register1`, params, { loading: false })
} }
// 获取菜单列表 // 获取菜单列表
export const getAuthMenuListApi = () => { export const getAuthMenuListApi = () => {
return http.get<Menu.MenuOptions[]>(`/sysFunction/getMenu`, {}, { loading: false }) return http.get<Menu.MenuOptions[]>(`/sysFunction/getMenu`, {}, {loading: false})
// return http.post<Menu.MenuOptions[]>(`/Register2`, {}, { loading: false }) // return http.post<Menu.MenuOptions[]>(`/Register2`, {}, { loading: false })
} }
// 获取按钮权限 // 获取按钮权限
export const getAuthButtonListApi = () => { export const getAuthButtonListApi = () => {
return http.get<Login.ResAuthButtons>(`/sysFunction/getButton`, {}, { loading: false }) return http.get<Login.ResAuthButtons>(`/sysFunction/getButton`, {}, {loading: false})
// return http.post<Login.ResAuthButtons>(`/Register3`, {}, { loading: false }) // return http.post<Login.ResAuthButtons>(`/Register3`, {}, { loading: false })
} }
// 用户退出登录 // 用户退出登录
export const logoutApi = () => { export const logoutApi = () => {
return http.post(`${rePrefix}/logout`) return http.post(`${rePrefix}/logout`)
} }
//获取下拉框列表 //获取下拉框列表
export const getDictList = () =>{ export const getDictList = () => {
return http.get<Dict>('/dictData/dictDataCache') return http.get<Dict>('/dictData/dictDataCache')
} }
//token刷新
export const refreshToken = () => {
return http.get<Login.ResLogin>(`${rePrefix}/refreshToken`,
{},
{loading: false}
)
}
//获取场景
export const getCurrentScene = () => {
return http.get('/sysTestConfig/getCurrentScene', {}, {loading: false})
}
/**
* 获取RSA公钥
*/
export const getPublicKey = (username: string) => {
return http.get(`/admin/getPublicKey?username=${username}`, {}, {loading: false})
}
/**
* 获取是否在检测中自动生成报告
*/
export const getAutoGenerate = () => {
return http.get('/sysTestConfig/getAutoGenerate', {}, {loading: false})
}

View File

@@ -37,3 +37,7 @@ export const updatePassWord = (params: User.ResPassWordUser) => {
return http.post(`/sysUser/updatePassword`,params) return http.post(`/sysUser/updatePassword`,params)
} }
// 获取所有用户
export const getAllUser= () => {
return http.get(`/sysUser/getAll`)
}

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.4 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 54 KiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 62 KiB

View File

@@ -1,19 +1,20 @@
<template> <template>
<div class="not-container"> <div class="not-container">
<img src="@/assets/images/403.png" class="not-img" alt="403" /> <img src="@/assets/images/403.png" class="not-img" alt="403" />
<div class="not-detail"> <div class="not-detail">
<h2>403</h2> <h2>403</h2>
<h4>抱歉您无权访问该页面~🙅🙅</h4> <h4>抱歉您无权访问该页面~🙅🙅</h4>
<el-button type="primary" @click="router.back"> 返回上一页 </el-button> <el-button type="primary" @click="router.back">返回上一页</el-button>
</div>
</div> </div>
</div>
</template> </template>
<script setup lang="ts" name="403"> <script setup lang="ts" name="403">
import { useRouter } from "vue-router"; import { useRouter } from 'vue-router'
const router = useRouter();
const router = useRouter()
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,19 +1,20 @@
<template> <template>
<div class="not-container"> <div class="not-container">
<img src="@/assets/images/404.png" class="not-img" alt="404" /> <img src="@/assets/images/404.png" class="not-img" alt="404" />
<div class="not-detail"> <div class="not-detail">
<h2>404</h2> <h2>404</h2>
<h4>抱歉您访问的页面不存在~🤷🤷</h4> <h4>抱歉您访问的页面不存在~🤷🤷</h4>
<el-button type="primary" @click="router.back"> 返回上一页 </el-button> <el-button type="primary" @click="router.back">返回上一页</el-button>
</div>
</div> </div>
</div>
</template> </template>
<script setup lang="ts" name="404"> <script setup lang="ts" name="404">
import { useRouter } from "vue-router"; import { useRouter } from 'vue-router'
const router = useRouter();
const router = useRouter()
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,19 +1,20 @@
<template> <template>
<div class="not-container"> <div class="not-container">
<img src="@/assets/images/500.png" class="not-img" alt="500" /> <img src="@/assets/images/500.png" class="not-img" alt="500" />
<div class="not-detail"> <div class="not-detail">
<h2>500</h2> <h2>500</h2>
<h4>抱歉您的网络不见了~🤦🤦</h4> <h4>抱歉您的网络不见了~🤦🤦</h4>
<el-button type="primary" @click="router.back"> 返回上一页 </el-button> <el-button type="primary" @click="router.back">返回上一页</el-button>
</div>
</div> </div>
</div>
</template> </template>
<script setup lang="ts" name="500"> <script setup lang="ts" name="500">
import { useRouter } from "vue-router"; import { useRouter } from 'vue-router'
const router = useRouter();
const router = useRouter()
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,151 +1,207 @@
<template> <template>
<el-dialog v-model="dialogVisible" :title="`批量添加${parameter.title}`" :destroy-on-close="true" width="580px" draggable> <el-dialog
<el-form class="drawer-multiColumn-form" label-width="100px"> v-model="dialogVisible"
<el-form-item label="模板下载 :"> :title="`批量添加${parameter.title}`"
<el-button type="primary" :icon="Download" @click="downloadTemp"> 点击下载 </el-button> :destroy-on-close="true"
</el-form-item> width="580px"
<el-form-item label="文件上传 :"> draggable
<el-upload >
action="#" <el-form class="drawer-multiColumn-form" label-width="100px">
class="upload" <el-form-item label="模板下载 :">
:drag="true" <el-button type="primary" :icon="Download" @click="downloadTemp">点击下载</el-button>
:limit="excelLimit" </el-form-item>
:multiple="true" <el-form-item label="文件上传 :">
:show-file-list="true" <el-upload
:http-request="uploadExcel" action="#"
:before-upload="beforeExcelUpload" class="upload"
:on-exceed="handleExceed" :drag="true"
:on-success="excelUploadSuccess" :limit="excelLimit"
:on-error="excelUploadError" :multiple="true"
:accept="parameter.fileType!.join(',')" :show-file-list="true"
> :http-request="uploadExcel"
<slot name="empty"> :before-upload="beforeExcelUpload"
<el-icon class="el-icon--upload"> :on-exceed="handleExceed"
<upload-filled /> :accept="parameter.fileType!.join(',')"
</el-icon> >
<div class="el-upload__text">将文件拖到此处<em>点击上传</em></div> <slot name="empty">
</slot> <el-icon class="el-icon--upload">
<template #tip> <upload-filled />
<slot name="tip"> </el-icon>
<div class="el-upload__tip">请上传 .xls , .xlsx 标准格式文件文件最大为 {{ parameter.fileSize }}M</div> <div class="el-upload__text">
</slot> 将文件拖到此处
</template> <em>点击上传</em>
</el-upload> </div>
</el-form-item> </slot>
<el-form-item v-if="parameter.showCover" label="数据覆盖 :"> <template #tip>
<el-switch v-model="isCover" /> <slot name="tip">
</el-form-item> <div class="el-upload__tip">
</el-form> 请上传 .xls , .xlsx 标准格式文件文件最大为 {{ parameter.fileSize }}M
</el-dialog> </div>
</slot>
</template>
</el-upload>
</el-form-item>
<el-form-item v-if="parameter.showCover" label="数据覆盖 :">
<el-switch v-model="isCover" />
</el-form-item>
</el-form>
</el-dialog>
</template> </template>
<script setup lang="ts" name="ImportExcel"> <script setup lang="ts" name="ImportExcel">
import { ref } from "vue"; import { ref } from 'vue'
import { useDownload } from "@/hooks/useDownload"; import { useDownload } from '@/hooks/useDownload'
import { Download } from "@element-plus/icons-vue"; import { Download } from '@element-plus/icons-vue'
import { ElNotification, UploadRequestOptions, UploadRawFile } from "element-plus"; import { ElMessage, ElNotification, UploadRawFile, UploadRequestOptions } from 'element-plus'
export interface ExcelParameterProps { export interface ExcelParameterProps {
title: string; // 标题 title: string // 标题
showCover?: boolean; // 是否显示”数据覆盖“选项 showCover?: boolean // 是否显示”数据覆盖“选项
fileSize?: number; // 上传文件的大小 patternId?: string // 模式ID
fileType?: File.ExcelMimeType[]; // 上传文件的类型 planId?: string | null //计划ID
tempApi?: (params: any) => Promise<any>; // 下载模板的Api fileSize?: number // 上传文件的大小
importApi?: (params: any) => Promise<any>; // 批量导入的Api fileType?: File.ExcelMimeType[] // 上传文件的类型
getTableList?: () => void; // 获取表格数据的Api tempApi?: (params: any) => Promise<any> // 下载模板的Api
importApi?: (params: any) => Promise<any> // 批量导入的Api
getTableList?: () => void // 获取表格数据的Api
} }
// 是否覆盖数据 // 是否覆盖数据
const isCover = ref(false); const isCover = ref(false)
// 最大文件上传数 // 最大文件上传数
const excelLimit = ref(1); const excelLimit = ref(1)
// dialog状态 // dialog状态
const dialogVisible = ref(false); const dialogVisible = ref(false)
// 父组件传过来的参数 // 父组件传过来的参数
const parameter = ref<ExcelParameterProps>({ const parameter = ref<ExcelParameterProps>({
title: "", title: '',
fileSize: 5, fileSize: 5,
fileType: ["application/vnd.ms-excel", "application/vnd.openxmlformats-officedocument.spreadsheetml.sheet"] fileType: ['application/vnd.ms-excel', 'application/vnd.openxmlformats-officedocument.spreadsheetml.sheet']
}); })
const emit = defineEmits<{
(e: 'result', data: boolean): void
}>()
// 接收父组件参数 // 接收父组件参数
const acceptParams = (params: ExcelParameterProps) => { const acceptParams = (params: ExcelParameterProps) => {
parameter.value = { ...parameter.value, ...params }; parameter.value = { ...parameter.value, ...params }
dialogVisible.value = true; dialogVisible.value = true
}; }
// Excel 导入模板下载 // Excel 导入模板下载
const downloadTemp = () => { const downloadTemp = () => {
if (!parameter.value.tempApi) return; if (!parameter.value.tempApi) return
useDownload(parameter.value.tempApi, `${parameter.value.title}模板`,{},true); useDownload(parameter.value.tempApi, `${parameter.value.title}模板`, { pattern: parameter.value.patternId }, false)
}; }
// 文件上传 // 文件上传
const uploadExcel = async (param: UploadRequestOptions) => { const uploadExcel = async (param: UploadRequestOptions) => {
debugger let excelFormData = new FormData()
let excelFormData = new FormData(); excelFormData.append('file', param.file)
excelFormData.append("file", param.file); if (parameter.value.patternId) {
isCover.value && excelFormData.append("isCover", isCover.value as unknown as Blob); excelFormData.append('patternId', parameter.value.patternId)
await parameter.value.importApi!(excelFormData); }
parameter.value.getTableList && parameter.value.getTableList();
dialogVisible.value = false; excelFormData.append('planId', parameter.value.planId)
};
isCover.value && excelFormData.append('isCover', isCover.value as unknown as Blob)
//await parameter.value.importApi!(excelFormData);
await parameter.value.importApi!(excelFormData).then(res => handleImportResponse(res))
parameter.value.getTableList && parameter.value.getTableList()
dialogVisible.value = false
}
async function handleImportResponse(res: any) {
console.log(res)
if (res.type === 'application/json') {
const fileReader = new FileReader()
fileReader.onloadend = () => {
try {
const jsonData = JSON.parse(fileReader.result)
if (jsonData.code === 'A0000') {
ElMessage.success('导入成功')
} else {
ElMessageBox.alert(jsonData.message, {
title: '导入结果',
type: 'error'
})
}
emit('result', jsonData.data)
} catch (err) {
console.log(err)
}
}
fileReader.readAsText(res)
} else {
emit('result', false)
ElMessage.error('导入失败,请查看下载附件!')
let blob = new Blob([res], { type: 'application/vnd.ms-excel' })
const url = window.URL.createObjectURL(blob)
const link = document.createElement('a')
link.href = url
link.download = '导入失败数据'
document.body.appendChild(link)
link.click()
link.remove()
}
}
/** /**
* @description 文件上传之前判断 * @description 文件上传之前判断
* @param file 上传的文件 * @param file 上传的文件
* */ * */
const beforeExcelUpload = (file: UploadRawFile) => { const beforeExcelUpload = (file: UploadRawFile) => {
const isExcel = parameter.value.fileType!.includes(file.type as File.ExcelMimeType); const isExcel = parameter.value.fileType!.includes(file.type as File.ExcelMimeType)
const fileSize = file.size / 1024 / 1024 < parameter.value.fileSize!; const fileSize = file.size / 1024 / 1024 < parameter.value.fileSize!
if (!isExcel) if (!isExcel)
ElNotification({ ElNotification({
title: "温馨提示", title: '温馨提示',
message: "上传文件只能是 xls / xlsx 格式!", message: '上传文件只能是 xls / xlsx 格式!',
type: "warning" type: 'warning'
}); })
if (!fileSize) if (!fileSize)
setTimeout(() => { setTimeout(() => {
ElNotification({ ElNotification({
title: "温馨提示", title: '温馨提示',
message: `上传文件大小不能超过 ${parameter.value.fileSize}MB`, message: `上传文件大小不能超过 ${parameter.value.fileSize}MB`,
type: "warning" type: 'warning'
}); })
}, 0); }, 0)
return isExcel && fileSize; return isExcel && fileSize
}; }
// 文件数超出提示 // 文件数超出提示
const handleExceed = () => { const handleExceed = () => {
ElNotification({ ElNotification({
title: "温馨提示", title: '温馨提示',
message: "最多只能上传一个文件!", message: '最多只能上传一个文件!',
type: "warning" type: 'warning'
}); })
}; }
// 上传错误提示 // 上传错误提示
const excelUploadError = () => { // const excelUploadError = () => {
ElNotification({ // ElNotification({
title: "温馨提示", // title: '温馨提示',
message: `批量添加${parameter.value.title}失败,请您重新上传!`, // message: `批量添加${parameter.value.title}失败,请您重新上传!`,
type: "error" // type: 'error',
}); // })
}; // }
// 上传成功提示 // 上传成功提示
const excelUploadSuccess = () => { // const excelUploadSuccess = () => {
ElNotification({ // ElNotification({
title: "温馨提示", // title: '温馨提示',
message: `批量添加${parameter.value.title}成功!`, // message: `批量添加${parameter.value.title}成功!`,
type: "success" // type: 'success',
}); // })
}; // }
defineExpose({ defineExpose({
acceptParams acceptParams
}); })
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -0,0 +1,3 @@
.upload {
width: 80%;
}

View File

@@ -0,0 +1,241 @@
<template>
<el-dialog
v-model="dialogVisible"
:title="parameter.title"
:destroy-on-close="true"
width="450px"
:close-on-click-modal="!parameter.progressBar"
:show-close="!disable"
draggable
>
<el-upload
ref="uploadRef"
action="#"
class="upload"
:limit="1"
:http-request="uploadZip"
accept=".zip"
:auto-upload="!parameter.confirmMessage"
:on-change="handleChange"
:on-remove="handleRemove"
:disabled="fileDisabled"
>
<slot name="empty">
<el-button type="primary" :disabled="fileDisabled" icon="Upload">点击上传</el-button>
</slot>
<template #tip>
<slot name="tip">
<div class="el-upload__tip">请上传 .zip 标准格式文件</div>
</slot>
</template>
</el-upload>
<el-text v-if="parameter.progressBar && progressData.status === 'exception'" size="small" type="danger">
{{ progressData.message }}
</el-text>
<el-text v-if="parameter.progressBar && progressData.status === 'success'" size="small" type="success">
{{ progressData.message }}
</el-text>
<el-text v-if="parameter.progressBar && progressData.status === ''" size="small" type="info">
{{ progressData.message }}
</el-text>
<el-progress
style="margin-top: 10px; margin-bottom: 10px"
v-if="parameter.progressBar"
:status="progressData.status"
:percentage="progressData.percentage"
:stroke-width="10"
striped
striped-flow
></el-progress>
<template #footer v-if="parameter.confirmMessage">
<el-button :disabled="disable" type="primary" @click="uploadSubmit">开始导入</el-button>
</template>
</el-dialog>
</template>
<script setup lang="ts" name="ImportZip">
import { ref } from 'vue'
import type { UploadInstance, UploadProps, UploadRequestOptions } from 'element-plus'
import http from '@/api'
export interface ZipParameterProps {
title: string // 标题
patternId?: string // 模式ID
planId?: string // 计划ID
importApi?: (params: any) => Promise<any> // 批量导入的Api
confirmMessage?: string // 提示信息
progressBar?: boolean // 进度条
}
// dialog状态
const dialogVisible = ref(false)
const disable = ref(true)
const fileDisabled = ref(false)
const uploadRef = ref<UploadInstance>()
// 父组件传过来的参数
const parameter = ref<ZipParameterProps>({
title: ''
})
const emit = defineEmits<{
(e: 'result', data: boolean): void
}>()
// 接收父组件参数
const acceptParams = (params: ZipParameterProps) => {
parameter.value = { ...parameter.value, ...params }
dialogVisible.value = true
}
// 文件上传
const uploadZip = (param: UploadRequestOptions) => {
let zipFormData = new FormData()
zipFormData.append('file', param.file)
if (parameter.value.patternId) {
zipFormData.append('patternId', parameter.value.patternId)
}
if (parameter.value.planId) {
zipFormData.append('planId', parameter.value.planId)
}
if (parameter.value.progressBar) {
initSSE()
}
setTimeout(() => {
parameter.value.importApi!(zipFormData)
.then(res => handleImportResponse(res))
.catch(err => {
fileDisabled.value = false
disable.value = false
})
}, 1000)
}
const handleImportResponse = (res: any) => {
if (!parameter.value.progressBar) {
if (res.code === 'A0000') {
ElMessage.success('导入成功')
emit('result', true)
dialogVisible.value = false
} else {
ElMessage.error(res.message)
fileDisabled.value = false
disable.value = false
}
} else {
if (res.code !== 'A0000') {
ElMessage.error(res.message)
}
}
}
const uploadSubmit = () => {
if (!uploadRef.value) {
return ElMessage.warning('请选择文件!')
}
progressData.value = {
percentage: 0,
status: '',
message: ''
}
ElMessageBox.confirm(parameter.value.confirmMessage, '温馨提示', {
confirmButtonText: '确定',
cancelButtonText: '取消',
type: 'warning'
})
.then(() => {
disable.value = true
fileDisabled.value = true
uploadRef.value?.submit()
})
.catch(() => {
disable.value = false
fileDisabled.value = false
})
}
const handleChange: UploadProps['onChange'] = (uploadFile, uploadFiles) => {
disable.value = uploadFiles.length === 0
progressData.value = {
percentage: 0,
status: '',
message: ''
}
}
const handleRemove: UploadProps['onRemove'] = (uploadFile, uploadFiles) => {
disable.value = uploadFiles.length === 0
progressData.value = {
percentage: 0,
status: '',
message: ''
}
}
const progressData = ref({
percentage: 0,
status: '',
message: ''
})
const eventSource = ref<EventSource | null>(null)
const initSSE = () => {
eventSource.value = http.sse('/sse/createSse')
eventSource.value.onmessage = event => {
console.log('收到消息内容是:', event.data)
const res = JSON.parse(event.data)
progressData.value.percentage = res.data
progressData.value.message = res.message
if (res.code === 'A0002') {
fileDisabled.value = false
disable.value = false
progressData.value.status = 'exception'
ElMessage.error(res.message)
}
if (progressData.value.percentage === 100) {
progressData.value.status = 'success'
eventSource.value!.close()
ElMessage.success('导入成功')
emit('result', true)
dialogVisible.value = false
}
}
eventSource.value.onerror = error => {
console.warn('SSE 连接出错:', error)
eventSource.value!.close()
}
}
// 添加一个手动关闭EventSource的函数
const closeEventSource = () => {
if (eventSource.value) {
eventSource.value.close()
eventSource.value = null
console.log('SSE连接已关闭')
}
}
// 监听 dialogVisible 的变化,确保在对话框关闭时清理资源
watch(dialogVisible, newVal => {
if (!newVal) {
// 延迟执行,确保在组件完全关闭后清理
setTimeout(() => {
closeEventSource()
fileDisabled.value = false
disable.value = false
progressData.value = {
percentage: 0,
status: '',
message: ''
}
}, 100)
}
})
onUnmounted(() => {
closeEventSource()
})
defineExpose({
acceptParams
})
</script>
<style lang="scss" scoped>
@use './index.scss';
</style>

View File

@@ -9,5 +9,5 @@
<script setup lang="ts" name="Loading"></script> <script setup lang="ts" name="Loading"></script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use "./index.scss";;
</style> </style>

View File

@@ -8,9 +8,8 @@
:search-param='searchParam' :search-param='searchParam'
:search-col='searchCol' :search-col='searchCol'
/> />
<!-- 表格主体 --> <!-- 表格主体 -->
<div class='card table-main'> <div class='table-main' :class='{ card: showCard }' >
<!-- 表格头部 操作按钮 --> <!-- 表格头部 操作按钮 -->
<div class='table-header'> <div class='table-header'>
<div class='header-button-lf'> <div class='header-button-lf'>
@@ -43,10 +42,11 @@
> >
<!-- 默认插槽 --> <!-- 默认插槽 -->
<slot /> <slot />
<template v-for='item in tableColumns' :key='item'> <template v-for='item in tableColumns' :key='item'>
<!-- selection || radio || index || expand || sort --> <!-- selection || radio || index || expand || sort -->
<el-table-column <el-table-column
v-if='item.type && columnTypes.includes(item.type)' v-if='item.type && columnTypes.includes(item.type) && item.isShow'
v-bind='item' v-bind='item'
:align="item.align ?? 'center'" :align="item.align ?? 'center'"
:reserve-selection="item.type == 'selection'" :reserve-selection="item.type == 'selection'"
@@ -119,6 +119,7 @@ import TableColumn from './components/TableColumn.vue'
import Sortable from 'sortablejs' import Sortable from 'sortablejs'
export interface ProTableProps { export interface ProTableProps {
columns: ColumnProps[]; // 列配置项 ==> 必传 columns: ColumnProps[]; // 列配置项 ==> 必传
data?: any[]; // 静态 table data 数据,若存在则不会使用 requestApi 返回的 data ==> 非必传 data?: any[]; // 静态 table data 数据,若存在则不会使用 requestApi 返回的 data ==> 非必传
requestApi?: (params: any) => Promise<any>; // 请求表格数据的 api ==> 非必传 requestApi?: (params: any) => Promise<any>; // 请求表格数据的 api ==> 非必传
@@ -126,6 +127,7 @@ export interface ProTableProps {
requestError?: (params: any) => void; // 表格 api 请求错误监听 ==> 非必传 requestError?: (params: any) => void; // 表格 api 请求错误监听 ==> 非必传
dataCallback?: (data: any) => any; // 返回数据的回调函数,可以对数据进行处理 ==> 非必传 dataCallback?: (data: any) => any; // 返回数据的回调函数,可以对数据进行处理 ==> 非必传
title?: string; // 表格标题 ==> 非必传 title?: string; // 表格标题 ==> 非必传
showCard?: boolean; // 下个是否需要卡片
pagination?: boolean; // 是否需要分页组件 ==> 非必传默认为true pagination?: boolean; // 是否需要分页组件 ==> 非必传默认为true
initParam?: any; // 初始化请求参数 ==> 非必传(默认为{} initParam?: any; // 初始化请求参数 ==> 非必传(默认为{}
border?: boolean; // 是否带有纵向边框 ==> 非必传默认为true border?: boolean; // 是否带有纵向边框 ==> 非必传默认为true
@@ -141,6 +143,7 @@ const props = withDefaults(defineProps<ProTableProps>(), {
pagination: true, pagination: true,
initParam: {}, initParam: {},
border: true, border: true,
showCard: true,
toolButton: true, toolButton: true,
rowKey: 'id', rowKey: 'id',
searchCol: () => ({ xs: 1, sm: 2, md: 2, lg: 3, xl: 4 }), searchCol: () => ({ xs: 1, sm: 2, md: 2, lg: 3, xl: 4 }),

View File

@@ -1,49 +1,49 @@
<template> <template>
<div class='icon-box'> <div class="icon-box">
<el-input <el-input
ref='inputRef' ref="inputRef"
v-model='valueIcon' v-model="valueIcon"
v-bind='$attrs' v-bind="$attrs"
:placeholder='placeholder' :placeholder="placeholder"
:clearable='clearable' :clearable="clearable"
@clear='clearIcon' @clear="clearIcon"
@click='openDialog' @click="openDialog"
> >
<template #append> <template #append>
<el-button :icon='customIcons[iconValue]' /> <el-button :icon="customIcons[iconValue]" />
</template> </template>
</el-input> </el-input>
<el-dialog v-model='dialogVisible' :title='placeholder' top='50px' width='66%'> <el-dialog v-model="dialogVisible" :title="placeholder" top="5%" width="30%">
<el-input v-model='inputValue' placeholder='搜索图标' size='large' :prefix-icon='Icons.Search' /> <el-input v-model="inputValue" placeholder="搜索图标" size="large" :prefix-icon="Icons.Search" />
<el-scrollbar v-if='Object.keys(iconsList).length'> <el-scrollbar v-if="Object.keys(iconsList).length">
<div class='icon-list'> <div class="icon-list">
<div v-for='item in iconsList' :key='item' class='icon-item' @click='selectIcon(item)'> <div v-for="item in iconsList" :key="item" class="icon-item" @click="selectIcon(item)">
<component :is='item'></component> <component :is="item"></component>
<span>{{ item.name }}</span> <span>{{ item.name }}</span>
</div> </div>
</div> </div>
</el-scrollbar> </el-scrollbar>
<el-empty v-else description='未搜索到您要找的图标~' /> <el-empty v-else description="未搜索到您要找的图标~" />
</el-dialog> </el-dialog>
</div> </div>
</template> </template>
<script lang='ts' setup name='SelectIcon'> <script lang="ts" setup name="SelectIcon">
import * as Icons from '@element-plus/icons-vue' import * as Icons from '@element-plus/icons-vue'
import { computed, ref } from 'vue'; import { computed, ref } from 'vue'
interface SelectIconProps { interface SelectIconProps {
iconValue: string| undefined; iconValue: string | undefined
title?: string; title?: string
clearable?: boolean; clearable?: boolean
placeholder?: string; placeholder?: string
} }
const props = withDefaults(defineProps<SelectIconProps>(), { const props = withDefaults(defineProps<SelectIconProps>(), {
iconValue: '', iconValue: '',
title: '请选择图标', title: '请选择图标',
clearable: true, clearable: true,
placeholder: '请选择图标', placeholder: '请选择图标'
}) })
// 重新接收一下,防止打包后 clearable 报错 // 重新接收一下,防止打包后 clearable 报错
@@ -55,37 +55,36 @@ const openDialog = () => (dialogVisible.value = true)
// 选择图标(触发更新父组件数据) // 选择图标(触发更新父组件数据)
const emit = defineEmits<{ const emit = defineEmits<{
'update:iconValue': [value: string]; 'update:iconValue': [value: string]
}>() }>()
const selectIcon = (item: any) => { const selectIcon = (item: any) => {
dialogVisible.value = false dialogVisible.value = false
valueIcon.value = item.name valueIcon.value = item.name
emit('update:iconValue', item.name) emit('update:iconValue', item.name)
setTimeout(() => inputRef.value.blur(), 0) setTimeout(() => inputRef.value.blur(), 0)
} }
// 清空图标 // 清空图标
const inputRef = ref() const inputRef = ref()
const clearIcon = () => { const clearIcon = () => {
valueIcon.value = '' valueIcon.value = ''
emit('update:iconValue', '') emit('update:iconValue', '')
setTimeout(() => inputRef.value.blur(), 0) setTimeout(() => inputRef.value.blur(), 0)
} }
// 监听搜索框值 // 监听搜索框值
const inputValue = ref('') const inputValue = ref('')
const customIcons: { [key: string]: any } = Icons const customIcons: { [key: string]: any } = Icons
const iconsList = computed((): { [key: string]: any } => { const iconsList = computed((): { [key: string]: any } => {
if (!inputValue.value) return Icons if (!inputValue.value) return Icons
let result: { [key: string]: any } = {} let result: { [key: string]: any } = {}
for (const key in customIcons) { for (const key in customIcons) {
if (key.toLowerCase().indexOf(inputValue.value.toLowerCase()) > -1) result[key] = customIcons[key] if (key.toLowerCase().indexOf(inputValue.value.toLowerCase()) > -1) result[key] = customIcons[key]
} }
return result return result
}) })
</script> </script>
<style scoped lang='scss'> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,97 +1,84 @@
<template> <template>
<div class='time-control'> <div class="time-control">
<el-select <el-select class="select" v-model="timeUnit" placeholder="选择时间单位" @change="handleChange">
class='select' <!-- 采用 v-for 动态渲染 -->
v-model='timeUnit' <el-option v-for="unit in timeUnits" :key="unit.value" :label="unit.label" :value="unit.value"></el-option>
placeholder='选择时间单位' </el-select>
@change='handleChange'
>
<!-- 采用 v-for 动态渲染 -->
<el-option
v-for='unit in timeUnits'
:key='unit.value'
:label='unit.label'
:value='unit.value'
></el-option>
</el-select>
<!-- 禁用时间选择器 --> <!-- 禁用时间选择器 -->
<div class='date-display'> <div class="date-display">
<el-date-picker <el-date-picker
class='date-picker' class="date-picker"
v-model='startDate' v-model="startDate"
type='date' type="date"
placeholder='起始时间' placeholder="起始时间"
@change='emitDateChange' @change="emitDateChange"
:disabled-date='disableStartDate' :disabled-date="disableStartDate"
:readonly="timeUnit != '自定义'" :readonly="timeUnit != '自定义'"
></el-date-picker> ></el-date-picker>
<el-text>~</el-text> <el-text>~</el-text>
<el-date-picker <el-date-picker
class='date-picker' class="date-picker"
v-model='endDate' v-model="endDate"
type='date' type="date"
placeholder='结束时间' placeholder="结束时间"
@change='emitDateChange' @change="emitDateChange"
:disabled-date='disableEndDate' :disabled-date="disableEndDate"
:readonly="timeUnit !== '自定义'" :readonly="timeUnit !== '自定义'"
></el-date-picker> ></el-date-picker>
</div>
<div class="date-display" v-if="timeUnit !== '自定义'">
<el-button
style="width: 10px"
class="triangle-button"
type="primary"
@click="prevPeriod"
@change="emitDateChange"
>
<div class="left_triangle"></div>
</el-button>
<el-button class="triangle-button" type="primary" @click="goToCurrent">当前</el-button>
<el-button
style="width: 10px"
class="triangle-button"
type="primary"
@click="nextPeriod"
:disabled="isNextDisabled"
>
<div class="right_triangle"></div>
</el-button>
</div>
</div> </div>
<div class='date-display' v-if="timeUnit !== '自定义'">
<el-button
style='width: 10px;'
class='triangle-button'
type='primary'
@click='prevPeriod'
@change='emitDateChange'
>
<div class='left_triangle'></div>
</el-button>
<el-button class='triangle-button' type='primary' @click='goToCurrent'>
当前
</el-button>
<el-button
style='width: 10px;'
class='triangle-button'
type='primary'
@click='nextPeriod'
:disabled='isNextDisabled'
>
<div class='right_triangle'></div>
</el-button>
</div>
</div>
</template> </template>
<script setup lang="ts">
<script setup lang='ts'> import { onMounted, ref } from 'vue'
import { ref, onMounted, defineProps, defineEmits } from 'vue'
// 定义时间单位的类型 // 定义时间单位的类型
interface TimeUnit { interface TimeUnit {
label: string; label: string
value: string; value: string
} }
// 定义组件的props包含包括和排除的时间单位 // 定义组件的props包含包括和排除的时间单位
const props = defineProps({ const props = defineProps({
include: { include: {
type: Array as () => string[], type: Array as () => string[],
default: () => ['日', '周', '月', '季度', '年', '自定义'], default: () => ['日', '周', '月', '季度', '年', '自定义']
}, },
exclude: { exclude: {
type: Array as () => string[], type: Array as () => string[],
default: () => [], default: () => []
}, },
default: { default: {
type: String, type: String,
default: '月', default: '月'
}, }
}) })
// 定义事件 // 定义事件
const emit = defineEmits<{ const emit = defineEmits<{
(e: 'update-dates', startDate: string, endDate: string): void; (e: 'update-dates', startDate: string, endDate: string): void
}>() }>()
const timeUnit = ref<string>(props.default) // 默认选择 const timeUnit = ref<string>(props.default) // 默认选择
const startDate = ref<Date>(new Date()) // 起始日期 const startDate = ref<Date>(new Date()) // 起始日期
@@ -100,201 +87,206 @@ const isNextDisabled = ref<boolean>(false) // 控制下一周期按钮的禁用
const today = ref<Date>(new Date()) // 当前日期 const today = ref<Date>(new Date()) // 当前日期
// 过滤出可用的时间单位 // 过滤出可用的时间单位
const timeUnits = ref<TimeUnit[]>( const timeUnits = ref<TimeUnit[]>(
props.include.filter(unit => !props.exclude.includes(unit)).map(unit => ({ props.include
label: unit, .filter(unit => !props.exclude.includes(unit))
value: unit, .map(unit => ({
})), label: unit,
value: unit
}))
) )
// 发出日期变化事件 // 发出日期变化事件
const emitDateChange = () => { const emitDateChange = () => {
emit('update-dates', formatDate(startDate.value), formatDate(endDate.value)) emit('update-dates', formatDate(startDate.value), formatDate(endDate.value))
} }
// 在组件挂载时更新日期范围 // 在组件挂载时更新日期范围
onMounted(() => { onMounted(() => {
updateDateRange() updateDateRange()
}) })
const handleChange = (unit: string) => { const handleChange = (unit: string) => {
// 根据选择的时间单位处理日期变化 // 根据选择的时间单位处理日期变化
if (unit !== '自定义') { if (unit !== '自定义') {
updateDateRange() updateDateRange()
} else { } else {
// 自定义选项逻辑 // 自定义选项逻辑
startDate.value = new Date(new Date().setDate(new Date().getDate() - 1)) startDate.value = new Date(new Date().setDate(new Date().getDate() - 1))
endDate.value = new Date() endDate.value = new Date()
} }
timeUnit.value = unit timeUnit.value = unit
emitDateChange() // 变化时也发出更新事件
updateNextButtonStatus() // 确保开始时间和结束时间不为空
if (!startDate.value) {
startDate.value = new Date()
}
if (!endDate.value) {
endDate.value = new Date()
}
emitDateChange() // 变化时也发出更新事件
updateNextButtonStatus()
} }
const updateDateRange = () => { const updateDateRange = () => {
// 根据选择的时间单位计算起始和结束日期
if (timeUnit.value === '日') {
startDate.value = today.value
endDate.value = today.value
} else if (timeUnit.value === '周') {
startDate.value = getStartOfWeek(today.value)
endDate.value = getEndOfWeek(today.value)
} else if (timeUnit.value === '月') {
// 获取本月的开始和结束日期
startDate.value = new Date(today.value.getFullYear(), today.value.getMonth(), 1)
endDate.value = new Date(today.value.getFullYear(), today.value.getMonth() + 1, 0)
// 根据选择的时间单位计算起始和结束日期 // // 确保结束日期不超过今天
if (timeUnit.value === '日') { // if (endDate.value > today.value) {
startDate.value = today.value // endDate.value = new Date(today.value);
endDate.value = today.value // endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
} else if (timeUnit.value === '周') { // }
startDate.value = getStartOfWeek(today.value) } else if (timeUnit.value === '季度') {
endDate.value = getEndOfWeek(today.value) const quarter = Math.floor(today.value.getMonth() / 3)
startDate.value = new Date(today.value.getFullYear(), quarter * 3, 1)
} else if (timeUnit.value === '月') { endDate.value = new Date(today.value.getFullYear(), quarter * 3 + 3, 0)
// 获取本月的开始和结束日期
startDate.value = new Date(today.value.getFullYear(), today.value.getMonth(), 1);
endDate.value = new Date(today.value.getFullYear(), today.value.getMonth() + 1, 0);
// // 确保结束日期不超过今天
// if (endDate.value > today.value) {
// endDate.value = new Date(today.value);
// endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
// }
} else if (timeUnit.value === '季度') { // // 确保结束日期不超过今天
const quarter = Math.floor(today.value.getMonth() / 3); // if (endDate.value > today.value) {
startDate.value = new Date(today.value.getFullYear(), quarter * 3, 1); // endDate.value = new Date(today.value);
endDate.value = new Date(today.value.getFullYear(), quarter * 3 + 3, 0); // endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
// }
} else if (timeUnit.value === '年') {
startDate.value = new Date(today.value.getFullYear(), 0, 1)
endDate.value = new Date(today.value.getFullYear(), 11, 31)
// // 确保结束日期不超过今天 // // 确保结束日期不超过今天
// if (endDate.value > today.value) { // if (endDate.value > today.value) {
// endDate.value = new Date(today.value);
// endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
// }
} else if (timeUnit.value === '年') { // endDate.value = new Date(today.value);
startDate.value = new Date(today.value.getFullYear(), 0, 1); // endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
endDate.value = new Date(today.value.getFullYear(), 11, 31); // }
}
// 确保开始时间和结束时间不为空
if (!startDate.value) {
startDate.value = new Date()
}
if (!endDate.value) {
endDate.value = new Date()
}
// // 确保结束日期不超过今天 updateNextButtonStatus()
// if (endDate.value > today.value) {
// endDate.value = new Date(today.value);
// endDate.value.setHours(23, 59, 59, 999); // 设置结束时间为今天的23:59:59.999
// }
}
updateNextButtonStatus()
} }
const getStartOfWeek = (date: Date) => { const getStartOfWeek = (date: Date) => {
const startOfWeek = new Date(date) const startOfWeek = new Date(date)
const day = startOfWeek.getDay() const day = startOfWeek.getDay()
const diff = day === 0 ? -6 : 1 - day // 星期天的情况 const diff = day === 0 ? -6 : 1 - day // 星期天的情况
startOfWeek.setDate(startOfWeek.getDate() + diff) startOfWeek.setDate(startOfWeek.getDate() + diff)
return startOfWeek return startOfWeek
} }
const getEndOfWeek = (date: Date) => { const getEndOfWeek = (date: Date) => {
const endOfWeek = new Date(date) const endOfWeek = new Date(date)
const day = endOfWeek.getDay() const day = endOfWeek.getDay()
const diff = day === 0 ? 0 : 7 - day // 星期天的情况 const diff = day === 0 ? 0 : 7 - day // 星期天的情况
endOfWeek.setDate(endOfWeek.getDate() + diff) endOfWeek.setDate(endOfWeek.getDate() + diff)
// 获取今天的日期 // 获取今天的日期
const today = new Date(); const today = new Date()
today.setHours(23, 59, 59, 999); // 设置今天的结束时间23:59:59.999 today.setHours(23, 59, 59, 999) // 设置今天的结束时间23:59:59.999
// 返回不超过今天的结束时间 // 返回不超过今天的结束时间
//return endOfWeek > today ? today : endOfWeek; //return endOfWeek > today ? today : endOfWeek;
return endOfWeek return endOfWeek
} }
const prevPeriod = () => { const prevPeriod = () => {
const prevStartDate = new Date(startDate.value) const prevStartDate = new Date(startDate.value)
const prevEndDate = new Date(endDate.value) const prevEndDate = new Date(endDate.value)
if (timeUnit.value === '日') { if (timeUnit.value === '日') {
prevStartDate.setDate(prevStartDate.getDate() - 1) prevStartDate.setDate(prevStartDate.getDate() - 1)
prevEndDate.setDate(prevEndDate.getDate() - 1) prevEndDate.setDate(prevEndDate.getDate() - 1)
} else if (timeUnit.value === '周') { } else if (timeUnit.value === '周') {
prevStartDate.setDate(prevStartDate.getDate() - 7) prevStartDate.setDate(prevStartDate.getDate() - 7)
prevEndDate.setDate(prevEndDate.getDate() - 7) prevEndDate.setDate(prevEndDate.getDate() - 7)
} else if (timeUnit.value === '月') { } else if (timeUnit.value === '月') {
prevStartDate.setMonth(prevStartDate.getMonth() - 1)
prevEndDate.setMonth(prevEndDate.getMonth() - 1)
} else if (timeUnit.value === '季度') {
prevStartDate.setMonth(prevStartDate.getMonth() - 3)
prevEndDate.setMonth(prevEndDate.getMonth() - 3)
} else if (timeUnit.value === '年') {
prevStartDate.setFullYear(prevStartDate.getFullYear() - 1)
prevEndDate.setFullYear(prevEndDate.getFullYear() - 1)
}
startDate.value = prevStartDate
prevStartDate.setMonth(prevStartDate.getMonth() - 1) endDate.value = prevEndDate
prevEndDate.setMonth(prevEndDate.getMonth() - 1) updateNextButtonStatus()
} else if (timeUnit.value === '季度') {
prevStartDate.setMonth(prevStartDate.getMonth() - 3)
prevEndDate.setMonth(prevEndDate.getMonth() - 3)
} else if (timeUnit.value === '年') {
prevStartDate.setFullYear(prevStartDate.getFullYear() - 1)
prevEndDate.setFullYear(prevEndDate.getFullYear() - 1)
}
startDate.value = prevStartDate
endDate.value = prevEndDate
updateNextButtonStatus()
} }
const goToCurrent = () => { const goToCurrent = () => {
if (timeUnit.value !== '自定义') { if (timeUnit.value !== '自定义') {
updateDateRange() // 更新为当前选择时间单位的时间范围 updateDateRange() // 更新为当前选择时间单位的时间范围
} }
} }
const nextPeriod = () => { const nextPeriod = () => {
const nextStartDate = new Date(startDate.value) const nextStartDate = new Date(startDate.value)
const nextEndDate = new Date(endDate.value) const nextEndDate = new Date(endDate.value)
if (timeUnit.value === '日') { if (timeUnit.value === '日') {
nextStartDate.setDate(nextStartDate.getDate() + 1) nextStartDate.setDate(nextStartDate.getDate() + 1)
nextEndDate.setDate(nextEndDate.getDate() + 1) nextEndDate.setDate(nextEndDate.getDate() + 1)
} else if (timeUnit.value === '周') { } else if (timeUnit.value === '周') {
nextStartDate.setDate(nextStartDate.getDate() + 7) nextStartDate.setDate(nextStartDate.getDate() + 7)
nextEndDate.setDate(nextEndDate.getDate() + 7) nextEndDate.setDate(nextEndDate.getDate() + 7)
} else if (timeUnit.value === '月') { } else if (timeUnit.value === '月') {
nextStartDate.setMonth(nextStartDate.getMonth() + 1) nextStartDate.setMonth(nextStartDate.getMonth() + 1)
nextEndDate.setMonth(nextEndDate.getMonth() + 1) nextEndDate.setMonth(nextEndDate.getMonth() + 1)
} else if (timeUnit.value === '季度') { } else if (timeUnit.value === '季度') {
nextStartDate.setMonth(nextStartDate.getMonth() + 3) nextStartDate.setMonth(nextStartDate.getMonth() + 3)
nextEndDate.setMonth(nextStartDate.getMonth() + 3) nextEndDate.setMonth(nextStartDate.getMonth() + 3)
} else if (timeUnit.value === '年') { } else if (timeUnit.value === '年') {
nextStartDate.setFullYear(nextStartDate.getFullYear() + 1) nextStartDate.setFullYear(nextStartDate.getFullYear() + 1)
nextEndDate.setFullYear(nextEndDate.getFullYear() + 1) nextEndDate.setFullYear(nextEndDate.getFullYear() + 1)
} }
startDate.value = nextStartDate startDate.value = nextStartDate
endDate.value = nextEndDate endDate.value = nextEndDate
updateNextButtonStatus() updateNextButtonStatus()
} }
const updateNextButtonStatus = () => { const updateNextButtonStatus = () => {
// 更新下一个按钮的禁用状态
// 更新下一个按钮的禁用状态 const maxDate = new Date() // 假设最新日期为今天
const maxDate = new Date() // 假设最新日期为今天 // 将 maxDate 设置为当天的开始时间
// 将 maxDate 设置为当天的开始时间 maxDate.setHours(0, 0, 0, 0)
maxDate.setHours(0, 0, 0, 0) // 将 endDate 设置为当天的开始时间并进行比较
// 将 endDate 设置为当天的开始时间并进行比较 const endDateAdjusted = new Date(endDate.value)
const endDateAdjusted = new Date(endDate.value) endDateAdjusted.setHours(0, 0, 0, 0)
endDateAdjusted.setHours(0, 0, 0, 0) // 仅比较日期部分
// 仅比较日期部分 isNextDisabled.value = endDateAdjusted >= maxDate
isNextDisabled.value = endDateAdjusted >= maxDate emitDateChange() // 变化时也发出更新事件
emitDateChange() // 变化时也发出更新事件
} }
// 限制开始日期不能选择超过当前日期 // 限制开始日期不能选择超过当前日期
const disableStartDate = (date: Date) => { const disableStartDate = (date: Date) => {
return date > today.value return date > today.value
} }
// 限制结束日期不能超过当前日期且必须大于开始日期 // 限制结束日期不能超过当前日期且必须大于开始日期
const disableEndDate = (date: Date) => { const disableEndDate = (date: Date) => {
if (timeUnit.value !== '自定义') return false // 如果不是自定义时间单位,则不限制 if (timeUnit.value !== '自定义') return false // 如果不是自定义时间单位,则不限制
const start = new Date(startDate.value) const start = new Date(startDate.value)
return date > today.value || (start && date <= start) return date > today.value || (start && date <= start)
} }
// 格式化日期yyyy-mm-dd // 格式化日期yyyy-mm-dd
function formatDate(date:Date) { function formatDate(date: Date | null): string {
const year = date.getFullYear(); if (!date) {
const month = String(date.getMonth() + 1).padStart(2, '0'); return ''
const day = String(date.getDate()).padStart(2, '0'); }
return `${year}-${month}-${day}`; const year = date.getFullYear()
const month = String(date.getMonth() + 1).padStart(2, '0')
const day = String(date.getDate()).padStart(2, '0')
return `${year}-${month}-${day}`
} }
</script> </script>
<style scoped lang='scss'> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -0,0 +1,282 @@
<template>
<div class="chart">
<div ref="chartRef" class="my-chart" />
</div>
</template>
<script setup lang="ts">
import { onBeforeUnmount, onMounted, ref, watch } from 'vue'
// import echarts from './echarts'
import * as echarts from 'echarts' // 全引入
// import 'echarts/lib/component/dataZoom'
const color = [
'var(--el-color-primary)',
'#07CCCA',
'#00BFF5',
'#FFBF00',
'#77DA63',
'#D5FF6B',
'#Ff6600',
'#FF9100',
'#5B6E96',
'#66FFCC',
'#B3B3B3'
]
const chartRef = ref<HTMLDivElement>()
const props = defineProps(['options', 'isInterVal', 'pieInterVal'])
let chart: echarts.ECharts | any = null
const resizeHandler = () => {
// 不在视野中的时候不进行resize
if (!chartRef.value) return
if (chartRef.value.offsetHeight == 0) return
chart.getZr().painter.getViewportRoot().style.display = 'none'
requestAnimationFrame(() => {
chart.resize()
chart.getZr().painter.getViewportRoot().style.display = ''
})
}
const initChart = () => {
if (!props.isInterVal && !props.pieInterVal) {
chart?.dispose()
}
// chart?.dispose()
chart = echarts.init(chartRef.value as HTMLDivElement)
const options = {
title: {
left: 'center',
// textStyle: {
color: '#000',
fontSize: 18,
// },
...(props.options?.title || null)
},
tooltip: {
trigger: 'axis',
axisPointer: {
type: 'shadow',
label: {
color: '#fff',
fontSize: 16
}
},
textStyle: {
color: '#fff',
fontStyle: 'normal',
opacity: 0.35,
fontSize: 14
},
backgroundColor: 'rgba(0,0,0,0.55)',
borderWidth: 0,
confine: true,
...(props.options?.tooltip || null)
},
legend: {
right: 20,
top: 0,
itemGap: 10,
itemStyle: {},
// textStyle: {
fontSize: 12,
padding: [2, 0, 0, 0], //[上、右、下、左]
// },
itemWidth: 15,
itemHeight: 10,
...(props.options?.legend || null)
},
grid: {
top: '60px',
left: '30px',
right: '70px',
bottom: props.options?.options?.dataZoom === null ? '10px' : '40px',
containLabel: true,
...(props.options?.grid || null)
},
xAxis: props.options?.xAxis ? handlerXAxis() : null,
yAxis: props.options?.yAxis ? handlerYAxis() : null,
dataZoom: props.options?.dataZoom || [
{
type: 'inside',
height: 13,
start: 0,
bottom: '20px',
end: 100
},
{
start: 0,
height: 13,
bottom: '20px',
end: 100
}
],
color: props.options?.color || color,
series: props.options?.series,
...props.options?.options
}
// console.log(options.series,"获取x轴");
handlerBar(options)
// 处理柱状图
chart.setOption(options, true)
setTimeout(() => {
chart.resize()
}, 0)
}
const handlerBar = (options: any) => {
if (Array.isArray(options.series)) {
options.series.forEach((item: any) => {
if (item.type === 'bar') {
item.barMinHeight = 10
item.barMaxWidth = 20
item.itemStyle = Object.assign(
{
color: (params: any) => {
if (params.value == 0 || params.value == 3.14159) {
return '#ccc'
} else {
return props.options?.color
? props.options?.color[params.seriesIndex]
: color[params.seriesIndex]
}
}
},
item.itemStyle
)
}
})
}
}
const handlerYAxis = () => {
let temp = {
type: 'value',
nameGap: 15,
nameTextStyle: {
color: '#000'
},
splitNumber: 5,
minInterval: 1,
axisLine: {
show: true,
lineStyle: {
color: '#000'
}
},
axisLabel: {
color: '#000',
fontSize: 14,
formatter: function (value) {
return value.toFixed(0) // 格式化显示为一位小数
}
},
splitLine: {
lineStyle: {
// 使用深浅的间隔色
color: ['#ccc'],
type: 'dashed',
opacity: 0.5
}
}
}
// props.options?.xAxis 是数组还是对象
if (Array.isArray(props.options?.yAxis)) {
return props.options?.yAxis.map((item: any) => {
return {
...temp,
...item
}
})
} else {
return {
...temp,
...props.options?.yAxis
}
}
}
const handlerXAxis = () => {
let temp = {
type: 'category',
axisTick: { show: false },
nameTextStyle: {
color: '#000'
},
axisLine: {
// lineStyle: {
color: '#000'
// }
},
axisLabel: {
// textStyle: {
fontFamily: 'dinproRegular',
color: '#000',
fontSize: '12'
// }
}
// boundaryGap: false,
}
// props.options?.xAxis 是数组还是对象
if (Array.isArray(props.options?.xAxis)) {
return props.options?.xAxis.map((item: any) => {
return {
...temp,
...item
}
})
} else {
return {
...temp,
...props.options?.xAxis
}
}
}
let throttle: ReturnType<typeof setTimeout>
// 动态计算table高度
const resizeObserver = new ResizeObserver(entries => {
for (const entry of entries) {
if (throttle) {
clearTimeout(throttle)
}
throttle = setTimeout(() => {
resizeHandler()
}, 100)
}
})
onMounted(() => {
initChart()
resizeObserver.observe(chartRef.value!)
})
defineExpose({ initChart })
onBeforeUnmount(() => {
resizeObserver.unobserve(chartRef.value!)
chart?.dispose()
})
watch(
() => props.options,
(newVal, oldVal) => {
initChart()
}
)
</script>
<style lang="scss" scoped>
.chart {
width: 100%;
height: 100%;
position: relative;
.el-button {
position: absolute;
right: 0px;
top: -60px;
}
.my-chart {
height: 100%;
width: 100%;
}
}
</style>

View File

@@ -4,6 +4,7 @@
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import * as echarts from "echarts"; import * as echarts from "echarts";
import { ref } from "vue";
const chartsRef = ref(); const chartsRef = ref();
const props = defineProps({ const props = defineProps({
//饼图数据 //饼图数据
@@ -21,11 +22,16 @@ const props = defineProps({
type: Object, type: Object,
default: () => ({}), default: () => ({}),
}, },
}); });
const customData: any = ref({}), const customData: any = ref({}),
legendData: any = ref({}), legendData: any = ref({}),
chart: any = ref(); chart: any = ref();
const labelIsShow = ref(true)//引导线台数和鼠标点击饼图是否弹出提示显示
const init = () => { const init = () => {
customData.value = { customData.value = {
title: "", //标题 title: "", //标题
textAlign: "left", //标题位置可选属性left 可选属性值 left,right,center textAlign: "left", //标题位置可选属性left 可选属性值 left,right,center
@@ -34,13 +40,16 @@ const init = () => {
isRadius: false, //是否圆角 isRadius: false, //是否圆角
isSpace: false, //是否显示间隔 isSpace: false, //是否显示间隔
isLabelLine: true, //是否显示引导线 isLabelLine: true, //是否显示引导线
titleFontSize: '14px', //标题字体大小
...props.customData, ...props.customData,
}; };
legendData.value = { legendData.value = {
icon: "roundRect", // 图例项的icon,类型包括 circle(圆形),rect(正方形),//roundRect(圆角正方形),triangle(三角形),diamond(菱形),//pin(大头针行),arrow(箭头形),none(无图例项的icon) icon: "roundRect", // 图例项的icon,类型包括 circle(圆形),rect(正方形),//roundRect(圆角正方形),triangle(三角形),diamond(菱形),//pin(大头针行),arrow(箭头形),none(无图例项的icon)
orient: "vertical", //图例排列方向 orient: "vertical", //图例排列方向
left: "right", //可选属性left,right,top,bottom,可选属性值 left,right,top,bottom,px,百分比,数值, left: "right", //可选属性left,right,top,bottom,可选属性值 left,right,top,bottom,px,百分比,数值,
itemGap: 1, // 设置图例项之间的间隔为20 itemGap: 10, // 设置图例项之间的间隔为20
...props.legendData, ...props.legendData,
}; };
chart.value = chartsRef.value && echarts.init(chartsRef.value); chart.value = chartsRef.value && echarts.init(chartsRef.value);
@@ -48,10 +57,27 @@ const init = () => {
title: { title: {
text: customData.value.title, text: customData.value.title,
left: customData.value.textAlign, left: customData.value.textAlign,
textStyle: {
fontSize: customData.value.titleFontSize, // 使用 titleFontSize 属性
},
}, },
legend: legendData.value, legend:legendData.value,
// legend: {
// icon: legendData.value.icon, // 图例项的icon,类型包括 circle(圆形),rect(正方形),//roundRect(圆角正方形),triangle(三角形),diamond(菱形),//pin(大头针行),arrow(箭头形),none(无图例项的icon)
// orient: "vertical", //图例排列方向
// left: legendData.value.left, //可选属性left,right,top,bottom,可选属性值 left,right,top,bottom,px,百分比,数值,
// itemGap: 1, // 设置图例项之间的间隔为20
// formatter: function (name) {
// const item = props.chartsData.filter(item=>item.name==name)
// console.log(item)
// if(item)
// return item[0].value;
// },
// },
tooltip: { tooltip: {
show: true, show: labelIsShow.value,
trigger: "item", trigger: "item",
formatter: customData.value.ratio ? `{b} : {c} ({d}%)` : "{b} :{c} ", formatter: customData.value.ratio ? `{b} : {c} ({d}%)` : "{b} :{c} ",
borderWidth: 1, borderWidth: 1,
@@ -61,7 +87,13 @@ const init = () => {
type: "pie", type: "pie",
radius: customData.value.isRing ? ["55", "75"] : "80%", radius: customData.value.isRing ? ["55", "75"] : "80%",
data: props.chartsData, data: props.chartsData,
center: ["55%", "50%"], // 设置饼图的中心位置 formatter: function (name: any) {
const item = props.chartsData.filter(item=>item.name==name)
//console.log(item)
if(item)
return item[0].value;
},
center: ["55%", "55%"], // 设置饼图的中心位置
// padAngle: 2, // padAngle: 2,
minAngle: 15, //最小角度 minAngle: 15, //最小角度
startAngle: 270, //起始角度 startAngle: 270, //起始角度
@@ -74,12 +106,15 @@ const init = () => {
}, },
label: { label: {
normal: { normal: {
show: true, show: labelIsShow.value,
position: "inside", position: "outside",
textStyle: { textStyle: {
color: "#fff", //color: "#fff",
fontSize: 12, fontSize: 12,
}, },
formatter: function (data) {
return labelIsShow.value ? data.value + '台' : '';
}
}, },
}, },
itemStyle: { itemStyle: {
@@ -93,6 +128,7 @@ const init = () => {
}, },
], ],
}; };
option && chart.value && chart.value.setOption(option); option && chart.value && chart.value.setOption(option);
setTimeout(() => { setTimeout(() => {
chart.value.resize(); chart.value.resize();
@@ -108,7 +144,7 @@ const reSize = (widthValue: number,heightValue: number,silentValue: boolean) =>
} }
}; };
const resizeCharts = () => { const resizeCharts = () => {
console.log(chart.value,111111); //console.log(chart.value,111111);
if (chart.value) { if (chart.value) {
chart.value.resize(); chart.value.resize();
@@ -128,6 +164,12 @@ watch(
() => props.chartsData, () => props.chartsData,
(val, oldVal) => { (val, oldVal) => {
if (val) { if (val) {
const item = props.chartsData.find(item => item.value === 0);
if(item != undefined){
labelIsShow.value = false;
}else{
labelIsShow.value = true;
}
init(); init();
} }
}, },

View File

@@ -3,11 +3,13 @@
// 首页地址(默认) // 首页地址(默认)
export const HOME_URL: string = "/home/index"; export const HOME_URL: string = "/home/index";
// export const HOME_URL: string = "/machine/controlSource";
// 登录页地址(默认) // 登录页地址(默认)
export const LOGIN_URL: string = "/login"; export const LOGIN_URL: string = "/login";
// 默认主题颜色 // 默认主题颜色
export const DEFAULT_PRIMARY: string = "#003078"; export const DEFAULT_PRIMARY: string = "#526ADE";
// 路由白名单地址(本地存在的路由 staticRouter.ts 中) // 路由白名单地址(本地存在的路由 staticRouter.ts 中)
export const ROUTER_WHITE_LIST: string[] = ["/500"]; export const ROUTER_WHITE_LIST: string[] = ["/500"];
@@ -17,3 +19,5 @@ export const AMAP_MAP_KEY: string = "";
// 百度地图 key // 百度地图 key
export const BAIDU_MAP_KEY: string = ""; export const BAIDU_MAP_KEY: string = "";

View File

@@ -6,19 +6,24 @@ import { useAuthStore } from '@/stores/modules/auth'
import type { Directive, DirectiveBinding } from 'vue' import type { Directive, DirectiveBinding } from 'vue'
const auth: Directive = { const auth: Directive = {
mounted(el: HTMLElement, binding: DirectiveBinding) { mounted(el: HTMLElement, binding: DirectiveBinding) {
const { value } = binding //console.log('binding',binding)
const authStore = useAuthStore() const { value, modifiers } = binding
const currentPageRoles = authStore.authButtonListGet[authStore.routeName] ?? [] let currentPageRoles = []
// console.log('1234',authStore.routeName) const authStore = useAuthStore()
// console.log('123',currentPageRoles) if (modifiers && Object.keys(modifiers).length) {
if (value instanceof Array && value.length) { currentPageRoles = authStore.authButtonListGet[Object.keys(modifiers)[0]] ?? []
const hasPermission = value.every(item => currentPageRoles.includes(item)) } else {
if (!hasPermission) el.remove() currentPageRoles = authStore.authButtonListGet[authStore.routeName] ?? []
} else { }
if (!currentPageRoles.includes(value)) el.remove() console.log('currentPageRoles', currentPageRoles)
if (value instanceof Array && value.length) {
const hasPermission = value.every(item => currentPageRoles.includes(item))
if (!hasPermission) el.remove()
} else {
if (!currentPageRoles.includes(value)) el.remove()
}
} }
},
} }
export default auth export default auth

View File

@@ -4,7 +4,8 @@
export enum ResultEnum { export enum ResultEnum {
SUCCESS = "A0000", SUCCESS = "A0000",
ERROR = 500, ERROR = 500,
OVERDUE = "401", ACCESSTOKEN_EXPIRED = "A0024",
OVERDUE = "A0025",
TIMEOUT = 30000, TIMEOUT = 30000,
TYPE = "success" TYPE = "success"
} }

View File

@@ -1,4 +1,5 @@
import { ElNotification } from "element-plus"; import { ElNotification } from "element-plus";
import type { AxiosResponse } from "axios";
/** /**
* @description 接收数据流生成 blob创建链接下载文件 * @description 接收数据流生成 blob创建链接下载文件
@@ -8,6 +9,55 @@ import { ElNotification } from "element-plus";
* @param {Boolean} isNotify 是否有导出消息提示 (默认为 true) * @param {Boolean} isNotify 是否有导出消息提示 (默认为 true)
* @param {String} fileType 导出的文件格式 (默认为.xlsx) * @param {String} fileType 导出的文件格式 (默认为.xlsx)
* */ * */
/**
* 从 Content-Disposition 头解析文件名
*/
const getFileNameFromContentDisposition = (contentDisposition: string | undefined | null): string | null => {
if (!contentDisposition) return null;
// 优先匹配 filename*=UTF-8'' 格式RFC 5987
const filenameStarRegex = /filename\*\s*=\s*UTF-8''([^;]+)/i;
const starMatches = filenameStarRegex.exec(contentDisposition);
if (starMatches && starMatches[1]) {
try {
return decodeURIComponent(starMatches[1]);
} catch (e) {
console.warn('解码 filename* 失败:', e);
}
}
// 其次匹配 filename="文件名" 或 filename=文件名 格式
const filenameRegex = /filename\s*=\s*([^;]+)/i;
const matches = filenameRegex.exec(contentDisposition);
if (matches && matches[1]) {
let filename = matches[1].trim();
// 去除引号
filename = filename.replace(/^["']|["']$/g, "");
// 尝试解码 URL 编码的文件名
try {
// 检查是否包含 URL 编码字符
if (filename.includes('%') || filename.includes('UTF-8')) {
// 处理可能的 UTF-8 前缀
if (filename.startsWith("UTF-8''") || filename.startsWith("utf-8''")) {
filename = filename.substring(7);
}
filename = decodeURIComponent(filename);
}
} catch (e) {
// 如果解码失败,返回原始文件名
console.warn('解码文件名失败,使用原始值:', e);
}
return filename;
}
return null;
};
export const useDownload = async ( export const useDownload = async (
api: (param: any) => Promise<any>, api: (param: any) => Promise<any>,
tempName: string, tempName: string,
@@ -39,6 +89,73 @@ export const useDownload = async (
document.body.removeChild(exportFile); document.body.removeChild(exportFile);
window.URL.revokeObjectURL(blobUrl); window.URL.revokeObjectURL(blobUrl);
} catch (error) { } catch (error) {
}
};
/**
* @description 支持服务器文件名的下载方法,会从 Content-Disposition 头获取文件名
* @param {Function} api 导出表格的api方法 (必传)
* @param {String} fallbackName 备用文件名,当服务器未提供时使用 (可选)
* @param {Object} params 导出的参数 (默认{})
* @param {Boolean} isNotify 是否有导出消息提示 (默认为 true)
* @param {String} fallbackFileType 备用文件格式,当服务器未提供时使用 (默认为.xlsx)
*/
export const useDownloadWithServerFileName = async (
api: (param: any) => Promise<AxiosResponse<Blob> | any>,
fallbackName: string = "",
params: any = {},
isNotify: boolean = true,
fallbackFileType: string = ".xlsx"
) => {
if (isNotify) {
ElNotification({
title: "温馨提示",
message: "如果数据庞大会导致下载缓慢哦,请您耐心等待!",
type: "info",
duration: 3000
});
}
try {
const res = await api(params);
// 检查响应是否包含 data 属性AxiosResponse还是直接是 Blob
const blob = res.data ? new Blob([res.data]) : new Blob([res]);
// 尝试从响应头获取文件名(如果存在)
let serverFileName: string | null = null;
if (res.headers) {
const headers = res.headers || {};
const contentDisposition = headers['content-disposition'] || headers['Content-Disposition'];
serverFileName = getFileNameFromContentDisposition(contentDisposition);
}
// 确定最终使用的文件名
let finalFileName: string;
if (serverFileName) {
finalFileName = serverFileName;
} else if (fallbackName) {
finalFileName = `${fallbackName}${fallbackFileType}`;
} else {
finalFileName = `download${fallbackFileType}`;
}
// 兼容 edge 不支持 createObjectURL 方法
if ("msSaveOrOpenBlob" in navigator) {
return window.navigator.msSaveOrOpenBlob(blob, finalFileName);
}
const blobUrl = window.URL.createObjectURL(blob);
const exportFile = document.createElement("a");
exportFile.style.display = "none";
exportFile.download = finalFileName;
exportFile.href = blobUrl;
document.body.appendChild(exportFile);
exportFile.click();
// 去除下载对 url 的影响
document.body.removeChild(exportFile);
window.URL.revokeObjectURL(blobUrl);
} catch (error) {
console.error('文件下载失败:', error);
} }
}; };

View File

@@ -78,6 +78,7 @@ export const useTable = (
}); });
dataCallBack && (data = dataCallBack(data)); dataCallBack && (data = dataCallBack(data));
state.tableData = isPageable ? data.records : data; state.tableData = isPageable ? data.records : data;
//console.log(data);
// 解构后台返回的分页数据 (如果有分页更新分页信息) // 解构后台返回的分页数据 (如果有分页更新分页信息)
if (isPageable) { if (isPageable) {
state.resPageable.total = data.total; state.resPageable.total = data.total;
@@ -85,6 +86,7 @@ export const useTable = (
state.resPageable.size = data.size; state.resPageable.size = data.size;
} }
} catch (error) { } catch (error) {
//console.log('123');
requestError && requestError(error); requestError && requestError(error);
} }
}; };

View File

@@ -27,6 +27,7 @@ export const useTheme = () => {
// 修改主题颜色 // 修改主题颜色
const changePrimary = (val: string | null) => { const changePrimary = (val: string | null) => {
if (!val) { if (!val) {
val = DEFAULT_PRIMARY; val = DEFAULT_PRIMARY;
ElMessage({ type: "success", message: `主题颜色已重置为 ${DEFAULT_PRIMARY}` }); ElMessage({ type: "success", message: `主题颜色已重置为 ${DEFAULT_PRIMARY}` });
@@ -40,10 +41,13 @@ export const useTheme = () => {
for (let i = 1; i <= 9; i++) { for (let i = 1; i <= 9; i++) {
const primaryColor = isDark.value ? `${getDarkColor(val, i / 10)}` : `${getLightColor(val, i / 10)}`; const primaryColor = isDark.value ? `${getDarkColor(val, i / 10)}` : `${getLightColor(val, i / 10)}`;
document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, primaryColor); document.documentElement.style.setProperty(`--el-color-primary-light-${i}`, primaryColor);
// const colorValue = document.documentElement.style.getPropertyValue(`--el-color-primary-light-${i}`).trim();
// console.log(`--el-color-primary-light-${i}: ${colorValue}`);
} }
globalStore.setGlobalState("primary", val); globalStore.setGlobalState("primary", val);
};
}
// 灰色和弱色切换 // 灰色和弱色切换
const changeGreyOrWeak = (type: Theme.GreyOrWeakType, value: boolean) => { const changeGreyOrWeak = (type: Theme.GreyOrWeakType, value: boolean) => {
const body = document.body as HTMLElement; const body = document.body as HTMLElement;

View File

@@ -26,6 +26,8 @@ export default {
changePassword: "Change Password", changePassword: "Change Password",
changeMode:"Change Mode", changeMode:"Change Mode",
versionRegister:"Version Register", versionRegister:"Version Register",
changeTheme: "Change Theme",
changeScene: "Change Scene",
logout: "Logout" logout: "Logout"
} }
}; };

View File

@@ -26,6 +26,8 @@ export default {
changePassword: "修改密码", changePassword: "修改密码",
changeMode:"模式切换", changeMode:"模式切换",
versionRegister:"版本注册", versionRegister:"版本注册",
changeTheme:"主题切换",
changeScene: "场景切换",
logout: "退出登录" logout: "退出登录"
} }
}; };

View File

@@ -1,63 +1,62 @@
<!-- 经典布局 --> <!-- 经典布局 -->
<template> <template>
<el-container class="layout"> <el-container class="layout">
<el-header>
<el-header> <div class="header-lf mask-image">
<div class="header-lf mask-image"> <div class="logo flx-center">
<div class="logo flx-center"> <img class="logo-img" src="@/assets/images/logo.svg" alt="logo" />
<img class="logo-img" src="@/assets/images/logo.svg" alt="logo" /> <span class="logo-text">{{ title }}</span>
<span class="logo-text">{{ title }}</span> </div>
</div> <ToolBarLeft />
<ToolBarLeft /> </div>
</div> <div class="header-ri">
<div class="header-ri"> <ToolBarRight />
<ToolBarRight /> </div>
</div> </el-header>
</el-header> <el-container class="classic-content">
<el-container class="classic-content"> <el-aside>
<el-aside> <div class="aside-box" :style="{ width: isCollapse ? '65px' : '210px' }">
<div class="aside-box" :style="{ width: isCollapse ? '65px' : '210px' }"> <el-scrollbar>
<el-scrollbar> <el-menu
<el-menu :router="false"
:router="false" :default-active="activeMenu"
:default-active="activeMenu" :collapse="isCollapse"
:collapse="isCollapse" :unique-opened="accordion"
:unique-opened="accordion" :collapse-transition="false"
:collapse-transition="false" >
> <SubMenu :menu-list="menuList" />
<SubMenu :menu-list="menuList" /> </el-menu>
</el-menu> </el-scrollbar>
</el-scrollbar> </div>
</div> </el-aside>
</el-aside> <el-container class="classic-main">
<el-container class="classic-main"> <Main />
<Main /> </el-container>
</el-container> </el-container>
</el-container> </el-container>
</el-container>
</template> </template>
<script setup lang="ts" name="layoutClassic"> <script setup lang="ts" name="layoutClassic">
import { computed } from "vue"; import { computed } from 'vue'
import { useRoute } from "vue-router"; import { useRoute } from 'vue-router'
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from '@/stores/modules/auth'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import Main from "@/layouts/components/Main/index.vue"; import Main from '@/layouts/components/Main/index.vue'
import SubMenu from "@/layouts/components/Menu/SubMenu.vue"; import SubMenu from '@/layouts/components/Menu/SubMenu.vue'
import ToolBarLeft from "@/layouts/components/Header/ToolBarLeft.vue"; import ToolBarLeft from '@/layouts/components/Header/ToolBarLeft.vue'
import ToolBarRight from "@/layouts/components/Header/ToolBarRight.vue"; import ToolBarRight from '@/layouts/components/Header/ToolBarRight.vue'
const title = import.meta.env.VITE_GLOB_APP_TITLE; const title = import.meta.env.VITE_GLOB_APP_TITLE
const route = useRoute(); const route = useRoute()
const authStore = useAuthStore(); const authStore = useAuthStore()
const globalStore = useGlobalStore(); const globalStore = useGlobalStore()
const accordion = computed(() => globalStore.accordion); const accordion = computed(() => globalStore.accordion)
const isCollapse = computed(() => globalStore.isCollapse); const isCollapse = computed(() => globalStore.isCollapse)
const menuList = computed(() => authStore.showMenuListGet); const menuList = computed(() => authStore.showMenuListGet)
const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string); const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string)
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,103 +1,105 @@
<!-- 分栏布局 --> <!-- 分栏布局 -->
<template> <template>
<el-container class="layout"> <el-container class="layout">
<div class="aside-split"> <div class="aside-split">
<div class="logo flx-center"> <div class="logo flx-center">
<img class="logo-img" src="@/assets/images/logo.svg" alt="logo" /> <img class="logo-img" src="@/assets/images/logo.svg" alt="logo" />
</div> </div>
<el-scrollbar> <el-scrollbar>
<div class="split-list"> <div class="split-list">
<div <div
v-for="item in menuList" v-for="item in menuList"
:key="item.path" :key="item.path"
class="split-item" class="split-item"
:class="{ 'split-active': splitActive === item.path || `/${splitActive.split('/')[1]}` === item.path }" :class="{
@click="changeSubMenu(item)" 'split-active': splitActive === item.path || `/${splitActive.split('/')[1]}` === item.path
> }"
<el-icon> @click="changeSubMenu(item)"
<component :is="item.meta.icon"></component> >
</el-icon> <el-icon>
<span class="title">{{ item.meta.title }}</span> <component :is="item.meta.icon"></component>
</div> </el-icon>
<span class="title">{{ item.meta.title }}</span>
</div>
</div>
</el-scrollbar>
</div> </div>
</el-scrollbar> <el-aside :class="{ 'not-aside': !subMenuList.length }" :style="{ width: isCollapse ? '65px' : '210px' }">
</div> <div class="logo flx-center">
<el-aside :class="{ 'not-aside': !subMenuList.length }" :style="{ width: isCollapse ? '65px' : '210px' }"> <span v-show="subMenuList.length" class="logo-text">{{ isCollapse ? 'G' : title }}</span>
<div class="logo flx-center"> </div>
<span v-show="subMenuList.length" class="logo-text">{{ isCollapse ? "G" : title }}</span> <el-scrollbar>
</div> <el-menu
<el-scrollbar> :router="false"
<el-menu :default-active="activeMenu"
:router="false" :collapse="isCollapse"
:default-active="activeMenu" :unique-opened="accordion"
:collapse="isCollapse" :collapse-transition="false"
:unique-opened="accordion" >
:collapse-transition="false" <SubMenu :menu-list="subMenuList" />
> </el-menu>
<SubMenu :menu-list="subMenuList" /> </el-scrollbar>
</el-menu> </el-aside>
</el-scrollbar> <el-container>
</el-aside> <el-header>
<el-container> <ToolBarLeft />
<el-header> <ToolBarRight />
<ToolBarLeft /> </el-header>
<ToolBarRight /> <Main />
</el-header> </el-container>
<Main />
</el-container> </el-container>
</el-container>
</template> </template>
<script setup lang="ts" name="layoutColumns"> <script setup lang="ts" name="layoutColumns">
import { ref, computed, watch } from "vue"; import { computed, ref, watch } from 'vue'
import { useRoute, useRouter } from "vue-router"; import { useRoute, useRouter } from 'vue-router'
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from '@/stores/modules/auth'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import Main from "@/layouts/components/Main/index.vue"; import Main from '@/layouts/components/Main/index.vue'
import ToolBarLeft from "@/layouts/components/Header/ToolBarLeft.vue"; import ToolBarLeft from '@/layouts/components/Header/ToolBarLeft.vue'
import ToolBarRight from "@/layouts/components/Header/ToolBarRight.vue"; import ToolBarRight from '@/layouts/components/Header/ToolBarRight.vue'
import SubMenu from "@/layouts/components/Menu/SubMenu.vue"; import SubMenu from '@/layouts/components/Menu/SubMenu.vue'
const title = import.meta.env.VITE_GLOB_APP_TITLE; const title = import.meta.env.VITE_GLOB_APP_TITLE
const route = useRoute(); const route = useRoute()
const router = useRouter(); const router = useRouter()
const authStore = useAuthStore(); const authStore = useAuthStore()
const globalStore = useGlobalStore(); const globalStore = useGlobalStore()
const accordion = computed(() => globalStore.accordion); const accordion = computed(() => globalStore.accordion)
const isCollapse = computed(() => globalStore.isCollapse); const isCollapse = computed(() => globalStore.isCollapse)
const menuList = computed(() => authStore.showMenuListGet); const menuList = computed(() => authStore.showMenuListGet)
const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string); const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string)
const subMenuList = ref<Menu.MenuOptions[]>([]); const subMenuList = ref<Menu.MenuOptions[]>([])
const splitActive = ref(""); const splitActive = ref('')
watch( watch(
() => [menuList, route], () => [menuList, route],
() => { () => {
// 当前菜单没有数据直接 return // 当前菜单没有数据直接 return
if (!menuList.value.length) return; if (!menuList.value.length) return
splitActive.value = route.path; splitActive.value = route.path
const menuItem = menuList.value.filter((item: Menu.MenuOptions) => { const menuItem = menuList.value.filter((item: Menu.MenuOptions) => {
return route.path === item.path || `/${route.path.split("/")[1]}` === item.path; return route.path === item.path || `/${route.path.split('/')[1]}` === item.path
}); })
if (menuItem[0].children?.length) return (subMenuList.value = menuItem[0].children); if (menuItem[0].children?.length) return (subMenuList.value = menuItem[0].children)
subMenuList.value = []; subMenuList.value = []
}, },
{ {
deep: true, deep: true,
immediate: true immediate: true
} }
); )
// change SubMenu // change SubMenu
const changeSubMenu = (item: Menu.MenuOptions) => { const changeSubMenu = (item: Menu.MenuOptions) => {
splitActive.value = item.path; splitActive.value = item.path
if (item.children?.length) return (subMenuList.value = item.children); if (item.children?.length) return (subMenuList.value = item.children)
subMenuList.value = []; subMenuList.value = []
router.push(item.path); router.push(item.path)
}; }
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -39,7 +39,9 @@
} }
.el-menu-item:hover { .el-menu-item:hover {
color: #fff; //一级导航划过颜色 color: #fff; //一级导航划过颜色
background-color: #5274a5 !important; //一级导航划过背景色 //background-color: #5274a5 !important; //一级导航划过背景色
background-color:var(--el-color-primary-light-3) !important;
} }
.el-sub-menu__hide-arrow { .el-sub-menu__hide-arrow {
width: 65px; width: 65px;
@@ -47,7 +49,9 @@
} }
.el-menu-item.is-active { .el-menu-item.is-active {
color: #fff !important; //一级导航文字选中颜色 color: #fff !important; //一级导航文字选中颜色
background-color: #5274a5 !important; //一级导航选中背景色 //background-color: #5274a5 !important; //一级导航选中背景色
background-color: var(--el-color-primary-light-3) !important;
border-bottom: 0 !important; border-bottom: 0 !important;
} }
.el-sub-menu__title { .el-sub-menu__title {
@@ -67,7 +71,9 @@
color: #fff !important; //二级导航文字选中颜色 color: #fff !important; //二级导航文字选中颜色
// background-color: var(--el-color-primary) !important; // background-color: var(--el-color-primary) !important;
// background-color: #5274a5 !important;//二级导航选中背景色 // background-color: #5274a5 !important;//二级导航选中背景色
background-color: #5274a5 !important;
//background-color: #7588e5 !important;
background-color: var(--el-color-primary-light-3) !important;
border-bottom-color: var(--el-color-primary) !important; border-bottom-color: var(--el-color-primary) !important;
} }
} }

View File

@@ -1,85 +1,78 @@
<!-- 横向布局 --> <!-- 横向布局 -->
<template> <template>
<el-container class="layout"> <el-container class="layout">
<el-header> <el-header>
<div class="logo flx-center"> <div class="logo flx-center">
<!-- <img class="logo-img" src="@/assets/images/logo.svg" alt="logo" /> --> <!-- <img class="logo-img" src="@/assets/images/logo.svg" alt="logo" /> -->
<img <img class="logo-img" src="@/assets/images/cn_pms9100_logo.png" alt="logo" />
class="logo-img" <span class="logo-text">{{ title }}</span>
src="@/assets/images/cn_pms9100_logo.png" </div>
alt="logo" <el-menu v-if="showMenuFlag" trigger="click" mode="horizontal" :router="false" :default-active="activeMenu">
/> <!-- 不能直接使用 SubMenu 组件无法触发 el-menu 隐藏省略功能 -->
<span class="logo-text">{{ title }} </span> <template v-for="subItem in menuList" :key="subItem.path">
</div> <el-sub-menu
<el-menu v-if="showMenuFlag" trigger="click" mode="horizontal" :router="false" :default-active="activeMenu"> v-if="subItem.children?.length"
<!-- 不能直接使用 SubMenu 组件无法触发 el-menu 隐藏省略功能 --> :key="subItem.path"
<template v-for="subItem in menuList" :key="subItem.path"> :index="subItem.path + 'el-sub-menu'"
<el-sub-menu >
v-if="subItem.children?.length" <template #title>
:key="subItem.path" <el-icon>
:index="subItem.path + 'el-sub-menu'" <component :is="subItem.meta.icon"></component>
> </el-icon>
<template #title> <span>{{ subItem.meta.title }}</span>
<el-icon> </template>
<component :is="subItem.meta.icon"></component> <SubMenu :menu-list="subItem.children" />
</el-icon> </el-sub-menu>
<span>{{ subItem.meta.title }}</span> <el-menu-item
</template> v-else
<SubMenu :menu-list="subItem.children" /> :key="subItem.path + 'el-menu-item'"
</el-sub-menu> :index="subItem.path"
<el-menu-item @click="handleClickMenu(subItem)"
v-else >
:key="subItem.path + 'el-menu-item'" <el-icon>
:index="subItem.path" <component :is="subItem.meta.icon"></component>
@click="handleClickMenu(subItem)" </el-icon>
> <template #title>
<el-icon> <span>{{ subItem.meta.title }}</span>
<component :is="subItem.meta.icon"></component> </template>
</el-icon> </el-menu-item>
<template #title> </template>
<span>{{ subItem.meta.title }}</span> </el-menu>
</template> <ToolBarRight />
</el-menu-item> </el-header>
</template> <Main />
</el-menu> </el-container>
<ToolBarRight />
</el-header>
<Main />
</el-container>
</template> </template>
<script setup lang="ts" name="layoutTransverse"> <script setup lang="ts" name="layoutTransverse">
import { computed } from "vue"; import { computed } from 'vue'
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from '@/stores/modules/auth'
import { useRoute, useRouter } from "vue-router"; import { useRoute, useRouter } from 'vue-router'
import Main from "@/layouts/components/Main/index.vue"; import Main from '@/layouts/components/Main/index.vue'
import ToolBarRight from "@/layouts/components/Header/ToolBarRight.vue"; import ToolBarRight from '@/layouts/components/Header/ToolBarRight.vue'
import SubMenu from "@/layouts/components/Menu/SubMenu.vue"; import SubMenu from '@/layouts/components/Menu/SubMenu.vue'
const title = import.meta.env.VITE_GLOB_APP_TITLE; const title = import.meta.env.VITE_GLOB_APP_TITLE
const route = useRoute(); const route = useRoute()
const router = useRouter(); const router = useRouter()
const authStore = useAuthStore(); const authStore = useAuthStore()
const menuList = computed(() => authStore.showMenuListGet); const menuList = computed(() => authStore.showMenuListGet)
const showMenuFlag=computed(()=>authStore.showMenuFlagGet) const showMenuFlag = computed(() => authStore.showMenuFlagGet)
const activeMenu = computed( const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string)
() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string
);
const handleClickMenu = (subItem: Menu.MenuOptions) => { const handleClickMenu = (subItem: Menu.MenuOptions) => {
if (subItem.meta.isLink) return window.open(subItem.meta.isLink, "_blank"); if (subItem.meta.isLink) return window.open(subItem.meta.isLink, '_blank')
router.push(subItem.path); router.push(subItem.path)
}; }
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
.logo{ .logo {
margin-right: 0 !important; margin-right: 0 !important;
} }
.logo-text { .logo-text {
letter-spacing: 2px; letter-spacing: 2px;
} }
</style> </style>

View File

@@ -1,56 +1,56 @@
<!-- 纵向布局 --> <!-- 纵向布局 -->
<template> <template>
<el-container class="layout"> <el-container class="layout">
<el-aside> <el-aside>
<div class="aside-box" :style="{ width: isCollapse ? '65px' : '210px' }"> <div class="aside-box" :style="{ width: isCollapse ? '65px' : '210px' }">
<div class="logo flx-center"> <div class="logo flx-center">
<img class="logo-img" src="@/assets/images/logo.svg" alt="logo" /> <img class="logo-img" src="@/assets/images/logo.svg" alt="logo" />
<span v-show="!isCollapse" class="logo-text">{{ title }}</span> <span v-show="!isCollapse" class="logo-text">{{ title }}</span>
</div> </div>
<el-scrollbar> <el-scrollbar>
<el-menu <el-menu
:router="false" :router="false"
:default-active="activeMenu" :default-active="activeMenu"
:collapse="isCollapse" :collapse="isCollapse"
:unique-opened="accordion" :unique-opened="accordion"
:collapse-transition="false" :collapse-transition="false"
> >
<SubMenu :menu-list="menuList" /> <SubMenu :menu-list="menuList" />
</el-menu> </el-menu>
</el-scrollbar> </el-scrollbar>
</div> </div>
</el-aside> </el-aside>
<el-container> <el-container>
<el-header> <el-header>
<ToolBarLeft /> <ToolBarLeft />
<ToolBarRight /> <ToolBarRight />
</el-header> </el-header>
<Main /> <Main />
</el-container>
</el-container> </el-container>
</el-container>
</template> </template>
<script setup lang="ts" name="layoutVertical"> <script setup lang="ts" name="layoutVertical">
import { computed } from "vue"; import { computed } from 'vue'
import { useRoute } from "vue-router"; import { useRoute } from 'vue-router'
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from '@/stores/modules/auth'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import Main from "@/layouts/components/Main/index.vue"; import Main from '@/layouts/components/Main/index.vue'
import ToolBarLeft from "@/layouts/components/Header/ToolBarLeft.vue"; import ToolBarLeft from '@/layouts/components/Header/ToolBarLeft.vue'
import ToolBarRight from "@/layouts/components/Header/ToolBarRight.vue"; import ToolBarRight from '@/layouts/components/Header/ToolBarRight.vue'
import SubMenu from "@/layouts/components/Menu/SubMenu.vue"; import SubMenu from '@/layouts/components/Menu/SubMenu.vue'
const title = import.meta.env.VITE_GLOB_APP_TITLE; const title = import.meta.env.VITE_GLOB_APP_TITLE
const route = useRoute(); const route = useRoute()
const authStore = useAuthStore(); const authStore = useAuthStore()
const globalStore = useGlobalStore(); const globalStore = useGlobalStore()
const accordion = computed(() => globalStore.accordion); const accordion = computed(() => globalStore.accordion)
const isCollapse = computed(() => globalStore.isCollapse); const isCollapse = computed(() => globalStore.isCollapse)
const menuList = computed(() => authStore.showMenuListGet); const menuList = computed(() => authStore.showMenuListGet)
const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string); const activeMenu = computed(() => (route.meta.activeMenu ? route.meta.activeMenu : route.path) as string)
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -1,110 +1,104 @@
<template> <template>
<div class="footer flx-align-center pl10"> <div class="footer flx-align-center pl10">
<el-dropdown> <el-dropdown>
<!-- <span class="el-dropdown-link"> <!-- <span class="el-dropdown-link">
{{ title }} {{ title }}
<el-icon class="el-icon--right"> <el-icon class="el-icon--right">
<arrow-down /> <arrow-down />
</el-icon> </el-icon>
</span> --> </span> -->
<!-- <el-button dictType="primary"> --> <!-- <el-button dictType="primary"> -->
<div class="change_mode"> <div class="change_mode">
{{ title }} {{ title }}
<el-icon class="el-icon--right change_mode_down" <el-icon class="el-icon--right change_mode_down"><arrow-down /></el-icon>
><arrow-down <el-icon class="el-icon--right change_mode_up"><arrow-up /></el-icon>
/></el-icon> </div>
<el-icon class="el-icon--right change_mode_up"><arrow-up /></el-icon> <!-- </el-button> -->
</div> <template #dropdown>
<!-- </el-button> --> <el-dropdown-menu>
<template #dropdown> <el-dropdown-item @click="handelOpen('模拟式')">模拟式模块</el-dropdown-item>
<el-dropdown-menu> <el-dropdown-item @click="handelOpen('数字式')">数字式模块</el-dropdown-item>
<el-dropdown-item @click="title = '模拟式模块'" <el-dropdown-item @click="handelOpen('比对式')">比对式模块</el-dropdown-item>
>模拟式模块</el-dropdown-item </el-dropdown-menu>
> </template>
<el-dropdown-item @click="title = '数字式模块'" </el-dropdown>
>数字式模块</el-dropdown-item <p style="margin: 0">
> <a href="http://www.shining-electric.com/" target="_blank">2024 © 南京灿能电力自动化股份有限公司</a>
<el-dropdown-item @click="title = '比对式模块'" </p>
>比对式模块</el-dropdown-item </div>
>
</el-dropdown-menu>
</template>
</el-dropdown>
<p style="margin: 0;">
<a href="http://www.shining-electric.com/" target="_blank">
2024 © 南京灿能电力自动化股份有限公司
</a>
</p>
</div>
</template> </template>
<script lang="ts" setup> <script lang="ts" setup>
import { ref, reactive, computed, onMounted, watch } from "vue"; import { computed } from 'vue'
const title = ref("模拟式模块"); import { useAuthStore } from '@/stores/modules/auth'
import { useModeStore } from '@/stores/modules/mode'; // 引入模式 store import { useModeStore } from '@/stores/modules/mode' // 引入模式 store
import { useRouter } from 'vue-router'
const modeStore = useModeStore(); const router = useRouter()
watch(() => modeStore.currentMode, (newVal) => { const authStore = useAuthStore()
//console.log('当前模式:', newVal); const modeStore = useModeStore()
switch (newVal) {
case '模拟式': const title = computed(() => {
title.value = '模拟式模块'; return modeStore.currentMode === '' ? '模拟式模块' : modeStore.currentMode + '模块'
break; })
case '数字式':
title.value = '数字式模块'; const handelOpen = async (item: string) => {
break; await authStore.setShowMenu()
case '比对式': modeStore.setCurrentMode(item) // 将模式code存入 store
title.value = '比对式模块';
break; //if (router.currentRoute.value.path === '/home/index') {
default: // 强制刷新页面
title.value = '模拟式模块'; window.location.reload()
} //} else {
}); // router.push({ path: '/home/index' });
//}
}
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
.footer { .footer {
position: relative; position: relative;
background-color: var(--el-color-primary); background-color: var(--el-color-primary);
// .el-button:hover { // .el-button:hover {
// background-color: var(--el-color-primary) !important; // background-color: var(--el-color-primary) !important;
// border: none !important; // border: none !important;
// outline: none !important; // outline: none !important;
// } // }
.change_mode { .change_mode {
color: #fff; color: #fff;
display: flex; display: flex;
align-items: center; align-items: center;
justify-content: flex-start; justify-content: flex-start;
height: 100%; height: 100%;
width: auto; width: auto;
font-size: 14px; font-size: 14px;
.change_mode_down { .change_mode_down {
display: block; display: block;
}
.change_mode_up {
display: none;
}
} }
.change_mode_up { .change_mode:hover {
display: none; .change_mode_down {
display: none;
}
.change_mode_up {
display: block;
}
} }
} .el-dropdown {
.change_mode:hover { z-index: 1001;
.change_mode_down {
display: none;
} }
.change_mode_up { p {
display: block; position: absolute;
width: 100%;
height: 100%;
text-align: right;
line-height: 40px;
a {
color: #fff;
margin-right: 25px; // 增加右边距
}
} }
}
.el-dropdown {
z-index: 1001;
}
p {
position: absolute;
width: 100%;
height: 100%;
text-align: center;
line-height: 40px;
a {
color: #fff;
}
}
} }
</style> </style>

View File

@@ -1,6 +1,14 @@
<template> <template>
<!-- <div class="userInfo">-->
<!-- <div class="icon">-->
<!-- <Avatar/>-->
<!-- </div>-->
<!-- <div class="username">-->
<!-- {{ username }}-->
<!-- </div>-->
<!-- </div>-->
<el-dropdown trigger="click"> <el-dropdown trigger="click">
<div class="userInfo"> <div class="userInfo">
<div class="icon"> <div class="icon">
<Avatar /> <Avatar />
</div> </div>
@@ -10,18 +18,43 @@
</div> </div>
<template #dropdown> <template #dropdown>
<el-dropdown-menu> <el-dropdown-menu>
<el-dropdown-item @click="openDialog('themeRef')">
<el-icon><Sunny /></el-icon>{{ t("header.changeTheme") }}
</el-dropdown-item>
<el-dropdown-item @click="openDialog('infoRef')"> <el-dropdown-item @click="openDialog('infoRef')">
<el-icon><User /></el-icon>{{ $t("header.personalData") }} <el-icon><User /></el-icon>{{ t("header.personalData") }}
</el-dropdown-item> </el-dropdown-item>
<el-dropdown-item @click="openDialog('passwordRef')"> <el-dropdown-item @click="openDialog('passwordRef')">
<el-icon><Edit /></el-icon>{{ $t("header.changePassword") }} <el-icon><Edit /></el-icon>{{ t("header.changePassword") }}
</el-dropdown-item> </el-dropdown-item>
<el-dropdown-item @click="changeMode"> <el-dropdown-item @click="changeMode">
<el-icon><Edit /></el-icon>{{ $t("header.changeMode") }} <el-icon><Switch /></el-icon>{{ t("header.changeMode") }}
</el-dropdown-item> </el-dropdown-item>
<el-dropdown-item @click="openDialog('versionRegisterRef')"> <el-dropdown-item @click="openDialog('versionRegisterRef')">
<el-icon><SetUp /></el-icon>{{ $t("header.versionRegister") }} <el-icon><SetUp /></el-icon>{{ t("header.versionRegister") }}
</el-dropdown-item> </el-dropdown-item>
<el-dropdown trigger="hover" placement="left-start" v-if="userStore.userInfo.loginName == 'root'">
<div class="custom-dropdown-trigger">
<el-icon><Tools /></el-icon>
<span>{{ t("header.changeScene") }}</span>
</div>
<template #dropdown>
<el-dropdown-menu>
<el-dropdown-item
v-for="item in dictStore.getDictData('app_scene')"
:key="item.value"
:class="{
'custom-dropdown-item': true,
active: item.value === appSceneStore.currentScene
}"
@click="changeScene(item.value?? '')"
:disabled = "item.value === appSceneStore.currentScene"
>
{{ item.name }}
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</el-dropdown-menu> </el-dropdown-menu>
</template> </template>
</el-dropdown> </el-dropdown>
@@ -34,6 +67,9 @@
<PasswordDialog ref="passwordRef"></PasswordDialog> <PasswordDialog ref="passwordRef"></PasswordDialog>
<!-- versionRegisterDialog --> <!-- versionRegisterDialog -->
<VersionDialog ref="versionRegisterRef"></VersionDialog> <VersionDialog ref="versionRegisterRef"></VersionDialog>
<!-- ThemeDialog -->
<ThemeDialog ref="themeRef"></ThemeDialog>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
@@ -42,12 +78,13 @@ import { LOGIN_URL } from "@/config";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
import { logoutApi } from "@/api/user/login"; import { logoutApi } from "@/api/user/login";
import { useUserStore } from "@/stores/modules/user"; import { useUserStore } from "@/stores/modules/user";
import { ElMessageBox, ElMessage } from "element-plus"; import { ElMessageBox, ElMessage, CHANGE_EVENT } from "element-plus";
import InfoDialog from "./InfoDialog.vue"; import InfoDialog from "./InfoDialog.vue";
import PasswordDialog from "./PasswordDialog.vue"; import PasswordDialog from "./PasswordDialog.vue";
import ThemeDialog from "./ThemeDialog.vue";
import VersionDialog from "@/views/system/versionRegister/index.vue"; import VersionDialog from "@/views/system/versionRegister/index.vue";
import { computed } from "vue"; import { computed } from "vue";
import { Avatar } from "@element-plus/icons-vue"; import { ArrowLeft, Avatar, Delete, Document, Sunny, Switch ,Tools} from "@element-plus/icons-vue";
import AssemblySize from "./components/AssemblySize.vue"; import AssemblySize from "./components/AssemblySize.vue";
import Language from "./components/Language.vue"; import Language from "./components/Language.vue";
import SearchMenu from "./components/SearchMenu.vue"; import SearchMenu from "./components/SearchMenu.vue";
@@ -56,13 +93,24 @@ import Message from "./components/Message.vue";
import Fullscreen from "./components/Fullscreen.vue"; import Fullscreen from "./components/Fullscreen.vue";
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from "@/stores/modules/auth";
import {useDictStore} from "@/stores/modules/dict"; import {useDictStore} from "@/stores/modules/dict";
import { useModeStore } from "@/stores/modules/mode"; import { useModeStore,useAppSceneStore } from "@/stores/modules/mode";
const userStore = useUserStore(); const userStore = useUserStore();
const dictStore = useDictStore(); const dictStore = useDictStore();
const username = computed(() => userStore.userInfo.name); const username = computed(() => userStore.userInfo.name);
const router = useRouter(); const router = useRouter();
const authStore = useAuthStore(); const authStore = useAuthStore();
const modeStore = useModeStore(); const modeStore = useModeStore();
const AppSceneStore = useAppSceneStore();
import { useTheme } from "@/hooks/useTheme";
import { useI18n } from "vue-i18n";
import {updateScene} from '@/api/system/base/index'
const { changePrimary} = useTheme();
// 初始化 i18n
const { t } = useI18n(); // 使用 t 方法替代 $t
// 退出登录 // 退出登录
const logout = () => { const logout = () => {
@@ -74,10 +122,14 @@ const logout = () => {
// 1.执行退出登录接口 // 1.执行退出登录接口
await logoutApi(); await logoutApi();
// 2.清除 Token // 2.清除 Token
userStore.setToken(""); userStore.setAccessToken("");
userStore.setUserInfo({name: ""}); userStore.setRefreshToken("");
userStore.setExp(0)
userStore.setUserInfo({id: "", name: ""});
userStore.setIsRefreshToken(false)
dictStore.setDictData([]); dictStore.setDictData([]);
modeStore.setCurrentMode(''); modeStore.setCurrentMode('');
AppSceneStore.setCurrentMode('');
// 3.重定向到登陆页 // 3.重定向到登陆页
router.replace(LOGIN_URL); router.replace(LOGIN_URL);
ElMessage.success("退出登录成功!"); ElMessage.success("退出登录成功!");
@@ -90,12 +142,26 @@ const logout = () => {
const infoRef = ref<InstanceType<typeof InfoDialog> | null>(null); const infoRef = ref<InstanceType<typeof InfoDialog> | null>(null);
const passwordRef = ref<InstanceType<typeof PasswordDialog> | null>(null); const passwordRef = ref<InstanceType<typeof PasswordDialog> | null>(null);
const versionRegisterRef = ref<InstanceType<typeof VersionDialog> | null>(null); const versionRegisterRef = ref<InstanceType<typeof VersionDialog> | null>(null);
const themeRef = ref<InstanceType<typeof ThemeDialog> | null>(null);
const openDialog = (ref: string) => { const openDialog = (ref: string) => {
if (ref == "infoRef") infoRef.value?.openDialog(); if (ref == "infoRef") infoRef.value?.openDialog();
if (ref == "passwordRef") passwordRef.value?.openDialog(); if (ref == "passwordRef") passwordRef.value?.openDialog();
if (ref == "versionRegisterRef") versionRegisterRef.value?.openDialog(); if (ref == "versionRegisterRef") versionRegisterRef.value?.openDialog();
if (ref == "themeRef") themeRef.value?.openDialog();
}; };
const appSceneStore = useAppSceneStore();
const changeScene = async (value: string) => {
appSceneStore.setCurrentMode(value);
await updateScene({scene :dictStore.getDictData('app_scene').find(item => item.value == value)?.id});
// 强制刷新页面
window.location.reload();
};
//模式切换 //模式切换
const changeMode = () => { const changeMode = () => {
authStore.changeModel(); authStore.changeModel();
@@ -135,4 +201,18 @@ const changeMode = () => {
// height: 100%; // height: 100%;
} }
} }
.custom-dropdown-trigger {
padding: 8px 15px;
cursor: pointer;
display: flex;
align-items: center;
gap: 8px;
}
:deep(.el-dropdown-menu__item.custom-dropdown-item.active),
:deep(.el-dropdown-menu__item.custom-dropdown-item.active:hover) {
background-color: var(--el-color-primary-light-9) !important;
color: var(--el-color-primary)
}
</style> </style>

View File

@@ -0,0 +1,45 @@
<template>
<el-dialog v-model="dialogVisible" title="主题切换" width="500px" draggable>
<el-divider content-position="center">主题颜色</el-divider>
<div style="display: flex; justify-content: center;">
<el-color-picker v-model="color" />
</div>
<template #footer>
<span class="dialog-footer">
<el-button @click="dialogVisible = false">取消</el-button>
<el-button type="primary" @click="sure">确认</el-button>
</span>
</template>
</el-dialog>
</template>
<script setup lang="ts">
import { onMounted, ref } from "vue";
import { useTheme } from "@/hooks/useTheme";
import { on } from "events";
const color = ref('')
const { changePrimary} = useTheme();
const dialogVisible = ref(false);
const openDialog = () => {
// 修复:使用可选链和空值合并运算符确保不会出现 null 或 undefined
const storedColor = JSON.parse(localStorage.getItem('cn-global') ?? '{}').primary;
color.value = storedColor ?? '#526ADE'; // 默认值为 '#526ADE'
dialogVisible.value = true;
};
const sure = () => {
changePrimary(color.value); // 切换主题
dialogVisible.value = false;
};
// onMounted(() => {
// // 修复:使用可选链和空值合并运算符确保不会出现 null 或 undefined
// const storedColor = JSON.parse(localStorage.getItem('cn-global') ?? '{}').primary;
// console.log('123',storedColor)
// color.value = storedColor ?? '#526ADE'; // 默认值为 '#526ADE'
// })
defineExpose({ openDialog });
</script>

View File

@@ -1,7 +1,7 @@
.el-main { .el-main {
box-sizing: border-box; box-sizing: border-box;
padding: 20px;//主体padding padding: 15px;//主体padding
overflow-x: hidden; overflow: hidden;
background-color: var(--el-bg-color-page); background-color: var(--el-bg-color-page);
} }
.el-footer { .el-footer {

View File

@@ -1,85 +1,81 @@
<template> <template>
<Maximize v-show="maximize" /> <Maximize v-show="maximize" />
<Tabs v-if="tabs && showMenuFlag" /> <Tabs v-if="tabs && showMenuFlag" />
<el-main> <el-main>
<!-- style="height: calc(100vh - 157px);overflow-y:auto; overflow-x:hidden;" :style='{height:showMenuFlag?"calc(100vh - 157px)":" calc(100vh - 120px)"}' --> <router-view v-slot="{ Component, route }" style="height: 100%">
<router-view v-slot="{ Component, route }" style="height:100%;"> <!-- {{ keepAliveName}} -->
<transition appear name="fade-transform" mode="out-in"> <!-- <transition name="slide-right" mode="out-in"> -->
<keep-alive :include="keepAliveName"> <keep-alive :include="tabsMenuList">
<component <component :is="Component" :key="route.fullPath" />
:is="Component" </keep-alive>
v-if="isRouterShow" <!-- </transition> -->
:key="route.fullPath" </router-view>
/> </el-main>
</keep-alive> <el-footer>
</transition> <Footer />
</router-view> </el-footer>
</el-main>
<el-footer >
<!-- v-show="footer" -->
<Footer />
</el-footer>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref, onBeforeUnmount, provide, watch, computed } from "vue"; import { computed, onBeforeUnmount, provide, ref, watch } from 'vue'
import { storeToRefs } from "pinia"; import { storeToRefs } from 'pinia'
import { useDebounceFn } from "@vueuse/core"; import { useDebounceFn } from '@vueuse/core'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import { useKeepAliveStore } from "@/stores/modules/keepAlive"; import { useKeepAliveStore } from '@/stores/modules/keepAlive'
import Maximize from "./components/Maximize.vue"; import Maximize from './components/Maximize.vue'
import Tabs from "@/layouts/components/Tabs/index.vue"; import Tabs from '@/layouts/components/Tabs/index.vue'
import Footer from "@/layouts/components/Footer/index.vue"; import Footer from '@/layouts/components/Footer/index.vue'
import { useAuthStore } from "@/stores/modules/auth"; import { useAuthStore } from '@/stores/modules/auth'
const globalStore = useGlobalStore(); import { useTabsStore } from '@/stores/modules/tabs'
const authStore = useAuthStore();
const { maximize, isCollapse, layout, tabs, footer } = storeToRefs(globalStore);
const keepAliveStore = useKeepAliveStore(); const tabStore = useTabsStore()
const { keepAliveName } = storeToRefs(keepAliveStore); const globalStore = useGlobalStore()
const tabsMenuList = computed(() => tabStore.tabsMenuList.map(item => item.name))
const authStore = useAuthStore()
const { maximize, isCollapse, layout, tabs, footer } = storeToRefs(globalStore)
const keepAliveStore = useKeepAliveStore()
const { keepAliveName } = storeToRefs(keepAliveStore)
//是否显示导航栏 //是否显示导航栏
const showMenuFlag = computed(() => authStore.showMenuFlagGet); const showMenuFlag = computed(() => authStore.showMenuFlagGet)
// 注入刷新页面方法 // 注入刷新页面方法
const isRouterShow = ref(true); const isRouterShow = ref(true)
const refreshCurrentPage = (val: boolean) => (isRouterShow.value = val); const refreshCurrentPage = (val: boolean) => (isRouterShow.value = val)
provide("refresh", refreshCurrentPage); provide('refresh', refreshCurrentPage)
// 监听当前页面是否最大化,动态添加 class // 监听当前页面是否最大化,动态添加 class
watch( watch(
() => maximize.value, () => maximize.value,
() => { () => {
const app = document.getElementById("app") as HTMLElement; const app = document.getElementById('app') as HTMLElement
if (maximize.value) app.classList.add("main-maximize"); if (maximize.value) app.classList.add('main-maximize')
else app.classList.remove("main-maximize"); else app.classList.remove('main-maximize')
}, },
{ immediate: true } { immediate: true }
); )
// 监听布局变化,在 body 上添加相对应的 layout class // 监听布局变化,在 body 上添加相对应的 layout class
watch( watch(
() => layout.value, () => layout.value,
() => { () => {
const body = document.body as HTMLElement; const body = document.body as HTMLElement
body.setAttribute("class", layout.value); body.setAttribute('class', layout.value)
}, },
{ immediate: true } { immediate: true }
); )
// 监听窗口大小变化,折叠侧边栏 // 监听窗口大小变化,折叠侧边栏
const screenWidth = ref(0); const screenWidth = ref(0)
const listeningWindow = useDebounceFn(() => { const listeningWindow = useDebounceFn(() => {
screenWidth.value = document.body.clientWidth; screenWidth.value = document.body.clientWidth
if (!isCollapse.value && screenWidth.value < 1200) if (!isCollapse.value && screenWidth.value < 1200) globalStore.setGlobalState('isCollapse', true)
globalStore.setGlobalState("isCollapse", true); if (isCollapse.value && screenWidth.value > 1200) globalStore.setGlobalState('isCollapse', false)
if (isCollapse.value && screenWidth.value > 1200) }, 100)
globalStore.setGlobalState("isCollapse", false); window.addEventListener('resize', listeningWindow, false)
}, 100);
window.addEventListener("resize", listeningWindow, false);
onBeforeUnmount(() => { onBeforeUnmount(() => {
window.removeEventListener("resize", listeningWindow); window.removeEventListener('resize', listeningWindow)
}); })
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -20,27 +20,36 @@
</template> </template>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { onBeforeMount } from "vue";
import { useRouter } from "vue-router"; import { useRouter } from "vue-router";
defineProps<{ menuList: Menu.MenuOptions[] }>(); defineProps<{ menuList: Menu.MenuOptions[] }>();
const router = useRouter(); const router = useRouter();
const handleClickMenu = (subItem: Menu.MenuOptions) => { const handleClickMenu = (subItem: Menu.MenuOptions) => {
//console.log('1456----------------',subItem);
if (subItem.meta.isLink) return window.open(subItem.meta.isLink, "_blank"); if (subItem.meta.isLink) return window.open(subItem.meta.isLink, "_blank");
router.push(subItem.path); router.push(subItem.path);
}; };
</script> </script>
<style lang="scss"> <style lang="scss">
.el-sub-menu .el-sub-menu__title:hover { .el-sub-menu .el-sub-menu__title:hover {
// color: var(--el-menu-hover-text-color) !important; // color: var(--el-menu-hover-text-color) !important;
// background-color: transparent !important; // background-color: transparent !important;
color: #fff !important;//一级导航文字选中颜色 color: #fff !important;//一级导航文字选中颜色
background-color: #5274a5 !important; //一级导航选中背景色 //background-color: #5274a5 !important; //一级导航选中背景色
background-color: var(--el-color-primary-light-3) !important;
} }
.el-menu--collapse { .el-menu--collapse {
.is-active { .is-active {
.el-sub-menu__title { .el-sub-menu__title {
color: #ffffff !important; color: #ffffff !important;
// background-color: var(--el-color-primary) !important; // background-color: var(--el-color-primary) !important;
background-color: #5274a5 !important; //background-color: #5274a5 !important;
background-color: var(--el-color-primary-light-3) !important;
border-bottom: 0 !important; border-bottom: 0 !important;
} }
} }
@@ -53,7 +62,9 @@ const handleClickMenu = (subItem: Menu.MenuOptions) => {
// color: var(--el-menu-active-color) !important; // color: var(--el-menu-active-color) !important;
// background-color: var(--el-menu-active-bg-color) !important; // background-color: var(--el-menu-active-bg-color) !important;
color: #fff !important;//一级导航文字选中颜色 color: #fff !important;//一级导航文字选中颜色
background-color: #5274a5 !important; //一级导航选中背景色 //background-color: #5274a5 !important; //一级导航选中背景色
background-color: var(--el-color-primary-light-3) !important;
&::before { &::before {
position: absolute; position: absolute;
top: 0; top: 0;

View File

@@ -1,81 +1,88 @@
<template> <template>
<el-dropdown trigger="click" :teleported="false"> <el-dropdown trigger="click" :teleported="false">
<div class="more-button"> <div class="more-button">
<i :class="'iconfont icon-xiala'"></i> <i :class="'iconfont icon-xiala'"></i>
</div> </div>
<template #dropdown> <template #dropdown>
<el-dropdown-menu> <el-dropdown-menu>
<el-dropdown-item @click="refresh"> <el-dropdown-item @click="refresh">
<el-icon><Refresh /></el-icon>{{ $t("tabs.refresh") }} <el-icon><Refresh /></el-icon>
</el-dropdown-item> {{ $t('tabs.refresh') }}
<el-dropdown-item @click="maximize"> </el-dropdown-item>
<el-icon><FullScreen /></el-icon>{{ $t("tabs.maximize") }} <el-dropdown-item @click="maximize">
</el-dropdown-item> <el-icon><FullScreen /></el-icon>
<el-dropdown-item divided @click="closeCurrentTab"> {{ $t('tabs.maximize') }}
<el-icon><Remove /></el-icon>{{ $t("tabs.closeCurrent") }} </el-dropdown-item>
</el-dropdown-item> <el-dropdown-item divided @click="closeCurrentTab">
<el-dropdown-item @click="tabStore.closeTabsOnSide(route.fullPath, 'left')"> <el-icon><Remove /></el-icon>
<el-icon><DArrowLeft /></el-icon>{{ $t("tabs.closeLeft") }} {{ $t('tabs.closeCurrent') }}
</el-dropdown-item> </el-dropdown-item>
<el-dropdown-item @click="tabStore.closeTabsOnSide(route.fullPath, 'right')"> <el-dropdown-item @click="tabStore.closeTabsOnSide(route.fullPath, 'left')">
<el-icon><DArrowRight /></el-icon>{{ $t("tabs.closeRight") }} <el-icon><DArrowLeft /></el-icon>
</el-dropdown-item> {{ $t('tabs.closeLeft') }}
<el-dropdown-item divided @click="tabStore.closeMultipleTab(route.fullPath)"> </el-dropdown-item>
<el-icon><CircleClose /></el-icon>{{ $t("tabs.closeOther") }} <el-dropdown-item @click="tabStore.closeTabsOnSide(route.fullPath, 'right')">
</el-dropdown-item> <el-icon><DArrowRight /></el-icon>
<el-dropdown-item @click="closeAllTab"> {{ $t('tabs.closeRight') }}
<el-icon><FolderDelete /></el-icon>{{ $t("tabs.closeAll") }} </el-dropdown-item>
</el-dropdown-item> <el-dropdown-item divided @click="tabStore.closeMultipleTab(route.fullPath)">
</el-dropdown-menu> <el-icon><CircleClose /></el-icon>
</template> {{ $t('tabs.closeOther') }}
</el-dropdown> </el-dropdown-item>
<el-dropdown-item @click="closeAllTab">
<el-icon><FolderDelete /></el-icon>
{{ $t('tabs.closeAll') }}
</el-dropdown-item>
</el-dropdown-menu>
</template>
</el-dropdown>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { inject, nextTick } from "vue"; import { inject, nextTick } from 'vue'
import { HOME_URL } from "@/config"; import { HOME_URL } from '@/config'
import { useTabsStore } from "@/stores/modules/tabs"; import { useTabsStore } from '@/stores/modules/tabs'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import { useKeepAliveStore } from "@/stores/modules/keepAlive"; import { useKeepAliveStore } from '@/stores/modules/keepAlive'
import { useRoute, useRouter } from "vue-router"; import { useRoute, useRouter } from 'vue-router'
const route = useRoute(); const route = useRoute()
const router = useRouter(); const router = useRouter()
const tabStore = useTabsStore(); const tabStore = useTabsStore()
const globalStore = useGlobalStore(); const globalStore = useGlobalStore()
const keepAliveStore = useKeepAliveStore(); const keepAliveStore = useKeepAliveStore()
// refresh current page // refresh current page
const refreshCurrentPage: Function = inject("refresh") as Function; const refreshCurrentPage: Function = inject('refresh') as Function
const refresh = () => { const refresh = () => {
setTimeout(() => { setTimeout(() => {
keepAliveStore.removeKeepAliveName(route.name as string); keepAliveStore.removeKeepAliveName(route.name as string)
refreshCurrentPage(false); refreshCurrentPage(false)
nextTick(() => { nextTick(() => {
keepAliveStore.addKeepAliveName(route.name as string); keepAliveStore.addKeepAliveName(route.name as string)
refreshCurrentPage(true); refreshCurrentPage(true)
}); })
}, 0); }, 0)
}; }
// maximize current page // maximize current page
const maximize = () => { const maximize = () => {
globalStore.setGlobalState("maximize", true); globalStore.setGlobalState('maximize', true)
}; }
// Close Current // Close Current
const closeCurrentTab = () => { const closeCurrentTab = () => {
if (route.meta.isAffix) return; if (route.meta.isAffix) return
tabStore.removeTabs(route.fullPath); tabStore.removeTabs(route.fullPath)
}; }
// Close All // Close All
const closeAllTab = () => { const closeAllTab = () => {
tabStore.closeMultipleTab(); tabStore.closeMultipleTab()
router.push(HOME_URL); router.push(HOME_URL)
}; }
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "../index.scss"; @use '../index.scss';
</style> </style>

View File

@@ -1,30 +1,35 @@
<template> <template>
<div class='tabs-box'> <div class="tabs-box">
<div class='tabs-menu'> <div class="tabs-menu">
<el-tabs v-model='tabsMenuValue' type='card' @tab-click='tabClick' @tab-remove='tabRemove'> <el-tabs v-model="tabsMenuValue" type="card" @tab-click="tabClick" @tab-remove="tabRemove">
<el-tab-pane v-for='item in tabsMenuList' :key='item.path' :label='item.title' :name='item.path' <el-tab-pane
:closable='item.close'> v-for="item in tabsMenuList"
<template #label> :key="item.path"
<el-icon v-show='item.icon && tabsIcon' class='tabs-icon'> :label="item.title"
<component :is='item.icon'></component> :name="item.path"
</el-icon> :closable="item.close"
{{ item.title }} >
</template> <template #label>
</el-tab-pane> <el-icon v-show="item.icon && tabsIcon" class="tabs-icon">
</el-tabs> <component :is="item.icon"></component>
<MoreButton /> </el-icon>
{{ item.title }}
</template>
</el-tab-pane>
</el-tabs>
<MoreButton />
</div>
</div> </div>
</div>
</template> </template>
<script setup lang='ts'> <script setup lang="ts">
import Sortable from 'sortablejs' import Sortable from 'sortablejs'
import { ref, computed, watch, onMounted } from 'vue' import { computed, onMounted, ref, watch } from 'vue'
import { useRoute, useRouter } from 'vue-router' import { useRoute, useRouter } from 'vue-router'
import { useGlobalStore } from '@/stores/modules/global' import { useGlobalStore } from '@/stores/modules/global'
import { useTabsStore } from '@/stores/modules/tabs' import { useTabsStore } from '@/stores/modules/tabs'
import { useAuthStore } from '@/stores/modules/auth' import { useAuthStore } from '@/stores/modules/auth'
import { TabsPaneContext, TabPaneName } from 'element-plus' import { TabPaneName, TabsPaneContext } from 'element-plus'
import MoreButton from './components/MoreButton.vue' import MoreButton from './components/MoreButton.vue'
const route = useRoute() const route = useRoute()
@@ -38,77 +43,77 @@ const tabsMenuList = computed(() => tabStore.tabsMenuList)
const tabsIcon = computed(() => globalStore.tabsIcon) const tabsIcon = computed(() => globalStore.tabsIcon)
onMounted(() => { onMounted(() => {
tabsDrop() tabsDrop()
initTabs() initTabs()
}) })
// 监听路由的变化(防止浏览器后退/前进不变化 tabsMenuValue // 监听路由的变化(防止浏览器后退/前进不变化 tabsMenuValue
watch( watch(
() => route.fullPath, () => route.fullPath,
() => { () => {
if (route.meta.isFull) return if (route.meta.isFull) return
if (route.meta.hideTab){ if (route.meta.hideTab) {
tabsMenuValue.value = route.meta.parentPath as string tabsMenuValue.value = route.meta.parentPath as string
}else{ } else {
tabsMenuValue.value = route.fullPath tabsMenuValue.value = route.fullPath
const tabsParams = { const tabsParams = {
icon: route.meta.icon as string, icon: route.meta.icon as string,
title: route.meta.title as string, title: route.meta.title as string,
path: route.fullPath, path: route.fullPath,
name: route.name as string, name: route.name as string,
close: !route.meta.isAffix, close: !route.meta.isAffix,
isKeepAlive: route.meta.isKeepAlive as boolean, isKeepAlive: route.meta.isKeepAlive as boolean
} }
tabStore.addTabs(tabsParams) tabStore.addTabs(tabsParams)
} }
}, },
{ immediate: true }, { immediate: true }
) )
// 初始化需要固定的 tabs // 初始化需要固定的 tabs
const initTabs = () => { const initTabs = () => {
authStore.flatMenuListGet.forEach(item => { authStore.flatMenuListGet.forEach(item => {
if (item.meta.isAffix && !item.meta.isHide && !item.meta.isFull) { if (item.meta.isAffix && !item.meta.isHide && !item.meta.isFull) {
const tabsParams = { const tabsParams = {
icon: item.meta.icon, icon: item.meta.icon,
title: item.meta.title, title: item.meta.title,
path: item.path, path: item.path,
name: item.name, name: item.name,
close: !item.meta.isAffix, close: !item.meta.isAffix,
isKeepAlive: item.meta.isKeepAlive, isKeepAlive: item.meta.isKeepAlive,
} unshift: true
tabStore.addTabs(tabsParams) }
} tabStore.addTabs(tabsParams)
}) }
})
} }
// tabs 拖拽排序 // tabs 拖拽排序
const tabsDrop = () => { const tabsDrop = () => {
Sortable.create(document.querySelector('.el-tabs__nav') as HTMLElement, { Sortable.create(document.querySelector('.el-tabs__nav') as HTMLElement, {
draggable: '.el-tabs__item', draggable: '.el-tabs__item',
animation: 300, animation: 300,
onEnd({ newIndex, oldIndex }) { onEnd({ newIndex, oldIndex }) {
const tabsList = [...tabStore.tabsMenuList] const tabsList = [...tabStore.tabsMenuList]
const currRow = tabsList.splice(oldIndex as number, 1)[0] const currRow = tabsList.splice(oldIndex as number, 1)[0]
tabsList.splice(newIndex as number, 0, currRow) tabsList.splice(newIndex as number, 0, currRow)
tabStore.setTabs(tabsList) tabStore.setTabs(tabsList)
}, }
}) })
} }
// Tab Click // Tab Click
const tabClick = (tabItem: TabsPaneContext) => { const tabClick = (tabItem: TabsPaneContext) => {
const fullPath = tabItem.props.name as string const fullPath = tabItem.props.name as string
router.push(fullPath) router.push(fullPath)
} }
// Remove Tab // Remove Tab
const tabRemove = (fullPath: TabPaneName) => { const tabRemove = (fullPath: TabPaneName) => {
tabStore.removeTabs(fullPath as string, fullPath == route.fullPath) tabStore.removeTabs(fullPath as string, fullPath == route.fullPath || '/machine/testScriptAdd' == route.fullPath)
} }
</script> </script>
<style scoped lang='scss'> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -46,7 +46,7 @@
border-radius: 3px; border-radius: 3px;
} }
.layout-light { .layout-light {
background-color: var(--el-color-primary-light-5); background-color: var(--el-color-primary-light-3);
border-radius: 3px; border-radius: 3px;
} }
.layout-content { .layout-content {

View File

@@ -1,186 +1,198 @@
<template> <template>
<el-drawer v-model="drawerVisible" title="布局设置" size="290px"> <el-drawer v-model="drawerVisible" title="布局设置" size="290px">
<!-- 布局样式 --> <!-- 布局样式 -->
<el-divider class="divider" content-position="center"> <el-divider class="divider" content-position="center">
<el-icon><Notification /></el-icon> <el-icon><Notification /></el-icon>
布局样式 布局样式
</el-divider> </el-divider>
<div class="layout-box"> <div class="layout-box">
<el-tooltip effect="dark" content="纵向" placement="top" :show-after="200"> <el-tooltip effect="dark" content="纵向" placement="top" :show-after="200">
<div :class="['layout-item layout-vertical', { 'is-active': layout == 'vertical' }]" @click="setLayout('vertical')"> <div
<div class="layout-dark"></div> :class="['layout-item layout-vertical', { 'is-active': layout == 'vertical' }]"
<div class="layout-container"> @click="setLayout('vertical')"
<div class="layout-light"></div> >
<div class="layout-content"></div> <div class="layout-dark"></div>
</div> <div class="layout-container">
<el-icon v-if="layout == 'vertical'"> <div class="layout-light"></div>
<CircleCheckFilled /> <div class="layout-content"></div>
</el-icon> </div>
<el-icon v-if="layout == 'vertical'">
<CircleCheckFilled />
</el-icon>
</div>
</el-tooltip>
<el-tooltip effect="dark" content="经典" placement="top" :show-after="200">
<div
:class="['layout-item layout-classic', { 'is-active': layout == 'classic' }]"
@click="setLayout('classic')"
>
<div class="layout-dark"></div>
<div class="layout-container">
<div class="layout-light"></div>
<div class="layout-content"></div>
</div>
<el-icon v-if="layout == 'classic'">
<CircleCheckFilled />
</el-icon>
</div>
</el-tooltip>
<el-tooltip effect="dark" content="横向" placement="top" :show-after="200">
<div
:class="['layout-item layout-transverse', { 'is-active': layout == 'transverse' }]"
@click="setLayout('transverse')"
>
<div class="layout-dark"></div>
<div class="layout-content"></div>
<el-icon v-if="layout == 'transverse'">
<CircleCheckFilled />
</el-icon>
</div>
</el-tooltip>
<el-tooltip effect="dark" content="分栏" placement="top" :show-after="200">
<div
:class="['layout-item layout-columns', { 'is-active': layout == 'columns' }]"
@click="setLayout('columns')"
>
<div class="layout-dark"></div>
<div class="layout-light"></div>
<div class="layout-content"></div>
<el-icon v-if="layout == 'columns'">
<CircleCheckFilled />
</el-icon>
</div>
</el-tooltip>
</div> </div>
</el-tooltip> <div class="theme-item">
<el-tooltip effect="dark" content="经典" placement="top" :show-after="200"> <span>
<div :class="['layout-item layout-classic', { 'is-active': layout == 'classic' }]" @click="setLayout('classic')"> 侧边栏反转色
<div class="layout-dark"></div> <el-tooltip effect="dark" content="侧边栏颜色变为深色模式" placement="top">
<div class="layout-container"> <el-icon><QuestionFilled /></el-icon>
<div class="layout-light"></div> </el-tooltip>
<div class="layout-content"></div> </span>
</div> <el-switch v-model="asideInverted" @change="setAsideTheme" />
<el-icon v-if="layout == 'classic'">
<CircleCheckFilled />
</el-icon>
</div> </div>
</el-tooltip> <div class="theme-item mb50">
<el-tooltip effect="dark" content="横向" placement="top" :show-after="200"> <span>
<div :class="['layout-item layout-transverse', { 'is-active': layout == 'transverse' }]" @click="setLayout('transverse')"> 头部反转色
<div class="layout-dark"></div> <el-tooltip effect="dark" content="头部颜色变为深色模式" placement="top">
<div class="layout-content"></div> <el-icon><QuestionFilled /></el-icon>
<el-icon v-if="layout == 'transverse'"> </el-tooltip>
<CircleCheckFilled /> </span>
</el-icon> <el-switch v-model="headerInverted" @change="setHeaderTheme" />
</div> </div>
</el-tooltip>
<el-tooltip effect="dark" content="分栏" placement="top" :show-after="200">
<div :class="['layout-item layout-columns', { 'is-active': layout == 'columns' }]" @click="setLayout('columns')">
<div class="layout-dark"></div>
<div class="layout-light"></div>
<div class="layout-content"></div>
<el-icon v-if="layout == 'columns'">
<CircleCheckFilled />
</el-icon>
</div>
</el-tooltip>
</div>
<div class="theme-item">
<span>
侧边栏反转色
<el-tooltip effect="dark" content="侧边栏颜色变为深色模式" placement="top">
<el-icon><QuestionFilled /></el-icon>
</el-tooltip>
</span>
<el-switch v-model="asideInverted" @change="setAsideTheme" />
</div>
<div class="theme-item mb50">
<span>
头部反转色
<el-tooltip effect="dark" content="头部颜色变为深色模式" placement="top">
<el-icon><QuestionFilled /></el-icon>
</el-tooltip>
</span>
<el-switch v-model="headerInverted" @change="setHeaderTheme" />
</div>
<!-- 全局主题 --> <!-- 全局主题 -->
<el-divider class="divider" content-position="center"> <el-divider class="divider" content-position="center">
<el-icon><ColdDrink /></el-icon> <el-icon><ColdDrink /></el-icon>
全局主题 全局主题
</el-divider> </el-divider>
<div class="theme-item"> <div class="theme-item">
<span>主题颜色</span> <span>主题颜色</span>
<el-color-picker v-model="primary" :predefine="colorList" @change="changePrimary" /> <el-color-picker v-model="primary" :predefine="colorList" @change="changePrimary" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>暗黑模式</span> <span>暗黑模式</span>
<SwitchDark /> <SwitchDark />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>灰色模式</span> <span>灰色模式</span>
<el-switch v-model="isGrey" @change="changeGreyOrWeak('grey', !!$event)" /> <el-switch v-model="isGrey" @change="changeGreyOrWeak('grey', !!$event)" />
</div> </div>
<div class="theme-item mb40"> <div class="theme-item mb40">
<span>色弱模式</span> <span>色弱模式</span>
<el-switch v-model="isWeak" @change="changeGreyOrWeak('weak', !!$event)" /> <el-switch v-model="isWeak" @change="changeGreyOrWeak('weak', !!$event)" />
</div> </div>
<!-- 界面设置 --> <!-- 界面设置 -->
<el-divider class="divider" content-position="center"> <el-divider class="divider" content-position="center">
<el-icon><Setting /></el-icon> <el-icon><Setting /></el-icon>
界面设置 界面设置
</el-divider> </el-divider>
<div class="theme-item"> <div class="theme-item">
<span>菜单折叠</span> <span>菜单折叠</span>
<el-switch v-model="isCollapse" /> <el-switch v-model="isCollapse" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>菜单手风琴</span> <span>菜单手风琴</span>
<el-switch v-model="accordion" /> <el-switch v-model="accordion" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>面包屑</span> <span>面包屑</span>
<el-switch v-model="breadcrumb" /> <el-switch v-model="breadcrumb" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>面包屑图标</span> <span>面包屑图标</span>
<el-switch v-model="breadcrumbIcon" /> <el-switch v-model="breadcrumbIcon" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>标签栏</span> <span>标签栏</span>
<el-switch v-model="tabs" /> <el-switch v-model="tabs" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>标签栏图标</span> <span>标签栏图标</span>
<el-switch v-model="tabsIcon" /> <el-switch v-model="tabsIcon" />
</div> </div>
<div class="theme-item"> <div class="theme-item">
<span>页脚</span> <span>页脚</span>
<el-switch v-model="footer" /> <el-switch v-model="footer" />
</div> </div>
</el-drawer> </el-drawer>
</template> </template>
<script setup lang="ts"> <script setup lang="ts">
import { ref } from "vue"; import { ref } from 'vue'
import { storeToRefs } from "pinia"; import { storeToRefs } from 'pinia'
import { useTheme } from "@/hooks/useTheme"; import { useTheme } from '@/hooks/useTheme'
import { useGlobalStore } from "@/stores/modules/global"; import { useGlobalStore } from '@/stores/modules/global'
import { LayoutType } from "@/stores/interface"; import { LayoutType } from '@/stores/interface'
import { DEFAULT_PRIMARY } from "@/config"; import { DEFAULT_PRIMARY } from '@/config'
import mittBus from "@/utils/mittBus"; import mittBus from '@/utils/mittBus'
import SwitchDark from "@/components/SwitchDark/index.vue"; import SwitchDark from '@/components/SwitchDark/index.vue'
const { changePrimary, changeGreyOrWeak, setAsideTheme, setHeaderTheme } = useTheme(); const { changePrimary, changeGreyOrWeak, setAsideTheme, setHeaderTheme } = useTheme()
const globalStore = useGlobalStore(); const globalStore = useGlobalStore()
const { const {
layout, layout,
primary, primary,
isGrey, isGrey,
isWeak, isWeak,
asideInverted, asideInverted,
headerInverted, headerInverted,
isCollapse, isCollapse,
accordion, accordion,
breadcrumb, breadcrumb,
breadcrumbIcon, breadcrumbIcon,
tabs, tabs,
tabsIcon, tabsIcon,
footer footer
} = storeToRefs(globalStore); } = storeToRefs(globalStore)
// 预定义主题颜色 // 预定义主题颜色
const colorList = [ const colorList = [
DEFAULT_PRIMARY, DEFAULT_PRIMARY,
"#daa96e", '#daa96e',
"#0c819f", '#0c819f',
"#409eff", '#409eff',
"#27ae60", '#27ae60',
"#ff5c93", '#ff5c93',
"#e74c3c", '#e74c3c',
"#fd726d", '#fd726d',
"#f39c12", '#f39c12',
"#9b59b6" '#9b59b6'
]; ]
// 设置布局方式 // 设置布局方式
const setLayout = (val: LayoutType) => { const setLayout = (val: LayoutType) => {
globalStore.setGlobalState("layout", val); globalStore.setGlobalState('layout', val)
setAsideTheme(); setAsideTheme()
}; }
// 打开主题设置 // 打开主题设置
const drawerVisible = ref(false); const drawerVisible = ref(false)
mittBus.on("openThemeDrawer", () => (drawerVisible.value = true)); mittBus.on('openThemeDrawer', () => (drawerVisible.value = true))
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
@import "./index.scss"; @use './index.scss';
</style> </style>

View File

@@ -65,5 +65,6 @@ const setupAll = async () => {
//挂载app //挂载app
setupAll().then(() => { setupAll().then(() => {
app.mount('#app') app.mount('#app')
}) })

View File

@@ -30,6 +30,7 @@ const routerMode = {
* @param meta.isKeepAlive ==> 当前路由是否缓存 * @param meta.isKeepAlive ==> 当前路由是否缓存
* */ * */
const router = createRouter({ const router = createRouter({
history: routerMode[mode](), history: routerMode[mode](),
routes: [...staticRouter], routes: [...staticRouter],
// 不区分路由大小写,非严格模式下提供了更宽松的路径匹配 // 不区分路由大小写,非严格模式下提供了更宽松的路径匹配
@@ -46,14 +47,13 @@ router.beforeEach(async (to, from, next) => {
const authStore = useAuthStore() const authStore = useAuthStore()
// 1.NProgress 开始 // 1.NProgress 开始
NProgress.start() NProgress.start()
// 2.动态设置标题 // 2.动态设置标题
const title = import.meta.env.VITE_GLOB_APP_TITLE const title = import.meta.env.VITE_GLOB_APP_TITLE
document.title = to.meta.title ? `${to.meta.title} - ${title}` : title document.title = to.meta.title ? `${to.meta.title} - ${title}` : title
// 3.判断是访问登陆页,有 Token 就在当前页面,没有 Token 重置路由到登陆页 // 3.判断是访问登陆页,有 Token 就在当前页面,没有 Token 重置路由到登陆页
if (to.path.toLocaleLowerCase() === LOGIN_URL) { if (to.path.toLocaleLowerCase() === LOGIN_URL) {
if (userStore.token) return next(from.fullPath) if (userStore.accessToken) return next(from.fullPath)
resetRouter() resetRouter()
return next() return next()
} }
@@ -62,17 +62,16 @@ router.beforeEach(async (to, from, next) => {
if (ROUTER_WHITE_LIST.includes(to.path)) return next() if (ROUTER_WHITE_LIST.includes(to.path)) return next()
// 5.判断是否有 Token没有重定向到 login 页面 // 5.判断是否有 Token没有重定向到 login 页面
if (!userStore.token) return next({ path: LOGIN_URL, replace: true }) if (!userStore.accessToken) return next({ path: LOGIN_URL, replace: true })
// 6.如果没有菜单列表,就重新请求菜单列表并添加动态路由 // 6.如果没有菜单列表,就重新请求菜单列表并添加动态路由
if (!authStore.authMenuListGet.length) { if (!authStore.authMenuListGet.length) {
await initDynamicRouter() await initDynamicRouter()
return next({ ...to, replace: true }) return next({ ...to, replace: true })
} }
//console.log(to)
// 7.存储 routerName 做按钮权限筛选 // 7.存储 routerName 做按钮权限筛选
authStore.setRouteName(to.name as string) authStore.setRouteName(to.name as string)
// 8.正常访问页面 // 8.正常访问页面
next() next()
}) })

View File

@@ -28,7 +28,9 @@ export const initDynamicRouter = async () => {
type: "warning", type: "warning",
duration: 3000 duration: 3000
}); });
userStore.setToken(""); userStore.setAccessToken("");
userStore.setRefreshToken("");
userStore.setExp(0)
router.replace(LOGIN_URL); router.replace(LOGIN_URL);
return Promise.reject("No permission"); return Promise.reject("No permission");
} }
@@ -36,9 +38,11 @@ export const initDynamicRouter = async () => {
// 3.添加动态路由 // 3.添加动态路由
authStore.flatMenuListGet.forEach(item => { authStore.flatMenuListGet.forEach(item => {
item.children && delete item.children; item.children && delete item.children;
if (item.component && typeof item.component == "string") { if (item.component && typeof item.component == "string") {
item.component = modules["/src/views" + item.component + ".vue"]; item.component = modules["/src/views" + item.component + ".vue"];
} }
if (item.meta.isFull) { if (item.meta.isFull) {
router.addRoute(item as unknown as RouteRecordRaw); router.addRoute(item as unknown as RouteRecordRaw);
} else { } else {
@@ -47,7 +51,9 @@ export const initDynamicRouter = async () => {
}); });
} catch (error) { } catch (error) {
// 当按钮 || 菜单请求出错时,重定向到登陆页 // 当按钮 || 菜单请求出错时,重定向到登陆页
userStore.setToken(""); userStore.setAccessToken("");
userStore.setRefreshToken("");
userStore.setExp(0)
router.replace(LOGIN_URL); router.replace(LOGIN_URL);
return Promise.reject(error); return Promise.reject(error);
} }

View File

@@ -1,122 +1,139 @@
import { RouteRecordRaw } from "vue-router"; import { RouteRecordRaw } from 'vue-router'
import { HOME_URL, LOGIN_URL } from "@/config"; import { HOME_URL, LOGIN_URL } from '@/config'
export const Layout = () => import('@/layouts/index.vue'); export const Layout = () => import('@/layouts/index.vue')
/** /**
* staticRouter (静态路由) * staticRouter (静态路由)
*/ */
export const staticRouter: RouteRecordRaw[] = [ export const staticRouter: RouteRecordRaw[] = [
{ {
path: "/", path: '/',
redirect: HOME_URL, redirect: HOME_URL
},
{
path: LOGIN_URL,
name: "login",
component: () => import("@/views/login/index.vue"),
meta: {
title: "登录",
}, },
}, {
{ path: LOGIN_URL,
path: "/layout", name: 'login',
name: "layout", component: () => import('@/views/login/index.vue'),
component: Layout, meta: {
children: [ title: '登录'
{ }
path: "/plan", },
name: "plan", {
redirect: "/plan/planList", path: '/layout',
name: 'layout',
component: Layout,
children: [ children: [
{ {
path: "/plan/planList", path: '/plan',
name: "planList", name: 'plan',
component: () => import("@/views/plan/planList/index.vue"), redirect: '/plan/planList',
meta: { children: [
title: "检测计划列表", // {
icon: "List", // path: "/plan/planList",
isLink: "", // name: "planList",
isHide: false, // component: () => import("@/views/plan/planList/index.vue"),
isFull: false, // meta: {
isAffix: false, // title: "检测计划列表",
isKeepAlive: false, // icon: "List",
// isLink: "",
// isHide: false,
// isFull: false,
// isAffix: false,
// isKeepAlive: true,//缓存改成true
// },
// },
// {
// path: "/plan/singlePlanList",
// name: "singlePlanList",
// component: () => import("@/views/plan/singlePlanList/index.vue"),
// meta: {
// title: "单个计划列表",
// icon: "List",
// isLink: "",
// isHide: false,
// isFull: false,
// isAffix: false,
// isKeepAlive: true,
// },
// },
// {
// path: "/plan/preTest",
// name: "preTest",
// component: () => import("@/views/plan/preTest/index.vue"),
// meta: {
// title: "预检测",
// icon: "List",
// isLink: "",
// isHide: false,
// isFull: false,
// isAffix: false,
// isKeepAlive: false,
// },
// },
// {
// path: "/plan/autoTest",
// name: "autoTest",
// component: () => import("@/views/plan/autoTest/index.vue"),
// meta: {
// title: "自动检测",
// icon: "List",
// isLink: "",
// hideTab:true,
// parentPath:'/system/proTable',
// isHide: false,
// isFull: false,
// isAffix: false,
// isKeepAlive: true,
// },
// },
// 错误页面路由
{
path: '/403',
name: '403',
component: () => import('@/components/ErrorMessage/403.vue'),
meta: {
title: '403页面'
}
},
{
path: '/404',
name: '404',
component: () => import('@/components/ErrorMessage/404.vue'),
meta: {
title: '404页面'
}
},
{
path: '/500',
name: '500',
component: () => import('@/components/ErrorMessage/500.vue'),
meta: {
title: '500页面'
}
},
// Resolve refresh page, route warnings
{
path: '/:pathMatch(.*)*',
component: () => import('@/components/ErrorMessage/404.vue')
}
]
}, },
}, {
{ path: '/machine/testScriptAdd',
path: "/plan/singlePlanList", name: 'testScriptAdd',
name: "singlePlanList", component: () => import('@/views/machine/testScript/components/testScriptPopup.vue'),
component: () => import("@/views/plan/singlePlanList/index.vue"), meta: {
meta: { title: '检测脚本配置',
title: "单个计划列表", icon: 'List',
icon: "List", isLink: '',
isLink: "", hideTab: true,
isHide: false, parentPath: '/machine/testScript',
isFull: false, isHide: false,
isAffix: false, isFull: false,
isKeepAlive: false, isAffix: false,
}, isKeepAlive: true
}, }
{ }
path: "/plan/preTest", ]
name: "preTest", }
component: () => import("@/views/plan/preTest/index.vue"), ]
meta: {
title: "预检测",
icon: "List",
isLink: "",
isHide: false,
isFull: false,
isAffix: false,
isKeepAlive: false,
},
},
{
path: "/plan/autoTest",
name: "autoTest",
component: () => import("@/views/plan/autoTest/index.vue"),
meta: {
title: "自动检测",
icon: "List",
isLink: "",
hideTab:true,
parentPath:'/system/proTable',
isHide: false,
isFull: false,
isAffix: false,
isKeepAlive: false,
},
},
// 错误页面路由
{
path: "/403",
name: "403",
component: () => import("@/components/ErrorMessage/403.vue"),
meta: {
title: "403页面",
},
},
{
path: "/404",
name: "404",
component: () => import("@/components/ErrorMessage/404.vue"),
meta: {
title: "404页面",
},
},
{
path: "/500",
name: "500",
component: () => import("@/components/ErrorMessage/500.vue"),
meta: {
title: "500页面",
},
},
// Resolve refresh page, route warnings
{
path: "/:pathMatch(.*)*",
component: () => import("@/components/ErrorMessage/404.vue"),
},
],
}
],
},
];

View File

@@ -17,3 +17,5 @@ export const USER_STORE_KEY = "cn-user";
// pinia中dict store的key // pinia中dict store的key
export const DICT_STORE_KEY = "cn-dictData"; export const DICT_STORE_KEY = "cn-dictData";
export const CHECK_STORE_KEY = "cn-check";

View File

@@ -6,57 +6,60 @@ export type LanguageType = 'zh' | 'en' | null;
/* GlobalState */ /* GlobalState */
export interface GlobalState { export interface GlobalState {
layout: LayoutType; layout: LayoutType;
assemblySize: AssemblySizeType; assemblySize: AssemblySizeType;
language: LanguageType; language: LanguageType;
maximize: boolean; maximize: boolean;
primary: string; primary: string;
isDark: boolean; isDark: boolean;
isGrey: boolean; isGrey: boolean;
isWeak: boolean; isWeak: boolean;
asideInverted: boolean; asideInverted: boolean;
headerInverted: boolean; headerInverted: boolean;
isCollapse: boolean; isCollapse: boolean;
accordion: boolean; accordion: boolean;
breadcrumb: boolean; breadcrumb: boolean;
breadcrumbIcon: boolean; breadcrumbIcon: boolean;
tabs: boolean; tabs: boolean;
tabsIcon: boolean; tabsIcon: boolean;
footer: boolean; footer: boolean;
} }
/* UserState */ /* UserState */
export interface UserState { export interface UserState {
token: string; accessToken: string;
userInfo: { name: string }; refreshToken: string;
isRefreshToken: boolean;
userInfo: { id: string, name: string,loginName:string };
} }
/* tabsMenuProps */ /* tabsMenuProps */
export interface TabsMenuProps { export interface TabsMenuProps {
icon: string; icon: string;
title: string; title: string;
path: string; path: string;
name: string; name: string;
close: boolean; close: boolean;
isKeepAlive: boolean; isKeepAlive: boolean;
unshift?: boolean;
} }
/* TabsState */ /* TabsState */
export interface TabsState { export interface TabsState {
tabsMenuList: TabsMenuProps[]; tabsMenuList: TabsMenuProps[];
} }
/* AuthState */ /* AuthState */
export interface AuthState { export interface AuthState {
routeName: string; routeName: string;
authButtonList: { authButtonList: {
[key: string]: string[]; [key: string]: string[];
}; };
authMenuList: Menu.MenuOptions[]; authMenuList: Menu.MenuOptions[];
showMenuFlag: boolean; showMenuFlag: boolean;
} }
/* KeepAliveState */ /* KeepAliveState */
export interface KeepAliveState { export interface KeepAliveState {
keepAliveName: string[]; keepAliveName: string[];
} }

Some files were not shown because too many files have changed in this diff Show More