1
0
forked from dyf/dyf-vue-ui

Compare commits

2 Commits
main ... main

Author SHA1 Message Date
589a1eafed Merge branch 'liubiao-main' 2025-09-09 17:04:10 +08:00
8a65f4690c 首页图表动态数据 2025-09-09 17:02:39 +08:00
3 changed files with 234 additions and 163 deletions

View File

@ -16,9 +16,28 @@ export const getEquipmentClassification = (params) => {
params: params params: params
}); });
}; };
// 获取设备使用情况
export const getEquipmentUsageData = (range, params = {}) => {
return request({
url: `/api/device/homepage/getEquipmentUsageData/${range}`,
method: 'get',
params: params
});
};
// 获取报警信息
export const getAlarmInformation = (params) => {
return request({
url: `/api/device/homepage/getAlarmInformation`,
method: 'get',
params: params
});
};
export default { export default {
getDataOverview, getDataOverview,
getEquipmentClassification getEquipmentClassification,
getEquipmentUsageData,
getAlarmInformation
} }

View File

@ -662,19 +662,15 @@ const handleDeviceTypeChange = async (deviceTypeId: string | number) => {
//两个都有值:显示两个字段 + 都加校验 //两个都有值:显示两个字段 + 都加校验
showMacField.value = true; showMacField.value = true;
showImeiField.value = true; showImeiField.value = true;
rules.value.deviceMac = [{ required: true, message: '请输入设备MAC', trigger: 'blur' }];
rules.value.deviceImei = [{ required: true, message: '请输入设备IMEI', trigger: 'blur' }];
console.log('两个字段都有值'); console.log('两个字段都有值');
} else if (hasMac) { } else if (hasMac) {
showMacField.value = true; showMacField.value = true;
showImeiField.value = false; showImeiField.value = false;
rules.value.deviceMac = [{ required: true, message: '请输入设备MAC', trigger: 'blur' }];
rules.value.deviceImei = []; rules.value.deviceImei = [];
console.log('只有 Mac 有值'); console.log('只有 Mac 有值');
} else if (hasImei) { } else if (hasImei) {
showImeiField.value = true; showImeiField.value = true;
showMacField.value = false; showMacField.value = false;
rules.value.deviceImei = [{ required: true, message: '请输入设备IMEI', trigger: 'blur' }];
rules.value.deviceMac = []; rules.value.deviceMac = [];
console.log('只有 Imei 有值'); console.log('只有 Imei 有值');
} }

View File

