diff --git a/package.json b/package.json index 1a934a4..adf63a9 100644 --- a/package.json +++ b/package.json @@ -15,11 +15,13 @@ "@mui/material": "^5.2.3", "@tauri-apps/api": "^1.0.0-beta.8", "axios": "^0.24.0", + "dayjs": "^1.10.7", "react": "^17.0.0", "react-dom": "^17.0.0", "react-router-dom": "^6.0.2", "react-virtuoso": "^2.3.1", - "recoil": "^0.5.2" + "recoil": "^0.5.2", + "swr": "^1.1.2-beta.0" }, "devDependencies": { "@tauri-apps/cli": "^1.0.0-beta.10", diff --git a/src/components/profile-item.tsx b/src/components/profile-item.tsx new file mode 100644 index 0000000..a6c1997 --- /dev/null +++ b/src/components/profile-item.tsx @@ -0,0 +1,131 @@ +import React from "react"; +import dayjs from "dayjs"; +import { + alpha, + Box, + ButtonBase, + styled, + Typography, + LinearProgress, + IconButton, +} from "@mui/material"; +import { MenuRounded } from "@mui/icons-material"; +import { ProfileItem } from "../services/command"; +import parseTraffic from "../utils/parse-traffic"; + +const Wrapper = styled(Box)(({ theme }) => ({ + width: "100%", + display: "block", + cursor: "pointer", + textAlign: "left", + borderRadius: theme.shape.borderRadius, + boxShadow: theme.shadows[2], + padding: "8px 16px", + boxSizing: "border-box", +})); + +interface Props { + selected: boolean; + itemData: ProfileItem; + onClick: () => void; +} + +const ProfileItemComp: React.FC = (props) => { + const { selected, itemData, onClick } = props; + + const { name = "Profile", extra } = itemData; + const { upload = 0, download = 0, total = 0 } = extra ?? {}; + const from = parseUrl(itemData.url); + const expire = parseExpire(extra?.expire); + const progress = Math.round(((download + upload) * 100) / (total + 0.1)); + + return ( + { + const { mode, primary, text, grey } = palette; + const isDark = mode === "dark"; + + if (selected) { + const bgcolor = isDark + ? alpha(primary.main, 0.35) + : alpha(primary.main, 0.15); + + return { + bgcolor, + color: isDark ? alpha(text.secondary, 0.6) : text.secondary, + "& h2": { + color: isDark ? primary.light : primary.main, + }, + }; + } + const bgcolor = isDark + ? alpha(grey[700], 0.35) + : palette.background.paper; + return { + bgcolor, + color: isDark ? alpha(text.secondary, 0.6) : text.secondary, + "& h2": { + color: isDark ? text.primary : text.primary, + }, + }; + }} + onClick={onClick} + > + + + {name} + + + { + e.stopPropagation(); + }} + > + + + + + + {from} + + + + + {parseTraffic(upload + download)} / {parseTraffic(total)} + + {expire} + + + + + ); +}; + +function parseUrl(url?: string) { + if (!url) return ""; + const regex = /https?:\/\/(.+?)\//; + const result = url.match(regex); + return result ? result[1] : "local file"; +} + +function parseExpire(expire?: number) { + if (!expire) return "-"; + return dayjs(expire * 1000).format("YYYY-MM-DD"); +} + +export default ProfileItemComp; diff --git a/src/pages/_layout.tsx b/src/pages/_layout.tsx index e081c98..143871e 100644 --- a/src/pages/_layout.tsx +++ b/src/pages/_layout.tsx @@ -1,4 +1,5 @@ import { useMemo } from "react"; +import { SWRConfig } from "swr"; import { Route, Routes } from "react-router-dom"; import { useRecoilValue } from "recoil"; import { createTheme, List, Paper, ThemeProvider } from "@mui/material"; @@ -55,6 +56,15 @@ const Layout = () => { } return createTheme({ + breakpoints: { + values: { + xs: 0, + sm: 650, + md: 900, + lg: 1200, + xl: 1536, + }, + }, palette: { mode: paletteMode, primary: { @@ -69,38 +79,40 @@ const Layout = () => { }, [paletteMode]); return ( - - -
-
- + + + +
+
+ +
+ + + {routers.map((router) => ( + + {router.label} + + ))} + + +
+ +
- - {routers.map((router) => ( - - {router.label} - - ))} - - -
- +
+ + } /> + } /> + } /> + } /> + } /> + } /> +
-
- -
- - } /> - } /> - } /> - } /> - } /> - } /> - -
-
-
+ + +
); }; diff --git a/src/pages/rules.tsx b/src/pages/rules.tsx index 72f54fb..030a0b8 100644 --- a/src/pages/rules.tsx +++ b/src/pages/rules.tsx @@ -1,14 +1,18 @@ import { useState } from "react"; -import { Box, Button, TextField, Typography } from "@mui/material"; -import { importProfile } from "../services/command"; +import useSWR, { useSWRConfig } from "swr"; +import { Box, Button, Grid, TextField, Typography } from "@mui/material"; +import { getProfiles, importProfile, putProfiles } from "../services/command"; +import ProfileItemComp from "../components/profile-item"; import useNotice from "../utils/use-notice"; const RulesPage = () => { const [url, setUrl] = useState(""); const [disabled, setDisabled] = useState(false); - const [notice, noticeElement] = useNotice(); + const { mutate } = useSWRConfig(); + const { data: profiles = {} } = useSWR("getProfiles", getProfiles); + const onClick = () => { if (!url) return; setUrl(""); @@ -19,14 +23,26 @@ const RulesPage = () => { .finally(() => setDisabled(false)); }; + const onProfileChange = (index: number) => { + putProfiles(index) + .then(() => { + mutate("getProfiles", { ...profiles, current: index }, true); + }) + .catch((err) => { + console.error(err); + }); + }; + return ( Rules - + { onChange={(e) => setUrl(e.target.value)} sx={{ mr: 4 }} /> - + + {profiles?.items?.map((item, idx) => ( + + onProfileChange(idx)} + /> + + ))} + + {noticeElement} ); diff --git a/src/services/command.ts b/src/services/command.ts index ccc44e2..57bd471 100644 --- a/src/services/command.ts +++ b/src/services/command.ts @@ -38,9 +38,13 @@ export interface ProfilesConfig { } export async function getProfiles() { - return (await invoke("get_profiles")) ?? []; + return invoke("get_profiles"); } export async function setProfiles(current: number, profile: ProfileItem) { return invoke("set_profiles", { current, profile }); } + +export async function putProfiles(current: number) { + return invoke("put_profiles", { current }); +}