91 lines
2.6 KiB
TypeScript
91 lines
2.6 KiB
TypeScript
"use client";
|
|
|
|
import { FriendsHomeSect } from "@/components/FriendsOnlineSection";
|
|
import { GameCard } from "@/components/gameCard";
|
|
import { HomeLoggedInHeader } from "@/components/loggedInHeader";
|
|
import { VerifiedIcon } from "@/components/RobloxIcons";
|
|
import { Alert, AlertDescription, AlertTitle } from "@/components/ui/alert";
|
|
import { Card, CardContent } from "@/components/ui/card";
|
|
import {
|
|
getOmniRecommendationsHome,
|
|
OmniRecommendation
|
|
} from "@/lib/omniRecommendation";
|
|
import { loadThumbnails } from "@/lib/thumbnailLoader";
|
|
import { AlertTriangleIcon } from "lucide-react";
|
|
import { useEffect, useState } from "react";
|
|
|
|
export default function Home() {
|
|
const SORTS_ALLOWED_IDS = [100000003, 100000001];
|
|
const [rec, setRec] = useState<OmniRecommendation | null>(null);
|
|
useEffect(() => {
|
|
setTimeout(async () => {
|
|
const r = await getOmniRecommendationsHome();
|
|
if (r) {
|
|
setRec(r);
|
|
loadThumbnails(
|
|
Object.entries(r.contentMetadata.Game).map((a) => ({
|
|
type: "GameThumbnail",
|
|
targetId: Number(a[1].rootPlaceId),
|
|
format: "webp",
|
|
size: "384x216"
|
|
}))
|
|
).catch((a) => {});
|
|
}
|
|
}, 1000);
|
|
}, []);
|
|
|
|
return (
|
|
<div className="overflow-scroll no-scrollbar w-screen max-h-screen h-screen">
|
|
<HomeLoggedInHeader />
|
|
<FriendsHomeSect className="pt-8" />
|
|
<div className="justify-center w-screen px-8 pt-6">
|
|
<Alert variant="default" className="space-x-2">
|
|
<AlertTriangleIcon />
|
|
<AlertTitle>Warning</AlertTitle>
|
|
<AlertDescription>
|
|
This is work in progess, you can follow the development
|
|
process on GitHub.
|
|
</AlertDescription>
|
|
</Alert>
|
|
</div>
|
|
<div className="p-4 space-y-8 no-scrollbar">
|
|
{!rec ? (
|
|
<Card>
|
|
<CardContent className="p-4">
|
|
<div className="h-[200px] flex items-center justify-center">
|
|
<div className="animate-pulse text-muted-foreground">
|
|
{"Loading..."}
|
|
</div>
|
|
</div>
|
|
</CardContent>
|
|
</Card>
|
|
) : (
|
|
rec.sorts
|
|
.filter((a) => SORTS_ALLOWED_IDS.includes(a.topicId))
|
|
.map((sort, idx) => (
|
|
<div key={idx}>
|
|
<h1 className="text-2xl pb-2">{sort.topic}</h1>
|
|
<div className="grid grid-cols-1 md:grid-cols-2 lg:grid-cols-4 gap-4">
|
|
{(sort.recommendationList || []).map(
|
|
(recommendation, idxb) => {
|
|
const game =
|
|
rec.contentMetadata.Game[
|
|
recommendation.contentId.toString()
|
|
];
|
|
return (
|
|
<GameCard
|
|
key={idxb}
|
|
game={game}
|
|
/>
|
|
);
|
|
}
|
|
)}
|
|
</div>
|
|
</div>
|
|
))
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
}
|