Files
zclaw_openfang/desktop/src/components/MemoryPanel.tsx
iven 5c74e74f2a fix(desktop): component cleanup + dead code removal + DeerFlow ai-elements
- ChatArea: DeerFlow ai-elements annotations for accessibility
- Conversation: remove unused Context, simplify message rendering
- Delete dead modules: audit-logger.ts, gateway-reconnect.ts
- Replace console.log with structured logger across components
- Add idb dependency for IndexedDB persistence
- Fix kernel-skills type safety improvements
2026-04-03 00:28:58 +08:00

380 lines
13 KiB
TypeScript

import { useCallback, useEffect, useState } from 'react';
import { motion, AnimatePresence } from 'framer-motion';
import {
Brain, Search, Trash2, Download, Star, Tag, Clock,
ChevronDown, ChevronUp,
} from 'lucide-react';
import { cardHover, defaultTransition } from '../lib/animations';
import { Button, Badge, EmptyState } from './ui';
import {
intelligenceClient,
type MemoryEntry,
type MemoryType,
type MemoryStats,
} from '../lib/intelligence-client';
import { useConversationStore } from '../store/chat/conversationStore';
const TYPE_LABELS: Record<MemoryType, { label: string; emoji: string; color: string }> = {
fact: { label: '事实', emoji: '📋', color: 'bg-blue-100 text-blue-700 dark:bg-blue-900/30 dark:text-blue-300' },
preference: { label: '偏好', emoji: '⭐', color: 'bg-amber-100 text-amber-700 dark:bg-amber-900/30 dark:text-amber-300' },
lesson: { label: '经验', emoji: '💡', color: 'bg-green-100 text-green-700 dark:bg-green-900/30 dark:text-green-300' },
context: { label: '上下文', emoji: '📌', color: 'bg-purple-100 text-purple-700 dark:bg-purple-900/30 dark:text-purple-300' },
task: { label: '任务', emoji: '📝', color: 'bg-red-100 text-red-700 dark:bg-red-900/30 dark:text-red-300' },
};
export function MemoryPanel() {
const currentAgent = useConversationStore((s) => s.currentAgent);
const agentId = currentAgent?.id || 'zclaw-main';
const [memories, setMemories] = useState<MemoryEntry[]>([]);
const [stats, setStats] = useState<MemoryStats | null>(null);
const [searchQuery, setSearchQuery] = useState('');
const [filterType, setFilterType] = useState<MemoryType | 'all'>('all');
const [expandedId, setExpandedId] = useState<string | null>(null);
const [isExporting, setIsExporting] = useState(false);
const loadMemories = useCallback(async () => {
const typeFilter = filterType !== 'all' ? { type: filterType as MemoryType } : {};
if (searchQuery.trim()) {
const results = await intelligenceClient.memory.search({
agentId,
query: searchQuery,
limit: 50,
...typeFilter,
});
setMemories(results);
} else {
const results = await intelligenceClient.memory.search({
agentId,
limit: 50,
...typeFilter,
});
setMemories(results);
}
const s = await intelligenceClient.memory.stats();
setStats(s);
}, [agentId, searchQuery, filterType]);
useEffect(() => {
loadMemories();
}, [loadMemories]);
const handleDelete = async (id: string) => {
await intelligenceClient.memory.delete(id);
loadMemories();
};
const handleExport = async () => {
setIsExporting(true);
try {
const memories = await intelligenceClient.memory.export();
const filtered = memories.filter(m => m.agentId === agentId);
const md = filtered.map(m =>
`## [${m.type}] ${m.content}\n` +
`- 重要度: ${m.importance}\n` +
`- 标签: ${m.tags.join(', ') || '无'}\n` +
`- 创建时间: ${m.createdAt}\n`
).join('\n---\n\n');
const blob = new Blob([md || '# 无记忆数据'], { type: 'text/markdown' });
const url = URL.createObjectURL(blob);
const a = document.createElement('a');
a.href = url;
a.download = `zclaw-memory-${agentId}-${new Date().toISOString().slice(0, 10)}.md`;
a.click();
URL.revokeObjectURL(url);
} finally {
setIsExporting(false);
}
};
const handlePrune = async () => {
// Find old, low-importance memories and delete them
const memories = await intelligenceClient.memory.search({
agentId,
minImportance: 0,
limit: 1000,
});
const thirtyDaysAgo = Date.now() - 30 * 24 * 60 * 60 * 1000;
const toDelete = memories.filter(m =>
new Date(m.createdAt).getTime() < thirtyDaysAgo && m.importance < 3
);
for (const m of toDelete) {
await intelligenceClient.memory.delete(m.id);
}
if (toDelete.length > 0) {
loadMemories();
}
};
return (
<div className="space-y-3">
{/* Stats */}
<motion.div
whileHover={cardHover}
transition={defaultTransition}
className="rounded-xl border border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800 p-3 shadow-sm"
>
<div className="flex items-center justify-between mb-2">
<h3 className="text-sm font-semibold text-gray-900 dark:text-gray-100 flex items-center gap-1.5">
<Brain className="w-4 h-4 text-orange-500" />
Agent
</h3>
<div className="flex gap-1">
<Button
variant="ghost"
size="sm"
onClick={handleExport}
disabled={isExporting}
title="导出记忆"
aria-label="Export memories"
className="p-1"
>
<Download className="w-3.5 h-3.5" />
</Button>
<Button
variant="ghost"
size="sm"
onClick={handlePrune}
title="清理旧记忆"
aria-label="Prune old memories"
className="p-1"
>
<Trash2 className="w-3.5 h-3.5" />
</Button>
</div>
</div>
{stats && (
<div className="grid grid-cols-3 gap-2 text-center">
<div className="bg-gray-50 dark:bg-gray-700/50 rounded-lg p-1.5">
<div className="text-lg font-bold text-orange-600">{stats.totalEntries}</div>
<div className="text-[10px] text-gray-500"></div>
</div>
<div className="bg-gray-50 dark:bg-gray-700/50 rounded-lg p-1.5">
<div className="text-lg font-bold text-blue-600">{stats.byType['fact'] || 0}</div>
<div className="text-[10px] text-gray-500"></div>
</div>
<div className="bg-gray-50 dark:bg-gray-700/50 rounded-lg p-1.5">
<div className="text-lg font-bold text-amber-600">{stats.byType['preference'] || 0}</div>
<div className="text-[10px] text-gray-500"></div>
</div>
</div>
)}
</motion.div>
{/* Search & Filter */}
<div className="space-y-2">
<div className="relative">
<Search className="absolute left-2.5 top-1/2 -translate-y-1/2 w-3.5 h-3.5 text-gray-400" />
<input
type="text"
value={searchQuery}
onChange={(e) => setSearchQuery(e.target.value)}
placeholder="搜索记忆..."
className="w-full text-sm border border-gray-200 dark:border-gray-600 rounded-lg pl-8 pr-3 py-1.5 bg-white dark:bg-gray-800 text-gray-900 dark:text-gray-100 focus:outline-none focus:ring-1 focus:ring-orange-400"
/>
</div>
<div className="flex gap-1 flex-wrap">
<FilterChip
label="全部"
active={filterType === 'all'}
onClick={() => setFilterType('all')}
/>
{(Object.keys(TYPE_LABELS) as MemoryType[]).map((type) => (
<FilterChip
key={type}
label={`${TYPE_LABELS[type].emoji} ${TYPE_LABELS[type].label}`}
active={filterType === type}
onClick={() => setFilterType(type)}
/>
))}
</div>
</div>
{/* Memory List */}
<div className="space-y-2">
{memories.length > 0 ? (
<AnimatePresence initial={false}>
{memories.map((entry) => (
<MemoryCard
key={entry.id}
entry={entry}
expanded={expandedId === entry.id}
onToggle={() => setExpandedId(expandedId === entry.id ? null : entry.id)}
onDelete={() => handleDelete(entry.id)}
/>
))}
</AnimatePresence>
) : (
<EmptyState
icon={<Brain className="w-8 h-8" />}
title={searchQuery ? '未找到匹配的记忆' : '暂无记忆'}
description={searchQuery ? '尝试不同的搜索词' : '与 Agent 交流后,记忆会自动积累'}
className="py-6"
/>
)}
</div>
</div>
);
}
function MemoryCard({
entry,
expanded,
onToggle,
onDelete,
}: {
entry: MemoryEntry;
expanded: boolean;
onToggle: () => void;
onDelete: () => void;
}) {
const typeInfo = TYPE_LABELS[entry.type];
const importanceStars = Math.round(entry.importance / 2);
const timeAgo = getTimeAgo(entry.createdAt);
return (
<motion.div
layout
initial={{ opacity: 0, y: 8 }}
animate={{ opacity: 1, y: 0 }}
exit={{ opacity: 0, y: -8 }}
className="rounded-lg border border-gray-200 dark:border-gray-700 bg-white dark:bg-gray-800 shadow-sm overflow-hidden"
>
<div
className="px-3 py-2 cursor-pointer hover:bg-gray-50 dark:hover:bg-gray-700/50 transition-colors"
onClick={onToggle}
>
<div className="flex items-start gap-2">
<span className={`text-[10px] font-medium px-1.5 py-0.5 rounded-full ${typeInfo.color}`}>
{typeInfo.emoji} {typeInfo.label}
</span>
<div className="flex-1 min-w-0">
<p className="text-sm text-gray-800 dark:text-gray-200 line-clamp-2">{entry.content}</p>
</div>
{expanded ? (
<ChevronUp className="w-3.5 h-3.5 text-gray-400 flex-shrink-0 mt-0.5" />
) : (
<ChevronDown className="w-3.5 h-3.5 text-gray-400 flex-shrink-0 mt-0.5" />
)}
</div>
<div className="flex items-center gap-3 mt-1.5 text-[10px] text-gray-500 dark:text-gray-400">
<span className="flex items-center gap-0.5">
<Star className="w-3 h-3" />
{'★'.repeat(importanceStars)}{'☆'.repeat(5 - importanceStars)}
</span>
<span className="flex items-center gap-0.5">
<Clock className="w-3 h-3" />
{timeAgo}
</span>
{entry.tags.length > 0 && (
<span className="flex items-center gap-0.5">
<Tag className="w-3 h-3" />
{entry.tags.slice(0, 2).join(', ')}
</span>
)}
</div>
</div>
<AnimatePresence>
{expanded && (
<motion.div
initial={{ height: 0, opacity: 0 }}
animate={{ height: 'auto', opacity: 1 }}
exit={{ height: 0, opacity: 0 }}
transition={{ duration: 0.2 }}
className="border-t border-gray-100 dark:border-gray-700"
>
<div className="px-3 py-2 space-y-2 text-xs">
<div className="grid grid-cols-2 gap-2">
<div>
<span className="text-gray-500"></span>
<span className="ml-1 font-medium">{entry.importance}/10</span>
</div>
<div>
<span className="text-gray-500"></span>
<span className="ml-1 font-medium">{entry.source === 'auto' ? '自动' : entry.source === 'user' ? '用户' : '反思'}</span>
</div>
<div>
<span className="text-gray-500">访</span>
<span className="ml-1 font-medium">{entry.accessCount}</span>
</div>
<div>
<span className="text-gray-500"></span>
<span className="ml-1 font-medium">{new Date(entry.createdAt).toLocaleDateString('zh-CN')}</span>
</div>
</div>
{entry.tags.length > 0 && (
<div className="flex flex-wrap gap-1">
{entry.tags.map((tag) => (
<Badge key={tag} variant="default">{tag}</Badge>
))}
</div>
)}
<div className="flex justify-end">
<Button
variant="ghost"
size="sm"
onClick={(e: React.MouseEvent) => {
e.stopPropagation();
onDelete();
}}
className="text-red-500 hover:text-red-700 hover:bg-red-50 dark:hover:bg-red-900/20 text-xs px-2 py-1"
>
<Trash2 className="w-3 h-3 mr-1" />
</Button>
</div>
</div>
</motion.div>
)}
</AnimatePresence>
</motion.div>
);
}
function FilterChip({
label,
active,
onClick,
}: {
label: string;
active: boolean;
onClick: () => void;
}) {
return (
<button
onClick={onClick}
className={`text-[10px] px-2 py-0.5 rounded-full border transition-colors ${
active
? 'bg-orange-100 border-orange-300 text-orange-700 dark:bg-orange-900/30 dark:border-orange-700 dark:text-orange-300'
: 'bg-gray-50 border-gray-200 text-gray-600 hover:bg-gray-100 dark:bg-gray-800 dark:border-gray-600 dark:text-gray-400'
}`}
>
{label}
</button>
);
}
function getTimeAgo(isoDate: string): string {
const now = Date.now();
const then = new Date(isoDate).getTime();
const diffMs = now - then;
const minutes = Math.floor(diffMs / 60_000);
if (minutes < 1) return '刚刚';
if (minutes < 60) return `${minutes}分钟前`;
const hours = Math.floor(minutes / 60);
if (hours < 24) return `${hours}小时前`;
const days = Math.floor(hours / 24);
if (days < 30) return `${days}天前`;
const months = Math.floor(days / 30);
return `${months}个月前`;
}