import { useState, useEffect, useMemo } from 'react'; import { RefreshCw, FolderOpen, Image as ImageIcon, Film } from 'lucide-react'; import { useAssets } from '@/hooks/useAssets'; import { AssetProjectGroup } from './AssetProjectGroup'; import { AssetPreview } from './AssetPreview'; import { Button } from '@/components/ui/button'; import { Select, SelectTrigger, SelectValue, SelectContent, SelectItem } from '@/components/ui/select'; import type { Asset } from '@/types'; interface ManifestGroup { manifest_path: string; account_id: string; image_count: number; video_count: number; latest_at: string; } type TypeTab = 'all' | 'image' | 'video'; function AssetGroupSkeleton() { return (
{Array.from({ length: 4 }).map((_, i) => (
))}
); } export function AssetGallery() { const [accountFilter, setAccountFilter] = useState(''); const [typeTab, setTypeTab] = useState('all'); const [accounts, setAccounts] = useState<{ id: string; name: string }[]>([]); const [manifests, setManifests] = useState([]); const [previewAsset, setPreviewAsset] = useState(null); const { assets, loading, remove } = useAssets({ accountId: accountFilter || undefined, }); useEffect(() => { fetch('/api/accounts').then((r) => r.json()).then(setAccounts).catch(() => {}); fetch('/api/assets/manifests').then((r) => r.json()).then(setManifests).catch(() => {}); }, []); const filtered = useMemo(() => { if (typeTab === 'all') return assets; return assets.filter((a) => a.type === typeTab); }, [assets, typeTab]); const grouped = useMemo(() => { const map = new Map(); for (const asset of filtered) { const key = asset.manifest_path || 'ungrouped'; if (!map.has(key)) map.set(key, []); map.get(key)!.push(asset); } return map; }, [filtered]); const imageCount = assets.filter((a) => a.type === 'image').length; const videoCount = assets.filter((a) => a.type === 'video').length; const handleScan = async () => { await fetch('/api/assets/scan', { method: 'POST' }); window.location.reload(); }; const getGroupName = (manifestPath: string) => { const parts = manifestPath.replace(/\\/g, '/').split('/'); const dir = parts.length >= 2 ? parts[parts.length - 2] : manifestPath; return dir; }; const getGroupSubtitle = (manifestPath: string) => { const m = manifests.find((m) => m.manifest_path === manifestPath); if (!m) return ''; const parts: string[] = []; if (m.image_count) parts.push(`${m.image_count} 图`); if (m.video_count) parts.push(`${m.video_count} 视频`); const acc = accounts.find((a) => a.id === m.account_id); if (acc) parts.push(acc.name); return parts.join(' · '); }; const filteredGroups = useMemo(() => { if (!accountFilter) return [...grouped.entries()]; return [...grouped.entries()].filter(([key]) => { if (key === 'ungrouped') return true; const m = manifests.find((m) => m.manifest_path === key); return m?.account_id === accountFilter; }); }, [grouped, manifests, accountFilter]); const tabs: { key: TypeTab; label: string; icon: typeof ImageIcon; count: number }[] = [ { key: 'all', label: '全部', icon: FolderOpen, count: assets.length }, { key: 'image', label: '图片', icon: ImageIcon, count: imageCount }, { key: 'video', label: '视频', icon: Film, count: videoCount }, ]; return (
{/* Header */}
{tabs.map(({ key, label, icon: Icon, count }) => ( ))}
{/* Content */}
{loading ? (
{Array.from({ length: 3 }).map((_, i) => )}
) : filteredGroups.length === 0 ? (

{typeTab === 'all' ? '暂无资产' : `暂无${typeTab === 'image' ? '图片' : '视频'}`}

点击「扫描」从 output 目录导入

) : ( filteredGroups.map(([manifestPath, groupAssets]) => ( )) )}
{previewAsset && setPreviewAsset(null)} onNavigate={setPreviewAsset} />}
); }