mhsf-dev/apps/www/src/components/feat/server-list/server-list.tsx

81 lines
2.3 KiB
TypeScript
Raw Normal View History

2025-02-15 17:18:58 -06:00
"use client";
2025-02-15 20:06:35 -06:00
import { Spinner } from "@/components/ui/spinner";
import { useServers } from "@/lib/hooks/use-servers";
import ServerCard from "./server-card";
import { Separator } from "@/components/ui/separator";
import { Statistics } from "./statistics";
2025-02-16 21:40:17 -06:00
import InfiniteScroll from "react-infinite-scroll-component";
import { useInfiniteScrolling } from "@/lib/hooks/use-infinite-scrolling";
2025-02-22 10:38:19 -06:00
import { useEffect, useState } from "react";
import MiniMessage from "minimessage-js";
2025-02-15 20:06:35 -06:00
2025-02-15 17:18:58 -06:00
export function ServerList() {
2025-02-15 20:06:35 -06:00
const { servers, loading, serverCount, playerCount } = useServers();
2025-02-16 21:40:17 -06:00
const { itemsLength, fetchMoreData, hasMoreData, data } =
useInfiniteScrolling(servers);
2025-02-22 10:38:19 -06:00
const [motdList, setMotdList] = useState<{ name: string; motd: string }[]>(
[]
);
useEffect(() => {
const result: Array<{ name: string; motd: string }> = [];
const mm = MiniMessage.miniMessage();
servers.forEach((c) => {
try {
result.push({
name: c.name,
motd: mm.toHTML(mm.deserialize(c.motd)),
});
} catch (e) {
console.log(e);
}
});
setMotdList(result);
}, [servers]);
2025-02-15 20:06:35 -06:00
if (loading)
return (
<div className="absolute top-[50%] left-[50%]">
<Spinner />
</div>
);
return (
<main className="px-3 lg:px-16">
<h1 className="scroll-m-20 text-2xl font-extrabold tracking-tight lg:text-4xl mb-3">
Statistics
</h1>
<Statistics
totalServers={serverCount}
totalPlayers={playerCount}
topServer={servers[0]}
/>
2025-02-15 22:56:37 -06:00
<Separator className="my-6" />
2025-02-15 20:06:35 -06:00
<h1 className="scroll-m-20 text-2xl font-extrabold tracking-tight lg:text-4xl">
Servers
</h1>
2025-02-16 21:40:17 -06:00
<InfiniteScroll
dataLength={itemsLength}
next={fetchMoreData}
hasMore={hasMoreData}
loader={
<span className="mt-2 left-[50%] right-[50%] absolute">
<Spinner />
</span>
}
2025-02-16 21:40:17 -06:00
>
<div className="grid grid-cols-1 sm:grid-cols-2 md:grid-cols-3 lg:grid-cols-4 gap-2 mt-3">
{data.map((c) => (
2025-02-22 10:38:19 -06:00
<ServerCard
server={c}
key={c.name}
motd={motdList.find((x) => x.name === c.name)?.motd}
/>
2025-02-16 21:40:17 -06:00
))}
</div>
</InfiniteScroll>
2025-02-15 20:06:35 -06:00
</main>
);
2025-02-15 17:18:58 -06:00
}