修改页面弹窗全屏

This commit is contained in:
2026-02-10 23:22:22 +08:00
parent 3bab32e8f8
commit 7fda86d890
5 changed files with 198 additions and 31 deletions

View File

@@ -164,9 +164,9 @@ public class vDate {
} }
public static Date get24Hours() { public static Date getHours(Integer afterHours) {
Calendar calendar = Calendar.getInstance(); Calendar calendar = Calendar.getInstance();
calendar.add(Calendar.HOUR_OF_DAY, -24); calendar.add(Calendar.HOUR_OF_DAY, afterHours);
return calendar.getTime(); return calendar.getTime();
} }
} }

View File

@@ -51,6 +51,8 @@ public class BizResourceMonitor extends DataEntity<BizResourceMonitor> implement
private String hostId; // host_id private String hostId; // host_id
private String hourTime; private String hourTime;
private Integer afterHours;
public BizResourceMonitor() { public BizResourceMonitor() {
this(null); this(null);
} }

View File

@@ -102,7 +102,7 @@ public class BizResourceMonitorController extends BaseController {
@RequestMapping(value = "listAll") @RequestMapping(value = "listAll")
@ResponseBody @ResponseBody
public List<BizResourceMonitor> listAll(BizResourceMonitor bizResourceMonitor) { public List<BizResourceMonitor> listAll(BizResourceMonitor bizResourceMonitor) {
bizResourceMonitor.setCreateTime(vDate.get24Hours()); bizResourceMonitor.setCreateTime(vDate.getHours(bizResourceMonitor.getAfterHours()));
return bizResourceMonitorService.findList(bizResourceMonitor); return bizResourceMonitorService.findList(bizResourceMonitor);
} }

View File

@@ -11,6 +11,7 @@ export interface BizResourceMonitor extends BasicModel<BizResourceMonitor> {
memoryTotal: number; // 服务器总内存大小 memoryTotal: number; // 服务器总内存大小
hourTime: string; // 时间 hourTime: string; // 时间
hostId: string; // 服务器主机编号 hostId: string; // 服务器主机编号
afterHours: number;
} }
export const bizResourceMonitorListAll = (params?: BizResourceMonitor | any) => export const bizResourceMonitorListAll = (params?: BizResourceMonitor | any) =>

View File

