mirror of
https://github.com/hamster1963/nezha-dash.git
synced 2025-04-24 21:10:45 +08:00
189 lines
16 KiB
TypeScript
189 lines
16 KiB
TypeScript
"use client"
|
|
|
|
import { ServerApi } from "@/app/types/nezha-api"
|
|
import { Loader } from "@/components/loading/Loader"
|
|
import { Card, CardContent } from "@/components/ui/card"
|
|
import getEnv from "@/lib/env-entry"
|
|
import { useFilter } from "@/lib/network-filter-context"
|
|
import { useStatus } from "@/lib/status-context"
|
|
import { cn, formatBytes, nezhaFetcher } from "@/lib/utils"
|
|
import blogMan from "@/public/blog-man.webp"
|
|
import { ArrowDownCircleIcon, ArrowUpCircleIcon } from "@heroicons/react/20/solid"
|
|
import { useTranslations } from "next-intl"
|
|
import Image from "next/image"
|
|
import useSWRImmutable from "swr/immutable"
|
|
|
|
export default function ServerOverviewClient() {
|
|
const { status, setStatus } = useStatus()
|
|
const { filter, setFilter } = useFilter()
|
|
const t = useTranslations("ServerOverviewClient")
|
|
|
|
const { data, error, isLoading } = useSWRImmutable<ServerApi>("/api/server", nezhaFetcher)
|
|
const disableCartoon = getEnv("NEXT_PUBLIC_DisableCartoon") === "true"
|
|
|
|
if (error) {
|
|
return (
|
|
<div className="flex flex-col items-center justify-center">
|
|
<p className="text-sm font-medium opacity-40">
|
|
Error status:{error.status} {error.info?.cause ?? error.message}
|
|
</p>
|
|
<p className="text-sm font-medium opacity-40">{t("error_message")}</p>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
return (
|
|
<>
|
|
<section className="grid grid-cols-2 gap-4 lg:grid-cols-4">
|
|
<Card
|
|
onClick={() => {
|
|
setFilter(false)
|
|
setStatus("all")
|
|
}}
|
|
className={cn("cursor-pointer hover:border-blue-500 transition-all group")}
|
|
>
|
|
<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">{t("p_816-881_Totalservers")}</p>
|
|
<div className="flex items-center gap-2">
|
|
<span className="relative flex h-2 w-2">
|
|
<span className="relative inline-flex h-2 w-2 rounded-full bg-blue-500"></span>
|
|
</span>
|
|
{data?.result ? (
|
|
<div className="text-lg font-semibold">{data?.result.length}</div>
|
|
) : (
|
|
<div className="flex h-7 items-center">
|
|
<Loader visible={true} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
</section>
|
|
</CardContent>
|
|
</Card>
|
|
<Card
|
|
onClick={() => {
|
|
setFilter(false)
|
|
setStatus("online")
|
|
}}
|
|
className={cn(
|
|
"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">
|
|
<section className="flex flex-col gap-1">
|
|
<p className="text-sm font-medium md:text-base">{t("p_1610-1676_Onlineservers")}</p>
|
|
<div className="flex items-center gap-2">
|
|
<span className="relative flex h-2 w-2">
|
|
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-green-500 opacity-75"></span>
|
|
<span className="relative inline-flex h-2 w-2 rounded-full bg-green-500"></span>
|
|
</span>
|
|
{data?.result ? (
|
|
<div className="text-lg font-semibold">{data?.live_servers}</div>
|
|
) : (
|
|
<div className="flex h-7 items-center">
|
|
<Loader visible={true} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
</section>
|
|
</CardContent>
|
|
</Card>
|
|
<Card
|
|
onClick={() => {
|
|
setFilter(false)
|
|
setStatus("offline")
|
|
}}
|
|
className={cn(
|
|
"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">
|
|
<section className="flex flex-col gap-1">
|
|
<p className="text-sm font-medium md:text-base">{t("p_2532-2599_Offlineservers")}</p>
|
|
<div className="flex items-center gap-2">
|
|
<span className="relative flex h-2 w-2">
|
|
<span className="absolute inline-flex h-full w-full animate-ping rounded-full bg-red-500 opacity-75"></span>
|
|
<span className="relative inline-flex h-2 w-2 rounded-full bg-red-500"></span>
|
|
</span>
|
|
{data?.result ? (
|
|
<div className="text-lg font-semibold">{data?.offline_servers}</div>
|
|
) : (
|
|
<div className="flex h-7 items-center">
|
|
<Loader visible={true} />
|
|
</div>
|
|
)}
|
|
</div>
|
|
</section>
|
|
</CardContent>
|
|
</Card>
|
|
<Card
|
|
onClick={() => {
|
|
setStatus("all")
|
|
setFilter(true)
|
|
}}
|
|
className={cn(
|
|
"cursor-pointer hover:ring-purple-500 ring-1 ring-transparent transition-all group",
|
|
{
|
|
"ring-purple-500 ring-2 border-transparent": filter === true,
|
|
},
|
|
)}
|
|
>
|
|
<CardContent className="flex h-full items-center relative px-6 py-3">
|
|
<section className="flex flex-col gap-1 w-full">
|
|
<div className="flex items-center w-full justify-between">
|
|
<p className="text-sm font-medium md:text-base">{t("network")}</p>
|
|
</div>
|
|
{data?.result ? (
|
|
<>
|
|
<section className="flex items-start flex-row z-[999] pr-2 sm:pr-0 gap-1">
|
|
<p className="sm:text-[12px] text-[10px] text-blue-800 dark:text-blue-400 text-nowrap font-medium">
|
|
↑{formatBytes(data?.total_out_bandwidth)}
|
|
</p>
|
|
<p className="sm:text-[12px] text-[10px] text-purple-800 dark:text-purple-400 text-nowrap font-medium">
|
|
↓{formatBytes(data?.total_in_bandwidth)}
|
|
</p>
|
|
</section>
|
|
<section className="flex flex-col sm:flex-row -mr-1 sm:items-center items-start gap-1">
|
|
<p className="text-[11px] flex items-center text-nowrap font-semibold">
|
|
<ArrowUpCircleIcon className="size-3 mr-0.5 sm:mb-[1px]" />
|
|
{formatBytes(data?.total_out_speed)}/s
|
|
</p>
|
|
<p className="text-[11px] flex items-center text-nowrap font-semibold">
|
|
<ArrowDownCircleIcon className="size-3 mr-0.5" />
|
|
{formatBytes(data?.total_in_speed)}/s
|
|
</p>
|
|
</section>
|
|
</>
|
|
) : (
|
|
<div className="flex h-[38px] items-center">
|
|
<Loader visible={true} />
|
|
</div>
|
|
)}
|
|
</section>
|
|
{!disableCartoon && (
|
|
<Image
|
|
className="absolute right-3 top-[-85px] z-10 w-20 scale-90 group-hover:opacity-50 md:scale-100 transition-all"
|
|
alt={"Hamster1963"}
|
|
src={blogMan}
|
|
priority
|
|
placeholder=""
|
|
/>
|
|
)}
|
|
</CardContent>
|
|
</Card>
|
|
</section>
|
|
{data?.result === undefined && !isLoading && (
|
|
<div className="flex flex-col items-center justify-center">
|
|
<p className="text-sm font-medium opacity-40">{t("error_message")}</p>
|
|
</div>
|
|
)}
|
|
</>
|
|
)
|
|
}
|