2025-02-22 11:13:48 -06:00
|
|
|
/*
|
|
|
|
|
* MHSF, Minehut Server List
|
|
|
|
|
* All external content is rather licensed under the ECA Agreement
|
|
|
|
|
* located here: https://mhsf.app/docs/legal/external-content-agreement
|
|
|
|
|
*
|
|
|
|
|
* All code under MHSF is licensed under the MIT License
|
|
|
|
|
* by open source contributors
|
|
|
|
|
*
|
|
|
|
|
* Copyright (c) 2025 dvelo
|
|
|
|
|
*
|
|
|
|
|
* Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
|
|
|
* of this software and associated documentation files (the "Software"), to
|
|
|
|
|
* deal in the Software without restriction, including without limitation the
|
|
|
|
|
* rights to use, copy, modify, merge, publish, distribute, sublicense, and/or
|
|
|
|
|
* sell copies of the Software, and to permit persons to whom the Software is
|
|
|
|
|
* furnished to do so, subject to the following conditions:
|
|
|
|
|
*
|
|
|
|
|
* The above copyright notice and this permission notice shall be included in all
|
|
|
|
|
* copies or substantial portions of the Software.
|
|
|
|
|
*
|
|
|
|
|
* THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
|
|
|
|
* EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES
|
|
|
|
|
* OF MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
|
|
|
|
* NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT
|
|
|
|
|
* HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY,
|
|
|
|
|
* WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING
|
|
|
|
|
* FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR
|
|
|
|
|
* OTHER DEALINGS IN THE SOFTWARE.
|
|
|
|
|
*/
|
|
|
|
|
|
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}
|
2025-02-17 19:12:11 -06:00
|
|
|
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
|
|
|
}
|