Files
roblox/app/page.tsx
2025-12-27 14:20:22 +02:00

109 lines
3.0 KiB
TypeScript

"use client";
import {
BestFriendsHomeSect,
FriendsHomeSect
} from "@/components/roblox/FriendsOnline";
import { GameCard } from "@/components/roblox/GameCard";
import { HomeLoggedInHeader } from "@/components/site/HomeUserHeader";
import { Alert, AlertDescription, AlertTitle } from "@/components/ui/alert";
import { Card, CardContent } from "@/components/ui/card";
import {
getOmniRecommendationsHome,
OmniRecommendation
} from "@/lib/omniRecommendation";
import { getThumbnails, ThumbnailRequest } from "@/lib/thumbnailLoader";
import { useQuery, useQueryClient } from "@tanstack/react-query";
import { AlertTriangleIcon } from "lucide-react";
export default function Home() {
const SORTS_ALLOWED_IDS = [100000003, 100000001];
const queryClient = useQueryClient();
const { data: rec, isLoading } = useQuery({
queryKey: ["omni-recommendations"],
queryFn: async () => {
const r = await getOmniRecommendationsHome();
if (r) {
// Prefetch game thumbnails into React Query cache
const gameRequests: ThumbnailRequest[] = Object.entries(
r.contentMetadata.Game
).map(([_, g]) => ({
type: "GameThumbnail" as const,
targetId: Number(g.rootPlaceId),
format: "webp",
size: "384x216"
}));
await queryClient.prefetchQuery({
queryKey: [
"thumbnails",
gameRequests.map((r) => r.targetId)
],
queryFn: () => getThumbnails(gameRequests)
});
}
return r;
},
staleTime: 1000 * 60 * 5, // 5 minutes
refetchOnWindowFocus: false
});
return (
<>
<HomeLoggedInHeader />
<div className="h-4" />
<BestFriendsHomeSect className="pt-2" />
<FriendsHomeSect className="pt-2" />
{/* <div className="justify-center w-screen px-8 pt-6">
<Alert variant="default" className="bg-base/50 space-x-2">
<AlertTriangleIcon />
<AlertTitle>Warning</AlertTitle>
<AlertDescription>
This is work in progress, you can follow the development
process on GitHub.
</AlertDescription>
</Alert>
</div> */}
<div className="p-4 space-y-8 no-scrollbar">
{isLoading || !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>
</>
);
}