SocialNetworks_duan/src/views/KeyNodeDiscern/bridgeCommunication/components/GraphPanel.vue

302 lines
8.0 KiB
Vue
Raw Normal View History

2025-07-17 10:28:56 +08:00
<template>
<div class="right-panel">
<img src="@/assets/images/head/bayi.png" alt="" class="head" />
2025-07-17 10:28:56 +08:00
<div class="key-node-recognition">
<div class="content-wrapper">
<div class="chart-container">
<BridgeCommunityGraph
ref="bridgeGraphRef"
v-if="currentComponent == 'BridgeCommunityNode'"
2025-07-17 10:28:56 +08:00
:timestamp="store.activeTimePoint"
@click:navigateToCommunityDetail="handleClickCommunity"
2025-07-17 10:28:56 +08:00
/>
<DetailCommunityGraph
v-else
ref="detailCommunityGraphRef"
2025-07-25 15:26:19 +08:00
:communityId="currentSelectedCommunityId"
@click:goback="handleClickGoBack"
/>
2025-07-17 10:28:56 +08:00
</div>
<div class="timeline-container">
<span class="time-label">2023.10.07 00:00:00</span>
2025-07-18 14:29:29 +08:00
<div class="timeline-track" :style="trackStyle">
2025-07-17 10:28:56 +08:00
<div
v-for="point in store.timePoints"
:key="point.id"
class="timeline-point-wrapper"
2025-07-18 14:29:29 +08:00
:style="{ left: `${pointPositions[point.id]}%` }"
2025-08-15 16:11:11 +08:00
@click="handleTimePointClick(point.id)"
2025-07-17 10:28:56 +08:00
>
<el-tooltip
class="timePoint-box-item"
effect="light"
:content="point.timestamp"
placement="bottom"
>
<div class="timeline-point" :class="{ active: store.activeTimePoint === point.id }">
<el-popover
v-if="store.activeTimePoint === point.id"
effect="dark"
placement="top"
:title="point.leaderId"
:width="50"
trigger="click"
content="发布贴文"
>
<template #reference>
<div class="active-pin"></div>
</template>
</el-popover>
</div>
</el-tooltip>
</div>
</div>
<span class="time-label">2023.10.15 00:00:00</span>
</div>
</div>
</div>
</div>
</template>
<script setup>
2025-08-15 16:25:30 +08:00
import { ref, defineExpose, watch, computed, defineProps } from "vue"
import { useKeyNodeStore2 } from "@/store/keyNodeStore2"
2025-07-17 10:28:56 +08:00
2025-08-15 16:11:11 +08:00
// 接收 stopAutomaticPlay 方法
const props = defineProps({
stopAutomaticPlay: { type: Function, required: true }
})
import BridgeCommunityGraph from "./graph/bridgeCommunityGraph.vue"
import DetailCommunityGraph from "./graph/detailCommunityGraph.vue"
2025-07-17 10:28:56 +08:00
const store = useKeyNodeStore2()
const leaderGraphRef = ref(null)
2025-07-24 17:02:04 +08:00
// 添加对BridgeCommunityGraph的引用
const bridgeGraphRef = ref(null)
2025-07-25 15:26:19 +08:00
// 设置当前激活的组件--默认是BridgeCommunityGraph
const currentComponent = ref("BridgeCommunityNode")
const currentSelectedCommunityId = ref(null)
2025-07-17 10:28:56 +08:00
const handleGraphNodeClick = (leaderData) => {
store.openLeaderDetail(leaderData)
}
2025-07-17 10:28:56 +08:00
2025-07-25 15:26:19 +08:00
const handleClickCommunity = (communityId) => {
if (communityId) {
currentComponent.value = "DetailCommunityNode"
currentSelectedCommunityId.value = communityId
2025-07-25 15:26:19 +08:00
}
}
const handleClickGoBack = (currentComponentName) => {
currentComponent.value = currentComponentName
}
2025-07-25 15:26:19 +08:00
2025-07-17 10:28:56 +08:00
const highlightNode = (leaderId) => {
if (leaderGraphRef.value) {
leaderGraphRef.value.highlightNode(leaderId)
2025-07-17 10:28:56 +08:00
}
2025-07-24 17:02:04 +08:00
// 调用BridgeCommunityGraph的highlightNode方法
if (bridgeGraphRef.value) {
bridgeGraphRef.value.highlightNode(leaderId)
2025-07-24 17:02:04 +08:00
}
}
2025-07-18 14:29:29 +08:00
// 计算时间点在时间轴上的位置百分比
const calculatePositions = (timestamp) => {
// 时间范围: 2023-10-07T00:00:00 到 2023-10-15T00:00:00
const startTime = new Date("2023-10-07T00:00:00").getTime()
const endTime = new Date("2023-10-15T00:00:00").getTime()
const timeRange = endTime - startTime
2025-07-18 14:29:29 +08:00
// 复制并排序时间点数据
const sortedPoints = [...store.timePoints].sort((a, b) => {
return new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime()
})
2025-07-18 14:29:29 +08:00
// 计算每个点的原始位置
const pointPositions = {}
sortedPoints.forEach((point) => {
const pointTime = new Date(point.timestamp).getTime()
const position = ((pointTime - startTime) / timeRange) * 100
pointPositions[point.id] = Math.max(0, Math.min(100, position))
})
2025-07-18 14:29:29 +08:00
// 处理重叠 (假设每个点宽度约为20px时间轴总宽度为可用宽度)
const pointWidthPercentage = 5 // 估算的点宽度百分比
const minSpacing = pointWidthPercentage // 最小间距百分比
2025-07-18 14:29:29 +08:00
// 调整重叠的点
for (let i = 1; i < sortedPoints.length; i++) {
const prevPoint = sortedPoints[i - 1]
const currPoint = sortedPoints[i]
2025-07-18 14:29:29 +08:00
const prevPosition = pointPositions[prevPoint.id]
const currPosition = pointPositions[currPoint.id]
2025-07-18 14:29:29 +08:00
// 检查是否重叠
if (currPosition - prevPosition < minSpacing) {
// 调整当前点位置
pointPositions[currPoint.id] = prevPosition + minSpacing
2025-07-18 14:29:29 +08:00
// 确保不超出范围
if (pointPositions[currPoint.id] > 100) {
pointPositions[currPoint.id] = 100
2025-07-18 14:29:29 +08:00
}
}
}
return pointPositions
2025-07-18 14:29:29 +08:00
}
// 响应式引用存储位置信息
const pointPositions = ref({})
2025-07-18 14:29:29 +08:00
// 监听时间点变化,重新计算位置
watch(
() => store.timePoints,
() => {
pointPositions.value = calculatePositions()
2025-07-18 14:29:29 +08:00
},
{ immediate: true, deep: true }
)
2025-07-18 14:29:29 +08:00
// 计算时间轴轨道样式
const trackStyle = computed(() => {
if (!store.activeTimePoint) return {}
const activePosition = pointPositions.value[store.activeTimePoint] || 0
2025-07-18 14:29:29 +08:00
return {
background: `linear-gradient(90deg, #3B7699 0%, #00F3FF ${activePosition}%, #3B7699 ${activePosition}%, #3B7699 100%)`
}
})
2025-07-18 14:29:29 +08:00
defineExpose({ highlightNode })
2025-08-15 16:11:11 +08:00
// 添加时间点点击事件处理函数
const handleTimePointClick = (pointId) => {
// 停止自动播放
props.stopAutomaticPlay()
// 设置当前激活的时间点
store.setActiveTimePoint(pointId)
}
2025-07-17 10:28:56 +08:00
</script>
2025-08-19 11:23:46 +08:00
<style scoped lang="scss">
2025-07-17 10:28:56 +08:00
.right-panel {
width: 100%;
height: 100%;
2025-07-17 10:28:56 +08:00
position: relative;
}
.head {
margin: 0 auto;
}
2025-07-17 10:28:56 +08:00
.key-node-recognition {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
}
.background-svg-wrapper {
position: absolute;
top: 0;
left: 0;
width: 100%;
height: 100%;
z-index: 1;
}
.content-wrapper {
width: 100%;
2025-07-17 10:28:56 +08:00
position: relative;
z-index: 2;
display: flex;
flex-direction: column;
height: 100%;
2025-07-17 10:28:56 +08:00
box-sizing: border-box;
2025-08-13 16:43:36 +08:00
.graph-img-title {
display: absolute;
top: 0;
margin: 0 auto;
}
2025-07-17 10:28:56 +08:00
}
.chart-container {
width: 100%;
height: 100%;
margin-top: vh(35);
2025-07-17 10:28:56 +08:00
}
.timeline-container {
width: 95%;
height: vh(40);
2025-07-17 10:28:56 +08:00
display: flex;
align-items: center;
justify-content: space-between;
padding: vh(0) vw(10);
2025-07-17 10:28:56 +08:00
box-sizing: border-box;
position: absolute;
2025-07-17 10:28:56 +08:00
z-index: 1;
bottom: vh(15);
left: vw(20);
border-radius: vw(4);
border: 1px solid #3aa1f8;
background: linear-gradient(270deg, rgba(0, 82, 125, 0.48) 0%, rgba(0, 200, 255, 0.23) 100%);
backdrop-filter: blur(3px);
2025-07-17 10:28:56 +08:00
}
.time-label {
font-size: vw(12);
2025-07-17 10:28:56 +08:00
color: #a9c2e0;
}
.timeline-track {
flex-grow: 1;
height: vh(4);
margin: vh(0) vw(15);
2025-07-17 10:28:56 +08:00
position: relative;
display: flex;
align-items: center;
}
.timeline-point-wrapper {
display: flex;
align-items: center;
justify-content: center;
height: vh(20);
2025-07-17 10:28:56 +08:00
cursor: pointer;
2025-07-18 14:29:29 +08:00
position: absolute;
transform: translateX(-50%);
2025-07-17 10:28:56 +08:00
}
.timeline-point {
width: vw(18);
height: vh(18);
2025-07-18 14:29:29 +08:00
background-color: transparent;
2025-07-17 10:28:56 +08:00
border-radius: 50%;
border: 1.6px solid #ffe5a4;
2025-07-17 10:28:56 +08:00
transition: transform 0.3s ease;
position: relative;
2025-07-18 14:29:29 +08:00
display: flex;
align-items: center;
justify-content: center;
2025-07-17 10:28:56 +08:00
}
2025-07-18 14:29:29 +08:00
.timeline-point::after {
content: "";
width: vw(10);
height: vh(10);
background-color: #f9bd25;
2025-07-18 14:29:29 +08:00
border-radius: 50%;
position: absolute;
}
2025-07-17 10:28:56 +08:00
.timeline-point-wrapper:hover .timeline-point {
transform: scale(1.5);
}
.timeline-point.active {
2025-07-18 14:29:29 +08:00
background-color: transparent;
border-color: #ffe5a4;
2025-07-17 10:28:56 +08:00
transform: scale(1.3);
}
.active-pin {
width: vw(30);
height: vh(34);
2025-07-17 10:28:56 +08:00
background-image: url("@/assets/images/point.png");
background-size: cover;
bottom: vh(6);
2025-07-17 10:28:56 +08:00
position: absolute;
}
</style>