2025-09-27 16:10:17 +08:00
|
|
|
|
<template>
|
2025-09-27 17:58:15 +08:00
|
|
|
|
<div class="alarm-table-container">
|
|
|
|
|
|
<div class="alarm-table-header">
|
|
|
|
|
|
<div class="header-item">报警时间</div>
|
|
|
|
|
|
<div class="header-item">设备类型</div>
|
|
|
|
|
|
<div class="header-item">设备IMEI</div>
|
|
|
|
|
|
<div class="header-item">报警事件</div>
|
2025-09-30 10:59:31 +08:00
|
|
|
|
<div class="header-item">报警位置</div>
|
2025-09-27 17:58:15 +08:00
|
|
|
|
</div>
|
2025-09-30 10:59:31 +08:00
|
|
|
|
<div class="alarm-table-body">
|
|
|
|
|
|
<div ref="tableBody" class="alarm-table-body-inner">
|
|
|
|
|
|
<!-- 第一份数据 -->
|
|
|
|
|
|
<div v-for="(item, index) in displayData" :key="`first-${getKey(item, index)}`" class="alarm-item">
|
|
|
|
|
|
<div class="item-cell">{{ item.startTime }}</div>
|
|
|
|
|
|
<div class="item-cell">{{ item.deviceTypeName }}</div>
|
|
|
|
|
|
<div class="item-cell">{{ item.deviceImei }}</div>
|
|
|
|
|
|
<div class="item-cell alarm-event">
|
|
|
|
|
|
{{ getEventName(item.deviceAction) }}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="item-cell loaction">{{ item.location }}</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<!-- 第二份数据(用于无缝滚动) -->
|
|
|
|
|
|
<div v-for="(item, index) in displayData" :key="`second-${getKey(item, index)}`" class="alarm-item">
|
|
|
|
|
|
<div class="item-cell">{{ item.startTime }}</div>
|
|
|
|
|
|
<div class="item-cell">{{ item.deviceTypeName }}</div>
|
|
|
|
|
|
<div class="item-cell">{{ item.deviceImei }}</div>
|
|
|
|
|
|
<div class="item-cell alarm-event">
|
|
|
|
|
|
{{ getEventName(item.deviceAction) }}
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div class="item-cell loaction">{{ item.location }}</div>
|
2025-09-28 15:28:15 +08:00
|
|
|
|
</div>
|
2025-09-27 17:58:15 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-09-27 16:10:17 +08:00
|
|
|
|
</template>
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
2025-09-27 17:58:15 +08:00
|
|
|
|
<script setup>
|
2025-09-28 15:28:15 +08:00
|
|
|
|
import { getRealtimeAlarm } from '@/api/homeIndex/index';
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
2025-09-29 11:27:36 +08:00
|
|
|
|
let alarmTimer = null;
|
2025-09-30 10:59:31 +08:00
|
|
|
|
const alarmData = ref([]);
|
|
|
|
|
|
const displayData = ref([]); // 显示的数据(复制一份用于无缝滚动)
|
|
|
|
|
|
const tableBody = ref(null);
|
|
|
|
|
|
const isScrolling = ref(true);
|
|
|
|
|
|
const animationId = ref(null);
|
|
|
|
|
|
const scrollPosition = ref(0);
|
|
|
|
|
|
const itemHeight = ref(0); // 动态计算每个条目的高度
|
|
|
|
|
|
|
|
|
|
|
|
// 获取报警事件名称
|
|
|
|
|
|
const getEventName = (action) => {
|
|
|
|
|
|
const eventMap = {
|
|
|
|
|
|
0: '强制报警',
|
|
|
|
|
|
1: '撞击闯入',
|
|
|
|
|
|
2: '自动报警',
|
|
|
|
|
|
3: '电子围栏告警'
|
|
|
|
|
|
};
|
|
|
|
|
|
return eventMap[action] || '未知报警';
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 获取实时报警数据
|
2025-09-28 15:28:15 +08:00
|
|
|
|
const getRealtimeAlarmData = () => {
|
|
|
|
|
|
getRealtimeAlarm().then((res) => {
|
2025-09-30 10:59:31 +08:00
|
|
|
|
if (res.data && res.data.length > 0) {
|
|
|
|
|
|
// 新数据插入到最前面
|
|
|
|
|
|
alarmData.value = [...res.data, ...alarmData.value];
|
|
|
|
|
|
|
|
|
|
|
|
// 限制数据量,避免性能问题
|
|
|
|
|
|
if (alarmData.value.length > 10) {
|
|
|
|
|
|
alarmData.value = alarmData.value.slice(0, 10);
|
|
|
|
|
|
}
|
|
|
|
|
|
// 更新显示数据
|
|
|
|
|
|
updateDisplayData();
|
|
|
|
|
|
// 新数据插入时重置滚动位置
|
|
|
|
|
|
scrollPosition.value = 0;
|
|
|
|
|
|
if (tableBody.value) {
|
|
|
|
|
|
tableBody.value.style.transform = `translateY(0)`;
|
|
|
|
|
|
}
|
|
|
|
|
|
}
|
2025-09-28 15:28:15 +08:00
|
|
|
|
});
|
2025-09-30 10:59:31 +08:00
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 更新显示数据(复制一份用于无缝滚动)
|
|
|
|
|
|
const updateDisplayData = () => {
|
|
|
|
|
|
displayData.value = [...alarmData.value];
|
|
|
|
|
|
// 计算每个条目的实际高度
|
|
|
|
|
|
nextTick(() => {
|
|
|
|
|
|
const items = document.querySelectorAll('.alarm-item');
|
|
|
|
|
|
if (items.length > 0) {
|
|
|
|
|
|
itemHeight.value = items[0].offsetHeight;
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 生成唯一的key
|
|
|
|
|
|
const getKey = (item, index) => {
|
|
|
|
|
|
return `${item.deviceImei}_${item.startTime}_${index}`;
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 开始滚动动画
|
|
|
|
|
|
const startScroll = () => {
|
|
|
|
|
|
if (!tableBody.value || displayData.value.length === 0) return;
|
|
|
|
|
|
stopScroll(); // 先停止之前的动画
|
|
|
|
|
|
const scrollSpeed = 0.3; // 滚动速度
|
|
|
|
|
|
const animate = () => {
|
|
|
|
|
|
if (!isScrolling.value) {
|
|
|
|
|
|
animationId.value = requestAnimationFrame(animate);
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
scrollPosition.value += scrollSpeed;
|
|
|
|
|
|
|
|
|
|
|
|
// 当滚动超过一份数据的高度时,重置位置
|
|
|
|
|
|
if (itemHeight.value > 0 && scrollPosition.value >= itemHeight.value * displayData.value.length) {
|
|
|
|
|
|
scrollPosition.value = 0;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
// 应用滚动效果
|
|
|
|
|
|
if (tableBody.value) {
|
|
|
|
|
|
tableBody.value.style.transform = `translateY(-${scrollPosition.value}px)`;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
animationId.value = requestAnimationFrame(animate);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
animationId.value = requestAnimationFrame(animate);
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 停止滚动动画
|
|
|
|
|
|
const stopScroll = () => {
|
|
|
|
|
|
if (animationId.value) {
|
|
|
|
|
|
cancelAnimationFrame(animationId.value);
|
|
|
|
|
|
animationId.value = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
// 开始报警定时器
|
2025-09-29 11:27:36 +08:00
|
|
|
|
const startAlarmTimer = () => {
|
|
|
|
|
|
if (alarmTimer) {
|
|
|
|
|
|
clearInterval(alarmTimer);
|
|
|
|
|
|
}
|
|
|
|
|
|
getRealtimeAlarmData();
|
2025-09-30 10:59:31 +08:00
|
|
|
|
alarmTimer = setInterval(getRealtimeAlarmData, 30 * 10000);
|
2025-09-29 11:27:36 +08:00
|
|
|
|
};
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
|
|
|
|
|
// 清除报警定时器
|
2025-09-29 11:27:36 +08:00
|
|
|
|
const clearAlarmTimer = () => {
|
|
|
|
|
|
if (alarmTimer) {
|
|
|
|
|
|
clearInterval(alarmTimer);
|
|
|
|
|
|
alarmTimer = null;
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
|
|
|
|
|
// 监听数据变化
|
|
|
|
|
|
watch(displayData, (newData) => {
|
|
|
|
|
|
if (newData.length > 0) {
|
|
|
|
|
|
// 延迟一点确保DOM已更新
|
|
|
|
|
|
setTimeout(() => {
|
|
|
|
|
|
startScroll();
|
|
|
|
|
|
}, 100);
|
|
|
|
|
|
}
|
|
|
|
|
|
});
|
2025-09-27 17:58:15 +08:00
|
|
|
|
|
|
|
|
|
|
onMounted(() => {
|
2025-09-29 11:27:36 +08:00
|
|
|
|
startAlarmTimer();
|
2025-09-27 17:58:15 +08:00
|
|
|
|
});
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
2025-09-29 11:27:36 +08:00
|
|
|
|
onUnmounted(() => {
|
2025-09-30 10:59:31 +08:00
|
|
|
|
clearAlarmTimer();
|
|
|
|
|
|
stopScroll();
|
|
|
|
|
|
});
|
2025-09-27 16:10:17 +08:00
|
|
|
|
</script>
|
2025-09-30 10:59:31 +08:00
|
|
|
|
|
2025-09-27 17:58:15 +08:00
|
|
|
|
<style scoped lang="scss">
|
|
|
|
|
|
.alarm-table-container {
|
|
|
|
|
|
width: 100%;
|
|
|
|
|
|
border-radius: 4px;
|
|
|
|
|
|
overflow: hidden;
|
|
|
|
|
|
margin-top: 4vh;
|
2025-09-29 14:06:18 +08:00
|
|
|
|
padding: 1vw 0vw;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.alarm-table-header {
|
|
|
|
|
|
display: flex;
|
2025-09-28 15:28:15 +08:00
|
|
|
|
color: #00C0FF;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
border-bottom: 1px dashed rgba(100, 150, 200, 0.3);
|
2025-09-28 15:28:15 +08:00
|
|
|
|
font-size: 0.8vw;
|
2025-09-29 14:06:18 +08:00
|
|
|
|
background-color: #0C2644;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.header-item {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
padding: 8px;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.alarm-table-body {
|
|
|
|
|
|
height: 18vh;
|
2025-09-30 10:59:31 +08:00
|
|
|
|
overflow: hidden;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
color: #fff;
|
2025-09-30 10:59:31 +08:00
|
|
|
|
position: relative;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-30 10:59:31 +08:00
|
|
|
|
.alarm-table-body-inner {
|
|
|
|
|
|
transition: transform 0.1s linear;
|
2025-09-29 09:31:03 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-27 17:58:15 +08:00
|
|
|
|
.alarm-item {
|
|
|
|
|
|
display: flex;
|
|
|
|
|
|
border-bottom: 1px dashed rgba(100, 150, 200, 0.3);
|
|
|
|
|
|
font-size: 0.6vw;
|
|
|
|
|
|
align-items: center;
|
2025-09-29 09:31:03 +08:00
|
|
|
|
padding: 0.8vw;
|
2025-09-30 10:59:31 +08:00
|
|
|
|
height: 7vh;
|
|
|
|
|
|
box-sizing: border-box;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.item-cell {
|
|
|
|
|
|
flex: 1;
|
|
|
|
|
|
text-align: center;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
.alarm-event {
|
|
|
|
|
|
color: #ff5252;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
2025-09-30 10:59:31 +08:00
|
|
|
|
/* 隐藏滚动条 */
|
|
|
|
|
|
.alarm-table-body::-webkit-scrollbar {
|
|
|
|
|
|
display: none;
|
|
|
|
|
|
}
|
2025-09-27 17:58:15 +08:00
|
|
|
|
|
2025-09-30 10:59:31 +08:00
|
|
|
|
.alarm-table-body {
|
|
|
|
|
|
-ms-overflow-style: none;
|
|
|
|
|
|
scrollbar-width: none;
|
2025-09-27 17:58:15 +08:00
|
|
|
|
}
|
|
|
|
|
|
</style>
|