86 Commits

Author SHA1 Message Date
2765bcab10 故障告警,搜索栏 告警等级宽度调整 2026-01-29 20:08:33 +08:00
b55d9e2d5c Merge branch 'develop' into single-develop 2026-01-29 20:07:34 +08:00
29ab53056a 故障告警,搜索栏 告警等级宽度调整 2026-01-29 19:57:10 +08:00
0c32439146 Merge branch 'develop' into single-develop 2026-01-28 21:45:43 +08:00
5250db915f Merge branch 'develop' of http://101.43.41.9:13000/xzzn/emsfront into develop 2026-01-28 21:36:14 +08:00
814103c881 统计报表优化 2026-01-28 21:34:58 +08:00
55b7fba021 修改图片 2026-01-27 16:58:50 +08:00
da4ecc4792 修改顺序 2026-01-27 13:45:14 +08:00
498dc117f2 变更图片 2026-01-27 10:43:54 +08:00
058879ec45 Merge branch 'develop' into single-develop 2026-01-26 23:36:47 +08:00
4d29de93a1 大数据图片更新 2026-01-26 23:30:01 +08:00
445e9dfc9f 大数据图片改为切换模式 2026-01-26 22:40:57 +08:00
3a26f6312b 更新注释 2026-01-25 17:02:21 +08:00
138b0753af Merge branch 'develop' into single-develop 2026-01-25 16:56:29 +08:00
ac1d1ae154 更新workStatus状态枚举 2026-01-25 16:45:33 +08:00
825243e741 电池簇备注 2026-01-25 16:36:23 +08:00
0389ed85f3 Merge branch 'develop' of http://101.43.41.9:13000/xzzn/emsfront into develop 2026-01-25 16:33:00 +08:00
aef94f406a 电池簇、电池堆工作状态更新 2026-01-25 16:30:43 +08:00
95d69fb7b1 簇枚举变更 2026-01-25 16:27:35 +08:00
684002ffc8 簇枚举变更 2026-01-25 15:12:07 +08:00
098dfa05f8 簇枚举变更 2026-01-25 14:34:01 +08:00
d6c9310e50 净置改成静置 2026-01-25 13:54:23 +08:00
085f6fd252 Merge branch 'develop' into single-develop 2026-01-24 21:19:27 +08:00
142de3102b 大数据图片更新 2026-01-24 21:14:43 +08:00
d3e51322ab Merge branch 'develop' into single-develop 2026-01-24 10:37:31 +08:00
9b5806a2c0 0或没有返回值隐藏alarmNum的展示 2026-01-24 10:27:43 +08:00
2b6697fa5a pointName改成 dataPointName 2026-01-24 10:02:11 +08:00
3bb859b693 BMS电池簇数据同步 2026-01-23 22:32:12 +08:00
5b3701afd0 BMS总览数据同步 2026-01-23 22:02:02 +08:00
f531075853 电表数据同步 2026-01-23 21:46:02 +08:00
81f8f75d7e pcs开关机状态判断更新 2026-01-23 20:06:47 +08:00
fccddab04a Merge branch 'develop' into single-develop 2026-01-23 19:52:13 +08:00
8cb6fbee3e 大数据展示页面 2026-01-23 18:00:14 +08:00
f454b02c99 pcs、bmszl设备工作状态更新 2026-01-23 16:30:20 +08:00
bfe72cf2c3 首页字段、收益表格 2026-01-23 14:18:22 +08:00
fb90d81bb3 站点首页优化 2026-01-22 17:27:03 +08:00
823c0949d0 文案修改,实时运行图表切换站点数据未更新问题修复,电表未知类型 2026-01-21 18:58:37 +08:00
1b712a70f6 Merge branch 'develop' into single-develop 2026-01-21 14:42:02 +08:00
c1c411e48a 用户归属站点 2026-01-21 14:37:39 +08:00
43153a791d 用户归属站点 2026-01-21 14:05:09 +08:00
5d84972b8a 菜单 2026-01-16 16:08:54 +08:00
b122ab35ab Merge branch 'develop' into single-develop
# Conflicts:
#	src/router/ems.js
2026-01-16 15:59:17 +08:00
11111d035b 优化 2026-01-13 16:38:21 +08:00
80f07876c6 Merge branch 'develop' into single-develop 2026-01-12 14:17:32 +08:00
ab9bb1e85d pcs\bmszl\bmsdcc 标题颜色规则更新 2026-01-12 14:15:21 +08:00
bef8f01d0d pcs开关机接口参数更新 2026-01-08 17:14:57 +08:00
b469190d9f Merge branch 'develop' into single-develop 2026-01-08 16:59:47 +08:00
4079c40e5d workStatus、deviceStatus枚举更新,pcs开关机接口参数更新 2026-01-08 16:59:15 +08:00
e98b4ed287 Merge branch 'develop' into single-develop 2025-12-31 17:25:47 +08:00
4ebd5f0988 样式优化 2025-12-31 17:23:08 +08:00
a0095b4054 pcs设备类型,新增pcs设备配置选项,策略配置时间问题修复 2025-12-31 17:15:26 +08:00
6140ca8f14 Merge branch 'develop' into single-develop 2025-12-26 21:01:54 +08:00
84bc29410a 新增设备:TCP设备新增从站地址
策略配置:新增放电状态、可输入负数、开始、结束时间组件更换
2025-12-26 20:19:20 +08:00
a3d6cf7c55 Merge branch 'develop' into single-develop 2025-12-19 21:51:51 +08:00
f3fda19c64 保护方案 2025-12-19 21:49:19 +08:00
6fb706b528 云下展示pcs开关机按钮 2025-12-18 16:37:59 +08:00
579920e007 Merge branch 'develop' into single-develop 2025-12-18 16:35:34 +08:00
9a8247f833 云上隐藏pcs开关机按钮 2025-12-18 16:26:57 +08:00
d9eeee106f Merge branch 'develop' into single-develop 2025-12-16 13:48:09 +08:00
c4c79aaa64 设备列表顶部下载按钮展示 2025-12-16 13:44:48 +08:00
c90f4f5766 Merge branch 'develop' into single-develop 2025-12-16 11:14:50 +08:00
0966813c25 设备列表,隐藏顶部上传、下载按钮,pcs开机关机按钮显示条件优化 2025-12-16 11:03:59 +08:00
4a4d26f018 生产、云下环境变量配置 2025-12-15 17:41:29 +08:00
1a11c67aad test 环境变量配置 2025-12-15 14:42:27 +08:00
71fd48918d Merge branch 'develop' into single-develop
# Conflicts:
#	src/assets/styles/common.scss
2025-12-12 17:47:57 +08:00
dd4fa36597 优化 2025-12-12 17:38:26 +08:00
9b14d96e24 pcs开关机,设备列表上传、下载 2025-12-11 17:59:34 +08:00
34d9e038e6 合并代码 2025-12-10 15:09:33 +08:00
ac54ce999e ems、箱型图 2025-12-10 15:00:12 +08:00
f49efb5edd ems接口、综合查询 2025-12-09 18:25:36 +08:00
97f9b3ff0e ems 2025-12-08 17:00:18 +08:00
674a0c6c33 ems、收益报表 2025-12-08 16:58:43 +08:00
b4c49c3e58 Merge branch 'develop' into single-develop 2025-12-05 09:49:54 +08:00
7f3d02b8fb 点位、报警点位 2025-12-05 09:46:31 +08:00
9cfce671a6 Merge branch 'develop' into single-develop 2025-12-05 09:16:54 +08:00
cd8871d45a 点位上传、下载 2025-12-04 17:46:11 +08:00
11d44de513 表格内容fong-size调小1px 2025-11-28 17:53:34 +08:00
87e3db9f0f 云下使用深色主题区分云上 2025-11-28 15:21:42 +08:00
4132d8e539 合并develop最新代码 2025-11-28 15:18:27 +08:00
78eb81549b 设备列表上传 2025-11-28 15:11:34 +08:00
65f0e92296 2025-11-20紧急优化内容修复 2025-11-27 15:58:23 +08:00
9a585e97db 2025-11-20紧急优化内容-动环、消防 2025-11-27 15:51:13 +08:00
7fb6d1aa47 2025-11-20紧急优化内容 2025-11-26 17:50:06 +08:00
1378947a9e 设备监控-点位清单 2025-11-25 17:56:12 +08:00
28176ce052 Merge branch 'develop' into single-develop 2025-11-13 16:53:19 +08:00
0b2e7d9b86 mqtt新增站点、qos 2025-11-13 16:49:39 +08:00
54 changed files with 5737 additions and 4357 deletions

View File

@ -5,8 +5,9 @@ VUE_APP_TITLE = 上动新能源-EMS管理系统
NODE_ENV = 'production' NODE_ENV = 'production'
# EMS管理系统/生产环境 # EMS管理系统/生产环境
VUE_APP_BASE_API= 'http://1.15.120.242:8089' VUE_APP_BASE_API= 'http://127.0.0.1:8089'
# EMS管理系统/生产环境 图片拼接地址 # EMS管理系统/生产环境 图片拼接地址
VUE_APP_IMG_URL = 'http://1.15.120.242:8089' VUE_APP_IMG_URL = 'http://127.0.0.1:8089'

View File

@ -2,314 +2,383 @@ import request from '@/utils/request'
//获取单个站点的信息 //获取单个站点的信息
export function getDzjkHomeView(siteId) { export function getDzjkHomeView(siteId) {
return request({ return request({
url: `/ems/siteMonitor/homeView?siteId=${siteId}`, url: `/ems/siteMonitor/homeView?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//站点首页 冲放曲线 //站点首页 冲放曲线
export function getSevenChargeData({siteId,startDate,endDate}) { export function getSevenChargeData({siteId, startDate, endDate}) {
return request({ return request({
url: `/ems/siteMap/getSevenChargeData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`, url: `/ems/siteMap/getSevenChargeData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`,
method: 'get' method: 'get'
}) })
} }
// 获取站点包含的设备种类 用来判断单站监控设备监控的菜单栏展示 // 获取站点包含的设备种类 用来判断单站监控设备监控的菜单栏展示
export function getSiteAllDeviceCategory(siteId) { export function getSiteAllDeviceCategory(siteId) {
return request({ return request({
url: `/ems/siteConfig/getSiteAllDeviceCategory?siteId=${siteId}`, url: `/ems/siteConfig/getSiteAllDeviceCategory?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//EMS
export function getEmsDataList(siteId) {
return request({
url: `/ems/siteMonitor/getEmsDataList?siteId=${siteId}`,
method: 'get',
})
}
//获取pcs、实时运行头部的设备信息 //获取pcs、实时运行头部的设备信息
export function getRunningHeadInfo(siteId) { export function getRunningHeadInfo(siteId) {
return request({ return request({
url: `/ems/siteMonitor/runningHeadInfo?siteId=${siteId}`, url: `/ems/siteMonitor/runningHeadInfo?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取pcs列表 //获取pcs列表
export function getPcsDetailInfo(siteId) { export function getPcsDetailInfo(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getPcsDetailInfo?siteId=${siteId}`, url: `/ems/siteMonitor/getPcsDetailInfo?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取BMS总览数据 //获取BMS总览数据
export function getBMSOverView(siteId) { export function getBMSOverView(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getBMSOverView?siteId=${siteId}`, url: `/ems/siteMonitor/getBMSOverView?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取BMS电池簇总览数据 //获取BMS电池簇总览数据
export function getBMSBatteryCluster(siteId) { export function getBMSBatteryCluster(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getBMSBatteryCluster?siteId=${siteId}`, url: `/ems/siteMonitor/getBMSBatteryCluster?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取单体电池 电池堆列表数据 //获取单体电池 电池堆列表数据
export function getStackNameList(siteId) { export function getStackNameList(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getStackNameList?siteId=${siteId}`, url: `/ems/siteMonitor/getStackNameList?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取单体电池 电池簇列表数据 //获取单体电池 电池簇列表数据
export function getClusterNameList({stackDeviceId,siteId}) { export function getClusterNameList({stackDeviceId, siteId}) {
return request({ return request({
url: `/ems/siteMonitor/getClusterNameList?stackDeviceId=${stackDeviceId}&siteId=${siteId}`, url: `/ems/siteMonitor/getClusterNameList?stackDeviceId=${stackDeviceId}&siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//单体电池表格数据 //单体电池表格数据
export function getClusterDataInfoList({siteId, stackDeviceId, clusterDeviceId,batteryId, pageSize, pageNum}) { export function getClusterDataInfoList({siteId, stackDeviceId, clusterDeviceId, batteryId, pageSize, pageNum}) {
return request({ return request({
url: `/ems/siteMonitor/getClusterDataInfoList?clusterDeviceId=${clusterDeviceId}&siteId=${siteId}&stackDeviceId=${stackDeviceId}&batteryId=${batteryId}&pageSize=${pageSize}&pageNum=${pageNum}`, url: `/ems/siteMonitor/getClusterDataInfoList?clusterDeviceId=${clusterDeviceId}&siteId=${siteId}&stackDeviceId=${stackDeviceId}&batteryId=${batteryId}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get' method: 'get'
}) })
} }
// 单体电池图表 // 单体电池图表
//http://localhost:8089/ems/siteMonitor/getSingleBatteryData?clusterDeviceId=BMSC01&siteId=021_FXX_01&deviceId=001&startDate=2025-07-11&endDate=2025-07-18 //http://localhost:8089/ems/siteMonitor/getSingleBatteryData?clusterDeviceId=BMSC01&siteId=021_FXX_01&deviceId=001&startDate=2025-07-11&endDate=2025-07-18
export function getSingleBatteryData({siteId,deviceId,clusterDeviceId,startDate,endDate}) { export function getSingleBatteryData({siteId, deviceId, clusterDeviceId, startDate, endDate}) {
return request({ return request({
url: `/ems/siteMonitor/getSingleBatteryData?siteId=${siteId}&deviceId=${deviceId}&startDate=${startDate}&endDate=${endDate}&clusterDeviceId=${clusterDeviceId}`, url: `/ems/siteMonitor/getSingleBatteryData?siteId=${siteId}&deviceId=${deviceId}&startDate=${startDate}&endDate=${endDate}&clusterDeviceId=${clusterDeviceId}`,
method: 'get' method: 'get'
}) })
} }
//获取液冷列表数据 //获取液冷列表数据
export function getCoolingDataList(siteId) { export function getCoolingDataList(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getCoolingDataList?siteId=${siteId}`, url: `/ems/siteMonitor/getCoolingDataList?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//获取动环数据
export function getDhDataList(siteId) {
return request({
url: `/ems/siteMonitor/getDhDataList?siteId=${siteId}`,
method: 'get'
})
}
//获取消防数据
export function getXfDataList(siteId) {
return request({
url: `/ems/siteMonitor/getXfDataList?siteId=${siteId}`,
method: 'get'
})
}
//获取电表数据 //获取电表数据
export function getAmmeterDataList(siteId) { export function getAmmeterDataList(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getAmmeterDataList?siteId=${siteId}`, url: `/ems/siteMonitor/getAmmeterDataList?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
// 故障告警 // 故障告警
export function getAlarmDetailList({status,siteId, deviceId, alarmLevel, alarmStartTime, alarmEndTime,pageSize,pageNum}) { export function getAlarmDetailList({
return request({ status,
url: `/ems/siteAlarm/getAlarmDetailList?siteId=${siteId}&status=${status}&deviceId=${deviceId}&alarmLevel=${alarmLevel}&alarmStartTime=${alarmStartTime}&alarmEndTime=${alarmEndTime}&pageSize=${pageSize}&pageNum=${pageNum}`, siteId,
method: 'get' deviceId,
}) alarmLevel,
alarmStartTime,
alarmEndTime,
pageSize,
pageNum
}) {
return request({
url: `/ems/siteAlarm/getAlarmDetailList?siteId=${siteId}&status=${status}&deviceId=${deviceId}&alarmLevel=${alarmLevel}&alarmStartTime=${alarmStartTime}&alarmEndTime=${alarmEndTime}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get'
})
} }
// 告警生成工单 // 告警生成工单
export function createTicketNo(data) { export function createTicketNo(data) {
return request({ return request({
url: `/ems/siteAlarm/createTicketNo`, url: `/ems/siteAlarm/createTicketNo`,
method: 'post', method: 'post',
data data
}) })
} }
// 概率统计 // 概率统计
//获取概率统计 电量指标接口 //获取概率统计 电量指标接口
export function getElectricData({siteId,startDate,endDate}) { export function getElectricData({siteId, startDate, endDate}) {
return request({ return request({
url: `/ems/statsReport/getElectricData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`, url: `/ems/statsReport/getElectricData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`,
method: 'get' method: 'get'
}) })
} }
//获取pcs列表 //获取pcs列表
export function getPcsNameList(siteId) { export function getPcsNameList(siteId) {
return request({ return request({
url: `/ems/siteMonitor/getPcsNameList?siteId=${siteId}`, url: `/ems/siteMonitor/getPcsNameList?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//pcs曲线 //pcs曲线
export function getPCSData({siteId,startTime,endTime,dataType}) { export function getPCSData({siteId, startTime, endTime, dataType}) {
return request({ return request({
url: `/ems/statsReport/getPCSData?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}&dataType=${dataType}`, url: `/ems/statsReport/getPCSData?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}&dataType=${dataType}`,
method: 'get' method: 'get'
}) })
} }
//电池堆曲线 //电池堆曲线
export function getStackData({siteId,startTime,endTime,dataType}) { export function getStackData({siteId, startTime, endTime, dataType}) {
return request({ return request({
url: `/ems/statsReport/getStackData?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}&dataType=${dataType}`, url: `/ems/statsReport/getStackData?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}&dataType=${dataType}`,
method: 'get' method: 'get'
}) })
} }
//电池温度 //电池温度
export function getClusterData({siteId,stackId,clusterId,dateTime,pageNum,pageSize}) { export function getClusterData({siteId, stackId, clusterId, dateTime, pageNum, pageSize}) {
return request({ return request({
url: `/ems/statsReport/getClusterData?siteId=${siteId}&stackId=${stackId}&clusterId=${clusterId}&dateTime=${dateTime}&pageNum=${pageNum}&pageSize=${pageSize}`, url: `/ems/statsReport/getClusterData?siteId=${siteId}&stackId=${stackId}&clusterId=${clusterId}&dateTime=${dateTime}&pageNum=${pageNum}&pageSize=${pageSize}`,
method: 'get' method: 'get'
}) })
} }
// 实时运行 // 实时运行
//储能 //储能
export function storagePower(siteId) { export function storagePower(siteId, startTime, endTime) {
return request({ return request({
url: `/ems/siteMonitor/runningGraph/storagePower?siteId=${siteId}`, url: `/ems/siteMonitor/runningGraph/storagePower?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}`,
method: 'get' method: 'get'
}) })
} }
//poc温度 //poc温度
export function pcsMaxTemp(siteId) { export function pcsMaxTemp(siteId, startTime, endTime) {
return request({ return request({
url: `/ems/siteMonitor/runningGraph/pcsMaxTemp?siteId=${siteId}`, url: `/ems/siteMonitor/runningGraph/pcsMaxTemp?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}`,
method: 'get' method: 'get'
}) })
} }
// 电池平均soc // 电池平均soc
export function batteryAveSoc(siteId) { export function batteryAveSoc(siteId, startTime, endTime) {
return request({ return request({
url: `/ems/siteMonitor/runningGraph/batteryAveSoc?siteId=${siteId}`, url: `/ems/siteMonitor/runningGraph/batteryAveSoc?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}`,
method: 'get' method: 'get'
}) })
} }
// 电池平均温度 // 电池平均温度
export function batteryAveTemp(siteId) { export function batteryAveTemp(siteId, startTime, endTime) {
return request({ return request({
url: `/ems/siteMonitor/runningGraph/batteryAveTemp?siteId=${siteId}`, url: `/ems/siteMonitor/runningGraph/batteryAveTemp?siteId=${siteId}&startDate=${startTime}&endDate=${endTime}`,
method: 'get' method: 'get'
}) })
} }
// 功率曲线 // 功率曲线
export function getPowerData({siteId,startDate,endDate}) { export function getPowerData({siteId, startDate, endDate}) {
return request({ return request({
url: `/ems/statsReport/getPowerData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`, url: `/ems/statsReport/getPowerData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`,
method: 'get' method: 'get'
}) })
} }
//电表列表 //电表列表
export function getLoadNameList(siteId) { export function getLoadNameList(siteId) {
return request({ return request({
url: `/ems/statsReport/getLoadNameList?siteId=${siteId}`, url: `/ems/statsReport/getLoadNameList?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
}
// 电表报表
export function getAmmeterData({siteId,startTime,endTime, pageSize, pageNum}) {
return request({
url: `/ems/statsReport/getAmmeterData?siteId=${siteId}&startTime=${startTime}&endTime=${endTime}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get'
})
} }
// 电表报表
export function getAmmeterData({siteId, startTime, endTime, pageSize, pageNum}) {
return request({
url: `/ems/statsReport/getAmmeterData?siteId=${siteId}&startTime=${startTime}&endTime=${endTime}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get'
})
}
// 电价报表
export function getAmmeterRevenueData(data) {
return request({
url: `/ems/statsReport/getAmmeterRevenueData`,
method: 'get',
params: data
})
}
//策略列表 //策略列表
export function strategyRunningList(siteId) { export function strategyRunningList(siteId) {
return request({ return request({
url: `/system/strategyRunning/list?siteId=${siteId}`, url: `/system/strategyRunning/list?siteId=${siteId}`,
method: 'get' method: 'get'
}) })
} }
//停止策略 //停止策略
export function stopStrategyRunning(id) { export function stopStrategyRunning(id) {
return request({ return request({
url: `/system/strategyRunning/stop?id=${id}`, url: `/system/strategyRunning/stop?id=${id}`,
method: 'get' method: 'get'
}) })
} }
// 获取所有主策略 // 获取所有主策略
export function getMainStrategyList() { export function getMainStrategyList() {
return request({ return request({
url: `/system/strategyRunning/getMainStrategyList`, url: `/system/strategyRunning/getMainStrategyList`,
method: 'get' method: 'get'
}) })
} }
//获取所有辅助策略 //获取所有辅助策略
export function getAuxStrategyList() { export function getAuxStrategyList() {
return request({ return request({
url: `/system/strategyRunning/getAuxStrategyList`, url: `/system/strategyRunning/getAuxStrategyList`,
method: 'get' method: 'get'
}) })
} }
//配置策略 //配置策略
export function configStrategy(data) { export function configStrategy(data) {
return request({ return request({
url: `/system/strategyRunning/configStrategy`, url: `/system/strategyRunning/configStrategy`,
method: 'post', method: 'post',
data data
}) })
} }
//http://localhost:8089/strategy/temp/getTempNameList?strategyId=1&siteId=021_FXX_01 //http://localhost:8089/strategy/temp/getTempNameList?strategyId=1&siteId=021_FXX_01
//获取策略下的所有模板列表 //获取策略下的所有模板列表
export function getTempNameList({siteId,strategyId}) { export function getTempNameList({siteId, strategyId}) {
return request({ return request({
url: `/strategy/temp/getTempNameList?siteId=${siteId}&strategyId=${strategyId}`, url: `/strategy/temp/getTempNameList?siteId=${siteId}&strategyId=${strategyId}`,
method: 'get', method: 'get',
}) })
} }
//获取模板详情 //获取模板详情
///strategy/temp/list?templateId=1 ///strategy/temp/list?templateId=1
export function getStrategyTempDetail(templateId) { export function getStrategyTempDetail(templateId) {
return request({ return request({
url: `/strategy/temp/list?templateId=${templateId}`, url: `/strategy/temp/list?templateId=${templateId}`,
method: 'get', method: 'get',
}) })
} }
//新增模板 //新增模板
export function addStrategyTemp(data) { export function addStrategyTemp(data) {
return request({ return request({
url: `/strategy/temp`, url: `/strategy/temp`,
method: 'post', method: 'post',
data data
}) })
} }
export function editStrategyTemp(data) { export function editStrategyTemp(data) {
return request({ return request({
url: `/strategy/temp`, url: `/strategy/temp`,
method: 'put', method: 'put',
data data
}) })
} }
//http://localhost:8089/strategy/temp/{id} //http://localhost:8089/strategy/temp/{id}
export function deleteStrategyTemp(id) { export function deleteStrategyTemp(id) {
return request({ return request({
url: `/strategy/temp/`+id, url: `/strategy/temp/` + id,
method: 'delete', method: 'delete',
}) })
} }
//http://localhost:8089/strategy/timeConfig/list?strategyId=1&siteId=021_FXX_01 //http://localhost:8089/strategy/timeConfig/list?strategyId=1&siteId=021_FXX_01
export function timeConfigList({siteId,strategyId}) { export function timeConfigList({siteId, strategyId}) {
return request({ return request({
url: `/strategy/timeConfig/list?siteId=${siteId}&strategyId=${strategyId}`, url: `/strategy/timeConfig/list?siteId=${siteId}&strategyId=${strategyId}`,
method: 'get', method: 'get',
}) })
} }
//保存时间配置 //保存时间配置
// http://localhost:8089/strategy/timeConfig // http://localhost:8089/strategy/timeConfig
export function setTimeConfigList(data) { export function setTimeConfigList(data) {
return request({ return request({
url: `/strategy/timeConfig`, url: `/strategy/timeConfig`,
method: 'post', method: 'post',
data data
}) })
} }
// 策略曲线图 // 策略曲线图
//http://localhost:8089/strategy/curve/curveList?strategyId=1&siteId=021_FXX_01 //http://localhost:8089/strategy/curve/curveList?strategyId=1&siteId=021_FXX_01
export function curveList({siteId,strategyId}) { export function curveList({siteId, strategyId}) {
return request({ return request({
url: `/strategy/curve/curveList?siteId=${siteId}&strategyId=${strategyId}`, url: `/strategy/curve/curveList?siteId=${siteId}&strategyId=${strategyId}`,
method: 'get', method: 'get',
}) })
} }
//单站监控 首页 当日功率曲线 //单站监控 首页 当日功率曲线
export function getPointData({siteId,startDate,endDate}) { export function getPointData({siteId, startDate, endDate}) {
return request({ return request({
url: `/ems/siteMonitor/getPointData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`, url: `/ems/siteMonitor/getPointData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`,
method: 'get', method: 'get',
}) })
} }

View File

@ -1,183 +1,203 @@
import request from '@/utils/request' import request from '@/utils/request'
// 站点列表 // 站点列表
export function getSiteInfoList({siteName,startTime, endTime,pageSize,pageNum}) { export function getSiteInfoList({siteName, startTime, endTime, pageSize, pageNum}) {
return request({ return request({
url: `/ems/siteConfig/getSiteInfoList?siteName=${siteName}&startTime=${startTime}&endTime=${endTime}&pageSize=${pageSize}&pageNum=${pageNum}`, url: `/ems/siteConfig/getSiteInfoList?siteName=${siteName}&startTime=${startTime}&endTime=${endTime}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get' method: 'get'
}) })
} }
// 设备列表 // 设备列表
export function getDeviceInfoList({siteId,pageSize,pageNum}) { export function getDeviceInfoList(data) {
return request({ return request({
url: `/ems/siteConfig/getDeviceInfoList?siteId=${siteId}&pageSize=${pageSize}&pageNum=${pageNum}`, url: `/ems/siteConfig/getDeviceInfoList`,
method: 'get' method: 'get',
}) params: data
})
} }
// 设备详情 // 设备详情
export function getDeviceDetailInfo(id) { export function getDeviceDetailInfo(id) {
return request({ return request({
url: `/ems/siteConfig/getDeviceDetailInfo?id=${id}`, url: `/ems/siteConfig/getDeviceDetailInfo?id=${id}`,
method: 'get' method: 'get'
}) })
} }
// 获取所有设备类别 // 获取所有设备类别
export function getDeviceCategory() { export function getDeviceCategory() {
return request({ return request({
url: `/ems/siteConfig/getDeviceCategory`, url: `/ems/siteConfig/getDeviceCategory`,
method: 'get' method: 'get'
}) })
} }
// 新增设备 // 新增设备
export function addDevice(data) { export function addDevice(data) {
return request({ return request({
url: `/ems/siteConfig/addDevice`, url: `/ems/siteConfig/addDevice`,
method: 'post', method: 'post',
data data
}) })
} }
// 编辑设备 // 编辑设备
export function updateDevice(data) { export function updateDevice(data) {
return request({ return request({
url: `/ems/siteConfig/updateDevice`, url: `/ems/siteConfig/updateDevice`,
method: 'post', method: 'post',
data data
}) })
} }
// 删除设备 // 删除设备
export function deleteService(id) { export function deleteService(id) {
return request({ return request({
url: `/ems/siteConfig/deleteService/`+id, url: `/ems/siteConfig/deleteService/` + id,
method: 'delete', method: 'delete',
}) })
} }
//pcs开、关机
export function updateDeviceStatus(data) {
return request({
url: `/ems/siteConfig/updateDeviceStatus`,
method: 'post',
data
})
}
// 获取上级设备id列表 // 获取上级设备id列表
export function getParentDeviceId({siteId,deviceCategory}) { export function getParentDeviceId({siteId, deviceCategory}) {
return request({ return request({
url: `/ems/siteConfig/getParentDeviceId?siteId=${siteId}&deviceCategory=${deviceCategory}`, url: `/ems/siteConfig/getParentDeviceId?siteId=${siteId}&deviceCategory=${deviceCategory}`,
method: 'get', method: 'get',
}) })
} }
//获取所有设备 //获取所有设备
export function getDeviceList(siteId) { export function getDeviceList(siteId) {
return request({ return request({
url: `/ems/siteConfig/getDeviceList?siteId=${siteId}`, url: `/ems/siteConfig/getDeviceList?siteId=${siteId}`,
method: 'get', method: 'get',
}) })
} }
//获取设备点位table //获取设备点位table
export function getDevicePointList({siteId,deviceId,deviceCategory,parentId,pageNum,pageSize,dataPointName='',sortMethod,sortData,dataPoint,lower,upper,ipAddress,ipPort}) { export function getDevicePointList(data) {
return request({ return request({
url: `/ems/siteConfig/getDevicePointList?siteId=${siteId}&deviceId=${deviceId}&pageNum=${pageNum}&pageSize=${pageSize}&deviceCategory=${deviceCategory}&dataPointName=${dataPointName}&parentId=${parentId}&dataPoint=${dataPoint}&lower=${lower}&upper=${upper}&pageNum=${pageNum}&sortMethod=${sortMethod}&sortData=${sortData}&ipAddress=${ipAddress}&ipPort=${ipPort}`, url: `/ems/siteConfig/getDevicePointList`,
method: 'get', method: 'get',
}) params: data
})
} }
//获取设备类型下面的所有设备列表 //获取设备类型下面的所有设备列表
export function getDeviceListBySiteAndCategory({siteId, deviceCategory}) { export function getDeviceListBySiteAndCategory({siteId, deviceCategory}) {
return request({ return request({
url: `/ems/siteConfig/getDeviceListBySiteAndCategory?siteId=${siteId}&deviceCategory=${deviceCategory}`, url: `/ems/siteConfig/getDeviceListBySiteAndCategory?siteId=${siteId}&deviceCategory=${deviceCategory}`,
method: 'get', method: 'get',
}) })
} }
//新增设备保护 //新增设备保护
export function addProtectPlan(data) { export function addProtectPlan(data) {
return request({ return request({
url: `/ems/protectPlan`, url: `/ems/protectPlan`,
method: 'post', method: 'post',
data data
}) })
} }
//修改设备保护 //修改设备保护
export function updateProtectPlan(data) { export function updateProtectPlan(data) {
return request({ return request({
url: `/ems/protectPlan`, url: `/ems/protectPlan`,
method: 'put', method: 'put',
data data
}) })
} }
//删除设备保护 //删除设备保护
export function deleteProtectPlan(id) { export function deleteProtectPlan(id) {
return request({ return request({
url: `/ems/protectPlan/${id}`, url: `/ems/protectPlan/${id}`,
method: 'delete', method: 'delete',
}) })
} }
//设备保护详情 //设备保护详情
export function getProtectPlan(id) { export function getProtectPlan(id) {
return request({ return request({
url: `/ems/protectPlan/${id}`, url: `/ems/protectPlan/${id}`,
method: 'get', method: 'get',
}) })
} }
//设备保护详情列表 //设备保护详情列表
//http://localhost:8089/ems/protectPlan/list?pageSize=10&pageNum=1&faultName=总压&siteId=021_DDS_01 //http://localhost:8089/ems/protectPlan/list?pageSize=10&pageNum=1&faultName=总压&siteId=021_DDS_01
export function protectPlanList({siteId, faultName,pageSize,pageNum}) { export function protectPlanList({siteId, faultName, pageSize, pageNum}) {
return request({ return request({
url: `/ems/protectPlan/list?siteId=${siteId}&faultName=${faultName}&pageSize=${pageSize}&pageNum=${pageNum}`, url: `/ems/protectPlan/list?siteId=${siteId}&faultName=${faultName}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get', method: 'get',
}) })
} }
// 点位导出 // 点位导出
export function exportPointList(data) { export function exportPointList(data) {
return request({ return request({
url: `/ems/pointMatch/export`, url: `/ems/pointMatch/export`,
method: 'post', method: 'post',
data data
}) })
}
// 点位导入
export function importPointList(data) {
return request({
url: `/ems/pointMatch/importData`,
method: 'post',
data
})
} }
// 点位导入
export function importPointList(data) {
return request({
url: `/ems/pointMatch/importData`,
method: 'post',
data
})
}
//mqtt //mqtt
export function getMqttList({pageSize,pageNum,mqttTopic,topicName}) { export function getMqttList({pageSize, pageNum, mqttTopic, topicName, siteId}) {
return request({ return request({
url: `/ems/mqttConfig/list?pageSize=${pageSize}&pageNum=${pageNum}&mqttTopic=${mqttTopic}&topicName=${topicName}`, url: `/ems/mqttConfig/list?pageSize=${pageSize}&pageNum=${pageNum}&mqttTopic=${mqttTopic}&topicName=${topicName}&siteId=${siteId}`,
method: 'get', method: 'get',
}) })
} }
export function getMqttDetail(id) { export function getMqttDetail(id) {
return request({ return request({
url: `/ems/mqttConfig/${id}`, url: `/ems/mqttConfig/${id}`,
method: 'get', method: 'get',
}) })
} }
export function addMqtt(data) { export function addMqtt(data) {
return request({ return request({
url: `/ems/mqttConfig`, url: `/ems/mqttConfig`,
method: 'post', method: 'post',
data data
}) })
} }
export function editMqtt(data) { export function editMqtt(data) {
return request({ return request({
url: `/ems/mqttConfig`, url: `/ems/mqttConfig`,
method: 'put', method: 'put',
data data
}) })
} }
export function deleteMqtt(id) { export function deleteMqtt(id) {
return request({ return request({
url: `/ems/mqttConfig/${id}`, url: `/ems/mqttConfig/${id}`,
method: 'delete', method: 'delete',
}) })
} }

Binary file not shown.

After

Width:  |  Height:  |  Size: 3.2 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 2.1 MiB

Binary file not shown.

After

Width:  |  Height:  |  Size: 11 MiB

View File

@ -9,60 +9,69 @@
padding: 24px; padding: 24px;
font-size: 12px; font-size: 12px;
} }
//除去顶部信息(如搜索栏、站点基本信息等)外的 白色背景内容区域 //除去顶部信息(如搜索栏、站点基本信息等)外的 白色背景内容区域
.ems-content-container{ .ems-content-container {
background-color: #ffffff; background-color: #ffffff;
margin-top: 24px; margin-top: 24px;
} }
//需要设置内padding的白色背景区域 //需要设置内padding的白色背景区域
.ems-content-container-padding{ .ems-content-container-padding {
padding: 24px; padding: 24px;
} }
//card通用样式 标题、body //card通用样式 标题、body
.common-card-container{ .common-card-container {
.el-card__header{ .el-card__header {
padding:14px; padding: 14px;
border-bottom: none; border-bottom: none;
font-size: 12px; font-size: 12px;
background: #F1F5FB ; background: #F1F5FB;
position: relative; position: relative;
.card-title{
font-weight: 500;
color:#333333;
}
.el-button--text{
color: #666666;
}
.card-title {
font-weight: 500;
color: #333333;
} }
}
.common-card-container-body-no-padding{ .el-button--text {
.el-card__body{ color: #666666;
padding:0;
} }
}
} }
.common-card-container-body-no-padding {
.el-card__body {
padding: 0;
}
}
.common-card-container-no-title-bg { .common-card-container-no-title-bg {
.el-card__header{
background-color: transparent;
}
}
//单站监控 设备监控card公共样式
.sbjk-card-container{
.el-card__header { .el-card__header {
background-color: transparent; background-color: transparent;
padding: 10px 14px; }
}
//单站监控 设备监控card公共样式
.sbjk-card-container {
.el-card__header {
background-color: transparent;
padding: 5px 14px;
color: #ffffff; color: #ffffff;
position: relative; position: relative;
border-radius: 5px 5px 0 0; border-radius: 5px 5px 0 0;
.large-title{
font-size: 20px; .large-title {
font-size: 18px;
font-weight: 500; font-weight: 500;
line-height: 40px; line-height: 40px;
padding: 0 50px 0 11px; padding: 0 50px 0 11px;
display: inline-block; display: inline-block;
vertical-align: middle; vertical-align: middle;
} }
.info { .info {
display: inline-block; display: inline-block;
vertical-align: middle; vertical-align: middle;
@ -70,33 +79,57 @@
font-size: 12px; font-size: 12px;
line-height: 20px; line-height: 20px;
} }
.el-button--text{
.el-button--text {
color: #666666; color: #666666;
} }
.alarm{
.alarm {
position: absolute; position: absolute;
right: 25px; right: 25px;
top: 50%; top: 50%;
transform: translateY(-50%); transform: translateY(-50%);
.alarm-icon {
font-size: 22px;
color: #fff;
display: block;
cursor: pointer;
}
} }
} }
//红色背景颜色标题 //红色背景颜色标题
&.warning-card-container{ &.warning-card-container {
.el-card__header { .el-card__header {
background-color: #fc6b69; background-color: #b64040; //#fc6b69;
}
.work-status {
color: #b64040 !important;;
} }
} }
//绿色背景颜色标题 //绿色背景颜色标题
&.running-card-container { &.running-card-container {
.el-card__header { .el-card__header {
background-color: #05aea3; background-color: #40b6a5; //#05aea3;
}
.work-status {
color: #40b6a5 !important;
} }
} }
//灰色背景颜色标题 //灰色背景颜色标题
&.timing-card-container { &.timing-card-container {
.el-card__header { .el-card__header {
background-color: #666666; background-color: #666666;
} }
.work-status {
color: #666666 !important;;
}
} }
} }
@ -105,11 +138,13 @@
&.common-card-container .el-card__header { &.common-card-container .el-card__header {
padding-top: 0; padding-top: 0;
padding-bottom: 0; padding-bottom: 0;
.time-range-header { .time-range-header {
height: 40px; height: 40px;
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;
align-items: center; align-items: center;
.card-title { .card-title {
line-height: 40px; line-height: 40px;
} }
@ -118,58 +153,100 @@
} }
//描述样式 PCS、BMS总览、BMS电池簇页面公共样式 //描述样式 PCS、BMS总览、BMS电池簇页面公共样式
.descriptions-main{ .descriptions-main {
padding:24px; padding: 24px;
position: relative; position: relative;
&.descriptions-main-bg-color{
background-color:#f1f5fc ; &.descriptions-main-bg-color {
.el-descriptions__body{ background-color: #f1f5fc;
background-color:#f1f5fc ;
.el-descriptions__body {
background-color: #f1f5fc;
} }
} }
.el-descriptions-item__cell[colspan='1']{
width:25% .el-descriptions-item__cell[colspan='1'] {
width: 25%
} }
.el-descriptions__body .el-descriptions__table{
.descriptions-direction{ .el-descriptions__body .el-descriptions__table {
.descriptions-direction {
line-height: 19px; line-height: 19px;
color: #666666; color: #666666;
font-size: 16px; font-size: 16px;
font-weight: 500; font-weight: 500;
} }
.descriptions-label{
.descriptions-label {
line-height: 14px; line-height: 14px;
color: #666666; color: #666666;
font-size: 12px; font-size: 12px;
} }
.danger{
color:#FC6B69; .danger {
color: #FC6B69;
} }
.save{
color:#09ADA3; .save {
color: #09ADA3;
} }
.keep{
color:#3C81FF; .keep {
color: #3C81FF;
} }
} }
} }
//电表、液冷公共样式
.device-info-row {
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
border-left: 1px solid #eee;
border-top: 1px solid #eee;
.device-info-col {
padding: 10px 0;
text-align: center;
font-size: 12px;
color: #666666;
line-height: 14px;
border-bottom: 1px solid #eee;
border-right: 1px solid #eee;
.left {
}
.right {
display: block;
font-weight: 500;
font-size: 16px;
line-height: 18px;
margin-top: 10px;
}
}
}
//公共表格样式 //公共表格样式
.common-table.el-table{ .el-table {
color:#333333; font-size: 13px;
}
.common-table.el-table {
color: #333333;
.el-table__header-wrapper th, .el-table__fixed-header-wrapper th { .el-table__header-wrapper th, .el-table__fixed-header-wrapper th {
background: #f1f5fc; background: #f1f5fc;
border-bottom: none; border-bottom: none;
.table-head { .table-head {
color: #515a6e; color: #515a6e;
} }
} }
.warning-status{
color:#FC6B69; .warning-status {
color: #FC6B69;
&.circle::before { &.circle::before {
content: ""; content: "";
display: inline-block; display: inline-block;
@ -183,25 +260,30 @@
} }
//二、三级菜单栏样式 //二、三级菜单栏样式
.ems-second-menu{ .ems-second-menu {
width:fit-content; width: fit-content;
.el-menu-item{
.el-menu-item {
line-height: 40px; line-height: 40px;
height: 40px; height: 40px;
padding:0; padding: 0;
} }
&.el-menu--horizontal > .el-menu-item.is-active,&.el-menu--horizontal > .el-menu-item{
border-bottom:none!important; &.el-menu--horizontal > .el-menu-item.is-active, &.el-menu--horizontal > .el-menu-item {
border-bottom: none !important;
} }
.el-menu-item.is-active{
background: #0366c1!important; .el-menu-item.is-active {
background: #0366c1 !important;
} }
} }
.ems-third-menu-container{
.ems-third-menu-container {
position: relative; position: relative;
padding-left: 160px; padding-left: 160px;
background-color: #ffffff; background-color: #ffffff;
.ems-third-menu{
.ems-third-menu {
border-right: none; border-right: none;
box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1); box-shadow: 0 2px 12px 0 rgba(0, 0, 0, 0.1);
height: fit-content; height: fit-content;
@ -215,22 +297,23 @@
width: 125px; width: 125px;
text-align: center; text-align: center;
} }
.el-menu-item:hover{
background: #67b1ff!important; .el-menu-item:hover {
color:#ffffff!important; background: #67b1ff !important;
color: #ffffff !important;
} }
.el-menu-item.is-active{
background: #409eff!important; .el-menu-item.is-active {
background: #409eff !important;
} }
} }
} }
//按钮栏 选中样式 //按钮栏 选中样式
.ems-btns-group{ .ems-btns-group {
.activeBtn{ .activeBtn {
background-color: #0366c1; background-color: #0366c1;
border-color: #0366c1; border-color: #0366c1;
color: #ffffff; color: #ffffff;
@ -239,11 +322,12 @@
} }
//搜索栏样式 //搜索栏样式
.select-container.el-form--inline .el-form-item{ .select-container.el-form--inline .el-form-item {
margin-right: 15px; margin-right: 15px;
} }
//红色背景颜色按钮 //红色背景颜色按钮
.alarm-btn,.alarm-btn:hover, .alarm-btn:focus{ .alarm-btn, .alarm-btn:hover, .alarm-btn:focus {
background-color: #FC6B69; background-color: #FC6B69;
border-color: #FC6B69; border-color: #FC6B69;
} }

View File

@ -0,0 +1,137 @@
<template>
<el-dialog
:fullscreen="true"
:append-to-body="true"
:visible.sync="show"
:show-close="false"
top="0"
custom-class="big-data-dialog"
>
<div class="swiper-container">
<div class="swiper-icon left-icon" v-show="imgIndex > 0">
<i class="el-icon-d-arrow-left icon" @click="toLeft"></i>
</div>
<div v-show="showRightIcon" class="swiper-icon right-icon">
<i class="el-icon-d-arrow-right icon" @click="toRight"></i>
</div>
<div
class="img-container"
:style="{ transform: 'translateX(' + imgIndex * -100 + 'vw)' }"
>
<img
v-for="index in maxImgNumber"
:key="'swiperImg' + index"
:src="require(`@/assets/images/ems/bigData-${index}.png`)"
alt=""
/>
</div>
</div>
<div class="close-btn" @click="show = false">
<i class="el-icon-close"></i>
</div>
</el-dialog>
</template>
<style lang="scss" scoped>
.close-btn {
position: absolute;
right: 10px;
top: 10px;
font-size: 23px;
line-height: 20px;
color: rgba(217, 242, 255, 1);
cursor: pointer;
}
.swiper-container {
height: 100%;
width: 100%;
overflow: hidden;
position: relative;
.swiper-icon {
color: rgba(217, 242, 255, 1);
position: absolute;
top: 50%;
transform: translateY(-50%);
z-index: 20;
cursor: pointer;
font-size: 30px;
padding: 20px;
background: transparent;
&.left-icon {
left: 20px;
}
&.right-icon {
right: 20px;
}
&:hover {
.icon {
opacity: 1;
}
}
.icon {
transition: all 0.6s;
opacity: 0;
}
}
.img-container {
height: 100%;
transition: all 1s;
display: flex;
flex-direction: row;
z-index: 0;
img {
width: 100vw;
height: 100vh;
display: block;
margin: 0;
flex-shrink: 0;
}
}
}
</style>
<style lang="scss">
.big-data-dialog {
.el-dialog__header {
display: none;
}
.el-dialog__body {
padding: 0;
margin: 0;
position: relative;
}
}
</style>
<script>
export default {
data() {
return {
show: false,
imgIndex: 0,
maxImgNumber: 3,
};
},
computed: {
showRightIcon() {
return this.imgIndex < this.maxImgNumber - 1;
},
},
watch: {
show: {
handler(newValue) {
if (!newValue) this.imgIndex = 0;
},
immediate: true,
},
},
methods: {
toLeft() {
if (this.imgIndex === 0) return;
this.imgIndex -= 1;
},
toRight() {
if (this.imgIndex >= this.maxImgNumber - 1) return;
this.imgIndex += 1;
},
},
};
</script>

View File

@ -1,36 +1,68 @@
<template> <template>
<div class="time-range"> <div class="time-range">
<el-date-picker <el-date-picker
v-model="dateRange" v-model="dateRange"
type="daterange" :class="miniTimePicker ? 'mini-date-picker' : ''"
range-separator="" type="daterange"
start-placeholder="开始时间" range-separator=""
value-format="yyyy-MM-dd" start-placeholder="开始时间"
:clearable="false" value-format="yyyy-MM-dd"
:picker-options="pickerOptions" :clearable="false"
end-placeholder="结束时间"> :picker-options="pickerOptions"
</el-date-picker> end-placeholder="结束时间">
<el-button size="mini" style="margin-left: 10px;" :loading="loading" @click="reset">重置</el-button> </el-date-picker>
<el-button type="primary" size="mini" :loading="loading" @click="search">搜索</el-button> <template v-if="!showIcon">
<el-button type="primary" size="mini" :loading="loading" @click="timeLine('before')">上一时段</el-button> <el-button size="mini" style="margin-left: 10px;" :loading="loading" @click="reset">重置</el-button>
<el-button type="primary" size="mini" :loading="loading" @click="timeLine('next')" :disabled="disabledNextBtn">下一时段</el-button> <el-button type="primary" size="mini" :loading="loading" @click="search">搜索</el-button>
</div> <el-button type="primary" size="mini" :loading="loading" @click="timeLine('before')">上一时段</el-button>
<el-button type="primary" size="mini" :loading="loading" @click="timeLine('next')" :disabled="disabledNextBtn">
下一时段
</el-button>
</template>
<template v-else>
<el-button class="btn-icon" icon="el-icon-refresh-right" circle size="mini" style="margin-left: 8px;"
:loading="loading"
@click="reset"></el-button>
<el-button class="btn-icon" type="primary" size="mini" icon="el-icon-search" circle :loading="loading"
@click="search"></el-button>
<el-button class="btn-icon" type="primary" size="mini" icon="el-icon-d-arrow-left" circle :loading="loading"
@click="timeLine('before')"></el-button>
<el-button class="btn-icon" type="primary" size="mini" icon="el-icon-d-arrow-right" circle :loading="loading"
@click="timeLine('next')"
:disabled="disabledNextBtn"></el-button>
</template>
</div>
</template> </template>
<script> <script>
import {formatDate} from '@/filters/ems' import {formatDate} from '@/filters/ems'
export default { export default {
computed:{ props: {
disabledNextBtn(){ showIcon: {
type: Boolean,
required: false,
default: false
},
miniTimePicker: {
type: Boolean,
required: false,
default: false
}
},
computed: {
disabledNextBtn() {
return new Date(this.dateRange[1]) >= new Date(this.defaultDateRange[1]) return new Date(this.dateRange[1]) >= new Date(this.defaultDateRange[1])
} }
}, },
data() { data() {
return { return {
loading:false, loading: false,
dateRange:[], dateRange: [],
defaultDateRange:[], defaultDateRange: [],
pickerOptions:{ pickerOptions: {
disabledDate(time) { disabledDate(time) {
return time.getTime() > Date.now(); return time.getTime() > Date.now();
}, },
@ -38,64 +70,81 @@ export default {
} }
}, },
methods: { methods: {
init(today=false){ init(today = false) {
const now = new Date(),formatNow = formatDate(now); const now = new Date(), formatNow = formatDate(now);
const weekAgo = formatDate(today ? new Date(now.getTime()) : new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000)) const weekAgo = formatDate(today ? new Date(now.getTime()) : new Date(now.getTime() - 7 * 24 * 60 * 60 * 1000))
this.dateRange = [weekAgo, formatNow]; this.dateRange = [weekAgo, formatNow];
this.defaultDateRange=[weekAgo, formatNow]; this.defaultDateRange = [weekAgo, formatNow];
this.$emit('updateDate',this.dateRange) this.$emit('updateDate', this.dateRange)
}, },
showBtnLoading(status){ showBtnLoading(status) {
this.loading = status this.loading = status
}, },
resetDate(){ resetDate() {
this.dateRange = this.defaultDateRange this.dateRange = this.defaultDateRange
}, },
//重置 设置时间范围为初始化时间段 //重置 设置时间范围为初始化时间段
reset(){ reset() {
this.resetDate() this.resetDate()
this.$emit('reset') this.$emit('reset')
this.$emit('updateDate',this.dateRange) this.$emit('updateDate', this.dateRange)
}, },
// 搜索 // 搜索
search(){ search() {
this.$emit('updateDate',this.dateRange) this.$emit('updateDate', this.dateRange)
}, },
timeLine(type){ timeLine(type) {
if(!this.dateRange || !this.dateRange[0] || !this.dateRange[1]) return if (!this.dateRange || !this.dateRange[0] || !this.dateRange[1]) return
const nowStartTimes = new Date(this.dateRange[0]).getTime(),nowEndTimes = new Date(this.dateRange[1]).getTime(),maxTime = new Date(this.defaultDateRange[1]).getTime() const nowStartTimes = new Date(this.dateRange[0]).getTime(), nowEndTimes = new Date(this.dateRange[1]).getTime(),
maxTime = new Date(this.defaultDateRange[1]).getTime()
const nowDis = nowEndTimes - nowStartTimes//用户当前选择时间差 可能=0 const nowDis = nowEndTimes - nowStartTimes//用户当前选择时间差 可能=0
//baseTime,maxTime 毫秒数 //baseTime,maxTime 毫秒数
const baseDis = 24 * 60 * 60 * 1000 const baseDis = 24 * 60 * 60 * 1000
const calcDis = nowDis === 0 ? baseDis : nowDis const calcDis = nowDis === 0 ? baseDis : nowDis
let start = type === 'before' ? nowStartTimes - calcDis : nowStartTimes + calcDis let start = type === 'before' ? nowStartTimes - calcDis : nowStartTimes + calcDis
if(start>maxTime) start=maxTime if (start > maxTime) start = maxTime
let end = type === 'before' ? nowEndTimes - calcDis : nowEndTimes + calcDis let end = type === 'before' ? nowEndTimes - calcDis : nowEndTimes + calcDis
if(end>maxTime) end=maxTime if (end > maxTime) end = maxTime
this.dateRange = [formatDate(start),formatDate(end)] this.dateRange = [formatDate(start), formatDate(end)]
this.$emit('updateDate',this.dateRange) this.$emit('updateDate', this.dateRange)
}, },
} }
} }
</script> </script>
<style lang="scss" scoped> <style lang="scss" scoped>
.time-range{ .time-range {
display: flex; display: flex;
::v-deep {
.el-range-editor--medium .el-range__icon, .el-range-editor--medium .el-range__close-icon{ ::v-deep {
line-height: 22px;
} .el-range-editor--medium .el-range__icon, .el-range-editor--medium .el-range__close-icon {
.el-range-editor--medium.el-input__inner{ line-height: 22px;
height: 30px;
}
.el-range-editor--medium .el-range-separator{
line-height: 24px;
}
.el-button--mini{
padding:3px 10px;
}
} }
.el-range-editor--medium.el-input__inner {
height: 30px;
}
.el-range-editor--medium .el-range-separator {
line-height: 22px;
}
.el-button--mini {
padding: 3px 10px;
}
// 展示icon的小组件
.btn-icon.el-button--mini {
padding: 3px 8px;
margin-left: 6px;
}
//小宽度时间选择框
.mini-date-picker {
width: 250px !important;
}
}
} }
</style> </style>

View File

@ -32,13 +32,13 @@ export default {
attr:'installCapacity' attr:'installCapacity'
},{ },{
title:'总充电量(MWh', title:'总充电量(KWh',
num:'', num:'',
color:'#A696FF', color:'#A696FF',
attr:'totalChargedCap' attr:'totalChargedCap'
},{ },{
title:'总放电量(MWh', title:'总放电量(KWh',
num:'', num:'',
color:'#A696FF', color:'#A696FF',
attr:'totalDischargedCap' attr:'totalDischargedCap'

View File

@ -1,19 +1,24 @@
<template> <template>
<div class="navbar"> <div class="navbar">
<hamburger id="hamburger-container" :is-active="sidebar.opened" class="hamburger-container" @toggleClick="toggleSideBar" /> <hamburger id="hamburger-container" :is-active="sidebar.opened" class="hamburger-container"
@toggleClick="toggleSideBar"/>
<breadcrumb v-if="!topNav" id="breadcrumb-container" class="breadcrumb-container" /> <breadcrumb v-if="!topNav" id="breadcrumb-container" class="breadcrumb-container"/>
<top-nav v-if="topNav" id="topmenu-container" class="topmenu-container" /> <top-nav v-if="topNav" id="topmenu-container" class="topmenu-container"/>
<div class="right-menu"> <div class="right-menu">
<template v-if="device!=='mobile'"> <template v-if="device!=='mobile'">
<search id="header-search" class="right-menu-item" /> <div class="big-data-container">
<i class="el-icon-s-marketing big-data-icon" @click.stop="showBigDataImg"></i>
</div>
<search id="header-search" class="right-menu-item"/>
<screenfull id="screenfull" class="right-menu-item hover-effect" /> <screenfull id="screenfull" class="right-menu-item hover-effect"/>
<el-tooltip content="布局大小" effect="dark" placement="bottom"> <el-tooltip content="布局大小" effect="dark" placement="bottom">
<size-select id="size-select" class="right-menu-item hover-effect" /> <size-select id="size-select" class="right-menu-item hover-effect"/>
</el-tooltip> </el-tooltip>
</template> </template>
@ -32,22 +37,23 @@
</el-dropdown-item> </el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</el-dropdown> </el-dropdown>
<div class="right-menu-item hover-effect setting" @click="setLayout" v-if="setting"> <div class="right-menu-item hover-effect setting" @click="setLayout" v-if="setting">
<svg-icon icon-class="more-up" /> <svg-icon icon-class="more-up"/>
</div> </div>
</div> </div>
<BigDataPopup ref="bigDataPopup"/>
</div> </div>
</template> </template>
<script> <script>
import { mapGetters } from 'vuex' import {mapGetters} from 'vuex'
import Breadcrumb from '@/components/Breadcrumb' import Breadcrumb from '@/components/Breadcrumb'
import TopNav from '@/components/TopNav' import TopNav from '@/components/TopNav'
import Hamburger from '@/components/Hamburger' import Hamburger from '@/components/Hamburger'
import Screenfull from '@/components/Screenfull' import Screenfull from '@/components/Screenfull'
import SizeSelect from '@/components/SizeSelect' import SizeSelect from '@/components/SizeSelect'
import Search from '@/components/HeaderSearch' import Search from '@/components/HeaderSearch'
import BigDataPopup from '@/components/BigDataPopup'
export default { export default {
emits: ['setLayout'], emits: ['setLayout'],
@ -57,7 +63,8 @@ export default {
Hamburger, Hamburger,
Screenfull, Screenfull,
SizeSelect, SizeSelect,
Search Search,
BigDataPopup
}, },
computed: { computed: {
...mapGetters([ ...mapGetters([
@ -78,6 +85,9 @@ export default {
} }
}, },
methods: { methods: {
showBigDataImg() {
this.$refs.bigDataPopup.show = true
},
toggleSideBar() { toggleSideBar() {
this.$store.dispatch('app/toggleSideBar') this.$store.dispatch('app/toggleSideBar')
}, },
@ -93,7 +103,8 @@ export default {
this.$store.dispatch('LogOut').then(() => { this.$store.dispatch('LogOut').then(() => {
location.href = '/index' location.href = '/index'
}) })
}).catch(() => {}) }).catch(() => {
})
} }
} }
} }
@ -105,7 +116,7 @@ export default {
overflow: hidden; overflow: hidden;
position: relative; position: relative;
background: #fff; background: #fff;
box-shadow: 0 1px 4px rgba(0,21,41,.08); box-shadow: 0 1px 4px rgba(0, 21, 41, .08);
.hamburger-container { .hamburger-container {
line-height: 46px; line-height: 46px;
@ -113,7 +124,7 @@ export default {
float: left; float: left;
cursor: pointer; cursor: pointer;
transition: background .3s; transition: background .3s;
-webkit-tap-highlight-color:transparent; -webkit-tap-highlight-color: transparent;
&:hover { &:hover {
background: rgba(0, 0, 0, .025) background: rgba(0, 0, 0, .025)
@ -139,6 +150,17 @@ export default {
height: 100%; height: 100%;
line-height: 50px; line-height: 50px;
.big-data-container {
display: inline-block;
padding: 0 8px;
height: 100%;
font-size: 24px;
color: #5a5e66;
vertical-align: text-bottom;
cursor: pointer;
}
&:focus { &:focus {
outline: none; outline: none;
} }
@ -177,7 +199,7 @@ export default {
border-radius: 50%; border-radius: 50%;
} }
.user-nickname{ .user-nickname {
position: relative; position: relative;
bottom: 10px; bottom: 10px;
font-size: 14px; font-size: 14px;

View File

@ -2,154 +2,265 @@ import Layout from "@/layout/index.vue";
//单站监控 //单站监控
// todo 本地设置了 hidden:true,不会显示在侧边栏,需要在系统管理、菜单管理中手动添加菜单后才会展示在侧边栏 // todo 本地设置了 hidden:true,不会显示在侧边栏,需要在系统管理、菜单管理中手动添加菜单后才会展示在侧边栏
export const dzjk=[ export const dzjk = [
{ {
path: '/dzjk', path: '/dzjk',
component: Layout, component: Layout,
redirect: '/dzjk/home', redirect: '/dzjk/home',
meta: { title: '单站监控', icon: 'server',}, meta: {title: '单站监控', icon: 'server',},
alwaysShow: false, alwaysShow: false,
name:'Dzjk', name: 'Dzjk',
children: [ children: [
{ {
path: '/dzjk/home', path: '/dzjk/home',
component: () => import('@/views/ems/dzjk/home/index.vue'), component: () => import('@/views/ems/dzjk/home/index.vue'),
name: 'DzjkHome', name: 'DzjkHome',
meta: { title: '站点首页',breadcrumb: false,activeMenu: '/dzjk/home',activeSecondMenuName:'DzjkHome' } meta: {title: '站点首页', breadcrumb: false, activeMenu: '/dzjk/home', activeSecondMenuName: 'DzjkHome'}
}, },
{ {
path: '/dzjk/zxlt', path: '/dzjk/zxlt',
component: () => import('@/views/ems/dzjk/zxlt/index.vue'), component: () => import('@/views/ems/dzjk/zxlt/index.vue'),
name: 'DzjkZxlt', name: 'DzjkZxlt',
meta: { title: '主线路图',breadcrumb: false,activeMenu: '/dzjk/zxlt',activeSecondMenuName:'DzjkZxlt' } meta: {title: '主线路图', breadcrumb: false, activeMenu: '/dzjk/zxlt', activeSecondMenuName: 'DzjkZxlt'}
}, },
{ {
path: '/dzjk/sbjk', path: '/dzjk/sbjk',
component: () => import('@/views/ems/dzjk/sbjk/index.vue'), component: () => import('@/views/ems/dzjk/sbjk/index.vue'),
name: 'DzjkSbjk', name: 'DzjkSbjk',
alwaysShow: false, alwaysShow: false,
meta: { title: '设备监控',breadcrumb: false,activeMenu: '/dzjk/sbjk'}, meta: {title: '设备监控', breadcrumb: false, activeMenu: '/dzjk/sbjk'},
hidden: false, hidden: false,
redirect: '/dzjk/sbjk/ssyx', redirect: '/dzjk/sbjk/ssyx',
children: [ children: [
{ {
path: 'ssyx', path: 'ssyx',
component: () => import('@/views/ems/dzjk/sbjk/ssyx/index.vue'), component: () => import('@/views/ems/dzjk/sbjk/ssyx/index.vue'),
name: 'DzjkSbjkSsyx', name: 'DzjkSbjkSsyx',
hidden: true, hidden: true,
meta: { title: '实时运行',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'SSYX'}, meta: {
}, title: '实时运行',
{ breadcrumb: false,
path: 'pcs', activeMenu: '/dzjk/sbjk',
component: () => import('@/views/ems/dzjk/sbjk/pcs/index.vue'), activeSecondMenuName: 'DzjkSbjk',
name: 'DzjkSbjkPcs', deviceCategory: 'SSYX'
hidden: true, },
meta: { title: 'PCS',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'PCS'}, },
}, {
{ path: 'ems',
path: 'bmszl', component: () => import('@/views/ems/dzjk/sbjk/ems/index.vue'),
component: () => import('@/views/ems/dzjk/sbjk/bmszl/index.vue'), name: 'DzjkSbjkEms',
name: 'DzjkSbjkBmszl', hidden: true,
hidden: true, meta: {
meta: { title: 'BMS总览',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk', deviceCategory:'STACK'}, title: 'EMS',
}, breadcrumb: false,
{ activeMenu: '/dzjk/sbjk',
path: 'bmsdcc', activeSecondMenuName: 'DzjkSbjk',
component: () => import('@/views/ems/dzjk/sbjk/bmsdcc/index.vue'), deviceCategory: 'EMS'
name: 'DzjkSbjkBmsdcc', },
hidden: true, },
meta: { title: 'BMS电池簇',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'CLUSTER'}, {
}, path: 'pcs',
{ component: () => import('@/views/ems/dzjk/sbjk/pcs/index.vue'),
path: 'dtdc', name: 'DzjkSbjkPcs',
component: () => import('@/views/ems/dzjk/sbjk/dtdc/index.vue'), hidden: true,
name: 'DzjkSbjkDtdc', meta: {
hidden: true, title: 'PCS',
meta: { title: '单体电池',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'BATTERY'}, breadcrumb: false,
}, activeMenu: '/dzjk/sbjk',
{ activeSecondMenuName: 'DzjkSbjk',
path: 'db', deviceCategory: 'PCS'
component: () => import('@/views/ems/dzjk/sbjk/db/index.vue'), },
name: 'DzjkSbjkDb', },
hidden: true, {
meta: { title: '电表',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'AMMETER'}, path: 'bmszl',
}, component: () => import('@/views/ems/dzjk/sbjk/bmszl/index.vue'),
{ name: 'DzjkSbjkBmszl',
path: 'yl', hidden: true,
component: () => import('@/views/ems/dzjk/sbjk/yl/index.vue'), meta: {
name: 'DzjkSbjkYl', title: 'BMS总览',
hidden: true, breadcrumb: false,
meta: { title: '液冷',breadcrumb: false,activeMenu: '/dzjk/sbjk',activeSecondMenuName:'DzjkSbjk',deviceCategory:'COOLING'}, activeMenu: '/dzjk/sbjk',
} activeSecondMenuName: 'DzjkSbjk',
] deviceCategory: 'STACK'
}, },
{ },
path: '/dzjk/gzgj', {
component: () => import('@/views/ems/dzjk/gzgj/index.vue'), path: 'bmsdcc',
name: 'DzjkGzgj', component: () => import('@/views/ems/dzjk/sbjk/bmsdcc/index.vue'),
meta: { title: '故障告警',breadcrumb: false,activeMenu: '/dzjk/gzgj',activeSecondMenuName:'DzjkGzgj' } name: 'DzjkSbjkBmsdcc',
}, hidden: true,
{ meta: {
path: '/dzjk/tjbb', title: 'BMS电池簇',
component: () => import('@/views/ems/dzjk/tjbb/index.vue'), breadcrumb: false,
name: 'DzjkTjbb', activeMenu: '/dzjk/sbjk',
alwaysShow: false, activeSecondMenuName: 'DzjkSbjk',
meta: {title: '统计报表', breadcrumb: false, activeMenu: '/dzjk/tjbb'}, deviceCategory: 'CLUSTER'
hidden: false, },
redirect: '/dzjk/tjbb/gltj', },
children: [ {
{ path: 'dtdc',
path: 'gltj', component: () => import('@/views/ems/dzjk/sbjk/dtdc/index.vue'),
component: () => import('@/views/ems/dzjk/tjbb/gltj/index.vue'), name: 'DzjkSbjkDtdc',
name: 'DzjkTjbbGltj', hidden: true,
hidden: true, meta: {
meta: { title: '概率统计',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, title: '单体电池',
}, breadcrumb: false,
{ activeMenu: '/dzjk/sbjk',
path: 'glqx', activeSecondMenuName: 'DzjkSbjk',
component: () => import('@/views/ems/dzjk/tjbb/glqx/index.vue'), deviceCategory: 'BATTERY'
name: 'DzjkTjbbGlqx', },
hidden: true, },
meta: { title: '功率曲线',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, {
}, path: 'db',
{ component: () => import('@/views/ems/dzjk/sbjk/db/index.vue'),
path: 'pcsqx', name: 'DzjkSbjkDb',
component: () => import('@/views/ems/dzjk/tjbb/pcsqx/index.vue'), hidden: true,
name: 'DzjkTjbbPcsqx', meta: {
hidden: true, title: '电表',
meta: { title: 'PCS曲线',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, breadcrumb: false,
}, activeMenu: '/dzjk/sbjk',
{ activeSecondMenuName: 'DzjkSbjk',
path: 'dcdqx', deviceCategory: 'AMMETER'
component: () => import('@/views/ems/dzjk/tjbb/dcdqx/index.vue'), },
name: 'DzjkTjbbDcdqx', },
hidden: true, {
meta: { title: '电池堆曲线',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, path: 'yl',
}, component: () => import('@/views/ems/dzjk/sbjk/yl/index.vue'),
{ name: 'DzjkSbjkYl',
path: 'dcwd', hidden: true,
component: () => import('@/views/ems/dzjk/tjbb/dcwd/index.vue'), meta: {
name: 'DzjkTjbbDcwd', title: '液冷',
hidden: true, breadcrumb: false,
meta: { title: '电池温度',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, activeMenu: '/dzjk/sbjk',
}, activeSecondMenuName: 'DzjkSbjk',
{ deviceCategory: 'COOLING'
path: 'dbbb', },
component: () => import('@/views/ems/dzjk/tjbb/dbbb/index.vue'), },
name: 'DzjkTjbbDbbb', {
hidden: true, path: 'dh',
meta: { title: '电表报表',breadcrumb: false,activeMenu: '/dzjk/tjbb',activeSecondMenuName:'DzjkTjbb'}, component: () => import('@/views/ems/dzjk/sbjk/dh/index.vue'),
} name: 'DzjkSbjkDh',
] hidden: true,
}, meta: {
{ title: '动环',
path: '/dzjk/clpz', breadcrumb: false,
component: () => import('@/views/ems/dzjk/clpz/clyx/index.vue'), activeMenu: '/dzjk',
name: 'DzjkClpz', activeSecondMenuName: 'DzjkSbjk',
meta: {title: '策略配置', breadcrumb: false, activeMenu: '/dzjk/clpz'}, deviceCategory: 'DH'
} },
},
{
path: 'xf',
component: () => import('@/views/ems/dzjk/sbjk/xf/index.vue'),
name: 'DzjkSbjkXf',
hidden: true,
meta: {
title: '消防',
breadcrumb: false,
activeMenu: '/dzjk',
activeSecondMenuName: 'DzjkSbjk',
deviceCategory: 'XF'
},
}
]
},
{
path: '/dzjk/gzgj',
component: () => import('@/views/ems/dzjk/gzgj/index.vue'),
name: 'DzjkGzgj',
meta: {title: '故障告警', breadcrumb: false, activeMenu: '/dzjk/gzgj', activeSecondMenuName: 'DzjkGzgj'}
},
{
path: '/dzjk/tjbb',
component: () => import('@/views/ems/dzjk/tjbb/index.vue'),
name: 'DzjkTjbb',
alwaysShow: false,
meta: {title: '统计报表', breadcrumb: false, activeMenu: '/dzjk/tjbb'},
hidden: false,
redirect: '/dzjk/tjbb/gltj',
children: [
{
path: 'gltj',
component: () => import('@/views/ems/dzjk/tjbb/gltj/index.vue'),
name: 'DzjkTjbbGltj',
hidden: true,
meta: {
title: '运行统计',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
},
{
path: 'glqx',
component: () => import('@/views/ems/dzjk/tjbb/glqx/index.vue'),
name: 'DzjkTjbbGlqx',
hidden: true,
meta: {
title: '功率曲线',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
},
{
path: 'pcsqx',
component: () => import('@/views/ems/dzjk/tjbb/pcsqx/index.vue'),
name: 'DzjkTjbbPcsqx',
hidden: true,
meta: {
title: 'PCS曲线',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
},
{
path: 'dcdqx',
component: () => import('@/views/ems/dzjk/tjbb/dcdqx/index.vue'),
name: 'DzjkTjbbDcdqx',
hidden: true,
meta: {
title: '电池堆曲线',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
},
{
path: 'dcwd',
component: () => import('@/views/ems/dzjk/tjbb/dcwd/index.vue'),
name: 'DzjkTjbbDcwd',
hidden: true,
meta: {
title: '电池温度',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
},
{
path: 'dbbb',
component: () => import('@/views/ems/dzjk/tjbb/dbbb/index.vue'),
name: 'DzjkTjbbDbbb',
hidden: true,
meta: {
title: '电表报表',
breadcrumb: false,
activeMenu: '/dzjk/tjbb',
activeSecondMenuName: 'DzjkTjbb'
},
}
]
},
{
path: '/dzjk/clpz',
component: () => import('@/views/ems/dzjk/clpz/clyx/index.vue'),
name: 'DzjkClpz',
meta: {title: '策略配置', breadcrumb: false, activeMenu: '/dzjk/clpz'},
}
] ]
} }
] ]

View File

@ -7,7 +7,7 @@ module.exports = {
/** /**
* 侧边栏主题 深色主题theme-dark浅色主题theme-light * 侧边栏主题 深色主题theme-dark浅色主题theme-light
*/ */
sideTheme: 'theme-light', sideTheme: 'theme-dark',
/** /**
* 系统布局配置 * 系统布局配置

View File

@ -1,62 +1,81 @@
import {getAllSites} from '@/api/ems/zddt' import {getAllSites} from '@/api/ems/zddt'
import {getAlarmDetailList,getSiteAllDeviceCategory} from'@/api/ems/dzjk' import {getAlarmDetailList, getSiteAllDeviceCategory} from '@/api/ems/dzjk'
const ems = {
state: {
dzjkAlarmLighting:false,//单站监控 告警统计红点标志
zdList:[],
zdDeviceCategoryOptions:{},//站点各个站点包含的设备种类 {021_DDS_01:["BATTERY","CLUSTER","STACK", "DH", "AMMETER", "PCS", "XF"],021_DDS_02:[]...}
workStatusOptions:{'0':'正常','1':'异常','2':'停止'},//工作状态
deviceStatusOptions:{'0':'离线','1':'待机','2':'运行','3':'故障','4':'停机'},//设备状态
gridStatusOptions:{'0':'并网','1':'未并网'},//并网状态
controlModeOptions:{'0':'远程','1':'本地'},//控制模式
warnOptions:{0:'正常', 1:'中断', 2:'不在线',3:'异常'},//告警状态
communicationStatusOptions:{'0':'正常','1':'通讯中断','2':'异常'},//通讯状态
workModeOptions:{'0':'正常','1':'停止'},//工作模式
alarmLevelOptions:{'A':'提示','B':'一般','C':'严重','D':'紧急'},//告警等级
alarmStatusOptions:{'0':'待处理','1':'已处理','2':'处理中'},//告警状态
deviceTypeOptions:{'TCP':'TCP','RTU':'RTU'},//设备类型
ticketStatusOptions:{1:'待处理', 2:'处理中', 3:'已处理'},//工单处理状态
strategyStatusOptions:{'0':'未启用', '1':'已运行', '2':'已暂停', '3':'禁用', '4':'删除'},//策略状态
chargeStatusOptions:{'1':'充电','2':'待机'},//冲放状态
deviceCategoryOptions:{'PCS':'PCS','STACK':'电池堆','CLUSTER':'电池簇','COOLING':'液冷','BATTERY':'单体电池','AMMETER':'电表'},//设备类别
comparisonOperatorOptions:{'>':'>','<':'<','=':'=','>=':'>=','<=':'<='},
relationWithPoint:{'||':'||','&&':'&&'}
},
mutations: {
SET_ZD_LIST(state, list) {
state.zdList = list || []
},
SET_DZJK_ALARM_LIGHTING(state, status) {
state.dzjkAlarmLighting = status
},
SET_ZD_DEVICE_CATEGORY_OPTIONS(state,{siteId,data}){
state.zdDeviceCategoryOptions = Object.assign({}, state.zdDeviceCategoryOptions, {[siteId]:data})
}
},
actions: {
getZdList({commit,state}){
if(state.zdList.length === 0){
getAllSites().then(response => {
commit('SET_ZD_LIST', response?.data || [])
console.log('store action getZdList 获取站点数据',state.zdList)
})
}
},
//查询站点的所有待处理0的告警 存在展示红点标志
getSiteAlarmNum({state,commit},siteId){
getAlarmDetailList({status:0,siteId,pageSize:10,pageNum:1,deviceId:'',alarmLevel:'',alarmStartTime:'',alarmEndTime:''}).then(response=>{
commit('SET_DZJK_ALARM_LIGHTING',!!response?.total || false)
})
},
getSiteDeviceCategory({state,commit},siteId){
getSiteAllDeviceCategory(siteId).then(response=>{
let data = response?.data || [];
data.unshift('SSYX');
commit('SET_ZD_DEVICE_CATEGORY_OPTIONS',{siteId,data})
})
}
} const ems = {
state: {
dzjkAlarmLighting: false,//单站监控 告警统计红点标志
zdList: [],
zdDeviceCategoryOptions: {},//站点各个站点包含的设备种类 {021_DDS_01:["BATTERY","CLUSTER","STACK", "DH", "AMMETER", "PCS", "XF"],021_DDS_02:[]...}
CLUSTERWorkStatusOptions: {'0': '静置', '1': '充电', '2': '放电', '3': '待机', '5': '运行', '9': "故障"},//电池簇工作状态
PCSWorkStatusOptions: {'0': '运行', '1': '停机', '2': '故障', '3': '待机', '4': '充电', '5': '放电'},//PCS工作状态
STACKWorkStatusOptions: {
"0": "静置",
"1": "充电",
"2": "放电",
"3": "浮充",
'4': '待机',
'5': '运行',
'9': "故障"
},//STACKBMS总览工作状态
deviceStatusOptions: {'0': '离线', '1': '在线'},//设备状态
gridStatusOptions: {'0': '并网', '1': '未并网'},//并网状态
controlModeOptions: {'0': '远程', '1': '本地'},//控制模式
warnOptions: {0: '正常', 1: '中断', 2: '不在线', 3: '异常'},//告警状态
communicationStatusOptions: {'0': '正常', '1': '通讯中断', '2': '异常'},//通讯状态
workModeOptions: {'0': '正常', '1': '停止'},//工作模式
alarmLevelOptions: {'A': '提示', 'B': '一般', 'C': '严重', 'D': '紧急'},//告警等级
alarmStatusOptions: {'0': '待处理', '1': '已处理', '2': '处理中'},//告警状态
deviceTypeOptions: {'TCP': 'TCP', 'RTU': 'RTU'},//设备类型
ticketStatusOptions: {1: '待处理', 2: '处理中', 3: '已处理'},//工单处理状态
strategyStatusOptions: {'0': '未启用', '1': '已运行', '2': '已暂停', '3': '禁用', '4': '删除'},//策略状态
chargeStatusOptions: {'1': '充电', '2': '待机', '3': '放电'},//冲放状态
comparisonOperatorOptions: {'>': '>', '<': '<', '=': '=', '>=': '>=', '<=': '<='},
relationWithPoint: {'||': '||', '&&': '&&'}
},
mutations: {
SET_ZD_LIST(state, list) {
state.zdList = list || []
},
SET_DZJK_ALARM_LIGHTING(state, status) {
state.dzjkAlarmLighting = status
},
SET_ZD_DEVICE_CATEGORY_OPTIONS(state, {siteId, data}) {
state.zdDeviceCategoryOptions = Object.assign({}, state.zdDeviceCategoryOptions, {[siteId]: data})
}
},
actions: {
getZdList({commit, state}) {
if (state.zdList.length === 0) {
getAllSites().then(response => {
commit('SET_ZD_LIST', response?.data || [])
console.log('store action getZdList 获取站点数据', state.zdList)
})
}
},
//查询站点的所有待处理0的告警 存在展示红点标志
getSiteAlarmNum({state, commit}, siteId) {
getAlarmDetailList({
status: 0,
siteId,
pageSize: 10,
pageNum: 1,
deviceId: '',
alarmLevel: '',
alarmStartTime: '',
alarmEndTime: ''
}).then(response => {
commit('SET_DZJK_ALARM_LIGHTING', !!response?.total || false)
})
},
getSiteDeviceCategory({state, commit}, siteId) {
getSiteAllDeviceCategory(siteId).then(response => {
let data = response?.data || [];
data.unshift('SSYX');
commit('SET_ZD_DEVICE_CATEGORY_OPTIONS', {siteId, data})
})
}
}
} }
export default ems export default ems

View File

@ -1,119 +1,140 @@
<template> <template>
<el-dialog :visible.sync="dialogTableVisible" class="ems-dialog" :title="mode === 'add'?'新增模板':`编辑模板` "> <el-dialog :visible.sync="dialogTableVisible" class="ems-dialog add-template-dialog"
<el-form ref="addTempForm" :model="formData" :rules="rules" size="medium" label-width="100px"> :title="mode === 'add'?'新增模板':`编辑模板` ">
<el-form-item label="模板名称" prop="templateName"> <el-form ref="addTempForm" :model="formData" :rules="rules" size="medium" label-width="100px">
<el-input v-model="formData.templateName" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="模板名称" prop="templateName">
</el-input> <el-input v-model="formData.templateName" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="soc限制" prop="sdcLimit" required> </el-form-item>
<el-switch :active-value="1" :inactive-value="0" v-model="formData.sdcLimit"></el-switch> <el-form-item label="soc限制" prop="sdcLimit" required>
</el-form-item> <el-switch :active-value="1" :inactive-value="0" v-model="formData.sdcLimit"></el-switch>
<!-- <template v-if="formData.sdcLimit === 1">--> </el-form-item>
<el-form-item label="soc下限" prop="sdcDown"> <!-- <template v-if="formData.sdcLimit === 1">-->
<el-input v-model="formData.sdcDown" placeholder="请输入" clearable :style="{width: '100%'}"></el-input> <el-form-item label="soc下限" prop="sdcDown">
<el-input v-model="formData.sdcDown" placeholder="请输入" clearable :style="{width: '100%'}"></el-input>
</el-form-item>
<el-form-item label="soc上限" prop="sdcUp">
<el-input v-model="formData.sdcUp" placeholder="请输入" clearable :style="{width: '100%'}"></el-input>
</el-form-item>
<!-- </template>-->
</el-form>
<el-button type="primary" size="mini" @click="addTime">新增</el-button>
<!-- 新增时间段表单-->
<el-collapse-transition>
<el-card v-show="showAddTime" shadow="always" class="common-card-container" style="margin-top:25px;">
<el-form class="add-time-form transition-box" ref="addTimeForm" :model="formInline" :rules="formInlineRule"
label-width="100px" style="margin-top:25px">
<!-- <el-form-item label="开始时间" prop="startTime">-->
<!-- <el-time-select-->
<!-- placeholder="开始时间"-->
<!-- v-model="formInline.startTime"-->
<!-- :picker-options="{-->
<!-- start: '00:00',-->
<!-- step: '00:01',-->
<!-- end: '23:00',-->
<!-- }">-->
<!-- </el-time-select>-->
<!-- </el-form-item>-->
<!-- <el-form-item label="结束时间" prop="endTime">-->
<!-- <el-time-select-->
<!-- placeholder="结束时间"-->
<!-- v-model="formInline.endTime"-->
<!-- :picker-options="{-->
<!-- start: '00:00',-->
<!-- step: '00:01',-->
<!-- end: '23:00',-->
<!-- minTime: formInline.startTime-->
<!-- }">-->
<!-- </el-time-select>-->
<!-- </el-form-item>-->
<el-form-item label="时间范围" prop="timeRange">
<el-time-picker
is-range
v-model="formInline.timeRange"
range-separator=""
start-placeholder="开始时间"
end-placeholder="结束时间"
placeholder="选择时间范围"
format="HH:mm"
value-format="HH:mm"
:style="{width: '100%'}">
</el-time-picker>
</el-form-item> </el-form-item>
<el-form-item label="soc上限" prop="sdcUp"> <el-form-item label="冲放功率" prop="chargeDischargePower">
<el-input v-model="formData.sdcUp" placeholder="请输入" clearable :style="{width: '100%'}"></el-input> <el-input v-model="formInline.chargeDischargePower" placeholder="请输入"
:style="{width: '100%'}"></el-input>
</el-form-item> </el-form-item>
<!-- </template>--> <el-form-item label="充电状态" prop="chargeStatus">
</el-form> <el-select v-model="formInline.chargeStatus" placeholder="请选择" :style="{width: '100%'}">
<el-button type="primary" size="mini" @click="addTime">新增</el-button> <el-option v-for="(value,key) in chargeStatusOptions" :key="key+'chargeStatusOptions'" :label="value"
<!-- 新增时间段表单--> :value="key"></el-option>
<el-collapse-transition> </el-select>
<el-card v-show="showAddTime" shadow="always" class="common-card-container" style="margin-top:25px;"> </el-form-item>
<el-form class="add-time-form transition-box" ref="addTimeForm" :model="formInline" :rules="formInlineRule" label-width="100px" style="margin-top:25px"> <el-form-item>
<el-form-item label="开始时间" prop="startTime"> <el-button type="primary" size="mini" @click="saveTime">保存</el-button>
<el-time-select <el-button size="mini" @click="cancelAddTime">取消</el-button>
placeholder="开始时间" </el-form-item>
v-model="formInline.startTime" </el-form>
:picker-options="{ </el-card>
start: '00:00', </el-collapse-transition>
step: '01:00', <el-table
end: '23:00',
}">
</el-time-select>
</el-form-item>
<el-form-item label="结束时间" prop="endTime">
<el-time-select
placeholder="结束时间"
v-model="formInline.endTime"
:picker-options="{
start: '00:00',
step: '01:00',
end: '23:00',
minTime: formInline.startTime
}">
</el-time-select>
</el-form-item>
<el-form-item label="冲放功率" prop="chargeDischargePower">
<el-input v-model="formInline.chargeDischargePower" placeholder="请输入"></el-input>
</el-form-item>
<el-form-item label="充电状态" prop="chargeStatus">
<el-select v-model="formInline.chargeStatus" placeholder="请选择">
<el-option v-for="(value,key) in chargeStatusOptions" :key="key+'chargeStatusOptions'" :label="value" :value="key"></el-option>
</el-select>
</el-form-item>
<el-form-item>
<el-button type="primary" size="mini" @click="saveTime">保存</el-button>
<el-button size="mini" @click="cancelAddTime">取消</el-button>
</el-form-item>
</el-form>
</el-card>
</el-collapse-transition>
<el-table
:data="tableData" :data="tableData"
border border
style="width: 100%;margin-top:25px"> style="width: 100%;margin-top:25px">
<!-- todo 如果要在span-method中使用column.property 在表格中必须定义prop="xxx"属性--> <el-table-column
<el-table-column
prop="startTime" prop="startTime"
label="开始时间"> label="开始时间">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="endTime" prop="endTime"
label="结束时间"> label="结束时间">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="chargeDischargePower" prop="chargeDischargePower"
label="充放功率kW"> label="充放功率kW">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="chargeStatus" prop="chargeStatus"
label="充电状态"> label="充电状态">
<template slot-scope="scope"> <template slot-scope="scope">
{{chargeStatusOptions[scope.row.chargeStatus]}} {{ chargeStatusOptions[scope.row.chargeStatus] }}
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
fixed="right" fixed="right"
label="操作" label="操作"
width="120"> width="120">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click.native.prevent="deleteRow(scope.$index, tableData)" @click.native.prevent="deleteRow(scope.$index, tableData)"
type="warning" type="warning"
size="mini"> size="mini">
删除 删除
</el-button> </el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<div slot="footer"> <div slot="footer">
<el-button @click="closeDialog">取消</el-button> <el-button @click="closeDialog">取消</el-button>
<el-button type="primary" @click="saveDialog">确定</el-button> <el-button type="primary" @click="saveDialog">确定</el-button>
</div> </div>
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
import {mapState} from 'vuex' import {mapState} from 'vuex'
import {addStrategyTemp,editStrategyTemp,getStrategyTempDetail} from '@/api/ems/dzjk' import {addStrategyTemp, editStrategyTemp, getStrategyTempDetail} from '@/api/ems/dzjk'
export default { export default {
inject:['$home'], inject: ['$home'],
data() { data() {
const now = new Date()
const year = now.getFullYear(), month = now.getMonth(), day = now.getDate()
const range = [new Date(year, month, day, 0), new Date(year, month, day, 23)]
return { return {
mode:'', mode: '',
editTempId:'', editTempId: '',
dialogTableVisible:false, dialogTableVisible: false,
secondRange: range,
formData: { formData: {
templateName: '', templateName: '',
sdcLimit: false, sdcLimit: false,
@ -127,44 +148,41 @@ export default {
trigger: 'blur' trigger: 'blur'
}], }],
sdcDown: [ sdcDown: [
{required: true,message: '请输入', trigger: 'blur'}, {required: true, message: '请输入', trigger: 'blur'},
{ pattern: /^(0|[1-9]\d*)(\.\d+)?$/, message: '请输入合法数字或小数' } {pattern: /^(0|[1-9]\d*)(\.\d+)?$/, message: '请输入合法数字或小数'}
], ],
sdcUp: [ sdcUp: [
{required: true,message: '请输入', trigger: 'blur'}, {required: true, message: '请输入', trigger: 'blur'},
{ pattern: /^(0|[1-9]\d*)(\.\d+)?$/, message: '请输入合法数字或小数' } {pattern: /^(0|[1-9]\d*)(\.\d+)?$/, message: '请输入合法数字或小数'}
], ],
}, },
showAddTime: false, showAddTime: false,
formInline:{ formInline: {
startTime:'',endTime:'',chargeDischargePower:'',chargeStatus:'' timeRange: range,
chargeDischargePower: '',
chargeStatus: ''
}, },
formInlineRule:{ formInlineRule: {
startTime: [{ timeRange: [{
required: true, required: true,
message: '请选择开始时间', message: '请选择时间范围',
trigger: 'change'
}],
endTime: [{
required: true,
message: '请选择结束时间',
trigger: 'change' trigger: 'change'
}], }],
chargeDischargePower: [{ chargeDischargePower: [{
required: true, required: true,
message: '请输入冲放功率', message: '请输入冲放功率',
trigger: 'blur' trigger: 'blur'
}, },
{ pattern: /^(0|[1-9]\d*)(\.\d+)?$/, message: '请输入合法数字或小数' } {pattern: /^-?\d*\.?\d*$/, message: '请输入合法数字或小数'}
], ],
chargeStatus:[{ chargeStatus: [{
required: true, required: true,
message: '请选择充放状态', message: '请选择充放状态',
trigger: ['blur','change'] trigger: ['blur', 'change']
} }
] ]
}, },
tableData:[], tableData: [],
} }
}, },
computed: { computed: {
@ -172,145 +190,136 @@ export default {
chargeStatusOptions: state => state?.ems?.chargeStatusOptions || {}, chargeStatusOptions: state => state?.ems?.chargeStatusOptions || {},
}) })
}, },
watch: {
"formInline.startTime":{
handler(newVal){
if(newVal && this.formInline.endTime){
const endTime = parseInt((this.formInline.endTime).split(':')[0] || 0)
const startTime =parseInt(newVal.split(':')[0])
if(endTime<=startTime){
this.formInline.endTime = `${startTime+1 <=9 ? '0'+(startTime+1) : startTime+1}:00`
}
}
},
deep:true
},
},
methods: { methods: {
changeSiteId(){ changeSiteId() {
this.dialogTableVisible=false this.dialogTableVisible = false
this.mode='' this.mode = ''
this.editTempId='' this.editTempId = ''
this.formData={ this.formData = {
templateName: '', templateName: '',
sdcLimit: false, sdcLimit: false,
sdcDown: '', sdcDown: '',
sdcUp: '', sdcUp: '',
} }
this.formInline={ this.formInline = {
startTime:'',endTime:'',chargeDischargePower:'',chargeStatus:'' timeRange: this.secondRange, chargeDischargePower: '', chargeStatus: ''
} }//startTime: '', endTime: '',
this.showAddTime = false this.showAddTime = false
this.tableData=[] this.tableData = []
}, },
show({mode = 'add', editTempId = ''}){ show({mode = 'add', editTempId = ''}) {
this.$nextTick(() => { this.$nextTick(() => {
this.dialogTableVisible = true this.dialogTableVisible = true
this.mode = mode this.mode = mode
this.editTempId=editTempId this.editTempId = editTempId
if(mode === 'edit' && editTempId){ if (mode === 'edit' && editTempId) {
getStrategyTempDetail(this.editTempId).then(response => { getStrategyTempDetail(this.editTempId).then(response => {
const data=JSON.parse(JSON.stringify(response?.data || [])); const data = JSON.parse(JSON.stringify(response?.data || []));
if(data.length>0){ if (data.length > 0) {
const {templateName,sdcLimit,sdcDown,sdcUp} =JSON.parse(JSON.stringify( data[0])); const {templateName, sdcLimit, sdcDown, sdcUp} = JSON.parse(JSON.stringify(data[0]));
this.formData.templateName=templateName this.formData.templateName = templateName
this.formData.sdcLimit=sdcLimit this.formData.sdcLimit = sdcLimit
this.formData.sdcDown=sdcDown this.formData.sdcDown = sdcDown
this.formData.sdcUp=sdcUp this.formData.sdcUp = sdcUp
} }
if(data.length === 1){ if (data.length === 1) {
const {startTime,endTime}=data; const {startTime, endTime} = data;
if(!startTime || !endTime){ if (!startTime || !endTime) {
this.tableData = [] this.tableData = []
}else{ } else {
this.tableData= data this.tableData = data
} }
}else{ } else {
this.tableData= data this.tableData = data
} }
}) })
} }
}) })
}, },
addTime(){ addTime() {
this.showAddTime=true this.showAddTime = true
}, },
cancelAddTime(){ cancelAddTime() {
this.$refs.addTimeForm.resetFields() this.$refs.addTimeForm.resetFields()
this.showAddTime=false this.showAddTime = false
this.formInline = {startTime:'',endTime:'',chargeDischargePower:'',chargeStatus:''} this.formInline = {timeRange: this.secondRange, chargeDischargePower: '', chargeStatus: ''}//startTime: '', endTime: '',
}, },
saveTime(){ saveTime() {
//表单校验校验成功添加到tableData里 //表单校验校验成功添加到tableData里
this.$refs.addTimeForm.validate(valid => { this.$refs.addTimeForm.validate(valid => {
if (!valid) return if (!valid) return
this.tableData.push(JSON.parse(JSON.stringify(this.formInline))); const {timeRange: [startTime, endTime], chargeDischargePower, chargeStatus} = this.formInline
this.$nextTick(() => {this.cancelAddTime()})
this.tableData.push({startTime, endTime, chargeDischargePower, chargeStatus})
this.$nextTick(() => {
this.cancelAddTime()
})
}) })
}, },
deleteRow(index){ deleteRow(index) {
this.tableData.splice(index,1) this.tableData.splice(index, 1)
}, },
saveDialog() { saveDialog() {
this.$refs.addTempForm.validate(valid => { this.$refs.addTempForm.validate(valid => {
if (!valid) return if (!valid) return
//校验时间选择范围是否冲突 const {templateName, sdcLimit, sdcDown, sdcUp} = this.formData
let status = true const {siteId, updateStrategyId} = this.$home
this.tableData.forEach((outer,outerIndex)=>{ const {tableData} = this
const {startTime, endTime}=outer if (this.mode === 'edit') {
const outerStart = parseInt(startTime),outerEnd = parseInt(endTime) editStrategyTemp({
if(outerStart>outerEnd){ siteId,
status = false strategyId: updateStrategyId,
}else{ templateId: this.editTempId,
this.tableData.forEach((inner,innerIndex)=>{ templateName,
if(innerIndex !== outerIndex){ sdcLimit,
const {startTime:innerStartTime, endTime:innerEndTime}=inner sdcDown,
const innerStart = parseInt(innerStartTime),innerEnd = parseInt(innerEndTime) sdcUp,
if((innerStart<outerStart && innerEnd>outerEnd) || !((innerStart<outerStart && innerEnd<=outerStart) || (innerStart>=outerEnd && innerEnd>outerEnd))){ timeConfigList: tableData
status=false }).then(response => {
} if (response?.code === 200) {
} this.closeDialog()
}) this.$emit('update')
this.$emit('updateTimeSetting')
} }
}) })
if(!status){ } else {
return this.$message.error('时间选择范围冲突'); addStrategyTemp({
} siteId,
const {templateName,sdcLimit,sdcDown,sdcUp} = this.formData strategyId: updateStrategyId,
const {siteId,updateStrategyId} =this.$home templateName,
const {tableData} = this sdcLimit,
if(this.mode==='edit'){ sdcDown,
editStrategyTemp({siteId,strategyId:updateStrategyId,templateId:this.editTempId,templateName,sdcLimit,sdcDown,sdcUp,timeConfigList:tableData}).then(response=>{ sdcUp,
if(response?.code === 200){ timeConfigList: tableData
this.closeDialog() }).then(response => {
this.$emit('update') if (response?.code === 200) {
this.$emit('updateTimeSetting') this.closeDialog()
} this.$emit('update')
}) }
}else{ })
addStrategyTemp({siteId,strategyId:updateStrategyId,templateName,sdcLimit,sdcDown,sdcUp,timeConfigList:tableData}).then(response=>{ }
if(response?.code === 200){
this.closeDialog()
this.$emit('update')
}
})
}
}) })
}, },
closeDialog(){ closeDialog() {
// 清空所有数据 // 清空所有数据
this.$refs.addTempForm.resetFields() this.$refs.addTempForm.resetFields()
this.formData={ this.formData = {
templateName: '', templateName: '',
sdcLimit:0, sdcLimit: 0,
sdcDown: '', sdcDown: '',
sdcUp: '', sdcUp: '',
} }
this.tableData=[] this.tableData = []
this.cancelAddTime() this.cancelAddTime()
this.dialogTableVisible=false this.dialogTableVisible = false
} }
} }
} }
</script> </script>
<style lang="scss" scoped>
.add-template-dialog {
max-height: 90vh;
overflow-y: auto;
}
</style>

View File

@ -1,27 +1,30 @@
<template> <template>
<div v-loading="loading" class="ems-dashboard-editor-container ems-content-container-padding"> <div v-loading="loading" class="ems-dashboard-editor-container ems-content-container-padding">
<!-- 搜索栏--> <!-- 搜索栏-->
<el-form :inline="true" class="select-container"> <el-form :inline="true" class="select-container">
<el-form-item label="设备清单"> <el-form-item label="设备清单">
<el-select v-model="search.deviceId" clearable placeholder="请选择" :loading="loading" loading-text="正在加载数据"> <el-select v-model="search.deviceId" clearable placeholder="请选择" :loading="loading"
<el-option :label="item.deviceName" :value="item.deviceId" v-for="(item,key) in deviceOptions" :key="key+'deviceIdOptions'"></el-option> loading-text="正在加载数据">
<el-option :label="item.deviceName" :value="item.deviceId" v-for="(item,key) in deviceOptions"
:key="key+'deviceIdOptions'"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="告警等级"> <el-form-item label="告警等级">
<el-select v-model="search.alarmLevel" clearable placeholder="请选择" :loading="loading" loading-text="正在加载数据"> <el-select v-model="search.alarmLevel" clearable placeholder="请选择" :loading="loading"
<el-option :label="value" :value="key" v-for="(value,key) in $store.state.ems.alarmLevelOptions" :key="key+'alarmLevelOptions'"></el-option> loading-text="正在加载数据" style="width: 130px">
<el-option :label="value" :value="key" v-for="(value,key) in $store.state.ems.alarmLevelOptions"
:key="key+'alarmLevelOptions'"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="时间选择"> <el-form-item label="时间选择">
<el-date-picker <el-date-picker
v-model="dateRange" v-model="dateRange"
type="daterange" type="daterange"
range-separator="" range-separator=""
start-placeholder="开始时间" start-placeholder="开始时间"
:picker-options="pickerOptions" :picker-options="pickerOptions"
:default-value="defaultDateRange" :default-value="defaultDateRange"
end-placeholder="结束时间"> end-placeholder="结束时间">
</el-date-picker> </el-date-picker>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
@ -36,7 +39,9 @@
<el-row style=""> <el-row style="">
<el-col :xs="24" :sm="24" :lg="24"> <el-col :xs="24" :sm="24" :lg="24">
<el-button-group class="ems-btns-group"> <el-button-group class="ems-btns-group">
<el-button v-for="(item,index) in btnList" :key="index+'dtdcBtns'" :class="{'activeBtn' : activeBtn === item.id}" @click="changeDataType(item.id)">{{item.name}}</el-button> <el-button v-for="(item,index) in btnList" :key="index+'dtdcBtns'"
:class="{'activeBtn' : activeBtn === item.id}" @click="changeDataType(item.id)">{{ item.name }}
</el-button>
</el-button-group> </el-button-group>
</el-col> </el-col>
</el-row> </el-row>
@ -47,64 +52,67 @@
stripe stripe
max-height="500" max-height="500"
style="width: 100%;margin-top:25px;"> style="width: 100%;margin-top:25px;">
<el-table-column <el-table-column
prop="deviceName" prop="deviceName"
label="设备名称"> label="设备名称">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="告警等级" label="告警等级"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{$store.state.ems.alarmLevelOptions[scope.row.alarmLevel]}}</span> <span>{{ $store.state.ems.alarmLevelOptions[scope.row.alarmLevel] }}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="alarmContent" prop="alarmContent"
show-overflow-tooltip show-overflow-tooltip
label="告警内容"> label="告警内容">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="alarmStartTime" prop="alarmStartTime"
label="告警发生时间"> label="告警发生时间">
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{formatDate(scope.row.alarmStartTime,true)}}</span> <span>{{ formatDate(scope.row.alarmStartTime, true) }}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="alarmEndTime" prop="alarmEndTime"
label="告警结束时间"> label="告警结束时间">
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{formatDate(scope.row.alarmEndTime,true)}}</span> <span>{{ formatDate(scope.row.alarmEndTime, true) }}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="状态"> label="状态">
<template slot-scope="scope"> <template slot-scope="scope">
<span :class="['0','2'].includes(scope.row.status) ? 'warning-status' : ''">{{$store.state.ems.alarmStatusOptions[scope.row.status]}}</span> <span
</template> :class="['0','2'].includes(scope.row.status) ? 'warning-status' : ''">{{ $store.state.ems.alarmStatusOptions[scope.row.status] }}</span>
</el-table-column> </template>
<el-table-column </el-table-column>
<el-table-column
label="工单" label="工单"
fixed="right" fixed="right"
width="250" width="250"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<el-button type="text" size="mini" v-if="scope.row.ticketNo" @click="toTicket">已生成工单(工单号:{{scope.row.ticketNo}})</el-button> <el-button type="text" size="mini" v-if="scope.row.ticketNo" @click="toTicket">
<el-button type="primary" size="mini" v-else @click="createTicket(scope.row.id)">生成工单</el-button> 已生成工单(工单号:{{ scope.row.ticketNo }})
</template> </el-button>
</el-table-column> <el-button type="primary" size="mini" v-else @click="createTicket(scope.row.id)">生成工单</el-button>
</el-table> </template>
</el-table-column>
</el-table>
<el-pagination <el-pagination
v-show="tableData.length>0" v-show="tableData.length>0"
background background
@size-change="handleSizeChange" @size-change="handleSizeChange"
@current-change="handleCurrentChange" @current-change="handleCurrentChange"
:current-page="pageNum" :current-page="pageNum"
:page-size="pageSize" :page-size="pageSize"
:page-sizes="[10, 20, 30, 40]" :page-sizes="[10, 20, 30, 40]"
layout="total, sizes, prev, pager, next, jumper" layout="total, sizes, prev, pager, next, jumper"
:total="totalSize" :total="totalSize"
style="margin-top:15px;text-align: center" style="margin-top:15px;text-align: center"
> >
</el-pagination> </el-pagination>
</div> </div>
@ -113,113 +121,127 @@
<script> <script>
import {getAlarmDetailList,createTicketNo} from'@/api/ems/dzjk' import {createTicketNo, getAlarmDetailList} from '@/api/ems/dzjk'
import {getDeviceList} from'@/api/ems/site' import {getDeviceList} from '@/api/ems/site'
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { formatDate } from '@/filters/ems' import {formatDate} from '@/filters/ems'
export default { export default {
name:'DzjkGzgj', name: 'DzjkGzgj',
mixins:[getQuerySiteId], mixins: [getQuerySiteId],
data() { data() {
return { return {
loading:false, loading: false,
btnList:[ btnList: [
{name:'未处理告警',id:'today'}, {name: '未处理告警', id: 'today'},
{name:'历史告警',id:'history'}, {name: '历史告警', id: 'history'},
], ],
deviceOptions:[],//设备列表 deviceOptions: [],//设备列表
pickerOptions:{ pickerOptions: {
disabledDate(time) { disabledDate(time) {
return time.getTime() > Date.now(); return time.getTime() > Date.now();
}, },
}, },
defaultDateRange:[],//默认展示的时间 defaultDateRange: [],//默认展示的时间
dateRange:[],//alarmStartTime,alarmEndTime dateRange: [],//alarmStartTime,alarmEndTime
activeBtn:'today', activeBtn: 'today',
search:{deviceId:'',alarmLevel:''}, search: {deviceId: '', alarmLevel: ''},
// 表格、分页 // 表格、分页
tableData:[], tableData: [],
pageSize:10,//分页栏当前每个数据总数 pageSize: 10,//分页栏当前每个数据总数
pageNum:1,//分页栏当前页数 pageNum: 1,//分页栏当前页数
totalSize:0,//table表格数据总数 totalSize: 0,//table表格数据总数
} }
}, },
methods:{ methods: {
formatDate, formatDate,
toTicket(){ toTicket() {
this.$router.push({path:'/ticket'}) this.$router.push({path: '/ticket'})
}, },
//生成工单 //生成工单
createTicket(id){ createTicket(id) {
this.loading = true this.loading = true
createTicketNo({id}).then(response=>{ createTicketNo({id}).then(response => {
response?.data && this.toTicket() response?.data && this.toTicket()
}).finally(()=>{this.loading = false}) }).finally(() => {
this.loading = false
})
}, },
// 判断是否是同一天 // 判断是否是同一天
isSameDay(day1, day2) { isSameDay(day1, day2) {
const date1 = new Date(day1),date2 = new Date(day2) const date1 = new Date(day1), date2 = new Date(day2)
return date1.getFullYear() === date2.getFullYear() && return date1.getFullYear() === date2.getFullYear() &&
date1.getMonth() === date2.getMonth() && date1.getMonth() === date2.getMonth() &&
date1.getDate() === date2.getDate(); date1.getDate() === date2.getDate();
}, },
// 分页 // 分页
handleSizeChange(val) { handleSizeChange(val) {
this.pageSize = val; this.pageSize = val;
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
handleCurrentChange(val) { handleCurrentChange(val) {
this.pageNum = val this.pageNum = val
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
// 搜索 // 搜索
onSearch(){ onSearch() {
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 重置 // 重置
onReset(){ onReset() {
this.search={deviceId:'',alarmLevel:''} this.search = {deviceId: '', alarmLevel: ''}
this.dateRange=[] this.dateRange = []
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 切换今日、历史告警 // 切换今日、历史告警
changeDataType(id){ changeDataType(id) {
if(id !== this.activeBtn){ if (id !== this.activeBtn) {
console.log('点击了不同的菜单,更新数据') console.log('点击了不同的菜单,更新数据')
this.activeBtn=id; this.activeBtn = id;
this.getData() this.getData()
} }
}, },
// 获取数据 // 获取数据
getData(){ getData() {
this.$store.dispatch('getSiteAlarmNum',this.siteId) this.$store.dispatch('getSiteAlarmNum', this.siteId)
this.loading=true this.loading = true
const {deviceId,alarmLevel} = this.search const {deviceId, alarmLevel} = this.search
const {siteId,pageNum,pageSize,activeBtn} =this const {siteId, pageNum, pageSize, activeBtn} = this
const [alarmStartTime='',alarmEndTime='']=(this.dateRange || []) const [alarmStartTime = '', alarmEndTime = ''] = (this.dateRange || [])
let status = activeBtn === 'today' ? '0' : '1,2' let status = activeBtn === 'today' ? '0' : '1,2'
getAlarmDetailList({status,deviceId,alarmLevel,siteId,pageSize,pageNum,alarmStartTime:formatDate(alarmStartTime),alarmEndTime:formatDate(alarmEndTime)}).then(response => { getAlarmDetailList({
this.tableData=response?.rows || []; status,
deviceId,
alarmLevel,
siteId,
pageSize,
pageNum,
alarmStartTime: formatDate(alarmStartTime),
alarmEndTime: formatDate(alarmEndTime)
}).then(response => {
this.tableData = response?.rows || [];
this.totalSize = response?.total || 0 this.totalSize = response?.total || 0
}).finally(() => {this.loading=false}) }).finally(() => {
this.loading = false
})
}, },
getDeviceOptions(){ getDeviceOptions() {
getDeviceList(this.siteId).then(response => { getDeviceList(this.siteId).then(response => {
this.deviceOptions = JSON.parse(JSON.stringify(response?.data || [])) this.deviceOptions = JSON.parse(JSON.stringify(response?.data || []))
}) })
}, },
init(){ init() {
this.getDeviceOptions() this.getDeviceOptions()
this.onReset() this.onReset()
}, },
}, },
mounted(){ mounted() {
const now = new Date(); const now = new Date();
const lastMonth = new Date(now.getFullYear(), now.getMonth() - 1, 1); const lastMonth = new Date(now.getFullYear(), now.getMonth() - 1, 1);
this.defaultDateRange = [lastMonth, now]; this.defaultDateRange = [lastMonth, now];

View File

@ -1,30 +1,29 @@
<template> <template>
<el-card shadow="always" class="common-card-container common-card-container-body-no-padding time-range-card"> <el-card shadow="always" class="common-card-container common-card-container-body-no-padding time-range-card">
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title">当日功率曲线</span> <span class="card-title">当日功率曲线</span>
<date-range-select ref="dateRangeSelect" @updateDate="updateDate"/> <date-range-select ref="dateRangeSelect" :showIcon="true" :mini-time-picker="true" @updateDate="updateDate"/>
</div> </div>
<div style="height: 310px" id="activeChart"></div> <div style="height: 310px" id="activeChart"></div>
</el-card> </el-card>
</template> </template>
<script> <script>
import * as echarts from 'echarts' import * as echarts from 'echarts'
import resize from '@/mixins/ems/resize' import resize from '@/mixins/ems/resize'
import DateRangeSelect from '@/components/Ems/DateRangeSelect/index.vue' import DateRangeSelect from '@/components/Ems/DateRangeSelect/index.vue'
import { getPointData } from '@/api/ems/dzjk' import {getPointData} from '@/api/ems/dzjk'
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
export default { export default {
mixins: [resize,intervalUpdate], mixins: [resize, intervalUpdate],
components: {DateRangeSelect}, components: {DateRangeSelect},
data() { data() {
return { return {
chart: null, chart: null,
timeRange:[], timeRange: [],
siteId:'', siteId: '',
isInit:true isInit: true
} }
}, },
mounted() { mounted() {
@ -41,23 +40,23 @@ export default {
}, },
methods: { methods: {
// 更新时间范围 重置图表 // 更新时间范围 重置图表
updateDate(data){ updateDate(data) {
this.timeRange=data this.timeRange = data
!this.isInit && this.getGVQXData() !this.isInit && this.getGVQXData()
this.isInit = false this.isInit = false
}, },
getGVQXData(){ getGVQXData() {
this.showLoading() this.showLoading()
const {siteId,timeRange}=this const {siteId, timeRange} = this
getPointData({siteId,startDate:timeRange[0],endDate:timeRange[1]}).then(response => { getPointData({siteId, startDate: timeRange[0], endDate: timeRange[1]}).then(response => {
this.setOption(response?.data || []) this.setOption(response?.data || [])
}).finally(()=>this.hideLoading()) }).finally(() => this.hideLoading())
}, },
init(siteId){ init(siteId) {
//初始化 清空数据 //初始化 清空数据
this.siteId = siteId this.siteId = siteId
this.isInit = true this.isInit = true
this.timeRange=[] this.timeRange = []
this.$refs.dateRangeSelect.init(true) this.$refs.dateRangeSelect.init(true)
this.getGVQXData() this.getGVQXData()
this.updateInterval(this.getGVQXData) this.updateInterval(this.getGVQXData)
@ -65,17 +64,17 @@ export default {
initChart() { initChart() {
this.chart = echarts.init(document.querySelector('#activeChart')) this.chart = echarts.init(document.querySelector('#activeChart'))
}, },
showLoading(){ showLoading() {
this.chart && this.chart.showLoading() this.chart && this.chart.showLoading()
}, },
hideLoading(){ hideLoading() {
this.chart && this.chart.hideLoading() this.chart && this.chart.hideLoading()
}, },
setOption(data) { setOption(data) {
const source = [['日期','电网功率','负载功率','储能功率','光伏功率','soc平均值','soh平均值','电池平均温度平均值']] const source = [['日期', '电网功率', '负载功率', '储能功率', '光伏功率', 'soc平均值', 'soh平均值', '电池平均温度平均值']]
console.log('source.slice(1)',source[0].slice(1)) console.log('source.slice(1)', source[0].slice(1))
this.chart && data.forEach((item)=>{ this.chart && data.forEach((item) => {
source.push([item.statisDate,item.gridPower,item.loadPower,item.storagePower,item.pvPower,item.avgSoc,item.avgSoh,item.avgTemp]) source.push([item.statisDate, item.gridPower, item.loadPower, item.storagePower, item.pvPower, item.avgSoc, item.avgSoh, item.avgTemp])
}) })
this.chart.setOption({ this.chart.setOption({
grid: { grid: {
@ -91,8 +90,8 @@ export default {
type: 'shadow' // 默认为直线,可选为:'line' | 'shadow' type: 'shadow' // 默认为直线,可选为:'line' | 'shadow'
} }
}, },
textStyle:{ textStyle: {
color:"#333333", color: "#333333",
}, },
xAxis: { xAxis: {
type: 'category', type: 'category',
@ -105,11 +104,17 @@ export default {
type: 'value', type: 'value',
}, },
], ],
dataset:{source}, dataset: {source},
series: source[0].slice(1).map((item,index)=>{ series: source[0].slice(1).map((item, index) => {
return { return {
type: 'line', type: 'line',//index === 5 ? 'bar' : 'line',
yAxisIndex:index<=4 ? 0 : 1 showSymbol: false,
symbolSize: 2,
smooth: true,
areaStyle: {
opacity: 0.5,
},
yAxisIndex: index <= 4 ? 0 : 1
} }
}) })
}) })

View File

@ -1,12 +1,11 @@
<template> <template>
<el-card shadow="always" class="common-card-container common-card-container-body-no-padding time-range-card"> <el-card shadow="always" class="common-card-container common-card-container-body-no-padding time-range-card">
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title">一周充放曲线</span> <span class="card-title">一周充放曲线</span>
<date-range-select ref="dateRangeSelect" @updateDate="updateDate"/> <date-range-select ref="dateRangeSelect" :showIcon="true" :mini-time-picker="true" @updateDate="updateDate"/>
</div> </div>
<div style="height: 310px" id="weekChart"></div> <div style="height: 310px" id="weekChart"></div>
</el-card> </el-card>
</template> </template>
<script> <script>
@ -14,14 +13,15 @@ import * as echarts from 'echarts'
import resize from '@/mixins/ems/resize' import resize from '@/mixins/ems/resize'
import DateRangeSelect from '@/components/Ems/DateRangeSelect/index.vue' import DateRangeSelect from '@/components/Ems/DateRangeSelect/index.vue'
import {getSevenChargeData} from '@/api/ems/dzjk' import {getSevenChargeData} from '@/api/ems/dzjk'
export default { export default {
mixins: [resize], mixins: [resize],
components: {DateRangeSelect}, components: {DateRangeSelect},
data() { data() {
return { return {
chart: null, chart: null,
timeRange:[], timeRange: [],
siteId:'', siteId: '',
} }
}, },
mounted() { mounted() {
@ -38,41 +38,40 @@ export default {
}, },
methods: { methods: {
// 更新时间范围 重置图表 // 更新时间范围 重置图表
updateDate(data){ updateDate(data) {
this.timeRange=data this.timeRange = data
this.getWeekKData() this.getWeekKData()
}, },
getWeekKData(){ getWeekKData() {
this.showLoading() this.showLoading()
const {siteId,timeRange}=this const {siteId, timeRange} = this
this.hideLoading() getSevenChargeData({siteId, startDate: timeRange[0], endDate: timeRange[1]}).then(response => {
getSevenChargeData({siteId,startDate:timeRange[0],endDate:timeRange[1]}).then(response => {
this.setOption(response?.data || []) this.setOption(response?.data || [])
}).finally(()=>this.hideLoading()) }).finally(() => this.hideLoading())
}, },
init(siteId){ init(siteId) {
//初始化 清空数据 //初始化 清空数据
this.siteId = siteId this.siteId = siteId
this.timeRange=[] this.timeRange = []
this.deviceId='' this.deviceId = ''
this.$refs.dateRangeSelect.init() this.$refs.dateRangeSelect.init()
}, },
initChart() { initChart() {
this.chart = echarts.init(document.querySelector('#weekChart')) this.chart = echarts.init(document.querySelector('#weekChart'))
}, },
showLoading(){ showLoading() {
this.chart && this.chart.showLoading() this.chart && this.chart.showLoading()
}, },
hideLoading(){ hideLoading() {
this.chart && this.chart.hideLoading() this.chart && this.chart.hideLoading()
}, },
setOption(data,unit) { setOption(data, unit) {
const source = [['日期','充电量','放电量']] const source = [['日期', '充电量', '放电量']]
data.forEach(item=>{ data.forEach(item => {
source.push([item.ammeterDate, item.chargedCap,item.disChargedCap]) source.push([item.ammeterDate, item.chargedCap, item.disChargedCap])
}) })
this.chart && this.chart.setOption({ this.chart && this.chart.setOption({
color:['#4472c4','#70ad47'],//所有充放电颜色保持统一 color: ['#4472c4', '#70ad47'],//所有充放电颜色保持统一
tooltip: { tooltip: {
trigger: 'axis', trigger: 'axis',
axisPointer: { // 坐标轴指示器,坐标轴触发有效 axisPointer: { // 坐标轴指示器,坐标轴触发有效
@ -88,29 +87,29 @@ export default {
}, },
xAxis: { xAxis: {
type: 'category', type: 'category',
name:unit, name: unit,
nameLocation:'center' nameLocation: 'center'
}, },
yAxis: [{ yAxis: [{
type: 'value', type: 'value',
name:'充电量/放电量kWh', name: '充电量/放电量kWh',
axisLine: { axisLine: {
lineStyle:{ lineStyle: {
color: '#333333', color: '#333333',
}, },
onZero:false onZero: false
} }
}], }],
dataset:{ dataset: {
source source
}, },
series: [ series: [
{ {
yAxisIndex:0, yAxisIndex: 0,
type: 'bar', type: 'bar',
}, },
{ {
yAxisIndex:0, yAxisIndex: 0,
type: 'bar', type: 'bar',
}, },
] ]

View File

@ -4,130 +4,164 @@
v-loading="loading" v-loading="loading"
> >
<el-row style="background: #fff" class="row-container" :gutter="15"> <el-row style="background: #fff" class="row-container" :gutter="15">
<el-col v-if="tableData.length > 0" :xs="24" :sm="24" :lg="24"> <el-col :xs="24" :sm="24" :lg="5">
<alarm-table :tableData="tableData" /> <!-- 站点信息-->
</el-col>
<el-col :xs="24" :sm="24" :lg="6">
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container common-card-container-body-no-padding" class="common-card-container common-card-container-body-no-padding"
> >
<div slot="header"> <div slot="header">
<span class="card-title">站点信息</span> <span class="card-title">站点信息</span>
<div class="alarm-msg" v-if="tableData.length > 0" @click="toAlarm">
<i class="el-icon-message-solid"></i> 设备告警
</div>
</div> </div>
<div <div
style="box-sizing: border-box; height: 250px; padding: 20px 15px" style="box-sizing: border-box; height: 218px; padding: 20px 15px"
> >
<el-descriptions class="home-normal-info" :column="1"> <!-- 地址运行时间-->
<el-descriptions-item <div class="site-info site-info-address">
size="mini" <div class="title">
v-for="(item, index) in singleZdInfo" <i class="el-icon-location"></i>
:key="index + 'singleZdInfo'" </div>
:label="item.title" <div class="value">{{ info.siteAddress }}</div>
>{{ info[item.attr] | formatNumber }}</el-descriptions-item </div>
> <div class="site-info">
</el-descriptions> <div class="title">
</div> <i class="el-icon-date"></i>
</el-card> </div>
</el-col> <div class="value">{{ info.runningTime || '-' }}</div>
<el-col :xs="24" :sm="24" :lg="8"> </div>
<el-card <!-- 装机功率容量 -->
shadow="always" <el-row :gutter="10" style="margin-top:20px;">
class="common-card-container common-card-container-body-no-padding"
>
<div slot="header">
<span class="card-title">总累计运行数据</span>
</div>
<div
style="box-sizing: border-box; height: 250px; padding: 20px 15px"
>
<el-row :gutter="20">
<el-col <el-col
:span="12" :span="12"
v-for="(item, index) in sjglData" class="sjgl-col power-col"
:key="index + 'sjglData'"
class="sjgl-data"
> >
<div class="sjgl-title">{{ item.title }}</div> <div class="sjgl-wrapper">
<div class="sjgl-value"> <div class="sjgl-title">装机功率(MW)</div>
{{ runningInfo[item.attr] | formatNumber }} <div class="sjgl-value">
{{ info.installPower | formatNumber }}
</div>
</div>
</el-col>
<el-col
:span="12"
class="sjgl-col power-col"
>
<div class="sjgl-wrapper">
<div class="sjgl-title">装机容量(MW)</div>
<div class="sjgl-value">
{{ info.installCapacity | formatNumber }}
</div>
</div> </div>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
</el-card> </el-card>
</el-col> </el-col>
<el-col :xs="24" :sm="24" :lg="10"> <!-- 总累计运行数据-->
<cl-info :info="runningInfo.strategyTempInfo" /> <el-col :xs="24" :sm="24" :lg="19">
<el-card
shadow="always"
class="common-card-container common-card-container-body-no-padding"
>
<div slot="header">
<span class="card-title">总累计运行数据</span>
<div class="total-count">
<span class="title">总收入</span>
<span class="value">{{ runningInfo.totalRevenue | formatNumber }}</span>
<span class="unit"></span>
</div>
</div>
<div
style="box-sizing: border-box; height: 218px; padding: 20px 15px"
>
<el-row :gutter="10">
<el-col
:span="6"
v-for="(item, index) in sjglData"
:key="index + 'sjglData'"
class="sjgl-col"
>
<div class="sjgl-wrapper">
<div class="sjgl-title">{{ item.title }}</div>
<div class="sjgl-value" :style="{color:item.color}">
{{ runningInfo[item.attr] | formatNumber }}
</div>
</div>
</el-col>
</el-row>
</div>
</el-card>
</el-col> </el-col>
<el-col :xs="24" :sm="24" :lg="24"> <el-col :xs="24" :sm="24" :lg="12">
<week-chart ref="weekChart" /> <week-chart ref="weekChart"/>
</el-col> </el-col>
<el-col :xs="24" :sm="24" :lg="24"> <el-col :xs="24" :sm="24" :lg="12">
<active-chart ref="activeChart" /> <active-chart ref="activeChart"/>
</el-col> </el-col>
</el-row> </el-row>
</div> </div>
</template> </template>
<script> <script>
import { getSingleSiteBaseInfo } from "@/api/ems/zddt"; import {getSingleSiteBaseInfo} from "@/api/ems/zddt";
import { getDzjkHomeView } from "@/api/ems/dzjk"; import {getDzjkHomeView} from "@/api/ems/dzjk";
import WeekChart from "./WeekChart.vue"; import WeekChart from "./WeekChart.vue";
import ActiveChart from "./ActiveChart.vue"; import ActiveChart from "./ActiveChart.vue";
import AlarmTable from "./AlarmTable.vue"; import AlarmTable from "./AlarmTable.vue";
import ClInfo from "./ClInfo.vue"; import ClInfo from "./ClInfo.vue";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
export default { export default {
name: "DzjkSbjkHome", name: "DzjkSbjkHome",
components: { WeekChart, ActiveChart, AlarmTable, ClInfo }, components: {WeekChart, ActiveChart, AlarmTable, ClInfo},
mixins: [getQuerySiteId, intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
data() { data() {
return { return {
loading: false, loading: false,
singleZdInfo: [
{
title: "电站位置",
attr: "siteAddress",
},
{
title: "投运时间",
attr: "runningTime",
},
{
title: "装机功率(MW)",
attr: "installPower",
},
{
title: "装机容量(MW)",
attr: "installCapacity",
},
],
sjglData: [ sjglData: [
{ {
title: "今日充电量kWh", title: "今日充电量kWh",
attr: "dayChargedCap", attr: "dayChargedCap",
color: '#4472c4'
}, },
{ {
title: "今日放电量kWh", title: "今日放电量kWh",
attr: "dayDisChargedCap", attr: "dayDisChargedCap",
color: '#70ad47'
}, },
{ {
title: "总充电量kWh", title: "总充电量kWh",
attr: "totalChargedCap", attr: "totalChargedCap",
color: '#4472c4'
},
{
title: "今日实时收入(元)",
attr: "dayRevenue",
color: '#f67438'
},
{
title: "昨日充电量kWh",
attr: "yesterdayChargedCap",
color: '#4472c4'
},
{
title: "昨日放电量kWh",
attr: "yesterdayDisChargedCap",
color: '#70ad47'
}, },
{ {
title: "总放电量kWh", title: "总放电量kWh",
attr: "totalDischargedCap", attr: "totalDischargedCap",
color: '#70ad47'
}, },
{ {
title: "收入(元)", title: "昨日实时收入(元)",
attr: "totalRevenue", attr: "yesterdayRevenue",
}, color: '#f67438'
{
title: "当日实时收入(元)",
attr: "dayRevenue",
}, },
], ],
info: {}, //基本信息 info: {}, //基本信息
@ -136,14 +170,13 @@ export default {
}, },
computed: { computed: {
tableData() { tableData() {
console.log(
"this.runningInfo?.siteMonitorHomeAlarmVo ",
this.runningInfo?.siteMonitorHomeAlarmVo
);
return this.runningInfo?.siteMonitorHomeAlarmVo || []; return this.runningInfo?.siteMonitorHomeAlarmVo || [];
}, },
}, },
methods: { methods: {
toAlarm() {
this.$router.push({path: "/dzjk/gzgj", query: this.$route.query});
},
getBaseInfo() { getBaseInfo() {
return getSingleSiteBaseInfo(this.siteId).then((response) => { return getSingleSiteBaseInfo(this.siteId).then((response) => {
this.info = response?.data || {}; this.info = response?.data || {};
@ -173,6 +206,73 @@ export default {
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
//设备告警
.alarm-msg {
background: #ff4949;
padding: 2px 5px;
font-size: 10px;
font-weight: bolder;
border-radius: 3px;
line-height: 20px;
cursor: pointer;
color: #fff;
position: absolute;
right: 10px;
top: 50%;
transform: translateY(-50%);
}
//基本信息-地址 运行️时间
.site-info {
display: flex;
font-size: 12px;
line-height: 20px;
margin-bottom: 10px;
align-items: center;
&.site-info-address {
height: 40px;
}
.title {
color: #1791ff;
font-size: 18px;
line-height: 20px;
margin-right: 7px;
}
.value {
color: #000;
font-size: 12px;
max-height: 40px;
overflow: hidden;
}
}
//总收入
.total-count {
position: absolute;
right: 10px;
top: 50%;
transform: translateY(-50%);
font-size: 12px;
font-weight: bolder;
color: #333;
line-height: 14px;
.unit {
font-style: italic;
}
.value {
font-size: 22px;
font-weight: bolder;
color: #ed2f1d;
font-style: italic;
padding: 0 5px;
}
}
.row-container { .row-container {
& > .el-col { & > .el-col {
margin-bottom: 20px; margin-bottom: 20px;
@ -180,25 +280,47 @@ export default {
} }
//数据概览 //数据概览
.sjgl-data { .sjgl-col {
text-align: center; .sjgl-wrapper {
&:nth-child(1), text-align: left;
padding: 15px 20px;
background-color: #f2f7fb;
}
&.power-col {
.sjgl-wrapper {
padding: 10px;
.sjgl-value {
color: #c44444;
}
}
}
&:nth-child(4),
&:nth-child(2), &:nth-child(2),
&:nth-child(3), &:nth-child(3),
&:nth-child(4) { &:nth-child(4) {
margin-bottom: 25px; margin-bottom: 10px;
} }
.sjgl-title { .sjgl-title {
color: #666666; color: #717171;
line-height: 14px; line-height: 14px;
font-weight: bold;
} }
.sjgl-value { .sjgl-value {
color: rgba(51, 51, 51, 1); color: rgba(51, 51, 51, 1);
font-size: 26px; font-size: 22px;
line-height: 26px; line-height: 26px;
font-weight: 500; font-weight: bolder;
font-style: italic;
margin-top: 14px; margin-top: 14px;
word-wrap: break-word; width: 100%;
overflow: hidden;
text-overflow: ellipsis;
white-space: nowrap;
} }
} }
</style> </style>
@ -206,10 +328,12 @@ export default {
<style lang="scss"> <style lang="scss">
.home-normal-info { .home-normal-info {
font-size: 12px; font-size: 12px;
.el-descriptions-item__container { .el-descriptions-item__container {
.el-descriptions-item__label { .el-descriptions-item__label {
color: #666666; color: #666666;
} }
.el-descriptions-item__content { .el-descriptions-item__content {
color: #333333; color: #333333;
} }

View File

@ -68,6 +68,7 @@ export default {
<style scoped lang="scss"> <style scoped lang="scss">
.dzjk-ems-content-container{ .dzjk-ems-content-container{
margin-top:0; margin-top:0;
min-height: 60vh;
} }
.lighting{ .lighting{
position: relative; position: relative;

View File

@ -1,20 +1,20 @@
<!--电位展示图表--> <!--电位展示图表-->
<template> <template>
<el-dialog <el-dialog
:visible.sync="show" :visible.sync="show"
:title="pointName" :title="pointName"
:close-on-click-modal="false" :close-on-click-modal="false"
show-close show-close
destroy-on-close destroy-on-close
lock-scroll lock-scroll
append-to-body append-to-body
width="1000px" width="1000px"
class="ems-dialog" class="ems-dialog"
:before-close="handleClosed" :before-close="handleClosed"
> >
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container common-card-container-body-no-padding time-range-card" class="common-card-container common-card-container-body-no-padding time-range-card"
> >
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<el-radio-group class="card-title" v-model="dataUnit"> <el-radio-group class="card-title" v-model="dataUnit">
@ -23,10 +23,10 @@
<el-radio :label="3"></el-radio> <el-radio :label="3"></el-radio>
</el-radio-group> </el-radio-group>
<date-time-select <date-time-select
ref="dateTimeSelect" ref="dateTimeSelect"
:data-unit="dataUnit" :data-unit="dataUnit"
@initDate="(e) => (dataRange = e || [])" @initDate="(e) => (dataRange = e || [])"
@updateDate="updateDate" @updateDate="updateDate"
/> />
</div> </div>
<div style="height: 350px" id="searchChart"></div> <div style="height: 350px" id="searchChart"></div>
@ -37,10 +37,11 @@
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import DateTimeSelect from "@/views/ems/search/DateTimeSelect.vue"; import DateTimeSelect from "@/views/ems/search/DateTimeSelect.vue";
import { getPointValueList } from "@/api/ems/search"; import {getPointValueList} from "@/api/ems/search";
import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue"; import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue";
export default { export default {
components: { DateRangeSelect, DateTimeSelect }, components: {DateRangeSelect, DateTimeSelect},
mixins: [resize], mixins: [resize],
props: { props: {
siteId: { siteId: {
@ -93,7 +94,7 @@ export default {
}; };
}, },
methods: { methods: {
showChart({ pointName, deviceCategory, deviceId, child = "" }) { showChart({pointName, deviceCategory, deviceId, child = ""}) {
//初始化数据 //初始化数据
this.pointName = pointName; this.pointName = pointName;
this.deviceCategory = deviceCategory; this.deviceCategory = deviceCategory;
@ -152,84 +153,94 @@ export default {
endDate, endDate,
siteDeviceMap, siteDeviceMap,
}) })
.then((response) => { .then((response) => {
this.setOption(response?.data || []); this.setOption(response?.data || []);
}) })
.finally(() => { .finally(() => {
this.hideLoading(); this.hideLoading();
}); });
}, },
setOption(data) { setOption(data) {
if (!this.chart) return; if (!this.chart) return;
this.chart.clear(); this.chart.clear();
console.log("返回的数据", data); console.log("返回的数据", data);
let dataset = []; if (!data || data.length <= 0) {
if (data.length > 0) {
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}${inner.deviceId}`
: `${inner.deviceId}`
}`,
type: "line",
markPoint: {
symbolSize: 30,
emphasis: {
disabled:false//打开 鼠标高亮
},
data: [//最大值、最小值
{
// type: 'max',
name: `最大值`,
coord:[inner.maxDate,inner.maxValue],
relativeTo:'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
},
},
{
// type: 'min',
name: `最小值`,
coord:[inner.minDate,inner.minValue],
relativeTo:'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
}
}
]
},
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
dataset[length - 1].xdata.push(value.valueDate);
dataset[length - 1].data.push(value.pointValue);
});
});
});
} else {
this.$message.warning("暂无数据"); this.$message.warning("暂无数据");
} }
console.log("图表数据", dataset); console.log('展示的图表类型chartType', data[0].chartType)
if (data[0].chartType === 2) {
// 箱型图
this.setBoxOption(data)
} else {
//折线图
this.setLineOption(data)
}
},
setLineOption(data) {
let dataset = [];
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}${inner.deviceId}`
: `${inner.deviceId}`
}`,
type: "line",
markPoint: {
symbolSize: 30,
emphasis: {
disabled: false//打开 鼠标高亮
},
data: [//最大值、最小值
{
// type: 'max',
name: `最大值`,
coord: [inner.maxDate, inner.maxValue],
relativeTo: 'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
},
},
{
// type: 'min',
name: `最小值`,
coord: [inner.minDate, inner.minValue],
relativeTo: 'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
}
}
]
},
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
dataset[length - 1].xdata.push(value.valueDate);
dataset[length - 1].data.push(value.pointValue);
});
});
});
console.log("折线图图表数据", dataset);
this.chart.setOption({ this.chart.setOption({
legend: { legend: {
// left: 'center', // left: 'center',
@ -241,14 +252,17 @@ export default {
tooltip: { tooltip: {
trigger: "axis", trigger: "axis",
axisPointer: { axisPointer: {
// 坐标轴指示器,坐标轴触发有效 type: 'cross',
type: "cross", // 默认为直线,可选为:'line' | 'shadow'
}, },
// axisPointer: {
// // 坐标轴指示器,坐标轴触发有效
// type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
// },
}, },
textStyle: { textStyle: {
color: "#333333", color: "#333333",
}, },
xAxis: { type: "category", data: dataset?.[0]?.xdata || [] }, xAxis: {type: "category", data: dataset?.[0]?.xdata || []},
yAxis: { yAxis: {
type: "value", type: "value",
}, },
@ -266,6 +280,122 @@ export default {
series: dataset, series: dataset,
}); });
}, },
setBoxOption(data) {
let dataset = [];
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}${inner.deviceId}`
: `${inner.deviceId}`
}`,
type: "boxplot",
// markPoint: {
// symbolSize: 30,
// emphasis: {
// disabled: false//打开 鼠标高亮
// },
// data: [//最大值、最小值
// {
// // type: 'max',
// name: `最大值`,
// coord: [inner.maxDate, inner.maxValue],
// relativeTo: 'coordinate',
// label: {
// position: "top",
// formatter: item.dataType === 2 ? ([
// `最大值:${inner.maxValue}`,
// // `平均值:${inner.avgValue}`,
// `差值:${inner.diffValue}`,
// ]).join('\n') : ([
// `最大值:${inner.maxValue}`,
// // `平均值:${inner.avgValue}`,
// ]).join('\n'),
// },
// },
// {
// // type: 'min',
// name: `最小值`,
// coord: [inner.minDate, inner.minValue],
// relativeTo: 'coordinate',
// label: {
// position: "top",
// formatter: item.dataType === 2 ? ([
// `最小值:${inner.minValue}`,
// // `平均值:${inner.avgValue}`,
// `差值:${inner.diffValue}`,
// ]).join('\n') : ([
// `最小值:${inner.minValue}`,
// // `平均值:${inner.avgValue}`,
// ]).join('\n'),
// }
// }
// ]
// },
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
const {valueDate, min, q1, median, q3, max} = value
// const mid = (max - min) / 2, minLine = min + Math.abs(median / 2),
// maxLine = max - Math.abs(median / 2)
dataset[length - 1].xdata.push(valueDate);
dataset[length - 1].data.push([min, q1, median, q3, max]);
});
});
});
console.log("箱型图图表数据", dataset);
this.chart.setOption({
legend: {
// left: 'center',
// top: '10',
},
grid: {
containLabel: true,
},
tooltip: {
trigger: 'item',
formatter: function (params) {
let data = params.data;
let result = params.marker + params.name + ' ' + params.seriesName + '<br/>';
result += '最小值: ' + data[1] + '<br/>';
result += '平均值: ' + data[3] + '<br/>';
result += '最大值: ' + data[5];
return result;
}
// trigger: "axis",
// axisPointer: {
// type: 'cross',
// },
// axisPointer: {
// // 坐标轴指示器,坐标轴触发有效
// type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
// },
},
textStyle: {
color: "#333333",
},
xAxis: {type: "category", data: dataset?.[0]?.xdata || []},
yAxis: {
type: "value",
},
dataZoom: [
{
type: "inside",
start: 0,
end: 100,
},
{
start: 0,
end: 100,
},
],
series: dataset,
});
},
updateDate(val) { updateDate(val) {
this.dataRange = val || []; this.dataRange = val || [];
this.getDate(); this.getDate();

View File

@ -1,142 +1,199 @@
<template> <template>
<div v-loading="loading"> <div v-loading="loading">
<div v-for="(baseInfo,index) in baseInfoList" :key="index+'bmsdccContainer'" style="margin-bottom:25px;"> <div v-for="(baseInfo,index) in baseInfoList" :key="index+'bmsdccContainer'" style="margin-bottom:25px;">
<el-card shadow="always" class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg" <el-card shadow="always"
:class="{ class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg"
'warning-card-container':baseInfo.workStatus && baseInfo.workStatus !== '0', :class="handleCardClass(baseInfo)">
'running-card-container':baseInfo.workStatus === '0'
}">
<div slot="header"> <div slot="header">
<span class="large-title">{{index+1}}#{{baseInfo.parentDeviceName?`${baseInfo.parentDeviceName} —> ` : ''}}{{baseInfo.deviceName}}</span> <span
</div> class="large-title">{{
<div class="descriptions-main"> baseInfo.parentDeviceName ? `${baseInfo.parentDeviceName} -> ` : ''
<el-descriptions direction="vertical" :column="3" :colon="false"> }}{{ baseInfo.deviceName }}</span>
<el-descriptions-item labelClassName="descriptions-label" :contentClassName="`descriptions-direction ${baseInfo.workStatus === '0' ? 'save' :'danger'}`" :span="1" label="工作状态" >{{$store.state.ems.workStatusOptions[baseInfo.workStatus]}}</el-descriptions-item> <div class="info">
<el-descriptions-item labelClassName="descriptions-label" contentClassName="descriptions-direction" :span="1" label="与PCS通信">{{$store.state.ems.communicationStatusOptions[baseInfo.pcsCommunicationStatus]}}</el-descriptions-item> <div>数据更新时间{{ baseInfo.dataUpdateTime || '-' }}</div>
<el-descriptions-item labelClassName="descriptions-label" contentClassName="descriptions-direction" :span="1" label="与EMS通信">{{$store.state.ems.communicationStatusOptions[baseInfo.emsCommunicationStatus]}}</el-descriptions-item>
</el-descriptions>
</div> </div>
<div class="descriptions-main descriptions-main-bg-color"> <div class="alarm">
<el-descriptions direction="vertical" :column="3" :colon="false"> <el-button type="primary" round size="small" style="margin-right:20px;"
<el-descriptions-item labelClassName="descriptions-label" contentClassName="descriptions-direction" v-for="(item,index) in infoData" :key="index+'pcsInfoData'" :span="1" :label="item.label"> @click="pointDetail(baseInfo,'point')">详细
</el-button>
<el-badge :hidden="!baseInfo.alarmNum" :value="baseInfo.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(baseInfo,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<div class="descriptions-main">
<el-descriptions direction="vertical" :column="3" :colon="false">
<el-descriptions-item
contentClassName="descriptions-direction work-status"
:span="1" label="工作状态">
{{ CLUSTERWorkStatusOptions[baseInfo.workStatus] }}
</el-descriptions-item>
<el-descriptions-item contentClassName="descriptions-direction"
:span="1" label="与PCS通信">
{{ $store.state.ems.communicationStatusOptions[baseInfo.pcsCommunicationStatus] }}
</el-descriptions-item>
<el-descriptions-item contentClassName="descriptions-direction"
:span="1" label="与EMS通信">
{{ $store.state.ems.communicationStatusOptions[baseInfo.emsCommunicationStatus] }}
</el-descriptions-item>
</el-descriptions>
</div>
<div class="descriptions-main descriptions-main-bg-color">
<el-descriptions direction="vertical" :column="3" :colon="false">
<el-descriptions-item labelClassName="descriptions-label" contentClassName="descriptions-direction"
v-for="(item,index) in infoData" :key="index+'pcsInfoData'" :span="1"
:label="item.label">
<span class="pointer" @click="showChart(item.pointName || '',baseInfo.deviceId)"> <span class="pointer" @click="showChart(item.pointName || '',baseInfo.deviceId)">
{{baseInfo[item.attr] | formatNumber}} <span v-if="item.unit" v-html="item.unit"></span> {{ baseInfo[item.attr] | formatNumber }} <span v-if="item.unit" v-html="item.unit"></span>
</span> </span>
</el-descriptions-item> </el-descriptions-item>
</el-descriptions> </el-descriptions>
<!-- 进度--> <!-- 进度-->
<div class="process-container"> <div class="process-container">
<div class="process-line-bg"> <div class="process-line-bg">
<div class="process-line" :style="{height:baseInfo.currentSoc+'%'}"></div> <div class="process-line" :style="{height:baseInfo.currentSoc+'%'}"></div>
</div> </div>
<div class="process pointer" @click="showChart( '当前SOC',baseInfo.deviceId)">当前SOC : {{baseInfo.currentSoc}}%</div> <div class="process pointer" @click="showChart( '当前SOC',baseInfo.deviceId)">当前SOC :
{{ baseInfo.currentSoc }}%
</div> </div>
</div> </div>
</div>
<el-table <el-table
class="common-table" class="common-table"
:data="baseInfo.batteryDataList" :data="baseInfo.batteryDataList"
stripe stripe
style="width: 100%;margin-top:25px;"> style="width: 100%;margin-top:25px;">
<el-table-column <el-table-column
prop="dataName" prop="dataName"
label="名称"> label="名称">
<template slot-scope="scope"> <template slot-scope="scope">
<span v-html="scope.row.dataName+''+unitObj[scope.row.dataName]+''"></span> <span v-html="scope.row.dataName+''+unitObj[scope.row.dataName]+''"></span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="avgData" prop="avgData"
label="单体平均值" label="单体平均值"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{scope.row.avgData}}</span> <span class="pointer"
@click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{
scope.row.avgData
}}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="minData" prop="minData"
label="单体最小值"> label="单体最小值">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{scope.row.minData}}</span> <span class="pointer"
@click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{
scope.row.minData
}}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="minDataID" prop="minDataID"
label="单体最小值ID"> label="单体最小值ID">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="maxData" prop="maxData"
label="单体最大值"> label="单体最大值">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer " @click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{scope.row.maxData}}</span> <span class="pointer "
@click="showChart( tablePointNameMap[scope.row.dataName+scope.column.label],baseInfo.deviceId)">{{
scope.row.maxData
}}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="maxDataID" prop="maxDataID"
label="单体最大值ID"> label="单体最大值ID">
</el-table-column> </el-table-column>
</el-table> </el-table>
</el-card> </el-card>
</div> </div>
<el-empty v-show="baseInfoList.length<=0" :image-size="200"></el-empty> <el-empty v-show="baseInfoList.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/> <point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div> </div>
</template> </template>
<script> <script>
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
import {getBMSBatteryCluster} from '@/api/ems/dzjk' import {getBMSBatteryCluster} from '@/api/ems/dzjk'
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
import {mapState} from "vuex";
export default { export default {
name:'DzjkSbjkBmsdcc', name: 'DzjkSbjkBmsdcc',
mixins:[getQuerySiteId,intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
components:{pointChart}, components: {PointTable, pointChart},
computed: {
...mapState({
CLUSTERWorkStatusOptions: state => state?.ems?.CLUSTERWorkStatusOptions || {},
})
},
data() { data() {
return { return {
loading:false, loading: false,
unitObj:{ unitObj: {
'电压':'V', '电压': 'V',
'温度':'&#8451;', '温度': '&#8451;',
'SOC':'%' 'SOC': '%'
}, },
tablePointNameMap:{ tablePointNameMap: {
'电压单体最小值':'最低单体电压', '电压单体最小值': '最低单体电压',
'电压单体平均值':'电压平均值', '电压单体平均值': '电压平均值',
'电压单体最大值':'最高单体电压', '电压单体最大值': '最高单体电压',
'温度单体最小值':'最低单体温度', '温度单体最小值': '最低单体温度',
'温度单体平均值':'平均单体温度', '温度单体平均值': '平均单体温度',
'温度单体最大值':'最高单体温度', '温度单体最大值': '最高单体温度',
'SOC单体最小值':'最低单体SOC', 'SOC单体最小值': '最低单体SOC',
'SOC单体平均值':'当前SOC', 'SOC单体平均值': '当前SOC',
'SOC单体最大值':'最高单体SOC', 'SOC单体最大值': '最高单体SOC',
}, },
baseInfoList:[], baseInfoList: [],
infoData:[ infoData: [
{label:'簇电压',attr:'clusterVoltage',unit:'V',pointName:'簇电压'}, {label: '簇电压', attr: 'clusterVoltage', unit: 'V', pointName: '簇电压'},
{label:'可充电量',attr:'chargeableCapacity',unit:'kWh',pointName:'可充电量'}, {label: '可充电量', attr: 'chargeableCapacity', unit: 'kWh', pointName: '可充电量'},
{label:'累计充电量',attr:'totalChargedCapacity',unit:'kWh',pointName:'累计充电量'}, {label: '累计充电量', attr: 'totalChargedCapacity', unit: 'kWh', pointName: '累计充电量'},
{label:'簇电流',attr:'clusterCurrent',unit:'A',pointName:'簇电流'}, {label: '簇电流', attr: 'clusterCurrent', unit: 'A', pointName: '簇电流'},
{label:'可放电量',attr:'dischargeableCapacity',unit:'kWh',pointName:'可放电量'}, {label: '可放电量', attr: 'dischargeableCapacity', unit: 'kWh', pointName: '可放电量'},
{label:'累计放电量',attr:'totalDischargedCapacity',unit:'kWh',pointName:'累计放电量'}, {label: '累计放电量', attr: 'totalDischargedCapacity', unit: 'kWh', pointName: '累计放电量'},
{label:'SOH',attr:'soh',unit:'%',pointName:'SOH'}, {label: 'SOH', attr: 'soh', unit: '%', pointName: 'SOH'},
{label:'平均温度',attr:'averageTemperature',unit:'&#8451;',pointName:'平均温度'}, {label: '平均温度', attr: 'averageTemperature', unit: '&#8451;', pointName: '平均温度'},
{label:'绝缘电阻',attr:'insulationResistance',unit:'&Omega;',pointName:'绝缘电阻'}, {label: '绝缘电阻', attr: 'insulationResistance', unit: '&Omega;', pointName: '绝缘电阻'},
], ],
} }
}, },
methods:{ methods: {
showChart(pointName,deviceId){ handleCardClass(item) {
pointName && this.$refs.pointChart.showChart({pointName,deviceCategory:'CLUSTER',deviceId}) const {workStatus = ''} = item
return !(Object.keys(this.CLUSTERWorkStatusOptions).includes(item.workStatus)) ? "timing-card-container" : workStatus === '9' ? 'warning-card-container' : 'running-card-container'
}, },
updateData(){ // 查看设备电位表格
pointDetail(row, dataType) {
const {siteId, deviceId} = row
this.$refs.pointTable.showTable({siteId, deviceId, deviceCategory: 'CLUSTER'}, dataType)
},
showChart(pointName, deviceId) {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory: 'CLUSTER', deviceId})
},
updateData() {
this.loading = true this.loading = true
getBMSBatteryCluster(this.siteId).then(response => { getBMSBatteryCluster(this.siteId).then(response => {
this.baseInfoList = JSON.parse(JSON.stringify(response?.data || [])); this.baseInfoList = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => {this.loading = false}) }).finally(() => {
this.loading = false
})
}, },
init(){ init() {
this.updateData() this.updateData()
this.updateInterval(this.updateData) this.updateInterval(this.updateData)
} }
@ -147,28 +204,32 @@ export default {
<style scoped lang="scss"> <style scoped lang="scss">
::v-deep { ::v-deep {
//描述列表样式 //描述列表样式
.descriptions-main{ .descriptions-main {
padding:24px 300px 24px 24px; padding: 24px 300px 24px 24px;
} }
.descriptions-main-bottom{
padding:14px 300px 14px 24px; .descriptions-main-bottom {
padding: 14px 300px 14px 24px;
} }
} }
// 进度条样式 // 进度条样式
.process-container{ .process-container {
width:100px; width: 100px;
position: absolute; position: absolute;
right:70px; right: 70px;
top:50%; top: 50%;
transform: translateY(-50%); transform: translateY(-50%);
.process-line-bg{
.process-line-bg {
position: relative; position: relative;
width:100%; width: 100%;
height: 110px; height: 110px;
background-color:#fff2cb ; background-color: #fff2cb;
border-radius: 6px; border-radius: 6px;
box-shadow: 0 0 10px #fff2cb, 0 0 0 rgba(255, 242, 203, 0.5); box-shadow: 0 0 10px #fff2cb, 0 0 0 rgba(255, 242, 203, 0.5);
.process-line{
.process-line {
position: absolute; position: absolute;
left: 0; left: 0;
bottom: 0; bottom: 0;
@ -179,9 +240,10 @@ export default {
box-shadow: 0 0 10px #ffbf14, 0 0 0 rgba(255, 191, 20, 0.5); box-shadow: 0 0 10px #ffbf14, 0 0 0 rgba(255, 191, 20, 0.5);
} }
} }
.process{
margin-top:15px; .process {
color:#666666; margin-top: 15px;
color: #666666;
} }
} }

View File

@ -1,27 +1,51 @@
<template> <template>
<div v-loading="loading"> <div v-loading="loading">
<div v-for="(baseInfo,index) in baseInfoList" :key="index+'bmszlContainer'" style="margin-bottom:25px;"> <div v-for="(baseInfo,index) in baseInfoList" :key="index+'bmszlContainer'" style="margin-bottom:25px;">
<el-card :class="{ <el-card
'warning-card-container':baseInfo.workStatus && baseInfo.workStatus !== '0', :class="handleCardClass(baseInfo)"
'running-card-container':baseInfo.workStatus === '0' class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg"
}" class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg" shadow="always">
shadow="always">
<div slot="header"> <div slot="header">
<span class="large-title">{{index+1}}#{{baseInfo.deviceName}}</span> <span class="large-title">{{ baseInfo.deviceName }}</span>
<div class="info">
<div>数据更新时间{{ baseInfo.dataUpdateTime || '-' }}</div>
</div>
<div class="alarm">
<el-button type="primary" round size="small" style="margin-right:20px;"
@click="pointDetail(baseInfo,'point')">详细
</el-button>
<el-badge :hidden="!baseInfo.alarmNum" :value="baseInfo.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(baseInfo,'alarmPoint')"
></i>
</el-badge>
</div>
</div> </div>
<div class="descriptions-main"> <div class="descriptions-main">
<el-descriptions :colon="false" :column="3" direction="vertical"> <el-descriptions :colon="false" :column="3" direction="vertical">
<el-descriptions-item :contentClassName="`descriptions-direction ${baseInfo.workStatus === '0' ? 'save' :'danger'}`" :span="1" label="工作状态" labelClassName="descriptions-label" >{{$store.state.ems.workStatusOptions[baseInfo.workStatus]}}</el-descriptions-item> <el-descriptions-item
<el-descriptions-item :span="1" contentClassName="descriptions-direction" label="与PCS通信" labelClassName="descriptions-label">{{$store.state.ems.communicationStatusOptions[baseInfo.pcsCommunicationStatus]}}</el-descriptions-item> contentClassName="descriptions-direction work-status"
<el-descriptions-item :span="1" contentClassName="descriptions-direction" label="与EMS通信" labelClassName="descriptions-label">{{$store.state.ems.communicationStatusOptions[baseInfo.emsCommunicationStatus]}}</el-descriptions-item> label="工作状态" labelClassName="descriptions-label">
{{ STACKWorkStatusOptions[baseInfo.workStatus] }}
</el-descriptions-item>
<el-descriptions-item :span="1" contentClassName="descriptions-direction" label="与PCS通信"
labelClassName="descriptions-label">
{{ $store.state.ems.communicationStatusOptions[baseInfo.pcsCommunicationStatus] }}
</el-descriptions-item>
<el-descriptions-item :span="1" contentClassName="descriptions-direction" label="与EMS通信"
labelClassName="descriptions-label">
{{ $store.state.ems.communicationStatusOptions[baseInfo.emsCommunicationStatus] }}
</el-descriptions-item>
</el-descriptions> </el-descriptions>
</div> </div>
<div class="descriptions-main descriptions-main-bg-color"> <div class="descriptions-main descriptions-main-bg-color">
<el-descriptions :colon="false" :column="3" direction="vertical"> <el-descriptions :colon="false" :column="3" direction="vertical">
<el-descriptions-item v-for="(item,index) in infoData" :key="index+'pcsInfoData'" :label="item.label" :span="1" contentClassName="descriptions-direction" labelClassName="descriptions-label"> <el-descriptions-item v-for="(item,index) in infoData" :key="index+'pcsInfoData'" :label="item.label"
<span class="pointer" @click="showChart(item.pointName || '',baseInfo.deviceId)"> :span="1" contentClassName="descriptions-direction"
{{baseInfo[item.attr] | formatNumber}}<span v-if="item.unit" v-html="item.unit"></span> labelClassName="descriptions-label">
<span class="pointer" @click="showChart(item.pointName || '',baseInfo.deviceId)">
{{ baseInfo[item.attr] | formatNumber }}<span v-if="item.unit" v-html="item.unit"></span>
</span> </span>
</el-descriptions-item> </el-descriptions-item>
</el-descriptions> </el-descriptions>
@ -30,80 +54,97 @@
<div class="process-line-bg"> <div class="process-line-bg">
<div :style="{height:baseInfo.stackSoc+'%'}" class="process-line"></div> <div :style="{height:baseInfo.stackSoc+'%'}" class="process-line"></div>
</div> </div>
<div class="process pointer" @click="showChart('当前SOC',baseInfo.deviceId)">当前SOC : {{baseInfo.stackSoc}}%</div> <div class="process pointer" @click="showChart('当前SOC',baseInfo.deviceId)">当前SOC :
{{ baseInfo.stackSoc }}%
</div>
</div> </div>
</div> </div>
<el-table <el-table
:data="baseInfo.batteryDataList" :data="baseInfo.batteryDataList"
class="common-table" class="common-table"
max-height="500" max-height="500"
stripe stripe
style="width: 100%;margin-top:25px;"> style="width: 100%;margin-top:25px;">
<el-table-column <el-table-column
label="簇号" label="簇号"
prop="clusterId"> prop="clusterId">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="簇电压" label="簇电压"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('簇电压',scope.row.clusterId,'CLUSTER')">{{scope.row.clusterVoltage}} V</span> <span class="pointer"
@click="showChart('簇电压',scope.row.clusterId,'CLUSTER')">{{ scope.row.clusterVoltage }} V</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="簇电流"> label="簇电流">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('簇电流',scope.row.clusterId,'CLUSTER')">{{scope.row.clusterCurrent}} A</span> <span class="pointer"
@click="showChart('簇电流',scope.row.clusterId,'CLUSTER')">{{ scope.row.clusterCurrent }} A</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="簇SOC"> label="簇SOC">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('当前SOC',scope.row.clusterId,'CLUSTER')">{{scope.row.currentSoc}} %</span> <span class="pointer"
@click="showChart('当前SOC',scope.row.clusterId,'CLUSTER')">{{ scope.row.currentSoc }} %</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="单体最高电压" label="单体最高电压"
prop="maxVoltage"> prop="maxVoltage">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('最高单体电压',scope.row.clusterId,'CLUSTER')">{{scope.row.maxCellVoltage}} V</span> <span class="pointer"
@click="showChart('最高单体电压',scope.row.clusterId,'CLUSTER')">{{
scope.row.maxCellVoltage
}} V</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="电池号码" label="电池号码"
prop="maxCellVoltageId"> prop="maxCellVoltageId">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="单体最低电压" label="单体最低电压"
prop="minVoltage"> prop="minVoltage">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('最低单体电压',scope.row.clusterId,'CLUSTER')">{{scope.row.minCellVoltage}} V</span> <span class="pointer"
@click="showChart('最低单体电压',scope.row.clusterId,'CLUSTER')">{{
scope.row.minCellVoltage
}} V</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="电池号码" label="电池号码"
prop="minCellVoltageId"> prop="minCellVoltageId">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="单体最高温度"> label="单体最高温度">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('最高单体温度',scope.row.clusterId,'CLUSTER')">{{scope.row.maxCellTemp}} &#8451;</span> <span class="pointer"
@click="showChart('最高单体温度',scope.row.clusterId,'CLUSTER')">{{
scope.row.maxCellTemp
}} &#8451;</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="电池号码" label="电池号码"
prop="maxCellTempId"> prop="maxCellTempId">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="单体最低温度" label="单体最低温度"
prop="minTemperature"> prop="minTemperature">
<template slot-scope="scope"> <template slot-scope="scope">
<span class="pointer" @click="showChart('最低单体温度',scope.row.clusterId,'CLUSTER')">{{scope.row.minCellTemp}} &#8451;</span> <span class="pointer"
@click="showChart('最低单体温度',scope.row.clusterId,'CLUSTER')">{{
scope.row.minCellTemp
}} &#8451;</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
label="电池号码" label="电池号码"
prop="minCellTempId"> prop="minCellTempId">
</el-table-column> </el-table-column>
</el-table> </el-table>
@ -111,6 +152,7 @@
</div> </div>
<el-empty v-show="baseInfoList.length<=0" :image-size="200"></el-empty> <el-empty v-show="baseInfoList.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/> <point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div> </div>
</template> </template>
@ -119,38 +161,58 @@ import {getBMSOverView} from '@/api/ems/dzjk'
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
import {mapState} from "vuex";
export default { export default {
name:'DzjkSbjkBmszl', name: 'DzjkSbjkBmszl',
components: {pointChart,}, components: {pointChart, PointTable},
mixins:[getQuerySiteId,intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
computed: {
...mapState({
STACKWorkStatusOptions: state => state?.ems?.STACKWorkStatusOptions || {},
})
},
data() { data() {
return { return {
loading:false, loading: false,
baseInfoList:[], baseInfoList: [],
infoData:[ infoData: [
{label:'电池堆总电压',attr:'stackVoltage',unit:'V',pointName:'电池堆电压'}, {label: '电池堆总电压', attr: 'stackVoltage', unit: 'V', pointName: '电池堆电压'},
{label:'可充电量',attr:'availableChargeCapacity',unit:'kWh',pointName:'可充电量'}, {label: '可充电量', attr: 'availableChargeCapacity', unit: 'kWh', pointName: '可充电量'},
{label:'累计充电量',attr:'totalChargeCapacity',unit:'kWh',pointName:'累计充电量'}, {label: '累计充电量', attr: 'totalChargeCapacity', unit: 'kWh', pointName: '累计充电量'},
{label:'电池堆总电流',attr:'stackCurrent',unit:'A',pointName:'电池堆总电流'}, {label: '电池堆总电流', attr: 'stackCurrent', unit: 'A', pointName: '电池堆总电流'},
{label:'可放电量',attr:'availableDischargeCapacity',unit:'kWh',pointName:'可放电量'}, {label: '可放电量', attr: 'availableDischargeCapacity', unit: 'kWh', pointName: '可放电量'},
{label:'累计放电量',attr:'totalDischargeCapacity',unit:'kWh',pointName:'累计放电量'}, {label: '累计放电量', attr: 'totalDischargeCapacity', unit: 'kWh', pointName: '累计放电量'},
{label:'SOH',attr:'stackSoh',unit:'%',pointName:'SOH'}, {label: 'SOH', attr: 'stackSoh', unit: '%', pointName: 'SOH'},
{label:'平均温度',attr:'operatingTemp',unit:'&#8451;',pointName:'平均温度'}, {label: '平均温度', attr: 'operatingTemp', unit: '&#8451;', pointName: '平均温度'},
{label:'绝缘电阻',attr:'stackInsulationResistance',unit:'&Omega;',pointName:'绝缘电阻'}, {label: '绝缘电阻', attr: 'stackInsulationResistance', unit: '&Omega;', pointName: '绝缘电阻'},
] ]
} }
}, },
methods:{ methods: {
showChart(pointName,deviceId,deviceCategory = 'STACK'){ handleCardClass(item) {
pointName && this.$refs.pointChart.showChart({pointName,deviceCategory,deviceId}) const {workStatus = ''} = item
return !Object.keys(this.STACKWorkStatusOptions).find(i => i === workStatus) ? "timing-card-container" : workStatus === '9' ? 'warning-card-container' : 'running-card-container'
}, },
updateData(){
// 查看设备电位表格
pointDetail(row, dataType) {
const {siteId, deviceId} = row
this.$refs.pointTable.showTable({siteId, deviceId, deviceCategory: 'STACK'}, dataType)
},
showChart(pointName, deviceId, deviceCategory = 'STACK') {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory, deviceId})
},
updateData() {
this.loading = true this.loading = true
getBMSOverView(this.siteId).then(response => { getBMSOverView(this.siteId).then(response => {
this.baseInfoList = JSON.parse(JSON.stringify(response?.data || [])); this.baseInfoList = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => {this.loading = false}) }).finally(() => {
this.loading = false
})
}, },
init(){ init() {
this.updateData() this.updateData()
this.updateInterval(this.updateData) this.updateInterval(this.updateData)
} }
@ -170,21 +232,24 @@ export default {
} }
} }
} }
// 进度条样式 // 进度条样式
.process-container{ .process-container {
width:100px; width: 100px;
position: absolute; position: absolute;
right:70px; right: 70px;
top:50%; top: 50%;
transform: translateY(-50%); transform: translateY(-50%);
.process-line-bg{
.process-line-bg {
position: relative; position: relative;
width:100%; width: 100%;
height: 110px; height: 110px;
background-color:#fff2cb ; background-color: #fff2cb;
border-radius: 6px; border-radius: 6px;
box-shadow: 0 0 10px #fff2cb, 0 0 0 rgba(255, 242, 203, 0.5); box-shadow: 0 0 10px #fff2cb, 0 0 0 rgba(255, 242, 203, 0.5);
.process-line{
.process-line {
position: absolute; position: absolute;
left: 0; left: 0;
bottom: 0; bottom: 0;
@ -195,9 +260,10 @@ export default {
box-shadow: 0 0 10px rgb(252 108 108), 0 0 0 rgba(252, 108, 108, 0.5); box-shadow: 0 0 10px rgb(252 108 108), 0 0 0 rgba(252, 108, 108, 0.5);
} }
} }
.process{
margin-top:15px; .process {
color:#666666; margin-top: 15px;
color: #666666;
} }
} }

View File

@ -1,149 +1,213 @@
<template> <template>
<div v-loading="loading"> <div v-loading="loading">
<el-card <el-card
v-for="(item,index) in list" v-for="(item,index) in list"
:key="index+'dbList'" :key="index+'dbList'"
shadow="always" shadow="always"
class="sbjk-card-container list" class="sbjk-card-container list"
:class="{ :class="{
'warning-card-container':item.emsCommunicationStatus && item.emsCommunicationStatus !== '0', 'timing-card-container':!['0','2'].includes(item.emsCommunicationStatus),
'warning-card-container':item.emsCommunicationStatus === '2',
'running-card-container':item.emsCommunicationStatus === '0' 'running-card-container':item.emsCommunicationStatus === '0'
}" }"
> >
<div slot="header"> <div slot="header">
<span class="large-title">{{index+1}}#{{ item.deviceName }}</span> <span class="large-title">{{ item.deviceName }}</span>
<div class="info"> <div class="info">
<div> <div>
{{ {{
$store.state.ems.communicationStatusOptions[ communicationStatusOptions[item.emsCommunicationStatus] || '-'
item.emsCommunicationStatus }}
] </div>
}} <div>数据更新时间{{ item.dataUpdateTime || '-' }}</div>
</div> </div>
<div>数据更新时间{{ item.dataUpdateTime }}</div> <div class="alarm">
</div> <el-button type="primary" round size="small" style="margin-right:20px;" @click="pointDetail(item,'point')">
</div> 详细
<el-row> </el-button>
<el-col v-for="(tempDataItem,tempDataIndex) in deviceIdTypeMsg[item.deviceId]" :key="tempDataIndex+'dbTempData'" :span="8"> <el-badge :hidden="!item.alarmNum" :value="item.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(item,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<el-row class="device-info-row">
<el-col v-for="(tempDataItem,tempDataIndex) in (deviceIdTypeMsg[item.deviceId] || otherTypeMsg)"
:key="tempDataIndex+'dbTempData'"
:span="8" class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.pointName,item.deviceId)"> <span class="pointer" @click="showChart(tempDataItem.pointName,item.deviceId)">
{{tempDataItem.name}}{{item[tempDataItem.attr]}}<span v-html="tempDataItem.unit"></span> <span class="left">{{ tempDataItem.name }}</span> <span class="right">{{ item[tempDataItem.attr] || '-' }}<span
v-html="tempDataItem.unit"></span></span>
</span> </span>
</el-col> </el-col>
</el-row> </el-row>
</el-card> </el-card>
<el-empty v-show="list.length<=0" :image-size="200"></el-empty> <el-empty v-show="list.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/> <point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div> </div>
</template> </template>
<script> <script>
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getAmmeterDataList } from "@/api/ems/dzjk"; import {getAmmeterDataList} from "@/api/ems/dzjk";
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
import {mapState} from "vuex";
export default { export default {
name: "DzjkSbjkDb", name: "DzjkSbjkDb",
mixins: [getQuerySiteId,intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
components:{pointChart}, components: {PointTable, pointChart},
computed: {
...mapState({
communicationStatusOptions: state => state?.ems?.communicationStatusOptions || {},
})
},
data() { data() {
return { return {
loading: false, loading: false,
list:[], list: [],
deviceIdTypeMsg:{ deviceIdTypeMsg: {
'LOAD':[ 'LOAD': [
{ {
name:'正向有功电能', name: '正向有功电能',
attr:'forwardActive', attr: 'forwardActive',
pointName:'正向有功电能' pointName: '正向有功电能',
unit: 'kWh'
}, },
{ {
name:'反向有功电能', name: '反向有功电能',
attr:'reverseActive', attr: 'reverseActive',
pointName:'反向有功电能' pointName: '反向有功电能',
unit: 'kWh'
}, },
{ {
name:'正向无功电能', name: '正向无功电能',
attr:'forwardReactive', attr: 'forwardReactive',
pointName:'正向无功电能' pointName: '正向无功电能',
unit: 'kvarh'
}, },
{ {
name:'反向无功电能', name: '反向无功电能',
attr:'reverseReactive', attr: 'reverseReactive',
pointName:'反向无功电能' pointName: '反向无功电能',
unit: 'kvarh'
}, },
{ {
name:'有功功率', name: '有功功率',
attr:'activePower', attr: 'activePower',
pointName:'总有功功率' pointName: '总有功功率',
unit: 'kW'
}, },
{ {
name:'无功功率', name: '无功功率',
attr:'reactivePower', attr: 'reactivePower',
pointName:'总无功功率' pointName: '总无功功率',
unit: 'kvar'
} }
], ],
'METE':[ 'METE': [
{ {
name:'正向有功电能', name: '正向有功电能',
attr:'forwardActive', attr: 'forwardActive',
pointName:'正向有功电能' pointName: '正向有功电能',
unit: 'kWh'
}, },
{ {
name:'反向有功电能', name: '反向有功电能',
attr:'reverseActive', attr: 'reverseActive',
pointName:'反向有功电能' pointName: '反向有功电能',
unit: 'kWh'
}, },
{ {
name:'正向无功电能', name: '正向无功电能',
attr:'forwardReactive', attr: 'forwardReactive',
pointName:'正向无功电能' pointName: '正向无功电能',
unit: 'kvarh'
}, },
{ {
name:'反向无功电能', name: '反向无功电能',
attr:'reverseReactive', attr: 'reverseReactive',
pointName:'反向无功电能' pointName: '反向无功电能',
unit: 'kvarh'
}, },
{ {
name:'有功功率', name: '有功功率',
attr:'activePower', attr: 'activePower',
pointName:'总有功功率' pointName: '总有功功率',
unit: 'kW'
}, },
{ {
name:'无功功率', name: '无功功率',
attr:'reactivePower', attr: 'reactivePower',
pointName:'总无功功率' pointName: '总无功功率',
unit: 'kvar'
} }
], ],
'METEGF':[ 'METEGF': [
{ {
name:'有功电能', name: '有功电能',
attr:'activeEnergy', attr: 'activeEnergy',
pointName:'有功电能' pointName: '有功电能',
unit: 'kWh'
}, },
{ {
name:'无功电能', name: '无功电能',
attr:'reactiveEnergy', attr: 'reactiveEnergy',
pointName:'无功电能' pointName: '无功电能',
unit: 'kvarh'
}, },
{ {
name:'有功功率', name: '有功功率',
attr:'activePower', attr: 'activePower',
pointName:'总有功功率' pointName: '总有功功率',
unit: 'kW'
}, },
{ {
name:'无功功率', name: '无功功率',
attr:'reactivePower', attr: 'reactivePower',
pointName:'总无功功率' pointName: '总无功功率',
unit: 'kvar'
} }
] ]
} },
otherTypeMsg: [
{
name: '正向有功电能',
attr: 'forwardActive',
pointName: '正向有功电能',
unit: 'kWh'
},
{
name: '反向有功电能',
attr: 'reverseActive',
pointName: '反向有功电能',
unit: 'kWh'
},
{
name: '有功功率',
attr: 'activePower',
pointName: '总有功功率',
unit: 'kW'
},
]
}; };
}, },
methods: { methods: {
showChart(pointName,deviceId){ // 查看设备电位表格
pointName && this.$refs.pointChart.showChart({pointName,deviceCategory:'AMMETER',deviceId}) pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'AMMETER'}, dataType)
}, },
updateData(){ showChart(pointName, deviceId) {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory: 'AMMETER', deviceId})
},
updateData() {
this.loading = true; this.loading = true;
getAmmeterDataList(this.siteId) getAmmeterDataList(this.siteId)
.then((response) => { .then((response) => {
@ -158,33 +222,16 @@ export default {
this.updateInterval(this.updateData) this.updateInterval(this.updateData)
}, },
}, },
mounted() {}, mounted() {
},
}; };
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
.sbjk-card-container { .sbjk-card-container {
&.list:not(:last-child){ &.list:not(:last-child) {
margin-bottom: 25px; margin-bottom: 25px;
} }
.el-row{
background-color: #ffffff;
border:1px solid #eeeeee;
font-size: 14px;
line-height: 16px;
color: #333333;
.el-col{
padding:12px 0;
text-align: center;
position: relative;
}
.el-col{
border-bottom: 1px solid #eeeeee;
}
.el-col:not(:nth-child(3n)){
border-right: 1px solid #eeeeee;
}
}
} }
</style> </style>

View File

@ -0,0 +1,120 @@
<template>
<div v-loading="loading">
<el-card
v-for="(item,index) in list"
:key="index+'ylLise'"
class="sbjk-card-container running-card-container"
shadow="always">
<div slot="header">
<span class="large-title">{{ item.deviceName }}</span>
<div class="info">
<div>数据更新时间{{ item.dataUpdateTime || '-' }}</div>
</div>
<div class="alarm">
<el-button type="primary" round size="small" style="margin-right:20px;" @click="pointDetail(item,'point')">
详细
</el-button>
<el-badge :hidden="!item.alarmNum" :value="item.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(item,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<el-row class="device-info-row">
<el-col v-for="(tempDataItem,tempDataIndex) in tempData" :key="tempDataIndex+'hdTempData'" :span="12"
class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.title,item.deviceId)">
<span class="left">{{ tempDataItem.title }}</span> <span
class="right">{{ item[tempDataItem.attr] || '-' }}<span
v-html="tempDataItem.unit"></span></span>
</span>
</el-col>
</el-row>
</el-card>
<el-empty v-show="list.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div>
</template>
<script>
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getDhDataList} from '@/api/ems/dzjk'
import intervalUpdate from "@/mixins/ems/intervalUpdate";
import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
export default {
name: 'DzjkSbjkDh',
mixins: [getQuerySiteId, intervalUpdate],
components: {pointChart, PointTable},
data() {
return {
loading: false,
list: [],
tempData: [
{title: '湿度', attr: 'humidity', unit: ''},
{title: '温度', attr: 'temperature', unit: '&#8451;'},
]
}
},
methods: {
// 查看设备电位表格
pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'DH'}, dataType)
},
showChart(pointName, deviceId) {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory: 'DH', deviceId})
},
updateData() {
this.loading = true
getDhDataList(this.siteId).then(response => {
this.list = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => {
this.loading = false
})
},
init() {
this.updateData()
this.updateInterval(this.updateData)
}
},
mounted() {
}
}
</script>
<style scoped lang="scss">
.sbjk-card-container {
&:not(:last-child) {
margin-bottom: 25px;
}
.el-row {
background-color: #ffffff;
border: 1px solid #eeeeee;
font-size: 14px;
line-height: 16px;
color: #333333;
.el-col {
padding: 12px 0;
text-align: center;
position: relative;
}
.el-col {
border-bottom: 1px solid #eeeeee;
}
.el-col:not(:nth-child(3n)) {
border-right: 1px solid #eeeeee;
}
}
}
</style>

View File

@ -1,19 +1,19 @@
<template> <template>
<div> <div>
<el-table <el-table
class="common-table" class="common-table"
:data="tableData" :data="tableData"
stripe stripe
style="width: 100%; margin-top: 25px" style="width: 100%; margin-top: 25px"
> >
<el-table-column prop="deviceId" label="单体编号"> </el-table-column> <el-table-column prop="deviceId" label="单体编号"></el-table-column>
<el-table-column prop="clusterDeviceId" label="簇号"> </el-table-column> <el-table-column prop="clusterDeviceId" label="簇号"></el-table-column>
<el-table-column prop="voltage" label="电压 (V)"> <el-table-column prop="voltage" label="电压 (V)">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click="chartDetail(scope.row, '电压 (V)')" @click="chartDetail(scope.row, '电压 (V)')"
type="text" type="text"
size="small" size="small"
> >
{{ scope.row.voltage }} {{ scope.row.voltage }}
</el-button> </el-button>
@ -22,9 +22,9 @@
<el-table-column prop="temperature" label="温度 (℃)"> <el-table-column prop="temperature" label="温度 (℃)">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click="chartDetail(scope.row, '温度 (℃)')" @click="chartDetail(scope.row, '温度 (℃)')"
type="text" type="text"
size="small" size="small"
> >
{{ scope.row.temperature }} {{ scope.row.temperature }}
</el-button> </el-button>
@ -33,9 +33,9 @@
<el-table-column prop="soc" label="SOC (%)"> <el-table-column prop="soc" label="SOC (%)">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click="chartDetail(scope.row, 'SOC (%)')" @click="chartDetail(scope.row, 'SOC (%)')"
type="text" type="text"
size="small" size="small"
> >
{{ scope.row.soc }} {{ scope.row.soc }}
</el-button> </el-button>
@ -44,21 +44,24 @@
<el-table-column prop="soh" label="SOH (%)"> <el-table-column prop="soh" label="SOH (%)">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click="chartDetail(scope.row, 'SOH (%)')" @click="chartDetail(scope.row, 'SOH (%)')"
type="text" type="text"
size="small" size="small"
> >
{{ scope.row.soh }} {{ scope.row.soh }}
</el-button> </el-button>
</template> </template>
</el-table-column> </el-table-column>
<!-- <el-table-column label="曲线图"> <el-table-column label="操作" width="160">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button @click="chartDetail(scope.row)" type="text" size="small"> <el-button @click="$emit('pointDetail',scope.row,'point')" type="text" size="small">
展示 详细
</el-button>
<el-button @click="$emit('pointDetail',scope.row,'alarmPoint')" type="text" size="small">
报警点位详细
</el-button> </el-button>
</template> </template>
</el-table-column> --> </el-table-column>
</el-table> </el-table>
<!-- <el-pagination <!-- <el-pagination
v-show="tableData.length > 0" v-show="tableData.length > 0"
@ -115,8 +118,7 @@ export default {
methods: { methods: {
//查看表格行图表 //查看表格行图表
chartDetail(row, dataType = "") { chartDetail(row, dataType = "") {
const { clusterDeviceId, deviceId } = row; this.$emit("chart", {...row, dataType});
this.$emit("chart", { ...row, dataType });
}, },
}, },
}; };

View File

@ -88,6 +88,7 @@
:totalSize="totalSize" :totalSize="totalSize"
:pointIdList="pointIdList" :pointIdList="pointIdList"
@chart="chartDetail" @chart="chartDetail"
@pointDetail="pointDetail"
></component> ></component>
<el-pagination <el-pagination
v-show="tableData.length > 0" v-show="tableData.length > 0"
@ -104,6 +105,7 @@
</el-pagination> </el-pagination>
<chart-detail ref="chartDetail" /> <chart-detail ref="chartDetail" />
<point-chart ref="pointChart" :site-id="siteId" /> <point-chart ref="pointChart" :site-id="siteId" />
<point-table ref="pointTable"/>
</el-card> </el-card>
</template> </template>
@ -119,10 +121,12 @@ import ChartDetail from "./ChartDetail.vue";
import Table from "./Table.vue"; import Table from "./Table.vue";
import List from "./List.vue"; import List from "./List.vue";
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
export default { export default {
name: "DzjkSbjkDtdc", name: "DzjkSbjkDtdc",
mixins: [getQuerySiteId], mixins: [getQuerySiteId],
components: { components: {
PointTable,
BarChart, BarChart,
ChartDetail, ChartDetail,
DtdcTable: Table, DtdcTable: Table,
@ -165,6 +169,11 @@ export default {
}; };
}, },
methods: { methods: {
// 查看设备电位表格
pointDetail(row,dataType){
const {deviceId,clusterDeviceId} = row
this.$refs.pointTable.showTable({siteId:this.siteId,deviceId,deviceCategory:'BATTERY',parentId:clusterDeviceId},dataType)
},
changeMenu(menu) { changeMenu(menu) {
const { activeBtn } = this; const { activeBtn } = this;
activeBtn !== menu && (this.activeBtn = menu); activeBtn !== menu && (this.activeBtn = menu);

View File

@ -0,0 +1,133 @@
<template>
<div v-loading="loading" class="ems">
<el-card
v-for="(item,index) in list"
:key="index+'emsList'"
class="sbjk-card-container list running-card-container"
shadow="always"
>
<div slot="header">
<span class="large-title">{{ item.deviceName }}</span>
<div class="info">
<div>
EMS控制模式: {{
item.emsStatus === 0 ? '自动' : '手动'
}}
</div>
<div>数据更新时间{{ item.dataUpdateTime }}</div>
</div>
<div class="alarm">
<el-button size="small" round style="margin-right:20px;" type="primary" @click="pointDetail(item,'point')">详细
</el-button>
<el-badge :hidden="!item.alarmNum" :value="item.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(item,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<el-row class="device-info-row">
<el-col v-for="(tempDataItem,tempDataIndex) in bmsDataList" :key="tempDataIndex+'bmsTempData'"
:span="6" class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.name,item.deviceId)">
<span class="left">{{ tempDataItem.name }}</span> <span class="right">{{ item[tempDataItem.attr] }}<span
v-html="tempDataItem.unit"></span></span>
</span>
</el-col>
<el-col v-for="(tempDataItem,tempDataIndex) in pcsDataList" :key="tempDataIndex+'pcsTempData'"
:span="6" class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.name,item.deviceId)">
<span class="left">{{ tempDataItem.name }}</span> <span class="right">{{ item[tempDataItem.attr] }}<span
v-html="tempDataItem.unit"></span></span>
</span>
</el-col>
</el-row>
</el-card>
<el-empty v-show="list.length <= 0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div>
</template>
<script>
import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getEmsDataList} from "@/api/ems/dzjk";
import intervalUpdate from "@/mixins/ems/intervalUpdate";
export default {
name: "DzjkSbjkEms",
components: {pointChart, PointTable},
mixins: [getQuerySiteId, intervalUpdate],
data() {
return {
loading: false,
list: [],
bmsDataList: [{
name: 'BMS1SOC',
attr: 'bms1Soc'
},
{
name: 'BMS2SOC',
attr: 'bms2Soc'
},
{
name: 'BMS3SOC',
attr: 'bms3Soc'
},
{
name: 'BMS4SOC',
attr: 'bms4Soc'
}],
pcsDataList: [{
name: 'PCS-1有功功率',
attr: 'pcs1Yggl'
},
{
name: 'PCS-2有功功率',
attr: 'pcs2Yggl'
},
{
name: 'PCS-3有功功率',
attr: 'pcs3Yggl'
},
{
name: 'PCS-4有功功率',
attr: 'pcs4Yggl'
}]
};
},
methods: {
// 查看设备电位表格
pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'EMS'}, dataType)
},
showChart(pointName, deviceId) {
pointName &&
this.$refs.pointChart.showChart({pointName, deviceCategory: 'EMS', deviceId});
},
getData() {
this.loading = true;
getEmsDataList(this.siteId)
.then((response) => {
const data = response?.data || {};
this.list = JSON.parse(JSON.stringify(data));
})
.finally(() => (this.loading = false));
},
updateData() {
this.getData();
},
init() {
this.updateData();
this.updateInterval(this.updateData);
},
},
};
</script>
<style lang="scss" scoped></style>

View File

@ -1,40 +1,46 @@
<template> <template>
<div v-loading="loading" class="pcs-ems-dashboard-editor-container"> <div v-loading="loading" class="pcs-ems-dashboard-editor-container">
<!-- 顶部六个方块--> <!-- 顶部六个方块-->
<real-time-base-info :data="runningHeadData" /> <real-time-base-info :data="runningHeadData"/>
<div <div
v-for="(pcsItem, pcsIndex) in pcsList" v-for="(pcsItem, pcsIndex) in pcsList"
:key="pcsIndex + 'PcsHome'" :key="pcsIndex + 'PcsHome'"
style="margin-bottom: 25px" style="margin-bottom: 25px"
> >
<el-card <el-card
:class="{ :class="handleCardClass(pcsItem)"
'warning-card-container': pcsItem.workStatus === '1', class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg"
'timing-card-container': pcsItem.workStatus === '2', shadow="always"
'running-card-container': !['1', '2'].includes(pcsItem.workStatus),
}"
class="sbjk-card-container common-card-container-body-no-padding common-card-container-no-title-bg"
shadow="always"
> >
<div slot="header"> <div slot="header">
<span class="large-title" <span class="large-title"
>{{ pcsIndex + 1 }}#{{ pcsItem.deviceName }}</span >{{ pcsItem.deviceName }}</span
> >
<div class="info"> <div class="info">
<div> <div>
{{ {{
$store.state.ems.communicationStatusOptions[ $store.state.ems.communicationStatusOptions[
pcsItem.communicationStatus pcsItem.communicationStatus
] ]
}} }}
</div> </div>
<div>数据更新时间{{ pcsItem.dataUpdateTime }}</div> <div>数据更新时间{{ pcsItem.dataUpdateTime }}</div>
</div> </div>
<div class="alarm"> <div class="alarm">
<el-badge :value="pcsItem.alarmNum || 0" class="item"> <pcs-switch style="margin-right:10px;"
:round="true"
size="small"
type="danger"
:data="pcsItem"
@updateSuccess="init"/>
<el-button type="primary" round size="small" style="margin-right:20px;"
@click="pointDetail(pcsItem,'point')">
详细
</el-button>
<el-badge :hidden="!pcsItem.alarmNum" :value="pcsItem.alarmNum || 0" class="item">
<i <i
class="el-icon-message-solid" class="el-icon-message-solid alarm-icon"
style="font-size: 26px; color: #fff; display: block" @click="pointDetail(pcsItem,'alarmPoint')"
></i> ></i>
</el-badge> </el-badge>
</div> </div>
@ -42,64 +48,66 @@
<div class="descriptions-main"> <div class="descriptions-main">
<el-descriptions :colon="false" :column="4" direction="vertical"> <el-descriptions :colon="false" :column="4" direction="vertical">
<el-descriptions-item <el-descriptions-item
:contentClassName="`descriptions-direction ${ contentClassName="descriptions-direction work-status"
pcsItem.workStatus === '0' ? 'save' : 'danger' :span="1"
}`" label="工作状态"
:span="1" labelClassName="descriptions-label"
label="工作状态" >{{
labelClassName="descriptions-label" PCSWorkStatusOptions[pcsItem.workStatus]
>{{ }}
$store.state.ems.workStatusOptions[pcsItem.workStatus] </el-descriptions-item
}}</el-descriptions-item
> >
<el-descriptions-item <el-descriptions-item
:span="1" :span="1"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
label="并网状态" label="并网状态"
labelClassName="descriptions-label" labelClassName="descriptions-label"
>{{ >{{
$store.state.ems.gridStatusOptions[pcsItem.gridStatus] $store.state.ems.gridStatusOptions[pcsItem.gridStatus]
}}</el-descriptions-item }}
</el-descriptions-item
> >
<el-descriptions-item <el-descriptions-item
:contentClassName="`descriptions-direction ${ :contentClassName="`descriptions-direction ${
pcsItem.deviceStatus === '2' ? 'save' : 'danger' pcsItem.deviceStatus === '1' ? 'save' : 'danger'
}`" }`"
:span="1" :span="1"
label="设备状态" label="设备状态"
labelClassName="descriptions-label" labelClassName="descriptions-label"
>{{ >{{
$store.state.ems.deviceStatusOptions[pcsItem.deviceStatus] $store.state.ems.deviceStatusOptions[pcsItem.deviceStatus]
}}</el-descriptions-item }}
</el-descriptions-item
> >
<el-descriptions-item <el-descriptions-item
:span="1" :span="1"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
label="控制模式" label="控制模式"
labelClassName="descriptions-label" labelClassName="descriptions-label"
>{{ >{{
$store.state.ems.controlModeOptions[pcsItem.controlMode] $store.state.ems.controlModeOptions[pcsItem.controlMode]
}}</el-descriptions-item }}
</el-descriptions-item
> >
</el-descriptions> </el-descriptions>
</div> </div>
<div class="descriptions-main descriptions-main-bg-color"> <div class="descriptions-main descriptions-main-bg-color">
<el-descriptions <el-descriptions
:colon="false" :colon="false"
:column="4" :column="4"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
direction="vertical" direction="vertical"
labelClassName="descriptions-label" labelClassName="descriptions-label"
> >
<el-descriptions-item <el-descriptions-item
v-for="(item, index) in infoData" v-for="(item, index) in infoData"
:key="index + 'pcsInfoData'" :key="index + 'pcsInfoData'"
:label="item.label" :label="item.label"
:span="1" :span="1"
> >
<span <span
class="pointer" class="pointer"
@click=" @click="
showChart(item.pointName || '', pcsItem.deviceId) showChart(item.pointName || '', pcsItem.deviceId)
" "
> >
@ -110,58 +118,59 @@
</el-descriptions> </el-descriptions>
</div> </div>
<div <div
v-for="(item, index) in pcsItem.pcsBranchInfoList" v-for="(item, index) in pcsItem.pcsBranchInfoList"
:key="index + 'pcsBranchInfoList'" :key="index + 'pcsBranchInfoList'"
class="descriptions-main" class="descriptions-main"
> >
<el-descriptions <el-descriptions
:colon="false" :colon="false"
:column="4" :column="4"
contentClassName="descriptions-direction keep" contentClassName="descriptions-direction keep"
direction="vertical" direction="vertical"
labelClassName="descriptions-label" labelClassName="descriptions-label"
> >
<el-descriptions-item <el-descriptions-item
:label="'支路' + (index + 1)" :label="'支路' + (index + 1)"
:span="4" :span="4"
contentClassName="descriptions-direction keep" contentClassName="descriptions-direction keep"
labelClassName="descriptions-label" labelClassName="descriptions-label"
>{{ item.dischargeStatus }}</el-descriptions-item >{{ item.dischargeStatus }}
</el-descriptions-item
> >
<el-descriptions-item <el-descriptions-item
:span="1" :span="1"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
label="直流功率" label="直流功率"
labelClassName="descriptions-label" labelClassName="descriptions-label"
> >
<span <span
class="pointer" class="pointer"
@click="showChart('直流功率', item.deviceId,true)" @click="showChart('直流功率', item.deviceId,true)"
>{{ item.dcPower }}kW</span >{{ item.dcPower }}kW</span
> >
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item <el-descriptions-item
:span="1" :span="1"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
label="直流电压" label="直流电压"
labelClassName="descriptions-label" labelClassName="descriptions-label"
> >
<span <span
class="pointer" class="pointer"
@click="showChart('直流电压', item.deviceId,true)" @click="showChart('直流电压', item.deviceId,true)"
>{{ item.dcVoltage }}V</span >{{ item.dcVoltage }}V</span
> >
</el-descriptions-item> </el-descriptions-item>
<el-descriptions-item <el-descriptions-item
:span="1" :span="1"
contentClassName="descriptions-direction" contentClassName="descriptions-direction"
label="直流电流" label="直流电流"
labelClassName="descriptions-label" labelClassName="descriptions-label"
> >
<span <span
class="pointer" class="pointer"
@click="showChart('直流电流', item.deviceId,true)" @click="showChart('直流电流', item.deviceId,true)"
>{{ item.dcCurrent }}A</span >{{ item.dcCurrent }}A</span
> >
</el-descriptions-item> </el-descriptions-item>
</el-descriptions> </el-descriptions>
@ -169,20 +178,30 @@
</el-card> </el-card>
</div> </div>
<el-empty v-show="pcsList.length <= 0" :image-size="200"></el-empty> <el-empty v-show="pcsList.length <= 0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId" /> <point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div> </div>
</template> </template>
<script> <script>
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
import RealTimeBaseInfo from "./../RealTimeBaseInfo.vue"; import RealTimeBaseInfo from "./../RealTimeBaseInfo.vue";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getRunningHeadInfo, getPcsDetailInfo } from "@/api/ems/dzjk"; import {getPcsDetailInfo, getRunningHeadInfo} from "@/api/ems/dzjk";
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
import PcsSwitch from "@/views/ems/site/sblb/PcsSwitch.vue";
import {mapState} from "vuex";
export default { export default {
name: "DzjkSbjkPcs", name: "DzjkSbjkPcs",
components: { RealTimeBaseInfo, pointChart }, components: {RealTimeBaseInfo, pointChart, PointTable, PcsSwitch},
mixins: [getQuerySiteId, intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
computed: {
...mapState({
PCSWorkStatusOptions: state => state?.ems?.PCSWorkStatusOptions || {},
})
},
data() { data() {
return { return {
loading: false, loading: false,
@ -190,10 +209,10 @@ export default {
pcsList: [], pcsList: [],
infoData: [ infoData: [
{ {
label: "总交流有功率", label: "总交流有功率",
attr: "totalActivePower", attr: "totalActivePower",
unit: "kW", unit: "kW",
pointName: "总交流有功率", pointName: "总交流有功率",
}, },
{ {
label: "当天交流充电量", label: "当天交流充电量",
@ -201,7 +220,7 @@ export default {
unit: "kWh", unit: "kWh",
pointName: "当天交流充电量 (kWh)", pointName: "当天交流充电量 (kWh)",
}, },
{ label: "A相电压", attr: "aPhaseVoltage", unit: "V", pointName: "" }, {label: "A相电压", attr: "aPhaseVoltage", unit: "V", pointName: ""},
{ {
label: "A相电流", label: "A相电流",
attr: "aPhaseCurrent", attr: "aPhaseCurrent",
@ -209,10 +228,10 @@ export default {
pointName: "A相电流", pointName: "A相电流",
}, },
{ {
label: "总交流无功率", label: "总交流无功率",
attr: "totalReactivePower", attr: "totalReactivePower",
unit: "kVar", unit: "kVar",
pointName: "总交流无功率", pointName: "总交流无功率",
}, },
{ {
label: "当天交流放电量", label: "当天交流放电量",
@ -220,7 +239,7 @@ export default {
unit: "kWh", unit: "kWh",
pointName: "当天交流放电量 (kWh)", pointName: "当天交流放电量 (kWh)",
}, },
{ label: "B相电压", attr: "bPhaseVoltage", unit: "V", pointName: "" }, {label: "B相电压", attr: "bPhaseVoltage", unit: "V", pointName: ""},
{ {
label: "B相电流", label: "B相电流",
attr: "bPhaseCurrent", attr: "bPhaseCurrent",
@ -239,7 +258,7 @@ export default {
unit: "&#8451;", unit: "&#8451;",
pointName: "", pointName: "",
}, },
{ label: "C相电压", attr: "cPhaseVoltage", unit: "V", pointName: "" }, {label: "C相电压", attr: "cPhaseVoltage", unit: "V", pointName: ""},
{ {
label: "C相电流", label: "C相电流",
attr: "cPhaseCurrent", attr: "cPhaseCurrent",
@ -269,9 +288,18 @@ export default {
}; };
}, },
methods: { methods: {
showChart(pointName, deviceId,isBranch=false) { handleCardClass(item) {
const {workStatus = ''} = item
return workStatus === '1' || !Object.keys(this.PCSWorkStatusOptions).find(i => i === workStatus) ? "timing-card-container" : workStatus === '2' ? 'warning-card-container' : 'running-card-container'
},
// 查看设备电位表格
pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'PCS'}, dataType)
},
showChart(pointName, deviceId, isBranch = false) {
pointName && pointName &&
this.$refs.pointChart.showChart({ pointName,deviceCategory:isBranch ? 'BRANCH' : 'PCS', deviceId }); this.$refs.pointChart.showChart({pointName, deviceCategory: isBranch ? 'BRANCH' : 'PCS', deviceId});
}, },
//6个方块数据 //6个方块数据
getRunningHeadData() { getRunningHeadData() {
@ -282,11 +310,11 @@ export default {
getPcsList() { getPcsList() {
this.loading = true; this.loading = true;
getPcsDetailInfo(this.siteId) getPcsDetailInfo(this.siteId)
.then((response) => { .then((response) => {
const data = response?.data || {}; const data = response?.data || {};
this.pcsList = JSON.parse(JSON.stringify(data)); this.pcsList = JSON.parse(JSON.stringify(data));
}) })
.finally(() => (this.loading = false)); .finally(() => (this.loading = false));
}, },
updateData() { updateData() {
this.getRunningHeadData(); this.getRunningHeadData();

View File

@ -1,12 +1,12 @@
<template> <template>
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container common-card-container-body-no-padding" class="common-card-container common-card-container-body-no-padding"
> >
<div slot="header"> <div slot="header">
<span class="card-title">PCS有功功率/PCS无功功率</span> <span class="card-title">PCS有功功率/PCS无功功率</span>
</div> </div>
<div style="height: 360px" id="cnglqxChart" /> <div style="height: 360px" id="cnglqxChart"/>
</el-card> </el-card>
</template> </template>
@ -14,8 +14,7 @@
<script> <script>
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import { formatDate } from "@/filters/ems"; import {storagePower} from "@/api/ems/dzjk";
import { storagePower } from "@/api/ems/dzjk";
export default { export default {
mixins: [resize], mixins: [resize],
@ -35,51 +34,58 @@ export default {
this.chart = null; this.chart = null;
}, },
methods: { methods: {
init(siteId) { init(siteId, timeRange) {
this.chart.showLoading(); this.chart.showLoading();
const x = []; const [startTime = '', endTime = ''] = timeRange;
const data1 = [], storagePower(siteId, startTime, endTime)
data2 = []; .then((response) => {
storagePower(siteId) this.setOption(response?.data?.pcsPowerList || []);
.then((response) => { })
this.setOption(response?.data?.pcsPowerList || []); .finally(() => {
}) this.chart.hideLoading();
.finally(() => { });
this.chart.hideLoading();
});
}, },
setOption(data) { setOption(data) {
// data=[{deviceId:'pcs1',energyStoragePowList:[{createDate,deviceId,pcsTotalActPower,pcsTotalReactivePower}]}]
let xdata = [], let xdata = [],
series = []; series = [];
data.forEach((element, index) => { data.forEach((element, index) => {
if (index === 0) { if (index === 0) {
xdata = (element.energyStoragePowList || []).map((i) => i.createDate); xdata = (element.energyStoragePowList || []).map((i) => i.createDate);
} }
series.push( series.push(
{ {
type: "line", type: "line",
name: `${element.deviceId}有功功率`, name: `${element.deviceId}有功功率`,
areaStyle: { areaStyle: {
// color:'#FFBD00' // color:'#FFBD00'
},
data: (element.energyStoragePowList || []).map(
(i) => {
return {
value: i.pcsTotalActPower,
year: i.dateDay || ''
}
}
)
}, },
data: (element.energyStoragePowList || []).map( {
(i) => i.pcsTotalActPower type: "line",
), name: `${element.deviceId}无功功率`,
}, areaStyle: {
{ // color:'#FFBD00'
type: "line", },
name: `${element.deviceId}无功功率`, data: (element.energyStoragePowList || []).map(
areaStyle: { (i) => {
// color:'#FFBD00' return {
}, value: i.pcsTotalReactivePower,
data: (element.energyStoragePowList || []).map( year: i.dateDay || ''
(i) => i.pcsTotalReactivePower }
), }
} ),
}
); );
}); });
this.chart.setOption({ this.chart && this.chart.setOption({
legend: { legend: {
left: "center", left: "center",
top: "5", top: "5",
@ -93,16 +99,29 @@ export default {
containLabel: true, containLabel: true,
}, },
tooltip: { tooltip: {
show: true,
trigger: "axis", trigger: "axis",
axisPointer: { axisPointer: {
// 坐标轴指示器,坐标轴触发有效 // 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow' type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
formatter: (params) => {
if (params.length <= 0) return
let result = (params[0].data.year || '') + ' ' + params[0].name + '<div>'
params.forEach(item => {
const {color, seriesName, value} = item
result += `<div style="position: relative;padding-left:20px;line-height: 20px;">
<div style="position: absolute;top:50%;left:0;width:12px;height:12px;border-radius:100%;background: ${color};transform: translateY(-50%)"></div>
<span>${seriesName}</span><span style="margin-left:20px;font-weight: 700">${value}</span></div>`
})
result += '</div>'
return result
}
}, },
textStyle: { textStyle: {
color: "#333333", color: "#333333",
}, },
xAxis: { type: "category", data: xdata }, xAxis: {type: "category", data: xdata},
yAxis: { yAxis: {
type: "value", type: "value",
}, },
@ -118,7 +137,7 @@ export default {
}, },
], ],
series, series,
}); }, true);
}, },
}, },
}; };

View File

@ -14,7 +14,6 @@
<script> <script>
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import { formatDate } from "@/filters/ems";
import { batteryAveSoc } from "@/api/ems/dzjk"; import { batteryAveSoc } from "@/api/ems/dzjk";
export default { export default {
mixins: [resize], mixins: [resize],
@ -34,9 +33,10 @@ export default {
this.chart = null; this.chart = null;
}, },
methods: { methods: {
init(siteId) { init(siteId,timeRange) {
this.chart.showLoading(); this.chart.showLoading();
batteryAveSoc(siteId) const [startTime='', endTime=''] = timeRange;
batteryAveSoc(siteId,startTime,endTime)
.then((response) => { .then((response) => {
this.setOption(response?.data?.batteryAveSOCList || []); this.setOption(response?.data?.batteryAveSOCList || []);
}) })
@ -49,9 +49,12 @@ export default {
ydata = []; ydata = [];
data.forEach((element) => { data.forEach((element) => {
xdata.push(element.createDate); xdata.push(element.createDate);
ydata.push(element.batterySOC); ydata.push({
value:element.batterySOC,
year:element.dateDay,
});
}); });
xdata = this.chart.setOption({ this.chart && this.chart.setOption({
legend: { legend: {
left: "center", left: "center",
top: "5", top: "5",
@ -65,11 +68,24 @@ export default {
containLabel: true, containLabel: true,
}, },
tooltip: { tooltip: {
show:true,
trigger: "axis", trigger: "axis",
axisPointer: { axisPointer: {
// 坐标轴指示器,坐标轴触发有效 // 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow' type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
formatter :(params)=>{
if(params.length <= 0) return
let result = (params[0].data.year || '')+' '+params[0].name + '<div>'
params.forEach(item=>{
const {color,seriesName,value} = item
result += `<div style="position: relative;padding-left:20px;line-height: 20px;">
<div style="position: absolute;top:50%;left:0;width:12px;height:12px;border-radius:100%;background: ${color};transform: translateY(-50%)"></div>
<span>${seriesName}</span><span style="margin-left:20px;font-weight: 700">${value}</span></div>`
})
result+='</div>'
return result
}
}, },
textStyle: { textStyle: {
color: "#333333", color: "#333333",
@ -99,7 +115,7 @@ export default {
data: ydata, data: ydata,
}, },
], ],
}); },true);
}, },
}, },
}; };

View File

@ -14,7 +14,6 @@
<script> <script>
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import { formatDate } from "@/filters/ems";
import { batteryAveTemp } from "@/api/ems/dzjk"; import { batteryAveTemp } from "@/api/ems/dzjk";
export default { export default {
@ -35,12 +34,10 @@ export default {
this.chart = null; this.chart = null;
}, },
methods: { methods: {
init(siteId) { init(siteId,timeRange) {
this.chart.showLoading(); this.chart.showLoading();
const x = []; const [startTime='', endTime=''] = timeRange;
const data1 = [], batteryAveTemp(siteId,startTime,endTime)
data2 = [];
batteryAveTemp(siteId)
.then((response) => { .then((response) => {
this.setOption(response?.data?.batteryAveTempList || []); this.setOption(response?.data?.batteryAveTempList || []);
}) })
@ -53,9 +50,14 @@ export default {
ydata = []; ydata = [];
data.forEach((element) => { data.forEach((element) => {
xdata.push(element.createDate); xdata.push(element.createDate);
ydata.push(element.batteryTemp); ydata.push(
{
value: element.batteryTemp,
year: element.dateDay
}
);
}); });
xdata = this.chart.setOption({ this.chart && this.chart.setOption({
legend: { legend: {
left: "center", left: "center",
top: "5", top: "5",
@ -69,11 +71,24 @@ export default {
containLabel: true, containLabel: true,
}, },
tooltip: { tooltip: {
show:true,
trigger: "axis", trigger: "axis",
axisPointer: { axisPointer: {
// 坐标轴指示器,坐标轴触发有效 // 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow' type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
formatter :(params)=>{
if(params.length <= 0) return
let result = (params[0].data.year || '')+' '+params[0].name + '<div>'
params.forEach(item=>{
const {color,seriesName,value} = item
result += `<div style="position: relative;padding-left:20px;line-height: 20px;">
<div style="position: absolute;top:50%;left:0;width:12px;height:12px;border-radius:100%;background: ${color};transform: translateY(-50%)"></div>
<span>${seriesName}</span><span style="margin-left:20px;font-weight: 700">${value}</span></div>`
})
result+='</div>'
return result
}
}, },
textStyle: { textStyle: {
color: "#333333", color: "#333333",
@ -103,7 +118,7 @@ export default {
data: ydata, data: ydata,
}, },
], ],
}); },true);
}, },
}, },
}; };

View File

@ -14,7 +14,6 @@
<script> <script>
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import { formatDate } from "@/filters/ems";
import { pcsMaxTemp } from "@/api/ems/dzjk"; import { pcsMaxTemp } from "@/api/ems/dzjk";
export default { export default {
@ -35,11 +34,10 @@ export default {
this.chart = null; this.chart = null;
}, },
methods: { methods: {
init(siteId) { init(siteId,timeRange) {
this.chart.showLoading(); this.chart.showLoading();
const x = []; const [startTime='', endTime=''] = timeRange;
const data = []; pcsMaxTemp(siteId,startTime,endTime)
pcsMaxTemp(siteId)
.then((response) => { .then((response) => {
this.setOption(response?.data?.pcsMaxTempList || []); this.setOption(response?.data?.pcsMaxTempList || []);
}) })
@ -60,10 +58,15 @@ export default {
areaStyle: { areaStyle: {
// color:'#FFBD00' // color:'#FFBD00'
}, },
data: (element.maxTempVoList || []).map((i) => i.temp), data: (element.maxTempVoList || []).map((i) => {
return {
value: i.temp,
year: i.dateDay
}
}),
}); });
}); });
this.chart.setOption({ this.chart && this.chart.setOption({
legend: { legend: {
left: "center", left: "center",
top: "5", top: "5",
@ -77,11 +80,24 @@ export default {
containLabel: true, containLabel: true,
}, },
tooltip: { tooltip: {
show:true,
trigger: "axis", trigger: "axis",
axisPointer: { axisPointer: {
// 坐标轴指示器,坐标轴触发有效 // 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow' type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
formatter :(params)=>{
if(params.length <= 0) return
let result = (params[0].data.year || '')+' '+params[0].name + '<div>'
params.forEach(item=>{
const {color,seriesName,value} = item
result += `<div style="position: relative;padding-left:20px;line-height: 20px;">
<div style="position: absolute;top:50%;left:0;width:12px;height:12px;border-radius:100%;background: ${color};transform: translateY(-50%)"></div>
<span>${seriesName}</span><span style="margin-left:20px;font-weight: 700">${value}</span></div>`
})
result+='</div>'
return result
}
}, },
textStyle: { textStyle: {
color: "#333333", color: "#333333",
@ -102,7 +118,7 @@ export default {
}, },
], ],
series, series,
}); },true);
}, },
}, },
}; };

View File

@ -3,6 +3,8 @@
<div class="ssyx-ems-dashboard-editor-container"> <div class="ssyx-ems-dashboard-editor-container">
<!-- 6个方块--> <!-- 6个方块-->
<real-time-base-info :data="runningHeadData"/> <real-time-base-info :data="runningHeadData"/>
<!-- 时间选择 -->
<date-range-select ref="dateRangeSelect" @updateDate="updateDate" style="margin-top:20px;"/>
<!-- echart图表--> <!-- echart图表-->
<el-row :gutter="32" style="background:#fff;margin:30px 0;"> <el-row :gutter="32" style="background:#fff;margin:30px 0;">
<el-col :xs="24" :sm="12" :lg="12"> <el-col :xs="24" :sm="12" :lg="12">
@ -27,6 +29,7 @@
</style> </style>
<script> <script>
import DateRangeSelect from '@/components/Ems/DateRangeSelect/index.vue'
import RealTimeBaseInfo from "./../RealTimeBaseInfo.vue"; import RealTimeBaseInfo from "./../RealTimeBaseInfo.vue";
import CnglqxChart from './CnglqxChart.vue' import CnglqxChart from './CnglqxChart.vue'
import PocpjwdChart from './PocpjwdChart.vue' import PocpjwdChart from './PocpjwdChart.vue'
@ -38,11 +41,13 @@ import intervalUpdate from "@/mixins/ems/intervalUpdate";
export default { export default {
name:'DzjkSbjkSsyx', name:'DzjkSbjkSsyx',
components:{RealTimeBaseInfo,CnglqxChart,PocpjwdChart,DcpjwdChart,DcpjsocChart}, components:{RealTimeBaseInfo,CnglqxChart,PocpjwdChart,DcpjwdChart,DcpjsocChart,DateRangeSelect},
mixins:[getQuerySiteId,intervalUpdate], mixins:[getQuerySiteId,intervalUpdate],
data() { data() {
return { return {
runningHeadData:{},//运行信息 runningHeadData:{},//运行信息
timeRange:[],
isInit:true
} }
}, },
methods:{ methods:{
@ -52,17 +57,27 @@ export default {
this.runningHeadData = response?.data || {} this.runningHeadData = response?.data || {}
}) })
}, },
// 更新时间范围 重置图表
updateDate(data){
this.timeRange=data
!this.isInit && this.updateChart()
this.isInit = false
},
updateChart(){
this.$refs.cnglqx.init(this.siteId,this.timeRange||[])
this.$refs.pocpjwd.init(this.siteId,this.timeRange||[])
this.$refs.dcpjsoc.init(this.siteId,this.timeRange||[])
this.$refs.dcpjwd.init(this.siteId,this.timeRange||[])
this.updateInterval(this.updateData)
},
updateData(){ updateData(){
this.getRunningHeadData() this.getRunningHeadData()
this.$refs.cnglqx.init(this.siteId) this.updateChart()
this.$refs.pocpjwd.init(this.siteId)
this.$refs.dcpjsoc.init(this.siteId)
this.$refs.dcpjwd.init(this.siteId)
}, },
init(){ init(){
this.$refs.dateRangeSelect.init(true)
this.$nextTick(()=>{ this.$nextTick(()=>{
this.updateData() this.updateData()
this.updateInterval(this.updateData)
}) })
} }
} }

View File

@ -0,0 +1,134 @@
<template>
<div v-loading="loading">
<el-card
v-for="(item,index) in list"
:key="index+'ylLise'"
class="sbjk-card-container running-card-container"
:class="{
'warning-card-container':item.emsCommunicationStatus && item.emsCommunicationStatus !== '0',
'running-card-container':item.emsCommunicationStatus === '0'
}"
shadow="always">
<div slot="header">
<span class="large-title">{{ item.deviceName }}</span>
<div class="info">
<div>
{{
$store.state.ems.communicationStatusOptions[
item.emsCommunicationStatus
]
}}
</div>
<div>数据更新时间{{ item.dataUpdateTime || '-' }}</div>
</div>
<div class="alarm">
<el-button type="primary" round size="small" style="margin-right:20px;" @click="pointDetail(item,'point')">
详细
</el-button>
<el-badge :hidden="!item.alarmNum" :value="item.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(item,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<el-row class="device-info-row">
<el-col v-for="(tempDataItem,tempDataIndex) in tempData" :key="tempDataIndex+'hdTempData'" :span="12"
class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.title,item.deviceId)">
<span class="left">{{ tempDataItem.title }}</span> <span class="right">{{
item[tempDataItem.attr] || '-'
}}<span
v-html="tempDataItem.unit"></span></span>
</span>
</el-col>
</el-row>
</el-card>
<el-empty v-show="list.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/>
<point-table ref="pointTable"/>
</div>
</template>
<script>
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getXfDataList} from '@/api/ems/dzjk'
import intervalUpdate from "@/mixins/ems/intervalUpdate";
import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
export default {
name: 'DzjkSbjkXf',
mixins: [getQuerySiteId, intervalUpdate],
components: {pointChart, PointTable},
data() {
return {
loading: false,
list: [],
tempData: [
{title: '主电源备用电池状态', attr: 'dczt', unit: ''},
{title: '手自动状态延时状态', attr: 'yszt', unit: ''},
{title: '启动喷洒气体喷洒状态', attr: 'pszt', unit: ''},
{title: '压力开关状态电磁阀状态', attr: 'dcfzt', unit: ''},
]
}
},
methods: {
// 查看设备电位表格
pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'XF'}, dataType)
},
showChart(pointName, deviceId) {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory: 'XF', deviceId})
},
updateData() {
this.loading = true
getXfDataList(this.siteId).then(response => {
this.list = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => {
this.loading = false
})
},
init() {
this.updateData()
this.updateInterval(this.updateData)
}
},
mounted() {
}
}
</script>
<style scoped lang="scss">
.sbjk-card-container {
&:not(:last-child) {
margin-bottom: 25px;
}
.el-row {
background-color: #ffffff;
border: 1px solid #eeeeee;
font-size: 14px;
line-height: 16px;
color: #333333;
.el-col {
padding: 12px 0;
text-align: center;
position: relative;
}
.el-col {
border-bottom: 1px solid #eeeeee;
}
.el-col:not(:nth-child(3n)) {
border-right: 1px solid #eeeeee;
}
}
}
</style>

View File

@ -1,25 +1,42 @@
<template> <template>
<div v-loading="loading"> <div v-loading="loading">
<el-card <el-card
v-for="(item,index) in list" v-for="(item,index) in list"
:key="index+'ylLise'" :key="index+'ylLise'"
class="sbjk-card-container running-card-container" class="sbjk-card-container running-card-container"
shadow="always"> shadow="always">
<div slot="header"> <div slot="header">
<span class="large-title">{{index+1}}#{{item.deviceName}}</span> <span class="large-title">{{ item.deviceName }}</span>
<div class="info">
<div>数据更新时间{{ item.dataUpdateTime || '-' }}</div>
</div> </div>
<el-row> <div class="alarm">
<el-col v-for="(tempDataItem,tempDataIndex) in tempData" :key="tempDataIndex+'ylTempData'" :span="8"> <el-button type="primary" round size="small" style="margin-right:20px;" @click="pointDetail(item,'point')">
详细
</el-button>
<el-badge :hidden="!item.alarmNum" :value="item.alarmNum || 0" class="item">
<i
class="el-icon-message-solid alarm-icon"
@click="pointDetail(item,'alarmPoint')"
></i>
</el-badge>
</div>
</div>
<el-row class="device-info-row">
<el-col v-for="(tempDataItem,tempDataIndex) in tempData" :key="tempDataIndex+'ylTempData'" :span="8"
class="device-info-col">
<span class="pointer" @click="showChart(tempDataItem.title,item.deviceId)"> <span class="pointer" @click="showChart(tempDataItem.title,item.deviceId)">
{{tempDataItem.title}}{{item[tempDataItem.attr]}}<span v-html="tempDataItem.unit"></span> <span class="left">{{ tempDataItem.title }}</span> <span
class="right">{{ item[tempDataItem.attr] || '-' }}<span
v-html="tempDataItem.unit"></span></span>
</span> </span>
</el-col> </el-col>
</el-row> </el-row>
</el-card> </el-card>
<el-empty v-show="list.length<=0" :image-size="200"></el-empty> <el-empty v-show="list.length<=0" :image-size="200"></el-empty>
<point-chart ref="pointChart" :site-id="siteId"/> <point-chart ref="pointChart" :site-id="siteId"/>
</div> <point-table ref="pointTable"/>
</div>
</template> </template>
@ -28,68 +45,59 @@ import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getCoolingDataList} from '@/api/ems/dzjk' import {getCoolingDataList} from '@/api/ems/dzjk'
import intervalUpdate from "@/mixins/ems/intervalUpdate"; import intervalUpdate from "@/mixins/ems/intervalUpdate";
import pointChart from "./../PointChart.vue"; import pointChart from "./../PointChart.vue";
import PointTable from "@/views/ems/site/sblb/PointTable.vue";
export default { export default {
name:'DzjkSbjkYl', name: 'DzjkSbjkYl',
mixins:[getQuerySiteId,intervalUpdate], mixins: [getQuerySiteId, intervalUpdate],
components:{pointChart}, components: {pointChart, PointTable},
data() { data() {
return { return {
loading:false, loading: false,
list:[], list: [],
tempData:[ tempData: [
{title:'供水温度',attr:'gsTemp',unit:'&#8451;'}, {title: '供水温度', attr: 'gsTemp', unit: '&#8451;'},
{title:'回水温度',attr:'hsTemp',unit:'&#8451;'}, {title: '回水温度', attr: 'hsTemp', unit: '&#8451;'},
{title:'供水压力',attr:'gsPressure',unit:'bar'}, {title: '供水压力', attr: 'gsPressure', unit: 'bar'},
{title:'回水压力',attr:'hsPressure',unit:'bar'}, {title: '回水压力', attr: 'hsPressure', unit: 'bar'},
{title:'冷源水温度',attr:'lysTemp',unit:'&#8451;'}, {title: '冷源水温度', attr: 'lysTemp', unit: '&#8451;'},
{title:'VB01开度',attr:'vb01Kd',unit:'%'}, {title: 'VB01开度', attr: 'vb01Kd', unit: '%'},
{title:'VB02开度',attr:'vb02Kd',unit:'%'}, {title: 'VB02开度', attr: 'vb02Kd', unit: '%'},
] ]
} }
}, },
methods:{ methods: {
showChart(pointName,deviceId){ // 查看设备电位表格
pointName && this.$refs.pointChart.showChart({pointName,deviceCategory:'COOLING',deviceId}) pointDetail(row, dataType) {
const {deviceId} = row
this.$refs.pointTable.showTable({siteId: this.siteId, deviceId, deviceCategory: 'COOLING'}, dataType)
}, },
updateData(){ showChart(pointName, deviceId) {
pointName && this.$refs.pointChart.showChart({pointName, deviceCategory: 'COOLING', deviceId})
},
updateData() {
this.loading = true this.loading = true
getCoolingDataList(this.siteId).then(response => { getCoolingDataList(this.siteId).then(response => {
this.list = JSON.parse(JSON.stringify(response?.data || [])); this.list = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => {this.loading = false}) }).finally(() => {
this.loading = false
})
}, },
init(){ init() {
this.updateData() this.updateData()
this.updateInterval(this.updateData) this.updateInterval(this.updateData)
} }
}, },
mounted(){ mounted() {
} }
} }
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
.sbjk-card-container{ .sbjk-card-container {
&:not(:last-child){ &:not(:last-child) {
margin-bottom: 25px; margin-bottom: 25px;
} }
.el-row{
background-color: #ffffff;
border:1px solid #eeeeee;
font-size: 14px;
line-height: 16px;
color: #333333;
.el-col{
padding:12px 0;
text-align: center;
position: relative;
}
.el-col{
border-bottom: 1px solid #eeeeee;
}
.el-col:not(:nth-child(3n)){
border-right: 1px solid #eeeeee;
}
}
} }
</style> </style>

View File

@ -1,4 +1,3 @@
<template> <template>
<div style="width:100%" v-loading="loading"> <div style="width:100%" v-loading="loading">
<!-- 搜索栏--> <!-- 搜索栏-->
@ -22,172 +21,185 @@
<el-form-item> <el-form-item>
<el-button @click="onReset" native-type="button">重置</el-button> <el-button @click="onReset" native-type="button">重置</el-button>
</el-form-item> </el-form-item>
<el-form-item>
<el-button type="primary" @click="exportTable" native-type="button">导出</el-button>
</el-form-item>
</el-form> </el-form>
<!--表格--> <!--表格-->
<el-table <el-table
class="common-table" class="common-table"
:data="tableData" :data="tableData"
stripe stripe
style="width: 100%;margin-top:25px;"> style="width: 100%;margin-top:25px;">
<!-- 汇总列--> <!-- 汇总列-->
<el-table-column label="汇总"> <el-table-column label="汇总">
<el-table-column <el-table-column
prop="dataTime" prop="dataTime"
label="日期" label="日期"
width="120"> width="120">
</el-table-column> </el-table-column>
</el-table-column> </el-table-column>
<!--充电量列--> <!--充电量列-->
<el-table-column label="充电量" align="center"> <el-table-column label="充电量" align="center">
<el-table-column <el-table-column
align="center" align="center"
prop="activePeakKwh" prop="activePeakKwh"
label="尖"> label="尖">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="activeHighKwh" prop="activeHighKwh"
label="峰"> label="峰">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="activeFlatKwh" prop="activeFlatKwh"
label="平"> label="平">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="activeValleyKwh" prop="activeValleyKwh"
label="谷"> label="谷">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="activeTotalKwh" prop="activeTotalKwh"
label="总"> label="总">
</el-table-column> </el-table-column>
</el-table-column> </el-table-column>
<!--充电量列--> <!--充电量列-->
<el-table-column label="放电量" align="center"> <el-table-column label="放电量" align="center">
<el-table-column <el-table-column
align="center" align="center"
prop="reActivePeakKwh" prop="reActivePeakKwh"
label="尖"> label="尖">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="reActiveHighKwh" prop="reActiveHighKwh"
label="峰"> label="峰">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="reActiveFlatKwh" prop="reActiveFlatKwh"
label="平"> label="平">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="reActiveValleyKwh" prop="reActiveValleyKwh"
label="谷"> label="谷">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
align="center" align="center"
prop="reActiveTotalKwh" prop="reActiveTotalKwh"
label="总"> label="总">
</el-table-column> </el-table-column>
</el-table-column> </el-table-column>
<!-- 效率--> <!-- 效率-->
<el-table-column label="效率(%)" align="center"> <el-table-column label="效率(%)" align="center">
<el-table-column <el-table-column
align="center" align="center"
prop="effect"> prop="effect">
</el-table-column> </el-table-column>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-pagination <el-pagination
v-show="tableData.length>0" v-show="tableData.length>0"
background background
@size-change="handleSizeChange" @size-change="handleSizeChange"
@current-change="handleCurrentChange" @current-change="handleCurrentChange"
:current-page="pageNum" :current-page="pageNum"
:page-size="pageSize" :page-size="pageSize"
:page-sizes="[10, 20, 30, 40]" :page-sizes="[10, 20, 30, 40]"
layout="total, sizes, prev, pager, next, jumper" layout="total, sizes, prev, pager, next, jumper"
:total="totalSize" :total="totalSize"
style="margin-top:15px;text-align: center" style="margin-top:15px;text-align: center"
> >
</el-pagination> </el-pagination>
</div> </div>
</template> </template>
<script> <script>
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getAmmeterData} from '@/api/ems/dzjk' import {getAmmeterData} from '@/api/ems/dzjk'
import {formatDate} from "@/filters/ems"; import {formatDate} from "@/filters/ems";
export default { export default {
name:'DzjkTjbbDbbb', name: 'DzjkTjbbDbbb',
mixins: [getQuerySiteId], mixins: [getQuerySiteId],
data() { data() {
return { return {
loading:false, loading: false,
pickerOptions:{ pickerOptions: {
disabledDate(time) { disabledDate(time) {
return time.getTime() > Date.now(); return time.getTime() > Date.now();
}, },
}, },
defaultDateRange:[],//默认展示的时间 defaultDateRange: [],//默认展示的时间
dateRange:[], dateRange: [],
tableData:[], tableData: [],
pageSize:10,//分页栏当前每个数据总数 pageSize: 10,//分页栏当前每个数据总数
pageNum:1,//分页栏当前页数 pageNum: 1,//分页栏当前页数
totalSize:0,//table表格数据总数 totalSize: 0,//table表格数据总数
} }
}, },
methods:{ methods: {
// 导出表格
exportTable() {
if (!this.dateRange?.length) return
const [startTime, endTime] = this.dateRange
this.download('ems/statsReport/exportAmmeterData', {
siteId: this.siteId,
startTime,
endTime,
}, `电表报表_${startTime}-${endTime}.xlsx`)
},
// 搜索 // 搜索
onSearch(){ onSearch() {
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 重置 // 重置
onReset(){ onReset() {
this.dateRange=this.defaultDateRange this.dateRange = this.defaultDateRange
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 分页 // 分页
handleSizeChange(val) { handleSizeChange(val) {
this.pageSize = val; this.pageSize = val;
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
handleCurrentChange(val) { handleCurrentChange(val) {
this.pageNum = val this.pageNum = val
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
// 获取数据 // 获取数据
getData(){ getData() {
this.loading=true this.loading = true
const {siteId,pageNum,pageSize} =this const {siteId, pageNum, pageSize} = this
const [startTime='',endTime='']=(this.dateRange || []) const [startTime = '', endTime = ''] = (this.dateRange || [])
getAmmeterData({siteId:siteId,startTime,endTime,pageSize,pageNum}).then(response=>{ getAmmeterData({siteId: siteId, startTime, endTime, pageSize, pageNum}).then(response => {
this.tableData=response?.rows || []; this.tableData = response?.rows || [];
this.totalSize = response?.total || 0 this.totalSize = response?.total || 0
}).finally(()=> { }).finally(() => {
this.loading = false this.loading = false
}) })
}, },
init(){ init() {
this.dateRange=[] this.dateRange = []
this.tableData=[] this.tableData = []
this.totalSize=0 this.totalSize = 0
this.pageSize=10 this.pageSize = 10
this.pageNum = 1 this.pageNum = 1
const now = new Date().getTime(); let now = new Date(), lastDay = now.getTime(), firstDay = new Date(now.setDate(1)).getTime();
const lastMonth = new Date(now-30 * 24 * 60 * 60 * 1000).getTime(); this.defaultDateRange = [formatDate(firstDay), formatDate(lastDay)];
this.defaultDateRange = [formatDate(lastMonth), formatDate(now)]; this.dateRange = [formatDate(firstDay), formatDate(lastDay)];
this.dateRange=[formatDate(lastMonth), formatDate(now)];
this.getData() this.getData()
}, },
}, },
@ -195,8 +207,8 @@ export default {
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
::v-deep{ ::v-deep {
.common-table.el-table .el-table__header-wrapper th, .common-table.el-table .el-table__fixed-header-wrapper th{ .common-table.el-table .el-table__header-wrapper th, .common-table.el-table .el-table__fixed-header-wrapper th {
border-bottom: 1px solid #dfe6ec; border-bottom: 1px solid #dfe6ec;
} }
} }

View File

@ -1,22 +1,23 @@
<template> <template>
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container time-range-card" class="common-card-container time-range-card"
style="margin-top: 20px" style="margin-top: 20px"
> >
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title"> </span> <span class="card-title"> </span>
<date-range-select ref="dateRangeSelect" @updateDate="updateDate" /> <date-range-select ref="dateRangeSelect" @updateDate="updateDate"/>
</div> </div>
<div class="card-main" v-loading="loading"> <div class="card-main" v-loading="loading">
<el-button-group class="ems-btns-group"> <el-button-group class="ems-btns-group">
<el-button <el-button
v-for="(item, index) in btnList" v-for="(item, index) in btnList"
:key="index + 'dcdqxBtns'" :key="index + 'dcdqxBtns'"
size="mini" size="mini"
:class="{ activeBtn: activeBtn === item.id }" :class="{ activeBtn: activeBtn === item.id }"
@click="changeDataType(item.id)" @click="changeDataType(item.id)"
>{{ item.name }}</el-button >{{ item.name }}
</el-button
> >
</el-button-group> </el-button-group>
<div id="dcdEchart" style="height: 310px"></div> <div id="dcdEchart" style="height: 310px"></div>
@ -28,12 +29,13 @@
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getStackData, getStackNameList } from "@/api/ems/dzjk"; import {getStackData} from "@/api/ems/dzjk";
import { formatDate } from "@/filters/ems"; import {formatDate} from "@/filters/ems";
import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue"; import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue";
export default { export default {
name: "DzjkTjbbDcdqx", name: "DzjkTjbbDcdqx",
components: { DateRangeSelect }, components: {DateRangeSelect},
mixins: [resize, getQuerySiteId], mixins: [resize, getQuerySiteId],
data() { data() {
return { return {
@ -46,10 +48,10 @@ export default {
loading: false, loading: false,
activeBtn: "1", activeBtn: "1",
btnList: [ btnList: [
{ name: "堆平均维度", id: "1", attr: ["temp"], source: ["有功功率"] }, {name: "堆平均维度", id: "1", attr: ["temp"], source: ["有功功率"]},
{ name: "堆电压", id: "2", attr: ["voltage"], source: ["堆电压"] }, {name: "堆电压", id: "2", attr: ["voltage"], source: ["堆电压"]},
{ name: "堆电流", id: "3", attr: ["current"], source: ["堆电流"] }, {name: "堆电流", id: "3", attr: ["current"], source: ["堆电流"]},
{ name: "堆soc", id: "4", attr: ["soc"], source: ["堆soc"] }, {name: "堆soc", id: "4", attr: ["soc"], source: ["堆soc"]},
], ],
}; };
}, },
@ -66,7 +68,7 @@ export default {
this.getData(); this.getData();
}, },
getData() { getData() {
const { siteId, activeBtn } = this; const {siteId, activeBtn} = this;
const [start = "", end = ""] = this.dateRange || []; const [start = "", end = ""] = this.dateRange || [];
//接口调用完成之后 设置图表、结束loading //接口调用完成之后 设置图表、结束loading
this.loading = true; this.loading = true;
@ -76,12 +78,12 @@ export default {
endTime: formatDate(end), endTime: formatDate(end),
dataType: activeBtn, dataType: activeBtn,
}) })
.then((response) => { .then((response) => {
this.setOption(response?.data || []); this.setOption(response?.data || []);
}) })
.finally(() => { .finally(() => {
this.loading = false; this.loading = false;
}); });
}, },
compareDate(date1, date2) { compareDate(date1, date2) {
console.log("比较时间", date1, date2); console.log("比较时间", date1, date2);
@ -92,9 +94,9 @@ export default {
const [date1_Y = "", date1_M = "", date1_D = ""] = date1.split("-"); //根据空格区分[年月日,小时] const [date1_Y = "", date1_M = "", date1_D = ""] = date1.split("-"); //根据空格区分[年月日,小时]
const [date2_Y = "", date2_M = "", date2_D = ""] = date2.split("-"); //根据空格区分[年月日,小时] const [date2_Y = "", date2_M = "", date2_D = ""] = date2.split("-"); //根据空格区分[年月日,小时]
return ( return (
(date1_Y === date2_Y && date1_M === date2_M && date1_D - date2_D) || (date1_Y === date2_Y && date1_M === date2_M && date1_D - date2_D) ||
(date1_Y === date2_Y && date1_M - date2_M) || (date1_Y === date2_Y && date1_M - date2_M) ||
date1_Y - date2_Y date1_Y - date2_Y
); );
}, },
setOption(data) { setOption(data) {
@ -106,7 +108,7 @@ export default {
const source = []; const source = [];
const sourceTop = ["日期"]; const sourceTop = ["日期"];
let map = {}, let map = {},
mapArr = []; mapArr = [];
// 生成所有{日期:[],日期:[]}格式的对象和所有包含所有日期的[日期1,日期2...] // 生成所有{日期:[],日期:[]}格式的对象和所有包含所有日期的[日期1,日期2...]
data.forEach((item) => { data.forEach((item) => {
item.dataList.forEach((inner) => { item.dataList.forEach((inner) => {
@ -136,38 +138,53 @@ export default {
}); });
source.unshift(sourceTop); source.unshift(sourceTop);
this.chart.setOption( this.chart.setOption(
{ {
grid: { grid: {
containLabel: true, containLabel: true,
},
legend: {
left: "center",
bottom: "15",
},
tooltip: {
trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
legend: {
left: "center",
top: "10",
},
tooltip: {
trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
},
},
textStyle: {
color: "#333333",
},
xAxis: {
type: "category",
},
yAxis: {
type: "value",
},
dataZoom: [
{
type: "inside",
start: 0,
end: 100,
},
{
start: 0,
end: 100,
},
],
dataset: {source},
series: source[0].slice(1).map((item) => {
return {
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
};
}),
}, },
textStyle: { true
color: "#333333",
},
xAxis: {
type: "category",
},
yAxis: {
type: "value",
},
dataset: { source },
series: source[0].slice(1).map((item) => {
return {
type: "line",
};
}),
},
true
); );
}, },
initChart() { initChart() {
@ -176,7 +193,7 @@ export default {
init() { init() {
this.$nextTick(() => { this.$nextTick(() => {
this.initChart(); this.initChart();
this.$refs.dateRangeSelect.init(); this.$refs.dateRangeSelect.init(true);
}); });
}, },
}, },

View File

@ -1,15 +1,14 @@
<template> <template>
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container time-range-card" class="common-card-container time-range-card"
style="margin-top: 20px" style="margin-top: 20px"
> >
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title">功率曲线</span> <span class="card-title">功率曲线</span>
<date-range-select <date-range-select
ref="dateRangeSelect" ref="dateRangeSelect"
@reset="resetTime" @updateDate="updateDate"
@updateDate="updateDate"
/> />
</div> </div>
<div class="card-main" v-loading="loading"> <div class="card-main" v-loading="loading">
@ -22,12 +21,13 @@
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getPcsNameList, getPowerData } from "@/api/ems/dzjk"; import {getPowerData} from "@/api/ems/dzjk";
import { formatDate } from "@/filters/ems"; import {formatDate} from "@/filters/ems";
import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue"; import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue";
export default { export default {
name: "DzjkTjbbGlqx", name: "DzjkTjbbGlqx",
components: { DateRangeSelect }, components: {DateRangeSelect},
mixins: [resize, getQuerySiteId], mixins: [resize, getQuerySiteId],
data() { data() {
return { return {
@ -38,7 +38,6 @@ export default {
}, },
dateRange: [], dateRange: [],
loading: false, loading: false,
dateRangeInit: true,
}; };
}, },
methods: { methods: {
@ -47,30 +46,22 @@ export default {
this.dateRange = data || []; this.dateRange = data || [];
this.getData(); this.getData();
}, },
resetTime() {
this.dateRangeInit = true;
},
getData() { getData() {
const { siteId } = this; const {siteId} = this;
let [start = "", end = ""] = this.dateRange || []; let [start = "", end = ""] = this.dateRange || [];
//接口调用完成之后 设置图表、结束loading //接口调用完成之后 设置图表、结束loading
this.loading = true; this.loading = true;
if (this.dateRangeInit) {
start = "";
end = "";
this.dateRangeInit = false;
}
getPowerData({ getPowerData({
siteId, siteId,
startDate: formatDate(start), startDate: formatDate(start),
endDate: formatDate(end), endDate: formatDate(end),
}) })
.then((response) => { .then((response) => {
this.setOption(response?.data || []); this.setOption(response?.data || []);
}) })
.finally(() => { .finally(() => {
this.loading = false; this.loading = false;
}); });
}, },
setOption(data) { setOption(data) {
const source = [["日期", "电网功率", "负载功率", "储能功率", "光伏功率"]]; const source = [["日期", "电网功率", "负载功率", "储能功率", "光伏功率"]];
@ -84,47 +75,74 @@ export default {
]); ]);
}); });
this.chart.setOption( this.chart.setOption(
{ {
grid: { grid: {
containLabel: true, containLabel: true,
},
legend: {
left: "center",
bottom: "15",
},
tooltip: {
trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
}, legend: {
textStyle: { left: "center",
color: "#333333", top: "10",
},
xAxis: {
type: "category",
},
yAxis: {
type: "value",
},
dataset: { source },
series: [
{
type: "line",
}, },
{ tooltip: {
type: "line", trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
},
}, },
{ textStyle: {
type: "line", color: "#333333",
}, },
{ xAxis: {
type: "line", type: "category",
}, },
], yAxis: {
}, type: "value",
true },
dataset: {source},
dataZoom: [
{
type: "inside",
start: 0,
end: 100,
},
{
start: 0,
end: 100,
},
],
series: [
{
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
},
{
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
},
{
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
},
{
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
},
],
},
true
); );
}, },
initChart() { initChart() {
@ -133,7 +151,6 @@ export default {
}, },
init() { init() {
this.$nextTick(() => { this.$nextTick(() => {
this.dateRangeInit = true;
this.initChart(); this.initChart();
this.$refs.dateRangeSelect.init(); this.$refs.dateRangeSelect.init();
}); });

View File

@ -1,22 +1,23 @@
<template> <template>
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container time-range-card" class="common-card-container time-range-card"
style="margin-top: 20px" style="margin-top: 20px"
> >
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title"> </span> <span class="card-title"> </span>
<date-range-select ref="dateRangeSelect" @updateDate="updateDate" /> <date-range-select ref="dateRangeSelect" @updateDate="updateDate"/>
</div> </div>
<div class="card-main" v-loading="loading"> <div class="card-main" v-loading="loading">
<el-button-group class="ems-btns-group"> <el-button-group class="ems-btns-group">
<el-button <el-button
v-for="(item, index) in btnList" v-for="(item, index) in btnList"
:key="index + 'flqxcBtns'" :key="index + 'flqxcBtns'"
size="mini" size="mini"
:class="{ activeBtn: activeBtn === item.id }" :class="{ activeBtn: activeBtn === item.id }"
@click="changeDataType(item.id)" @click="changeDataType(item.id)"
>{{ item.name }}</el-button >{{ item.name }}
</el-button
> >
</el-button-group> </el-button-group>
<div id="pcsEchart" style="height: 310px"></div> <div id="pcsEchart" style="height: 310px"></div>
@ -28,12 +29,13 @@
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { getPCSData, getPcsNameList } from "@/api/ems/dzjk"; import {getPCSData} from "@/api/ems/dzjk";
import { formatDate } from "@/filters/ems"; import {formatDate} from "@/filters/ems";
import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue"; import DateRangeSelect from "@/components/Ems/DateRangeSelect/index.vue";
export default { export default {
name: "DzjkTjbbPcsqx", name: "DzjkTjbbPcsqx",
components: { DateRangeSelect }, components: {DateRangeSelect},
mixins: [resize, getQuerySiteId], mixins: [resize, getQuerySiteId],
data() { data() {
return { return {
@ -82,7 +84,7 @@ export default {
this.getData(); this.getData();
}, },
getData() { getData() {
const { siteId, activeBtn } = this; const {siteId, activeBtn} = this;
const [start = "", end = ""] = this.dateRange || []; const [start = "", end = ""] = this.dateRange || [];
this.loading = true; this.loading = true;
//接口调用完成之后 设置图表、结束loading //接口调用完成之后 设置图表、结束loading
@ -92,12 +94,12 @@ export default {
endTime: formatDate(end), endTime: formatDate(end),
dataType: activeBtn, dataType: activeBtn,
}) })
.then((response) => { .then((response) => {
this.setOption(response?.data || []); this.setOption(response?.data || []);
}) })
.finally(() => { .finally(() => {
this.loading = false; this.loading = false;
}); });
}, },
compareDate(date1, date2) { compareDate(date1, date2) {
console.log("比较时间", date1, date2); console.log("比较时间", date1, date2);
@ -108,9 +110,9 @@ export default {
const [date1_Y = "", date1_M = "", date1_D = ""] = date1.split("-"); //根据空格区分[年月日,小时] const [date1_Y = "", date1_M = "", date1_D = ""] = date1.split("-"); //根据空格区分[年月日,小时]
const [date2_Y = "", date2_M = "", date2_D = ""] = date2.split("-"); //根据空格区分[年月日,小时] const [date2_Y = "", date2_M = "", date2_D = ""] = date2.split("-"); //根据空格区分[年月日,小时]
return ( return (
(date1_Y === date2_Y && date1_M === date2_M && date1_D - date2_D) || (date1_Y === date2_Y && date1_M === date2_M && date1_D - date2_D) ||
(date1_Y === date2_Y && date1_M - date2_M) || (date1_Y === date2_Y && date1_M - date2_M) ||
date1_Y - date2_Y date1_Y - date2_Y
); );
}, },
setOption(data) { setOption(data) {
@ -122,7 +124,7 @@ export default {
const source = []; const source = [];
const sourceTop = ["日期"]; const sourceTop = ["日期"];
let map = {}, let map = {},
mapArr = []; mapArr = [];
// 生成所有{日期:[],日期:[]}格式的对象和所有包含所有日期的[日期1,日期2...] // 生成所有{日期:[],日期:[]}格式的对象和所有包含所有日期的[日期1,日期2...]
data.forEach((item) => { data.forEach((item) => {
item.dataList.forEach((inner) => { item.dataList.forEach((inner) => {
@ -152,38 +154,53 @@ export default {
}); });
source.unshift(sourceTop); source.unshift(sourceTop);
this.chart.setOption( this.chart.setOption(
{ {
grid: { grid: {
containLabel: true, containLabel: true,
},
legend: {
left: "center",
bottom: "15",
},
tooltip: {
trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
}, },
legend: {
left: "center",
top: "10",
},
tooltip: {
trigger: "axis",
axisPointer: {
// 坐标轴指示器,坐标轴触发有效
type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
},
},
textStyle: {
color: "#333333",
},
xAxis: {
type: "category",
},
yAxis: {
type: "value",
},
dataZoom: [
{
type: "inside",
start: 0,
end: 100,
},
{
start: 0,
end: 100,
},
],
dataset: {source},
series: source[0].slice(1).map((item) => {
return {
type: "line",
smooth: true,
areaStyle: {
opacity: 0.7,
},
};
}),
}, },
textStyle: { true
color: "#333333",
},
xAxis: {
type: "category",
},
yAxis: {
type: "value",
},
dataset: { source },
series: source[0].slice(1).map((item) => {
return {
type: "line",
};
}),
},
true
); );
}, },
initChart() { initChart() {
@ -192,7 +209,7 @@ export default {
init() { init() {
this.$nextTick(() => { this.$nextTick(() => {
this.initChart(); this.initChart();
this.$refs.dateRangeSelect.init(); this.$refs.dateRangeSelect.init(true);
}); });
}, },
}, },

View File

@ -0,0 +1,228 @@
<template>
<div style="width:100%" v-loading="loading">
<!-- 搜索栏-->
<el-form :inline="true" class="select-container">
<el-form-item label="时间选择">
<el-date-picker
v-model="dateRange"
type="daterange"
range-separator=""
start-placeholder="开始日期"
end-placeholder="结束日期"
value-format="yyyy-MM-dd"
:clearable="false"
:picker-options="pickerOptions"
:default-value="defaultDateRange"
></el-date-picker>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="onSearch" native-type="button">搜索</el-button>
</el-form-item>
<el-form-item>
<el-button @click="onReset" native-type="button">重置</el-button>
</el-form-item>
<el-form-item>
<el-button type="primary" @click="exportTable" native-type="button">导出</el-button>
</el-form-item>
</el-form>
<!--表格-->
<el-table
class="common-table"
:data="tableData"
show-summary
stripe
style="width: 100%;margin-top:25px;">
<!-- 汇总列-->
<el-table-column label="汇总" min-width="100px" align="center">
<el-table-column
prop="dataTime"
label="日期"
min-width="100px" align="center">
</el-table-column>
</el-table-column>
<!--充电量列-->
<el-table-column label="充电价格" align="center">
<el-table-column
align="center"
prop="activePeakPrice"
label="尖">
</el-table-column>
<el-table-column
align="center"
prop="activeHighPrice"
label="峰">
</el-table-column>
<el-table-column
align="center"
prop="activeFlatPrice"
label="平">
</el-table-column>
<el-table-column
align="center"
prop="activeValleyPrice"
label="谷">
</el-table-column>
<el-table-column
align="center"
prop="activeTotalPrice"
label="总">
</el-table-column>
</el-table-column>
<!--充电量列-->
<el-table-column label="放电价格" align="center">
<el-table-column
align="center"
prop="reActivePeakPrice"
label="尖">
</el-table-column>
<el-table-column
align="center"
prop="reActiveHighPrice"
label="峰">
</el-table-column>
<el-table-column
align="center"
prop="reActiveFlatPrice"
label="平">
</el-table-column>
<el-table-column
align="center"
prop="reActiveValleyPrice"
label="谷">
</el-table-column>
<el-table-column
align="center"
prop="reActiveTotalPrice"
label="总">
</el-table-column>
</el-table-column>
<!-- 实际收益-->
<el-table-column label="" align="center" fixed="right">
<el-table-column
prop="actualRevenue"
label="实际收益"
align="center">
</el-table-column>
</el-table-column>
</el-table>
<el-pagination
v-show="tableData.length>0"
background
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
:current-page="pageNum"
:page-size="pageSize"
:page-sizes="[10, 20, 30, 40]"
layout="total, sizes, prev, pager, next, jumper"
:total="totalSize"
style="margin-top:15px;text-align: center"
>
</el-pagination>
</div>
</template>
<script>
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getAmmeterRevenueData} from '@/api/ems/dzjk'
import {formatDate} from "@/filters/ems";
export default {
name: 'DzjkTjbbSybb',
mixins: [getQuerySiteId],
data() {
return {
loading: false,
pickerOptions: {
disabledDate(time) {
return time.getTime() > Date.now();
},
},
defaultDateRange: [],//默认展示的时间
dateRange: [],
tableData: [],
pageSize: 10,//分页栏当前每个数据总数
pageNum: 1,//分页栏当前页数
totalSize: 0,//table表格数据总数
}
},
methods: {
// 导出表格
exportTable() {
if (!this.dateRange?.length) return
const [startTime, endTime] = this.dateRange
this.download('ems/statsReport/exportAmmeterRevenueData', {
siteId: this.siteId,
startTime,
endTime,
}, `收益报表_${startTime}-${endTime}.xlsx`)
},
// 搜索
onSearch() {
this.pageNum = 1//每次搜索从1开始搜索
this.getData()
},
// 重置
onReset() {
this.dateRange = this.defaultDateRange
this.pageNum = 1//每次搜索从1开始搜索
this.getData()
},
// 分页
handleSizeChange(val) {
this.pageSize = val;
this.$nextTick(() => {
this.getData()
})
},
handleCurrentChange(val) {
this.pageNum = val
this.$nextTick(() => {
this.getData()
})
},
// 获取数据
getData() {
this.loading = true
const {siteId, pageNum, pageSize} = this
const [startTime = '', endTime = ''] = (this.dateRange || [])
getAmmeterRevenueData({siteId: siteId, startTime, endTime, pageSize, pageNum}).then(response => {
this.tableData = response?.rows || [];
this.totalSize = response?.total || 0
}).finally(() => {
this.loading = false
})
},
init() {
this.dateRange = []
this.tableData = []
this.totalSize = 0
this.pageSize = 10
this.pageNum = 1
let now = new Date(), lastDay = now.getTime(), firstDay = new Date(now.setDate(1)).getTime();
this.defaultDateRange = [formatDate(firstDay), formatDate(lastDay)];
this.dateRange = [formatDate(firstDay), formatDate(lastDay)];
this.getData()
},
},
}
</script>
<style scoped lang="scss">
::v-deep {
.common-table.el-table {
.el-table__header-wrapper th, .common-table.el-table .el-table__fixed-header-wrapper th {
border-bottom: 1px solid #dfe6ec;
}
.el-table__footer-wrapper {
tbody td.el-table__cell {
color: #000;
font-weight: bolder;
}
}
}
}
</style>

View File

@ -1,577 +0,0 @@
<template>
<div class="ems-dashboard-editor-container" v-loading="loading">
<div class="container" v-show="!empty">
<!-- 电脑 -->
<div class="top">
<div class="cloud-container">
<div class="cloud">
<span style="z-index: 2; position: relative"></span>
</div>
</div>
<div class="double-arrows">
<div class="top-arrows"></div>
<div class="bottom-arrows"></div>
</div>
<div class="computer">
<img src="@/assets/images/ems/computer.png" alt="" />
<span style="z-index: 2; position: relative">ems</span>
</div>
</div>
<div class="outer-border">
<!-- 电表-->
<div class="row-lists-container" v-if="showDb">
<div class="row-title">电表({{ db.length }})</div>
<div class="row-lists">
<div v-for="item in db" :key="item.deviceId" class="row-items">
<div
class="status"
:class="
item.communicationStatus === '0' ? 'status-running' : ''
"
>
{{ communicationStatusOptions[item.communicationStatus] }}
</div>
<div class="row-items-img">
<img
class="img-db"
:src="require('@/assets/images/ems/db.png')"
/>
<div class="name">{{ item.deviceName }}</div>
</div>
</div>
</div>
</div>
<!-- 液冷-->
<div class="row-lists-container" v-if="showLq">
<div class="row-title">冷却({{ lq.length }})</div>
<div class="row-lists">
<div v-for="item in lq" :key="item.deviceId" class="row-items">
<div
class="status"
:class="
item.communicationStatus === '0' ? 'status-running' : ''
"
>
{{ communicationStatusOptions[item.communicationStatus] }}
</div>
<div class="row-items-img">
<img
class="img-lq"
:src="require('@/assets/images/ems/lq.png')"
/>
<div class="name">{{ item.deviceName }}</div>
</div>
</div>
</div>
</div>
<!-- PCS-->
<div class="row-lists-container" v-if="showPcs">
<div class="row-lists">
<div class="row-title">PCS({{ pcs.length }})</div>
<div
v-for="(item, index) in pcs"
:key="item.deviceId"
class="row-items row-items-pcs"
>
<!-- pcs -->
<div class="parent-dash">
<div
class="status"
:class="
item.communicationStatus === '0' ? 'status-running' : ''
"
>
{{ communicationStatusOptions[item.communicationStatus] }}
</div>
<div class="row-items-img">
<img
class="img-pcs"
:src="require('@/assets/images/ems/pcs.png')"
/>
<div class="name">{{ item.deviceName }}</div>
</div>
</div>
<!-- 子设备 bms -->
<div
v-if="item.children && item.children.length > 0"
class="children-dash"
>
<div
class="row-children-title"
v-if="bmsHasParentLength > 0 && index === 0"
>
BMS({{ bmsHasParentLength }})
</div>
<div
v-for="(childrenItem, childrenIndex) in item.children"
:key="childrenIndex + 'childrenBms'"
class="children-dash-items"
>
<div
class="status"
:class="
childrenItem.communicationStatus === '0'
? 'status-running'
: ''
"
>
{{
communicationStatusOptions[
childrenItem.communicationStatus
]
}}
</div>
<div class="row-items-img">
<img
class="img-pcs"
:src="require('@/assets/images/ems/bms.png')"
/>
<div class="name">{{ childrenItem.deviceName }}</div>
</div>
</div>
</div>
</div>
</div>
</div>
<div class="row-lists-container" v-if="showPcs">
<div class="row-title">PCS({{ pcs.length }})</div>
<div class="row-lists">
<div
v-for="item in bmsNoParent"
:key="item.deviceId"
class="row-items row-items-pcs"
>
<!-- pcs -->
<div class="parent-dash">
<div
class="status"
:class="
item.communicationStatus === '0' ? 'status-running' : ''
"
>
{{ communicationStatusOptions[item.communicationStatus] }}
</div>
<div class="row-items-img">
<img :src="require('@/assets/images/ems/bms.png')" />
<div class="name">{{ item.deviceName }}</div>
</div>
</div>
</div>
</div>
</div>
<!-- bms没有上级设备-->
<div class="row-lists-container" v-if="bmsNoParent.length > 0">
<div class="row-title">BMS({{ bmsNoParent.length }})</div>
<div class="row-lists">
<div
v-for="item in bmsNoParent"
:key="item.deviceId"
class="row-items row-items-pcs"
>
<!-- pcs -->
<div class="parent-dash">
<div
class="status"
:class="
item.communicationStatus === '0' ? 'status-running' : ''
"
>
{{ communicationStatusOptions[item.communicationStatus] }}
</div>
<div class="row-items-img">
<img :src="require('@/assets/images/ems/bms.png')" />
<div class="name">{{ item.deviceName }}</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<el-empty v-show="empty" :image-size="200"></el-empty>
</div>
</template>
<script>
import { getDeviceList } from "@/api/ems/site";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { mapState } from "vuex";
export default {
name: "DzjkZxlt",
mixins: [getQuerySiteId],
data() {
return {
loading: false,
pcs: [],
bms: [],
db: [],
lq: [],
pcsHasChildren: [],
pcsNoChildren: [],
bmsNoParent: [],
};
},
computed: {
...mapState({
communicationStatusOptions: (state) =>
state.ems.communicationStatusOptions,
}),
showPcs() {
return this.pcs.length > 0;
},
showBms() {
return this.bms.length > 0;
},
showDb() {
return this.db.length > 0;
},
showLq() {
return this.lq.length > 0;
},
bmsHasParentLength() {
let count = 0;
this.pcs.forEach((item) => (count += item.children.length));
return count;
},
empty() {
return !this.showBms && !this.showPcs && !this.showDb && !this.showLq;
},
},
methods: {
init() {
this.pcs = [];
this.bms = [];
this.lq = [];
this.db = [];
this.bmsNoParent = [];
this.loading = true;
getDeviceList(this.siteId)
.then((response) => {
const data = JSON.parse(JSON.stringify(response?.data || []));
let pcs = [],
bms = [],
db = [],
lq = [],
bmsNoParent = [];
data.forEach((item) => {
// 电表
if (item.deviceCategory === "AMMETER") {
db.push({ ...item, children: [] });
} else if (item.deviceCategory === "PCS") {
// pcs
pcs.push({ ...item, children: [] });
} else if (item.deviceCategory === "STACK") {
// bms
bms.push({ ...item, children: [] });
} else if (item.deviceCategory === "COOLING") {
// 液冷
lq.push({ ...item, children: [] });
}
});
bms.forEach((item, index) => {
if (item.parentId) {
pcs
.find((pcsItem) => pcsItem.deviceId === item.parentId)
.children.push(item);
} else {
bmsNoParent.push(item);
}
});
this.pcs = pcs;
this.bms = bms;
this.lq = lq;
this.db = db;
this.pcsHasChildren = pcs.filter((item) => item.children.length > 0);
this.pcsNoChildren = pcs.filter((item) => item.children.length === 0);
this.bmsNoParent = bmsNoParent;
})
.finally(() => {
this.loading = false;
});
},
},
};
</script>
<style lang="scss" scoped>
$sqDistance: 30px;
$borderColor: #174a8e;
$lineColor: #86bcc7;
.ems-dashboard-editor-container {
background-color: #ffffff;
padding: 0;
color: #666666;
.container {
display: flex;
position: relative;
}
//云 、计算机 、箭头
.top {
z-index: 2;
width: fit-content;
display: flex;
justify-content: center;
align-items: center;
// position: absolute;
// top: 50%;
// left: 0;
// transform: translateY(-50%);
//云 样式
.cloud-container {
margin: 0 auto;
.cloud {
width: 60px;
height: 26px;
background: #cbebfd;
border-radius: 100px;
position: relative;
text-align: center;
font-weight: bold;
font-size: 14px;
line-height: 26px;
}
.cloud:before,
.cloud:after {
content: "";
position: absolute;
background: #cbebfd;
width: 30px;
height: 30px;
border-radius: 100%;
}
.cloud:before {
top: -9px;
left: 8px;
}
.cloud:after {
top: -6px;
right: 9px;
}
}
//双箭头
.double-arrows {
height: fit-content;
margin: 0 10px;
text-align: center;
.top-arrows,
.bottom-arrows {
height: 4px;
width: 30px;
background-color: #5ea9df;
margin: 0 10px;
position: relative;
&::after {
content: "";
position: absolute;
left: 0;
width: 0;
height: 0;
}
}
.top-arrows {
vertical-align: super;
}
.top-arrows::after {
top: -4px;
border-bottom: 6px solid transparent;
border-left: 6px solid transparent;
border-right: 6px solid #5ea9df;
border-top: 6px solid transparent;
left: -11px;
}
.bottom-arrows {
margin-top: 8px;
&::after {
top: -4px;
border-top: 6px solid transparent;
border-left: 6px solid #5ea9df;
border-right: 6px solid transparent;
border-bottom: 6px solid transparent;
right: -11px;
left: auto;
}
}
}
//电脑
.computer {
text-align: center;
font-size: 14px;
line-height: 16px;
font-weight: bold;
position: relative;
background: #fff;
img {
width: 80px;
height: auto;
display: block;
}
}
}
.outer-border {
position: relative;
width: fit-content;
border: 1.5px solid $borderColor;
border-radius: 5px;
padding-left: 120px;
padding-right: 20px;
margin-left: -40px;
}
// 设备列表
.row-lists-container {
font-size: 10px;
position: relative;
padding: 10px;
.row-title {
position: absolute;
left: -$sqDistance - 30px;
top: calc(50% + 10px);
transform: translateY(-50%);
color: #000;
font-weight: bolder;
}
.row-lists {
display: flex;
position: relative;
.row-items {
position: relative;
padding: 5px 0;
&:not(:first-child) {
margin-left: $sqDistance; //和外层父元素上下padding一致
}
&::before {
content: "";
display: block;
height: 3px;
width: $sqDistance - 2px;
background: $lineColor;
position: absolute;
left: -$sqDistance;
top: calc(50% + 10px);
transform: scale(1, 0.4);
}
// 一列 第一个设备最上面的线
&:first-child {
&::before {
width: $sqDistance + 20px;
// top: -$sqDistance - 20px;
}
}
// 一列 最后一个设备最下面的线
// &:last-child {
// &::after {
// content: "";
// display: block;
// width: 3px;
// height: $sqDistance - 2px;
// background: $lineColor;
// position: absolute;
// bottom: -$sqDistance;
// left: 50%;
// transform: scale(0.4, 1);
// }
// }
// 设备状态
.status {
margin: 0 auto 4px;
width: fit-content;
height: 18px;
padding: 0 8px;
box-sizing: border-box;
text-align: center;
font-size: 8px;
line-height: 18px;
border: 1px solid #08ffff;
border-radius: 2px;
background: #aaaaaa;
color: #ffffff;
&.status-running {
background: #00c69c;
}
}
// 图片+设备名称
.row-items-img {
position: relative;
padding-top: 12px;
img {
width: 80px;
height: auto;
display: block;
&.img-lq {
width: 50px;
}
&.img-pcs {
width: 50px;
}
&.img-db {
width: 56px;
}
}
.name {
position: absolute;
top: 1px;
left: 0;
color: #666;
white-space: nowrap;
}
}
}
}
}
//子设备
.row-lists-container-children {
margin: 10px 0 0 $sqDistance;
.parent-dash {
position: relative;
&::before {
content: "";
display: block;
height: 40px;
width: 3px;
background: #ec7f8c;
position: absolute;
left: 20%;
top: -40px;
transform: scale(0.4, 1) rotate(-40deg);
}
}
}
.parent-dash {
width: fit-content;
}
.children-dash {
margin: $sqDistance 0 0 $sqDistance;
position: relative;
.row-children-title {
position: absolute;
left: -$sqDistance - 30px;
top: calc(50% + 10px);
transform: translateY(-50%);
color: #000;
font-weight: bolder;
}
.children-dash-items {
position: relative;
&::before {
content: "";
display: block;
height: $sqDistance;
width: 3px;
background: #ec7f8c;
position: absolute;
left: 20%;
top: -$sqDistance;
transform: scale(0.4, 1) rotate(-40deg);
}
}
}
}
</style>

View File

@ -1,614 +0,0 @@
<template>
<div class="ems-dashboard-editor-container" v-loading="loading" >
<div class="container" v-show="!empty">
<div class="top">
<div class="cloud-container">
<div class="cloud">
<span style="z-index:2;position: relative;"></span>
</div>
</div>
<div class="double-arrows">
<div class="top-arrows"></div>
<div class="bottom-arrows"></div>
</div>
<div class="computer">
<img src="@/assets/images/ems/computer.png" alt="">
<span style="z-index:2;position: relative;">ems</span>
</div>
<div class="arrow"></div>
</div>
<div class="bottom">
<!-- 四列设备-->
<div class="zxlt-row">
<!-- bmspcs 下级和上级在一列 -->
<div class="row-lists pcs-row-lists" v-if="showPcsAndBms">
<div class="item-square">
<div class="row-lists-title" v-if="showPcs">PCS({{pcs.length}})</div>
<div class="row-lists-title" v-if="showBms">BMS({{bms.length}})</div>
</div>
<!-- 上下级块 class区分-->
<div class="item-square pcs-has-children-item-square" :class="{'no-bms-list':!showBms}" v-for="(item,index) in pcsHasChildren" :key="index+'pcsHasChildren'">
<!-- 左边的上级 上级只有一个-->
<div class="item-lists parent-item-lists">
<!-- 上级设备-->
<div class="items normal-items-arrow">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="item.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[item.communicationStatus] || '-'}}</div>
</div>
<img v-if="item.pictureUrl" :src="item.pictureUrl">
<img v-else :src="require('@/assets/images/ems/pcs.png')"/>
<div class="name">{{item.deviceName}}</div>
</div>
</div>
</div>
<!-- 右边的下级 下级有多个-->
<div class="item-lists children-item-lists">
<!-- 下级设备 循环生成-->
<div class="items children-items-arrow bms-children-arrow" v-for="children in item.children" :key="children.deviceId">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="children.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[children.communicationStatus] || '-'}}</div>
</div>
<img v-if="children.pictureUrl" :src="children.pictureUrl">
<img v-else :src="require('@/assets/images/ems/bms.png')"/>
<div class="name">{{children.deviceName}}</div>
</div>
</div>
</div>
</div>
<!-- 没有上下级关系的bmspcs-->
<div class="item-square" :class="{'no-bms-list':!showBms}">
<!-- 左边没有下级的pcs-->
<div class="item-lists">
<div class="items normal-items-arrow" v-for="item in pcsNoChildren" :key="item.deviceId">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="item.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[item.communicationStatus] || '-'}}</div>
</div>
<img v-if="item.pictureUrl" :src="item.pictureUrl">
<img v-else :src="require('@/assets/images/ems/pcs.png')"/>
<div class="name">{{item.deviceName}}</div>
</div>
</div>
</div>
<!-- 右边没有上级的bms-->
<div class="item-lists">
<!-- 下级设备 循环生成-->
<div class="items children-items-arrow" v-for="item in bmsNoParent" :key="item.deviceId">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="item.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[item.communicationStatus] || '-'}}</div>
</div>
<img v-if="item.pictureUrl" :src="item.pictureUrl">
<img v-else :src="require('@/assets/images/ems/bms.png')"/>
<div class="name">{{item.deviceName}}</div>
</div>
</div>
</div>
</div>
</div>
<!-- 电表-->
<div class="row-lists" v-if="showDb">
<div class="item-square">
<div class="row-lists-title" style="width:100%;">电表({{db.length}})</div>
</div>
<div class="item-square">
<!-- 左边的下级 下级有多个-->
<div class="item-lists">
<!-- 下级设备 循环生成-->
<div class="items normal-items-arrow" v-for="item in db" :key="item.deviceId">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="item.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[item.communicationStatus] || '-'}}</div>
</div>
<img v-if="item.pictureUrl" :src="item.pictureUrl">
<img v-else :src="require('@/assets/images/ems/bms.png')"/>
<div class="name">{{item.deviceName}}</div>
</div>
</div>
</div>
</div>
</div>
<!--冷却-->
<div class="row-lists" v-if="showLq">
<div class="item-square">
<div class="row-lists-title" style="width:100%;">冷却({{lq.length}})</div>
</div>
<div class="item-square">
<div class="item-lists">
<div class="items normal-items-arrow" v-for="item in lq" :key="item.deviceId">
<div class="items-inner">
<div style="text-align: center;margin-bottom:10px;">
<div class="status" :class="item.communicationStatus === '0' ?'status-normal' : 'status-warn'">通讯状态:{{communicationStatusOptions[item.communicationStatus] || '-'}}</div>
</div>
<img v-if="item.pictureUrl" :src="item.pictureUrl">
<img v-else :src="require('@/assets/images/ems/bms.png')"/>
<div class="name">{{item.deviceName}}</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
</div>
<el-empty v-show="empty" :image-size="200"></el-empty>
</div>
</template>
<script>
import {getDeviceList} from'@/api/ems/site'
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {mapState} from "vuex";
export default {
name: 'DzjkZxlt',
mixins: [getQuerySiteId],
data() {
return {
loading:false,
pcs :[],
bms:[],
db:[],
lq:[],
pcsHasChildren:[],
pcsNoChildren:[],
bmsNoParent:[]
}
},
computed:{
...mapState({
communicationStatusOptions:(state)=>state.ems.communicationStatusOptions
}),
showPcs(){
return this.pcs.length>0
},
showBms(){
return this.bms.length>0
},
showDb(){
return this.db.length>0
},
showLq(){
return this.lq.length>0
},
showPcsAndBms(){
return this.showPcs || this.showBms
},
empty(){
return !this.showBms && !this.showPcs && !this.showDb && !this.showLq
},
},
methods: {
init(){
this.pcs = []
this.bms = []
this.lq=[]
this.db=[]
this.bmsNoParent=[]
this.loading = true
getDeviceList(this.siteId).then(response => {
const data =JSON.parse(JSON.stringify(response?.data || []))
let pcs = [],bms=[],db=[],lq=[],bmsNoParent=[]
data.forEach(item=>{
// 电表
if(item.deviceCategory === 'AMMETER'){
db.push({...item,children:[]})
}else if(item.deviceCategory === 'PCS'){
// pcs
pcs.push({...item,children:[]})
}else if(item.deviceCategory === 'STACK'){
// bms
bms.push({...item,children:[]})
}else if(item.deviceCategory === 'COOLING'){
// 液冷
lq.push({...item,children:[]})
}
})
bms.forEach((item,index)=>{
if(item.parentId){
pcs.find(pcsItem=>pcsItem.deviceId === item.parentId).children.push(item)
}else{
bmsNoParent.push(item)
}
})
this.pcs = pcs
this.bms = bms
this.lq=lq
this.db=db
this.pcsHasChildren = pcs.filter(item=>item.children.length > 0)
this.pcsNoChildren = pcs.filter(item=>item.children.length === 0)
this.bmsNoParent = bmsNoParent
}).finally(() => {
this.loading = false
})
}
},
}
</script>
<style lang="scss" scoped>
$distance:60px;
$arrowDistance:80px;//margin:60+quare的padding10
$arrowColoe:#5ea9df;
$lineColoe:#5ea9df;
.ems-dashboard-editor-container {
background-color: #ffffff;
padding:0;
.container{
position: relative;
overflow-x: auto;
}
//云 、计算机 、箭头
.top{
width: 280px;
font-size: 30px;
line-height: 40px;
font-weight: 500;
display: flex;
flex-direction: column;
//云 样式
.cloud-container{
padding-top:40px;
margin:0 auto;
.cloud {
width: 150px;
height: 60px;
background: #cbebfd;
border-radius: 200px;
position: relative;
text-align: center;
color:#666666;
}
.cloud:before, .cloud:after {
content: '';
position: absolute;
background:#cbebfd;
width: 80px;
height: 80px;
border-radius: 50%;
}
.cloud:before {
top: -28px;
left: 20px;
}
.cloud:after {
top: -31px;
right: 20px;
}
}
//双箭头
.double-arrows {
height: 50px;
margin:20px 0;
text-align: center;
.top-arrows,.bottom-arrows{
height: 100%;
width: 6px;
background-color: $arrowColoe;
display: inline-block;
margin: 0 10px;
position: relative;
vertical-align: super;
&::after {
content: '';
position: absolute;
left:0;
width: 0;
height: 0;
}
}
.top-arrows{
vertical-align: super;
}
.top-arrows::after {
bottom: -24px;
border-bottom: 12px solid transparent;
border-left: 12px solid transparent;
border-right: 12px solid transparent;
border-top: 14px solid $arrowColoe;
left: -9px;
}
.bottom-arrows{
margin-top:12px;
&::after {
top: -24px;
border-top: 12px solid transparent;
border-left: 12px solid transparent;
border-right: 12px solid transparent;
border-bottom: 14px solid $arrowColoe;
left: -9px;
}
}
}
//电脑
.computer{
margin:20px auto;
text-align: center;
color:#666666;
position: relative;
img {
width: auto;
height: 100px;
display: block;
}
}
.arrow{
height: 50px;
width: 30px;
border-radius: 5px;
background-color: $arrowColoe;
position: relative;
margin:0 auto;
&::after{
content: "";
position: absolute;
width: 0;
height: 0;
left: -9px;
border-top: 24px solid $arrowColoe;
border-left: 24px solid transparent;
border-bottom: 24px solid transparent;
border-right: 24px solid transparent;
bottom: -44px;
}
}
}
.bottom{
z-index:1;
box-sizing: border-box;
margin-top:50px;
.zxlt-row{
display: flex;
padding:20px $distance;
position: relative;
width: fit-content;
&:before{
content: '';
display: block;
width:calc(100% - 100px);
height:1px;
background-color: $lineColoe;
position:absolute;
top:0;
left: $distance/2;
}
.row-lists{
height: fit-content;
position: relative;
&:before{
content: '';
display: block;
height: 100%;
width: 1px;
position: absolute;
left:-($distance/2);
top:-20px;
background-color: $lineColoe;
}
//pcs列 bms右侧的边框
&.pcs-row-lists{
&:after{
content: '';
display: block;
height: 100%;
width: 1px;
position: absolute;
right:-(($distance/2) + 1);
top:-20px;
background-color: $lineColoe;
}
}
&:not(:last-child){
margin-right: $distance;
}
.item-square{
//左右 两列
display: flex;
vertical-align: middle;
align-items: flex-start;
padding:10px;
border-radius: 5px;
&:not(:last-child){
margin-bottom: 40px;
}
.row-lists-title{
font-size: 20px;
line-height: 20px;
color: #333333;
font-weight: 500;
text-align: center;
flex: 1;
}
.item-lists{
position: relative;
&:not(:last-child){
margin-right:$distance;
}
//每个设备
.items{
background-color: #cbebfd;
position: relative;
border-radius: 5px;
padding: 10px;
box-shadow: 0 0 10px rgba(0, 0, 0, 0.1), 0 0 0 rgba(0, 0, 0, 0.5);
//普通设备 箭头方向
&.normal-items-arrow{
&:before{
content: '';
display: block;
width:($arrowDistance/2) - 15;
height: 4px;
background-color: $arrowColoe;
position: absolute;
top:50%;
left: -($arrowDistance/2);
transform: translateY(-50%);
}
&:after{
content: '';
display: block;
height: 0;
width: 0;
border-left: 10px solid #5ea9df;
border-right: 10px solid transparent;
border-bottom: 10px solid transparent;
border-top: 10px solid transparent;
position: absolute;
top: 50%;
left: -15px;
transform: translateY(-50%);
}
}
//下级的箭头
&.children-items-arrow{
&:before{
content: '';
display: block;
width:($arrowDistance/2) - 15;
height: 4px;
background-color: $arrowColoe;
position: absolute;
top:50%;
right: -($arrowDistance/2);
transform: translateY(-50%);
}
&:after{
content: '';
display: block;
height: 0;
width: 0;
border-right: 10px solid #5ea9df;
border-left: 10px solid transparent;
border-bottom: 10px solid transparent;
border-top: 10px solid transparent;
position: absolute;
top: 50%;
right: -15px;
transform: translateY(-50%);
}
}
&:not(:last-child){
margin-bottom: 15px;
}
.items-inner{
background-color: #ffffff;
border-radius: 5px;
padding:10px;
width:130px;
text-align: center;
}
img{
width: 80px;
height: auto;
display: block;
z-index:2;
margin: 0 auto;
}
.name{
text-align: center;
margin-top:10px;
font-size: 14px;
line-height: 20px;
z-index:2;
}
.status{
z-index:2;
margin-top:10px;
font-size: 14px;
line-height: 20px;
position: relative;
padding-left:20px;
display: inline;
&.status-normal {
&:before {
content: "";
display: block;
width: 15px;
height: 15px;
border-radius: 50%;
background-color: #05AEA3;
position: absolute;
top:50%;
left:0;
transform: translate(0,-50%);
}
}
&.status-warn{
&:before{
content: "";
display: inline-block;
width: 15px;
height: 15px;
border-radius: 50%;
background-color: #FC6B69;
position: absolute;
top:50%;
left:0;
transform: translate(0,-50%);
}
}
}
}
}
.children-item-lists{
//todo 手动修改
&:before{
content: '';
display: block;
width:40px;
height: 4px;
background-color: $arrowColoe;
position: absolute;
top:50%;
left: -50px;
transform:translateY(-50%);
}
&:after{
content: '';
display: block;
height: 0;
width: 0;
border-left: 10px solid #5ea9df;
border-right: 10px solid transparent;
border-bottom: 10px solid transparent;
border-top: 10px solid transparent;
position: absolute;
top: 50%;
left: -14px;
transform:translateY(-50%);
}
}
}
.pcs-has-children-item-square{
vertical-align: middle;
align-items: center;
background-color: #ffefad;
}
.no-bms-list{
.item-lists{
&:not(:last-child){
margin-right:0;
}
}
}
}
}
}
}
</style>

View File

@ -13,7 +13,7 @@
<div class="bottom-arrows"></div> <div class="bottom-arrows"></div>
</div> </div>
<div class="computer"> <div class="computer">
<img src="@/assets/images/ems/computer.png" alt="" /> <img src="@/assets/images/ems/computer.png" alt=""/>
<span style="z-index: 2; position: relative">ems</span> <span style="z-index: 2; position: relative">ems</span>
</div> </div>
</div> </div>
@ -22,27 +22,27 @@
<div class="row-lists-container" v-if="showDb"> <div class="row-lists-container" v-if="showDb">
<div class="row-title">电表({{ db.length }})</div> <div class="row-title">电表({{ db.length }})</div>
<div <div
class="row-lists" class="row-lists"
v-for="outter in Math.ceil(db.length / 3)" v-for="outter in Math.ceil(db.length / 3)"
:key="outter + 'row'" :key="outter + 'row'"
> >
<template v-for="(item, index) in handlerList('db', outter)"> <template v-for="(item, index) in handlerList('db', outter)">
<div :key="index" class="row-items"> <div :key="index" class="row-items">
<div <div
style="position: relative; z-index: 2; background-color: #fff" style="position: relative; z-index: 2; background-color: #fff"
> >
<div <div
class="status" class="status"
:class=" :class="
item.runningStatus === '2' ? 'status-running' : '' item.deviceStatus === '1' ? 'status-running' : ''
" "
> >
{{ deviceStatusOptions[item.runningStatus] }} {{ deviceStatusOptions[item.deviceStatus] }}
</div> </div>
<div class="row-items-img"> <div class="row-items-img">
<img <img
class="img-db" class="img-db"
:src="require('@/assets/images/ems/db.png')" :src="require('@/assets/images/ems/db.png')"
/> />
<div class="name">{{ item.deviceName }}</div> <div class="name">{{ item.deviceName }}</div>
</div> </div>
@ -56,27 +56,27 @@
<div class="row-lists-container" v-if="showLq"> <div class="row-lists-container" v-if="showLq">
<div class="row-title">冷却({{ lq.length }})</div> <div class="row-title">冷却({{ lq.length }})</div>
<div <div
class="row-lists" class="row-lists"
v-for="outter in Math.ceil(lq.length / 3)" v-for="outter in Math.ceil(lq.length / 3)"
:key="outter + 'row'" :key="outter + 'row'"
> >
<template v-for="(item, index) in handlerList('lq', outter)"> <template v-for="(item, index) in handlerList('lq', outter)">
<div :key="index" class="row-items"> <div :key="index" class="row-items">
<div <div
style="position: relative; z-index: 2; background-color: #fff" style="position: relative; z-index: 2; background-color: #fff"
> >
<div <div
class="status" class="status"
:class=" :class="
item.runningStatus === '2' ? 'status-running' : '' item.deviceStatus === '1' ? 'status-running' : ''
" "
> >
{{ deviceStatusOptions[item.runningStatus] }} {{ deviceStatusOptions[item.deviceStatus] }}
</div> </div>
<div class="row-items-img"> <div class="row-items-img">
<img <img
class="img-lq" class="img-lq"
:src="require('@/assets/images/ems/lq.png')" :src="require('@/assets/images/ems/lq.png')"
/> />
<div class="name">{{ item.deviceName }}</div> <div class="name">{{ item.deviceName }}</div>
</div> </div>
@ -88,37 +88,37 @@
<!-- 没有上级设备的bms --> <!-- 没有上级设备的bms -->
<div <div
class="row-lists-container" class="row-lists-container"
v-if="bmsHasParentLength !== bms.length" v-if="bmsHasParentLength !== bms.length"
> >
<div class="row-title">BMS({{ bmsNoParent.length }})</div> <div class="row-title">BMS({{ bmsNoParent.length }})</div>
<div <div
class="row-lists" class="row-lists"
v-for="outter in Math.ceil(bmsNoParent.length / 3)" v-for="outter in Math.ceil(bmsNoParent.length / 3)"
:key="outter + 'row'" :key="outter + 'row'"
> >
<template <template
v-for="(item, index) in handlerList('bmsNoParent', outter)" v-for="(item, index) in handlerList('bmsNoParent', outter)"
> >
<div :key="index" class="row-items"> <div :key="index" class="row-items">
<div <div
style="position: relative; z-index: 2; background-color: #fff" style="position: relative; z-index: 2; background-color: #fff"
> >
<div <div
class="status" class="status"
:class=" :class="
item.runningStatus === '2' ? 'status-running' : '' item.deviceStatus === '1' ? 'status-running' : ''
" "
> >
{{ deviceStatusOptions[item.runningStatus] }} {{ deviceStatusOptions[item.deviceStatus] }}
</div> </div>
<div class="row-items-img row-items-img-bms"> <div class="row-items-img row-items-img-bms">
<div style="position:relative;"> <div style="position:relative;">
<img <img
class="img-bms" class="img-bms"
:src="require('@/assets/images/ems/bms.png')" :src="require('@/assets/images/ems/bms.png')"
/> />
<div class="num">{{item.batteryNum || 0}}</div> <div class="num">{{ item.batteryNum || 0 }}</div>
</div> </div>
<div class="name">{{ item.deviceName }}</div> <div class="name">{{ item.deviceName }}</div>
</div> </div>
@ -130,8 +130,8 @@
<!-- pcs --> <!-- pcs -->
<div <div
class="row-lists-container row-lists-container-with-children" class="row-lists-container row-lists-container-with-children"
v-if="showPcs" v-if="showPcs"
> >
<div class="row-title"> <div class="row-title">
PCS({{ pcs.length }}){{ PCS({{ pcs.length }}){{
@ -139,56 +139,56 @@
}} }}
</div> </div>
<div <div
class="row-lists" class="row-lists"
v-for="outter in Math.ceil(pcs.length / 3)" v-for="outter in Math.ceil(pcs.length / 3)"
:key="outter + 'row'" :key="outter + 'row'"
> >
<template v-for="(item, index) in handlerList('pcs', outter)"> <template v-for="(item, index) in handlerList('pcs', outter)">
<div :key="index" class="row-items"> <div :key="index" class="row-items">
<!-- 上级设备 --> <!-- 上级设备 -->
<div class="parent-dash"> <div class="parent-dash">
<div <div
class="status" class="status"
:class=" :class="
item.runningStatus === '2' ? 'status-running' : '' item.deviceStatus === '1' ? 'status-running' : ''
" "
> >
{{ deviceStatusOptions[item.runningStatus] }} {{ deviceStatusOptions[item.deviceStatus] }}
</div> </div>
<div class="row-items-img"> <div class="row-items-img">
<img <img
class="img-pcs" class="img-pcs"
:src="require('@/assets/images/ems/pcs.png')" :src="require('@/assets/images/ems/pcs.png')"
/> />
<div class="name">{{ item.deviceName }}</div> <div class="name">{{ item.deviceName }}</div>
</div> </div>
</div> </div>
<!-- 下级设备 --> <!-- 下级设备 -->
<div <div
class="children-dash" class="children-dash"
v-if="item.children && item.children.length > 0" v-if="item.children && item.children.length > 0"
> >
<div <div
class="status" class="status"
:class=" :class="
item.children[0].runningStatus === '2' item.children[0].deviceStatus === '1'
? 'status-running' ? 'status-running'
: '' : ''
" "
> >
{{ {{
deviceStatusOptions[ deviceStatusOptions[
item.children[0].runningStatus item.children[0].deviceStatus
] ]
}} }}
</div> </div>
<div class="row-items-img row-items-img-bms"> <div class="row-items-img row-items-img-bms">
<div style="position: relative"> <div style="position: relative">
<img <img
class="img-bms" class="img-bms"
:src="require('@/assets/images/ems/bms.png')" :src="require('@/assets/images/ems/bms.png')"
/> />
<div class="num">{{item.batteryNum || 0}}</div> <div class="num">{{ item.batteryNum || 0 }}</div>
</div> </div>
<div class="name">{{ item.children[0].deviceName }}</div> <div class="name">{{ item.children[0].deviceName }}</div>
</div> </div>
@ -204,9 +204,10 @@
</template> </template>
<script> <script>
import { getDeviceList } from "@/api/ems/site"; import {getDeviceList} from "@/api/ems/site";
import getQuerySiteId from "@/mixins/ems/getQuerySiteId"; import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { mapState } from "vuex"; import {mapState} from "vuex";
export default { export default {
name: "DzjkZxlt", name: "DzjkZxlt",
mixins: [getQuerySiteId], mixins: [getQuerySiteId],
@ -223,7 +224,7 @@ export default {
computed: { computed: {
...mapState({ ...mapState({
deviceStatusOptions: (state) => deviceStatusOptions: (state) =>
state.ems.deviceStatusOptions, state.ems.deviceStatusOptions,
}), }),
showPcs() { showPcs() {
@ -245,7 +246,7 @@ export default {
}, },
pcsHasChildren() { pcsHasChildren() {
return this.pcs.filter( return this.pcs.filter(
(item) => item.children && item.children.length > 0 (item) => item.children && item.children.length > 0
); );
}, },
empty() { empty() {
@ -270,46 +271,46 @@ export default {
this.bmsNoParent = []; this.bmsNoParent = [];
this.loading = true; this.loading = true;
getDeviceList(this.siteId) getDeviceList(this.siteId)
.then((response) => { .then((response) => {
const data = JSON.parse(JSON.stringify(response?.data || [])); const data = JSON.parse(JSON.stringify(response?.data || []));
let pcs = [], let pcs = [],
bms = [], bms = [],
db = [], db = [],
lq = [], lq = [],
bmsNoParent = []; bmsNoParent = [];
data.forEach((item) => { data.forEach((item) => {
// 电表 // 电表
if (item.deviceCategory === "AMMETER") { if (item.deviceCategory === "AMMETER") {
db.push({ ...item, children: [] }); db.push({...item, children: []});
} else if (item.deviceCategory === "PCS") { } else if (item.deviceCategory === "PCS") {
// pcs // pcs
pcs.push({ ...item, children: [] }); pcs.push({...item, children: []});
} else if (item.deviceCategory === "STACK") { } else if (item.deviceCategory === "STACK") {
// bms // bms
bms.push({ ...item, children: [] }); bms.push({...item, children: []});
} else if (item.deviceCategory === "COOLING") { } else if (item.deviceCategory === "COOLING") {
// 液冷 // 液冷
lq.push({ ...item, children: [] }); lq.push({...item, children: []});
} }
});
bms.forEach((item, index) => {
if (item.parentId) {
pcs
.find((pcsItem) => pcsItem.deviceId === item.parentId)
.children.push(item);
} else {
bmsNoParent.push(item);
}
});
this.pcs = pcs;
this.bms = bms;
this.lq = lq;
this.db = db;
this.bmsNoParent = bmsNoParent;
})
.finally(() => {
this.loading = false;
}); });
bms.forEach((item, index) => {
if (item.parentId) {
pcs
.find((pcsItem) => pcsItem.deviceId === item.parentId)
.children.push(item);
} else {
bmsNoParent.push(item);
}
});
this.pcs = pcs;
this.bms = bms;
this.lq = lq;
this.db = db;
this.bmsNoParent = bmsNoParent;
})
.finally(() => {
this.loading = false;
});
}, },
}, },
}; };
@ -329,6 +330,7 @@ $lineColor: #86bcc7;
display: flex; display: flex;
position: relative; position: relative;
} }
//云 、计算机 、箭头 //云 、计算机 、箭头
.top { .top {
z-index: 2; z-index: 2;
@ -343,6 +345,7 @@ $lineColor: #86bcc7;
//云 样式 //云 样式
.cloud-container { .cloud-container {
margin: 0 auto; margin: 0 auto;
.cloud { .cloud {
width: 60px; width: 60px;
height: 26px; height: 26px;
@ -354,6 +357,7 @@ $lineColor: #86bcc7;
font-size: 14px; font-size: 14px;
line-height: 26px; line-height: 26px;
} }
.cloud:before, .cloud:before,
.cloud:after { .cloud:after {
content: ""; content: "";
@ -363,20 +367,24 @@ $lineColor: #86bcc7;
height: 30px; height: 30px;
border-radius: 100%; border-radius: 100%;
} }
.cloud:before { .cloud:before {
top: -9px; top: -9px;
left: 8px; left: 8px;
} }
.cloud:after { .cloud:after {
top: -6px; top: -6px;
right: 9px; right: 9px;
} }
} }
//双箭头 //双箭头
.double-arrows { .double-arrows {
height: fit-content; height: fit-content;
margin: 0 10px; margin: 0 10px;
text-align: center; text-align: center;
.top-arrows, .top-arrows,
.bottom-arrows { .bottom-arrows {
height: 4px; height: 4px;
@ -384,6 +392,7 @@ $lineColor: #86bcc7;
background-color: #5ea9df; background-color: #5ea9df;
margin: 0 10px; margin: 0 10px;
position: relative; position: relative;
&::after { &::after {
content: ""; content: "";
position: absolute; position: absolute;
@ -392,9 +401,11 @@ $lineColor: #86bcc7;
height: 0; height: 0;
} }
} }
.top-arrows { .top-arrows {
vertical-align: super; vertical-align: super;
} }
.top-arrows::after { .top-arrows::after {
top: -4px; top: -4px;
border-bottom: 6px solid transparent; border-bottom: 6px solid transparent;
@ -403,8 +414,10 @@ $lineColor: #86bcc7;
border-top: 6px solid transparent; border-top: 6px solid transparent;
left: -11px; left: -11px;
} }
.bottom-arrows { .bottom-arrows {
margin-top: 8px; margin-top: 8px;
&::after { &::after {
top: -4px; top: -4px;
border-top: 6px solid transparent; border-top: 6px solid transparent;
@ -425,6 +438,7 @@ $lineColor: #86bcc7;
font-weight: bold; font-weight: bold;
position: relative; position: relative;
background: #fff; background: #fff;
img { img {
width: 80px; width: 80px;
height: auto; height: auto;
@ -432,6 +446,7 @@ $lineColor: #86bcc7;
} }
} }
} }
.outer-border { .outer-border {
position: relative; position: relative;
width: fit-content; width: fit-content;
@ -448,11 +463,11 @@ $lineColor: #86bcc7;
&:not(:first-child) { &:not(:first-child) {
background: linear-gradient( background: linear-gradient(
to top, to top,
transparent 0%, transparent 0%,
transparent 50%, transparent 50%,
#ccc 50%, #ccc 50%,
#ccc 100% #ccc 100%
); );
background-size: 1px 10px; background-size: 1px 10px;
background-repeat: repeat-y; background-repeat: repeat-y;
@ -467,8 +482,10 @@ $lineColor: #86bcc7;
color: #000; color: #000;
font-weight: bolder; font-weight: bolder;
} }
.row-lists { .row-lists {
position: relative; position: relative;
.row-items { .row-items {
position: relative; position: relative;
padding: 5px 0; padding: 5px 0;
@ -503,17 +520,21 @@ $lineColor: #86bcc7;
border-radius: 2px; border-radius: 2px;
background: #aaaaaa; background: #aaaaaa;
color: #ffffff; color: #ffffff;
&.status-running { &.status-running {
background: #00c69c; background: #00c69c;
} }
} }
// 图片+设备名称 // 图片+设备名称
.row-items-img { .row-items-img {
position: relative; position: relative;
padding-top: 12px; padding-top: 12px;
&.row-items-img-bms{
&.row-items-img-bms {
padding-top: 14px; padding-top: 14px;
.num{
.num {
position: absolute; position: absolute;
top: -2px; top: -2px;
right: -2px; right: -2px;
@ -526,21 +547,26 @@ $lineColor: #86bcc7;
color: #fff; color: #fff;
} }
} }
img { img {
width: 80px; width: 80px;
height: auto; height: auto;
display: block; display: block;
position: relative; position: relative;
&.img-lq { &.img-lq {
width: 50px; width: 50px;
} }
&.img-pcs { &.img-pcs {
width: 50px; width: 50px;
} }
&.img-db { &.img-db {
width: 56px; width: 56px;
} }
} }
.name { .name {
position: absolute; position: absolute;
top: 1px; top: 1px;
@ -559,19 +585,24 @@ $lineColor: #86bcc7;
&:not(:last-child) { &:not(:last-child) {
margin-right: 30px; margin-right: 30px;
} }
.row-items { .row-items {
display: flex; display: flex;
&::after { &::after {
display: none; display: none;
} }
.parent-dash { .parent-dash {
position: relative; position: relative;
z-index: 2; z-index: 2;
background-color: #fff; background-color: #fff;
> div { > div {
z-index: 2; z-index: 2;
position: inherit; position: inherit;
} }
&::after { &::after {
z-index: 1; z-index: 1;
content: ""; content: "";
@ -585,16 +616,19 @@ $lineColor: #86bcc7;
transform: scale(0.4, 1); transform: scale(0.4, 1);
} }
} }
.children-dash { .children-dash {
position: relative; position: relative;
z-index: 2; z-index: 2;
background-color: #fff; background-color: #fff;
margin: -20px 0 0 26px; margin: -20px 0 0 26px;
height: fit-content; height: fit-content;
> div { > div {
z-index: 2; z-index: 2;
position: inherit; position: inherit;
} }
&::after { &::after {
z-index: 1; z-index: 1;
content: ""; content: "";

View File

@ -1,68 +1,68 @@
<template> <template>
<div <div
v-loading="loading" v-loading="loading"
class="ems-dashboard-editor-container" class="ems-dashboard-editor-container"
style="background-color: #ffffff" style="background-color: #ffffff"
> >
<el-form ref="form" :model="form" label-position="top"> <el-form ref="form" :model="form" label-position="top">
<el-form-item <el-form-item
label="站点" label="站点"
prop="siteIds" prop="siteIds"
:rules="[{ required: true, message: '请选择站点' }]" :rules="[{ required: true, message: '请选择站点' }]"
> >
<el-radio-group v-model="form.siteIds" @input="changeSiteIds"> <el-radio-group v-model="form.siteIds" @input="changeSiteIds">
<el-radio <el-radio
v-for="(item, index) in siteList" v-for="(item, index) in siteList"
:key="index + 'zdListSearch'" :key="index + 'zdListSearch'"
:label="item.siteId" :label="item.siteId"
> >
{{ item.siteName }} {{ item.siteName }}
</el-radio> </el-radio>
</el-radio-group> </el-radio-group>
</el-form-item> </el-form-item>
<el-form-item <el-form-item
label="设备" label="设备"
prop="deviceCategory" prop="deviceCategory"
:rules="[{ required: true, message: '请选择设备' }]" :rules="[{ required: true, message: '请选择设备' }]"
> >
<el-radio-group v-model="form.deviceCategory" @input="changeSiteIds"> <el-radio-group v-model="form.deviceCategory" @input="changeSiteIds">
<el-radio <el-radio
v-for="(item, index) in deviceCategoryList" v-for="(item, index) in deviceCategoryList"
:key="index + 'sbListSearch'" :key="index + 'sbListSearch'"
:label="item.code" :label="item.code"
> >
{{ item.name }} {{ item.name }}
</el-radio> </el-radio>
</el-radio-group> </el-radio-group>
</el-form-item> </el-form-item>
<el-form-item <el-form-item
v-if="isDtdc" v-if="isDtdc"
label="单体电池(不超过5个)" label="单体电池(不超过5个)"
prop="child" prop="child"
:rules="[{ required: true, message: '请选择单体电池' }]" :rules="[{ required: true, message: '请选择单体电池' }]"
> >
<el-cascader <el-cascader
v-model="form.child" v-model="form.child"
style="width: 400px" style="width: 400px"
:props="{ multiple: true }" :props="{ multiple: true }"
:show-all-levels="false" :show-all-levels="false"
:options="childOptions" :options="childOptions"
@change="handleChildChange" @change="handleChildChange"
></el-cascader> ></el-cascader>
</el-form-item> </el-form-item>
<div style="display: flex"> <div style="display: flex">
<el-form-item <el-form-item
label="点位" label="点位"
prop="pointName" prop="pointName"
:rules="[{ required: true, message: '请输入点位' }]" :rules="[{ required: true, message: '请输入点位' }]"
style="margin-right: 50px" style="margin-right: 50px"
> >
<el-autocomplete <el-autocomplete
v-model="form.pointName" v-model="form.pointName"
placeholder="请输入点位" placeholder="请输入点位"
clearable clearable
:fetch-suggestions="querySearchAsync" :fetch-suggestions="querySearchAsync"
@select="handleSelect" @select="handleSelect"
></el-autocomplete> ></el-autocomplete>
</el-form-item> </el-form-item>
</div> </div>
@ -71,8 +71,8 @@
</el-form-item> </el-form-item>
</el-form> </el-form>
<el-card <el-card
shadow="always" shadow="always"
class="common-card-container common-card-container-body-no-padding time-range-card" class="common-card-container common-card-container-body-no-padding time-range-card"
> >
<div slot="header" class="time-range-header"> <div slot="header" class="time-range-header">
<span class="card-title"> <span class="card-title">
@ -83,10 +83,10 @@
</el-radio-group> </el-radio-group>
</span> </span>
<date-time-select <date-time-select
ref="dateTimeSelect" ref="dateTimeSelect"
:data-unit="form.dataUnit" :data-unit="form.dataUnit"
@initDate="(e) => (form.dataRange = e || [])" @initDate="(e) => (form.dataRange = e || [])"
@updateDate="updateDate" @updateDate="updateDate"
/> />
</div> </div>
<div style="height: 350px" id="searchChart"></div> <div style="height: 350px" id="searchChart"></div>
@ -97,18 +97,14 @@
<script> <script>
import * as echarts from "echarts"; import * as echarts from "echarts";
import resize from "@/mixins/ems/resize"; import resize from "@/mixins/ems/resize";
import { getAllSites } from "@/api/ems/zddt"; import {getAllSites} from "@/api/ems/zddt";
import { import {getAllBatteryIdsBySites, getAllDeviceCategory, getPointValueList, pointFuzzyQuery,} from "@/api/ems/search";
getAllDeviceCategory,
getPointValueList,
pointFuzzyQuery,
getAllBatteryIdsBySites,
} from "@/api/ems/search";
import DateTimeSelect from "./DateTimeSelect.vue"; import DateTimeSelect from "./DateTimeSelect.vue";
export default { export default {
name: "Search", name: "Search",
mixins: [resize], mixins: [resize],
components: { DateTimeSelect }, components: {DateTimeSelect},
computed: { computed: {
isDtdc() { isDtdc() {
return this.form.deviceCategory === "BATTERY"; return this.form.deviceCategory === "BATTERY";
@ -161,7 +157,7 @@ export default {
getChildList() { getChildList() {
this.childOptions = []; this.childOptions = [];
this.form.child = []; this.form.child = [];
const { siteIds } = this.form; const {siteIds} = this.form;
getAllBatteryIdsBySites([siteIds]).then((response) => { getAllBatteryIdsBySites([siteIds]).then((response) => {
const data = response?.data || {}; const data = response?.data || {};
const base = 50; const base = 50;
@ -178,7 +174,7 @@ export default {
if (num === 0) return; if (num === 0) return;
for (let i = 1; i <= num; i++) { for (let i = 1; i <= num; i++) {
const start = (i - 1) * base + 1, const start = (i - 1) * base + 1,
end = i * base; end = i * base;
options[index].children.push({ options[index].children.push({
value: i, value: i,
label: `${start}-${end}`, label: `${start}-${end}`,
@ -214,76 +210,88 @@ export default {
this.getDate(); this.getDate();
}, },
setOption(data) { setOption(data) {
// 点位类型 dataType 1-瞬时值 2-累计值 仅当值为2展示差值
// 图表类型 chartType 1-曲线图2-箱线图
if (!this.chart) return; if (!this.chart) return;
this.chart.clear(); this.chart.clear();
console.log("返回的数据", data); console.log("返回的数据", data);
let dataset = []; if (!data || data.length <= 0) {
if (data.length > 0) {
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}`
: ""
}${inner.deviceId}`,
type: "line",
markPoint: {
symbolSize: 30,
emphasis: {
disabled:false//打开 鼠标高亮
},
data: [//最大值、最小值
{
// type: 'max',
name: `最大值`,
coord:[inner.maxDate,inner.maxValue],
relativeTo:'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
},
},
{
// type: 'min',
name: `最小值`,
coord:[inner.minDate,inner.minValue],
relativeTo:'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
}
}
]
},
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
dataset[length - 1].xdata.push(value.valueDate);
dataset[length - 1].data.push(value.pointValue);
});
});
});
} else {
this.$message.warning("暂无数据"); this.$message.warning("暂无数据");
} }
console.log("图表数据", dataset); console.log('展示的图表类型chartType', data[0].chartType)
if (data[0].chartType === 2) {
// 箱型图
this.setBoxOption(data)
} else {
//折线图
this.setLineOption(data)
}
},
setLineOption(data) {
let dataset = [];
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}`
: ""
}${inner.deviceId}`,
type: "line",
markPoint: {
symbolSize: 30,
emphasis: {
disabled: false//打开 鼠标高亮
},
data: [//最大值、最小值
{
// type: 'max',
name: `最大值`,
coord: [inner.maxDate, inner.maxValue],
relativeTo: 'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最大值:${inner.maxValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
},
},
{
// type: 'min',
name: `最小值`,
coord: [inner.minDate, inner.minValue],
relativeTo: 'coordinate',
label: {
position: "top",
formatter: item.dataType === 2 ? ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
`差值:${inner.diffValue}`,
]).join('\n') : ([
`最小值:${inner.minValue}`,
// `平均值:${inner.avgValue}`,
]).join('\n'),
}
}
]
},
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
dataset[length - 1].xdata.push(value.valueDate);
dataset[length - 1].data.push(value.pointValue);
});
});
});
console.log("折线图图表数据", dataset);
this.chart.setOption({ this.chart.setOption({
legend: { legend: {
// left: 'center', // left: 'center',
@ -305,7 +313,120 @@ export default {
textStyle: { textStyle: {
color: "#333333", color: "#333333",
}, },
xAxis: { type: "category", data: dataset?.[0]?.xdata || [] }, xAxis: {type: "category", data: dataset?.[0]?.xdata || []},
yAxis: {
type: "value",
},
dataZoom: [
{
type: "inside",
start: 0,
end: 100,
},
{
start: 0,
end: 100,
},
],
series: dataset,
});
},
setBoxOption(data) {
let dataset = [];
data.forEach((item, index) => {
item.deviceList.forEach((inner) => {
dataset.push({
name: `${
this.isDtdc
? `${inner.parentDeviceId ? inner.parentDeviceId + "-" : ""}`
: ""
}${inner.deviceId}`,
type: "boxplot",
// markPoint: {
// symbolSize: 30,
// emphasis: {
// disabled: false//打开 鼠标高亮
// },
// data: [//最大值、最小值
// {
// // type: 'max',
// name: `最大值`,
// coord: [inner.maxDate, inner.maxValue],
// relativeTo: 'coordinate',
// label: {
// position: "top",
// formatter: item.dataType === 2 ? ([
// `最大值:${inner.maxValue}`,
// // `平均值:${inner.avgValue}`,
// `差值:${inner.diffValue}`,
// ]).join('\n') : ([
// `最大值:${inner.maxValue}`,
// // `平均值:${inner.avgValue}`,
// ]).join('\n'),
// },
// },
// {
// // type: 'min',
// name: `最小值`,
// coord: [inner.minDate, inner.minValue],
// relativeTo: 'coordinate',
// label: {
// position: "top",
// formatter: item.dataType === 2 ? ([
// `最小值:${inner.minValue}`,
// // `平均值:${inner.avgValue}`,
// `差值:${inner.diffValue}`,
// ]).join('\n') : ([
// `最小值:${inner.minValue}`,
// // `平均值:${inner.avgValue}`,
// ]).join('\n'),
// }
// }
// ]
// },
xdata: [],
data: [],
});
const length = dataset.length;
inner.pointValueList.forEach((value) => {
const {valueDate, min, q1, median, q3, max} = value
dataset[length - 1].xdata.push(valueDate);
dataset[length - 1].data.push([min, q1, median, q3, max]);
});
});
});
console.log("箱型图图表数据", dataset);
this.chart.setOption({
legend: {
// left: 'center',
// top: '10',
},
grid: {
containLabel: true,
},
tooltip: {
trigger: 'item',
formatter: function (params) {
let data = params.data;
let result = params.marker + params.name + ' ' + params.seriesName + '<br/>';
result += '最小值: ' + data[1] + '<br/>';
result += '平均值: ' + data[3] + '<br/>';
result += '最大值: ' + data[5];
return result;
}
// trigger: "axis",
// axisPointer: {
// type: 'cross',
// },
// axisPointer: {
// // 坐标轴指示器,坐标轴触发有效
// type: "shadow", // 默认为直线,可选为:'line' | 'shadow'
// },
},
textStyle: {
color: "#333333",
},
xAxis: {type: "category", data: dataset?.[0]?.xdata || []},
yAxis: { yAxis: {
type: "value", type: "value",
}, },
@ -345,9 +466,9 @@ export default {
}).then((response) => { }).then((response) => {
const data = response?.data || []; const data = response?.data || [];
cb( cb(
data.map((item) => { data.map((item) => {
return { name: item, value: item }; return {name: item, value: item};
}) })
); );
}); });
}, },
@ -359,10 +480,11 @@ export default {
}, },
getZdList() { getZdList() {
return getAllSites() return getAllSites()
.then((response) => { .then((response) => {
this.siteList = response.data || []; this.siteList = response.data || [];
}) })
.finally(() => {}); .finally(() => {
});
}, },
getDate() { getDate() {
this.$refs.form.validate((valid) => { this.$refs.form.validate((valid) => {
@ -371,8 +493,8 @@ export default {
return this.$message.error("请输入正确的点位"); return this.$message.error("请输入正确的点位");
} }
if ( if (
this.isDtdc && this.isDtdc &&
(this.form.child.length === 0 || this.form.child.length > 5) (this.form.child.length === 0 || this.form.child.length > 5)
) { ) {
return this.$message.error("请选择单体电池且不能超过5个"); return this.$message.error("请选择单体电池且不能超过5个");
} }
@ -417,12 +539,12 @@ export default {
endDate, endDate,
siteDeviceMap, siteDeviceMap,
}) })
.then((response) => { .then((response) => {
this.setOption(response?.data || []); this.setOption(response?.data || []);
}) })
.finally(() => { .finally(() => {
this.loading = false; this.loading = false;
}); });
} }
}); });
}, },
@ -440,7 +562,7 @@ export default {
this.initChart(); this.initChart();
this.$refs.dateTimeSelect.init(); this.$refs.dateTimeSelect.init();
Promise.all([this.getDeviceCategory(), this.getZdList()]).finally( Promise.all([this.getDeviceCategory(), this.getZdList()]).finally(
() => (this.loading = false) () => (this.loading = false)
); );
}); });
}, },

View File

@ -1,11 +1,23 @@
<template> <template>
<el-dialog :visible.sync="dialogTableVisible" :close-on-press-escape="false" :close-on-click-modal="false" :show-close="false" destroy-on-close lock-scroll append-to-body width="400px" class="ems-dialog" :title="mode === 'add'?'新增配置':`编辑配置` " > <el-dialog :visible.sync="dialogTableVisible" :close-on-press-escape="false" :close-on-click-modal="false" :show-close="false" destroy-on-close lock-scroll append-to-body width="400px" class="ems-dialog" :title="mode === 'add'?'新增配置':`编辑配置` " >
<el-form v-loading="loading>0" ref="addTempForm" :model="formData" :rules="rules" size="medium" label-width="140px"> <el-form v-loading="loading>0" ref="addTempForm" :model="formData" :rules="rules" size="medium" label-width="140px">
<el-form-item label="订阅topic" prop="deviceName"> <el-form-item label="站点" prop="siteId">
<el-select v-model="formData.siteId" :disabled="mode === 'edit'" placeholder="请选择站点" :loading="searchLoading" loading-text="正在加载数据">
<el-option v-for="(item,index) in siteList" :key="index+'zdxeSelect'" :label="item.siteName" :value="item.siteId" ></el-option>
</el-select>
</el-form-item>
<el-form-item label="消息等级" prop="qos">
<el-select v-model="formData.qos" placeholder="请选择消息等级">
<el-option :value="1">1</el-option>
<el-option :value="2">2</el-option>
<el-option :value="3">3</el-option>
</el-select>
</el-form-item>
<el-form-item label="订阅topic" prop="mqttTopic">
<el-input v-model="formData.mqttTopic" placeholder="请输入" clearable :style="{width: '100%'}"> <el-input v-model="formData.mqttTopic" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input> </el-input>
</el-form-item> </el-form-item>
<el-form-item label="topic描述" prop="description"> <el-form-item label="topic描述" prop="topicName">
<el-input v-model="formData.topicName" type="textarea" placeholder="请输入" clearable :style="{width: '100%'}"> <el-input v-model="formData.topicName" type="textarea" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input> </el-input>
</el-form-item> </el-form-item>
@ -18,18 +30,29 @@
</template> </template>
<script> <script>
import {editMqtt,addMqtt,getMqttDetail} from "@/api/ems/site"; import {editMqtt,addMqtt,getMqttDetail} from "@/api/ems/site";
import {getAllSites} from '@/api/ems/zddt'
export default { export default {
data() { data() {
return { return {
loading:0, loading:0,
siteList:[],
searchLoading:false,
dialogTableVisible:false, dialogTableVisible:false,
mode:'', mode:'',
formData: { formData: {
id:'',//设备唯一标识 id:'',//设备唯一标识
siteId:'',
qos:'',
topicName:'', topicName:'',
mqttTopic:'' mqttTopic:''
}, },
rules: { rules: {
siteId:[
{ required: true, message: '请选择站点', trigger: 'blur'},
],
qos:[
{ required: true, message: '请选择消息等级', trigger: 'blur'},
],
mqttTopic:[ mqttTopic:[
{ required: true, message: '请输入订阅topic', trigger: 'blur'}, { required: true, message: '请输入订阅topic', trigger: 'blur'},
], ],
@ -42,6 +65,7 @@ export default {
methods: { methods: {
showDialog(id){ showDialog(id){
this.dialogTableVisible = true this.dialogTableVisible = true
this.getZdList()
if(id){ if(id){
this.mode = 'edit' this.mode = 'edit'
this.formData.id = id this.formData.id = id
@ -49,12 +73,21 @@ export default {
}else{ }else{
this.mode = 'add' this.mode = 'add'
} }
},
//获取站点列表
getZdList(){
this.searchLoading=true
getAllSites().then(response => {
this.siteList = response?.data || []
}).finally(() => {this.searchLoading=false})
}, },
getDetail(id){ getDetail(id){
getMqttDetail(id).then(response => { getMqttDetail(id).then(response => {
const {topicName='',mqttTopic=''} = JSON.parse(JSON.stringify(response?.data || {})); const {topicName='',mqttTopic='',qos='',siteId=''} = JSON.parse(JSON.stringify(response?.data || {}));
this.formData.mqttTopic=mqttTopic; this.formData.mqttTopic=mqttTopic;
this.formData.topicName=topicName; this.formData.topicName=topicName;
this.formData.qos=qos;
this.formData.siteId=siteId;
}) })
}, },
saveDialog() { saveDialog() {
@ -63,11 +96,13 @@ export default {
this.loading+=1 this.loading+=1
const { const {
id='', id='',
siteId='',
qos='',
mqttTopic='',//站点ID mqttTopic='',//站点ID
topicName='',//设备id topicName='',//设备id
}= this.formData; }= this.formData;
if(this.mode === 'add'){ if(this.mode === 'add'){
addMqtt( {mqttTopic,topicName}).then(response => { addMqtt( {mqttTopic,topicName,siteId,qos}).then(response => {
if(response.code === 200){ if(response.code === 200){
//新增成功 //新增成功
// 关闭弹窗 更新表格 // 关闭弹窗 更新表格
@ -78,7 +113,7 @@ export default {
this.loading-=1 this.loading-=1
}) })
}else{ }else{
editMqtt({mqttTopic,topicName,id}).then(response => { editMqtt({mqttTopic,topicName,id,siteId,qos}).then(response => {
if(response.code === 200){ if(response.code === 200){
//新增成功 //新增成功
// 关闭弹窗 更新表格 // 关闭弹窗 更新表格

View File

@ -1,6 +1,11 @@
<template> <template>
<div class="ems-dashboard-editor-container" style="background-color: #ffffff" v-loading="loading"> <div class="ems-dashboard-editor-container" style="background-color: #ffffff" v-loading="loading">
<el-form :inline="true" class="select-container"> <el-form :inline="true" class="select-container">
<el-form-item label="站点选择">
<el-select v-model="form.siteId" placeholder="请选择换电站名称" :loading="searchLoading" loading-text="正在加载数据" clearable>
<el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList" :key="index+'zdxeSelect'"></el-option>
</el-select>
</el-form-item>
<el-form-item label="订阅topic"> <el-form-item label="订阅topic">
<el-input <el-input
v-model="form.mqttTopic" v-model="form.mqttTopic"
@ -29,6 +34,14 @@
max-height="600px" max-height="600px"
stripe stripe
style="width: 100%;margin-top: 25px"> style="width: 100%;margin-top: 25px">
<el-table-column
label="站点"
prop="siteId">
</el-table-column>
<el-table-column
label="消息等级"
prop="qos">
</el-table-column>
<el-table-column <el-table-column
label="订阅topic" label="订阅topic"
prop="mqttTopic"> prop="mqttTopic">
@ -77,6 +90,7 @@
<script> <script>
import {deleteMqtt,getMqttList} from '@/api/ems/site' import {deleteMqtt,getMqttList} from '@/api/ems/site'
import {getAllSites} from '@/api/ems/zddt'
import AddMqtt from './AddMqtt.vue' import AddMqtt from './AddMqtt.vue'
export default { export default {
name: "Mqtt", name: "Mqtt",
@ -85,9 +99,12 @@ export default {
data() { data() {
return { return {
form:{ form:{
siteId:"",
topicName:'', topicName:'',
mqttTopic:'' mqttTopic:''
}, },
siteList:[],
searchLoading:false,
loading:false, loading:false,
tableData:[], tableData:[],
pageSize:10,//分页栏当前每个数据总数 pageSize:10,//分页栏当前每个数据总数
@ -116,17 +133,26 @@ export default {
}, },
onReset(){ onReset(){
this.form={ this.form={
siteId:'',
topicName:'', topicName:'',
mqttTopic:'' mqttTopic:''
} }
this.pageNum =1//每次搜索从1开始搜索 this.pageNum =1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
//获取站点列表
getZdList(){
this.searchLoading=true
return getAllSites().then(response => {
this.siteList = response?.data || []
// if( this.siteList.length>0 ) this.siteId = this.siteList[0].siteId
}).finally(() => {this.searchLoading=false})
},
getData(){ getData(){
this.loading=true; this.loading=true;
const {mqttTopic,topicName} = this.form; const {mqttTopic,topicName,siteId} = this.form;
const {pageNum,pageSize} = this; const {pageNum,pageSize} = this;
getMqttList({pageNum,pageSize,mqttTopic,topicName}).then(response => { getMqttList({pageNum,pageSize,mqttTopic,topicName,siteId}).then(response => {
this.tableData=response?.rows || []; this.tableData=response?.rows || [];
this.totalSize = response?.total || 0 this.totalSize = response?.total || 0
}).finally(() => {this.loading=false}) }).finally(() => {this.loading=false})
@ -168,6 +194,7 @@ export default {
}, },
mounted() { mounted() {
this.loading=true this.loading=true
this.getZdList()
this.getData() this.getData()
} }
} }

View File

@ -1,42 +1,42 @@
<template> <template>
<el-dialog <el-dialog
v-loading="loading" v-loading="loading"
width="90%" width="90%"
:visible.sync="dialogTableVisible" :visible.sync="dialogTableVisible"
class="ems-dialog" class="ems-dialog"
title="保护方案" title="保护方案"
:close-on-click-modal="false" :close-on-click-modal="false"
:show-close="false" :show-close="false"
> >
<el-form <el-form
v-loading="loading > 0" v-loading="loading > 0"
ref="addTempForm" ref="addTempForm"
:model="formData" :model="formData"
:rules="rules" :rules="rules"
size="medium" size="medium"
label-width="140px" label-width="140px"
> >
<el-form-item label="站点" prop="siteId"> <el-form-item label="站点" prop="siteId">
<el-select <el-select
v-model="formData.siteId" v-model="formData.siteId"
placeholder="请选择" placeholder="请选择"
:style="{ width: '50%' }" :style="{ width: '50%' }"
@change="changeType" @change="changeType"
> >
<el-option <el-option
:label="item.siteName" :label="item.siteName"
:value="item.siteId" :value="item.siteId"
v-for="(item, index) in siteList" v-for="(item, index) in siteList"
:key="index + 'siteOptions'" :key="index + 'siteOptions'"
></el-option> ></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="设备保护名称" prop="faultName"> <el-form-item label="设备保护名称" prop="faultName">
<el-input <el-input
v-model="formData.faultName" v-model="formData.faultName"
placeholder="请输入" placeholder="请输入"
clearable clearable
:style="{ width: '50%' }" :style="{ width: '50%' }"
> >
</el-input> </el-input>
</el-form-item> </el-form-item>
@ -53,9 +53,9 @@
</el-form-item> </el-form-item>
<el-form-item label="是否告警" prop="isAlert"> <el-form-item label="是否告警" prop="isAlert">
<el-checkbox <el-checkbox
v-model="formData.isAlert" v-model="formData.isAlert"
:true-label="1" :true-label="1"
:false-label="0" :false-label="0"
></el-checkbox> ></el-checkbox>
</el-form-item> </el-form-item>
<el-form-item label="告警等级" prop="faultLevel"> <el-form-item label="告警等级" prop="faultLevel">
@ -81,11 +81,11 @@
</div> </div>
<div> <div>
<el-button <el-button
@click.native.prevent="addRow('protectionSettings')" @click.native.prevent="addRow('protectionSettings')"
block block
type="primary" type="primary"
size="mini" size="mini"
style="margin-bottom: 20px" style="margin-bottom: 20px"
> >
新增保护前提 新增保护前提
</el-button> </el-button>
@ -103,38 +103,40 @@
<div>操作</div> <div>操作</div>
</div> </div>
<div <div
class="time-lists" class="time-lists"
v-for="(item, index) in protectionSettings" v-for="(item, index) in protectionSettings"
:key="'protectionSettings' + index" :key="'protectionSettings' + index"
> >
<div> <div>
<el-cascader <el-cascader
v-model="item.deviceId" v-model="item.deviceId"
:options="childOptions" :options="childOptions"
:props="props"
:show-all-levels="false" :show-all-levels="false"
:ref="'protectionSettings'+index"
@change="(v)=>handleChange(v,'protectionSettings',index)" @change="(v)=>handleChange(v,'protectionSettings',index)"
></el-cascader> ></el-cascader>
</div> </div>
<div> <div>
<el-autocomplete <el-autocomplete
v-model="item.point" v-model="item.point"
placeholder="请输入点位" placeholder="请输入点位"
clearable clearable
:fetch-suggestions=" :fetch-suggestions="
(q, c) => (q, c) =>
querySearchAsync(q, c, index, 'protectionSettings') querySearchAsync(q, c, index, 'protectionSettings')
" "
@select="(v) => handleSelect(v, index, 'protectionSettings')" @select="(v) => handleSelect(v, index, 'protectionSettings')"
></el-autocomplete> ></el-autocomplete>
</div> </div>
<div> <div>
<el-select v-model="item.faultOperator" placeholder="请选择"> <el-select v-model="item.faultOperator" placeholder="请选择">
<el-option <el-option
v-for="(value, key) in comparisonOperatorOptions" v-for="(value, key) in comparisonOperatorOptions"
:key="key + 'faultOperator'" :key="key + 'faultOperator'"
:label="key" :label="key"
:value="value" :value="value"
></el-option> ></el-option>
</el-select> </el-select>
</div> </div>
@ -145,35 +147,35 @@
<div> <div>
<el-select v-model="item.releaseOperator" placeholder="请选择"> <el-select v-model="item.releaseOperator" placeholder="请选择">
<el-option <el-option
v-for="(value, key) in comparisonOperatorOptions" v-for="(value, key) in comparisonOperatorOptions"
:key="key + 'releaseOperator'" :key="key + 'releaseOperator'"
:label="key" :label="key"
:value="value" :value="value"
></el-option> ></el-option>
</el-select> </el-select>
</div> </div>
<div> <div>
<el-input <el-input
placeholder="请输入释放值" placeholder="请输入释放值"
v-model="item.releaseValue" v-model="item.releaseValue"
> >
</el-input> </el-input>
</div> </div>
<div> <div>
<el-select v-model="item.relationNext" placeholder="请选择"> <el-select v-model="item.relationNext" placeholder="请选择">
<el-option <el-option
v-for="(value, key) in relationWithPoint" v-for="(value, key) in relationWithPoint"
:key="key + 'relation'" :key="key + 'relation'"
:label="key" :label="key"
:value="value" :value="value"
></el-option> ></el-option>
</el-select> </el-select>
</div> </div>
<div> <div>
<el-button <el-button
@click.native.prevent="deleteRow(index,'protectionSettings')" @click.native.prevent="deleteRow(index,'protectionSettings')"
type="warning" type="warning"
size="mini" size="mini"
> >
删除 删除
</el-button> </el-button>
@ -198,11 +200,11 @@
</div> </div>
<div> <div>
<el-button <el-button
@click.native.prevent="addRow('protectionPlan')" @click.native.prevent="addRow('protectionPlan')"
block block
type="primary" type="primary"
size="mini" size="mini"
style="margin-bottom: 20px" style="margin-bottom: 20px"
> >
新增保护方案 新增保护方案
</el-button> </el-button>
@ -217,27 +219,29 @@
<div>操作</div> <div>操作</div>
</div> </div>
<div <div
class="time-lists" class="time-lists"
v-for="(item, index) in protectionPlan" v-for="(item, index) in protectionPlan"
:key="'protectionPlan' + index" :key="'protectionPlan' + index"
> >
<div> <div>
<el-cascader <el-cascader
v-model="item.deviceId" v-model="item.deviceId"
:show-all-levels="false" :show-all-levels="false"
:options="childOptions" :options="childOptions"
:props="props"
:ref="'protectionPlan'+index"
@change="(v)=>handleChange(v,'protectionPlan',index)" @change="(v)=>handleChange(v,'protectionPlan',index)"
></el-cascader> ></el-cascader>
</div> </div>
<div> <div>
<el-autocomplete <el-autocomplete
v-model="item.point" v-model="item.point"
placeholder="请输入点位" placeholder="请输入点位"
clearable clearable
:fetch-suggestions=" :fetch-suggestions="
(q, c) => querySearchAsync(q, c, index, 'protectionPlan') (q, c) => querySearchAsync(q, c, index, 'protectionPlan')
" "
@select="(v) => handleSelect(v, index, 'protectionPlan')" @select="(v) => handleSelect(v, index, 'protectionPlan')"
></el-autocomplete> ></el-autocomplete>
</div> </div>
@ -248,9 +252,9 @@
</div> </div>
<div> <div>
<el-button <el-button
@click.native.prevent="deleteRow(index,'protectionPlan')" @click.native.prevent="deleteRow(index,'protectionPlan')"
type="warning" type="warning"
size="mini" size="mini"
> >
删除 删除
</el-button> </el-button>
@ -267,16 +271,12 @@
</el-dialog> </el-dialog>
</template> </template>
<script> <script>
import { mapState } from "vuex"; import {mapState} from "vuex";
import { getAllSites } from "@/api/ems/zddt"; import {getAllSites} from "@/api/ems/zddt";
import { validText } from "@/utils/validate"; import {validText} from "@/utils/validate";
import { import {addProtectPlan, getDeviceListBySiteAndCategory, getProtectPlan, updateProtectPlan} from "@/api/ems/site";
updateProtectPlan, import {getAllDeviceCategory, pointFuzzyQuery} from "@/api/ems/search";
addProtectPlan,
getProtectPlan,
getDeviceListBySiteAndCategory
} from "@/api/ems/site";
import { getAllDeviceCategory, pointFuzzyQuery } from "@/api/ems/search";
export default { export default {
data() { data() {
const validateText = (rule, value, callback) => { const validateText = (rule, value, callback) => {
@ -287,9 +287,17 @@ export default {
} }
}; };
return { return {
mode:'', props: {
// emitPath: false,//在选中节点改变时,是否返回由该节点所在的各级菜单的值所组成的数组,若设置 false则只返回该节点的值
lazy: true,
lazyLoad: (node, resolve) => {
console.log('---------node', node)
this.getDeviceList(node, resolve)
}
},
mode: '',
loading: 0, loading: 0,
childOptions:[], childOptions: [],
protectionSettings: [], protectionSettings: [],
protectionPlan: [], protectionPlan: [],
dialogTableVisible: false, dialogTableVisible: false,
@ -302,7 +310,7 @@ export default {
faultLevel: 1, //告警等级 faultLevel: 1, //告警等级
faultDelaySeconds: "", //故障延时 faultDelaySeconds: "", //故障延时
releaseDelaySeconds: "", //释放延时 releaseDelaySeconds: "", //释放延时
description:'',//方案描述 description: '',//方案描述
}, },
rules: { rules: {
siteId: [ siteId: [
@ -313,22 +321,22 @@ export default {
}, },
], ],
faultName: [ faultName: [
{ required: true, message: "请输入设备保护名称", trigger: "blur" }, {required: true, message: "请输入设备保护名称", trigger: "blur"},
], ],
isAlert: [ isAlert: [
{ required: true, message: "请选择是否告警", trigger: "blur" }, {required: true, message: "请选择是否告警", trigger: "blur"},
], ],
description: [ description: [
{ required: true, message: "请输入设备描述", trigger: "blur" }, {required: true, message: "请输入设备描述", trigger: "blur"},
{ validator: validateText, trigger: "blur" }, {validator: validateText, trigger: "blur"},
], ],
faultDelaySeconds: [ faultDelaySeconds: [
{ required: true, message: "请输入保护前提延时", trigger: "blur" }, {required: true, message: "请输入保护前提延时", trigger: "blur"},
{ validator: validateText, trigger: "blur" }, {validator: validateText, trigger: "blur"},
], ],
releaseDelaySeconds: [ releaseDelaySeconds: [
{ required: true, message: "请输入保护方案延时", trigger: "blur" }, {required: true, message: "请输入保护方案延时", trigger: "blur"},
{ validator: validateText, trigger: "blur" }, {validator: validateText, trigger: "blur"},
], ],
}, },
}; };
@ -336,24 +344,24 @@ export default {
computed: { computed: {
...mapState({ ...mapState({
communicationStatusOptions: (state) => communicationStatusOptions: (state) =>
state?.ems?.communicationStatusOptions || {}, state?.ems?.communicationStatusOptions || {},
deviceTypeOptions: (state) => state?.ems?.deviceTypeOptions || {}, deviceTypeOptions: (state) => state?.ems?.deviceTypeOptions || {},
comparisonOperatorOptions: (state) => comparisonOperatorOptions: (state) =>
state?.ems?.comparisonOperatorOptions || {}, state?.ems?.comparisonOperatorOptions || {},
relationWithPoint: (state) => state?.ems?.relationWithPoint || {}, relationWithPoint: (state) => state?.ems?.relationWithPoint || {},
}), }),
}, },
methods: { methods: {
open(id,siteId){ open(id, siteId) {
this.dialogTableVisible=true this.dialogTableVisible = true
this.getZdList(); this.getZdList();
this.getDeviceCategoryList().then(()=>{ this.getDeviceCategoryList().then(() => {
if(id && siteId) { if (id && siteId) {
this.getDeviceList('PCS',siteId) // this.getDeviceList('PCS', siteId)
this.getDeviceList('STACK',siteId) // this.getDeviceList('STACK', siteId)
} }
}); });
if(id){ if (id) {
this.formData.id = id this.formData.id = id
this.mode = 'edit' this.mode = 'edit'
getProtectPlan(id).then(response => { getProtectPlan(id).then(response => {
@ -365,53 +373,74 @@ export default {
isAlert: data?.isAlert || 0, //是否告警 isAlert: data?.isAlert || 0, //是否告警
faultLevel: data?.faultLevel || 1, //告警等级 faultLevel: data?.faultLevel || 1, //告警等级
faultDelaySeconds: data?.faultDelaySeconds || "", //故障延时 faultDelaySeconds: data?.faultDelaySeconds || "", //故障延时
releaseDelaySeconds: data?.releaseDelaySeconds ||"", //释放延时 releaseDelaySeconds: data?.releaseDelaySeconds || "", //释放延时
description: data?.description ||'',//方案描述 description: data?.description || '',//方案描述
} }
const plan =(JSON.parse(data?.protectionPlan || [])).map(item=>{ const plan = (JSON.parse(data?.protectionPlan || [])).map(item => {
return Object.assign({},item,{ const index = this.childOptions.findIndex(i => i.value === item.deviceCategory)
deviceId:[item.deviceCategory || '',item.deviceId || ''], if (index > -1) {
!this.childOptions[index].children.find(i => i.value === item.deviceId) && this.childOptions[index].children.push({
value: item.deviceId,
label: item.deviceName,
leaf: true
})
}
return Object.assign({}, item, {
deviceId: [item.deviceCategory || '', item.deviceId || ''],
deviceName: [item.categoryName || '', item.deviceName || ''],
}) })
}) })
const settings =(JSON.parse(data?.protectionSettings || [])).map(item=>{ const settings = (JSON.parse(data?.protectionSettings || [])).map(item => {
return Object.assign({},item,{ const index = this.childOptions.findIndex(i => i.value === item.deviceCategory)
deviceId:[item.deviceCategory || '',item.deviceId || ''], if (index > -1) {
!this.childOptions[index].children.find(i => i.value === item.deviceId) && this.childOptions[index].children.push({
value: item.deviceId,
label: item.deviceName,
leaf: true
})
}
return Object.assign({}, item, {
deviceId: [item.deviceCategory || '', item.deviceId || ''],
deviceName: [item.categoryName || '', item.deviceName || ''],
}) })
}) })
this.$nextTick(()=>{ this.$nextTick(() => {
this.protectionPlan.splice(0,0,...plan) this.protectionPlan.splice(0, 0, ...plan)
this.protectionSettings.splice(0,0,...settings) this.protectionSettings.splice(0, 0, ...settings)
}) })
console.log('获取设备保护详情并初始化',this.formData,this.protectionPlan,this.protectionSettings) console.log('获取设备保护详情并初始化', this.formData, this.protectionPlan, this.protectionSettings, this.childOptions)
}) })
}else{ } else {
this.mode = 'add' this.mode = 'add'
} }
}, },
// 新增设备保护前提、设备保护方案 // 新增设备保护前提、设备保护方案
addRow(type) { addRow(type) {
const item = type === 'protectionSettings' ? { const item = type === 'protectionSettings' ? {
deviceId:[],//设备ID deviceId: [],//设备ID
deviceName: [],
deviceCategory: "",//设备类型 英文 deviceCategory: "",//设备类型 英文
categoryName:'',//设备类型名称 中文 categoryName: '',//设备类型名称 中文
point: "",//点位 英文 point: "",//点位 英文
pointName:"",//点位 中文 pointName: "",//点位 中文
faultValue: "",//故障值 faultValue: "",//故障值
releaseValue: "",//释放值 releaseValue: "",//释放值
faultOperator: "",//故障值比较关系 faultOperator: "",//故障值比较关系
releaseOperator: "",//释放值比较关系 releaseOperator: "",//释放值比较关系
relationNext: "",//与下一个点位的关系 relationNext: "",//与下一个点位的关系
} : { } : {
deviceId:[], deviceId: [],
deviceName: [],
deviceCategory: "",//设备类型 英文 deviceCategory: "",//设备类型 英文
categoryName:'',//设备类型名称 中文 categoryName: '',//设备类型名称 中文
point: "", point: "",
pointName:"", pointName: "",
value: "",//设置值 value: "",//设置值
} }
// this.$set(this[type], this[type].length, item); // this.$set(this[type], this[type].length, item);
this[type].splice(this[type].length,0,item) this[type].splice(this[type].length, 0, item)
console.log('新增设备保护前提、方案',type,this[type]) console.log('新增设备保护前提、方案', type, this[type])
}, },
// 删除设备保护前提、设备保护方案 // 删除设备保护前提、设备保护方案
deleteRow(index, type) { deleteRow(index, type) {
@ -435,25 +464,25 @@ export default {
}).then((response) => { }).then((response) => {
const data = response?.data || []; const data = response?.data || [];
cb( cb(
data.map((item) => { data.map((item) => {
return { name: item, value: item }; return {name: item, value: item};
}) })
); );
}); });
}, },
// 点位选择 // 点位选择
handleSelect(data, index, type) { handleSelect(data, index, type) {
console.log('选择点位',data,index,type) console.log('选择点位', data, index, type)
// this.$set(this[type], index, Object.assign({},this[type][index],{ // this.$set(this[type], index, Object.assign({},this[type][index],{
// point:data.value, // point:data.value,
// pointName:data.value, // pointName:data.value,
// })); // }));
let line = Object.assign({},this[type][index],{ let line = Object.assign({}, this[type][index], {
point:data.value, point: data.value,
pointName:data.value, pointName: data.value,
}) })
this[type].splice(index,1,line); this[type].splice(index, 1, line);
console.log('选择点位配置完成',this[type][index]) console.log('选择点位配置完成', this[type][index])
}, },
// 获取设备类别-不区分站点 // 获取设备类别-不区分站点
@ -461,79 +490,86 @@ export default {
this.loading += 1; this.loading += 1;
return getAllDeviceCategory() return getAllDeviceCategory()
.then((response) => { .then((response) => {
const data = (response?.data || []).filter(item => ['PCS','STACK'].includes(item.code)); const data = (response?.data || []);
// this.childOptions=[] this.$set(this, 'childOptions', [])
this.$set(this,'childOptions',[]) let arr = []
let arr =[]
data.forEach((item) => { data.forEach((item) => {
arr.push({ arr.push({
value: item.code, value: item.code,
label: item.name, label: item.name,
children:[] leaf: false,
children: []
}) })
}) })
this.childOptions.splice(0,0,...arr) this.childOptions.splice(0, 0, ...arr)
console.log('获取设备类型',data,this.childOptions) console.log('获取设备类型', data, this.childOptions)
}) })
.finally(() => { .finally(() => {
this.loading -= 1; this.loading -= 1;
}); });
}, },
//获取设备列表-区分站点 //获取设备列表-区分站点
getDeviceList(deviceCategory,siteId){ getDeviceList(node, resolve) {
this.$nextTick(()=>{ if (!this.formData.siteId) {
getDeviceListBySiteAndCategory({siteId:siteId || this.formData.siteId,deviceCategory}).then((response) => { this.$message({
const data = (response?.data || []).map(item => { type: "warning",
return { message: "请先选择站点",
label: item.deviceName, });
value: item.id, return resolve([])
} }
}) getDeviceListBySiteAndCategory({
const index = this.childOptions.findIndex(item=>item.value === deviceCategory) siteId: this.formData.siteId,
if(index>-1){ deviceCategory: node.value
const length = this.childOptions[index].children.length }).then((response) => {
this.childOptions[index].children.splice(0,length,...data) const data = (response?.data || []).map(item => {
return {
label: item.deviceName,
value: item.id,
leaf: true
} }
}) })
resolve(data)
}).catch(() => {
resolve([])
}) })
}, },
//更新站点下面的设备列表 //更新站点下面的设备列表
updateSiteDeviceList(){ updateSiteDeviceList() {
this.childOptions.forEach(item => { this.childOptions.forEach(item => {
const length = item.children.length const length = item.children.length
item.children.splice(0,length) item.children.splice(0, length)
}) })
this.getDeviceList('PCS')
this.getDeviceList('STACK')
}, },
//选中设备类型、设备 //选中设备类型、设备
handleChange(data,type,index){ handleChange(data, type, index) {
const deviceCategory = data[0],deviceId=data[1] console.log('设置选中设备类型、设备', data, type, index)
console.log('设置选中设备类型、设备',deviceCategory,deviceId,type,index) const {label = '', parent = {}} = this.$refs[type + index][0].getCheckedNodes()[0]
const item = Object.assign({},this[type][index],{ console.log('选中设备的名称', parent.label, label)
deviceId:data, const item = Object.assign({}, this[type][index], {
deviceCategory, deviceId: data,
categoryName : this.childOptions.find(i=>i.value === deviceCategory).label, deviceName: [parent.label, label],
pointName:'', deviceCategory: data[0],
point:'' categoryName: parent.label,
pointName: '',
point: ''
}) })
this.$nextTick(()=>{ this.$nextTick(() => {
// this.$set(this[type], index, item); // this.$set(this[type], index, item);
this[type].splice(index,1,item); this[type].splice(index, 1, item);
}) })
console.log('设置选中设备类型、设备配置完成',this[type][index]) console.log('设置选中设备类型、设备配置完成', this[type][index])
}, },
//获取站点列表 //获取站点列表
getZdList() { getZdList() {
this.loading += 1; this.loading += 1;
getAllSites() getAllSites()
.then((response) => { .then((response) => {
this.siteList = response?.data || []; this.siteList = response?.data || [];
}) })
.finally(() => { .finally(() => {
this.loading -= 1; this.loading -= 1;
}); });
}, },
// 切换站点 // 切换站点
// 重新获取设备列表 // 重新获取设备列表
@ -541,62 +577,65 @@ export default {
changeType() { changeType() {
//获取当前站点下的pcs和bms //获取当前站点下的pcs和bms
this.updateSiteDeviceList() this.updateSiteDeviceList()
if(this.protectionSettings.length>0){ if (this.protectionSettings.length > 0) {
const list =this.protectionSettings const list = this.protectionSettings
list.forEach((item) => { list.forEach((item) => {
item.point = "" item.point = ""
item.pointName = "" item.pointName = ""
item.deviceId=[] item.deviceId = []
item.categoryName='' item.deviceName = []
item.deviceCategory='' item.categoryName = ''
item.deviceCategory = ''
}); });
// this.$set(this,'protectionSettings',list) // this.$set(this,'protectionSettings',list)
this.$nextTick(()=>{ this.$nextTick(() => {
this.protectionSettings.splice(0,this.protectionSettings.length,...list) this.protectionSettings.splice(0, this.protectionSettings.length, ...list)
}) })
} }
if(this.protectionPlan.length>0){ if (this.protectionPlan.length > 0) {
const list =this.protectionPlan const list = this.protectionPlan
list.forEach((item) => { list.forEach((item) => {
item.point = "" item.point = ""
item.pointName = "" item.pointName = ""
item.deviceId=[] item.deviceId = []
item.categoryName='' item.deviceName = []
item.deviceCategory='' item.categoryName = ''
item.deviceCategory = ''
}); });
// this.$set(this,'protectionPlan',list) // this.$set(this,'protectionPlan',list)
this.$nextTick(()=>{ this.$nextTick(() => {
this.protectionPlan.splice(0,this.protectionPlan.length,...list) this.protectionPlan.splice(0, this.protectionPlan.length, ...list)
}) })
} }
}, },
saveDialog() { saveDialog() {
function getToastMsg(name,type,index){ function getToastMsg(name, type, index) {
return { return {
protectionSettings:{ protectionSettings: {
deviceId:`请选择保护前提第${index}行的设备`,//设备ID deviceId: `请选择保护前提第${index}行的设备`,//设备ID
deviceCategory: `请选择保护前提第${index}行的设备类型`,//设备类型 英文 deviceCategory: `请选择保护前提第${index}行的设备类型`,//设备类型 英文
categoryName:`请选择保护前提第${index}行的设备类型`,//设备类型名称 中文 categoryName: `请选择保护前提第${index}行的设备类型`,//设备类型名称 中文
point: `请选择保护前提第${index}行的点位`,//点位 英文 point: `请选择保护前提第${index}行的点位`,//点位 英文
pointName:`请选择保护前提第${index}行的点位`,//点位 中文 pointName: `请选择保护前提第${index}行的点位`,//点位 中文
faultValue: `请输入保护前提第${index}行的故障值`,//故障值 faultValue: `请输入保护前提第${index}行的故障值`,//故障值
releaseValue: `请输入保护前提第${index}行的释放值`,//释放值 releaseValue: `请输入保护前提第${index}行的释放值`,//释放值
faultOperator: `请选择保护前提第${index}行的故障值比较关系`,//故障值比较关系 faultOperator: `请选择保护前提第${index}行的故障值比较关系`,//故障值比较关系
releaseOperator: `请选择保护前提第${index}行的释放值比较关系`,//释放值比较关系 releaseOperator: `请选择保护前提第${index}行的释放值比较关系`,//释放值比较关系
relationNext: `请选择保护前提第${index}行与下一个点位的关系`,//与下一个点位的关系 relationNext: `请选择保护前提第${index}行与下一个点位的关系`,//与下一个点位的关系
}, },
protectionPlan :{ protectionPlan: {
deviceId:`请选择保护方案第${index}行的设备`, deviceId: `请选择保护方案第${index}行的设备`,
deviceCategory: `请选择保护方案第${index}行的设备类型`,//设备类型 英文 deviceCategory: `请选择保护方案第${index}行的设备类型`,//设备类型 英文
categoryName:`请选择保护方案第${index}行的设备类型`,//设备类型名称 中文 categoryName: `请选择保护方案第${index}行的设备类型`,//设备类型名称 中文
point: `请选择保护方案第${index}行的点位`, point: `请选择保护方案第${index}行的点位`,
pointName:`请选择保护方案第${index}行的点位`, pointName: `请选择保护方案第${index}行的点位`,
value: `请输入保护方案第${index}行的故障值`,//设置值 value: `请输入保护方案第${index}行的故障值`,//设置值
} }
}[type][name] }[type][name]
} }
this.$refs.addTempForm.validate((valid) => { this.$refs.addTempForm.validate((valid) => {
if (!valid) return; if (!valid) return;
const { const {
@ -607,67 +646,69 @@ export default {
faultLevel = 1, //告警等级 faultLevel = 1, //告警等级
faultDelaySeconds = "", //故障延时 faultDelaySeconds = "", //故障延时
releaseDelaySeconds = "", //释放延时 releaseDelaySeconds = "", //释放延时
description="",//方案描述 description = "",//方案描述
} = this.formData; } = this.formData;
const {protectionSettings,protectionPlan} = this const {protectionSettings, protectionPlan} = this
let protectionSettingsValidateStatus= true , protectionPlanValidateStatus= true let protectionSettingsValidateStatus = true, protectionPlanValidateStatus = true
for(let i = 0;i<protectionSettings.length;i++){ for (let i = 0; i < protectionSettings.length; i++) {
let valueMap = Object.entries(protectionSettings[i]); let valueMap = Object.entries(protectionSettings[i]);
for(let inner = 0;inner < valueMap.length;inner++){ for (let inner = 0; inner < valueMap.length; inner++) {
const key =valueMap[inner][0],value =valueMap[inner][1] const key = valueMap[inner][0], value = valueMap[inner][1]
if(key === 'relationNext'){ if (key === 'relationNext') {
if(protectionSettings[i+1] && !value){//有下一个点位 if (protectionSettings[i + 1] && !value) {//有下一个点位
this.$message.error(getToastMsg(key,'protectionSettings',i+1)) this.$message.error(getToastMsg(key, 'protectionSettings', i + 1))
protectionSettingsValidateStatus=false protectionSettingsValidateStatus = false
break break
} }
}else{ } else {
if(![0,'0'].includes(value) && !value){ if (![0, '0'].includes(value) && !value) {
this.$message.error(getToastMsg(key,'protectionSettings',i+1)) this.$message.error(getToastMsg(key, 'protectionSettings', i + 1))
protectionSettingsValidateStatus=false protectionSettingsValidateStatus = false
break break
} }
} }
} }
if(!protectionSettingsValidateStatus) break if (!protectionSettingsValidateStatus) break
} }
for(let i = 0;i<protectionPlan.length;i++){ for (let i = 0; i < protectionPlan.length; i++) {
let valueMap = Object.entries(protectionPlan[i]); let valueMap = Object.entries(protectionPlan[i]);
for(let inner = 0;inner < valueMap.length;inner++){ for (let inner = 0; inner < valueMap.length; inner++) {
const key =valueMap[inner][0],value =valueMap[inner][1] const key = valueMap[inner][0], value = valueMap[inner][1]
if(key === 'relationNext'){ if (key === 'relationNext') {
if(protectionPlan[i+1] && !value){//有下一个点位 if (protectionPlan[i + 1] && !value) {//有下一个点位
this.$message.error(getToastMsg(key,'protectionPlan',i+1)) this.$message.error(getToastMsg(key, 'protectionPlan', i + 1))
protectionPlanValidateStatus=false protectionPlanValidateStatus = false
break break
}else{ } else {
// protectionPlan[i][key] = ''//清空选择的关系 // protectionPlan[i][key] = ''//清空选择的关系
} }
}else{ } else {
if(![0,'0'].includes(value) && !value){ if (![0, '0'].includes(value) && !value) {
this.$message.error(getToastMsg(key,'protectionPlan',i+1)) this.$message.error(getToastMsg(key, 'protectionPlan', i + 1))
protectionPlanValidateStatus=false protectionPlanValidateStatus = false
break break
} }
} }
} }
if(!protectionPlanValidateStatus) break if (!protectionPlanValidateStatus) break
} }
if(!protectionSettingsValidateStatus || !protectionPlanValidateStatus) return if (!protectionSettingsValidateStatus || !protectionPlanValidateStatus) return
const settings = protectionSettings.map(item=>{ const settings = protectionSettings.map(item => {
return Object.assign({},item,{ return Object.assign({}, item, {
deviceId:item.deviceId[1], deviceId: item.deviceId[1],
deviceName: item.deviceName[1],
}) })
}) })
const plan = protectionPlan.map(item=>{ const plan = protectionPlan.map(item => {
return Object.assign({},item,{ return Object.assign({}, item, {
deviceId:item.deviceId[1], deviceId: item.deviceId[1],
deviceName: item.deviceName[1],
}) })
}) })
this.loading += 1; this.loading += 1;
const params= { const params = {
siteId, siteId,
faultName, faultName,
isAlert, isAlert,
@ -675,49 +716,49 @@ export default {
faultDelaySeconds, faultDelaySeconds,
releaseDelaySeconds, releaseDelaySeconds,
description, description,
protectionSettings:JSON.stringify(settings), protectionSettings: JSON.stringify(settings),
protectionPlan:JSON.stringify(plan), protectionPlan: JSON.stringify(plan),
} }
if (this.mode === "add") { if (this.mode === "add") {
addProtectPlan(params) addProtectPlan(params)
.then((response) => { .then((response) => {
if (response.code === 200) { if (response.code === 200) {
//新增成功 //新增成功
// 关闭弹窗 更新表格 // 关闭弹窗 更新表格
this.$emit("update"); this.$emit("update");
this.closeDialog(); this.closeDialog();
} }
}) })
.finally(() => { .finally(() => {
this.loading -= 1; this.loading -= 1;
}); });
} else { } else {
params.id = id params.id = id
updateProtectPlan(params) updateProtectPlan(params)
.then((response) => { .then((response) => {
if (response.code === 200) { if (response.code === 200) {
//新增成功 //新增成功
// 关闭弹窗 更新表格 // 关闭弹窗 更新表格
this.$emit("update"); this.$emit("update");
this.closeDialog(); this.closeDialog();
} }
}) })
.finally(() => { .finally(() => {
this.loading -= 1; this.loading -= 1;
}); });
} }
}); });
}, },
closeDialog() { closeDialog() {
this.$emit("clear"); this.$emit("clear");
// 清空所有数据 // 清空所有数据
for(let key in this.formData) { for (let key in this.formData) {
this.formData[key] = key === 'isAlert' ? 0 : key === 'faultLevel' ? 1 : '' this.formData[key] = key === 'isAlert' ? 0 : key === 'faultLevel' ? 1 : ''
} }
this.$refs.addTempForm.resetFields(); this.$refs.addTempForm.resetFields();
this.$set(this,'protectionSettings',[]) this.$set(this, 'protectionSettings', [])
this.$set(this,'protectionPlan',[]) this.$set(this, 'protectionPlan', [])
this.$set(this,'childOptions',[]) this.$set(this, 'childOptions', [])
this.dialogTableVisible = false; this.dialogTableVisible = false;
}, },
}, },
@ -727,35 +768,43 @@ export default {
.items-container { .items-container {
margin-top: 40px; margin-top: 40px;
margin-bottom: 20px; margin-bottom: 20px;
.item-title { .item-title {
line-height: 16px; line-height: 16px;
padding: 10px 0; padding: 10px 0;
color: #000; color: #000;
} }
} }
.time-lists-container { .time-lists-container {
width: 100%; width: 100%;
border: 1px solid #eee; border: 1px solid #eee;
.time-lists { .time-lists {
&:not(:last-child) { &:not(:last-child) {
border-bottom: 1px solid #eee; border-bottom: 1px solid #eee;
} }
display: flex; display: flex;
& > div { & > div {
width: 16%; width: 16%;
box-sizing: border-box; box-sizing: border-box;
text-align: center; text-align: center;
padding: 10px 15px; padding: 10px 15px;
&:not(:last-child) { &:not(:last-child) {
width: 28%; width: 28%;
border-right: 1px solid #eee; border-right: 1px solid #eee;
} }
.el-date-editor.el-input, .el-date-editor.el-input,
.el-date-editor.el-input__inner { .el-date-editor.el-input__inner {
width: 100%; width: 100%;
} }
} }
} }
.time-lists-title { .time-lists-title {
color: #000; color: #000;
font-size: 12px; font-size: 12px;

View File

@ -1,77 +1,136 @@
<template> <template>
<el-dialog :visible.sync="dialogTableVisible" :close-on-press-escape="false" :close-on-click-modal="false" :show-close="false" destroy-on-close lock-scroll append-to-body width="600px" class="ems-dialog" :title="mode === 'add'?'新增设备':`编辑设备` " > <el-dialog :visible.sync="dialogTableVisible" :close-on-press-escape="false" :close-on-click-modal="false"
<el-form v-loading="loading>0" ref="addTempForm" :model="formData" :rules="rules" size="medium" label-width="140px"> :show-close="false" destroy-on-close lock-scroll append-to-body width="800px" class="ems-dialog"
<el-form-item label="站点" prop="siteId"> :title="mode === 'add'?'新增设备':`编辑设备` ">
<el-select v-model="formData.siteId" placeholder="请选择" :style="{width: '100%'}" @change="changeType"> <div v-loading="loading>0">
<el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList" :key="index+'siteOptions'"></el-option> <el-form v-loading="loading>0" ref="addTempForm" inline :model="formData" :rules="rules" size="medium"
</el-select> label-width="120px" class="device-form">
</el-form-item> <el-form-item label="站点" prop="siteId">
<el-form-item label="设备id" prop="deviceId" > <el-select v-model="formData.siteId" placeholder="请选择" :style="{width: '100%'}" @change="changeType">
<el-input v-model="formData.deviceId" placeholder="请输入" maxlength="60" clearable :style="{width: '100%'}"> <el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList"
</el-input> :key="index+'siteOptions'"></el-option>
</el-form-item> </el-select>
<el-form-item label="设备名称" prop="deviceName"> </el-form-item>
<el-input v-model="formData.deviceName" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="设备id" prop="deviceId">
</el-input> <el-input v-model="formData.deviceId" placeholder="请输入" maxlength="60" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="设备描述" prop="description"> </el-form-item>
<el-input v-model="formData.description" type="textarea" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="设备名称" prop="deviceName">
</el-input> <el-input v-model="formData.deviceName" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="工作状态" prop="communicationStatus"> </el-form-item>
<el-select v-model="formData.communicationStatus" placeholder="请选择" :style="{width: '100%'}"> <el-form-item label="设备描述" prop="description">
<el-option :label="value" :value="key" v-for="(value,key) in communicationStatusOptions" :key="key+'communicationStatusOptions'"></el-option> <el-input v-model="formData.description" type="textarea" placeholder="请输入" clearable
</el-select> :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="设备类型" prop="deviceType"> </el-form-item>
<el-select v-model="formData.deviceType" placeholder="请选择" :style="{width: '100%'}"> <el-form-item label="工作状态" prop="communicationStatus">
<el-option :label="value" :value="key" v-for="(value,key) in deviceTypeOptions" :key="key+'deviceTypeOptions'"></el-option> <el-select v-model="formData.communicationStatus" placeholder="请选择" :style="{width: '100%'}">
</el-select> <el-option :label="value" :value="key" v-for="(value,key) in communicationStatusOptions"
</el-form-item> :key="key+'communicationStatusOptions'"></el-option>
<el-form-item label="设备类别" prop="deviceCategory"> </el-select>
<el-select v-model="formData.deviceCategory" placeholder="请选择" :style="{width: '100%'}" @change="changeType"> </el-form-item>
<el-option :label="item.name" :value="item.code" v-for="(item,index) in deviceCategoryList" :key="index+'deviceCategoryList'"></el-option> <el-form-item label="设备类型" prop="deviceType">
</el-select> <el-select v-model="formData.deviceType" placeholder="请选择" :style="{width: '100%'}">
</el-form-item> <el-option :label="value" :value="key" v-for="(value,key) in deviceTypeOptions"
<el-form-item label="上级设备" prop="parentId" v-if="dccDeviceCategoryList.includes(formData.deviceCategory)"> :key="key+'deviceTypeOptions'"></el-option>
<el-select v-model="formData.parentId" :placeholder="parentDeviceList.length === 0 && !formData.siteId ? '请先选择站点' : '请选择'" :style="{width: '100%'}"> </el-select>
<el-option :label="item.deviceName" :value="item.id" v-for="(item,index) in parentDeviceList" :key="index+'parentDeviceList'" ></el-option> </el-form-item>
</el-select> <el-form-item label="设备类别" prop="deviceCategory">
</el-form-item> <el-select v-model="formData.deviceCategory" placeholder="请选择" :style="{width: '100%'}"
<el-form-item label="TCP设备的ip地址" prop="ipAddress" v-if="formData.deviceType === 'TCP'"> @change="changeType">
<el-input v-model="formData.ipAddress" placeholder="请输入" clearable :style="{width: '100%'}"> <el-option :label="item.name" :value="item.code" v-for="(item,index) in deviceCategoryList"
</el-input> :key="index+'deviceCategoryList'"></el-option>
</el-form-item> </el-select>
<el-form-item label="TCP设备的端口号" prop="ipPort" v-if="formData.deviceType === 'TCP'"> </el-form-item>
<el-input v-model="formData.ipPort" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="上级设备" prop="parentId" v-if="dccDeviceCategoryList.includes(formData.deviceCategory)">
</el-input> <el-select v-model="formData.parentId"
</el-form-item> :placeholder="parentDeviceList.length === 0 && !formData.siteId ? '请先选择站点' : '请选择'"
:style="{width: '100%'}">
<el-option :label="item.deviceName" :value="item.id" v-for="(item,index) in parentDeviceList"
:key="index+'parentDeviceList'"></el-option>
</el-select>
</el-form-item>
<el-form-item label="TCP设备的ip地址" prop="ipAddress" v-if="formData.deviceType === 'TCP'">
<el-input v-model="formData.ipAddress" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="TCP设备的端口号" prop="ipPort" v-if="formData.deviceType === 'TCP'">
<el-input v-model="formData.ipPort" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="从站地址" prop="slaveId" v-if="formData.deviceType === 'TCP'">
<el-input v-model="formData.slaveId" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="串口路径" prop="serialPort">
<el-input v-model="formData.serialPort" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="波特率" prop="baudRate">
<el-input v-model="formData.baudRate" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="数据位" prop="dataBits">
<el-input v-model="formData.dataBits" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="停止位" prop="stopBits">
<el-input v-model="formData.stopBits" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="校验位" prop="parity">
<el-input v-model="formData.parity" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="图片" prop="pictureUrl">
<image-upload :limit="1" :drag="false" @input="uploadImage" :value="formData.pictureUrl"/>
</el-form-item>
<el-form-item label="串口路径" prop="serialPort"> </el-form>
<el-input v-model="formData.serialPort" placeholder="请输入" clearable :style="{width: '100%'}"> <!-- pcs配置-->
</el-input> <el-form v-if="isPcs" ref="pcsSettingForm" inline :model="pcsSetting" size="medium"
</el-form-item> label-width="120px" class="device-form" :rules="pcsSettingRules">
<el-form-item label="波特率" prop="baudRate"> <div style="font-size: 14px;padding: 10px 0 20px;font-weight: 600;">PCS配置</div>
<el-input v-model="formData.baudRate" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="开关机地址" prop="pointAddress">
</el-input> <el-input v-model="pcsSetting.pointAddress" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="数据位" prop="dataBits"> </el-form-item>
<el-input v-model="formData.dataBits" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="功率地址" prop="powerAddress">
</el-input> <el-input v-model="pcsSetting.powerAddress" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="停止位" prop="stopBits"> </el-form-item>
<el-input v-model="formData.stopBits" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="开机指令" prop="startCommand">
</el-input> <el-input v-model="pcsSetting.startCommand" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="校验位" prop="parity"> </el-form-item>
<el-input v-model="formData.parity" placeholder="请输入" clearable :style="{width: '100%'}"> <el-form-item label="开机目标功率" prop="startPower">
</el-input> <el-input v-model="pcsSetting.startPower" placeholder="请输入" clearable :style="{width: '100%'}">
</el-form-item> </el-input>
<el-form-item label="图片" prop="pictureUrl"> </el-form-item>
<image-upload :limit="1" :drag="false" @input="uploadImage" :value="formData.pictureUrl"/> <el-form-item label="关机指令" prop="stopCommand">
</el-form-item> <el-input v-model="pcsSetting.stopCommand" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form> </el-form-item>
<el-form-item label="关机目标功率" prop="stopPower">
<el-input v-model="pcsSetting.stopPower" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<el-form-item label="电池簇数" prop="clusterNum">
<el-input v-model="pcsSetting.clusterNum" placeholder="请输入" clearable :style="{width: '100%'}">
</el-input>
</el-form-item>
<br>
<template v-for="index in parseInt(pcsSetting.clusterNum) || 0">
<el-form-item :label="'电池簇'+(index)+'地址'"
prop="clusterPointAddress">
<el-input v-model="pcsSetting.clusterPointAddress[index-1]" placeholder="请输入" clearable
:style="{width: '100%'}">
</el-input>
</el-form-item>
</template>
</el-form>
</div>
<div slot="footer"> <div slot="footer">
<el-button @click="closeDialog">取消</el-button> <el-button @click="closeDialog">取消</el-button>
<el-button type="primary" @click="saveDialog">确定</el-button> <el-button type="primary" @click="saveDialog">确定</el-button>
@ -82,257 +141,404 @@
import {mapState} from "vuex"; import {mapState} from "vuex";
import {getAllSites} from '@/api/ems/zddt' import {getAllSites} from '@/api/ems/zddt'
import {validText} from '@/utils/validate' import {validText} from '@/utils/validate'
import {getDeviceDetailInfo,updateDevice,addDevice,getParentDeviceId} from "@/api/ems/site"; import {addDevice, getDeviceDetailInfo, getParentDeviceId, updateDevice} from "@/api/ems/site";
import {getAllDeviceCategory} from '@/api/ems/search' import {getAllDeviceCategory} from '@/api/ems/search'
export default { export default {
props:{ props: {
mode:{ mode: {
type:String, type: String,
default:"add" default: "add"
}, },
id:{ id: {
type:String|Number, type: String | Number,
required:false required: false
} }
}, },
data() { data() {
const validateText=(rule, value, callback) =>{ const validateText = (rule, value, callback) => {
if (value !== '' && !validText(value)) { if (value !== '' && !validText(value)) {
callback(new Error('只能输入中文、英文、数字和特殊字符!')); callback(new Error('只能输入中文、英文、数字和特殊字符!'));
} else { } else {
callback(); callback();
} }
} }
const validateDeviceId=(rule, value, callback) =>{ const validateDeviceId = (rule, value, callback) => {
if (value !== '' && !/^[a-zA-Z0-9]+$/.test(value)) { if (value !== '' && !/^[a-zA-Z0-9]+$/.test(value)) {
callback(new Error('只能输入英文和数字!')); callback(new Error('只能输入英文和数字!'));
} else { } else {
callback(); callback();
} }
} }
const validateNumber = (rule, value, callback) => {
if (value !== '' && !/^[0-9]+$/.test(value)) {
callback(new Error('只能输入数字!'));
} else {
callback();
}
}
return { return {
loading:0, loading: 0,
dccDeviceCategoryList:['CLUSTER','BATTERY'],//需要展示上级设备的设备类型 dccDeviceCategoryList: ['CLUSTER', 'BATTERY'],//需要展示上级设备的设备类型
dialogTableVisible:false, dialogTableVisible: false,
parentDeviceList:[],//上级设备列表 从接口获取数据 parentDeviceList: [],//上级设备列表 从接口获取数据
siteList:[],//站点列表 从接口获取数据 siteList: [],//站点列表 从接口获取数据
deviceCategoryList:[],//设备类别列表 从接口获取数据 deviceCategoryList: [],//设备类别列表 从接口获取数据
formData: { formData: {
id:'',//设备唯一标识 id: '',//设备唯一标识
siteId:'',//站点ID siteId: '',//站点ID
deviceId:'',//设备id deviceId: '',//设备id
deviceName:'',//设备名称 deviceName: '',//设备名称
description:'',//设备描述 description: '',//设备描述
communicationStatus:'',//工作状态 communicationStatus: '',//工作状态
deviceType:'',//设备类型 deviceType: '',//设备类型
deviceCategory:'',//设备类别 deviceCategory: '',//设备类别
parentId:'',//上级设备id parentId: '',//上级设备id
ipAddress:'',//TCP设备的ip地址 ipAddress: '',//TCP设备的ip地址
ipPort:"",//TCP端口号 ipPort: "",//TCP端口号
serialPort:'',//串口路径 serialPort: '',//串口路径
baudRate:'',//波特率 baudRate: '',//波特率
dataBits:'',//数据位 dataBits: '',//数据位
stopBits:'',//停止位 stopBits: '',//停止位
parity:'',//校验位 parity: '',//校验位
pictureUrl:'',//设备图片 pictureUrl: '',//设备图片
slaveId: '',//从站地址
},
pcsSetting: {
deviceSettingId: '',
powerAddress: '',//功率地址
pointAddress: "",//开关机地址
startCommand: "",//开机指令
stopCommand: "",//关机指令
startPower: '',//开机目标功率
stopPower: '',//关机目标功率
clusterNum: '',//电池簇数
clusterPointAddress: []//电池簇地址
}, },
rules: { rules: {
siteId:[ siteId: [
{ required: true, message: '请选择站点', trigger: ['blur','change']} {required: true, message: '请选择站点', trigger: ['blur', 'change']}
], ],
deviceId:[ deviceId: [
{ required: true, message: '请输入设备id', trigger: 'blur'}, {required: true, message: '请输入设备id', trigger: 'blur'},
{ validator: validateDeviceId, trigger: 'blur' } {validator: validateDeviceId, trigger: 'blur'}
], ],
deviceName:[ deviceName: [
{ required: true, message: '请输入设备名称', trigger: 'blur'}, {required: true, message: '请输入设备名称', trigger: 'blur'},
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
description:[ description: [
{ required: true, message: '请输入设备描述', trigger: 'blur'}, {required: true, message: '请输入设备描述', trigger: 'blur'},
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
communicationStatus:[ communicationStatus: [
{ required: true, message: '请选择工作状态', trigger: ['blur','change']} {required: true, message: '请选择工作状态', trigger: ['blur', 'change']}
], ],
deviceType:[ deviceType: [
{ required: true, message: '请选择设备类型', trigger:['blur','change']} {required: true, message: '请选择设备类型', trigger: ['blur', 'change']}
], ],
deviceCategory:[ deviceCategory: [
{ required: true, message: '请选择设备类别', trigger: ['blur','change']} {required: true, message: '请选择设备类别', trigger: ['blur', 'change']}
], ],
ipAddress:[ ipAddress: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
ipPort:[ ipPort: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
serialPort:[ slaveId: [
{ validator: validateText, trigger: 'blur' } {validator: validateNumber, trigger: 'blur'}
], ],
baudRate:[ serialPort: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
dataBits:[ baudRate: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
stopBits:[ dataBits: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
], ],
parity:[ stopBits: [
{ validator: validateText, trigger: 'blur' } {validator: validateText, trigger: 'blur'}
],
parity: [
{validator: validateText, trigger: 'blur'}
], ],
// pictureUrl:[ // pictureUrl:[
// { required: true, message: '请上传图片', trigger: ['blur', 'change']} // { required: true, message: '请上传图片', trigger: ['blur', 'change']}
// ], // ],
}, },
pcsSettingRules: {
pointAddress: [
{required: true, message: '请输入开关机地址', trigger: 'blur'},
{validator: validateText, trigger: 'blur'}
],
powerAddress: [
{validator: validateText, trigger: 'blur'}
],
startCommand: [
{required: true, message: '请输入开机指令', trigger: 'blur'},
{validator: validateText, trigger: 'blur'}
],
startPower: [
{validator: validateText, trigger: 'blur'}
],
stopCommand: [
{required: true, message: '请输入关机指令', trigger: 'blur'},
{validator: validateText, trigger: 'blur'}
],
stopPower: [
{validator: validateText, trigger: 'blur'}
],
clusterNum: [
{required: true, message: '请输入电池簇数', trigger: 'blur'},
{validator: validateNumber, trigger: 'blur'}
],
clusterPointAddress: [
{required: true, message: '请输入电池簇地址', trigger: 'blur'},
{validator: validateText, trigger: 'blur'}
]
}
} }
}, },
computed: { computed: {
...mapState({ ...mapState({
communicationStatusOptions: state => state?.ems?.communicationStatusOptions || {}, communicationStatusOptions: state => state?.ems?.communicationStatusOptions || {},
deviceTypeOptions:state=>state?.ems?.deviceTypeOptions || {} deviceTypeOptions: state => state?.ems?.deviceTypeOptions || {}
}) }),
isPcs() {
return this.formData.deviceCategory === 'PCS'
}
}, },
watch:{ watch: {
dialogTableVisible:{ dialogTableVisible: {
handler(newVal){ handler(newVal) {
//打开弹窗 //打开弹窗
if(newVal){ if (newVal) {
this.getZdList() this.getZdList()
this.getDeviceCategoryList() this.getDeviceCategoryList()
} }
}, },
immediate: true, immediate: true,
}, },
id:{ id: {
handler(newVal){ handler(newVal) {
if((newVal || newVal===0) && this.mode !== 'add'){ if ((newVal || newVal === 0) && this.mode !== 'add') {
this.loading+=1 this.loading += 1
getDeviceDetailInfo(newVal).then(response => { getDeviceDetailInfo(newVal).then(response => {
this.formData = JSON.parse(JSON.stringify(response?.data || {})); const {pcsSetting, ...data} = JSON.parse(JSON.stringify(response?.data || {}))
if(this.dccDeviceCategoryList.includes(this.formData.deviceCategory)){ this.formData = data;
if (pcsSetting && JSON.stringify(pcsSetting) !== '{}') {
this.pcsSetting = JSON.parse(JSON.stringify({
...pcsSetting,
clusterPointAddress: JSON.parse(pcsSetting.clusterPointAddress || [])
}));
}
if (this.dccDeviceCategoryList.includes(this.formData.deviceCategory)) {
this.getParentDeviceList(true) this.getParentDeviceList(true)
} }
}).finally(() => {this.loading-=1}) }).finally(() => {
this.loading -= 1
})
} }
}, },
immediate: true, immediate: true,
} }
}, },
methods: { methods: {
changeType(){ changeType() {
if(this.dccDeviceCategoryList.includes(this.formData.deviceCategory)){ if (this.dccDeviceCategoryList.includes(this.formData.deviceCategory)) {
this.getParentDeviceList() this.getParentDeviceList()
} }
}, },
uploadImage(data){ uploadImage(data) {
this.formData.pictureUrl = data this.formData.pictureUrl = data
}, },
//获取站点列表 //获取站点列表
getZdList(){ getZdList() {
this.loading+=1 this.loading += 1
getAllSites().then(response => { getAllSites().then(response => {
this.siteList = response?.data || [] this.siteList = response?.data || []
}).finally(() => {this.loading-=1}) }).finally(() => {
this.loading -= 1
})
}, },
// 获取设备类别 // 获取设备类别
getDeviceCategoryList(){ getDeviceCategoryList() {
this.loading+=1 this.loading += 1
getAllDeviceCategory().then(response => { getAllDeviceCategory().then(response => {
this.deviceCategoryList = response?.data || [] this.deviceCategoryList = response?.data || []
}).finally(() => {this.loading-=1}) }).finally(() => {
this.loading -= 1
})
}, },
//获取上级id列表 //获取上级id列表
getParentDeviceList(init=false){ getParentDeviceList(init = false) {
if(!this.formData.siteId){ if (!this.formData.siteId) {
return console.log('请先选择站点') return console.log('请先选择站点')
} }
!init && (this.formData.parentId='') !init && (this.formData.parentId = '')
this.loading= this.loading+1 this.loading = this.loading + 1
getParentDeviceId({siteId:this.formData.siteId,deviceCategory:this.formData.deviceCategory}).then(response => { getParentDeviceId({siteId: this.formData.siteId, deviceCategory: this.formData.deviceCategory}).then(response => {
this.parentDeviceList = JSON.parse(JSON.stringify(response?.data || [])); this.parentDeviceList = JSON.parse(JSON.stringify(response?.data || []));
}).finally(() => { }).finally(() => {
this.loading=this.loading -1 this.loading = this.loading - 1
}) })
}, },
saveData() {
this.loading += 1
const {
id = '',
siteId = '',//站点ID
deviceId = '',//设备id
deviceName = '',//设备名称
description = '',//设备描述
communicationStatus = '',//工作状态
deviceType = '',//设备类型
deviceCategory = '',//设备类别
parentId = '',//上级设备id
ipAddress = '',//TCP设备的ip地址
ipPort = "",//TCP端口号
serialPort = '',//串口路径
baudRate = '',//波特率
dataBits = '',//数据位
stopBits = '',//停止位
parity = '',//校验位
pictureUrl = '',//设备图片
slaveId = '',//从站地址
} = this.formData;
const {
deviceSettingId,
powerAddress,
pointAddress,
startCommand,
stopCommand,
startPower,
stopPower,
clusterNum,
clusterPointAddress
} = this.pcsSetting
let params = {
siteId,
deviceId,
deviceName,
description,
communicationStatus,
deviceType,
deviceCategory,
parentId,
ipAddress,
ipPort,
serialPort,
baudRate,
dataBits,
stopBits,
parity,
pictureUrl,
slaveId,
}
if (this.isPcs) {
params.pcsSetting = {
powerAddress,
pointAddress,
startCommand,
stopCommand,
startPower,
stopPower,
clusterNum,
clusterPointAddress: JSON.stringify(!clusterNum ? [] : (clusterPointAddress || []).slice(0, clusterNum))
}
}
if (this.mode === 'add') {
addDevice(params).then(response => {
if (response.code === 200) {
//新增成功
// 关闭弹窗 更新表格
this.$emit('update')
this.closeDialog()
}
}).finally(() => {
this.loading -= 1
})
} else {
params.id = id
params.pcsSetting && (params.pcsSetting.deviceSettingId = deviceSettingId)
updateDevice(params).then(response => {
if (response.code === 200) {
//新增成功
// 关闭弹窗 更新表格
this.$emit('update')
this.closeDialog()
}
}).finally(() => {
this.loading -= 1
})
}
},
saveDialog() { saveDialog() {
this.$refs.addTempForm.validate(valid => { this.$refs.addTempForm.validate(valid => {
if (!valid) return if (!valid) return
this.loading+=1 if (this.isPcs) {
const { this.$refs.pcsSettingForm.validate(pcsValidate => {
id='', if (!pcsValidate) return
siteId='',//站点ID this.saveData()
deviceId='',//设备id
deviceName='',//设备名称
description='',//设备描述
communicationStatus='',//工作状态
deviceType='',//设备类型
deviceCategory='',//设备类别
parentId='',//上级设备id
ipAddress='',//TCP设备的ip地址
ipPort="",//TCP端口号
serialPort='',//串口路径
baudRate='',//波特率
dataBits='',//数据位
stopBits='',//停止位
parity='',//校验位
pictureUrl='',//设备图片
}= this.formData;
if(this.mode === 'add'){
addDevice({siteId,deviceId,deviceName,description,communicationStatus,deviceType,deviceCategory,parentId,ipAddress,ipPort,serialPort,baudRate,dataBits,stopBits,parity,pictureUrl}).then(response => {
if(response.code === 200){
//新增成功
// 关闭弹窗 更新表格
this.$emit('update')
this.closeDialog()
}
}).finally(() => {
this.loading-=1
})
}else{
updateDevice({id,siteId,deviceId,deviceName,description,communicationStatus,deviceType,deviceCategory,parentId,ipAddress,ipPort,serialPort,baudRate,dataBits,stopBits,parity,pictureUrl}).then(response => {
if(response.code === 200){
//新增成功
// 关闭弹窗 更新表格
this.$emit('update')
this.closeDialog()
}
}).finally(() => {
this.loading-=1
}) })
} else {
this.saveData()
} }
}) })
}, },
closeDialog(){ closeDialog() {
this.$emit('clear') this.$emit('clear')
// 清空所有数据 // 清空所有数据
this.formData= { this.formData = {
id:'',//设备唯一标识 id: '',//设备唯一标识
siteId:'',//站点ID siteId: '',//站点ID
deviceId:'',//设备id deviceId: '',//设备id
deviceName:'',//设备名称 deviceName: '',//设备名称
description:'',//设备描述 description: '',//设备描述
communicationStatus:'',//工作状态 communicationStatus: '',//工作状态
deviceType:'',//设备类型 deviceType: '',//设备类型
deviceCategory:'',//设备类别 deviceCategory: '',//设备类别
parentId:'',//上级设备id parentId: '',//上级设备id
ipAddress:'',//TCP设备的ip地址 ipAddress: '',//TCP设备的ip地址
ipPort:"",//TCP端口号 ipPort: "",//TCP端口号
serialPort:'',//串口路径 serialPort: '',//串口路径
baudRate:'',//波特率 baudRate: '',//波特率
dataBits:'',//数据位 dataBits: '',//数据位
stopBits:'',//停止位 stopBits: '',//停止位
parity:'',//校验位 parity: '',//校验位
pictureUrl:'',//设备图片 pictureUrl: '',//设备图片
slaveId: '',//从站地址
}
this.pcsSetting = {
deviceSettingId: '',
powerAddress: '',//功率地址
pointAddress: "",//开关机地址
startCommand: "",//开机指令
stopCommand: "",//关机指令
startPower: '',//开机目标功率
stopPower: '',//关机目标功率
clusterNum: '',//电池簇数
clusterPointAddress: []//电池簇地址
} }
this.$refs.addTempForm.resetFields() this.$refs.addTempForm.resetFields()
this.dialogTableVisible=false this.$refs?.pcsSettingForm?.resetFields()
this.dialogTableVisible = false
} }
} }
} }
</script> </script>
<style scoped> <style scoped lang="scss">
.device-form {
::v-deep .el-form-item--medium .el-form-item__content {
width: 260px;
}
.el-form-item {
width: 50%;
margin-right: 0;
}
}
</style> </style>

View File

@ -0,0 +1,98 @@
<template>
<el-button :size="size" :type="type" :round="round" @click="switchStatus"
>
{{ label }}
</el-button>
</template>
<style scoped lang="scss">
</style>
<script>
import {updateDeviceStatus} from "@/api/ems/site";
export default {
props: {
size: {
type: String,
default: 'mini',
required: false
},
round: {
type: Boolean,
default: false,
required: false
},
type: {
type: String,
default: 'primary',
required: false
},
data: {
type: Object,
default: () => {
return {
workStatus: null,
deviceId: null,
deviceName: null,
}
},
required: true
}
},
data() {
return {
runningStatusMap: ['0', '3', '4', '6']
}
},
computed: {
label() {
return this.runningStatusMap.includes(this.data.workStatus) ? '关机' : '开机'
}
},
methods: {
switchStatus() {
const {workStatus, deviceId, deviceName, siteId} = this.data
this.$confirm(`确认要${this.label}设备${deviceName || ''}吗?`, {
confirmButtonText: "确定",
cancelButtonText: "取消",
showClose: false,
closeOnClickModal: false,
type: "warning",
beforeClose: (action, instance, done) => {
if (action === "confirm") {
instance.confirmButtonLoading = true;
//做开关机操作,更新成功后刷新表格
updateDeviceStatus({
siteId,
workStatus: this.runningStatusMap.includes(workStatus) ? "1" : '0',//1 关机 0开机
deviceId
})
.then((response) => {
response.code === 200 && done();
})
.finally(() => {
instance.confirmButtonLoading = false;
});
} else {
done();
}
},
})
.then(() => {
//只有在废弃成功的情况下会走到这里
this.$message({
type: "success",
message: `${deviceName}${this.label}成功!`,
});
this.$emit('updateSuccess')
//调用接口 更新表格数据
})
.catch(() => {
//取消关机
});
}
}
}
</script>

View File

@ -2,171 +2,177 @@
<template> <template>
<div> <div>
<el-dialog <el-dialog
v-loading="loading" v-loading="loading"
:close-on-click-modal="false" :close-on-click-modal="false"
:visible.sync="show" :visible.sync="show"
append-to-body append-to-body
class="ems-dialog" class="ems-dialog"
destroy-on-close destroy-on-close
lock-scroll lock-scroll
show-close show-close
title="点位清单" :title="dataType === 'point' ? '点位清单' : '报警点位'"
width="950px" width="950px"
> >
<el-form :inline="true" label-width="100px"> <el-form :inline="true" label-width="100px">
<el-form-item label="点位名称"> <el-form-item label="点位名称">
<el-input <el-input
v-model="form.dataPointName" v-model="form.dataPointName"
clearable clearable
placeholder="请输入点位名称" placeholder="请输入点位名称"
style="width: 150px" style="width: 150px"
></el-input> ></el-input>
</el-form-item> </el-form-item>
<el-form-item label="点位"> <el-form-item label="点位">
<el-input <el-input
v-model="form.dataPoint" v-model="form.dataPoint"
clearable
placeholder="请输入点位"
style="width: 150px"
></el-input>
</el-form-item>
<el-form-item label="modbus ip">
<el-input
v-model="form.ipAddress"
clearable clearable
placeholder="请输入ip" placeholder="请输入点位"
style="width: 150px" style="width: 150px"
></el-input> ></el-input>
</el-form-item> </el-form-item>
<!-- <el-form-item label="modbus ip">-->
<!-- <el-input-->
<!-- v-model="form.ipAddress"-->
<!-- clearable-->
<!-- placeholder="请输入ip"-->
<!-- style="width: 150px"-->
<!-- ></el-input>-->
<!-- </el-form-item>-->
<br>
<el-form-item label="最小值"> <el-form-item label="最小值">
<el-input <el-input
v-model="form.lower" v-model="form.lower"
clearable clearable
placeholder="请输入最小值" placeholder="请输入最小值"
style="width: 150px" style="width: 150px"
></el-input> ></el-input>
</el-form-item> </el-form-item>
<el-form-item label="最大值"> <el-form-item label="最大值">
<el-input <el-input
v-model="form.upper" v-model="form.upper"
clearable
placeholder="请输入最大值"
style="width: 150px"
></el-input>
</el-form-item>
<el-form-item label="modbus 端口">
<el-input
v-model="form.ipPort"
clearable clearable
placeholder="请输入端口" placeholder="请输入最大值"
style="width: 150px" style="width: 150px"
></el-input> ></el-input>
</el-form-item> </el-form-item>
<!-- <el-form-item label="modbus 端口">-->
<!-- <el-input-->
<!-- v-model="form.ipPort"-->
<!-- clearable-->
<!-- placeholder="请输入端口"-->
<!-- style="width: 150px"-->
<!-- ></el-input>-->
<!-- </el-form-item>-->
<el-form-item style="margin-left: 20px"> <el-form-item style="margin-left: 20px">
<el-button type="primary" @click="search">搜索</el-button> <el-button type="primary" @click="search">搜索</el-button>
</el-form-item> </el-form-item>
</el-form> </el-form>
<el-table <el-table
:data="tableData" :data="tableData"
class="common-table" class="common-table"
max-height="400px" max-height="400px"
stripe stripe
style="width: 100%" style="width: 100%"
:default-sort="defaultSort" :default-sort="defaultSort"
@sort-change="handleSortChange" @sort-change="handleSortChange"
> >
<el-table-column label="数据点位" prop="dataPoint"> </el-table-column> <el-table-column label="数据点位" prop="dataPoint"></el-table-column>
<el-table-column label="数据点位名称" prop="pointName"></el-table-column>
<el-table-column <el-table-column
label="modbus地址" label="数据点位名称"
prop="dataPointName"
></el-table-column>
<!-- <el-table-column label="modbus地址">-->
<!-- <template slot-scope="scope">-->
<!-- <span>{{-->
<!-- `${scope.row.ipAddress || ""} ${scope.row.ipPort || ""}`-->
<!-- }}</span>-->
<!-- </template>-->
<!-- </el-table-column>
<el-table-column label="寄存器地址" prop="寄存器地址"></el-table-column>-->
<el-table-column
label="最新值"
prop="pointValue"
align="right"
sortable="custom"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{`${scope.row.ipAddress || ''} ${scope.row.ipPort || ''}`}}</span> <span class="pointer" @click="showChart(scope.row)">{{
</template> scope.row.pointValue
</el-table-column> }}</span>
<el-table-column
label="最新值"
prop="pointValue"
align="right"
sortable="custom"
>
<template slot-scope="scope">
<span
class="pointer"
@click="
showChart(
scope.row
)
"
>{{ scope.row.pointValue }}</span
>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="单位" prop="dataUnit"></el-table-column> <el-table-column label="单位" prop="dataUnit"></el-table-column>
<el-table-column label="更新时间" min-width="160px" prop="updateTime" sortable="custom"> <el-table-column
label="更新时间"
min-width="160px"
prop="updateTime"
sortable="custom"
>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-pagination <el-pagination
v-show="tableData.length > 0" v-show="tableData.length > 0"
:current-page="pageNum" :current-page="pageNum"
:page-size="pageSize" :page-size="pageSize"
:page-sizes="[10, 20, 30, 40]" :page-sizes="[10, 20, 30, 40]"
:pager-count="5" :pager-count="5"
:total="totalSize" :total="totalSize"
background background
layout="total, sizes, prev, pager, next, jumper" layout="total, sizes, prev, pager, next, jumper"
small small
style="margin-top: 15px; text-align: center" style="margin-top: 15px; text-align: center"
@size-change="handleSizeChange" @size-change="handleSizeChange"
@current-change="handleCurrentChange" @current-change="handleCurrentChange"
> >
</el-pagination> </el-pagination>
</el-dialog> </el-dialog>
<point-chart ref="pointChart" :site-id="siteId" /> <point-chart ref="pointChart" :site-id="siteId"/>
</div> </div>
</template> </template>
<script> <script>
import { getDevicePointList } from "@/api/ems/site"; import {getDevicePointList} from "@/api/ems/site";
import pointChart from "@/views/ems/dzjk/sbjk/PointChart.vue"; import pointChart from "@/views/ems/dzjk/sbjk/PointChart.vue";
export default { export default {
components: { pointChart }, components: {pointChart},
watch: { watch: {
show(val) { show(val) {
if (!val) { if (!val) {
this.tableData = []; this.tableData = [];
this.deviceId = ""; this.deviceId = "";
this.parentId=''; this.parentId = "";
this.siteId = ""; this.siteId = "";
this.pageSize = 10; this.pageSize = 10;
this.pageNum = 1; this.pageNum = 1;
this.totalSize = 0; this.totalSize = 0;
this.dataType = '';
this.form = { this.form = {
sortMethod: "desc", //升序不传或者asc、降序desc sortMethod: "desc", //升序不传或者asc、降序desc
sortData:this.defaultSort.prop, sortData: this.defaultSort.prop,
dataPointName: "", //点位名称 dataPointName: "", //点位名称
dataPoint: "", //点位名称 dataPoint: "", //点位名称
lower: "", // lower: "", //
upper: "", // upper: "", //
ipAddress:'', // ipAddress: "",
ipPort:'' // ipPort: "",
}; };
this.loading = false; this.loading = false;
} }
}, },
}, },
computed:{ computed: {
isDtdc(){ isDtdc() {
return this.deviceCategory === 'BATTERY' return this.deviceCategory === "BATTERY";
} },
}, },
data() { data() {
return { return {
// 默认排序 // 默认排序
defaultSort: { prop: "updateTime", order: "descending" }, defaultSort: {prop: "updateTime", order: "descending"},
show: false, show: false,
loading: false, loading: false,
dataType: '',//展示的数据类型 point点位/alarmPoint报警点位
form: { form: {
sortData: "updateTime", //最新值升序不传或者asc、降序desc sortData: "updateTime", //最新值升序不传或者asc、降序desc
sortMethod: "desc", //升序不传或者asc、降序desc sortMethod: "desc", //升序不传或者asc、降序desc
@ -174,12 +180,12 @@ export default {
dataPoint: "", //点位名称 dataPoint: "", //点位名称
lower: "", // lower: "", //
upper: "", // upper: "", //
ipAddress:'', // ipAddress: "",
ipPort:'' // ipPort: "",
}, },
deviceCategory: "", deviceCategory: "",
deviceId: "", deviceId: "",
parentId:'', parentId: "",
siteId: "", siteId: "",
tableData: [], tableData: [],
pageSize: 10, //分页栏当前每个数据总数 pageSize: 10, //分页栏当前每个数据总数
@ -189,32 +195,56 @@ export default {
}, },
methods: { methods: {
showChart({pointName}) { showChart({pointName}) {
if(pointName){ if (pointName) {
const {deviceCategory,deviceId}=this const {deviceCategory, deviceId} = this;
if(this.isDtdc) this.$refs.pointChart.showChart({ pointName, deviceCategory, deviceId:this.parentId,child:[deviceId] }); if (this.isDtdc)
else this.$refs.pointChart.showChart({ pointName, deviceCategory, deviceId }); this.$refs.pointChart.showChart({
pointName,
deviceCategory,
deviceId: this.parentId,
child: [deviceId],
});
else
this.$refs.pointChart.showChart({
pointName,
deviceCategory,
deviceId,
});
} }
}, },
handleSortChange(column) { handleSortChange(column) {
this.form.sortData = column.prop this.form.sortData = column.prop;
this.form.sortMethod = column.order === "descending" ? "desc" : "asc"; this.form.sortMethod = column.order === "descending" ? "desc" : "asc";
console.log("切换排序方式", column, this.form); console.log("切换排序方式", column, this.form);
this.getData(); this.getData()
}, },
search() { search() {
this.pageNum = 1; this.pageNum = 1;
this.getData(); this.getData()
}, },
showTable({ deviceCategory, siteId, deviceId ,parentId=''}) { // 分页
handleSizeChange(val) {
this.pageSize = val;
this.$nextTick(() => {
this.getData()
});
},
handleCurrentChange(val) {
this.pageNum = val;
this.$nextTick(() => {
this.getData()
});
},
showTable({deviceCategory, siteId, deviceId, parentId = ""}, dataType) {
this.dataType = dataType;
this.deviceCategory = deviceCategory; this.deviceCategory = deviceCategory;
this.siteId = siteId; this.siteId = siteId;
this.deviceId = deviceId; this.deviceId = deviceId;
this.parentId=deviceCategory === 'BATTERY' ? parentId : ''//只有单体电池需要这个值 this.parentId = deviceCategory === "BATTERY" ? parentId : ""; //只有单体电池需要这个值
this.show = true; this.show = true;
this.getData(); this.getData()
}, },
getData() { getData() {
this.loading = true;
const { const {
siteId, siteId,
deviceId, deviceId,
@ -229,11 +259,11 @@ export default {
dataPoint, dataPoint,
lower, lower,
upper, upper,
ipAddress, // ipAddress,
ipPort // ipPort,
}, },
} = this; } = this;
getDevicePointList({ const params = {
siteId, siteId,
deviceId, deviceId,
deviceCategory, deviceCategory,
@ -246,29 +276,19 @@ export default {
dataPoint, dataPoint,
lower, lower,
upper, upper,
ipAddress, // ipAddress: '',
ipPort // ipPort: '',
}) }
.then((response) => { params.isAlarm = this.dataType === 'point' ? 0 : 1
this.tableData = response?.rows || []; this.loading = true;
this.totalSize = response?.total || 0; getDevicePointList(params)
}) .then((response) => {
.finally(() => { this.tableData = response?.rows || [];
this.loading = false; this.totalSize = response?.total || 0;
}); })
}, .finally(() => {
// 分页 this.loading = false;
handleSizeChange(val) { });
this.pageSize = val;
this.$nextTick(() => {
this.getData();
});
},
handleCurrentChange(val) {
this.pageNum = val;
this.$nextTick(() => {
this.getData();
});
}, },
}, },
}; };

View File

@ -1,12 +1,15 @@
<template> <template>
<el-dialog :visible.sync="show" class="ems-dialog" title="点位上传" width="400px" append-to-body :close-on-click-modal="false" :close-on-press-escape="false"> <el-dialog :visible.sync="show" class="ems-dialog" title="点位上传" width="400px" append-to-body
<!-- <file-upload :fileType='["xls", "xlsx"]' value="String" :limit="1" :drag="false" @input="updateFile"/>--> :close-on-click-modal="false" :close-on-press-escape="false">
<el-upload ref="upload" :limit="1" accept=".xlsx, .xls" :headers="upload.headers" :action="upload.url" :disabled="upload.isUploading" :on-progress="handleFileUploadProgress" :on-success="handleFileSuccess" :auto-upload="false" drag> <!-- <file-upload :fileType='["xls", "xlsx"]' value="String" :limit="1" :drag="false" @input="updateFile"/>-->
<el-upload ref="upload" :limit="1" accept=".xlsx, .xls" :headers="upload.headers" :action="upload.url"
:data="upload.data" :disabled="upload.isUploading" :on-progress="handleFileUploadProgress"
:on-success="handleFileSuccess" :auto-upload="false" drag>
<i class="el-icon-upload"></i> <i class="el-icon-upload"></i>
<div class="el-upload__text">将文件拖到此处<em>点击上传</em></div> <div class="el-upload__text">将文件拖到此处<em>点击上传</em></div>
<div class="el-upload__tip text-center" slot="tip"> <div class="el-upload__tip text-center" slot="tip">
<span>仅允许导入xlsxlsx格式文件</span> <span>仅允许导入xlsxlsx格式文件</span>
<!-- <el-link type="primary" :underline="false" style="font-size: 12px; vertical-align: baseline" @click="importTemplate">下载模板</el-link>--> <!-- <el-link type="primary" :underline="false" style="font-size: 12px; vertical-align: baseline" @click="importTemplate">下载模板</el-link>-->
</div> </div>
</el-upload> </el-upload>
<div slot="footer"> <div slot="footer">
@ -20,30 +23,34 @@
</style> </style>
<script> <script>
import { getToken } from "@/utils/auth" import {getToken} from "@/utils/auth"
export default { export default {
data(){ data() {
return { return {
show:false, show: false,
// 用户导入参数 // 用户导入参数
upload: { upload: {
data: {},
// 是否禁用上传 // 是否禁用上传
isUploading: false, isUploading: false,
// 设置上传的请求头部 // 设置上传的请求头部
headers: { Authorization: "Bearer " + getToken() }, headers: {Authorization: "Bearer " + getToken()},
// 上传的地址 // 上传的地址
url: process.env.VUE_APP_BASE_API + "/ems/pointMatch/importData" url: process.env.VUE_APP_BASE_API + "/ems/pointMatch/importDataByDevice"
}, },
} }
}, },
methods:{ methods: {
showDialog(){ showDialog({siteId, deviceCategory, deviceId}) {
this.upload.data = {siteId, deviceCategory, deviceId}
this.show = true; this.show = true;
}, },
//关闭弹窗 重置数据 //关闭弹窗 重置数据
handleClosed(){ handleClosed() {
this.$refs.upload.clearFiles()//写在before-closed无效 this.$refs.upload.clearFiles()//写在before-closed无效
this.show=false this.upload.data = {}
this.show = false
}, },
// updateFile(data){ // updateFile(data){
// console.log('上传文件emit',data) // console.log('上传文件emit',data)
@ -51,16 +58,16 @@ export default {
// saveDialog(){ // saveDialog(){
// this.show=true // this.show=true
// }, // },
// 文件上传中处理 // 文件上传中处理 返回参数event, file, fileList
handleFileUploadProgress(event, file, fileList) { handleFileUploadProgress() {
this.upload.isUploading = true this.upload.isUploading = true
}, },
// 文件上传成功处理 // 文件上传成功处理 返回参数response, file, fileList
handleFileSuccess(response, file, fileList) { handleFileSuccess(response) {
console.log('上传文件结果',response) console.log('上传文件结果', response)
this.upload.isUploading = false this.upload.isUploading = false
if(response.code === 500 && !response.msg) response.msg = '上传失败' if (response.code === 500 && !response.msg) response.msg = '上传失败'
this.$alert("<div style='overflow: auto;overflow-x: hidden;max-height: 70vh;padding: 10px 20px 0;'>" + (response.msg || '上传成功' )+ "</div>", "导入结果", { dangerouslyUseHTMLString: true }) this.$alert("<div style='overflow: auto;overflow-x: hidden;max-height: 70vh;padding: 10px 20px 0;'>" + (response.msg || '上传成功') + "</div>", "导入结果", {dangerouslyUseHTMLString: true})
this.$emit('update') this.$emit('update')
this.handleClosed() this.handleClosed()
}, },

View File

@ -1,111 +1,168 @@
<template> <template>
<div class="ems-dashboard-editor-container" style="background-color: #ffffff" v-loading="loading"> <div class="ems-dashboard-editor-container" style="background-color: #ffffff" v-loading="loading">
<el-form :inline="true" class="select-container"> <el-form :inline="true" class="select-container">
<el-form-item label="站点选择"> <el-form-item label="站点选择">
<el-select v-model="siteId" placeholder="请选择换电站名称" :loading="searchLoading" loading-text="正在加载数据" @change="onSearch" clearable> <el-select v-model="siteId" placeholder="请选择换电站名称" :loading="searchLoading" loading-text="正在加载数据"
<el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList" :key="index+'zdxeSelect'"></el-option> @change="onSearch" clearable>
</el-select> <el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList"
</el-form-item> :key="index+'zdxeSelect'"></el-option>
</el-select>
</el-form-item>
<el-form-item label="设备类型">
<el-select v-model="deviceCategory" placeholder="请选择设备类型" @change="onSearch" clearable>
<el-option
v-for="(item,index) in deviceCategoryList"
:key="index+'deviceCategorySelect'"
:label="item.name"
:value="item.code">
</el-option>
</el-select>
</el-form-item>
<el-form-item> <el-form-item>
<!-- <el-button type="primary" @click="onSearch" native-type="button">搜索</el-button>--> <!-- <el-button type="primary" @click="onSearch" native-type="button">搜索</el-button>-->
<!-- <el-button @click="onReset" native-type="button">重置</el-button>--> <!-- <el-button @click="onReset" native-type="button">重置</el-button>-->
</el-form-item> </el-form-item>
</el-form> </el-form>
<el-button type="primary" @click="addDevice" native-type="button">新增设备</el-button> <el-button type="primary" @click="addDevice" native-type="button">新增设备</el-button>
<el-button <el-dropdown @command="(val)=>downloadPointDetail(val,false)">
@click="uploadPointDetail" <el-button
type="success" style="margin-left:10px;"
plain> type="primary"
上传点位清单 plain>
</el-button> 下载点位清单
</el-button>
<el-dropdown-menu slot="dropdown">
<el-dropdown-item v-for="(item,index) in deviceCategoryList" :key="index+'deviceCategoryList'"
:command="item">
{{ item.name }}
</el-dropdown-item>
</el-dropdown-menu>
</el-dropdown>
<!-- <el-dropdown @command="(val)=>uploadPointDetail(val,false)">-->
<!-- <el-button-->
<!-- style="margin-left:10px;"-->
<!-- type="success"-->
<!-- plain>-->
<!-- 上传点位清单-->
<!-- </el-button>-->
<!-- <el-dropdown-menu slot="dropdown">-->
<!-- <el-dropdown-item v-for="(item,index) in deviceCategoryList" :key="index+'deviceCategoryList'"-->
<!-- :command="item">-->
<!-- {{ item.name }}-->
<!-- </el-dropdown-item>-->
<!-- </el-dropdown-menu>-->
<!-- </el-dropdown>-->
<el-table <el-table
class="common-table" class="common-table"
:data="tableData" :data="tableData"
stripe stripe
max-height="600px" max-height="600px"
style="width: 100%;margin-top: 25px"> style="width: 100%;margin-top: 25px">
<el-table-column <el-table-column
prop="siteId" prop="siteId"
label="站点ID"> label="站点ID">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="siteName" prop="siteName"
label="站点名称"> label="站点名称">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="deviceId" prop="deviceId"
label="设备ID" label="设备ID"
> >
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="deviceName" prop="deviceName"
label="设备名称"> label="设备名称">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="categoryName" prop="categoryName"
label="设备类别"> label="设备类别">
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="runningStatus" prop="deviceStatus"
label="在线状态"> label="在线状态">
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{$store.state.ems.deviceStatusOptions[scope.row.runningStatus]}}</span> <span>{{ $store.state.ems.deviceStatusOptions[scope.row.deviceStatus] }}</span>
<pcs-switch v-if="scope.row.deviceCategory === 'PCS' && ![null,'',undefined].includes(scope.row.workStatus)"
style="margin-left:5px;"
:data="{siteId:scope.row.siteId,workStatus:scope.row.workStatus,deviceId:scope.row.deviceId,deviceName:scope.row.deviceName}"
@updateSuccess="getData"/>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
fixed="right" fixed="right"
label="操作" label="操作"
width="250"> width="250">
<template slot-scope="scope"> <template slot-scope="scope">
<el-button <el-button
@click="pointDetail(scope.row)" @click="pointDetail(scope.row,'point')"
type="primary" type="primary"
size="mini"> size="mini">
点位清单 点位清单
</el-button> </el-button>
<el-button <el-button
@click="downloadPointDetail(scope.row)" @click="pointDetail(scope.row,'alarmPoint')"
type="primary"
size="mini">
报警点位清单
</el-button>
<br>
<el-button
@click="downloadPointDetail(scope.row,true)"
style="margin-top:10px;"
type="primary" type="primary"
plain plain
size="mini"> size="mini">
下载点位清单 下载点位清单
</el-button> </el-button>
<el-button
@click="uploadPointDetail(scope.row,true)"
style="margin-top:10px;"
type="success"
plain
size="mini">
上传点位清单
</el-button>
<br> <br>
<el-button <el-button
@click="editDevice(scope.row)" @click="editDevice(scope.row)"
style="margin-top:10px;" style="margin-top:10px;"
type="warning" type="warning"
size="mini"> size="mini">
编辑 编辑
</el-button> </el-button>
<el-button <el-button
type="danger" type="danger"
style="margin-top:10px;" style="margin-top:10px;"
@click="deleteDevice(scope.row)" @click="deleteDevice(scope.row)"
size="mini"> size="mini">
删除 删除
</el-button> </el-button>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<el-pagination <el-pagination
v-show="tableData.length>0" v-show="tableData.length>0"
background background
@size-change="handleSizeChange" @size-change="handleSizeChange"
@current-change="handleCurrentChange" @current-change="handleCurrentChange"
:current-page="pageNum" :current-page="pageNum"
:page-size="pageSize" :page-size="pageSize"
:page-sizes="[10, 20, 30, 40]" :page-sizes="[10, 20, 30, 40]"
layout="total, sizes, prev, pager, next, jumper" layout="total, sizes, prev, pager, next, jumper"
:total="totalSize" :total="totalSize"
style="margin-top:15px;text-align: center" style="margin-top:15px;text-align: center"
> >
</el-pagination> </el-pagination>
<el-dialog :visible.sync="dialogTableVisible" class="ems-dialog" title=" 详细信息" :close-on-click-modal="false" :before-close="handleClosed"> <el-dialog :visible.sync="dialogTableVisible" class="ems-dialog" title="详细信息" :close-on-click-modal="false"
<div class="descriptions-main" style="padding: 0" > :before-close="handleClosed">
<div class="descriptions-main" style="padding: 0">
<el-descriptions direction="vertical" :column="2" :colon="false" border> <el-descriptions direction="vertical" :column="2" :colon="false" border>
<el-descriptions-item v-for="(item,index) in detailData" :key="index+'descriptions'" labelClassName="descriptions-label" contentClassName="descriptions-direction" :span="1" :label="item.label">{{item.value}}</el-descriptions-item> <el-descriptions-item v-for="(item,index) in detailData" :key="index+'descriptions'"
labelClassName="descriptions-label" contentClassName="descriptions-direction" :span="1"
:label="item.label">{{ item.value }}
</el-descriptions-item>
</el-descriptions> </el-descriptions>
</div> </div>
</el-dialog> </el-dialog>
@ -116,90 +173,111 @@
</template> </template>
<script> <script>
import {getDeviceInfoList,getDeviceDetailInfo,deleteService,exportPointList,importPointList} from'@/api/ems/site' import {deleteService, getDeviceDetailInfo, getDeviceInfoList} from '@/api/ems/site'
import {getAllSites} from '@/api/ems/zddt' import {getAllSites} from '@/api/ems/zddt'
import {formatNumber} from "@/filters/ems"; import {formatNumber} from "@/filters/ems";
import {getAllDeviceCategory} from '@/api/ems/search'
import PointTable from './PointTable.vue' import PointTable from './PointTable.vue'
import AddDevice from "./AddDevice.vue"; import AddDevice from "./AddDevice.vue";
import PointUpload from "./PointUpload.vue"; import PointUpload from "./PointUpload.vue";
import {download} from "@/utils/request"; import PcsSwitch from "./PcsSwitch.vue";
export default { export default {
name: "Sblb", name: "Sblb",
components:{AddDevice,PointTable,PointUpload}, components: {AddDevice, PointTable, PointUpload, PcsSwitch},
data() { data() {
return { return {
loading:false, loading: false,
searchLoading:false, searchLoading: false,
mode:'',//新增、编辑设备 mode: '',//新增、编辑设备
editDeviceId:'',//编辑设备id editDeviceId: '',//编辑设备id
siteId:'', siteId: '',
siteList:[], siteList: [],
tableData:[], deviceCategory: '',//搜索栏设备类型
pageSize:10,//分页栏当前每个数据总数 deviceCategoryList: [],//设备类别
pageNum:1,//分页栏当前页数 tableData: [],
totalSize:0,//table表格数据总数 pageSize: 10,//分页栏当前每个数据总数
dialogTableVisible:false, pageNum: 1,//分页栏当前页数
detailData:[ totalSize: 0,//table表格数据总数
{label:'设备ID',value:'',attr:'deviceId'}, dialogTableVisible: false,
{label:'设备名称',value:'',attr:'deviceName'}, detailData: [
{label:'设备类型',value:'',attr:'deviceType'}, {label: '设备ID', value: '', attr: 'deviceId'},
{label:'通信状态',value:'',attr:'communicationStatus'}, {label: '设备名称', value: '', attr: 'deviceName'},
{label:'TCP设备的IP地址',value:'',attr:'ipAddress'}, {label: '设备类型', value: '', attr: 'deviceType'},
{label:'TCP端口号',value:'',attr:'ipPort'}, {label: '通信状态', value: '', attr: 'communicationStatus'},
{label:'串口路径',value:'',attr:'serialPort'}, {label: 'TCP设备的IP地址', value: '', attr: 'ipAddress'},
{label:'波特率',value:'',attr:'baudRate'}, {label: 'TCP端口号', value: '', attr: 'ipPort'},
{label:'数据位',value:'',attr:'dataBits'}, {label: '串口路径', value: '', attr: 'serialPort'},
{label:'停止位',value:'',attr:'stopBits'}, {label: '波特率', value: '', attr: 'baudRate'},
{label:'校验位',value:'',attr:'parity'}, {label: '数据位', value: '', attr: 'dataBits'},
{label:'设备描述',value:'',attr:'description'}, {label: '停止位', value: '', attr: 'stopBits'},
{label:'设备类别',value:'',attr:'deviceCategory'}, {label: '校验位', value: '', attr: 'parity'},
{label:'上级设备ID',value:'',attr:'parentId'}, {label: '设备描述', value: '', attr: 'description'},
{label:'设备唯一标识符',value:'',attr:'deviceId'}, {label: '设备类别', value: '', attr: 'deviceCategory'},
{label: '上级设备ID', value: '', attr: 'parentId'},
{label: '设备唯一标识符', value: '', attr: 'deviceId'},
], ],
detailformatAttr:{ detailformatAttr: {
'deviceType':'deviceTypeOptions', 'deviceType': 'deviceTypeOptions',
'communicationStatus':'communicationStatusOptions', 'communicationStatus': 'communicationStatusOptions',
} }
} }
}, },
methods:{ methods: {
// 获取设备类别
getDeviceCategoryList() {
getAllDeviceCategory().then(response => {
this.deviceCategoryList = response?.data || []
})
},
// 查看设备电位表格 // 查看设备电位表格
pointDetail(row){ pointDetail(row, dataType) {
this.$refs.pointTable.showTable(row) this.$refs.pointTable.showTable(row, dataType)
}, },
// 下载点位清单 // 下载点位清单
downloadPointDetail({siteId,deviceCategory,deviceName}){ downloadPointDetail(command, isDetail = false) {
const siteId = isDetail ? command.siteId : this.siteId
const deviceCategory = isDetail ? command.deviceCategory : command.code
const categoryName = isDetail ? command.categoryName : command.name
const deviceId = isDetail ? command.deviceId : null
console.log('下载', command, isDetail)
this.download('ems/pointMatch/export', { this.download('ems/pointMatch/export', {
siteId, siteId,
deviceCategory deviceCategory,
}, `点位清单_${deviceName}_${new Date().getTime()}.xlsx`) deviceId,
}, `点位清单_${categoryName}_${new Date().getTime()}.xlsx`)
}, },
// 上传点位清单 // 上传点位清单
uploadPointDetail(){ uploadPointDetail(command, isDetail = false) {
this.$refs.pointUpload.showDialog() const siteId = isDetail ? command.siteId : this.siteId
const deviceCategory = isDetail ? command.deviceCategory : command.code
const categoryName = isDetail ? command.categoryName : command.name
const deviceId = isDetail ? command.deviceId : ''
console.log('上传', command, isDetail)
this.$refs.pointUpload.showDialog({siteId, deviceCategory, categoryName, deviceId})
}, },
clearEditDeviceData(){ clearEditDeviceData() {
this.mode = ''; this.mode = '';
this.editDeviceId='' this.editDeviceId = ''
}, },
// 新增设备 展示弹窗 // 新增设备 展示弹窗
addDevice(){ addDevice() {
this.mode = 'add'; this.mode = 'add';
this.$refs.addDevice.dialogTableVisible = true this.$refs.addDevice.dialogTableVisible = true
}, },
// 编辑设备 // 编辑设备
editDevice(row){ editDevice(row) {
this.mode = 'edit'; this.mode = 'edit';
this.editDeviceId = row.id this.editDeviceId = row.id
this.$refs.addDevice.dialogTableVisible = true; this.$refs.addDevice.dialogTableVisible = true;
}, },
//删除设备 //删除设备
deleteDevice(row){ deleteDevice(row) {
this.$confirm(`确认要设备${row.deviceName}吗?`, { this.$confirm(`确认要设备${row.deviceName}吗?`, {
confirmButtonText: '确定', confirmButtonText: '确定',
cancelButtonText: '取消', cancelButtonText: '取消',
showClose:false, showClose: false,
closeOnClickModal:false, closeOnClickModal: false,
type: 'warning', type: 'warning',
beforeClose: (action, instance, done) => { beforeClose: (action, instance, done) => {
if (action === 'confirm') { if (action === 'confirm') {
@ -225,73 +303,82 @@ export default {
//取消关机 //取消关机
}); });
}, },
handleClosed(done){ handleClosed(done) {
this.detailData.forEach((item)=>{item.value = ''}) this.detailData.forEach((item) => {
item.value = ''
})
done() done()
}, },
toDetail(id){ toDetail(id) {
this.loading = true this.loading = true
getDeviceDetailInfo(id).then(response => { getDeviceDetailInfo(id).then(response => {
const data = response?.data || {}; const data = response?.data || {};
this.detailData.forEach(item=>{ this.detailData.forEach(item => {
if(this.detailformatAttr[item.attr]){ if (this.detailformatAttr[item.attr]) {
item.value=formatNumber(this.$store.state.ems[this.detailformatAttr[item.attr]][data[item.attr]]); item.value = formatNumber(this.$store.state.ems[this.detailformatAttr[item.attr]][data[item.attr]]);
}else{ } else {
item.value=formatNumber(data[item.attr]); item.value = formatNumber(data[item.attr]);
} }
}) })
this.dialogTableVisible = true this.dialogTableVisible = true
}).finally(() => {this.loading = false}) }).finally(() => {
this.loading = false
})
}, },
// 分页 // 分页
handleSizeChange(val) { handleSizeChange(val) {
this.pageSize = val; this.pageSize = val;
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
handleCurrentChange(val) { handleCurrentChange(val) {
this.pageNum = val this.pageNum = val
this.$nextTick(()=>{ this.$nextTick(() => {
this.getData() this.getData()
}) })
}, },
// 搜索 // 搜索
onSearch(){ onSearch() {
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 重置 // 重置
onReset(){ onReset() {
this.siteId='' this.siteId = ''
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getData() this.getData()
}, },
// 获取数据 // 获取数据
getData(){ getData() {
this.loading=true this.loading = true
const {siteId,pageNum,pageSize} =this const {siteId, deviceCategory, pageNum, pageSize} = this
getDeviceInfoList({siteId,pageNum,pageSize}).then(response => { getDeviceInfoList({siteId, deviceCategory, pageNum, pageSize}).then(response => {
this.tableData=response?.rows || []; this.tableData = response?.rows || [];
this.totalSize = response?.total || 0 this.totalSize = response?.total || 0
}).finally(() => {this.loading=false}) }).finally(() => {
this.loading = false
})
}, },
//获取站点列表 //获取站点列表
getZdList(){ getZdList() {
this.searchLoading=true this.searchLoading = true
return getAllSites().then(response => { return getAllSites().then(response => {
this.siteList = response?.data || [] this.siteList = response?.data || []
if(this.siteList.length>0) this.siteId = this.siteList[0].siteId if (this.siteList.length > 0) this.siteId = this.siteList[0].siteId
}).finally(() => {this.searchLoading=false}) }).finally(() => {
this.searchLoading = false
})
} }
}, },
mounted() { mounted() {
this.loading=true this.loading = true
this.siteId='' this.siteId = ''
this.pageNum =1//每次搜索从1开始搜索 this.pageNum = 1//每次搜索从1开始搜索
this.getZdList().then(()=>{ this.getDeviceCategoryList()
this.getZdList().then(() => {
this.getData() this.getData()
}) })
} }

View File

@ -6,30 +6,39 @@
<pane size="16"> <pane size="16">
<el-col> <el-col>
<div class="head-container"> <div class="head-container">
<el-input v-model="deptName" placeholder="请输入部门名称" clearable size="small" prefix-icon="el-icon-search" style="margin-bottom: 20px" /> <el-input v-model="deptName" placeholder="请输入部门名称" clearable size="small"
prefix-icon="el-icon-search" style="margin-bottom: 20px"/>
</div> </div>
<div class="head-container"> <div class="head-container">
<el-tree :data="deptOptions" :props="defaultProps" :expand-on-click-node="false" :filter-node-method="filterNode" ref="tree" node-key="id" default-expand-all highlight-current @node-click="handleNodeClick" /> <el-tree :data="deptOptions" :props="defaultProps" :expand-on-click-node="false"
:filter-node-method="filterNode" ref="tree" node-key="id" default-expand-all highlight-current
@node-click="handleNodeClick"/>
</div> </div>
</el-col> </el-col>
</pane> </pane>
<!--用户数据--> <!--用户数据-->
<pane size="84"> <pane size="84">
<el-col> <el-col>
<el-form :model="queryParams" ref="queryForm" size="small" :inline="true" v-show="showSearch" label-width="68px"> <el-form :model="queryParams" ref="queryForm" size="small" :inline="true" v-show="showSearch"
label-width="68px">
<el-form-item label="用户名称" prop="userName"> <el-form-item label="用户名称" prop="userName">
<el-input v-model="queryParams.userName" placeholder="请输入用户名称" clearable style="width: 240px" @keyup.enter.native="handleQuery" /> <el-input v-model="queryParams.userName" placeholder="请输入用户名称" clearable style="width: 240px"
@keyup.enter.native="handleQuery"/>
</el-form-item> </el-form-item>
<el-form-item label="手机号码" prop="phonenumber"> <el-form-item label="手机号码" prop="phonenumber">
<el-input v-model="queryParams.phonenumber" placeholder="请输入手机号码" clearable style="width: 240px" @keyup.enter.native="handleQuery" /> <el-input v-model="queryParams.phonenumber" placeholder="请输入手机号码" clearable style="width: 240px"
@keyup.enter.native="handleQuery"/>
</el-form-item> </el-form-item>
<el-form-item label="状态" prop="status"> <el-form-item label="状态" prop="status">
<el-select v-model="queryParams.status" placeholder="用户状态" clearable style="width: 240px"> <el-select v-model="queryParams.status" placeholder="用户状态" clearable style="width: 240px">
<el-option v-for="dict in dict.type.sys_normal_disable" :key="dict.value" :label="dict.label" :value="dict.value" /> <el-option v-for="dict in dict.type.sys_normal_disable" :key="dict.value" :label="dict.label"
:value="dict.value"/>
</el-select> </el-select>
</el-form-item> </el-form-item>
<el-form-item label="创建时间"> <el-form-item label="创建时间">
<el-date-picker v-model="dateRange" style="width: 240px" value-format="yyyy-MM-dd" type="daterange" range-separator="-" start-placeholder="开始日期" end-placeholder="结束日期"></el-date-picker> <el-date-picker v-model="dateRange" style="width: 240px" value-format="yyyy-MM-dd" type="daterange"
range-separator="-" start-placeholder="开始日期"
end-placeholder="结束日期"></el-date-picker>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button> <el-button type="primary" icon="el-icon-search" size="mini" @click="handleQuery">搜索</el-button>
@ -39,33 +48,48 @@
<el-row :gutter="10" class="mb8"> <el-row :gutter="10" class="mb8">
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAdd" v-hasPermi="['system:user:add']">新增</el-button> <el-button type="primary" plain icon="el-icon-plus" size="mini" @click="handleAdd"
v-hasPermi="['system:user:add']">新增
</el-button>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="success" plain icon="el-icon-edit" size="mini" :disabled="single" @click="handleUpdate" v-hasPermi="['system:user:edit']">修改</el-button> <el-button type="success" plain icon="el-icon-edit" size="mini" :disabled="single" @click="handleUpdate"
v-hasPermi="['system:user:edit']">修改
</el-button>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="danger" plain icon="el-icon-delete" size="mini" :disabled="multiple" @click="handleDelete" v-hasPermi="['system:user:remove']">删除</el-button> <el-button type="danger" plain icon="el-icon-delete" size="mini" :disabled="multiple"
@click="handleDelete" v-hasPermi="['system:user:remove']">删除
</el-button>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="info" plain icon="el-icon-upload2" size="mini" @click="handleImport" v-hasPermi="['system:user:import']">导入</el-button> <el-button type="info" plain icon="el-icon-upload2" size="mini" @click="handleImport"
v-hasPermi="['system:user:import']">导入
</el-button>
</el-col> </el-col>
<el-col :span="1.5"> <el-col :span="1.5">
<el-button type="warning" plain icon="el-icon-download" size="mini" @click="handleExport" v-hasPermi="['system:user:export']">导出</el-button> <el-button type="warning" plain icon="el-icon-download" size="mini" @click="handleExport"
v-hasPermi="['system:user:export']">导出
</el-button>
</el-col> </el-col>
<right-toolbar :showSearch.sync="showSearch" @queryTable="getList" :columns="columns"></right-toolbar> <right-toolbar :showSearch.sync="showSearch" @queryTable="getList" :columns="columns"></right-toolbar>
</el-row> </el-row>
<el-table v-loading="loading" :data="userList" @selection-change="handleSelectionChange"> <el-table v-loading="loading" :data="userList" @selection-change="handleSelectionChange">
<el-table-column type="selection" width="50" align="center" /> <el-table-column type="selection" width="50" align="center"/>
<el-table-column label="用户编号" align="center" key="userId" prop="userId" v-if="columns[0].visible" /> <el-table-column label="用户编号" align="center" key="userId" prop="userId" v-if="columns[0].visible"/>
<el-table-column label="用户名称" align="center" key="userName" prop="userName" v-if="columns[1].visible" :show-overflow-tooltip="true" /> <el-table-column label="用户名称" align="center" key="userName" prop="userName" v-if="columns[1].visible"
<el-table-column label="用户昵称" align="center" key="nickName" prop="nickName" v-if="columns[2].visible" :show-overflow-tooltip="true" /> :show-overflow-tooltip="true"/>
<el-table-column label="部门" align="center" key="deptName" prop="dept.deptName" v-if="columns[3].visible" :show-overflow-tooltip="true" /> <el-table-column label="用户昵称" align="center" key="nickName" prop="nickName" v-if="columns[2].visible"
<el-table-column label="手机号码" align="center" key="phonenumber" prop="phonenumber" v-if="columns[4].visible" width="120" /> :show-overflow-tooltip="true"/>
<el-table-column label="部门" align="center" key="deptName" prop="dept.deptName" v-if="columns[3].visible"
:show-overflow-tooltip="true"/>
<el-table-column label="手机号码" align="center" key="phonenumber" prop="phonenumber"
v-if="columns[4].visible" width="120"/>
<el-table-column label="状态" align="center" key="status" v-if="columns[5].visible"> <el-table-column label="状态" align="center" key="status" v-if="columns[5].visible">
<template slot-scope="scope"> <template slot-scope="scope">
<el-switch v-model="scope.row.status" active-value="0" inactive-value="1" @change="handleStatusChange(scope.row)"></el-switch> <el-switch v-model="scope.row.status" active-value="0" inactive-value="1"
@change="handleStatusChange(scope.row)"></el-switch>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column label="创建时间" align="center" prop="createTime" v-if="columns[6].visible" width="160"> <el-table-column label="创建时间" align="center" prop="createTime" v-if="columns[6].visible" width="160">
@ -75,20 +99,30 @@
</el-table-column> </el-table-column>
<el-table-column label="操作" align="center" width="160" class-name="small-padding fixed-width"> <el-table-column label="操作" align="center" width="160" class-name="small-padding fixed-width">
<template slot-scope="scope" v-if="scope.row.userId !== 1"> <template slot-scope="scope" v-if="scope.row.userId !== 1">
<el-button size="mini" type="text" icon="el-icon-edit" @click="handleUpdate(scope.row)" v-hasPermi="['system:user:edit']">修改</el-button> <el-button size="mini" type="text" icon="el-icon-edit" @click="handleUpdate(scope.row)"
<el-button size="mini" type="text" icon="el-icon-delete" @click="handleDelete(scope.row)" v-hasPermi="['system:user:remove']">删除</el-button> v-hasPermi="['system:user:edit']">修改
<el-dropdown size="mini" @command="(command) => handleCommand(command, scope.row)" v-hasPermi="['system:user:resetPwd', 'system:user:edit']"> </el-button>
<el-button size="mini" type="text" icon="el-icon-delete" @click="handleDelete(scope.row)"
v-hasPermi="['system:user:remove']">删除
</el-button>
<el-dropdown size="mini" @command="(command) => handleCommand(command, scope.row)"
v-hasPermi="['system:user:resetPwd', 'system:user:edit']">
<el-button size="mini" type="text" icon="el-icon-d-arrow-right">更多</el-button> <el-button size="mini" type="text" icon="el-icon-d-arrow-right">更多</el-button>
<el-dropdown-menu slot="dropdown"> <el-dropdown-menu slot="dropdown">
<el-dropdown-item command="handleResetPwd" icon="el-icon-key" v-hasPermi="['system:user:resetPwd']">重置密码</el-dropdown-item> <el-dropdown-item command="handleResetPwd" icon="el-icon-key"
<el-dropdown-item command="handleAuthRole" icon="el-icon-circle-check" v-hasPermi="['system:user:edit']">分配角色</el-dropdown-item> v-hasPermi="['system:user:resetPwd']">重置密码
</el-dropdown-item>
<el-dropdown-item command="handleAuthRole" icon="el-icon-circle-check"
v-hasPermi="['system:user:edit']">分配角色
</el-dropdown-item>
</el-dropdown-menu> </el-dropdown-menu>
</el-dropdown> </el-dropdown>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </el-table>
<pagination v-show="total > 0" :total="total" :page.sync="queryParams.pageNum" :limit.sync="queryParams.pageSize" @pagination="getList" /> <pagination v-show="total > 0" :total="total" :page.sync="queryParams.pageNum"
:limit.sync="queryParams.pageSize" @pagination="getList"/>
</el-col> </el-col>
</pane> </pane>
</splitpanes> </splitpanes>
@ -100,36 +134,38 @@
<el-row> <el-row>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="用户昵称" prop="nickName"> <el-form-item label="用户昵称" prop="nickName">
<el-input v-model="form.nickName" placeholder="请输入用户昵称" maxlength="30" /> <el-input v-model="form.nickName" placeholder="请输入用户昵称" maxlength="30"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="归属部门" prop="deptId"> <el-form-item label="归属部门" prop="deptId">
<treeselect v-model="form.deptId" :options="enabledDeptOptions" :show-count="true" placeholder="请选择归属部门" /> <treeselect v-model="form.deptId" :options="enabledDeptOptions" :show-count="true"
placeholder="请选择归属部门"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="手机号码" prop="phonenumber"> <el-form-item label="手机号码" prop="phonenumber">
<el-input v-model="form.phonenumber" placeholder="请输入手机号码" maxlength="11" /> <el-input v-model="form.phonenumber" placeholder="请输入手机号码" maxlength="11"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="邮箱" prop="email"> <el-form-item label="邮箱" prop="email">
<el-input v-model="form.email" placeholder="请输入邮箱" maxlength="50" /> <el-input v-model="form.email" placeholder="请输入邮箱" maxlength="50"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
<el-row> <el-row>
<el-col :span="12"> <el-col :span="12">
<el-form-item v-if="form.userId == undefined" label="用户名称" prop="userName"> <el-form-item v-if="form.userId == undefined" label="用户名称" prop="userName">
<el-input v-model="form.userName" placeholder="请输入用户名称" maxlength="30" /> <el-input v-model="form.userName" placeholder="请输入用户名称" maxlength="30"/>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item v-if="form.userId == undefined" label="用户密码" prop="password"> <el-form-item v-if="form.userId == undefined" label="用户密码" prop="password">
<el-input v-model="form.password" placeholder="请输入用户密码" type="password" maxlength="20" show-password /> <el-input v-model="form.password" placeholder="请输入用户密码" type="password" maxlength="20"
show-password/>
</el-form-item> </el-form-item>
</el-col> </el-col>
</el-row> </el-row>
@ -137,14 +173,17 @@
<el-col :span="12"> <el-col :span="12">
<el-form-item label="用户性别"> <el-form-item label="用户性别">
<el-select v-model="form.sex" placeholder="请选择性别"> <el-select v-model="form.sex" placeholder="请选择性别">
<el-option v-for="dict in dict.type.sys_user_sex" :key="dict.value" :label="dict.label" :value="dict.value"></el-option> <el-option v-for="dict in dict.type.sys_user_sex" :key="dict.value" :label="dict.label"
:value="dict.value"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="状态"> <el-form-item label="状态">
<el-radio-group v-model="form.status"> <el-radio-group v-model="form.status">
<el-radio v-for="dict in dict.type.sys_normal_disable" :key="dict.value" :label="dict.value">{{ dict.label }}</el-radio> <el-radio v-for="dict in dict.type.sys_normal_disable" :key="dict.value" :label="dict.value">
{{ dict.label }}
</el-radio>
</el-radio-group> </el-radio-group>
</el-form-item> </el-form-item>
</el-col> </el-col>
@ -153,14 +192,33 @@
<el-col :span="12"> <el-col :span="12">
<el-form-item label="岗位"> <el-form-item label="岗位">
<el-select v-model="form.postIds" multiple placeholder="请选择岗位"> <el-select v-model="form.postIds" multiple placeholder="请选择岗位">
<el-option v-for="item in postOptions" :key="item.postId" :label="item.postName" :value="item.postId" :disabled="item.status == 1" ></el-option> <el-option v-for="item in postOptions" :key="item.postId" :label="item.postName" :value="item.postId"
:disabled="item.status == 1"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
</el-col> </el-col>
<el-col :span="12"> <el-col :span="12">
<el-form-item label="角色"> <el-form-item label="角色">
<el-select v-model="form.roleIds" multiple placeholder="请选择角色"> <el-select v-model="form.roleIds" multiple placeholder="请选择角色">
<el-option v-for="item in roleOptions" :key="item.roleId" :label="item.roleName" :value="item.roleId" :disabled="item.status == 1"></el-option> <el-option v-for="item in roleOptions" :key="item.roleId" :label="item.roleName" :value="item.roleId"
:disabled="item.status == 1"></el-option>
</el-select>
</el-form-item>
</el-col>
</el-row>
<el-row>
<el-col :span="24">
<el-form-item label="归属站点" name="belongSite">
<el-select
v-model="form.belongSite"
multiple
collapse-tags
placeholder="请选择"
style="width:100%"
@change="selectBelongSite">
<el-option :label="item.siteName" :value="item.siteId" v-for="(item,index) in siteList"
:disabled="item.siteId!== 'all' && (form.belongSite || []).includes('all')"
:key="index+'zdxeSelect'"></el-option>
</el-select> </el-select>
</el-form-item> </el-form-item>
</el-col> </el-col>
@ -181,15 +239,20 @@
<!-- 用户导入对话框 --> <!-- 用户导入对话框 -->
<el-dialog :title="upload.title" :visible.sync="upload.open" width="400px" append-to-body> <el-dialog :title="upload.title" :visible.sync="upload.open" width="400px" append-to-body>
<el-upload ref="upload" :limit="1" accept=".xlsx, .xls" :headers="upload.headers" :action="upload.url + '?updateSupport=' + upload.updateSupport" :disabled="upload.isUploading" :on-progress="handleFileUploadProgress" :on-success="handleFileSuccess" :auto-upload="false" drag> <el-upload ref="upload" :limit="1" accept=".xlsx, .xls" :headers="upload.headers"
:action="upload.url + '?updateSupport=' + upload.updateSupport" :disabled="upload.isUploading"
:on-progress="handleFileUploadProgress" :on-success="handleFileSuccess" :auto-upload="false" drag>
<i class="el-icon-upload"></i> <i class="el-icon-upload"></i>
<div class="el-upload__text">将文件拖到此处<em>点击上传</em></div> <div class="el-upload__text">将文件拖到此处<em>点击上传</em></div>
<div class="el-upload__tip text-center" slot="tip"> <div class="el-upload__tip text-center" slot="tip">
<div class="el-upload__tip" slot="tip"> <div class="el-upload__tip" slot="tip">
<el-checkbox v-model="upload.updateSupport" />是否更新已经存在的用户数据 <el-checkbox v-model="upload.updateSupport"/>
是否更新已经存在的用户数据
</div> </div>
<span>仅允许导入xlsxlsx格式文件</span> <span>仅允许导入xlsxlsx格式文件</span>
<el-link type="primary" :underline="false" style="font-size: 12px; vertical-align: baseline" @click="importTemplate">下载模板</el-link> <el-link type="primary" :underline="false" style="font-size: 12px; vertical-align: baseline"
@click="importTemplate">下载模板
</el-link>
</div> </div>
</el-upload> </el-upload>
<div slot="footer" class="dialog-footer"> <div slot="footer" class="dialog-footer">
@ -201,17 +264,27 @@
</template> </template>
<script> <script>
import { listUser, getUser, delUser, addUser, updateUser, resetUserPwd, changeUserStatus, deptTreeSelect } from "@/api/system/user" import {
import { getToken } from "@/utils/auth" addUser,
changeUserStatus,
delUser,
deptTreeSelect,
getUser,
listUser,
resetUserPwd,
updateUser
} from "@/api/system/user"
import {getAllSites} from '@/api/ems/zddt'
import {getToken} from "@/utils/auth"
import Treeselect from "@riophae/vue-treeselect" import Treeselect from "@riophae/vue-treeselect"
import "@riophae/vue-treeselect/dist/vue-treeselect.css" import "@riophae/vue-treeselect/dist/vue-treeselect.css"
import { Splitpanes, Pane } from "splitpanes" import {Pane, Splitpanes} from "splitpanes"
import "splitpanes/dist/splitpanes.css" import "splitpanes/dist/splitpanes.css"
export default { export default {
name: "User", name: "User",
dicts: ['sys_normal_disable', 'sys_user_sex'], dicts: ['sys_normal_disable', 'sys_user_sex'],
components: { Treeselect, Splitpanes, Pane }, components: {Treeselect, Splitpanes, Pane},
data() { data() {
return { return {
// 遮罩层 // 遮罩层
@ -228,6 +301,8 @@ export default {
total: 0, total: 0,
// 用户表格数据 // 用户表格数据
userList: null, userList: null,
// 站点列表数据
siteList: [],
// 弹出层标题 // 弹出层标题
title: "", title: "",
// 所有部门树选项 // 所有部门树选项
@ -263,7 +338,7 @@ export default {
// 是否更新已经存在的用户数据 // 是否更新已经存在的用户数据
updateSupport: 0, updateSupport: 0,
// 设置上传的请求头部 // 设置上传的请求头部
headers: { Authorization: "Bearer " + getToken() }, headers: {Authorization: "Bearer " + getToken()},
// 上传的地址 // 上传的地址
url: process.env.VUE_APP_BASE_API + "/system/user/importData" url: process.env.VUE_APP_BASE_API + "/system/user/importData"
}, },
@ -278,27 +353,27 @@ export default {
}, },
// 列信息 // 列信息
columns: [ columns: [
{ key: 0, label: `用户编号`, visible: true }, {key: 0, label: `用户编号`, visible: true},
{ key: 1, label: `用户名称`, visible: true }, {key: 1, label: `用户名称`, visible: true},
{ key: 2, label: `用户昵称`, visible: true }, {key: 2, label: `用户昵称`, visible: true},
{ key: 3, label: `部门`, visible: true }, {key: 3, label: `部门`, visible: true},
{ key: 4, label: `手机号码`, visible: true }, {key: 4, label: `手机号码`, visible: true},
{ key: 5, label: `状态`, visible: true }, {key: 5, label: `状态`, visible: true},
{ key: 6, label: `创建时间`, visible: true } {key: 6, label: `创建时间`, visible: true}
], ],
// 表单校验 // 表单校验
rules: { rules: {
userName: [ userName: [
{ required: true, message: "用户名称不能为空", trigger: "blur" }, {required: true, message: "用户名称不能为空", trigger: "blur"},
{ min: 2, max: 20, message: '用户名称长度必须介于 2 和 20 之间', trigger: 'blur' } {min: 2, max: 20, message: '用户名称长度必须介于 2 和 20 之间', trigger: 'blur'}
], ],
nickName: [ nickName: [
{ required: true, message: "用户昵称不能为空", trigger: "blur" } {required: true, message: "用户昵称不能为空", trigger: "blur"}
], ],
password: [ password: [
{ required: true, message: "用户密码不能为空", trigger: "blur" }, {required: true, message: "用户密码不能为空", trigger: "blur"},
{ min: 5, max: 20, message: '用户密码长度必须介于 5 和 20 之间', trigger: 'blur' }, {min: 5, max: 20, message: '用户密码长度必须介于 5 和 20 之间', trigger: 'blur'},
{ pattern: /^[^<>"'|\\]+$/, message: "不能包含非法字符:< > \" ' \\\ |", trigger: "blur" } {pattern: /^[^<>"'|\\]+$/, message: "不能包含非法字符:< > \" ' \\\ |", trigger: "blur"}
], ],
email: [ email: [
{ {
@ -313,7 +388,8 @@ export default {
message: "请输入正确的手机号码", message: "请输入正确的手机号码",
trigger: "blur" trigger: "blur"
} }
] ],
} }
} }
}, },
@ -324,6 +400,7 @@ export default {
} }
}, },
created() { created() {
this.getZdList()
this.getList() this.getList()
this.getDeptTree() this.getDeptTree()
this.getConfigKey("sys.user.initPassword").then(response => { this.getConfigKey("sys.user.initPassword").then(response => {
@ -331,14 +408,37 @@ export default {
}) })
}, },
methods: { methods: {
selectBelongSite(data) {
console.log('选中的站点', data)
if (data.includes("all")) {
this.form.belongSite = ['all']
return
}
if (this.siteList.length && data.length === (this.siteList.length - 1)) {
this.form.belongSite = ['all']
}
},
//获取站点列表
getZdList() {
return getAllSites().then(response => {
this.siteList = response?.data || []
if (this.siteList.length > 0) {
this.siteList.unshift({
id: 'all',
siteId: "all",
siteName: "全部"
})
}
})
},
/** 查询用户列表 */ /** 查询用户列表 */
getList() { getList() {
this.loading = true this.loading = true
listUser(this.addDateRange(this.queryParams, this.dateRange)).then(response => { listUser(this.addDateRange(this.queryParams, this.dateRange)).then(response => {
this.userList = response.rows this.userList = response.rows
this.total = response.total this.total = response.total
this.loading = false this.loading = false
} }
) )
}, },
/** 查询部门下拉树结构 */ /** 查询部门下拉树结构 */
@ -373,11 +473,11 @@ export default {
// 用户状态修改 // 用户状态修改
handleStatusChange(row) { handleStatusChange(row) {
let text = row.status === "0" ? "启用" : "停用" let text = row.status === "0" ? "启用" : "停用"
this.$modal.confirm('确认要"' + text + '""' + row.userName + '"用户吗?').then(function() { this.$modal.confirm('确认要"' + text + '""' + row.userName + '"用户吗?').then(function () {
return changeUserStatus(row.userId, row.status) return changeUserStatus(row.userId, row.status)
}).then(() => { }).then(() => {
this.$modal.msgSuccess(text + "成功") this.$modal.msgSuccess(text + "成功")
}).catch(function() { }).catch(function () {
row.status = row.status === "0" ? "1" : "0" row.status = row.status === "0" ? "1" : "0"
}) })
}, },
@ -400,7 +500,8 @@ export default {
status: "0", status: "0",
remark: undefined, remark: undefined,
postIds: [], postIds: [],
roleIds: [] roleIds: [],
belongSite: []
} }
this.resetForm("form") this.resetForm("form")
}, },
@ -455,6 +556,7 @@ export default {
this.form = response.data this.form = response.data
this.postOptions = response.posts this.postOptions = response.posts
this.roleOptions = response.roles this.roleOptions = response.roles
this.$set(this.form, "belongSite", response?.data?.belongSite ? JSON.parse(response.data.belongSite) : [])
this.$set(this.form, "postIds", response.postIds) this.$set(this.form, "postIds", response.postIds)
this.$set(this.form, "roleIds", response.roleIds) this.$set(this.form, "roleIds", response.roleIds)
this.open = true this.open = true
@ -475,29 +577,30 @@ export default {
return "不能包含非法字符:< > \" ' \\\ |" return "不能包含非法字符:< > \" ' \\\ |"
} }
}, },
}).then(({ value }) => { }).then(({value}) => {
resetUserPwd(row.userId, value).then(response => { resetUserPwd(row.userId, value).then(response => {
this.$modal.msgSuccess("修改成功,新密码是:" + value) this.$modal.msgSuccess("修改成功,新密码是:" + value)
}) })
}).catch(() => {}) }).catch(() => {
})
}, },
/** 分配角色操作 */ /** 分配角色操作 */
handleAuthRole: function(row) { handleAuthRole: function (row) {
const userId = row.userId const userId = row.userId
this.$router.push("/system/user-auth/role/" + userId) this.$router.push("/system/user-auth/role/" + userId)
}, },
/** 提交按钮 */ /** 提交按钮 */
submitForm: function() { submitForm: function () {
this.$refs["form"].validate(valid => { this.$refs["form"].validate(valid => {
if (valid) { if (valid) {
if (this.form.userId != undefined) { if (this.form.userId != undefined) {
updateUser(this.form).then(response => { updateUser({...this.form, belongSite: JSON.stringify(this.form.belongSite)}).then(response => {
this.$modal.msgSuccess("修改成功") this.$modal.msgSuccess("修改成功")
this.open = false this.open = false
this.getList() this.getList()
}) })
} else { } else {
addUser(this.form).then(response => { addUser({...this.form, belongSite: JSON.stringify(this.form.belongSite)}).then(response => {
this.$modal.msgSuccess("新增成功") this.$modal.msgSuccess("新增成功")
this.open = false this.open = false
this.getList() this.getList()
@ -509,12 +612,13 @@ export default {
/** 删除按钮操作 */ /** 删除按钮操作 */
handleDelete(row) { handleDelete(row) {
const userIds = row.userId || this.ids const userIds = row.userId || this.ids
this.$modal.confirm('是否确认删除用户编号为"' + userIds + '"的数据项?').then(function() { this.$modal.confirm('是否确认删除用户编号为"' + userIds + '"的数据项?').then(function () {
return delUser(userIds) return delUser(userIds)
}).then(() => { }).then(() => {
this.getList() this.getList()
this.$modal.msgSuccess("删除成功") this.$modal.msgSuccess("删除成功")
}).catch(() => {}) }).catch(() => {
})
}, },
/** 导出按钮操作 */ /** 导出按钮操作 */
handleExport() { handleExport() {
@ -529,8 +633,7 @@ export default {
}, },
/** 下载模板操作 */ /** 下载模板操作 */
importTemplate() { importTemplate() {
this.download('system/user/importTemplate', { this.download('system/user/importTemplate', {}, `user_template_${new Date().getTime()}.xlsx`)
}, `user_template_${new Date().getTime()}.xlsx`)
}, },
// 文件上传中处理 // 文件上传中处理
handleFileUploadProgress(event, file, fileList) { handleFileUploadProgress(event, file, fileList) {
@ -541,7 +644,7 @@ export default {
this.upload.open = false this.upload.open = false
this.upload.isUploading = false this.upload.isUploading = false
this.$refs.upload.clearFiles() this.$refs.upload.clearFiles()
this.$alert("<div style='overflow: auto;overflow-x: hidden;max-height: 70vh;padding: 10px 20px 0;'>" + response.msg + "</div>", "导入结果", { dangerouslyUseHTMLString: true }) this.$alert("<div style='overflow: auto;overflow-x: hidden;max-height: 70vh;padding: 10px 20px 0;'>" + response.msg + "</div>", "导入结果", {dangerouslyUseHTMLString: true})
this.getList() this.getList()
}, },
// 提交上传文件 // 提交上传文件