mirror of
https://github.com/clash-verge-rev/clash-verge-rev
synced 2025-05-06 00:03:44 +08:00
193 lines
5.4 KiB
TypeScript
193 lines
5.4 KiB
TypeScript
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<TestViewerRef>(null);
|
||
|
||
// 使用useMemo优化测试列表,避免每次渲染重新计算
|
||
const testList = useMemo(() => {
|
||
return verge?.test_list ?? DEFAULT_TEST_LIST;
|
||
}, [verge?.test_list]);
|
||
|
||
// 使用useCallback优化函数引用,避免不必要的重新渲染
|
||
const onTestListItemChange = useCallback(
|
||
(uid: string, patch?: Partial<IVergeTestItem>) => {
|
||
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(() => (
|
||
<Grid container spacing={1} columns={12}>
|
||
<SortableContext items={testList.map((x) => x.uid)}>
|
||
{testList.map((item) => (
|
||
<Grid key={item.uid} size={3}>
|
||
<TestItem
|
||
id={item.uid}
|
||
itemData={item}
|
||
onEdit={() => viewerRef.current?.edit(item)}
|
||
onDelete={onDeleteTestListItem}
|
||
/>
|
||
</Grid>
|
||
))}
|
||
</SortableContext>
|
||
</Grid>
|
||
), [testList, onDeleteTestListItem]);
|
||
|
||
const handleTestAll = useCallback(() => {
|
||
emit("verge://test-all");
|
||
}, []);
|
||
|
||
const handleCreateTest = useCallback(() => {
|
||
viewerRef.current?.create();
|
||
}, []);
|
||
|
||
return (
|
||
<EnhancedCard
|
||
title={t("Website Tests")}
|
||
icon={<NetworkCheck />}
|
||
action={
|
||
<Box sx={{ display: "flex", gap: 1 }}>
|
||
<Tooltip title={t("Test All")} arrow>
|
||
<IconButton size="small" onClick={handleTestAll}>
|
||
<NetworkCheck fontSize="small" />
|
||
</IconButton>
|
||
</Tooltip>
|
||
<Tooltip title={t("Create Test")} arrow>
|
||
<IconButton size="small" onClick={handleCreateTest}>
|
||
<Add fontSize="small" />
|
||
</IconButton>
|
||
</Tooltip>
|
||
</Box>
|
||
}
|
||
>
|
||
<ScrollBox>
|
||
<DndContext
|
||
sensors={sensors}
|
||
collisionDetection={closestCenter}
|
||
onDragEnd={onDragEnd}
|
||
>
|
||
{renderTestItems}
|
||
</DndContext>
|
||
</ScrollBox>
|
||
|
||
<TestViewer ref={viewerRef} onChange={onTestListItemChange} />
|
||
</EnhancedCard>
|
||
);
|
||
};
|