- Sign all image URLs server-side with 60s expiry presigned URLs - Add /api/pages endpoint for batched page fetching (7 per batch) - PageReader prefetches next batch when user scrolls to 3rd page - Move chapter count badge outside overflow-hidden for 3D effect - Fix missing URL signing on search and genre pages - Extract signCoverUrls helper to reduce duplication - Clamp API limit param to prevent abuse Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
33 lines
1007 B
TypeScript
33 lines
1007 B
TypeScript
import { prisma } from "@/lib/db";
|
|
import { signCoverUrls } from "@/lib/r2";
|
|
import { TrendingCarousel } from "@/components/TrendingCarousel";
|
|
import { GenreTabs } from "@/components/GenreTabs";
|
|
|
|
export const dynamic = "force-dynamic";
|
|
|
|
export default async function Home() {
|
|
const manga = await prisma.manga.findMany({
|
|
where: { status: "PUBLISHED" },
|
|
orderBy: { updatedAt: "desc" },
|
|
include: { _count: { select: { chapters: true } } },
|
|
});
|
|
|
|
const signedManga = await signCoverUrls(manga);
|
|
|
|
// Top 10 for trending
|
|
const trending = signedManga.slice(0, 10);
|
|
|
|
// Extract unique genres
|
|
const genres = [...new Set(signedManga.map((m) => m.genre))].sort();
|
|
|
|
return (
|
|
<div className="max-w-6xl mx-auto px-4 py-5 space-y-8">
|
|
{/* Trending section — Webtoon-style ranked carousel */}
|
|
<TrendingCarousel manga={trending} />
|
|
|
|
{/* Genre browsing section — horizontal tabs + filtered grid */}
|
|
<GenreTabs manga={signedManga} genres={genres} />
|
|
</div>
|
|
);
|
|
}
|