@@ -1,10 +1,25 @@
<template> <template>
<Card title="监控信息" style="width: 100%; height: 30vh; margin: 0; padding: 0;"> <Card title="监控信息" style="width: 100%; height: 30vh; margin: 0; padding: 0;">
<template #extra>
<div class="status-filter-container">
<div class="status-filter">
<span
v-for="item in statusOptions"
:key="item.value"
:class="['status-item', { active: currentStatus === item.value }]"
@click="handleStatusChange(item.value)"
>
{{ item.label }}
</span>
</div>
</div>
</template>
<div ref="chartDom" style="width: 100%; height: calc(100% - 32px - 4px); padding: 2px;"></div> <div ref="chartDom" style="width: 100%; height: calc(100% - 32px - 4px); padding: 2px;"></div>
</Card> </Card>
</template> </template>
<script lang="ts" setup name="Monitor"> <script lang="ts" setup name="Monitor">
import { ref, onMounted, watch, onUnmounted } from 'vue'; import { ref, onMounted, watch, onUnmounted, nextTick } from 'vue';
import { Card } from 'ant-design-vue'; import { Card } from 'ant-design-vue';
import { BizResourceMonitor, bizResourceMonitorListAll } from '@jeesite/biz/api/biz/resourceMonitor'; import { BizResourceMonitor, bizResourceMonitorListAll } from '@jeesite/biz/api/biz/resourceMonitor';
import * as echarts from 'echarts'; import * as echarts from 'echarts';
@@ -17,15 +32,48 @@ const monitorList = ref<BizResourceMonitor[]>([]);
const chartInstance = ref<echarts.ECharts | null>(null); const chartInstance = ref<echarts.ECharts | null>(null);
const chartDom = ref<HTMLDivElement | null>(null); const chartDom = ref<HTMLDivElement | null>(null);
// 状态选项配置
const statusOptions = ref([
{ value: '-12', label: '12小时' },
{ value: '-24', label: '24小时' },
{ value: '-36', label: '36小时' },
{ value: '-72', label: '72小时' },
]);
const currentStatus = ref<string>('-12');
// 防抖处理resize避免频繁触发
const resizeHandler = () => { const resizeHandler = () => {
chartInstance.value?.resize(); if (chartInstance.value) {
// 使用nextTick确保DOM更新完成后再调整大小
nextTick(() => {
chartInstance.value?.resize({
animation: { duration: 200 }
});
});
}
};
const handleStatusChange = (status: string) => {
if (currentStatus.value === status) return;
currentStatus.value = status;
fetchList(props.formParams);
}; };
const fetchList = async (params: Record<string, any>) => { const fetchList = async (params: Record<string, any>) => {
try { try {
const result = await bizResourceMonitorListAll(params); const reqParams = {
...params,
afterHours: currentStatus.value
};
const result = await bizResourceMonitorListAll(reqParams);
monitorList.value = result || []; monitorList.value = result || [];
renderChart(); // 确保图表实例初始化完成后再渲染
if (chartInstance.value) {
renderChart();
} else {
// 如果实例未初始化,先初始化再渲染
initChart().then(() => renderChart());
}
} catch (error) { } catch (error) {
console.error('获取数据列表失败:', error); console.error('获取数据列表失败:', error);
monitorList.value = []; monitorList.value = [];
@@ -33,26 +81,91 @@ const fetchList = async (params: Record<string, any>) => {
} }
}; };
const initChart = () => { // 改为异步函数,确保初始化完成
const initChart = async () => {
if (!chartDom.value) return; if (!chartDom.value) return;
// 先销毁旧实例(如果存在)
if (chartInstance.value) { if (chartInstance.value) {
chartInstance.value.dispose(); chartInstance.value.dispose();
chartInstance.value = null; chartInstance.value = null;
} }
// 等待DOM更新完成
await nextTick();
// 初始化图表实例
chartInstance.value = echarts.init(chartDom.value); chartInstance.value = echarts.init(chartDom.value);
// 移除重复的事件监听
window.removeEventListener('resize', resizeHandler);
window.addEventListener('resize', resizeHandler); window.addEventListener('resize', resizeHandler);
// 监听图例点击事件,增加错误捕获
chartInstance.value.on('legendselectchanged', (params) => {
try {
// 可以在这里自定义图例点击逻辑,也可以留空仅做错误捕获
console.log('图例选中状态变更:', params);
} catch (e) {
console.error('图例点击事件处理异常:', e);
}
});
}; };
const renderChart = () => { const renderChart = () => {
// 核心保护:实例不存在时直接返回
if (!chartInstance.value) return; if (!chartInstance.value) return;
// 空数据时的完整配置,避免交互报错
if (monitorList.value.length === 0) { if (monitorList.value.length === 0) {
// 空数据时不直接隐藏tooltip仅清空数据避免影响tooltip初始化
chartInstance.value.setOption({ chartInstance.value.setOption({
series: [{ data: [] }, { data: [] }], legend: {
xAxis: { data: [] } data: ['CPU使用率', '内存使用率'],
top: 5,
left: 'center',
itemGap: 15,
textStyle: { fontSize: 11 }
},
grid: {
left: '2%',
right: '2%',
bottom: '8%',
top: '18%',
containLabel: true
},
xAxis: {
type: 'category',
data: [],
axisLabel: { rotate: 20, fontSize: 10, margin: 5 },
axisTick: { alignWithLabel: true }
},
yAxis: {
type: 'value',
name: '使用率(%)',
nameTextStyle: { fontSize: 10 },
min: 0,
max: 100,
interval: 20,
axisLabel: { formatter: '{value}%', fontSize: 10, margin: 5 }
},
tooltip: {
trigger: 'axis',
textStyle: { fontSize: 10 }
},
series: [
{
name: 'CPU使用率',
type: 'line',
data: [],
smooth: true
},
{
name: '内存使用率',
type: 'line',
data: [],
smooth: true
}
]
}); });
return; return;
} }
@@ -67,21 +180,10 @@ const renderChart = () => {
return Math.max(0, Math.min(100, value)); return Math.max(0, Math.min(100, value));
}); });
const allData = [...cpuData, ...memoryData]; const yMin = 0;
const minValue = Math.min(...allData); const yMax = 100;
const maxValue = Math.max(...allData); const yInterval = 20;
const yMin = Math.max(0, minValue - 5);
const yMax = Math.min(100, maxValue + 5);
const getInterval = (min: number, max: number) => {
const range = max - min;
if (range <= 10) return 2;
if (range <= 20) return 5;
if (range <= 50) return 10;
return 20;
};
const yInterval = getInterval(yMin, yMax);
const option = { const option = {
title: { left: 'center' }, title: { left: 'center' },
legend: { legend: {
@@ -91,9 +193,18 @@ const renderChart = () => {
itemGap: 15, itemGap: 15,
textStyle: { fontSize: 11 } textStyle: { fontSize: 11 }
}, },
tooltip: {
trigger: 'axis',
textStyle: { fontSize: 10 },
// 增加tooltip配置避免交互时的默认行为报错
axisPointer: {
type: 'line',
lineStyle: { width: 1, color: '#ccc' }
}
},
grid: { grid: {
left: '2%', // 从1%改为5%,扩大触发区域 left: '2%',
right: '2%', // 从1%改为5%,扩大触发区域 right: '2%',
bottom: '8%', bottom: '8%',
top: '18%', top: '18%',
containLabel: true containLabel: true
@@ -153,9 +264,10 @@ const renderChart = () => {
}); });
}; };
onMounted(() => { onMounted(async () => {
initChart(); // 关键修复:先初始化图表,再获取数据渲染
fetchList(props.formParams); await initChart();
await fetchList(props.formParams);
}); });
watch( watch(
@@ -167,8 +279,11 @@ watch(
); );
onUnmounted(() => { onUnmounted(() => {
// 移除所有事件监听
window.removeEventListener('resize', resizeHandler); window.removeEventListener('resize', resizeHandler);
if (chartInstance.value) { if (chartInstance.value) {
// 移除图例点击事件监听
chartInstance.value.off('legendselectchanged');
chartInstance.value.dispose(); chartInstance.value.dispose();
chartInstance.value = null; chartInstance.value = null;
} }
@@ -176,6 +291,55 @@ onUnmounted(() => {
</script> </script>
<style scoped> <style scoped>
/* extra插槽容器状态标签 + 更多按钮 横向排列 */
.status-filter-container {
display: flex;
align-items: center;
gap: 16px;
}
/* 状态筛选栏(非按钮样式) */
.status-filter {
display: flex;
align-items: center;
flex-wrap: wrap;
gap: 12px; /* 标签间距 */
font-size: 14px;
}
/* 状态项样式(纯文本标签) */
.status-item {
cursor: pointer;
color: #666;
position: relative;
padding-bottom: 2px;
transition: all 0.2s ease;
user-select: none;
}
/* 选中状态样式(下划线 + 高亮色) */
.status-item.active {
color: #1890ff;
font-weight: 500;
}
/* 选中状态下划线 */
.status-item.active::after {
content: '';
position: absolute;
bottom: 0;
left: 0;
width: 100%;
height: 2px;
background-color: #1890ff;
border-radius: 1px;
}
/* 悬浮效果 */
.status-item:not(.active):hover {
color: #40a9ff;
}
:deep(.ant-card) { :deep(.ant-card) {
border-radius: 4px; border-radius: 4px;
box-shadow: 0 1px 2px rgba(0,0,0,0.05); box-shadow: 0 1px 2px rgba(0,0,0,0.05);
@@ -199,4 +363,4 @@ onUnmounted(() => {
margin: 0; margin: 0;
height: calc(100% - 32px); height: calc(100% - 32px);
} }
</style> </style>