import React, { type FC, useEffect, useState, useRef, useCallback } from 'react' import { useTranslation } from 'react-i18next' import { useParams, useNavigate } from 'react-router-dom' import { Col, Row, Space, Button } from 'antd' import dayjs from 'dayjs' import RbCard from '@/components/RbCard/Card' import ReactEcharts from 'echarts-for-react' import detailEmpty from '@/assets/images/userMemory/detail_empty.png' import type { Node, Edge, GraphData, StatementNodeProperties, ExtractedEntityNodeProperties } from '../types' import { getMemorySearchEdges, } from '@/api/memory' import Empty from '@/components/Empty' import Tag from '@/components/Tag' const colors = ['#155EEF', '#369F21', '#4DA8FF', '#FF5D34', '#9C6FFF', '#FF8A4C', '#8BAEF7', '#FFB048'] const RelationshipNetwork:FC = () => { const { t } = useTranslation() const { id } = useParams() const chartRef = useRef(null) const resizeScheduledRef = useRef(false) const [nodes, setNodes] = useState([]) const [links, setLinks] = useState([]) const [categories, setCategories] = useState<{ name: string }[]>([]) const [selectedNode, setSelectedNode] = useState(null) // const [fullScreen, setFullScreen] = useState(false) const navigate = useNavigate() console.log('categories', categories) // 关系网络 const getEdgeData = useCallback(() => { if (!id) return setSelectedNode(null) getMemorySearchEdges(id).then((res) => { const { nodes, edges, statistics } = res as GraphData const curNodes: Node[] = [] const curEdges: Edge[] = [] const curNodeTypes = Object.keys(statistics.node_types).filter(vo => vo !== 'Dialogue') // 计算每个节点的连接数 const connectionCount: Record = {} edges.forEach(edge => { connectionCount[edge.source] = (connectionCount[edge.source] || 0) + 1 connectionCount[edge.target] = (connectionCount[edge.target] || 0) + 1 }) // 处理节点数据 nodes.filter(vo => vo.label !== 'Dialogue').forEach(node => { const connections = connectionCount[node.id] || 0 const categoryIndex = curNodeTypes.indexOf(node.label) // 根据节点类型获取显示名称 let displayName = '' switch (node.label) { // case 'Statement': // displayName = 'statement' in node.properties ? node.properties.statement?.slice(0, 5) || '' : '' // break case 'ExtractedEntity': displayName = 'name' in node.properties ? node.properties.name || '' : '' break // default: // displayName = 'content' in node.properties ? node.properties.content?.slice(0, 5) || '' : '' // break } let symbolSize = 0 if (connections <= 1) { symbolSize = 5 } else if (connections <= 10) { symbolSize = 10 } else if (connections <= 15) { symbolSize = 15 } else if (connections <= 20) { symbolSize = 25 } else { symbolSize = 35 } curNodes.push({ ...node, name: displayName, category: categoryIndex >= 0 ? categoryIndex : 0, symbolSize: symbolSize, // 根据连接数调整节点大小 itemStyle: { color: colors[categoryIndex % 8] } }) }) // 处理边数据 edges.forEach(edge => { curEdges.push({ ...edge, source: edge.source, target: edge.target, value: edge.weight || 1 }) }) // 设置分类 const curCategories = curNodeTypes.map(type => ({ name: type })) setNodes(curNodes) setLinks(curEdges) setCategories(curCategories) }) }, [id]) useEffect(() => { if (!id) return getEdgeData() }, [id]) useEffect(() => { const handleResize = () => { if (chartRef.current && !resizeScheduledRef.current) { resizeScheduledRef.current = true requestAnimationFrame(() => { chartRef.current?.getEchartsInstance().resize(); resizeScheduledRef.current = false }); } } const resizeObserver = new ResizeObserver(handleResize) const chartElement = chartRef.current?.getEchartsInstance().getDom().parentElement if (chartElement) { resizeObserver.observe(chartElement) } return () => { resizeObserver.disconnect() } }, [nodes]) const handleViewAll = () => { if (!selectedNode) return const params = new URLSearchParams({ nodeId: selectedNode.id, nodeLabel: selectedNode.label, nodeName: selectedNode.name || '' }) navigate(`/user-memory/detail/${id}/GRAPH?${params.toString()}`) } return ( {/* 关系网络 */} //
// {t('userMemory.fullScreen')} // // } >
{nodes.length === 0 ? ( ) : ( ({ name: t(`userMemory.${vo.name}`) })) || [], roam: true, label: { show: true, position: 'right', formatter: '{b}', }, lineStyle: { color: '#5B6167', curveness: 0.3 }, force: { repulsion: 100, // 启用类别聚合 edgeLength: 80, gravity: 0.3, // 同类别的节点相互吸引 layoutAnimation: true, // 防止点击时重新计算布局 preventOverlap: true, // 点击节点后保持布局稳定 edgeSymbol: ['none', 'arrow'], edgeSymbolSize: [4, 10], // 初始布局完成后关闭力导向 initLayout: 'force' }, selectedMode: 'single', draggable: true, // 防止数据更新时重新计算布局 animationDurationUpdate: 0, select: { itemStyle: { borderWidth: 2, borderColor: '#ffffff', shadowBlur: 10, } } } ] }} style={{ height: '518px', width: '100%' }} notMerge={false} lazyUpdate={true} onEvents={{ // 节点点击事件处理 click: (params: { dataType: string; data: Node; name: string }) => { if (params.dataType === 'node') { // 处理节点点击事件 console.log('Node clicked:', params.data); // 使用函数式更新避免状态依赖问题 setSelectedNode(params.data) } } }} /> )}
{/* 记忆详情 */}
{t('userMemory.completeMemory')} } >
{!selectedNode ? : <>
{selectedNode.name}
<>
{t('userMemory.memoryContent')}
{['Chunk', 'Dialogue', 'MemorySummary'].includes(selectedNode.label) && 'content' in selectedNode.properties ? selectedNode.properties.content : selectedNode.label === 'ExtractedEntity' && 'description' in selectedNode.properties ? selectedNode.properties.description : selectedNode.label === 'Statement' && 'statement' in selectedNode.properties ? selectedNode.properties.statement : '' }
{t('userMemory.created_at')}
{dayjs(selectedNode?.properties.created_at).format('YYYY-MM-DD HH:mm:ss')}
{selectedNode?.properties.associative_memory > 0 &&
{t('userMemory.associative_memory')}
{selectedNode?.properties.associative_memory} {t('userMemory.unix')}{t('userMemory.associative_memory')}
} {selectedNode.label === 'Statement' && <> {(['emotion_keywords', 'emotion_type', 'emotion_subject', 'importance_score'] as const).map(key => { const statementProps = selectedNode.properties as StatementNodeProperties; if ((key === 'emotion_keywords' && statementProps[key]?.length > 0) || statementProps[key]) { return (
{t(`userMemory.Statement_${key}`)}
{key === 'emotion_keywords' ? {statementProps.emotion_keywords.map((vo, index) => {vo})} : statementProps[key] }
) } return null })} } {selectedNode.label === 'ExtractedEntity' && <> {(['name', 'entity_type', 'aliases', 'connect_strngth', 'importance_score'] as const).map(key => { const entityProps = selectedNode.properties as ExtractedEntityNodeProperties; if (entityProps[key]) { return (
{t(`userMemory.ExtractedEntity_${key}`)}
{entityProps[key]}
) } return null })} }
}
) } // 使用React.memo包装组件,避免不必要的渲染 export default React.memo(RelationshipNetwork)