From 50dbe7c27b31c32e8e9f62de67f858f9e35a30ea Mon Sep 17 00:00:00 2001 From: naarang <93020785+naarang@users.noreply.github.com> Date: Wed, 13 Nov 2024 11:08:59 +0900 Subject: [PATCH] =?UTF-8?q?Feature=20#40:=20pagination=20=EC=BB=B4?= =?UTF-8?q?=ED=8F=AC=EB=84=8C=ED=8A=B8=20=EA=B5=AC=ED=98=84=ED=95=98?= =?UTF-8?q?=EA=B8=B0=20(#43)?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit * feat(#36): 레이아웃 적용 및 헤더 컴포넌트 구현하기 - 로그인 여부에 따른 헤더 컴포넌트 구현하기 - 디렉토리 기반 라우팅으로 레이아웃 적용하기 * feat(#40): 페이지네이션 간단히 구현하기 - shadcn 페이지네이션 컴포넌트 적용하기 - 3개 이상 양끝 페이지와 차이가 나면 ...으로 표시하기 * feat(#40): 페이지네이션 위젯을 카드 리스트 위젯 안으로 넣기 및 페이지 선택 시 스크롤 상단으로 이동하기 - 추가로 현재 페이지 번호를 또 클릭했을 때는 페이지네이션 동작 안하도록하기 * fix(#40): 불필요한 console.log 제거하기 --- apps/frontend/package.json | 2 +- .../src/feature/Pagination/usePagination.tsx | 57 +++++++++++++ .../src/widget/LotusList/LotusCardList.tsx | 49 +++++++----- apps/frontend/src/widget/Pagination.tsx | 48 +++++++++++ packages/design/src/components/index.tsx | 1 + .../design/src/components/ui/pagination.tsx | 80 +++++++++++++++++++ pnpm-lock.yaml | 57 +++++++++++-- 7 files changed, 266 insertions(+), 28 deletions(-) create mode 100644 apps/frontend/src/feature/Pagination/usePagination.tsx create mode 100644 apps/frontend/src/widget/Pagination.tsx create mode 100644 packages/design/src/components/ui/pagination.tsx diff --git a/apps/frontend/package.json b/apps/frontend/package.json index 6bc3f87d..146beb3e 100644 --- a/apps/frontend/package.json +++ b/apps/frontend/package.json @@ -26,7 +26,7 @@ "react": "^18.3.1", "react-dom": "^18.3.1", "react-hook-form": "^7.53.2", - "react-icons": "^5.3.0" + "react-icons": "^5.3.0", "zod": "^3.23.8" }, "devDependencies": { diff --git a/apps/frontend/src/feature/Pagination/usePagination.tsx b/apps/frontend/src/feature/Pagination/usePagination.tsx new file mode 100644 index 00000000..1be14e19 --- /dev/null +++ b/apps/frontend/src/feature/Pagination/usePagination.tsx @@ -0,0 +1,57 @@ +import { useState } from 'react'; + +interface UsePaginationProps { + totalPages: number; + initialPage?: number; + onChangePage?: (page: number) => void; +} + +export function usePagination({ totalPages, initialPage = 1, onChangePage }: UsePaginationProps) { + const [currentPage, setCurrentPage] = useState(initialPage); + + const onClickPage = (page: number) => { + setCurrentPage(page); + onChangePage?.(page); + window.scrollTo({ top: 0, behavior: 'smooth' }); + }; + + const onClickPrevious = () => { + if (currentPage > 1) onClickPage(currentPage - 1); + }; + + const onClickNext = () => { + if (currentPage < totalPages) onClickPage(currentPage + 1); + }; + + // "첫 페이지 ... 현재 페이지와 앞뒤 1페이지 ... 마지막 페이지 "로 구성되도록 구현함 + const getPaginationItems = () => { + const items: (number | 'ellipsis')[] = []; + const showStartEllipsis = currentPage > 4; + const showEndEllipsis = currentPage < totalPages - 3; + + items.push(1); + + if (showStartEllipsis) items.push('ellipsis'); + + const startPage = showStartEllipsis ? Math.max(2, currentPage - 1) : 2; + const endPage = showEndEllipsis ? Math.min(totalPages - 1, currentPage + 1) : totalPages - 1; + + for (let page = startPage; page <= endPage; page++) { + items.push(page); + } + + if (showEndEllipsis) items.push('ellipsis'); + + if (totalPages > 1) items.push(totalPages); + + return items; + }; + + return { + currentPage, + onClickPage, + onClickPrevious, + onClickNext, + getPaginationItems + }; +} diff --git a/apps/frontend/src/widget/LotusList/LotusCardList.tsx b/apps/frontend/src/widget/LotusList/LotusCardList.tsx index 2cd0d0b0..20afcb46 100644 --- a/apps/frontend/src/widget/LotusList/LotusCardList.tsx +++ b/apps/frontend/src/widget/LotusList/LotusCardList.tsx @@ -1,5 +1,6 @@ import { Lotus } from '@/feature/Lotus'; import { LotusType } from '@/feature/Lotus/type'; +import { Pagination } from '@/widget/Pagination'; const LotusDummyData: LotusType = { link: 'https://example.com', @@ -14,29 +15,33 @@ const LotusDummyData: LotusType = { } }; +// TODO: 나중에 Props로 size 받아서 재사용하기 export function LotusCardList() { return ( -
- {new Array(10).fill(0).map((_, index) => ( - - - - -
- - -
- {LotusDummyData?.tags?.length ? ( - <> -
- - - ) : ( - <> - )} - - - ))} -
+ <> +
+ {new Array(10).fill(0).map((_, index) => ( + + + + +
+ + +
+ {LotusDummyData?.tags?.length ? ( + <> +
+ + + ) : ( + <> + )} + + + ))} +
+ console.log(page)} /> + ); } diff --git a/apps/frontend/src/widget/Pagination.tsx b/apps/frontend/src/widget/Pagination.tsx new file mode 100644 index 00000000..7bbd470e --- /dev/null +++ b/apps/frontend/src/widget/Pagination.tsx @@ -0,0 +1,48 @@ +import { + Pagination as PaginationBox, + PaginationContent, + PaginationEllipsis, + PaginationItem, + PaginationLink, + PaginationNext, + PaginationPrevious +} from '@froxy/design/components'; +import { usePagination } from '@/feature/Pagination/usePagination'; + +interface PaginationProps { + totalPages: number; + initialPage?: number; + onChangePage?: (page: number) => void; +} + +export function Pagination({ totalPages, initialPage = 1, onChangePage }: PaginationProps) { + const { currentPage, onClickPage, onClickPrevious, onClickNext, getPaginationItems } = usePagination({ + totalPages, + initialPage, + onChangePage + }); + + return ( + + + + + + {getPaginationItems().map((item, index) => ( + + {typeof item === 'number' ? ( + item !== currentPage && onClickPage(item)} isActive={item === currentPage}> + {item} + + ) : ( + + )} + + ))} + + + + + + ); +} diff --git a/packages/design/src/components/index.tsx b/packages/design/src/components/index.tsx index 86afb753..fac30e49 100644 --- a/packages/design/src/components/index.tsx +++ b/packages/design/src/components/index.tsx @@ -2,6 +2,7 @@ export * from './ui/badge'; export * from './ui/button'; export * from './ui/carousel'; export * from './ui/input'; +export * from './ui/pagination'; export * from './ui/typography'; export * from './ui/tabs'; export * from './Slot'; diff --git a/packages/design/src/components/ui/pagination.tsx b/packages/design/src/components/ui/pagination.tsx new file mode 100644 index 00000000..5be67417 --- /dev/null +++ b/packages/design/src/components/ui/pagination.tsx @@ -0,0 +1,80 @@ +import * as React from 'react'; +import { ChevronLeft, ChevronRight, MoreHorizontal } from 'lucide-react'; +import { ButtonProps, buttonVariants } from './button'; +import { cn } from '@/lib/utils'; + +const Pagination = ({ className, ...props }: React.ComponentProps<'nav'>) => ( +