zeroclaw/web/src/pages/Memory.tsx
Argenis bcdbce0bee
feat(web): add theme system with CSS variables and settings modal (#4133)
- Add ThemeContext with light/dark/system theme support
- Migrate all hardcoded colors to CSS variables
- Add SettingsModal for theme customization
- Add font loader for dynamic font selection
- Add i18n support for Chinese and Turkish locales
- Fix accessibility: add aria-live to pairing error message

Co-authored-by: nanyuantingfeng <nanyuantingfeng@163.com>
2026-03-21 06:22:30 -04:00

276 lines
12 KiB
TypeScript

import { useState, useEffect } from 'react';
import {
Brain,
Search,
Plus,
Trash2,
X,
Filter,
} from 'lucide-react';
import type { MemoryEntry } from '@/types/api';
import { getMemory, storeMemory, deleteMemory } from '@/lib/api';
import { t } from '@/lib/i18n';
function truncate(text: string, max: number): string {
if (text.length <= max) return text;
return text.slice(0, max) + '...';
}
function formatDate(iso: string): string {
const d = new Date(iso);
return d.toLocaleString();
}
export default function Memory() {
const [entries, setEntries] = useState<MemoryEntry[]>([]);
const [loading, setLoading] = useState(true);
const [error, setError] = useState<string | null>(null);
const [search, setSearch] = useState('');
const [categoryFilter, setCategoryFilter] = useState('');
const [showForm, setShowForm] = useState(false);
const [confirmDelete, setConfirmDelete] = useState<string | null>(null);
// Form state
const [formKey, setFormKey] = useState('');
const [formContent, setFormContent] = useState('');
const [formCategory, setFormCategory] = useState('');
const [formError, setFormError] = useState<string | null>(null);
const [submitting, setSubmitting] = useState(false);
const fetchEntries = (q?: string, cat?: string) => {
setLoading(true);
getMemory(q || undefined, cat || undefined)
.then(setEntries)
.catch((err) => setError(err.message))
.finally(() => setLoading(false));
};
useEffect(() => {
fetchEntries();
}, []);
const handleSearch = () => {
fetchEntries(search, categoryFilter);
};
const handleKeyDown = (e: React.KeyboardEvent) => {
if (e.key === 'Enter') handleSearch();
};
const categories = Array.from(new Set(entries.map((e) => e.category))).sort();
const handleAdd = async () => {
if (!formKey.trim() || !formContent.trim()) { setFormError(t('memory.validation_error')); return; }
setSubmitting(true);
setFormError(null);
try {
await storeMemory(
formKey.trim(),
formContent.trim(),
formCategory.trim() || undefined,
);
fetchEntries(search, categoryFilter);
setShowForm(false);
setFormKey(''); setFormContent(''); setFormCategory('');
} catch (err: unknown) {
setFormError(err instanceof Error ? err.message : t('memory.store_error'));
} finally { setSubmitting(false); }
};
const handleDelete = async (key: string) => {
try {
await deleteMemory(key);
setEntries((prev) => prev.filter((e) => e.key !== key));
} catch (err: unknown) {
setError(err instanceof Error ? err.message : t('memory.delete_error'));
} finally {
setConfirmDelete(null);
}
};
if (error && entries.length === 0) {
return (
<div className="p-6 animate-fade-in">
<div className="rounded-2xl border p-4" style={{ background: 'rgba(239, 68, 68, 0.08)', borderColor: 'rgba(239, 68, 68, 0.2)', color: '#f87171' }}>
{t('memory.load_error')}: {error}
</div>
</div>
);
}
return (
<div className="flex flex-col h-full p-6 gap-6 animate-fade-in overflow-hidden">
{/* Header */}
<div className="flex items-center justify-between">
<div className="flex items-center gap-2">
<Brain className="h-5 w-5" style={{ color: 'var(--pc-accent)' }} />
<h2 className="text-sm font-semibold uppercase tracking-wider" style={{ color: 'var(--pc-text-primary)' }}>
{t('memory.memory_title')} ({entries.length})
</h2>
</div>
<button onClick={() => setShowForm(true)} className="btn-electric flex items-center gap-2 text-sm px-4 py-2">
<Plus className="h-4 w-4" />{t('memory.add_memory')}
</button>
</div>
{/* Search and Filter */}
<div className="flex flex-col sm:flex-row gap-3">
<div className="relative flex-1">
<Search className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4" style={{ color: 'var(--pc-text-faint)' }} />
<input type="text" value={search} onChange={(e) => setSearch(e.target.value)} onKeyDown={handleKeyDown} placeholder={t('memory.search_placeholder')} className="input-electric w-full pl-10 pr-4 py-2.5 text-sm" />
</div>
<div className="relative">
<Filter className="absolute left-3 top-1/2 -translate-y-1/2 h-4 w-4" style={{ color: 'var(--pc-text-faint)' }} />
<select value={categoryFilter} onChange={(e) => setCategoryFilter(e.target.value)} className="input-electric pl-10 pr-8 py-2.5 text-sm appearance-none cursor-pointer">
<option value="">{t('memory.all_categories')}</option>
{categories.map((cat) => (
<option key={cat} value={cat}>
{cat}
</option>
))}
</select>
</div>
<button onClick={handleSearch} className="btn-electric px-4 py-2.5 text-sm">{t('memory.search_button')}</button>
</div>
{/* Error banner (non-fatal) */}
{error && (
<div className="rounded-xl border p-3 text-sm animate-fade-in" style={{ background: 'rgba(239, 68, 68, 0.08)', borderColor: 'rgba(239, 68, 68, 0.2)', color: '#f87171' }}>
{error}
</div>
)}
{/* Add Memory Form Modal */}
{showForm && (
<div className="fixed inset-0 modal-backdrop flex items-center justify-center z-50">
<div className="surface-panel p-6 w-full max-w-md mx-4 animate-fade-in-scale">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold" style={{ color: 'var(--pc-text-primary)' }}>{t('memory.add_modal_title')}</h3>
<button
onClick={() => {
setShowForm(false);
setFormError(null);
}}
className="btn-icon">
<X className="h-5 w-5" />
</button>
</div>
{formError && (
<div className="mb-4 rounded-xl border p-3 text-sm animate-fade-in" style={{ background: 'rgba(239, 68, 68, 0.08)', borderColor: 'rgba(239, 68, 68, 0.2)', color: '#f87171' }}>
{formError}
</div>
)}
<div className="space-y-4">
<div>
<label className="block text-xs font-semibold mb-1.5 uppercase tracking-wider" style={{ color: 'var(--pc-text-secondary)' }}>
{t('memory.key_required')} <span style={{ color: 'var(--color-status-error)' }}>*</span>
</label>
<input type="text" value={formKey} onChange={(e) => setFormKey(e.target.value)} placeholder="e.g. user_preferences" className="input-electric w-full px-3 py-2.5 text-sm" />
</div>
<div>
<label className="block text-xs font-semibold mb-1.5 uppercase tracking-wider" style={{ color: 'var(--pc-text-secondary)' }}>
{t('memory.content_required')} <span style={{ color: 'var(--color-status-error)' }}>*</span>
</label>
<textarea value={formContent} onChange={(e) => setFormContent(e.target.value)} placeholder="Memory content..." rows={4} className="input-electric w-full px-3 py-2.5 text-sm resize-none" />
</div>
<div>
<label className="block text-xs font-semibold mb-1.5 uppercase tracking-wider" style={{ color: 'var(--pc-text-secondary)' }}>
{t('memory.category_optional')}
</label>
<input type="text" value={formCategory} onChange={(e) => setFormCategory(e.target.value)} placeholder="e.g. preferences, context, facts" className="input-electric w-full px-3 py-2.5 text-sm" />
</div>
</div>
<div className="flex justify-end gap-3 mt-6">
<button
onClick={() => {
setShowForm(false);
setFormError(null);
}}
className="btn-secondary px-4 py-2 text-sm font-medium"
>
{t('memory.cancel')}
</button>
<button
onClick={handleAdd} disabled={submitting} className="btn-electric px-4 py-2 text-sm font-medium">{submitting ? t('memory.saving') : t('common.save')}</button>
</div>
</div>
</div>
)}
{/* Memory Table */}
{loading ? (
<div className="flex items-center justify-center h-32">
<div className="h-8 w-8 border-2 rounded-full animate-spin" style={{ borderColor: 'var(--pc-border)', borderTopColor: 'var(--pc-accent)' }} />
</div>
) : entries.length === 0 ? (
<div className="card p-8 text-center">
<Brain className="h-10 w-10 mx-auto mb-3" style={{ color: 'var(--pc-text-faint)' }} />
<p style={{ color: 'var(--pc-text-muted)' }}>{t('memory.empty')}</p>
</div>
) : (
<div className="card overflow-x-auto rounded-2xl">
<table className="table-electric">
<thead>
<tr>
<th>{t('memory.key')}</th>
<th>{t('memory.content')}</th>
<th>{t('memory.category')}</th>
<th>{t('memory.timestamp')}</th>
<th className="text-right">{t('common.actions')}</th>
</tr>
</thead>
<tbody>
{entries.map((entry) => (
<tr key={entry.id}>
<td className="font-mono text-xs" style={{ color: 'var(--pc-text-primary)' }}>
{entry.key}
</td>
<td className="max-w-[300px] text-sm" style={{ color: 'var(--pc-text-secondary)' }}>
<span title={entry.content}>
{truncate(entry.content, 80)}
</span>
</td>
<td>
<span className="inline-flex items-center px-2.5 py-0.5 rounded-full text-[10px] font-semibold capitalize border" style={{ borderColor: 'var(--pc-border)', color: 'var(--pc-text-secondary)', background: 'var(--pc-accent-glow)' }}>
{entry.category}
</span>
</td>
<td className="text-xs whitespace-nowrap" style={{ color: 'var(--pc-text-muted)' }}>
{formatDate(entry.timestamp)}
</td>
<td className="text-right">
{confirmDelete === entry.key ? (
<div className="flex items-center justify-end gap-2 animate-fade-in">
<span className="text-xs" style={{ color: 'var(--color-status-error)' }}>
{t('memory.delete_confirm')}
</span>
<button
onClick={() => handleDelete(entry.key)}
className="text-xs font-medium" style={{ color: 'var(--color-status-error)' }}
>
{t('memory.yes')}
</button>
<button
onClick={() => setConfirmDelete(null)}
className="text-xs font-medium" style={{ color: 'var(--pc-text-muted)' }}>
{t('memory.no')}
</button>
</div>
) : (
<button
onClick={() => setConfirmDelete(entry.key)}
className="btn-icon"
>
<Trash2 className="h-4 w-4" />
</button>
)}
</td>
</tr>
))}
</tbody>
</table>
</div>
)}
</div>
);
}