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} />}
);
}