From 78e9eee87c402dcaaa111c2c605f5aa34fe5af57 Mon Sep 17 00:00:00 2001 From: Haider Alshamma Date: Thu, 16 Nov 2023 16:54:50 -0500 Subject: [PATCH] feat: optionally scroll to the top after paginating --- src/Pagination/Pagination.story.tsx | 31 ++++++++++++++++++++- src/Pagination/Pagination.tsx | 43 ++++++++++++++++++++++++++--- 2 files changed, 69 insertions(+), 5 deletions(-) diff --git a/src/Pagination/Pagination.story.tsx b/src/Pagination/Pagination.story.tsx index cf2f2dfd5..de9844765 100644 --- a/src/Pagination/Pagination.story.tsx +++ b/src/Pagination/Pagination.story.tsx @@ -1,5 +1,8 @@ -import React from "react"; +import React, { useRef, useState } from "react"; import { action } from "@storybook/addon-actions"; +import { Flex } from "../Flex"; +import { Heading1 } from "../Type"; +import { Box } from "../Box"; import { Pagination } from "."; export default { @@ -24,6 +27,32 @@ export const _Pagination = () => ( ); +export const WithScrollTop = () => { + const [currentPage, setCurrentPage] = useState(1); + const ref = useRef(null); + + return ( + + + After paginating, this section should not be in the viewport + + + After paginating, the page should scroll to the top of this section + + setCurrentPage((p) => p + 1)} + onPrevious={() => setCurrentPage((p) => p - 1)} + onSelectPage={(page) => { + setCurrentPage(Number(page)); + }} + paginationScrollTargetRef={ref} + /> + + ); +}; + export const OnTheFirstPage = () => ; OnTheFirstPage.story = { diff --git a/src/Pagination/Pagination.tsx b/src/Pagination/Pagination.tsx index f67ba3290..49fe35f55 100644 --- a/src/Pagination/Pagination.tsx +++ b/src/Pagination/Pagination.tsx @@ -36,6 +36,7 @@ type PaginationProps = FlexProps & { nextAriaLabel?: string; previousLabel?: ReactNode; previousAriaLabel?: string; + paginationScrollTargetRef?: RefObject; }; function Pagination({ @@ -48,17 +49,34 @@ function Pagination({ nextLabel, previousAriaLabel, previousLabel, - previousLabel, + paginationScrollTargetRef, "aria-label": ariaLabel, ...restProps }: PaginationProps) { const { t } = useTranslation(); + const scrollToTop = () => { + if (paginationScrollTargetRef && paginationScrollTargetRef.current) { + const { top } = paginationScrollTargetRef.current.getBoundingClientRect(); + + window.scrollTo({ + top: window.scrollY + top, + behavior: "smooth", + }); + } + }; + return ( { + flushSync(() => { + onPrevious(); + }); + + scrollToTop(); + }} ariaLabel={previousAriaLabel} label={previousLabel} /> @@ -80,13 +98,30 @@ function Pagination({ disabled={isCurrentPage} aria-label={isCurrentPage ? null : t("go to page", { count: Number(page) })} key={page} - onClick={() => onSelectPage(page)} + onClick={() => { + flushSync(() => { + onSelectPage(page); + }); + + scrollToTop(); + }} > {page} ); })} - + { + flushSync(() => { + onNext(); + }); + + scrollToTop(); + }} + ariaLabel={nextAriaLabel} + label={nextLabel} + /> ); }