feat(web): memory ui upgrade

This commit is contained in:
zhaoying
2026-03-16 15:10:55 +08:00
parent f09de3a11c
commit 88598fb9fb
26 changed files with 1732 additions and 1186 deletions

View File

@@ -1,7 +1,16 @@
/*
* @Author: ZhaoYing
* @Date: 2026-01-10 17:35:17
* @Last Modified by: ZhaoYing
* @Last Modified time: 2026-03-16 15:05:06
*/
import { type FC, useEffect, useState, useRef } from 'react'
import { useTranslation } from 'react-i18next'
import { useParams } from 'react-router-dom'
import { Skeleton, Row, Col } from 'antd'
import { Skeleton, Row, Col, Flex } from 'antd'
import * as echarts from 'echarts'
import 'echarts-wordcloud'
import RbCard from '@/components/RbCard/Card'
import {
getExplicitMemory,
@@ -10,40 +19,67 @@ import { formatDateTime } from '@/utils/format'
import Empty from '@/components/Empty'
import ExplicitDetailModal from '../components/ExplicitDetailModal'
/** An episodic (event-based) memory entry with a title and free-text content. */
export interface EpisodicMemory {
id: string;
title: string;
content: string;
created_at: number;
}
/** A semantic (concept-based) memory entry extracted as a named entity. */
export interface SemanticMemory {
id: string;
/** Entity name displayed in the word cloud. */
name: string;
/** Classification of the entity (e.g. person, location, concept). */
entity_type: string;
/** Brief definition or description of the entity. */
core_definition: string;
created_at: number;
}
/** Combined API response containing both memory categories. */
interface Data {
episodic_memories: EpisodicMemory[];
semantic_memories: SemanticMemory[]
}
/** Imperative handle exposed by ExplicitDetailModal for opening the detail drawer. */
export interface ExplicitDetailModalRef {
handleOpen: (vo: EpisodicMemory | SemanticMemory) => void;
}
/** Rotating colour palette used for word-cloud text. */
const DEFAULT_COLORS = ['#FF8A4C', '#FF5D34', '#155EEF', '#9C6FFF', '#4DA8FF', '#369F21']
/**
* ExplicitDetail Two-column view of a user's explicit memories.
*
* Left column: scrollable list of episodic memory cards (title + content).
* Right column: ECharts word cloud built from semantic memory entity names;
* clicking a word opens the detail modal.
*
* Route param `id` is the end-user ID whose memories are displayed.
*/
const ExplicitDetail: FC = () => {
const { t } = useTranslation()
const { id } = useParams()
const explicitDetailModalRef = useRef<ExplicitDetailModalRef>(null)
/** Container element for the ECharts word-cloud instance. */
const wordCloudRef = useRef<HTMLDivElement>(null)
/** Keeps a stable reference to the ECharts instance for cleanup. */
const chartInstance = useRef<echarts.ECharts | null>(null)
const [loading, setLoading] = useState<boolean>(false)
const [data, setData] = useState<Data>({ episodic_memories: [], semantic_memories: [] })
/* Fetch data whenever the route user ID changes. */
useEffect(() => {
if (!id) return
getData()
}, [id])
/** Load both episodic and semantic memories for the current user. */
const getData = () => {
if (!id) return
setLoading(true)
@@ -56,54 +92,98 @@ const ExplicitDetail: FC = () => {
setLoading(false)
})
}
/** Open the detail modal for a given memory item. */
const handleView = (item: EpisodicMemory | SemanticMemory) => {
explicitDetailModalRef.current?.handleOpen(item)
}
return (
<div className="rb:h-full rb:w-full">
<div className="rb:bg-[rgba(21,94,239,0.12)] rb:px-3 rb:py-2.5 rb:font-medium rb:leading-5 rb:mt-3 rb:rounded-md rb:mb-4">{t('explicitDetail.episodic_memories')}</div>
{loading ?
<Skeleton active />
: data.episodic_memories?.length > 0 ? (
<Row gutter={[16, 16]}>
{data.episodic_memories.map(item => (
<Col key={item.id} span={6}>
<RbCard
title={item.title}
className="rb:h-full! rb:cursor-pointer"
onClick={() => handleView(item)}
>
<div>{formatDateTime(item.created_at)}</div>
<div>{item.content}</div>
</RbCard>
</Col>
))}
</Row>
) : <Empty />}
<div className="rb:bg-[rgba(21,94,239,0.12)] rb:px-3 rb:py-2.5 rb:font-medium rb:leading-5 rb:mt-6 rb:rounded-md rb:mb-4">{t('explicitDetail.semantic_memories')}</div>
{loading ?
<Skeleton active />
: data.semantic_memories?.length > 0 ? (
<Row gutter={[16, 16]}>
{data.semantic_memories.map(item => (
<Col key={item.id} span={6}>
<RbCard
title={item.name}
className="rb:h-full! rb:cursor-pointer"
onClick={() => handleView(item)}
>
<div>{item.core_definition}</div>
</RbCard>
</Col>
))}
</Row>
) : <Empty />}
/**
* Initialise / re-render the word cloud whenever semantic memories change.
* Each word is clickable and opens the detail modal for that entity.
* The chart instance is disposed on cleanup to prevent memory leaks.
*/
useEffect(() => {
if (!wordCloudRef.current || !data.semantic_memories?.length) return
if (chartInstance.current) chartInstance.current.dispose()
chartInstance.current = echarts.init(wordCloudRef.current)
chartInstance.current.setOption({
series: [{
type: 'wordCloud',
gridSize: 8,
sizeRange: [14, 56],
rotationRange: [-45, 45],
shape: 'pentagon',
width: '100%',
height: '100%',
textStyle: { fontFamily: 'sans-serif', fontWeight: 'bold' },
emphasis: { textStyle: { shadowBlur: 10, shadowColor: '#333' } },
data: data.semantic_memories.map((item, index) => ({
name: item.name,
value: 50 + (index % 5) * 10,
itemIndex: index,
textStyle: { color: DEFAULT_COLORS[index % DEFAULT_COLORS.length] }
}))
}]
})
chartInstance.current.on('click', (params) => {
const item = data.semantic_memories[(params.data as any).itemIndex]
if (item) handleView(item)
})
return () => { chartInstance.current?.dispose(); chartInstance.current = null }
}, [data.semantic_memories])
return (
<Row gutter={12} className="rb:h-full">
<Col span={12}>
<RbCard
title={t('explicitDetail.episodic_memories')}
headerType="borderless"
headerClassName="rb:min-h-[50px]! rb:font-[MiSans-Bold] rb:font-bold"
bodyClassName="rb:p-3! rb:pt-0! rb:h-[calc(100%-50px)] rb:overflow-y-auto!"
className="rb:h-[calc(100vh-88px)]!"
>
{loading ?
<Skeleton active />
: data.episodic_memories?.length > 0 ? (
<Flex gap={12} vertical>
{data.episodic_memories.map(item => (
<div
key={item.id}
className="rb:cursor-pointer rb:bg-[#F6F6F6] rb:rounded-xl rb:pt-2.5 rb:px-3 rb:pb-3"
onClick={() => handleView(item)}
>
<Flex align="center" justify="space-between">
<span className="rb:font-medium rb:pl-1">{item.title}</span>
<div className="rb:textt-[#5B6167] rb:leading-4.25 rb:text-[12px]">{formatDateTime(item.created_at)}</div>
</Flex>
<div className="rb:bg-white rb:rounded-lg rb:py-2.5 rb:px-3 rb:mt-2.5 rb:leading-5">{item.content}</div>
</div>
))}
</Flex>
) : <Empty />
}
</RbCard>
</Col>
<Col span={12}>
<RbCard
title={t('explicitDetail.semantic_memories')}
headerType="borderless"
headerClassName="rb:min-h-[54px]! rb:font-[MiSans-Bold] rb:font-bold"
bodyClassName="rb:p-3! rb:pt-0! rb:h-[calc(100%-54px)] rb:overflow-y-auto!"
className="rb:h-[calc(100vh-88px)]!"
>
{loading ?
<Skeleton active />
: data.semantic_memories?.length > 0
? <div ref={wordCloudRef} className="rb:h-full rb:w-full rb:cursor-pointer" />
: <Empty />
}
</RbCard>
</Col>
<ExplicitDetailModal
ref={explicitDetailModalRef}
/>
</div>
</Row>
)
}
export default ExplicitDetail