Files
blog-frontend/src/app/category/[id]/page.tsx
2025-12-27 15:45:12 +09:00

144 lines
5.2 KiB
TypeScript

'use client';
import { use, useState, useEffect } from 'react';
import { useQuery } from '@tanstack/react-query';
import { getPostsByCategory } from '@/api/posts';
import PostCard from '@/components/post/PostCard';
import PostListItem from '@/components/post/PostListItem'; // 추가
import { Loader2, ChevronLeft, ChevronRight, LayoutGrid, List } from 'lucide-react';
import { notFound } from 'next/navigation';
import { clsx } from 'clsx';
export default function CategoryPage({ params }: { params: Promise<{ id: string }> }) {
const { id } = use(params);
const categoryName = decodeURIComponent(id);
const apiCategoryName = categoryName === 'uncategorized' ? '미분류' : categoryName;
const [page, setPage] = useState(0);
const [viewMode, setViewMode] = useState<'grid' | 'list'>('grid');
const PAGE_SIZE = 10;
// 💡 로컬 스토리지에서 뷰 모드 불러오기
useEffect(() => {
const savedMode = localStorage.getItem('postViewMode') as 'grid' | 'list';
if (savedMode) setViewMode(savedMode);
}, []);
const handleViewModeChange = (mode: 'grid' | 'list') => {
setViewMode(mode);
localStorage.setItem('postViewMode', mode);
};
const { data: postsData, isLoading, error, isPlaceholderData } = useQuery({
queryKey: ['posts', 'category', apiCategoryName, page],
queryFn: () => getPostsByCategory(apiCategoryName, page, PAGE_SIZE),
placeholderData: (previousData) => previousData,
});
if (isLoading || (postsData === undefined && !error)) {
return (
<div className="flex justify-center items-center h-64">
<Loader2 className="animate-spin text-blue-500" size={40} />
</div>
);
}
if (error) {
return (
<div className="text-center py-10 text-red-500">
.<br/>
<span className="text-sm text-gray-400"> .</span>
</div>
);
}
const posts = postsData?.content || [];
const handlePrevPage = () => setPage((old) => Math.max(old - 1, 0));
const handleNextPage = () => {
if (!postsData?.last) {
setPage((old) => old + 1);
}
};
return (
<div className="max-w-4xl mx-auto px-4 py-8">
{/* 헤더 영역 수정 */}
<div className="flex items-center justify-between mb-6">
<h1 className="text-2xl font-bold text-gray-800 flex items-baseline gap-2">
{apiCategoryName} <span className="text-gray-400 text-lg font-normal"> </span>
</h1>
{/* 뷰 모드 버튼 */}
<div className="flex items-center gap-1 bg-gray-100 p-1 rounded-lg">
<button
onClick={() => handleViewModeChange('grid')}
className={clsx(
"p-2 rounded-md transition-all duration-200",
viewMode === 'grid' ? "bg-white text-blue-600 shadow-sm" : "text-gray-400 hover:text-gray-600"
)}
title="카드형 보기"
>
<LayoutGrid size={18} />
</button>
<button
onClick={() => handleViewModeChange('list')}
className={clsx(
"p-2 rounded-md transition-all duration-200",
viewMode === 'list' ? "bg-white text-blue-600 shadow-sm" : "text-gray-400 hover:text-gray-600"
)}
title="리스트형 보기"
>
<List size={18} />
</button>
</div>
</div>
{posts.length === 0 ? (
<div className="text-center py-20 bg-gray-50 rounded-lg border border-gray-100">
<p className="text-gray-400 mb-2"> .</p>
</div>
) : (
<>
{viewMode === 'grid' ? (
<div className="grid gap-6 md:grid-cols-2">
{posts.map((post) => (
<PostCard key={post.id} post={post} />
))}
</div>
) : (
<div className="flex flex-col border-t border-gray-100">
{posts.map((post) => (
<PostListItem key={post.id} post={post} />
))}
</div>
)}
<div className="flex justify-center items-center gap-6 mt-12 mb-8">
<button
onClick={handlePrevPage}
disabled={page === 0}
className="p-2 rounded-full hover:bg-gray-100 disabled:opacity-30 disabled:hover:bg-transparent transition-colors disabled:cursor-not-allowed"
aria-label="이전 페이지"
>
<ChevronLeft size={24} />
</button>
<span className="text-sm font-medium text-gray-600">
Page <span className="text-gray-900 font-bold">{page + 1}</span> {postsData && postsData.totalPages > 0 && `/ ${postsData.totalPages}`}
</span>
<button
onClick={handleNextPage}
disabled={postsData?.last || isPlaceholderData}
className="p-2 rounded-full hover:bg-gray-100 disabled:opacity-30 disabled:hover:bg-transparent transition-colors disabled:cursor-not-allowed"
aria-label="다음 페이지"
>
<ChevronRight size={24} />
</button>
</div>
</>
)}
</div>
);
}