feat: refactor overview button

This commit is contained in:
hamster1963 2024-12-06 22:55:13 +08:00
parent 1d9e59a9df
commit a3bbd7b2eb
10 changed files with 170 additions and 21 deletions

View File

@ -110,8 +110,8 @@ export default function ServerDetailChart({
gpuName={`#${index + 1}`}
key={index}
/>
)
)) : (
))
) : (
<></>
)}
<ProcessChart now={nezhaWsData.now} data={server} />

View File

@ -6,6 +6,8 @@ import {
ArrowDownCircleIcon,
ArrowUpCircleIcon,
} from "@heroicons/react/20/solid";
import { useStatus } from "@/hooks/use-status";
import useFilter from "@/hooks/use-filter";
type ServerOverviewProps = {
online: number;
@ -27,11 +29,19 @@ export default function ServerOverview({
downSpeed,
}: ServerOverviewProps) {
const { t } = useTranslation();
const { status, setStatus } = useStatus();
const { filter, setFilter } = useFilter();
return (
<>
<section className="grid grid-cols-2 gap-4 lg:grid-cols-4">
<Card className={cn("hover:border-blue-500 transition-all")}>
<Card
onClick={() => {
setFilter(false);
setStatus("all");
}}
className={cn("hover:border-blue-500 cursor-pointer transition-all")}
>
<CardContent className="flex h-full items-center px-6 py-3">
<section className="flex flex-col gap-1">
<p className="text-sm font-medium md:text-base">
@ -47,8 +57,15 @@ export default function ServerOverview({
</CardContent>
</Card>
<Card
onClick={() => {
setFilter(false);
setStatus("online");
}}
className={cn(
" hover:ring-green-500 ring-1 ring-transparent transition-all",
"cursor-pointer hover:ring-green-500 ring-1 ring-transparent transition-all",
{
"ring-green-500 ring-2 border-transparent": status === "online",
},
)}
>
<CardContent className="flex h-full items-center px-6 py-3">
@ -68,8 +85,15 @@ export default function ServerOverview({
</CardContent>
</Card>
<Card
onClick={() => {
setFilter(false);
setStatus("offline");
}}
className={cn(
" hover:ring-red-500 ring-1 ring-transparent transition-all",
"cursor-pointer hover:ring-red-500 ring-1 ring-transparent transition-all",
{
"ring-red-500 ring-2 border-transparent": status === "offline",
},
)}
>
<CardContent className="flex h-full items-center px-6 py-3">
@ -88,8 +112,15 @@ export default function ServerOverview({
</CardContent>
</Card>
<Card
onClick={() => {
setStatus("all");
setFilter(true);
}}
className={cn(
" hover:ring-purple-500 ring-1 ring-transparent transition-all",
"cursor-pointer hover:ring-purple-500 ring-1 ring-transparent transition-all",
{
"ring-purple-500 ring-2 border-transparent": filter === true,
},
)}
>
<CardContent className="flex h-full items-center relative px-6 py-3">

View File

@ -0,0 +1,10 @@
import { createContext } from "react";
export interface FilterContextType {
filter: boolean;
setFilter: (filter: boolean) => void;
}
export const FilterContext = createContext<FilterContextType | undefined>(
undefined,
);

View File

@ -0,0 +1,14 @@
"use client";
import { ReactNode, useState } from "react";
import { FilterContext } from "./filter-context";
export function FilterProvider({ children }: { children: ReactNode }) {
const [filter, setFilter] = useState<boolean>(false);
return (
<FilterContext.Provider value={{ filter, setFilter }}>
{children}
</FilterContext.Provider>
);
}

View File

@ -0,0 +1,12 @@
import { createContext } from "react";
export type Status = "all" | "online" | "offline";
export interface StatusContextType {
status: Status;
setStatus: (status: Status) => void;
}
export const StatusContext = createContext<StatusContextType | undefined>(
undefined,
);

View File

@ -0,0 +1,14 @@
"use client";
import { ReactNode, useState } from "react";
import { Status, StatusContext } from "./status-context";
export function StatusProvider({ children }: { children: ReactNode }) {
const [status, setStatus] = useState<Status>("all");
return (
<StatusContext.Provider value={{ status, setStatus }}>
{children}
</StatusContext.Provider>
);
}

12
src/hooks/use-filter.tsx Normal file
View File

@ -0,0 +1,12 @@
import { useContext } from "react";
import { FilterContext, FilterContextType } from "@/context/filter-context";
const useFilter = (): FilterContextType => {
const context = useContext(FilterContext);
if (context === undefined) {
throw new Error("useFilter must be used within a FilterProvider");
}
return context;
};
export default useFilter;

10
src/hooks/use-status.tsx Normal file
View File

@ -0,0 +1,10 @@
import { useContext } from "react";
import { StatusContext } from "../context/status-context";
export function useStatus() {
const context = useContext(StatusContext);
if (context === undefined) {
throw new Error("useStatus must be used within a StatusProvider");
}
return context;
}

View File

@ -9,6 +9,8 @@ import { QueryClient, QueryClientProvider } from "@tanstack/react-query";
import { Toaster } from "sonner";
import { MotionProvider } from "./components/motion/motion-provider";
import { WebSocketProvider } from "./context/websocket-provider";
import { StatusProvider } from "./context/status-provider";
import { FilterProvider } from "./context/network-filter-context";
const queryClient = new QueryClient();
@ -18,6 +20,8 @@ ReactDOM.createRoot(document.getElementById("root")!).render(
<ThemeProvider storageKey="vite-ui-theme">
<QueryClientProvider client={queryClient}>
<WebSocketProvider url="/api/v1/ws/server">
<StatusProvider>
<FilterProvider>
<App />
<Toaster
duration={1000}
@ -31,6 +35,8 @@ ReactDOM.createRoot(document.getElementById("root")!).render(
className={"flex items-center justify-center"}
/>
<ReactQueryDevtools />
</FilterProvider>
</StatusProvider>
</WebSocketProvider>
</QueryClientProvider>
</ThemeProvider>

View File

@ -19,6 +19,8 @@ import { ServiceTracker } from "@/components/ServiceTracker";
import ServerCardInline from "@/components/ServerCardInline";
import { Loader } from "@/components/loading/Loader";
import GlobalMap from "@/components/GlobalMap";
import { useStatus } from "@/hooks/use-status";
import useFilter from "@/hooks/use-filter";
export default function Servers() {
const { t } = useTranslation();
@ -27,7 +29,8 @@ export default function Servers() {
queryFn: () => fetchServerGroup(),
});
const { lastMessage, connected } = useWebSocketContext();
const { status } = useStatus();
const { filter } = useFilter();
const [showServices, setShowServices] = useState<string>("0");
const [showMap, setShowMap] = useState<string>("0");
const [inline, setInline] = useState<string>("0");
@ -83,7 +86,7 @@ export default function Servers() {
);
}
const filteredServers =
let filteredServers =
nezhaWsData?.servers?.filter((server) => {
if (currentGroup === "All") return true;
const group = groupData?.data?.find(
@ -138,6 +141,43 @@ export default function Servers() {
0,
) || 0;
filteredServers =
status === "all"
? filteredServers
: filteredServers.filter((server) =>
[status].includes(
formatNezhaInfo(nezhaWsData.now, server).online
? "online"
: "offline",
),
);
if (filter) {
filteredServers.sort((a, b) => {
if (
!formatNezhaInfo(nezhaWsData.now, a).online &&
formatNezhaInfo(nezhaWsData.now, b).online
)
return 1;
if (
formatNezhaInfo(nezhaWsData.now, a).online &&
!formatNezhaInfo(nezhaWsData.now, b).online
)
return -1;
if (
!formatNezhaInfo(nezhaWsData.now, a).online &&
!formatNezhaInfo(nezhaWsData.now, b).online
)
return 0;
return (
formatNezhaInfo(nezhaWsData.now, b).state.net_in_speed +
formatNezhaInfo(nezhaWsData.now, b).state.net_out_speed -
(formatNezhaInfo(nezhaWsData.now, a).state.net_in_speed +
formatNezhaInfo(nezhaWsData.now, a).state.net_out_speed)
);
});
}
return (
<div className="mx-auto w-full max-w-5xl px-0">
<ServerOverview