@ -77,14 +77,21 @@
<div class="card-header"> <div class="card-header">
<h2>设备使用频次</h2> <h2>设备使用频次</h2>
<div class="chart-controls"> <div class="chart-controls">
<!-- <el-select v-model="timeRange" placeholder="选择时间范围" @change="updateFrequencyChart"> <el-select v-model="deviceType" placeholder="设备类型" style="width: 150px;"
<el-option label="近半年" value="halfYear"></el-option> @change="handleDeviceTypeChange">
<el-option label="近一年" value="oneYear"></el-option>
</el-select> -->
<el-select v-model="deviceType" placeholder="设备类型" style="width: 150px;">
<el-option v-for="item in deviceTypeOptions" :key="item.value" :label="item.typeName" <el-option v-for="item in deviceTypeOptions" :key="item.value" :label="item.typeName"
:value="item.deviceTypeId" /> :value="item.deviceTypeId" />
</el-select> </el-select>
<div class="tab-group">
<div class="tab-item" :class="{ 'tab-item--active': activeTab === '1' }"
@click="updateFrequencyChart('1')">
近半年
</div>
<div class="tab-item" :class="{ 'tab-item--active': activeTab === '2' }"
@click="updateFrequencyChart('2')">
近一年
</div>
</div>
</div> </div>
</div> </div>
<div class="card-body"> <div class="card-body">
@ -105,12 +112,12 @@
<!-- 环形图容器今日报警处理占比 --> <!-- 环形图容器今日报警处理占比 -->
<div ref="alarmRingChartRef" class="chart-container"></div> <div ref="alarmRingChartRef" class="chart-container"></div>
<div class="alarm-stats"> <div class="alarm-stats">
<div class="stat-item"> <div class="stat-item" v-if="alarmsData">
<div class="stat red">365</div> <div class="stat red">{{ alarmsData.alarmsTotal }}</div>
<div class="label">报警总数</div> <div class="label">报警总数</div>
</div> </div>
<div class="stat-item"> <div class="stat-item" v-if="alarmsData">
<div class="stat green">300</div> <div class="stat green">{{ alarmsData.processingAlarm }}</div>
<div class="label">总处理报警</div> <div class="label">总处理报警</div>
</div> </div>
</div> </div>
@ -148,6 +155,8 @@ const DataOverview = ref<DataOverviewType>({
}); });
const deviceTypeOptions = ref([]); //设备类型 const deviceTypeOptions = ref([]); //设备类型
const deviceType = ref() const deviceType = ref()
const activeTab = ref('1');
const alarmsData = ref()
// ---------------------- 基础数据 ---------------------- // ---------------------- 基础数据 ----------------------
// 设备分类数据 // 设备分类数据
const deviceList = ref([ const deviceList = ref([
@ -155,9 +164,6 @@ const deviceList = ref([
{ name: "蓝牙设备", current: 0, total: 0 }, { name: "蓝牙设备", current: 0, total: 0 },
{ name: "4G&蓝牙设备", current: 0, total: 0 }, { name: "4G&蓝牙设备", current: 0, total: 0 },
]); ]);
// 时间范围(控制设备使用频次图表)
const timeRange = ref('halfYear');
// ---------------------- 图表Ref用于挂载图表实例 ---------------------- // ---------------------- 图表Ref用于挂载图表实例 ----------------------
const frequencyChartRef = ref<HTMLDivElement | null>(null); // 设备使用频次折线图 const frequencyChartRef = ref<HTMLDivElement | null>(null); // 设备使用频次折线图
const alarmRingChartRef = ref<HTMLDivElement | null>(null); // 报警环形图 const alarmRingChartRef = ref<HTMLDivElement | null>(null); // 报警环形图
@ -183,114 +189,117 @@ const handleControlPanel = () => {
}; };
// ---------------------- 图表初始化方法 ---------------------- // ---------------------- 图表初始化方法 ----------------------
/** const initFrequencyChart = async (range: any = '1', deviceTypeId: any) => {
* 1. 设备使用频次折线图
*/
const initFrequencyChart = () => {
if (!frequencyChartRef.value) return; if (!frequencyChartRef.value) return;
// 初始化图表实例
frequencyChartInstance = echarts.init(frequencyChartRef.value); frequencyChartInstance = echarts.init(frequencyChartRef.value);
try {
// 假数据:按时间范围区分 let data = {
const chartData = timeRange.value === 'halfYear' deviceTypeId: deviceTypeId
? {
xAxis: ['1月', '2月', '3月', '4月', '5月', '6月'],
yAxis: [320, 280, 450, 380, 520, 480], // 近半年使用次数
peak: { name: '峰值', value: 520, month: '5月' }
} }
: { const res = await api.getEquipmentUsageData(range, data);
xAxis: ['1月', '2月', '3月', '4月', '5月', '6月', '7月', '8月', '9月', '10月', '11月', '12月'], const monthData = res.data[0] || {};
yAxis: [320, 280, 450, 380, 520, 480, 550, 620, 580, 490, 530, 650], // 近一年使用次数 const monthKeys = ['m1', 'm2', 'm3', 'm4', 'm5', 'm6', 'm7', 'm8', 'm9', 'm10', 'm11', 'm12'];
peak: { name: '峰值', value: 650, month: '12月' } const monthNames = ['1月', '2月', '3月', '4月', '5月', '6月', '7月', '8月', '9月', '10月', '11月', '12月'];
// 3. 计算时间范围(核心逻辑)
let filteredKeys, filteredNames, yAxisData;
const today = new Date();
const currentMonth = today.getMonth();
if (range === '1') {
const result = [];
for (let i = 0; i < 6; i++) {
const targetMonth = (currentMonth - i + 12) % 12;
result.push(targetMonth);
}
const recent6Months = result.reverse();
// 匹配接口字段和名称
filteredKeys = recent6Months.map(monthIndex => monthKeys[monthIndex]);
filteredNames = recent6Months.map(monthIndex => monthNames[monthIndex]);
yAxisData = filteredKeys.map(key => monthData[key] || 0);
} else {
// 近一年全部12个月1月→12月
filteredKeys = monthKeys;
filteredNames = monthNames;
yAxisData = filteredKeys.map(key => monthData[key] || 0);
}
const chartData = {
xAxis: filteredNames,
yAxis: yAxisData,
peak: { name: '峰值', value: 0, month: '' }
}; };
if (yAxisData.length) {
// ECharts配置项 const maxVal = Math.max(...yAxisData);
const maxIndex = yAxisData.indexOf(maxVal);
chartData.peak = {
name: '峰值',
value: maxVal,
month: chartData.xAxis[maxIndex] || ''
};
}
const option = { const option = {
tooltip: { tooltip: { trigger: 'axis', formatter: '{b}: {c} 次' },
trigger: 'axis', grid: { left: '3%', right: '4%', bottom: '3%', containLabel: true },
formatter: '{b}: {c} 次'
},
grid: {
left: '3%',
right: '4%',
bottom: '3%',
containLabel: true
},
xAxis: { xAxis: {
type: 'category', type: 'category',
data: chartData.xAxis, data: chartData.xAxis,
axisLabel: { axisLabel: { interval: 0 }
interval: 0 // 强制显示所有x轴标签
}
}, },
yAxis: { yAxis: {
type: 'value', type: 'value',
name: '使用次数', name: '使用次数',
min: 0, min: 0,
max: Math.max(...chartData.yAxis) + 100 // y轴最大值留有余量 max: chartData.yAxis.length ? Math.max(...chartData.yAxis) + 100 : 100
}, },
series: [ series: [
{ {
name: '使用频次', name: '使用频次',
type: 'line', type: 'line',
data: chartData.yAxis, data: chartData.yAxis,
smooth: false, // 平滑折线 smooth: false,
lineStyle: { lineStyle: { width: 3, color: '#409eff' },
width: 3, itemStyle: { color: '#409eff', radius: 5 },
color: '#409eff'
},
itemStyle: {
color: '#409eff',
radius: 5
},
areaStyle: { areaStyle: {
color: { color: {
type: 'linear', type: 'linear',
x: 0, x: 0, y: 0, x2: 0, y2: 1,
y: 0, colorStops: [
x2: 0, { offset: 0, color: 'rgba(64, 158, 255, 0.3)' },
y2: 1, { offset: 1, color: 'rgba(64, 158, 255, 0)' }
colorStops: [{ ]
offset: 0, color: 'rgba(64, 158, 255, 0.3)'
}, {
offset: 1, color: 'rgba(64, 158, 255, 0)'
}]
} }
}, },
// 标记峰值
markPoint: { markPoint: {
data: [{ data: chartData.peak.value
? [{
name: chartData.peak.name, name: chartData.peak.name,
value: chartData.peak.value, value: chartData.peak.value,
xAxis: chartData.xAxis.indexOf(chartData.peak.month), xAxis: chartData.xAxis.indexOf(chartData.peak.month),
yAxis: chartData.peak.value, yAxis: chartData.peak.value,
itemStyle: { itemStyle: { color: '#409eff' }
color: '#ff4d4f'
}
}] }]
: []
} }
} }
] ]
}; };
// 渲染图表
frequencyChartInstance.setOption(option); frequencyChartInstance.setOption(option);
} catch (error) {
}
}; };
/** /**
* 2. 报警环形图(今日报警处理占比) * 2. 报警环形图(今日报警处理占比)
*/ */
const initAlarmRingChart = () => { const initAlarmRingChart = async () => {
if (!alarmRingChartRef.value) return; if (!alarmRingChartRef.value) return;
alarmRingChartInstance = echarts.init(alarmRingChartRef.value); alarmRingChartInstance = echarts.init(alarmRingChartRef.value);
try {
// 假数据今日报警6次已处理6次 const res = await api.getAlarmInformation({});
const { alarmsTotalToday = 0, processingAlarmToday = 0 } = res.data || {};
alarmsData.value = res.data || '0'
alarmRingChartInstance = echarts.init(alarmRingChartRef.value);
const option = { const option = {
tooltip: { tooltip: {
trigger: 'item', trigger: 'item',
formatter: '{b}: {c} 次 ({d}%)' formatter: '{b}: {c} 次 ({d}%)' // 显示数量和百分比
}, },
series: [ series: [
{ {
@ -301,23 +310,34 @@ const initAlarmRingChart = () => {
label: { label: {
show: true, show: true,
position: 'center', position: 'center',
formatter: '6/6\n今日报警/处理', // 中心显示“今日 已处理/总数” formatter: `${alarmsTotalToday}/${processingAlarmToday}\n今日报警/处理`,
fontSize: 16, fontSize: 16,
fontWeight: 'bold', fontWeight: 'bold',
color: '#333' color: '#333'
}, },
labelLine: { labelLine: {
show: false show: false // 隐藏标签连接线
}, },
data: [ data: [
{ value: 6, name: '已处理报警', itemStyle: { color: '#07BE75' } }, {
{ value: 0, name: '未处理报警', itemStyle: { color: '#F65757' } } value: alarmsTotalToday,
name: '已处理报警',
itemStyle: { color: '#07BE75' } // 绿色:已处理
},
{
value: processingAlarmToday,
name: '未处理报警',
itemStyle: { color: '#F65757' } // 红色:未处理
}
] ]
} }
] ]
}; };
alarmRingChartInstance.setOption(option); alarmRingChartInstance.setOption(option);
// 报警柱状图
initAlarmBarChart()
} catch (error) {
}
}; };
/** /**
@ -325,14 +345,23 @@ const initAlarmRingChart = () => {
*/ */
const initAlarmBarChart = () => { const initAlarmBarChart = () => {
if (!alarmBarChartRef.value) return; if (!alarmBarChartRef.value) return;
const alarmTypeMap = [
{ name: '强制报警', field: 'alarmForced' }, // alarmForced
{ name: '撞击闯入', field: 'intrusionImpact' }, // intrusionImpact
{ name: '手动报警', field: 'alarmManual' }, // alarmManual
{ name: '电子围栏', field: 'fenceElectronic' } // fenceElectronic
];
alarmBarChartInstance = echarts.init(alarmBarChartRef.value); const alarmTypes = alarmTypeMap.map(item => item.name);
const alarmCounts = alarmTypeMap.map(item => {
// 假数据:各类型报警次数 const value = alarmsData.value[item.field]; // 提取对应字段值
const alarmTypes = ['强制报警', '撞击闯入', '手动报警', '电子围栏']; console.log(`${item.name}数值:`, value); // 打印每个类型的数值
const alarmCounts = [50, 35, 65, 50]; // 对应各类型次数 return value;
});
// ECharts配置项 const commonGradient = new echarts.graphic.LinearGradient(0, 0, 0, 1, [
{ offset: 0, color: 'rgba(246, 87, 87, 1)' }, // 渐变起点:#F65757不透明
{ offset: 1, color: 'rgba(224, 52, 52, 0)' } // 渐变终点:#E03434全透明
]);
const option = { const option = {
tooltip: { tooltip: {
trigger: 'axis', trigger: 'axis',
@ -348,8 +377,7 @@ const initAlarmBarChart = () => {
type: 'category', type: 'category',
data: alarmTypes, data: alarmTypes,
axisLabel: { axisLabel: {
interval: 0, interval: 0 // 强制显示所有标签
//rotate: 15 // 标签旋转,避免重叠
} }
}, },
yAxis: { yAxis: {
@ -362,28 +390,29 @@ const initAlarmBarChart = () => {
name: '报警次数', name: '报警次数',
type: 'bar', type: 'bar',
data: alarmCounts, data: alarmCounts,
barWidth: '40%', barWidth: '20%',
itemStyle: { itemStyle: {
color: (params: any) => { color: commonGradient, // 所有柱子共用同一渐变色
// 不同类型报警用不同颜色 borderRadius: 4 // 统一4px圆角
const colors = ['#ff4d4f', '#e6a23c', '#409eff', '#67c23a'];
return colors[params.dataIndex];
}
} }
} }
] ]
}; };
// 4. 初始化并渲染图表
alarmBarChartInstance = echarts.init(alarmBarChartRef.value);
alarmBarChartInstance.setOption(option); alarmBarChartInstance.setOption(option);
}; };
// ---------------------- 图表更新方法(时间范围切换时) ---------------------- // ---------------------- 图表更新方法(时间范围切换时) ----------------------
const updateFrequencyChart = () => { const updateFrequencyChart = (tabValue: any) => {
// 先销毁旧实例,再重新初始化 activeTab.value = tabValue;
if (frequencyChartInstance) { if (frequencyChartInstance) {
frequencyChartInstance.dispose(); frequencyChartInstance.dispose();
} }
initFrequencyChart(); initFrequencyChart(tabValue, '');
};
const handleDeviceTypeChange = (all) => {
initFrequencyChart(activeTab.value, all);
}; };
// 首页统计接口 // 首页统计接口
const getData = async () => { const getData = async () => {
@ -437,9 +466,8 @@ const getData = async () => {
// ---------------------- 生命周期钩子(初始化/销毁图表) ---------------------- // ---------------------- 生命周期钩子(初始化/销毁图表) ----------------------
onMounted(() => { onMounted(() => {
// 页面加载时初始化所有图表 // 页面加载时初始化所有图表
initFrequencyChart(); initFrequencyChart('1', '');
initAlarmRingChart(); initAlarmRingChart();
initAlarmBarChart();
getData() getData()
// 监听窗口 resize自动调整图表大小 // 监听窗口 resize自动调整图表大小
@ -461,7 +489,7 @@ onUnmounted(() => {
<style lang="scss" scoped> <style lang="scss" scoped>
.home { .home {
padding: 20px; padding:0px 20px 10px 20px;
background-color: #f5f7fa; background-color: #f5f7fa;
min-height: calc(100vh - 84px); min-height: calc(100vh - 84px);
@ -573,6 +601,34 @@ onUnmounted(() => {
} }
} }
.tab-group {
display: flex;
gap: 12px;
/* 两个标签的间距 */
align-items: center;
margin-top: 15px;
}
.tab-item {
padding: 3px 10px;
cursor: pointer;
color: #666;
/* 未选中时的文字颜色 */
transition: all 0.2s ease;
font-size: 13px;
}
.tab-item--active {
border-color: #409eff;
/* 选中时的边框颜色(示例用 Element UI 主色) */
color: #409eff;
/* 选中时的文字颜色 */
background-color: rgba(64, 158, 255, 0.05);
/* 可选:选中时的浅背景 */
border: 1px solid rgba(2, 124, 251, 1);
border-radius: 4px;
}
.card-header { .card-header {
display: flex; display: flex;
justify-content: space-between; justify-content: space-between;