diff --git a/src/components/profile-item.tsx b/src/components/profile-item.tsx index cbc5e6a212ddd55eeac5cc18866d7dc31a7ee7df..fb6f02175898a9e98aaf8f24b73a0b0ccada0dd5 100644 --- a/src/components/profile-item.tsx +++ b/src/components/profile-item.tsx @@ -1,18 +1,20 @@ -import React from "react"; +import React, { useState } from "react"; import dayjs from "dayjs"; import { alpha, Box, - ButtonBase, styled, Typography, LinearProgress, IconButton, + keyframes, } from "@mui/material"; -import { MenuRounded } from "@mui/icons-material"; +import { useSWRConfig } from "swr"; +import { RefreshRounded } from "@mui/icons-material"; import { CmdType } from "../services/types"; import parseTraffic from "../utils/parse-traffic"; import relativeTime from "dayjs/plugin/relativeTime"; +import { updateProfile } from "../services/cmds"; dayjs.extend(relativeTime); @@ -27,15 +29,23 @@ const Wrapper = styled(Box)(({ theme }) => ({ boxSizing: "border-box", })); +const round = keyframes` +from { transform: rotate(0deg); } +to { transform: rotate(360deg); } +`; + interface Props { + index: number; selected: boolean; itemData: CmdType.ProfileItem; onClick: () => void; - onUpdate: () => void; } const ProfileItemComp: React.FC<Props> = (props) => { - const { selected, itemData, onClick, onUpdate } = props; + const { index, selected, itemData, onClick } = props; + + const { mutate } = useSWRConfig(); + const [loading, setLoading] = useState(false); const { name = "Profile", extra, updated = 0 } = itemData; const { upload = 0, download = 0, total = 0 } = extra ?? {}; @@ -44,6 +54,19 @@ const ProfileItemComp: React.FC<Props> = (props) => { const progress = Math.round(((download + upload) * 100) / (total + 0.1)); const fromnow = updated > 0 ? dayjs(updated * 1000).fromNow() : ""; + const onUpdate = async () => { + if (loading) return; + setLoading(true); + try { + await updateProfile(index); + mutate("getProfiles"); + } catch (err) { + console.error(err); + } finally { + setLoading(false); + } + }; + return ( <Wrapper sx={({ palette }) => { @@ -88,14 +111,19 @@ const ProfileItemComp: React.FC<Props> = (props) => { </Typography> <IconButton - sx={{ width: 30, height: 30 }} + sx={{ + width: 30, + height: 30, + animation: loading ? `1s linear infinite ${round}` : "none", + }} color="inherit" + disabled={loading} onClick={(e) => { e.stopPropagation(); onUpdate(); }} > - <MenuRounded /> + <RefreshRounded /> </IconButton> </Box> diff --git a/src/pages/profile.tsx b/src/pages/profile.tsx index 8cac0c48d35953bece4302a1cf53af1ddc7579be..16be950b86880c905a1d90bff3aa7ae981cfede8 100644 --- a/src/pages/profile.tsx +++ b/src/pages/profile.tsx @@ -1,12 +1,7 @@ import { useRef, useState } from "react"; import useSWR, { useSWRConfig } from "swr"; import { Box, Button, Grid, TextField, Typography } from "@mui/material"; -import { - getProfiles, - importProfile, - putProfiles, - updateProfile, -} from "../services/cmds"; +import { getProfiles, importProfile, putProfiles } from "../services/cmds"; import { getProxies } from "../services/api"; import ProfileItemComp from "../components/profile-item"; import useNotice from "../utils/use-notice"; @@ -54,16 +49,6 @@ const ProfilePage = () => { }); }; - const onUpdateProfile = (index: number) => { - updateProfile(index) - .then(() => { - mutate("getProfiles"); - }) - .catch((err) => { - console.error(err); - }); - }; - return ( <Box sx={{ width: 0.9, maxWidth: "850px", mx: "auto", mb: 2 }}> <Typography variant="h4" component="h1" sx={{ py: 2, mb: 1 }}> @@ -94,10 +79,10 @@ const ProfilePage = () => { {profiles?.items?.map((item, idx) => ( <Grid item xs={12} sm={6} key={item.file}> <ProfileItemComp + index={idx} selected={profiles.current === idx} itemData={item} onClick={() => onProfileChange(idx)} - onUpdate={() => onUpdateProfile(idx)} /> </Grid> ))}