Compare commits

..

9 Commits

Author SHA1 Message Date
cacd939dd6 概率统计-电量指标接口联调,站点信息6个方块样式调整 2025-07-01 22:41:45 +08:00
b59b564544 取消单体电池搜索栏的清空按钮 2025-07-01 15:20:51 +08:00
0861c3e15f 搜索栏清空按钮 2025-07-01 15:14:07 +08:00
edab894447 Merge branch 'develop' into baicai 2025-07-01 14:43:33 +08:00
99b019ebc9 故障告警接口联调 2025-07-01 14:42:53 +08:00
9b78635fd8 故障告警 2025-07-01 14:11:41 +08:00
46490aebee 故障告警接口联调 2025-07-01 10:45:13 +08:00
b8779bee61 Merge branch 'develop' into baicai 2025-07-01 09:35:40 +08:00
ca7e9a60b2 故障告警部分功能 2025-06-30 22:44:18 +08:00
6 changed files with 216 additions and 80 deletions

View File

@ -75,3 +75,20 @@ export function getAmmeterDataList(siteId) {
method: 'get' method: 'get'
}) })
} }
// 故障告警
export function getAlarmDetailList({siteId, deviceType, alarmLevel, alarmStartTime, alarmEndTime,pageSize,pageNum}) {
return request({
url: `/ems/siteAlarm/getAlarmDetailList?siteId=${siteId}&deviceType=${deviceType}&alarmLevel=${alarmLevel}&alarmStartTime=${alarmStartTime}&alarmEndTime=${alarmEndTime}&pageSize=${pageSize}&pageNum=${pageNum}`,
method: 'get'
})
}
// 概率统计
//获取概率统计 电量指标接口
export function getElectricData({siteId,startDate,endDate}) {
return request({
url: `/ems/statsReport/getElectricData?siteId=${siteId}&startDate=${startDate}&endDate=${endDate}`,
method: 'get'
})
}

View File

