Add genre management functionality and update sidebar navigation

This commit is contained in:
2026-04-06 18:24:05 +00:00
parent d3f3d9c91a
commit c811135b92
7 changed files with 456 additions and 39 deletions
+83 -30
View File
@@ -6,6 +6,8 @@ import { readerApiFetch, readerApiFetchNullable } from "@/lib/server-api"
export const dynamic = "force-dynamic"
const PAGE_SIZE = 24
type GenreItem = {
id: string
name: string
@@ -29,41 +31,50 @@ type BrowseNovel = {
type BrowseResponse = {
items: BrowseNovel[]
totalCount: number
totalPages: number
currentPage: number
}
function collapseSeriesRows<T extends { id: string; seriesId?: string | null }>(rows: T[]): T[] {
const pickedSeries = new Set<string>()
const output: T[] = []
for (const row of rows) {
if (!row.seriesId) {
output.push(row)
continue
}
if (pickedSeries.has(row.seriesId)) continue
pickedSeries.add(row.seriesId)
output.push(row)
}
return output
}
export default async function GenreDetailPage({ params }: { params: Promise<{ slug: string }> }) {
export default async function GenreDetailPage({
params,
searchParams,
}: {
params: Promise<{ slug: string }>
searchParams: Promise<{ [key: string]: string | undefined }>
}) {
const { slug } = await params
const resolved = await searchParams
const sort = resolved.sort || "latest"
const requestedPage = Math.max(1, Number(resolved.page || "1") || 1)
const genres = await readerApiFetch<GenreItem[]>("/api/genres")
const genre = genres.find((item) => item.slug === slug) || null
const genre = await readerApiFetchNullable<GenreItem>(`/api/genres/${encodeURIComponent(slug)}`)
if (!genre) {
notFound()
}
const browse = await readerApiFetch<BrowseResponse>(`/api/novels/browse?genre=${encodeURIComponent(slug)}&sort=latest&page=1&limit=80`)
const browse = await readerApiFetch<BrowseResponse>(
`/api/novels/browse?genre=${encodeURIComponent(slug)}&sort=${sort}&page=${requestedPage}&limit=${PAGE_SIZE}&collapse_series=true`
)
const allNovels = collapseSeriesRows(browse.items).slice(0, 20)
const totalPages = Math.max(1, browse.totalPages || 1)
const currentPage = Math.min(requestedPage, totalPages)
const pageRangeStart = Math.max(1, currentPage - 2)
const pageRangeEnd = Math.min(totalPages, currentPage + 2)
const pageNumbers = Array.from(
{ length: pageRangeEnd - pageRangeStart + 1 },
(_, i) => pageRangeStart + i
)
const buildPageHref = (page: number) => {
const p = new URLSearchParams()
if (sort !== "latest") p.set("sort", sort)
p.set("page", String(page))
return `/the-loai/${slug}?${p.toString()}`
}
// Basic layout without sort for purely server side representation without search params. Optional searchParams can be added later if needed.
return (
<div className="mx-auto max-w-6xl px-4 py-6">
<div className="mb-6">
@@ -71,26 +82,68 @@ export default async function GenreDetailPage({ params }: { params: Promise<{ sl
<ChevronLeft className="h-4 w-4" /> Thể Loại
</Link>
<h1 className="text-2xl font-bold text-foreground">{genre.name}</h1>
<p className="mt-1 text-sm text-muted-foreground">{genre.description}</p>
{genre.description && (
<p className="mt-1 text-sm text-muted-foreground">{genre.description}</p>
)}
</div>
<div className="mb-4 flex items-center justify-between">
<p className="text-sm text-muted-foreground">{allNovels.length} truyện</p>
<div className="w-40" /> {/* Spacer for symmetry if we add sort later */}
<div className="mb-4 flex flex-wrap items-center justify-between gap-3">
<p className="text-sm text-muted-foreground">
{browse.totalCount} truyện {totalPages > 1 && `(Trang ${currentPage}/${totalPages})`}
</p>
<form method="GET" action={`/the-loai/${slug}`} className="flex items-center gap-2">
<select
name="sort"
defaultValue={sort}
className="flex h-9 items-center rounded-md border border-input bg-transparent px-3 py-2 text-sm shadow-sm focus:outline-none focus:ring-1 focus:ring-ring"
>
<option value="latest">Mới nhất</option>
<option value="popular">Xem nhiều</option>
<option value="rating">Đánh giá cao</option>
<option value="name">Theo tên</option>
</select>
<button type="submit" className="h-9 rounded-md bg-primary px-4 text-sm font-medium text-primary-foreground">Lọc</button>
</form>
</div>
{allNovels.length === 0 ? (
{browse.items.length === 0 ? (
<div className="flex flex-col items-center justify-center py-20 text-muted-foreground">
<p className="text-lg font-medium">Chưa truyện nào</p>
<p className="text-sm">Thể loại này chưa truyện, hãy quay lại sau.</p>
</div>
) : (
<div className="grid grid-cols-2 gap-4 sm:grid-cols-3 lg:grid-cols-4">
{allNovels.map((novel) => (
{browse.items.map((novel) => (
<NovelCard key={novel.id} novel={novel} />
))}
</div>
)}
{totalPages > 1 && (
<div className="mt-6 flex flex-wrap items-center justify-center gap-2">
<Link
href={buildPageHref(Math.max(1, currentPage - 1))}
className={`rounded-md border px-3 py-1.5 text-sm ${currentPage <= 1 ? "pointer-events-none opacity-50" : "hover:bg-muted"}`}
>
Trước
</Link>
{pageNumbers.map((page) => (
<Link
key={page}
href={buildPageHref(page)}
className={`rounded-md border px-3 py-1.5 text-sm ${page === currentPage ? "border-primary bg-primary text-primary-foreground" : "hover:bg-muted"}`}
>
{page}
</Link>
))}
<Link
href={buildPageHref(Math.min(totalPages, currentPage + 1))}
className={`rounded-md border px-3 py-1.5 text-sm ${currentPage >= totalPages ? "pointer-events-none opacity-50" : "hover:bg-muted"}`}
>
Sau
</Link>
</div>
)}
</div>
)
}