This commit is contained in:
chatgpt-yunju 2025-07-08 08:48:05 +08:00
parent ddcba3b96e
commit f6105fb622
2 changed files with 521 additions and 0 deletions

View File

@ -0,0 +1,207 @@
<template>
<div class="right-panel">
<div class="key-node-recognition">
<div class="background-svg-wrapper">
<svg
width="100%"
height="100%"
viewBox="0 0 800 540"
preserveAspectRatio="none"
xmlns="http://www.w3.org/2000/svg"
>
<defs>
<linearGradient
id="paint0_linear_bg"
x1="0"
y1="167.1"
x2="800"
y2="167.1"
gradientUnits="userSpaceOnUse"
>
<stop stop-color="#063D71" stop-opacity="0.2" />
<stop offset="1" stop-color="#081E38" stop-opacity="0.8" />
</linearGradient>
<linearGradient
id="paint1_linear_border"
x1="400"
y1="0"
x2="400"
y2="540"
gradientUnits="userSpaceOnUse"
>
<stop stop-color="#3AA1F8" />
<stop offset="1" stop-color="#3AA1F8" stop-opacity="0.2" />
</linearGradient>
</defs>
<path
d="M798 0.5H2C1.17159 0.5 0.500003 1.17158 0.5 2V538C0.5 538.828 1.17159 539.5 2 539.5H798C798.828 539.5 799.5 538.828 799.5 538V2C799.5 1.17157 798.828 0.5 798 0.5Z"
fill="url(#paint0_linear_bg)"
fill-opacity="0.48"
stroke="url(#paint1_linear_border)"
/>
</svg>
</div>
<div class="content-wrapper">
<img src="@/assets/images/chuanboGraphTitle.png" alt="" />
<div class="chart-container">
<DynamicGraph
ref="leaderGraphRef"
:timestamp="store.activeTimePoint"
:allLeaderData="store.allLeaderData"
@handle:openDialog="handleGraphNodeClick"
/>
</div>
<div class="timeline-container">
<span class="time-label">2023.10.07 00:00:00</span>
<div class="timeline-track">
<div
v-for="point in store.timePoints"
:key="point.id"
class="timeline-point-wrapper"
@click="store.setActiveTimePoint(point.id)"
>
<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>
import { ref, defineExpose } from 'vue';
import { useKeyNodeStore } from '@/store/keyNodeStore';
import DynamicGraph from "./graph/dynamicGraph.vue";
const store = useKeyNodeStore();
const leaderGraphRef = ref(null);
const handleGraphNodeClick = (leaderData) => {
store.openLeaderDetail(leaderData);
};
const highlightNode = (leaderId) => {
if (leaderGraphRef.value) {
leaderGraphRef.value.highlightNode(leaderId);
}
};
defineExpose({ highlightNode });
</script>
<style scoped>
.right-panel {
flex-grow: 1;
position: relative;
}
.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 {
position: relative;
z-index: 2;
display: flex;
flex-direction: column;
height: 100%;
padding: 15px 20px;
box-sizing: border-box;
}
.chart-container {
flex-grow: 1;
width: 100%;
height: calc(100% - 100px);
}
.timeline-container {
width: 100%;
height: 50px;
display: flex;
align-items: center;
justify-content: space-between;
padding: 0 10px;
box-sizing: border-box;
background-color: rgba(4, 67, 92, 0.6);
border-radius: 5px;
z-index: 1;
}
.time-label {
font-size: 12px;
color: #a9c2e0;
}
.timeline-track {
flex-grow: 1;
height: 4px;
background: linear-gradient(90deg, #1b62a9, #3aa1f8, #1b62a9);
margin: 0 15px;
position: relative;
display: flex;
justify-content: space-between;
align-items: center;
}
.timeline-point-wrapper {
display: flex;
align-items: center;
justify-content: center;
height: 20px;
cursor: pointer;
}
.timeline-point {
width: 10px;
height: 10px;
background-color: #8dc5ff;
border-radius: 50%;
border: 1px solid #cce7ff;
transition: transform 0.3s ease;
position: relative;
}
.timeline-point-wrapper:hover .timeline-point {
transform: scale(1.5);
}
.timeline-point.active {
background-color: #ffc94d;
border-color: #fff;
transform: scale(1.3);
}
.active-pin {
width: 30px;
height: 34px;
background-image: url("@/assets/images/point.png");
background-size: cover;
bottom: 5px;
left: -11px;
position: absolute;
}
</style>

View File

@ -0,0 +1,314 @@
<template>
<div id="container"></div>
</template>
<script setup>
import * as echarts from "echarts";
import {
nextTick,
onMounted,
ref,
onUnmounted,
watch,
defineEmits,
defineExpose,
defineProps
} from "vue";
import { cropToCircleAsync } from "@/utils/transform";
import Cache from "@/utils/cache";
// --- 11 Follower ---
import follower1 from "@/assets/images/followers/Israel Defense Forces.png";
import follower2 from "@/assets/images/followers/Laura Loomer.png";
import follower3 from "@/assets/images/followers/🇺🇸 Mike Davis 🇺🇸.png";
import follower4 from "@/assets/images/followers/Rep. Carlos A. Gimenez.png";
import follower5 from "@/assets/images/followers/Israel War Room.png";
import follower6 from "@/assets/images/followers/Meme Knight 🇺🇸.png";
import follower7 from "@/assets/images/followers/AIPAC 🇺🇸🇮🇱🎗️.png";
import follower8 from "@/assets/images/followers/Sarah Larchmont.png";
import follower9 from "@/assets/images/followers/Faraz Pervaiz.png";
import follower10 from "@/assets/images/followers/Jamie Bryson.png";
import follower11 from "@/assets/images/followers/大桥_daqiao.png";
const emit = defineEmits(["handle:openDialog"]);
const props = defineProps({
timestamp: {
type: Number,
default: 0
},
allLeaderData: {
type: Array,
default: () => []
}
});
// --- Follower ---
const commonFollowers = [
{ name: "Israel Defense Forces", avatar: follower1 },
{ name: "Laura Loomer", avatar: follower2 },
{ name: "🇺🇸 Mike Davis 🇺🇸", avatar: follower3 },
{ name: "Rep. Carlos A. Gimenez", avatar: follower4 },
{ name: "Israel War Room", avatar: follower5 },
{ name: "Meme Knight 🇺🇸", avatar: follower6 },
{ name: "AIPAC 🇺🇸🇮🇱🎗️", avatar: follower7 },
{ name: "Sarah Larchmont", avatar: follower8 },
{ name: "Faraz Pervaiz", avatar: follower9 },
{ name: "Jamie Bryson", avatar: follower10 },
{ name: "大桥_daqiao", avatar: follower11 }
];
const allLeaderData = ref(props.allLeaderData);
const chart = ref(null);
const allGraphData = ref({ nodes: [], edges: [] });
const selectedLeaderId = ref(null);
const getCircleAvatar = async (avatarUrl) => {
const avatarCache = Cache.getItem(avatarUrl);
if (avatarCache) {
return avatarCache;
}
const base64 = await cropToCircleAsync(avatarUrl);
Cache.setItem(avatarUrl, base64); // avatarUrl key
return base64;
};
// timestamp
const initAllGraphData = async () => {
const nodes = [];
const edges = [];
// 使 .entries() leader
for (const [leaderIndex, leader] of allLeaderData.value.entries()) {
//
const avatar = await getCircleAvatar(leader.default_avatar);
nodes.push({
id: leader.id,
name: leader.name,
symbol: `image://${avatar}`,
symbolSize: 80,
category: 0,
value: leader.followers,
leaderOriginInfo: leader,
label: { show: false }
});
const userCount = Math.floor(Math.random() * 11) + 5; // 515
for (let i = 0; i < userCount; i++) {
const userId = `user_${leader.id}_${i}`;
// : 1010使
if (leaderIndex < 10 && i < 10) {
// commonFollowers[10]i<10
const followerData = commonFollowers[i];
const followerAvatar = await getCircleAvatar(followerData.avatar);
nodes.push({
id: userId,
name: followerData.name,
symbol: `image://${followerAvatar}`,
// --- 35 25 ---
symbolSize: 25,
category: 1,
value: "",
label: { show: false }
});
} else {
// 使
nodes.push({
id: userId,
name: `user ${i}`,
symbol: "circle",
symbolSize: 25,
category: 1,
value: "",
label: { show: false },
itemStyle: {
color: {
type: "linear",
colorStops: [
{ offset: 0, color: "#035e96" },
{ offset: 1, color: "#34a7b0" }
]
}
}
});
}
//
edges.push({
source: leader.id,
target: userId
});
}
}
//
const splatteringCount = Math.floor(Math.random() * 11) + 10; // 10~20
for (let i = 0; i < splatteringCount; i++) {
const userId = `user_splattering_${i}`;
nodes.push({
id: userId,
name: `user ${i}`,
symbol: "circle",
symbolSize: 25,
category: 1,
value: "",
label: { show: false },
itemStyle: {
color: {
type: "linear",
colorStops: [
{ offset: 0, color: "#035e96" },
{ offset: 1, color: "#34a7b0" }
]
}
}
});
}
allGraphData.value = { nodes, edges };
console.log(allGraphData.value);
};
// timestamp
const getVisibleGraphData = () => {
const leaders = allLeaderData.value.slice(0, props.timestamp);
const leaderIds = new Set(leaders.map((l) => l.id));
const nodes = [];
const edges = [];
for (const node of allGraphData.value.nodes) {
let shouldShow = false;
if (leaderIds.has(node.id) || node.id.startsWith("user_splattering_")) {
shouldShow = true;
} else if (node.id.startsWith("user_")) {
const parts = node.id.split("_");
// Correctly handle parent IDs that contain underscores (e.g., 'levi_godman')
const parentId = parts.slice(1, parts.length - 1).join("_");
if (leaderIds.has(parentId)) {
shouldShow = true;
}
}
if (shouldShow) {
nodes.push(node);
}
}
for (const edge of allGraphData.value.edges) {
if (leaderIds.has(edge.source)) {
edges.push(edge);
}
}
return { nodes, edges };
};
const renderChart = () => {
if (!chart.value) {
chart.value = echarts.init(document.getElementById("container"));
}
const { nodes, edges } = getVisibleGraphData();
chart.value.setOption({
tooltip: {
formatter: (params) => {
if (params.data.category === 0) {
return `Name: ${params.data.name}<br>Followers: ${params.data.value}`;
}
return params.data.name;
}
},
series: [
{
type: "graph",
layout: "force",
roam: true,
draggable: true,
data: nodes.map((node) => {
let symbol = node.symbol;
if (node.category === 0) {
//
symbol =
selectedLeaderId.value === node.id
? `image://${node.leaderOriginInfo.active_avatar}`
: `image://${node.leaderOriginInfo.default_avatar}`;
}
return {
...node,
symbol,
itemStyle: {
...node.itemStyle,
shadowBlur: 10,
shadowColor: "rgba(0,207,255,0.5)",
borderWidth: 1,
borderColor: "#32c6fc"
},
emphasis: {
itemStyle: {
shadowBlur: 20,
shadowColor: "#c4a651",
borderColor: "#fcd267",
borderWidth: node.category === 0 ? 10 : 2,
borderType: "solid"
}
},
symbolKeepAspect: true
};
}),
links: edges,
categories: [{ name: "Leader" }, { name: "User" }],
force: {
repulsion: 100,
edgeLength: 80
},
label: {
position: "center"
}
}
]
});
};
const handleClickNode = () => {
chart.value.on("click", (params) => {
if (params.data && params.data.category === 0) {
selectedLeaderId.value = params.data.id;
renderChart();
emit("handle:openDialog", params.data);
}
});
};
onMounted(async () => {
await nextTick();
await initAllGraphData();
renderChart();
handleClickNode();
});
watch(
() => props.timestamp,
() => {
renderChart();
}
);
onUnmounted(() => {
if (chart.value) {
chart.value.dispose();
}
});
// chart
defineExpose({
chart, // chart
//
highlightNode(leaderId) {
if (chart.value) {
selectedLeaderId.value = leaderId;
renderChart();
}
}
});
</script>
<style scoped>
#container {
width: 100%;
height: 100%;
}
</style>