@ -65,11 +65,13 @@ export default {
</script> </script>
<style scoped lang="scss"> <style scoped lang="scss">
.card{ .card{
width: 150px; min-width: 150px;
height: 96px; height: 96px;
margin-right: 27px; margin-right: 27px;
border-bottom: 4px solid transparent; border-bottom: 4px solid transparent;
text-align: center; text-align: center;
box-sizing: border-box;
padding:0 10px;
.info{ .info{
color: #666666; color: #666666;
line-height: 14px; line-height: 14px;

View File

@ -4,3 +4,16 @@ export const formatNumber = (val) => {
return val return val
}else {return '-'} }else {return '-'}
} }
export const formatDate = (val,toSeconds = false) => {
if(!val) return ''
const date = new Date(val)
const month = date.getMonth() + 1,day = date.getDate()
if(!toSeconds){
return `${date.getFullYear()}-${month<10?'0'+month : month}-${day<10 ? '0'+day : day}`
}else{
const hours = date.getHours(),minuets=date.getMinutes(),seconds = date.getSeconds();
return `${date.getFullYear()}-${month<10?'0'+month : month}-${day<10 ? '0'+day : day} ${hours<10 ? '0'+hours : hours}:${minuets<10 ? '0'+minuets : minuets}:${seconds<10 ? '0'+seconds : seconds}`
}
}

View File

@ -8,6 +8,9 @@ const ems = {
warnOptions:{0:'正常', 1:'中断', 2:'不在线',3:'异常'},//告警状态 warnOptions:{0:'正常', 1:'中断', 2:'不在线',3:'异常'},//告警状态
communicationStatusOptions:{'0':'正常','1':'通讯中断','2':'异常'},//通讯状态 communicationStatusOptions:{'0':'正常','1':'通讯中断','2':'异常'},//通讯状态
workModeOptions:{'0':'正常','1':'停止'},//工作模式 workModeOptions:{'0':'正常','1':'停止'},//工作模式
alarmLevelOptions:{'A':'提示','B':'一般','C':'严重','D':'紧急'},//告警等级
alarmStatusOptions:{'0':'待处理','1':'已处理','2':'处理中'},//告警状态
deviceTypeOptions:{'TCP':'TCP','RTU':'RTU'},//设备类型
}, },
mutations: { mutations: {
SET_ZD_LIST(state, list) { SET_ZD_LIST(state, list) {

View File

@ -1,16 +1,16 @@
<template> <template>
<el-card shadow="always" class="common-card-container common-card-container-no-title-bg"> <el-card v-loading="loading" gshadow="always" class="common-card-container common-card-container-no-title-bg">
<!-- 搜索栏--> <!-- 搜索栏-->
<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.sblx" placeholder="请选择" :loading="loading" loading-text="正在加载数据"> <el-select v-model="search.deviceType" clearable placeholder="请选择" :loading="loading" loading-text="正在加载数据">
<el-option :label="item.name" :value="item.id" v-for="(item,index) in sblxOptions" :key="index+'sblxOptions'"></el-option> <el-option :label="value" :value="key" v-for="(value,key) in $store.state.ems.deviceTypeOptions" :key="key+'deviceTypeOptions'"></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.gjdj" placeholder="请选择" :loading="loading" loading-text="正在加载数据"> <el-select v-model="search.alarmLevel" clearable placeholder="请选择" :loading="loading" loading-text="正在加载数据">
<el-option :label="item.name" :value="item.id" v-for="(item,index) in gjdjOptions" :key="index+'gjdjOptions'"></el-option> <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="时间选择">
@ -48,7 +48,7 @@
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="name" prop="deviceName"
width="200" width="200"
label="设备名称"> label="设备名称">
</el-table-column> </el-table-column>
@ -56,43 +56,66 @@
label="告警等级" label="告警等级"
> >
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{scope.row.gjdj}}</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"
show-overflow-tooltip
label="告警内容"> label="告警内容">
</el-table-column>
<el-table-column
prop="alarmStartTime"
label="告警发生时间">
<template slot-scope="scope"> <template slot-scope="scope">
<span>{{scope.row.gjnr}}</span> <span>{{formatDate(scope.row.alarmStartTime,true)}}</span>
</template> </template>
</el-table-column> </el-table-column>
<el-table-column <el-table-column
prop="startTime" prop="alarmEndTime"
label="告警发生时间">
</el-table-column>
<el-table-column
prop="finishTime"
label="告警结束时间"> label="告警结束时间">
<template slot-scope="scope">
<span>{{formatDate(scope.row.alarmEndTime,true)}}</span>
</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="scope.row.status === 2 ? 'warning-status' : ''">{{scope.row.status === 2 ? '告警中' :'已结束'}}</span> <span :class="['0','2'].includes(scope.row.status) ? 'warning-status' : ''">{{$store.state.ems.alarmStatusOptions[scope.row.status]}}</span>
</template> </template>
</el-table-column> </el-table-column>
</el-table> </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> </div>
</el-card> </el-card>
</template> </template>
<script> <script>
import {getAlarmDetailList} from'@/api/ems/dzjk'
import getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import { formatDate } from '@/filters/ems'
export default { export default {
name:'DzjkGzgj', name:'DzjkGzgj',
mixins:[getQuerySiteId],
data() { data() {
return { return {
loading:false,
btnList:[ btnList:[
{name:'今日告警',id:'today'}, {name:'今日告警',id:'today'},
{name:'历史告警',id:'yesterday'}, {name:'历史告警',id:'history'},
], ],
pickerOptions:{ pickerOptions:{
disabledDate(time) { disabledDate(time) {
@ -100,56 +123,113 @@ export default {
}, },
}, },
defaultDateRange:[],//默认展示的时间 defaultDateRange:[],//默认展示的时间
dateRange:[], dateRange:[],//alarmStartTime,alarmEndTime
activeBtn:'today', activeBtn:'today',
loading:false, search:{deviceType:'',alarmLevel:''},
search:{sblx:'',gjdj:''}, // 表格、分页
sblxOptions:[ tableData:[],
{name:'类型一',id:1}, pageSize:10,//分页栏当前每个数据总数
{name:'类型二',id:2}, pageNum:1,//分页栏当前页数
{name:'类型三',id:3}, totalSize:0,//table表格数据总数
],
gjdjOptions:[
{name:'等级一',id:1},
{name:'等级二',id:2},
],
tableData:[
{name:'告警一',gjdj:'等级一',gjnr:'设备一温度过高',startTime:'2025/06/20 18:09:40',finishTime:'2025/06/20 20:07:01',status:1},
{name:'告警二',gjdj:'等级一',gjnr:'设备二温度过高',startTime:'2025/06/20 18:09:40',finishTime:'2025/06/20 20:07:01',status:2},
{name:'告警三',gjdj:'等级一',gjnr:'设备一温度过高',startTime:'2025/06/20 18:09:40',finishTime:'2025/06/20 20:07:01',status:1},
{name:'告警四',gjdj:'等级一',gjnr:'设备二温度过高',startTime:'2025/06/20 18:09:40',finishTime:'2025/06/20 20:07:01',status:2}
]
} }
}, },
methods:{ methods:{
formatDate,
// 判断是否是同一天
isSameDay(day1, day2) {
const date1 = new Date(day1),date2 = new Date(day2)
return date1.getFullYear() === date2.getFullYear() &&
date1.getMonth() === date2.getMonth() &&
date1.getDate() === date2.getDate();
},
// 分页
handleSizeChange(val) {
this.pageSize = val;
this.$nextTick(()=>{
this.getTableData()
})
},
handleCurrentChange(val) {
this.pageNum = val
this.$nextTick(()=>{
this.getTableData()
})
},
// 搜索 // 搜索
onSearch(){ onSearch(){
const [alarmStartTime='',alarmEndTime='']=(this.dateRange || [])
// 选中了时间范围
if(alarmStartTime && alarmStartTime){
// 如果选择的时间范围是今天
if(this.isSameDay(alarmStartTime,alarmEndTime) && this.isSameDay(alarmStartTime,new Date())){
this.activeBtn = 'today'
}else {
this.activeBtn = 'history'
}
}else{
//没有选择时间范围 还是按照选中的今日告警、历史告警查询
}
this.getData() this.getData()
}, },
// 重置 // 重置
onReset(){ onReset(){
this.search.sblx='' this.search={deviceType:'',alarmLevel:''}
this.search.gjdj=''
this.dateRange=[] this.dateRange=[]
this.getData() this.getData()
}, },
// 获取数据 // 切换今日、历史告警
getData(){
this.loading=false
},
changeDataType(id){ changeDataType(id){
if(id !== this.activeBtn){ if(id !== this.activeBtn){
console.log('点击了不同的菜单,更新数据') console.log('点击了不同的菜单,更新数据')
this.activeBtn=id; this.activeBtn=id;
const [alarmStartTime,alarmEndTime]=(this.dateRange || [])
// 切换到今日告警,如果已经选择了时间范围清空
if(alarmStartTime && alarmEndTime){
// 如果切换到了今日告警,时间范围不相等或者相等但是不是今天 清空时间选择范围
if(id === 'today' && !this.isSameDay(alarmStartTime,alarmEndTime) || (this.isSameDay(alarmStartTime,alarmEndTime) && !this.isSameDay(alarmStartTime,new Date()))){
this.dateRange = []
}else if(id === 'history' && this.isSameDay(alarmStartTime,alarmEndTime) && this.isSameDay(alarmStartTime,new Date())){
// 切换成历史告警,但是选择时间范围是当天,清空时间范围
this.dateRange = []
}
}
this.getData()
} }
}, },
// 获取数据
getData(){
this.loading=true
const {deviceType,alarmLevel} = this.search
const {siteId,pageNum,pageSize,activeBtn} =this
const [alarmStartTime='',alarmEndTime='']=(this.dateRange || [])
let start='',end = '',now =new Date()
if(activeBtn === 'today'){
start = end = now
}else{
if(alarmStartTime && alarmEndTime){
start = alarmStartTime
end = alarmEndTime
}else{
start=''
end = ''
// now
// end.setDate(end.getDate() - 1);
}
}
getAlarmDetailList({deviceType,alarmLevel,siteId,pageSize,pageNum,alarmStartTime:formatDate(start),alarmEndTime:formatDate(end)}).then(response => {
this.tableData=response?.rows || [];
this.totalSize = response?.total || 0
}).finally(() => {this.loading=false})
},
init(){
this.onReset()
},
}, },
mounted(){ mounted(){
this.getData()
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];
} }
} }
</script> </script>

View File

@ -3,7 +3,7 @@
<div slot="header"> <div slot="header">
<span class="card-title">电量指标</span> <span class="card-title">电量指标</span>
</div> </div>
<div class="card-main"> <div class="card-main" v-loading="loading">
<!-- 搜索栏--> <!-- 搜索栏-->
<div class="select-container"> <div class="select-container">
<el-form :inline="true"> <el-form :inline="true">
@ -19,7 +19,7 @@
</el-date-picker> </el-date-picker>
</el-form-item> </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="getData" native-type="button">搜索</el-button>
</el-form-item> </el-form-item>
<el-form-item> <el-form-item>
<el-button @click="onReset" native-type="button">重置</el-button> <el-button @click="onReset" native-type="button">重置</el-button>
@ -27,9 +27,9 @@
</el-form> </el-form>
</div> </div>
<div class="total-data"> <div class="total-data">
<div>总充电量:<span class="point">1390988kWh</span></div> <div>总充电量:<span class="point">{{totalChargedCap | formatNumber}}kWh</span></div>
<div>总放电量:<span class="point">988kWh</span></div> <div>总放电量:<span class="point">{{totalDisChargedCap | formatNumber}}kWh</span></div>
<div>综合效率:<span class="point">99%</span></div> <div>综合效率:<span class="point">{{efficiency | formatNumber}}%</span></div>
</div> </div>
<div id="dlzbChart" style="height: 310px"></div> <div id="dlzbChart" style="height: 310px"></div>
</div> </div>
@ -39,8 +39,11 @@
<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 getQuerySiteId from "@/mixins/ems/getQuerySiteId";
import {getElectricData} from '@/api/ems/dzjk'
import {formatDate} from '@/filters/ems'
export default { export default {
mixins: [resize], mixins: [resize,getQuerySiteId],
data() { data() {
return { return {
pickerOptions:{ pickerOptions:{
@ -51,74 +54,92 @@ export default {
defaultDateRange:[],//默认展示的时间 defaultDateRange:[],//默认展示的时间
dateRange:[], dateRange:[],
loading:false, loading:false,
chart: null chart: null,
totalChargedCap:'',
totalDisChargedCap:'',
efficiency:''
} }
}, },
methods: { methods: {
// 搜索
onSearch(){
this.getData()
},
// 重置 // 重置
onReset(){ onReset(){
this.dateRange=[] this.dateRange=[]
this.getData() this.getData()
}, },
getData(){
this.loading=true;
this.chart.showLoading()
//接口调用完成之后 设置图表、结束loading
this.setOption()
this.loading=false;
this.chart.hideLoading()
},
setOption(data){ setOption(data){
// 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,item.dailyEfficiency])
// }) })
this.chart.setOption({ this.chart.setOption({
color:['#FFBD00','#3C81FF'], color:['#FFBD00','#3C81FF','#05AEA3'],
// legend: { // legend: {
// left: 'right', // left: 'right',
// bottom: '10', // bottom: '10',
// }, // },
tooltip: {}, tooltip: {},
xAxis: { type: 'category' }, xAxis: { type: 'category' },
yAxis: { type: 'value' }, yAxis: [{
type: 'value',
axisLine: {
lineStyle:{
color: '#333333',
},
onZero:false
}
},{
type: 'value',
axisLine: {
lineStyle:{
color: '#333333',
},
onZero:false
}
}],
grid:{top:30}, grid:{top:30},
dataset:{ dataset:{
// source source
source:[//格式
['product','充电量','放电量'],
['6/10',101,220],
['6/11',210,390],
['6/12',220,360],
['6/13',520,340],
['6/14',270,430],
]
}, },
series: [ series: [
{ {
yAxisIndex:0,
type: 'bar',//柱状图 type: 'bar',//柱状图
}, },
{ {
yAxisIndex:0,
type: 'bar',//柱状图 type: 'bar',//柱状图
}, },
{
yAxisIndex:1,
type: 'line',//柱状图
},
] ]
}) })
}, },
initChart() { getData(){
this.loading=true;
//接口调用完成之后 设置图表、结束loading
const [start='',end='']=this.dateRange || [];
const startDate=formatDate(start),endDate = formatDate(end)
getElectricData({siteId:this.siteId,startDate,endDate}).then(response => {
const {totalChargedCap='',totalDisChargedCap='',efficiency='',sevenDayDisChargeStats=[]}=response?.data || {}
this.setOption(sevenDayDisChargeStats)
this.totalChargedCap=totalChargedCap
this.totalDisChargedCap=totalDisChargedCap
this.efficiency=efficiency
}).finally(() => {
this.loading=false;
})
},
init(){
this.chart = echarts.init(document.querySelector('#dlzbChart')); this.chart = echarts.init(document.querySelector('#dlzbChart'));
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];
this.initChart()
this.getData()
}, },
beforeDestroy() { beforeDestroy() {
if (!this.chart) { if (!this.chart) {