Files
komari-theme-purcarte/src/components/sections/StatsBar.tsx
2025-08-15 21:34:42 +08:00

224 lines
7.5 KiB
TypeScript

import {
DropdownMenu,
DropdownMenuContent,
DropdownMenuItem,
DropdownMenuLabel,
DropdownMenuSeparator,
DropdownMenuTrigger,
} from "@/components/ui/dropdown-menu";
import { Settings2 } from "lucide-react";
import { Switch } from "@/components/ui/switch";
import { Button } from "@/components/ui/button";
import { formatBytes } from "@/utils";
import { useIsMobile } from "@/hooks/useMobile";
interface StatsBarProps {
displayOptions: {
time: boolean;
online: boolean;
regions: boolean;
traffic: boolean;
speed: boolean;
};
setDisplayOptions: (options: any) => void;
stats: {
onlineCount: number;
totalCount: number;
uniqueRegions: number;
totalTrafficUp: number;
totalTrafficDown: number;
currentSpeedUp: number;
currentSpeedDown: number;
};
loading: boolean;
currentTime: Date;
}
export const StatsBar = ({
displayOptions,
setDisplayOptions,
stats,
loading,
currentTime,
}: StatsBarProps) => {
const isMobile = useIsMobile();
// 获取已启用的统计项列表
const enabledStats = Object.keys(displayOptions).filter(
(key) => displayOptions[key as keyof typeof displayOptions]
);
// 渲染统计项
const renderStatItem = (key: string) => {
switch (key) {
case "time":
return (
displayOptions.time && (
<div className="w-full py-1" key="time">
<div className="rt-Flex rt-r-fd-column rt-r-gap-2">
<label className="text-secondary-foreground text-sm">
</label>
<label className="font-medium -mt-2 text-md">
{currentTime.toLocaleTimeString()}
</label>
</div>
</div>
)
);
case "online":
return (
displayOptions.online && (
<div className="w-full py-1" key="online">
<div className="rt-Flex rt-r-fd-column rt-r-gap-2">
<label className="text-secondary-foreground text-sm">
线
</label>
<label className="font-medium -mt-2 text-md">
{loading
? "..."
: `${stats.onlineCount} / ${stats.totalCount}`}
</label>
</div>
</div>
)
);
case "regions":
return (
displayOptions.regions && (
<div className="w-full py-1" key="regions">
<div className="rt-Flex rt-r-fd-column rt-r-gap-2">
<label className="text-secondary-foreground text-sm">
</label>
<label className="font-medium -mt-2 text-md">
{loading ? "..." : stats.uniqueRegions}
</label>
</div>
</div>
)
);
case "traffic":
return (
displayOptions.traffic && (
<div className="w-full py-1" key="traffic">
<div className="rt-Flex rt-r-fd-column rt-r-gap-2">
<label className="text-secondary-foreground text-sm">
</label>
<div className="font-medium -mt-2 text-md">
{loading ? (
"..."
) : (
<div className="flex flex-col items-center">
<span>{`${formatBytes(stats.totalTrafficUp)}`}</span>
<span>{`${formatBytes(stats.totalTrafficDown)}`}</span>
</div>
)}
</div>
</div>
</div>
)
);
case "speed":
return (
displayOptions.speed && (
<div className="w-full py-1" key="speed">
<div className="rt-Flex rt-r-fd-column rt-r-gap-2">
<label className="text-secondary-foreground text-sm">
</label>
<div className="font-medium -mt-2 text-md">
{loading ? (
"..."
) : (
<div className="flex flex-col items-center">
<span>{`${formatBytes(stats.currentSpeedUp)}/s`}</span>
<span>{`${formatBytes(
stats.currentSpeedDown
)}/s`}</span>
</div>
)}
</div>
</div>
</div>
)
);
default:
return null;
}
};
return (
<div className="bg-card backdrop-blur-[10px] min-w-[300px] rounded-lg box-border border text-secondary-foreground my-6 mx-4 px-4 md:text-base text-sm relative flex items-center min-h-[5rem]">
<div className="absolute top-2 right-2">
<DropdownMenu modal={false}>
<DropdownMenuTrigger asChild>
<Button variant="ghost" size="icon" className="cursor-pointer">
<Settings2 />
</Button>
</DropdownMenuTrigger>
<DropdownMenuContent align="end">
<DropdownMenuLabel></DropdownMenuLabel>
<DropdownMenuSeparator />
<DropdownMenuItem className="flex items-center justify-between cursor-pointer">
<span></span>
<Switch
checked={displayOptions.time}
onCheckedChange={(checked) =>
setDisplayOptions({ ...displayOptions, time: checked })
}
/>
</DropdownMenuItem>
<DropdownMenuItem className="flex items-center justify-between cursor-pointer">
<span>线</span>
<Switch
checked={displayOptions.online}
onCheckedChange={(checked) =>
setDisplayOptions({ ...displayOptions, online: checked })
}
/>
</DropdownMenuItem>
<DropdownMenuItem className="flex items-center justify-between cursor-pointer">
<span></span>
<Switch
checked={displayOptions.regions}
onCheckedChange={(checked) =>
setDisplayOptions({ ...displayOptions, regions: checked })
}
/>
</DropdownMenuItem>
<DropdownMenuItem className="flex items-center justify-between cursor-pointer">
<span></span>
<Switch
checked={displayOptions.traffic}
onCheckedChange={(checked) =>
setDisplayOptions({ ...displayOptions, traffic: checked })
}
/>
</DropdownMenuItem>
<DropdownMenuItem className="flex items-center justify-between cursor-pointer">
<span></span>
<Switch
checked={displayOptions.speed}
onCheckedChange={(checked) =>
setDisplayOptions({ ...displayOptions, speed: checked })
}
/>
</DropdownMenuItem>
</DropdownMenuContent>
</DropdownMenu>
</div>
<div
className="grid w-full gap-2 text-center items-center py-3"
style={{
gridTemplateColumns: isMobile
? "repeat(2, 1fr)"
: "repeat(auto-fit, minmax(180px, 1fr))",
gridAutoRows: "min-content",
}}>
{enabledStats.map((key) => renderStatItem(key))}
</div>
</div>
);
};