This commit is contained in:
2026-03-05 16:34:25 +08:00
parent 13ee9e66cc
commit 69e199e9cc
15 changed files with 1398 additions and 677 deletions

View File

@ -1,24 +1,48 @@
<template>
<view class="page-container">
<!-- 顶部6个数据 -->
<uni-grid class="info-grid" :square="false" :column="2" :showBorder="false">
<uni-grid-item v-for="(item,index) in runningHeadData" :key="index+'head'">
<view class="grid-item-box">
<image :src="require('@/static/images/ems/pcs/'+item.img+'.jpg')" class="icon" alt=""/>
<view class="title">{{item.title}}</view>
<view class="text">{{runningHeadInfo[item.attr] | formatNumber}}</view>
</view>
</uni-grid-item>
</uni-grid>
<uni-collapse ref="collapse" accordion v-if="list.length > 0">
<uni-collapse-item v-for="(item,index) in list" :key="index+'pcs'" :open="index===0"
class="common-collapse-item" :class="handleCardClass(item)">
<template>
<view class="page-container">
<!-- 顶部总览横向展示 -->
<scroll-view class="info-overview-scroll" scroll-x>
<view class="info-overview-row">
<view class="info-overview-card" v-for="(item,index) in runningHeadCards" :key="index+'head'">
<view class="grid-item-box">
<image :src="require('@/static/images/ems/pcs/'+item.img+'.jpg')" class="icon" alt="" />
<view class="title">{{item.title}}</view>
<view class="text">{{item.value | formatNumber}}</view>
</view>
</view>
</view>
</scroll-view>
<!-- 设备标签横向展示 -->
<scroll-view class="pcs-tags-scroll" scroll-x>
<view class="pcs-tags-row">
<view
class="pcs-tag-item"
:class="{ active: !selectedPcsId }"
@click="handleTagClick('')"
>
全部
</view>
<view
v-for="(item, index) in pcsDeviceList"
:key="index + 'pcsTag'"
class="pcs-tag-item"
:class="{ active: selectedPcsId === (item.deviceId || item.id || '') }"
@click="handleTagClick(item.deviceId || item.id || '')"
>
{{ item.deviceName || item.deviceId || item.id || 'PCS' }}
</view>
</view>
</scroll-view>
<uni-collapse ref="collapse" accordion v-if="filteredList.length > 0">
<uni-collapse-item v-for="(item,index) in filteredList" :key="index+'pcs'" :open="index===0"
class="common-collapse-item" :class="handleCardClass(item)">
<template v-slot:title>
<view class='title-wrapper'>
<view class="top">
<view class="status">{{PCSWorkStatusOptions[item.workStatus] || '暂无数据'}}</view>
<view class="status">{{formatDictValue((PCSWorkStatusOptions || {}), item.workStatus, '暂无数据')}}</view>
<text class="name">{{item.deviceName}}</text>
</view>
</view>
@ -31,25 +55,25 @@
<view class="grid-item-box">
<view class="title">工作状态</view>
<text
class="text work-status-color">{{PCSWorkStatusOptions[item.workStatus] || '-'}}</text>
class="text work-status-color">{{formatDictValue((PCSWorkStatusOptions || {}), item.workStatus)}}</text>
</view>
</view>
<view class="flex-lists">
<view class="grid-item-box">
<view class="title">并网状态</view>
<text class="text">{{gridStatusOptions[item.gridStatus] || '-'}}</text>
<text class="text">{{formatDictValue((gridStatusOptions || {}), item.gridStatus)}}</text>
</view>
</view>
<view class="flex-lists">
<view class="grid-item-box">
<view class="title">设备状态</view>
<text class="text">{{deviceStatusOptions[item.deviceStatus] || '-'}}</text>
<text class="text">{{formatDictValue((deviceStatusOptions || {}), item.deviceStatus)}}</text>
</view>
</view>
<view class="flex-lists">
<view class="grid-item-box">
<view class="title">控制模式</view>
<text class="text">{{controlModeOptions[item.controlMode] || '-'}}</text>
<text class="text">{{formatDictValue((controlModeOptions || {}), item.controlMode)}}</text>
</view>
</view>
</view>
@ -96,25 +120,25 @@
</uni-group>
</view>
</uni-collapse-item>
</uni-collapse>
<view class="no-data" v-else>
暂无数据
</uni-collapse>
<view class="no-data" v-else>
暂无数据
</view>
</view>
</template>
<script>
import {
getRunningHeadInfo,
getPcsDetailInfo
} from '@/api/ems/site.js'
import {
mapState
} from 'vuex'
export default {
computed: {
...mapState({
<script>
import {
getProjectDisplayData,
getPcsNameList
} from '@/api/ems/site.js'
import {
mapState
} from 'vuex'
export default {
computed: {
...mapState({
PCSWorkStatusOptions: (state) =>
state.ems.PCSWorkStatusOptions,
communicationStatusOptions: (state) =>
@ -123,46 +147,31 @@
state.ems.deviceStatusOptions,
gridStatusOptions: (state) =>
state.ems.gridStatusOptions,
controlModeOptions: (state) =>
state.ems.controlModeOptions,
})
},
data() {
return {
runningHeadData: [{
title: '实时有功功率kW',
bgColor: '#FFF2CB',
attr: 'totalActivePower',
img: 'ssyggl'
}, {
title: '实时无功功率kVar',
bgColor: '#CBD6FF',
attr: 'totalReactivePower',
img: 'sswggl'
}, {
title: '电池堆SOC',
bgColor: '#DCCBFF',
attr: 'soc',
img: 'soc'
}, {
title: '电池堆SOH',
bgColor: '#FFD4CB',
attr: 'soh',
img: 'soh'
}, {
title: '今日充电量kWh',
bgColor: '#FFD6F8',
attr: 'dayChargedCap',
img: 'jrcdl'
}, {
title: '今日放电量kWh',
bgColor: '#E1FFCA',
attr: 'dayDisChargedCap',
img: 'jrfdl'
}],
runningHeadInfo: {},
list: [],
siteId: '',
controlModeOptions: (state) =>
state.ems.controlModeOptions,
}),
runningHeadCards() {
const sectionData = (this.runningDisplayData || []).filter(item => item.sectionName === '运行概览')
return sectionData.map((item, index) => ({
title: item.fieldName,
value: item.fieldValue,
img: this.getHeadCardImg(item, index)
}))
},
filteredList() {
if (!this.selectedPcsId) {
return this.list || []
}
return (this.list || []).filter(item => (item.deviceId || '') === this.selectedPcsId)
}
},
data() {
return {
runningDisplayData: [],
pcsDeviceList: [],
selectedPcsId: '',
list: [],
siteId: '',
infoData: [{
label: "总交流有功功率",
attr: "totalActivePower",
@ -257,79 +266,228 @@
},
]
}
},
methods: {
handleCardClass(item) {
const {
workStatus = ''
} = item
return workStatus === '1' || !Object.keys(this.PCSWorkStatusOptions).find(i => i === workStatus) ?
"timing-collapse-item" : workStatus === '2' ? 'warning-collapse-item' : 'running-collapse-item'
},
},
onLoad(options) {
uni.showLoading()
this.siteId = options.siteId || ''
getRunningHeadInfo({
siteId: this.siteId
}).then(response => {
this.runningHeadInfo = response?.data || {}
})
getPcsDetailInfo({
siteId: this.siteId
}).then(response => {
this.list = response?.data || []
if (this.list.length > 0) {
this.$nextTick(() => {
setTimeout(() => {
this.$refs.collapse.resize()
uni.hideLoading()
}, 100)
})
} else {
uni.hideLoading()
}
}).catch(() => {
uni.hideLoading()
})
}
},
methods: {
normalizeDictKey(value) {
const raw = String(value == null ? '' : value).trim()
if (!raw) return ''
if (/^-?\d+(\.0+)?$/.test(raw)) {
return String(parseInt(raw, 10))
}
return raw
},
formatDictValue(options, value, emptyText = '-') {
const dict = (options && typeof options === 'object') ? options : {}
const key = this.normalizeDictKey(value)
if (!key) return emptyText
return dict[key] || key
},
normalizeDeviceId(value) {
return String(value == null ? '' : value).trim().toUpperCase()
},
getModuleRows(menuCode, sectionName) {
return (this.runningDisplayData || []).filter(item => item.menuCode === menuCode && item.sectionName === sectionName)
},
getFieldName(fieldCode) {
if (!fieldCode) {
return ''
}
const index = fieldCode.lastIndexOf('__')
return index >= 0 ? fieldCode.slice(index + 2) : fieldCode
},
getFieldRowMap(rows = [], deviceId = '') {
const map = {}
const targetDeviceId = this.normalizeDeviceId(deviceId || '')
rows.forEach(item => {
if (!item || !item.fieldCode) {
return
}
const itemDeviceId = this.normalizeDeviceId(item.deviceId || '')
if (itemDeviceId !== targetDeviceId) {
return
}
map[this.getFieldName(item.fieldCode)] = item
})
rows.forEach(item => {
if (!item || !item.fieldCode) {
return
}
const itemDeviceId = this.normalizeDeviceId(item.deviceId || '')
if (itemDeviceId !== '') {
return
}
const fieldName = this.getFieldName(item.fieldCode)
if (map[fieldName] === undefined || map[fieldName] === null || map[fieldName] === '') {
map[fieldName] = item
}
})
return map
},
getFieldMap(rows = [], deviceId = '') {
const rowMap = this.getFieldRowMap(rows, deviceId)
return Object.keys(rowMap).reduce((acc, fieldName) => {
const row = rowMap[fieldName] || {}
acc[fieldName] = row.fieldValue
return acc
}, {})
},
getLatestTime(menuCode) {
const times = (this.runningDisplayData || [])
.filter(item => item.menuCode === menuCode && item.valueTime)
.map(item => new Date(item.valueTime).getTime())
.filter(ts => !isNaN(ts))
if (times.length === 0) {
return '-'
}
const date = new Date(Math.max(...times))
const p = (n) => String(n).padStart(2, '0')
return `${date.getFullYear()}-${p(date.getMonth() + 1)}-${p(date.getDate())} ${p(date.getHours())}:${p(date.getMinutes())}:${p(date.getSeconds())}`
},
getPcsDeviceList() {
return getPcsNameList(this.siteId).then((response) => {
this.pcsDeviceList = response?.data || []
}).catch(() => {
this.pcsDeviceList = []
})
},
buildPcsList() {
const devices = this.pcsDeviceList || []
this.list = devices.map((device) => ({
...this.getFieldMap(this.getModuleRows('SBJK_PCS', '电参量'), device.deviceId || device.id || ''),
deviceId: device.deviceId || device.id || '',
deviceName: device.deviceName || device.name || device.deviceId || device.id || 'PCS',
...this.getFieldMap(this.getModuleRows('SBJK_PCS', '状态'), device.deviceId || device.id || ''),
dataUpdateTime: this.getLatestTime('SBJK_PCS'),
alarmNum: 0,
pcsBranchInfoList: [],
}))
},
getHeadCardImg(item, index) {
const imgMap = {
totalActivePower: 'ssyggl',
totalReactivePower: 'sswggl',
soc: 'soc',
soh: 'soh',
dayChargedCap: 'jrcdl',
dayChargedCap_rt: 'jrcdl',
dayDisChargedCap: 'jrfdl',
dayDisChargedCap_rt: 'jrfdl'
}
const defaultImgs = ['ssyggl', 'sswggl', 'soc', 'soh', 'jrcdl', 'jrfdl']
return imgMap[item.fieldCode] || defaultImgs[index % defaultImgs.length]
},
handleCardClass(item) {
const workStatus = this.normalizeDictKey((item && item.workStatus) || '')
const statusOptions = (this.PCSWorkStatusOptions && typeof this.PCSWorkStatusOptions === 'object')
? this.PCSWorkStatusOptions
: {}
const hasStatus = Object.prototype.hasOwnProperty.call(statusOptions, workStatus)
return workStatus === '1' || !hasStatus
? "timing-collapse-item"
: workStatus === '2'
? 'warning-collapse-item'
: 'running-collapse-item'
},
handleTagClick(deviceId) {
this.selectedPcsId = deviceId || ''
},
},
onLoad(options) {
uni.showLoading()
this.siteId = options.siteId || ''
Promise.all([
getProjectDisplayData(this.siteId),
this.getPcsDeviceList()
]).then(([displayResponse]) => {
this.runningDisplayData = displayResponse?.data || []
this.buildPcsList()
if (this.list.length > 0) {
this.$nextTick(() => {
setTimeout(() => {
this.$refs.collapse.resize()
uni.hideLoading()
}, 100)
})
} else {
uni.hideLoading()
}
}).catch(() => {
uni.hideLoading()
})
}
}
</script>
<style lang="scss" scoped>
.info-grid {
background: #fff;
padding: 0 20rpx;
.uni-grid-item {
padding: 20rpx;
.grid-item-box {
box-shadow: 0 0 5px 1px rgba(0, 0, 0, 0.08);
border-radius: 20rpx;
.title {
color: #333;
margin-top: 10rpx;
}
.text {
color: #000;
white-space: nowrap;
text-overflow: ellipsis;
width: 100%;
overflow-x: hidden;
text-align: center;
}
}
}
.icon {
height: 100rpx;
width: 100rpx;
display: block;
border-radius: 20rpx;
}
}
</style>
<style lang="scss" scoped>
.info-overview-scroll {
background: #fff;
padding: 0 20rpx;
white-space: nowrap;
}
.info-overview-row {
display: inline-flex;
gap: 20rpx;
padding: 20rpx 0;
}
.info-overview-card {
width: 280rpx;
flex: 0 0 auto;
}
.grid-item-box {
box-shadow: 0 0 5px 1px rgba(0, 0, 0, 0.08);
border-radius: 20rpx;
padding: 20rpx;
background: #fff;
}
.grid-item-box .title {
color: #333;
margin-top: 10rpx;
}
.grid-item-box .text {
color: #000;
white-space: nowrap;
text-overflow: ellipsis;
width: 100%;
overflow-x: hidden;
text-align: center;
}
.icon {
height: 100rpx;
width: 100rpx;
display: block;
border-radius: 20rpx;
}
.pcs-tags-scroll {
background: #fff;
padding: 0 20rpx 20rpx;
white-space: nowrap;
}
.pcs-tags-row {
display: inline-flex;
gap: 16rpx;
}
.pcs-tag-item {
flex: 0 0 auto;
padding: 8rpx 22rpx;
border-radius: 999rpx;
border: 1px solid #dcdfe6;
color: #606266;
font-size: 24rpx;
line-height: 1.4;
background: #fff;
}
.pcs-tag-item.active {
color: #fff;
background: #007aff;
border-color: #007aff;
}
</style>