污染值报表

This commit is contained in:
sjl
2025-11-12 13:23:15 +08:00
parent c9a47917e4
commit d6336f7ba8
5 changed files with 871 additions and 496 deletions

View File

@@ -35,21 +35,21 @@ const setupAll = async () => {
const app = createApp(App)
//开启离线地图
app.use(BaiduMapOffline, {
offline: true,
offlineConfig: {
imgext: '.png',
customstyle: '',
tiles_dir: '',
tiles_hybrid: '',
tiles_self: '',
tiles_v_dir: '',
tiles_satellite_dir: '',
tiles_road_dir: '',
tiles_v_road_dir: '',
home: './plugin/offline/'
}
})
// app.use(BaiduMapOffline, {
// offline: true,
// offlineConfig: {
// imgext: '.png',
// customstyle: '',
// tiles_dir: '',
// tiles_hybrid: '',
// tiles_self: '',
// tiles_v_dir: '',
// tiles_satellite_dir: '',
// tiles_road_dir: '',
// tiles_v_road_dir: '',
// home: './plugin/offline/'
// }
// })
app.use(BaiduMap, {
// ak: 'Yp57V71dkOPiXjiN8VdcFRsVELzlVNKK',
ak: 'RpQi6WNFZ9tseKzhdwOQsXwFsoVntnsN',

View File

@@ -0,0 +1,390 @@
<template>
<div class="tab-content">
<TableHeader datePicker ref="tableHeaderRef" @selectChange="handleSelectChange">
<template #select>
<el-form-item label="区域">
<el-cascader
v-bind="$attrs"
:options="areOptions"
:props="cascaderProps"
v-model="selectedArea"
@change="handleFilterChange"
/>
</el-form-item>
<el-form-item label="统计类型">
<el-select
v-model="tableStore.table.params.statisticalType"
value-key="id"
placeholder="请选择统计类型"
@change="handleStatisticalTypeChange"
>
<el-option v-for="item in options" :key="item.id" :label="item.name" :value="item" />
</el-select>
</el-form-item>
<el-form-item label="电网标志">
<el-select
v-model="tableStore.table.params.powerFlag"
placeholder="请选择电网标志"
@change="handleFilterChange"
>
<el-option label="全部" value="0"></el-option>
<el-option label="电网侧" value="1"></el-option>
<el-option label="非电网侧" value="2"></el-option>
</el-select>
</el-form-item>
<el-form-item label="筛选">
<el-input
v-model="searchKeyword"
placeholder="请输入变电站/终端/监测点"
clearable
@input="handleFilterChange"
:show-word-limit=true
:maxlength="32"
/>
</el-form-item>
</template>
<template #operation>
<el-button icon="el-icon-Download" type="primary" @click="exportEvent">导出</el-button>
</template>
</TableHeader>
<div v-loading="tableStore.table.loading" class="main-container">
<vxe-table
class="full-height-table"
ref="tableRef"
auto-resize
:data="tableStore.table.data"
v-bind="defaultAttribute"
:height="tableHeight"
resizable
show-overflow
>
<vxe-column title="序号" width="80" type="seq" align="center"></vxe-column>
<vxe-column field="gdName" title="供电公司" align="center" min-width="120"></vxe-column>
<vxe-column field="subStationName" :show-overflow="true" title="变电站" align="center" min-width="150"></vxe-column>
<vxe-column field="devName" title="终端名称" align="center" min-width="120"></vxe-column>
<vxe-column field="devType" title="终端型号" align="center" min-width="150"></vxe-column>
<vxe-column field="loginTime" title="投运时间" align="center" min-width="120"></vxe-column>
<vxe-column field="lineName" title="监测点名称" align="center" min-width="150"></vxe-column>
<vxe-column field="powerFlag" title="监测位置" align="center" min-width="100"></vxe-column>
<vxe-column field="lineVoltage" title="监测点电压等级" align="center" min-width="120"></vxe-column>
<vxe-column field="loadType" title="干扰源类型" align="center" min-width="120"></vxe-column>
<vxe-column field="objName" title="监测对象名称" align="center" min-width="150"></vxe-column>
<vxe-column field="interval" title="统计间隔" align="center" min-width="100"></vxe-column>
<vxe-column field="onlineRate" title="在线率(%)" align="center" min-width="100"></vxe-column>
<vxe-column field="integrity" title="完整率(%)" align="center" min-width="100"></vxe-column>
<vxe-column field="harmonicValue" :title="harmonicValueTitle" align="center" min-width="120"></vxe-column>
<vxe-column field="upCounts" title="暂升次数(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="downCounts" title="电压暂降(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="breakCounts" title="短时中断(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="monitorId" title="一类监测点" align="center" min-width="120" :formatter="formatMonitorId"></vxe-column>
</vxe-table>
</div>
<div class="pagination-container">
<el-pagination
:current-page="tableStore.table.params.pageNum"
:page-size="tableStore.table.params.pageSize"
:page-sizes="[10, 20, 50, 100]"
background
layout="sizes, total, prev, pager, next, jumper"
:total="tableStore.table.total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
class="custom-pagination"/>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, provide, onBeforeUnmount, computed, watch } from 'vue'
import TableStore from '@/utils/tableStore'
import TableHeader from '@/components/table/header/index.vue'
import { defaultAttribute } from '@/components/table/defaultAttribute'
import { useDictData } from '@/stores/dictData'
import { debounce } from 'lodash-es'
const props = defineProps({
active: {
type: Boolean,
default: false
}
})
const emit = defineEmits(['initialized'])
const dictData = useDictData()
const options = dictData.getBasicData('Pollution_Calc')
const tableHeight = ref(500)
const tableRef = ref()
const tableHeaderRef = ref()
const selectedArea = ref()
const areOptions: any = dictData.state.area
const allData = ref<PollutionItem[]>([])
const searchKeyword = ref('')
const harmonicValueTitle = ref('谐波电压污染值')
const cascaderProps = {
label: 'name',
value: 'id',
checkStrictly: true,
emitPath: false
}
interface PollutionItem {
gdName?: string
subStationName?: string
devName?: string
lineName?: string
powerFlag?: string
}
const formatMonitorId = (row: any) => {
return row.row.monitorId || '/'
}
const handleSelectChange = (isExpanded: boolean) => {
if (isExpanded) {
tableHeight.value = tableHeight.value - 55
} else {
tableHeight.value = tableHeight.value + 55
}
}
const calculateTableHeight = () => {
const windowHeight = window.innerHeight
const headerHeight = 160
const paginationHeight = 120
const padding = 30
const calculatedHeight = windowHeight - headerHeight - paginationHeight - padding
tableHeight.value = Math.max(calculatedHeight, 300)
}
const debouncedCalculateTableHeight = debounce(() => {
calculateTableHeight()
}, 300)
const filteredData = computed(() => {
let result = [...allData.value]
if (selectedArea.value) {
let areaName = ''
let areaLevel = -1
const findAreaName = (areas: any[]) => {
for (const area of areas) {
if (area.id === selectedArea.value) {
areaName = area.name
areaLevel = area.level !== undefined ? area.level : -1
break
}
if (area.children && area.children.length > 0) {
findAreaName(area.children)
}
}
}
findAreaName(areOptions)
if (areaName && areaLevel > 1) {
result = result.filter(item => item.gdName && item.gdName.includes(areaName))
}
}
if (tableStore.table.params.powerFlag === '1') {
result = result.filter(item => item.powerFlag && !item.powerFlag.includes('非'))
} else if (tableStore.table.params.powerFlag === '2') {
result = result.filter(item => item.powerFlag && item.powerFlag.includes('非'))
}
if (searchKeyword.value) {
const keyword = searchKeyword.value.toLowerCase()
result = result.filter(item =>
(item.subStationName && item.subStationName.toLowerCase().includes(keyword)) ||
(item.devName && item.devName.toLowerCase().includes(keyword)) ||
(item.lineName && item.lineName.toLowerCase().includes(keyword))
)
}
return result
})
const currentPageData = computed(() => {
const pageSize = tableStore.table.params.pageSize
const pageNum = tableStore.table.params.pageNum
const start = (pageNum - 1) * pageSize
const end = start + pageSize
return filteredData.value.slice(start, end)
})
const updateTotal = computed(() => {
return filteredData.value.length
})
const tableStore = new TableStore({
url: '/harmonic-boot/PollutionSubstation/downPollutionLineCalc',
method: 'POST',
column: [],
beforeSearchFun: () => {
delete tableStore.table.params.deptIndex
delete tableStore.table.params.interval
delete tableStore.table.params.searchEndTime
delete tableStore.table.params.searchBeginTime
delete tableStore.table.params.timeFlag
},
loadCallback: () => {
allData.value = tableStore.table.data || []
tableStore.table.total = updateTotal.value
tableStore.table.data = currentPageData.value
}
})
provide('tableStore', tableStore)
watch(
() => areOptions,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
selectedArea.value = newOptions[0].id
tableStore.table.params.id = newOptions[0].id
}
},
{ immediate: true }
)
watch(
() => options,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
if (!tableStore.table.params.statisticalType) {
tableStore.table.params.statisticalType = newOptions[0]
tableStore.table.params.ids = [newOptions[0].id]
}
}
},
{ immediate: true }
)
tableStore.table.params.powerFlag = "0"
tableStore.table.params.isUpToGrid = 0
tableStore.table.params.type = 1
const handleStatisticalTypeChange = (newVal: { id: any }) => {
if (newVal) {
tableStore.table.params.statisticalType = newVal
tableStore.table.params.ids = [newVal.id]
if (newVal.name) {
harmonicValueTitle.value = newVal.name + '污染值'
}
}
// 重新调用接口
tableStore.index()
}
const handleFilterChange = () => {
tableStore.table.params.pageNum = 1
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const handleSizeChange = (val: number) => {
tableStore.table.params.pageSize = val
tableStore.table.params.pageNum = 1
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const handleCurrentChange = (val: number) => {
tableStore.table.params.pageNum = val
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const exportEvent = () => {
const allFilteredData = filteredData.value
tableRef.value.exportData({
filename: '污染值报告',
sheetName: 'Sheet1',
type: 'xlsx',
useStyle: true,
data: allFilteredData,
columnFilterMethod: function (column, $columnIndex) {
return !(column.$columnIndex === 0)
}
})
}
onMounted(() => {
if (props.active) {
tableStore.index()
emit('initialized', 'pollutionReport')
}
calculateTableHeight()
window.addEventListener('resize', debouncedCalculateTableHeight)
})
onBeforeUnmount(() => {
window.removeEventListener('resize', debouncedCalculateTableHeight)
})
defineExpose({
refresh: () => {
tableStore.index()
}
})
</script>
<style scoped>
.pagination-container {
border: 1px solid #ebeef5;
border-top: none;
padding: 10px;
margin: 0 0 5px 0;
background-color: #fff;
}
:deep(.custom-pagination) {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
}
:deep(.custom-pagination .el-pagination__sizes) {
margin-right: 10px;
}
:deep(.custom-pagination .el-pagination__total) {
margin-right: auto;
}
:deep(.custom-pagination .el-pagination__jump) {
margin-left: 20px;
}
/* 响应式处理 */
@media screen and (max-width: 768px) {
:deep(.custom-pagination) {
flex-direction: column;
align-items: stretch;
}
:deep(.custom-pagination .el-pagination__total) {
margin-right: 0;
order: 1;
margin-top: 10px;
}
:deep(.custom-pagination .el-pagination__sizes) {
order: 2;
}
:deep(.custom-pagination .btn-prev),
:deep(.custom-pagination .el-pager),
:deep(.custom-pagination .btn-next),
:deep(.custom-pagination .el-pagination__jump) {
order: 3;
justify-content: center;
margin-top: 10px;
}
}
</style>

View File

@@ -0,0 +1,428 @@
<template>
<div class="tab-content">
<TableHeader datePicker ref="tableHeaderRef" @selectChange="handleSelectChange">
<template #select>
<el-form-item label="区域">
<el-cascader
v-bind="$attrs"
:options="areOptions"
:props="cascaderProps"
v-model="selectedArea"
@change="handleFilterChange"
/>
</el-form-item>
<el-form-item label="统计类型">
<el-select
v-model="tableStore.table.params.statisticalType"
value-key="id"
placeholder="请选择统计类型"
@change="handleStatisticalTypeChange"
>
<el-option v-for="item in options" :key="item.id" :label="item.name" :value="item" />
</el-select>
</el-form-item>
<el-form-item label="电网标志">
<el-select
v-model="tableStore.table.params.powerFlag"
placeholder="请选择电网标志"
@change="handleFilterChange"
>
<el-option label="全部" value="0"></el-option>
<el-option label="电网侧" value="1"></el-option>
<el-option label="非电网侧" value="2"></el-option>
</el-select>
</el-form-item>
<el-form-item label="筛选">
<el-input
v-model="searchKeyword"
placeholder="请输入变电站/终端/监测点"
clearable
@input="handleFilterChange"
:show-word-limit=true
:maxlength="32"
/>
</el-form-item>
</template>
<template #operation>
<el-button icon="el-icon-Download" type="primary" @click="exportEvent">导出</el-button>
</template>
</TableHeader>
<div v-loading="tableStore.table.loading" class="main-container">
<vxe-table
class="full-height-table"
ref="tableRef"
auto-resize
:data="tableStore.table.data"
v-bind="defaultAttribute"
:height="tableHeight"
resizable
show-overflow
>
<vxe-column title="序号" width="80" type="seq" align="center"></vxe-column>
<vxe-column field="gdName" title="供电公司" align="center" min-width="120" ></vxe-column>
<vxe-column field="subStationName" :show-overflow="true" title="变电站" align="center" min-width="150"></vxe-column>
<vxe-column field="subVStationValue" :title="harmonicValueTitle" align="center" min-width="150"></vxe-column>
<vxe-column field="subVoltage" title="变电站电压等级" align="center" min-width="150"></vxe-column>
<vxe-column field="lineName" title="监测点名称" align="center" min-width="150"></vxe-column>
<vxe-column field="devName" title="终端名称" align="center" min-width="150"></vxe-column>
<vxe-column field="devType" title="终端型号" align="center" min-width="120"></vxe-column>
<vxe-column field="manufacturer" title="终端厂家" align="center" min-width="150"></vxe-column>
<vxe-column field="loginTime" title="投运时间" align="center" min-width="150"></vxe-column>
<vxe-column field="interval" title="统计间隔" align="center" min-width="150"></vxe-column>
<vxe-column field="powerFlag" title="监测位置" align="center" min-width="150"></vxe-column>
<vxe-column field="vharmonicValue" title="监测点污染值" align="center" min-width="150"></vxe-column>
</vxe-table>
</div>
<div class="pagination-container">
<el-pagination
:current-page="tableStore.table.params.pageNum"
:page-size="tableStore.table.params.pageSize"
:page-sizes="[10, 20, 50, 100]"
background
layout="sizes, total, prev, pager, next, jumper"
:total="tableStore.table.total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
class="custom-pagination"/>
</div>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, provide, onBeforeUnmount, computed, watch } from 'vue'
import TableStore from '@/utils/tableStore'
import TableHeader from '@/components/table/header/index.vue'
import { defaultAttribute } from '@/components/table/defaultAttribute'
import { useDictData } from '@/stores/dictData'
import { debounce } from 'lodash-es'
const props = defineProps({
active: {
type: Boolean,
default: false
}
})
const emit = defineEmits(['initialized'])
const dictData = useDictData()
const options = dictData.getBasicData('Pollution_Calc')
const tableHeight = ref(500)
const tableRef = ref()
const tableHeaderRef = ref()
const selectedArea = ref()
const areOptions: any = dictData.state.area
const allData = ref<PollutionItem[]>([])
const searchKeyword = ref('')
const harmonicValueTitle = ref('电站谐波电压污染值')
const cascaderProps = {
label: 'name',
value: 'id',
checkStrictly: true,
emitPath: false
}
interface PollutionItem {
gdName?: string
subStationName?: string
subVStationValue?: string
subVoltage?: string
devName?: string
lineName?: string
powerFlag?: string
devType?: string
manufacturer?: string
loginTime?: string
interval?: string
vharmonicValue?: string
}
const handleSelectChange = (isExpanded: boolean) => {
if (isExpanded) {
tableHeight.value = tableHeight.value - 55
} else {
tableHeight.value = tableHeight.value + 55
}
}
const calculateTableHeight = () => {
const windowHeight = window.innerHeight
const headerHeight = 160
const paginationHeight = 120
const padding = 30
const calculatedHeight = windowHeight - headerHeight - paginationHeight - padding
tableHeight.value = Math.max(calculatedHeight, 300)
}
const debouncedCalculateTableHeight = debounce(() => {
calculateTableHeight()
}, 300)
const filteredData = computed(() => {
let result = [...allData.value]
if (selectedArea.value) {
let areaName = ''
let areaLevel = -1
const findAreaName = (areas: any[]) => {
for (const area of areas) {
if (area.id === selectedArea.value) {
areaName = area.name
areaLevel = area.level !== undefined ? area.level : -1
break
}
if (area.children && area.children.length > 0) {
findAreaName(area.children)
}
}
}
findAreaName(areOptions)
if (areaName && areaLevel > 1) {
result = result.filter(item => item.gdName && item.gdName.includes(areaName))
}
}
if (tableStore.table.params.powerFlag === '1') {
result = result.filter(item => item.powerFlag && !item.powerFlag.includes('非'))
} else if (tableStore.table.params.powerFlag === '2') {
result = result.filter(item => item.powerFlag && item.powerFlag.includes('非'))
}
if (searchKeyword.value) {
const keyword = searchKeyword.value.toLowerCase()
result = result.filter(item =>
(item.subStationName && item.subStationName.toLowerCase().includes(keyword)) ||
(item.devName && item.devName.toLowerCase().includes(keyword)) ||
(item.lineName && item.lineName.toLowerCase().includes(keyword))
)
}
return result
})
const currentPageData = computed(() => {
const pageSize = tableStore.table.params.pageSize
const pageNum = tableStore.table.params.pageNum
const start = (pageNum - 1) * pageSize
const end = start + pageSize
return filteredData.value.slice(start, end)
})
const updateTotal = computed(() => {
return filteredData.value.length
})
const processNullValue = (value: any) => {
return value === null || value === undefined || value === '' || value === 'null'? '/' : value
}
const tableStore = new TableStore({
url: '/harmonic-boot/PollutionSubstation/downPollutionSubCalc',
method: 'POST',
column: [],
beforeSearchFun: () => {
delete tableStore.table.params.deptIndex
delete tableStore.table.params.interval
delete tableStore.table.params.searchEndTime
delete tableStore.table.params.searchBeginTime
delete tableStore.table.params.timeFlag
},
loadCallback: () => {
// 展开数据
const expandedData: PollutionItem[] = [];
(tableStore.table.data || []).forEach((item: any) => {
if (item.powerFlagPollutionList && item.powerFlagPollutionList.length > 0) {
// 为每个监测点创建一行数据
item.powerFlagPollutionList.forEach((point: any) => {
expandedData.push({
gdName: processNullValue(item.gdName),
subStationName: processNullValue(item.subStationName),
subVStationValue: processNullValue(item.subVStationValue),
subVoltage: processNullValue(item.subVoltage),
devName: processNullValue(point.devName),
lineName: processNullValue(point.lineName),
powerFlag: processNullValue(point.powerFlag),
devType: processNullValue(point.devType),
manufacturer: processNullValue(point.manufacturer),
loginTime: processNullValue(point.loginTime),
interval: processNullValue(point.interval),
vharmonicValue: processNullValue(point.vharmonicValue)
});
});
} else {
// 如果没有监测点数据,也保留变电站基本信息行
expandedData.push({
gdName: processNullValue(item.gdName),
subStationName: processNullValue(item.subStationName),
subVStationValue: processNullValue(item.subVStationValue),
subVoltage: processNullValue(item.subVoltage)
});
}
});
allData.value = expandedData;
tableStore.table.total = updateTotal.value;
tableStore.table.data = currentPageData.value;
}
})
provide('tableStore', tableStore)
watch(
() => areOptions,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
selectedArea.value = newOptions[0].id
tableStore.table.params.id = newOptions[0].id
}
},
{ immediate: true }
)
watch(
() => options,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
if (!tableStore.table.params.statisticalType) {
tableStore.table.params.statisticalType = newOptions[0]
tableStore.table.params.ids = [newOptions[0].id]
}
}
},
{ immediate: true }
)
tableStore.table.params.powerFlag = "0"
tableStore.table.params.isUpToGrid = 0
tableStore.table.params.type = 1
const handleStatisticalTypeChange = (newVal: { id: any }) => {
if (newVal) {
tableStore.table.params.statisticalType = newVal
tableStore.table.params.ids = [newVal.id]
if (newVal.name) {
harmonicValueTitle.value = '电站' + newVal.name + '污染值'
}
}
// 重新调用接口
tableStore.index()
}
const handleFilterChange = () => {
tableStore.table.params.pageNum = 1
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const handleSizeChange = (val: number) => {
tableStore.table.params.pageSize = val
tableStore.table.params.pageNum = 1
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const handleCurrentChange = (val: number) => {
tableStore.table.params.pageNum = val
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const exportEvent = () => {
const allFilteredData = filteredData.value
tableRef.value.exportData({
filename: '污染值报告',
sheetName: 'Sheet1',
type: 'xlsx',
useStyle: true,
data: allFilteredData,
columnFilterMethod: function (column, $columnIndex) {
return !(column.$columnIndex === 0)
}
})
}
onMounted(() => {
if (props.active) {
tableStore.index()
emit('initialized', 'pollutionReport')
}
calculateTableHeight()
window.addEventListener('resize', debouncedCalculateTableHeight)
})
onBeforeUnmount(() => {
window.removeEventListener('resize', debouncedCalculateTableHeight)
})
defineExpose({
refresh: () => {
tableStore.index()
}
})
</script>
<style scoped>
.pagination-container {
border: 1px solid #ebeef5;
border-top: none;
padding: 10px;
margin: 0 0 5px 0;
background-color: #fff;
}
:deep(.custom-pagination) {
display: flex;
justify-content: space-between;
align-items: center;
flex-wrap: wrap;
}
:deep(.custom-pagination .el-pagination__sizes) {
margin-right: 10px;
}
:deep(.custom-pagination .el-pagination__total) {
margin-right: auto;
}
:deep(.custom-pagination .el-pagination__jump) {
margin-left: 20px;
}
/* 响应式处理 */
@media screen and (max-width: 768px) {
:deep(.custom-pagination) {
flex-direction: column;
align-items: stretch;
}
:deep(.custom-pagination .el-pagination__total) {
margin-right: 0;
order: 1;
margin-top: 10px;
}
:deep(.custom-pagination .el-pagination__sizes) {
order: 2;
}
:deep(.custom-pagination .btn-prev),
:deep(.custom-pagination .el-pager),
:deep(.custom-pagination .btn-next),
:deep(.custom-pagination .el-pagination__jump) {
order: 3;
justify-content: center;
margin-top: 10px;
}
}
</style>

View File

@@ -1,493 +1,50 @@
<template>
<div class="default-main">
<TableHeader datePicker ref="TableHeaderRef" @selectChange="handleTableHeaderSelectChange">
<template #select>
<el-tabs v-model="activeTab" type="border-card" @tab-change="handleTabChange">
<el-tab-pane label="监测点评估" name="pollutionReport">
<MonitoringPointTab
ref="monitoringPointTabRef"
:active="activeTab === 'pollutionReport'"
@initialized="handleTabInitialized"
/>
</el-tab-pane>
<el-form-item label="区域">
<el-cascader
v-bind="$attrs"
:options="areOptions"
:props="cascaderProps"
v-model="selectedArea"
@change="handleFilterChange"
/>
</el-form-item>
<el-form-item label="统计类型">
<el-select
v-model="tableStore.table.params.statisticalType"
value-key="id"
placeholder="请选择统计类型"
@change="handleStatisticalTypeChange"
>
<el-option v-for="item in options" :key="item.id" :label="item.name" :value="item" />
</el-select>
</el-form-item>
<el-form-item label="电网标志">
<el-select
v-model="tableStore.table.params.powerFlag"
placeholder="请选择电网标志"
@change="handleFilterChange"
>
<el-option label="全部" value="0"></el-option>
<el-option label="电网侧" value="1"></el-option>
<el-option label="非电网侧" value="2"></el-option>
</el-select>
</el-form-item>
<el-form-item label="筛选">
<el-input
v-model="searchKeyword"
placeholder="请输入变电站/终端/监测点"
clearable
@input="handleFilterChange"
:show-word-limit=true
:maxlength="32"
/>
</el-form-item>
</template>
<template #operation>
<el-button icon="el-icon-Download" type="primary" @click="exportEvent">导出</el-button>
</template>
</TableHeader>
<div v-loading="tableStore.table.loading" class="main-container">
<vxe-table
class="full-height-table"
ref="positioningtableRef"
auto-resize
:data="tableStore.table.data"
v-bind="defaultAttribute"
:height="tableHeight"
resizable
show-overflow
>
<vxe-column title="序号" width="80" type="seq" align="center"></vxe-column>
<vxe-column field="gdName" title="供电公司" align="center" min-width="120"></vxe-column>
<vxe-column field="subStationName" :show-overflow="true" title="变电站" align="center" min-width="150" ></vxe-column>
<vxe-column field="devName" title="终端名称" align="center" min-width="120"></vxe-column>
<vxe-column field="devType" title="终端型号" align="center" min-width="150" ></vxe-column>
<vxe-column field="loginTime" title="投运时间" align="center" min-width="120"></vxe-column>
<vxe-column field="lineName" title="监测点名称" align="center" min-width="150"></vxe-column>
<vxe-column field="powerFlag" title="监测位置" align="center" min-width="100"></vxe-column>
<vxe-column field="lineVoltage" title="监测点电压等级" align="center" min-width="120"></vxe-column>
<vxe-column field="loadType" title="干扰源类型" align="center" min-width="120"></vxe-column>
<vxe-column field="objName" title="监测对象名称" align="center" min-width="150"></vxe-column>
<vxe-column field="interval" title="统计间隔" align="center" min-width="100"></vxe-column>
<vxe-column field="onlineRate" title="在线率(%)" align="center" min-width="100"></vxe-column>
<vxe-column field="integrity" title="完整率(%)" align="center" min-width="100"></vxe-column>
<vxe-column field="harmonicValue" :title="harmonicValueTitle" align="center" min-width="120"></vxe-column>
<vxe-column field="upCounts" title="暂升次数(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="downCounts" title="电压暂降(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="breakCounts" title="短时中断(次)" align="center" min-width="100"></vxe-column>
<vxe-column field="monitorId" title="一类监测点" align="center" min-width="120" :formatter="formatMonitorId"></vxe-column>
</vxe-table>
</div>
<!-- 修改分页控件 -->
<div class="pagination-container">
<el-pagination
:current-page="tableStore.table.params.pageNum"
:page-size="tableStore.table.params.pageSize"
:page-sizes="[10, 20, 50, 100]"
background
layout="sizes, total, prev, pager, next, jumper"
:total="tableStore.table.total"
@size-change="handleSizeChange"
@current-change="handleCurrentChange"
></el-pagination>
</div>
</div>
<el-tab-pane label="场站级评估" name="anotherReport">
<SubstationTab
ref="substationTabRef"
:active="activeTab === 'anotherReport'"
@initialized="handleTabInitialized"
/>
</el-tab-pane>
</el-tabs>
</div>
</template>
<script setup lang="ts">
import { ref, onMounted, provide, onBeforeUnmount, computed, reactive, watch } from 'vue'
import TableStore from '@/utils/tableStore'
import TableHeader from '@/components/table/header/index.vue'
import { defaultAttribute } from '@/components/table/defaultAttribute'
import { useDictData } from '@/stores/dictData'
import { debounce } from 'lodash-es'
import { ref } from 'vue'
import MonitoringPointTab from './MonitoringPointTab.vue'
import SubstationTab from './SubstationTab.vue'
defineOptions({
name: 'harmonic-boot/qydetailedAnalysis/pollutionReport'
const activeTab = ref('pollutionReport')
const monitoringPointTabRef = ref()
const substationTabRef = ref()
const tabInitialized = ref({
pollutionReport: false,
anotherReport: false
})
const dictData = useDictData()
const options = dictData.getBasicData('Pollution_Calc')
const tableHeight = ref(500) // 默认高度
const positioningtableRef = ref()
// 添加区域选择的响应式变量
const selectedArea = ref()
const areOptions:any = dictData.state.area
const allData = ref<PollutionItem[]>([])
const TableHeaderRef = ref()
// 添加响应式标题变量
const harmonicValueTitle = ref('谐波电压污染值')
const searchKeyword = ref('')
const cascaderProps = {
label: 'name',
value: 'id',
checkStrictly: true,
emitPath: false
}
// 存储所有数据
interface PollutionItem {
gdName?: string
subStationName?: string
devName?: string
lineName?: string
powerFlag?: string
}
// 格式化一类监测点字段
const formatMonitorId = (row: any) => {
return row.row.monitorId || '/'
}
// 处理 TableHeader 展开/收起事件
const handleTableHeaderSelectChange = (isExpanded: boolean) => {
if(isExpanded){
tableHeight.value = tableHeight.value - 55
}else{
tableHeight.value = tableHeight.value + 55
}
};
const calculateTableHeight = () => {
const windowHeight = window.innerHeight
const headerHeight = 120
const paginationHeight = 80 // 增加分页控件高度预留
const padding = 30 // 增加padding避免被遮挡
const calculatedHeight = windowHeight - headerHeight - paginationHeight - padding
tableHeight.value = Math.max(calculatedHeight, 300)
}
// 防抖处理窗口大小变化
const debouncedCalculateTableHeight = debounce(() => {
calculateTableHeight()
}, 300)
// 计算过滤后的数据
const filteredData = computed(() => {
let result = [...allData.value]
// 区域过滤
if (selectedArea.value) {
// 查找匹配的区域名称
let areaName = ''
let areaLevel = -1
const findAreaName = (areas: any[]) => {
for (const area of areas) {
if (area.id === selectedArea.value) {
areaName = area.name
areaLevel = area.level !== undefined ? area.level : -1
break
}
if (area.children && area.children.length > 0) {
findAreaName(area.children)
}
}
}
findAreaName(areOptions)
// 根据区域名称过滤数据但只有当层级大于1时才过滤
if (areaName && areaLevel > 1) {
result = result.filter(item => item.gdName && item.gdName.includes(areaName))
const handleTabChange = (tabName: string) => {
if (!tabInitialized.value[tabName]) {
if (tabName === 'pollutionReport' && monitoringPointTabRef.value) {
monitoringPointTabRef.value.refresh()
} else if (tabName === 'anotherReport' && substationTabRef.value) {
substationTabRef.value.refresh()
}
}
// 电网标志过滤
if (tableStore.table.params.powerFlag === '1') {
// 电网侧
result = result.filter(item => item.powerFlag && !item.powerFlag.includes('非'))
} else if (tableStore.table.params.powerFlag === '2') {
// 非电网侧
result = result.filter(item => item.powerFlag && item.powerFlag.includes('非'))
}
// '0' 表示全部,不过滤
// 统一搜索过滤
if (searchKeyword.value) {
const keyword = searchKeyword.value.toLowerCase()
result = result.filter(item =>
(item.subStationName && item.subStationName.toLowerCase().includes(keyword)) ||
(item.devName && item.devName.toLowerCase().includes(keyword)) ||
(item.lineName && item.lineName.toLowerCase().includes(keyword))
)
}
return result
})
// 计算当前页数据
const currentPageData = computed(() => {
const pageSize = tableStore.table.params.pageSize
const pageNum = tableStore.table.params.pageNum
const start = (pageNum - 1) * pageSize
const end = start + pageSize
return filteredData.value.slice(start, end)
})
// 更新总条数
const updateTotal = computed(() => {
return filteredData.value.length
})
const tableStore = new TableStore({
url: '/harmonic-boot/PollutionSubstation/downPollutionLineCalc',
method: 'POST',
column: [],
beforeSearchFun: () => {
//delete tableStore.table.params.statisticalType
delete tableStore.table.params.deptIndex
delete tableStore.table.params.interval
delete tableStore.table.params.searchEndTime
delete tableStore.table.params.searchBeginTime
delete tableStore.table.params.timeFlag
},
loadCallback: () => {
// 将所有数据存储到 allData 中
allData.value = tableStore.table.data || []
// 更新总条数
tableStore.table.total = updateTotal.value
// 更新当前页数据
tableStore.table.data = currentPageData.value
}
})
provide('tableStore', tableStore)
// 监听区域选项变化,设置默认值
watch(
() => areOptions,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
// 设置默认选中第一项
selectedArea.value = newOptions[0].id
tableStore.table.params.id = newOptions[0].id
}
},
{ immediate: true }
)
// 初始化统计类型
watch(
() => options,
(newOptions) => {
if (newOptions && newOptions.length > 0) {
// 检查是否已经设置了统计类型
if (!tableStore.table.params.statisticalType) {
tableStore.table.params.statisticalType = newOptions[0]
tableStore.table.params.ids = [newOptions[0].id]
}
}
},
{ immediate: true }
)
tableStore.table.params.powerFlag = "0"
tableStore.table.params.isUpToGrid = 0
tableStore.table.params.type = 1
// 监听统计类型变化同步更新ids
const handleStatisticalTypeChange = (newVal: { id: any }) => {
console.log("统计类型变化", newVal)
if (newVal) {
tableStore.table.params.ids = [newVal.id]
// 根据统计类型动态更新标题
if (newVal.name) {
harmonicValueTitle.value = newVal.name + '污染值'
}
}
// 重新调用接口
tableStore.index()
}
// 处理过滤条件变化
const handleFilterChange = () => {
// 重置分页到第一页
tableStore.table.params.pageNum = 1
// 更新数据和总条数
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
const handleTabInitialized = (tabName: string) => {
tabInitialized.value[tabName] = true
}
// 分页事件处理
const handleSizeChange = (val: number) => {
tableStore.table.params.pageSize = val
tableStore.table.params.pageNum = 1
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
const handleCurrentChange = (val: number) => {
tableStore.table.params.pageNum = val
tableStore.table.data = currentPageData.value
tableStore.table.total = updateTotal.value
}
// 导出
const exportEvent = () => {
// 获取当前过滤后的所有数据
const allFilteredData = filteredData.value;
// 使用 vxe-table 的导出功能
positioningtableRef.value.exportData({
filename: '污染值报告',
sheetName: 'Sheet1',
type: 'xlsx',
useStyle: true,
data: allFilteredData,
columnFilterMethod: function (column, $columnIndex) {
return !(column.$columnIndex === 0)
}
})
}
onMounted(() => {
tableStore.index()
calculateTableHeight()
window.addEventListener('resize', debouncedCalculateTableHeight)
})
onBeforeUnmount(() => {
window.removeEventListener('resize', debouncedCalculateTableHeight)
})
</script>
<style scoped lang="scss">
.default-main {
height: calc(100vh - 20px); /* 减去一些边距避免被任务栏遮挡 */
display: flex;
flex-direction: column;
}
.table-container {
flex: 1;
display: flex;
flex-direction: column;
overflow: hidden;
position: relative;
}
.full-height-table {
flex: 1;
overflow: hidden;
}
:deep .vxe-table {
height: 100% !important;
overflow: hidden !important;
}
:deep .vxe-table--body-wrapper {
overflow: auto !important;
}
.pagination-container {
flex: 0 0 auto; /* 不伸缩,保持固定高度 */
display: flex;
justify-content: center; /* 居中对齐 */
align-items: center;
min-height: 60px;
padding: 10px 20px;
background-color: #fff;
border-top: 1px solid #ebeef5;
z-index: 100; /* 提高层级确保可见 */
position: relative; /* 确保定位上下文 */
width: 100%;
margin-bottom: 20px;
}
:deep .el-pagination {
display: flex;
flex-wrap: nowrap;
align-items: center;
width: 100%;
.el-pagination__sizes {
.el-select {
min-width: 100px;
margin-right: 10px;
}
}
.el-pagination__total {
margin-right: auto; /* 关键:将总共条数推到左侧 */
white-space: nowrap;
}
/* 将除了 sizes 和 total 之外的所有元素都推到最右边 */
.btn-prev,
.btn-pager,
.btn-next,
.el-pagination__jump {
margin-left: auto;
}
.btn-prev {
margin-right: 5px;
}
.btn-next {
margin-left: 5px;
}
.el-pagination__jump {
margin-left: 10px;
}
.el-pagination__editor {
width: 50px;
margin: 0 5px;
}
}
/* 响应式处理 */
@media screen and (max-width: 768px) {
.pagination-container {
padding: 8px 10px;
min-height: 50px;
}
:deep .el-pagination {
.el-pagination__sizes {
.el-select {
min-width: 80px;
}
}
.el-pagination__total {
font-size: 12px;
}
.el-pagination__jump {
font-size: 12px;
}
}
}
@media screen and (max-width: 480px) {
:deep .el-pagination {
.el-pagination__sizes {
display: none;
}
.el-pagination__total {
display: none;
}
.btn-pager {
display: none;
}
.el-pagination__jump {
display: none;
}
}
}
</style>

View File

@@ -368,10 +368,10 @@ const siteList = ref<any>([])
const polyline = ref<any>([])
const lineId = ref('')
const center = ref({
lng: 115.64636, //江西
lat: 27.232634
// lng: 122.42588,//辽宁
// lat: 40.810977
//lng: 115.64636, //江西
//lat: 27.232634
lng: 122.42588,//辽宁
lat: 40.810977
})
const infoWindowPoint = ref<anyObj>({
lng: 0,