mirror of
https://github.com/clash-verge-rev/clash-verge-rev
synced 2025-05-05 04:33:45 +08:00
236 lines
6.3 KiB
TypeScript
236 lines
6.3 KiB
TypeScript
import {
|
|
alpha,
|
|
Box,
|
|
ListItemText,
|
|
ListItemButton,
|
|
Typography,
|
|
styled,
|
|
} from "@mui/material";
|
|
import {
|
|
ExpandLessRounded,
|
|
ExpandMoreRounded,
|
|
InboxRounded,
|
|
} from "@mui/icons-material";
|
|
import { HeadState } from "./use-head-state";
|
|
import { ProxyHead } from "./proxy-head";
|
|
import { ProxyItem } from "./proxy-item";
|
|
import { ProxyItemMini } from "./proxy-item-mini";
|
|
import type { IRenderItem } from "./use-render-list";
|
|
import { useVerge } from "@/hooks/use-verge";
|
|
import { useThemeMode } from "@/services/states";
|
|
import { useEffect, useMemo, useState } from "react";
|
|
import { convertFileSrc } from "@tauri-apps/api/core";
|
|
import { downloadIconCache } from "@/services/cmds";
|
|
|
|
interface RenderProps {
|
|
item: IRenderItem;
|
|
indent: boolean;
|
|
onLocation: (group: IProxyGroupItem) => void;
|
|
onCheckAll: (groupName: string) => void;
|
|
onHeadState: (groupName: string, patch: Partial<HeadState>) => void;
|
|
onChangeProxy: (group: IProxyGroupItem, proxy: IProxyItem) => void;
|
|
}
|
|
|
|
export const ProxyRender = (props: RenderProps) => {
|
|
const { indent, item, onLocation, onCheckAll, onHeadState, onChangeProxy } =
|
|
props;
|
|
const { type, group, headState, proxy, proxyCol } = item;
|
|
const { verge } = useVerge();
|
|
const enable_group_icon = verge?.enable_group_icon ?? true;
|
|
const mode = useThemeMode();
|
|
const isDark = mode === "light" ? false : true;
|
|
const itembackgroundcolor = isDark ? "#282A36" : "#ffffff";
|
|
const [iconCachePath, setIconCachePath] = useState("");
|
|
|
|
useEffect(() => {
|
|
initIconCachePath();
|
|
}, [group]);
|
|
|
|
async function initIconCachePath() {
|
|
if (group.icon && group.icon.trim().startsWith("http")) {
|
|
const fileName =
|
|
group.name.replaceAll(" ", "") + "-" + getFileName(group.icon);
|
|
const iconPath = await downloadIconCache(group.icon, fileName);
|
|
setIconCachePath(convertFileSrc(iconPath));
|
|
}
|
|
}
|
|
|
|
function getFileName(url: string) {
|
|
return url.substring(url.lastIndexOf("/") + 1);
|
|
}
|
|
|
|
if (type === 0) {
|
|
return (
|
|
<ListItemButton
|
|
dense
|
|
style={{
|
|
background: itembackgroundcolor,
|
|
height: "100%",
|
|
margin: "8px 8px",
|
|
borderRadius: "8px",
|
|
}}
|
|
onClick={() => onHeadState(group.name, { open: !headState?.open })}
|
|
>
|
|
{enable_group_icon &&
|
|
group.icon &&
|
|
group.icon.trim().startsWith("http") && (
|
|
<img
|
|
src={iconCachePath === "" ? group.icon : iconCachePath}
|
|
width="32px"
|
|
style={{ marginRight: "12px", borderRadius: "6px" }}
|
|
/>
|
|
)}
|
|
{enable_group_icon &&
|
|
group.icon &&
|
|
group.icon.trim().startsWith("data") && (
|
|
<img
|
|
src={group.icon}
|
|
width="32px"
|
|
style={{ marginRight: "12px", borderRadius: "6px" }}
|
|
/>
|
|
)}
|
|
{enable_group_icon &&
|
|
group.icon &&
|
|
group.icon.trim().startsWith("<svg") && (
|
|
<img
|
|
src={`data:image/svg+xml;base64,${btoa(group.icon)}`}
|
|
width="32px"
|
|
/>
|
|
)}
|
|
<ListItemText
|
|
primary={<StyledPrimary>{group.name}</StyledPrimary>}
|
|
secondary={
|
|
<Box
|
|
sx={{
|
|
overflow: "hidden",
|
|
display: "flex",
|
|
alignItems: "center",
|
|
pt: "2px",
|
|
}}
|
|
>
|
|
<Box component="span" sx={{ marginTop: "2px" }}>
|
|
<StyledTypeBox>{group.type}</StyledTypeBox>
|
|
<StyledSubtitle sx={{ color: "text.secondary" }}>
|
|
{group.now}
|
|
</StyledSubtitle>
|
|
</Box>
|
|
</Box>
|
|
}
|
|
slotProps={{
|
|
secondary: {
|
|
component: "div",
|
|
sx: { display: "flex", alignItems: "center", color: "#ccc" },
|
|
},
|
|
}}
|
|
/>
|
|
{headState?.open ? <ExpandLessRounded /> : <ExpandMoreRounded />}
|
|
</ListItemButton>
|
|
);
|
|
}
|
|
|
|
if (type === 1) {
|
|
return (
|
|
<ProxyHead
|
|
sx={{ pl: 2, pr: 3, mt: indent ? 1 : 0.5, mb: 1 }}
|
|
url={group.testUrl}
|
|
groupName={group.name}
|
|
headState={headState!}
|
|
onLocation={() => onLocation(group)}
|
|
onCheckDelay={() => onCheckAll(group.name)}
|
|
onHeadState={(p) => onHeadState(group.name, p)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
if (type === 2) {
|
|
return (
|
|
<ProxyItem
|
|
group={group}
|
|
proxy={proxy!}
|
|
selected={group.now === proxy?.name}
|
|
showType={headState?.showType}
|
|
sx={{ py: 0, pl: 2 }}
|
|
onClick={() => onChangeProxy(group, proxy!)}
|
|
/>
|
|
);
|
|
}
|
|
|
|
if (type === 3) {
|
|
return (
|
|
<Box
|
|
sx={{
|
|
py: 2,
|
|
pl: 0,
|
|
display: "flex",
|
|
flexDirection: "column",
|
|
alignItems: "center",
|
|
justifyContent: "center",
|
|
}}
|
|
>
|
|
<InboxRounded sx={{ fontSize: "2.5em", color: "inherit" }} />
|
|
<Typography sx={{ color: "inherit" }}>No Proxies</Typography>
|
|
</Box>
|
|
);
|
|
}
|
|
|
|
if (type === 4) {
|
|
const proxyColItemsMemo = useMemo(() => {
|
|
return proxyCol?.map((proxy) => (
|
|
<ProxyItemMini
|
|
key={item.key + proxy.name}
|
|
group={group}
|
|
proxy={proxy!}
|
|
selected={group.now === proxy.name}
|
|
showType={headState?.showType}
|
|
onClick={() => onChangeProxy(group, proxy!)}
|
|
/>
|
|
));
|
|
}, [proxyCol, group, headState]);
|
|
return (
|
|
<Box
|
|
sx={{
|
|
height: 56,
|
|
display: "grid",
|
|
gap: 1,
|
|
pl: 2,
|
|
pr: 2,
|
|
pb: 1,
|
|
gridTemplateColumns: `repeat(${item.col! || 2}, 1fr)`,
|
|
}}
|
|
>
|
|
{proxyColItemsMemo}
|
|
</Box>
|
|
);
|
|
}
|
|
|
|
return null;
|
|
};
|
|
|
|
const StyledPrimary = styled("span")`
|
|
font-size: 16px;
|
|
font-weight: 700;
|
|
line-height: 1.5;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
`;
|
|
const StyledSubtitle = styled("span")`
|
|
font-size: 13px;
|
|
overflow: hidden;
|
|
color: text.secondary;
|
|
text-overflow: ellipsis;
|
|
white-space: nowrap;
|
|
`;
|
|
|
|
const StyledTypeBox = styled(Box)(({ theme }) => ({
|
|
display: "inline-block",
|
|
border: "1px solid #ccc",
|
|
borderColor: alpha(theme.palette.primary.main, 0.5),
|
|
color: alpha(theme.palette.primary.main, 0.8),
|
|
borderRadius: 4,
|
|
fontSize: 10,
|
|
padding: "0 4px",
|
|
lineHeight: 1.5,
|
|
marginRight: "8px",
|
|
}));
|