import { useEffect, useRef, useMemo, useCallback } from "react"; import { useVerge } from "@/hooks/use-verge"; import { Box, IconButton, Tooltip, alpha, styled, Grid } from "@mui/material"; import { DndContext, closestCenter, PointerSensor, useSensor, useSensors, DragEndEvent, } from "@dnd-kit/core"; import { SortableContext } from "@dnd-kit/sortable"; import { useTranslation } from "react-i18next"; import { TestViewer, TestViewerRef } from "@/components/test/test-viewer"; import { TestItem } from "@/components/test/test-item"; import { emit } from "@tauri-apps/api/event"; import { nanoid } from "nanoid"; import { Add, NetworkCheck } from "@mui/icons-material"; import { EnhancedCard } from "./enhanced-card"; // test icons import apple from "@/assets/image/test/apple.svg?raw"; import github from "@/assets/image/test/github.svg?raw"; import google from "@/assets/image/test/google.svg?raw"; import youtube from "@/assets/image/test/youtube.svg?raw"; // 自定义滚动条样式 const ScrollBox = styled(Box)(({ theme }) => ({ maxHeight: "180px", overflowY: "auto", overflowX: "hidden", "&::-webkit-scrollbar": { width: "6px", }, "&::-webkit-scrollbar-thumb": { backgroundColor: alpha(theme.palette.text.primary, 0.2), borderRadius: "3px", }, })); // 默认测试列表,移到组件外部避免重复创建 const DEFAULT_TEST_LIST = [ { uid: nanoid(), name: "Apple", url: "https://www.apple.com", icon: apple, }, { uid: nanoid(), name: "GitHub", url: "https://www.github.com", icon: github, }, { uid: nanoid(), name: "Google", url: "https://www.google.com", icon: google, }, { uid: nanoid(), name: "Youtube", url: "https://www.youtube.com", icon: youtube, }, ]; export const TestCard = () => { const { t } = useTranslation(); const sensors = useSensors(useSensor(PointerSensor)); const { verge, mutateVerge, patchVerge } = useVerge(); const viewerRef = useRef(null); // 使用useMemo优化测试列表,避免每次渲染重新计算 const testList = useMemo(() => { return verge?.test_list ?? DEFAULT_TEST_LIST; }, [verge?.test_list]); // 使用useCallback优化函数引用,避免不必要的重新渲染 const onTestListItemChange = useCallback( (uid: string, patch?: Partial) => { if (!patch) { mutateVerge(); return; } const newList = testList.map((x) => x.uid === uid ? { ...x, ...patch } : x ); mutateVerge({ ...verge, test_list: newList }, false); }, [testList, verge, mutateVerge] ); const onDeleteTestListItem = useCallback( (uid: string) => { const newList = testList.filter((x) => x.uid !== uid); patchVerge({ test_list: newList }); mutateVerge({ ...verge, test_list: newList }, false); }, [testList, verge, patchVerge, mutateVerge] ); const onDragEnd = useCallback( async (event: DragEndEvent) => { const { active, over } = event; if (!over || active.id === over.id) return; const old_index = testList.findIndex((x) => x.uid === active.id); const new_index = testList.findIndex((x) => x.uid === over.id); if (old_index >= 0 && new_index >= 0) { const newList = [...testList]; const [removed] = newList.splice(old_index, 1); newList.splice(new_index, 0, removed); await mutateVerge({ ...verge, test_list: newList }, false); await patchVerge({ test_list: newList }); } }, [testList, verge, mutateVerge, patchVerge] ); // 仅在verge首次加载时初始化测试列表 useEffect(() => { if (verge && !verge.test_list) { patchVerge({ test_list: DEFAULT_TEST_LIST }); } }, [verge, patchVerge]); // 使用useMemo优化UI内容,减少渲染计算 const renderTestItems = useMemo(() => ( x.uid)}> {testList.map((item) => ( viewerRef.current?.edit(item)} onDelete={onDeleteTestListItem} /> ))} ), [testList, onDeleteTestListItem]); const handleTestAll = useCallback(() => { emit("verge://test-all"); }, []); const handleCreateTest = useCallback(() => { viewerRef.current?.create(); }, []); return ( } action={ } > {renderTestItems} ); };