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 ? (c.createdAt as Date).toISOString() : new 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 (
{/* Novel Header */}
{/* Cover */} {novel.title} {/* Info */}

{novel.title}

Tác giả: {novel.authorName} {novel.originalAuthorName && ({novel.originalAuthorName})}
{novel.originalTitle &&
Tên gốc: {novel.originalTitle}
}
Trạng thái: {novel.status}
{novelGenres.map((g, i) => ( {g.name} ))}
{/* Stats Row */}
{novel.totalChapters} Chương
{novel.views} Lượt đọc
{novel.bookmarkCount} Cất giữ
{novel.ratingCount} Đề cử
{/* Description */}

Giới Thiệu

{novel.description}
{/* Chapter list or series volumes */} {novel.seriesId ? (

Danh Sách Quyển

{seriesVolumes.map((volume, idx) => ( {idx + 1} {volume.title}

{volume.title}

{volume.totalChapters} chương

{volume.status} ))}
) : (

Danh Sách Chương

)} {/* Comments */}
) }