import { notFound } from "next/navigation" import Link from "next/link" import { BookOpen, Eye, BookMarked, User, Clock, Layers } from "lucide-react" import { formatViews } from "@/lib/utils" import { GenreBadge } from "@/components/genre-badge" import { StarRating } from "@/components/star-rating" import { ChapterList } from "@/components/chapter-list" import { CommentSection } from "@/components/comment-section" import { NovelDetailActions } from "./novel-detail-actions" import { prisma } from "@/lib/prisma" import connectToMongoDB from "@/lib/mongoose" import { Chapter } from "@/lib/models/chapter" import { getNovelStatusBadgeClass } from "@/lib/novel-status" export const dynamic = "force-dynamic" export default async function NovelDetailPage({ params, searchParams }: { params: Promise<{ slug: string }>, searchParams: Promise<{ page?: string }> }) { const { slug } = await params const { page } = await searchParams const currentPage = parseInt(page || "1") const limit = 20 const novel = await prisma.novel.findUnique({ where: { slug }, include: { genres: { include: { genre: true } } } }) if (!novel) { notFound() } let formattedChapters: any[] = [] let totalChapters = 0 let totalPages = 1 let firstChapterNumber: number | undefined let seriesVolumes: Array<{ id: string slug: string title: string status: string totalChapters: number coverUrl: string | null updatedAt: Date }> = [] await connectToMongoDB() if (novel.seriesId) { const [firstChapter, volumes] = await Promise.all([ Chapter.findOne({ novelId: novel.id }).sort({ number: 1 }).select("number").lean(), prisma.novel.findMany({ where: { seriesId: novel.seriesId }, select: { id: true, slug: true, title: true, status: true, totalChapters: true, coverUrl: true, updatedAt: true, }, orderBy: { createdAt: "asc" }, }), ]) firstChapterNumber = (firstChapter as any)?.number seriesVolumes = volumes } else { const skip = (currentPage - 1) * limit const [chapters, chaptersCount, firstChapter] = await Promise.all([ Chapter.find({ novelId: novel.id }) .sort({ number: 1 }) .skip(skip) .limit(limit) .select("id novelId number title createdAt views volumeNumber volumeTitle volumeChapterNumber") .lean(), Chapter.countDocuments({ novelId: novel.id }), Chapter.findOne({ novelId: novel.id }).sort({ number: 1 }).select("number").lean(), ]) totalChapters = chaptersCount totalPages = Math.ceil(totalChapters / limit) firstChapterNumber = (firstChapter as any)?.number formattedChapters = chapters.map(c => ({ id: c._id.toString(), novelId: c.novelId, number: c.number, volumeNumber: (c as any).volumeNumber ?? null, volumeTitle: (c as any).volumeTitle ?? null, volumeChapterNumber: (c as any).volumeChapterNumber ?? null, title: c.title, createdAt: (c.createdAt as Date).toISOString(), views: c.views || 0, content: "" })) } const commentsData = await prisma.comment.findMany({ where: { novelId: novel.id, chapterId: null }, include: { user: true }, orderBy: { createdAt: "desc" } }) // Format explicitly as the CommentProp type const comments = commentsData.map(c => ({ id: c.id, userId: c.user.id, username: c.user.name || "User", avatarColor: c.user.image || "bg-primary", novelId: c.novelId, content: c.content, createdAt: c.createdAt.toISOString().split("T")[0] })) const chapterCommentsData = await prisma.comment.findMany({ where: { novelId: novel.id, chapterId: { not: null } }, include: { user: true }, orderBy: { createdAt: "desc" } }) // Format explicitly as the CommentProp type const chapterComments = chapterCommentsData.map(c => ({ id: c.id, userId: c.user.id, username: c.user.name || "User", avatarColor: c.user.image || "bg-primary", novelId: c.novelId, content: c.content, createdAt: c.createdAt.toISOString().split("T")[0] })) const novelGenres = novel.genres.map(ng => ng.genre) || [] return (
{volume.title}
{volume.totalChapters} chương