@windstream/react-shared-components 0.1.38 → 0.1.41
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +635 -635
- package/dist/contentful/index.d.ts +80 -51
- package/dist/contentful/index.esm.js +4 -2
- package/dist/contentful/index.esm.js.map +1 -1
- package/dist/contentful/index.js +4 -2
- package/dist/contentful/index.js.map +1 -1
- package/dist/core.d.ts +4 -4
- package/dist/index.d.ts +3 -3
- package/dist/index.esm.js +7 -5
- package/dist/index.esm.js.map +1 -1
- package/dist/index.js +7 -5
- package/dist/index.js.map +1 -1
- package/dist/next/index.esm.js +3 -1
- package/dist/next/index.esm.js.map +1 -1
- package/dist/next/index.js +3 -1
- package/dist/next/index.js.map +1 -1
- package/dist/styles.css +1 -1
- package/dist/utils/index.d.ts +35 -6
- package/dist/utils/index.esm.js +1 -1
- package/dist/utils/index.esm.js.map +1 -1
- package/dist/utils/index.js +1 -1
- package/dist/utils/index.js.map +1 -1
- package/package.json +185 -182
- package/src/components/accordion/Accordion.stories.tsx +230 -230
- package/src/components/accordion/index.tsx +70 -70
- package/src/components/accordion/types.ts +12 -12
- package/src/components/alert-card/AlertCard.stories.tsx +171 -171
- package/src/components/alert-card/index.tsx +41 -41
- package/src/components/alert-card/types.ts +13 -13
- package/src/components/brand-button/BrandButton.stories.tsx +223 -223
- package/src/components/brand-button/helpers.ts +35 -35
- package/src/components/brand-button/index.tsx +114 -114
- package/src/components/brand-button/types.ts +37 -37
- package/src/components/button/Button.stories.tsx +108 -108
- package/src/components/button/index.tsx +27 -27
- package/src/components/button/types.ts +14 -14
- package/src/components/call-button/CallButton.stories.tsx +324 -324
- package/src/components/call-button/index.tsx +86 -86
- package/src/components/call-button/types.ts +11 -11
- package/src/components/checkbox/Checkbox.stories.tsx +247 -247
- package/src/components/checkbox/index.tsx +197 -197
- package/src/components/checkbox/types.ts +27 -27
- package/src/components/checklist/Checklist.stories.tsx +150 -150
- package/src/components/checklist/index.tsx +61 -61
- package/src/components/checklist/types.ts +17 -17
- package/src/components/collapse/Collapse.stories.tsx +255 -255
- package/src/components/collapse/index.tsx +46 -46
- package/src/components/collapse/types.ts +6 -6
- package/src/components/divider/Divider.stories.tsx +205 -205
- package/src/components/divider/index.tsx +22 -22
- package/src/components/divider/type.ts +3 -3
- package/src/components/image/Image.stories.tsx +113 -113
- package/src/components/image/index.tsx +25 -25
- package/src/components/image/types.ts +40 -40
- package/src/components/input/Input.stories.tsx +325 -325
- package/src/components/input/index.tsx +177 -177
- package/src/components/input/types.ts +37 -37
- package/src/components/link/Link.stories.tsx +163 -163
- package/src/components/link/index.tsx +109 -109
- package/src/components/link/types.ts +25 -25
- package/src/components/list/List.stories.tsx +272 -272
- package/src/components/list/index.tsx +88 -88
- package/src/components/list/list-item/index.tsx +38 -38
- package/src/components/list/list-item/types.ts +13 -13
- package/src/components/list/types.ts +29 -29
- package/src/components/material-icon/MaterialIcon.stories.tsx +322 -322
- package/src/components/material-icon/constants.ts +98 -98
- package/src/components/material-icon/index.tsx +47 -47
- package/src/components/material-icon/types.ts +31 -31
- package/src/components/modal/Modal.stories.tsx +171 -171
- package/src/components/modal/index.tsx +164 -164
- package/src/components/modal/types.ts +24 -24
- package/src/components/next-image/index.tsx +54 -54
- package/src/components/next-image/types.ts +1 -1
- package/src/components/pagination/index.tsx +100 -100
- package/src/components/pagination/types.ts +6 -6
- package/src/components/radio-button/RadioButton.stories.tsx +307 -307
- package/src/components/radio-button/index.tsx +75 -75
- package/src/components/radio-button/types.ts +21 -21
- package/src/components/see-more/SeeMore.stories.tsx +181 -181
- package/src/components/see-more/index.tsx +44 -44
- package/src/components/see-more/types.ts +4 -4
- package/src/components/select/Select.stories.tsx +411 -411
- package/src/components/select/index.tsx +155 -155
- package/src/components/select/types.ts +36 -36
- package/src/components/select-plan-button/SelectPlanButton.stories.tsx +184 -184
- package/src/components/select-plan-button/index.tsx +63 -63
- package/src/components/select-plan-button/types.ts +17 -17
- package/src/components/skeleton/Skeleton.stories.tsx +179 -179
- package/src/components/skeleton/index.tsx +61 -61
- package/src/components/skeleton/types.ts +4 -4
- package/src/components/spinner/Spinner.stories.tsx +335 -335
- package/src/components/spinner/index.tsx +44 -44
- package/src/components/spinner/types.ts +5 -5
- package/src/components/text/Text.stories.tsx +321 -321
- package/src/components/text/index.tsx +25 -25
- package/src/components/text/types.ts +45 -45
- package/src/components/tooltip/Tooltip.stories.tsx +219 -219
- package/src/components/tooltip/index.tsx +74 -74
- package/src/components/tooltip/types.ts +7 -7
- package/src/components/view-cart-button/ViewCartButton.stories.tsx +252 -252
- package/src/components/view-cart-button/index.tsx +42 -42
- package/src/components/view-cart-button/types.ts +5 -5
- package/src/contentful/blocks/accordion/Accordion.stories.tsx +29 -29
- package/src/contentful/blocks/accordion/index.tsx +62 -62
- package/src/contentful/blocks/accordion/types.ts +17 -17
- package/src/contentful/blocks/address-input-banner/index.tsx +52 -52
- package/src/contentful/blocks/address-input-banner/types.ts +14 -14
- package/src/contentful/blocks/anchored-bottom-banner/index.tsx +70 -70
- package/src/contentful/blocks/anchored-bottom-banner/types.ts +10 -10
- package/src/contentful/blocks/blogs-grid/index.tsx +134 -134
- package/src/contentful/blocks/blogs-grid/types.ts +26 -26
- package/src/contentful/blocks/breadcrumbs/index.tsx +51 -51
- package/src/contentful/blocks/breadcrumbs/types.ts +5 -5
- package/src/contentful/blocks/button/Button.stories.tsx +40 -40
- package/src/contentful/blocks/button/index.tsx +129 -129
- package/src/contentful/blocks/button/types.ts +39 -39
- package/src/contentful/blocks/callout/Callout.stories.tsx +23 -23
- package/src/contentful/blocks/callout/index.tsx +88 -88
- package/src/contentful/blocks/callout/types.ts +15 -15
- package/src/contentful/blocks/cards/Cards.stories.tsx +23 -23
- package/src/contentful/blocks/cards/blog-card/index.tsx +110 -110
- package/src/contentful/blocks/cards/blog-card/types.ts +18 -18
- package/src/contentful/blocks/cards/index.tsx +13 -13
- package/src/contentful/blocks/cards/product-card/index.tsx +251 -251
- package/src/contentful/blocks/cards/product-card/types.ts +28 -28
- package/src/contentful/blocks/cards/simple-card/index.tsx +89 -89
- package/src/contentful/blocks/cards/simple-card/types.ts +28 -28
- package/src/contentful/blocks/cards/testimonial-card/index.tsx +90 -90
- package/src/contentful/blocks/cards/testimonial-card/types.tsx +12 -12
- package/src/contentful/blocks/cards/types.ts +1 -1
- package/src/contentful/blocks/carousel/Carousel.stories.tsx +23 -23
- package/src/contentful/blocks/carousel/helper.tsx +440 -440
- package/src/contentful/blocks/carousel/index.tsx +85 -85
- package/src/contentful/blocks/carousel/types.ts +144 -144
- package/src/contentful/blocks/comparison-table/index.tsx +27 -27
- package/src/contentful/blocks/comparison-table/types.ts +6 -6
- package/src/contentful/blocks/cookiebanner/index.tsx +146 -146
- package/src/contentful/blocks/cookiebanner/type.ts +7 -7
- package/src/contentful/blocks/cta-callout/CtaCallout.stories.tsx +46 -46
- package/src/contentful/blocks/cta-callout/index.tsx +69 -69
- package/src/contentful/blocks/cta-callout/types.ts +26 -26
- package/src/contentful/blocks/dynamic-tabs/index.tsx +204 -204
- package/src/contentful/blocks/dynamic-tabs/types.ts +21 -21
- package/src/contentful/blocks/find-kinetic/FindKinetic.stories.tsx +23 -23
- package/src/contentful/blocks/find-kinetic/index.tsx +130 -130
- package/src/contentful/blocks/find-kinetic/types.ts +19 -19
- package/src/contentful/blocks/floating-banner/FloatingBanner.stories.tsx +34 -34
- package/src/contentful/blocks/floating-banner/index.tsx +97 -97
- package/src/contentful/blocks/floating-banner/types.ts +22 -22
- package/src/contentful/blocks/footer/Footer.stories.tsx +30 -30
- package/src/contentful/blocks/footer/index.tsx +90 -90
- package/src/contentful/blocks/footer/types.ts +13 -13
- package/src/contentful/blocks/image-promo-bar/ImagePromoBar.stories.tsx +23 -23
- package/src/contentful/blocks/image-promo-bar/helper.tsx +28 -28
- package/src/contentful/blocks/image-promo-bar/index.tsx +246 -240
- package/src/contentful/blocks/image-promo-bar/types.ts +44 -44
- package/src/contentful/blocks/image-promo-bar/vimeo-embed.tsx +93 -93
- package/src/contentful/blocks/image-promo-bar/youtube-embed.tsx +46 -46
- package/src/contentful/blocks/modal/constants.ts +53 -53
- package/src/contentful/blocks/modal/index.tsx +107 -103
- package/src/contentful/blocks/modal/types.ts +12 -12
- package/src/contentful/blocks/navigation/desktop-link-groups.tsx/index.tsx +113 -113
- package/src/contentful/blocks/navigation/index.tsx +394 -394
- package/src/contentful/blocks/navigation/mobile-link-groups.tsx/index.tsx +82 -82
- package/src/contentful/blocks/navigation/types.ts +41 -41
- package/src/contentful/blocks/primary-hero/PrimaryHero.stories.tsx +23 -23
- package/src/contentful/blocks/primary-hero/index.tsx +241 -234
- package/src/contentful/blocks/primary-hero/types.ts +37 -35
- package/src/contentful/blocks/search-block/index.tsx +90 -90
- package/src/contentful/blocks/search-block/types.ts +15 -15
- package/src/contentful/blocks/shape-background-wrapper/ShapeBackgroundWrapper.stories.tsx +26 -26
- package/src/contentful/blocks/shape-background-wrapper/index.tsx +124 -124
- package/src/contentful/blocks/shape-background-wrapper/types.ts +36 -36
- package/src/contentful/blocks/text/Text.stories.tsx +23 -23
- package/src/contentful/blocks/text/index.tsx +12 -12
- package/src/contentful/blocks/text/types.ts +1 -1
- package/src/contentful/index.ts +96 -81
- package/src/hooks/contentful/use-contentful-rich-text.tsx +310 -0
- package/src/hooks/contentful/use-processed-check-list.ts +63 -0
- package/src/hooks/use-body-scroll-lock.ts +34 -34
- package/src/hooks/use-carousel-swipe.ts +264 -264
- package/src/hooks/use-outside-click.ts +17 -17
- package/src/index.ts +101 -101
- package/src/next/index.ts +5 -5
- package/src/setupTests.ts +46 -46
- package/src/stories/DocsTemplate.tsx +24 -24
- package/src/styles/globals.css +343 -343
- package/src/types/global.d.ts +9 -9
- package/src/types/micro-components.ts +99 -99
- package/src/types/utm.ts +49 -49
- package/src/utils/contentful/to-document.ts +24 -0
- package/src/utils/cookie.ts +84 -80
- package/src/utils/cx.ts +49 -0
- package/src/utils/index.ts +38 -65
- package/src/utils/utm.ts +221 -221
|
@@ -1,24 +1,24 @@
|
|
|
1
|
-
import { Props as RModalProps } from "react-modal";
|
|
2
|
-
|
|
3
|
-
export type Size = "xl" | "lg" | "md" | "sm" | "xs";
|
|
4
|
-
export type Shape = "rounded" | "default";
|
|
5
|
-
export type Animation = "popper" | "bottomSheet";
|
|
6
|
-
|
|
7
|
-
type OwnProps = {
|
|
8
|
-
size?: Size;
|
|
9
|
-
shape?: Shape;
|
|
10
|
-
title?: string;
|
|
11
|
-
bodyClassName?: string | null | undefined;
|
|
12
|
-
closeWrapperClassName?: string | null | undefined;
|
|
13
|
-
closeButtonClassName?: string | null | undefined;
|
|
14
|
-
bodyStyle?: React.CSSProperties;
|
|
15
|
-
centered?: boolean;
|
|
16
|
-
hideScrollOnIsOpenFalse?: boolean;
|
|
17
|
-
parentSelector?: string;
|
|
18
|
-
animation?: Animation;
|
|
19
|
-
hideCloseButton?: boolean;
|
|
20
|
-
["data-testid"]?: string;
|
|
21
|
-
["data-cy"]?: string;
|
|
22
|
-
};
|
|
23
|
-
|
|
24
|
-
export type ModalProps = OwnProps & RModalProps;
|
|
1
|
+
import { Props as RModalProps } from "react-modal";
|
|
2
|
+
|
|
3
|
+
export type Size = "xl" | "lg" | "md" | "sm" | "xs";
|
|
4
|
+
export type Shape = "rounded" | "default";
|
|
5
|
+
export type Animation = "popper" | "bottomSheet";
|
|
6
|
+
|
|
7
|
+
type OwnProps = {
|
|
8
|
+
size?: Size;
|
|
9
|
+
shape?: Shape;
|
|
10
|
+
title?: string;
|
|
11
|
+
bodyClassName?: string | null | undefined;
|
|
12
|
+
closeWrapperClassName?: string | null | undefined;
|
|
13
|
+
closeButtonClassName?: string | null | undefined;
|
|
14
|
+
bodyStyle?: React.CSSProperties;
|
|
15
|
+
centered?: boolean;
|
|
16
|
+
hideScrollOnIsOpenFalse?: boolean;
|
|
17
|
+
parentSelector?: string;
|
|
18
|
+
animation?: Animation;
|
|
19
|
+
hideCloseButton?: boolean;
|
|
20
|
+
["data-testid"]?: string;
|
|
21
|
+
["data-cy"]?: string;
|
|
22
|
+
};
|
|
23
|
+
|
|
24
|
+
export type ModalProps = OwnProps & RModalProps;
|
|
@@ -1,54 +1,54 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
|
|
3
|
-
import { forwardRef } from "react";
|
|
4
|
-
import NextJsImage, {
|
|
5
|
-
type ImageLoaderProps,
|
|
6
|
-
type ImageProps as NextImageProps,
|
|
7
|
-
} from "next/image";
|
|
8
|
-
|
|
9
|
-
import { cx } from "@shared/utils";
|
|
10
|
-
|
|
11
|
-
export interface NextImageComponentProps extends NextImageProps {
|
|
12
|
-
className?: string;
|
|
13
|
-
}
|
|
14
|
-
|
|
15
|
-
/**
|
|
16
|
-
* Image loader that uses Contentful's Image API to serve optimized WebP images
|
|
17
|
-
* at the requested width and quality, avoiding an extra round-trip through
|
|
18
|
-
* the Next.js image optimization server.
|
|
19
|
-
*/
|
|
20
|
-
const contentfulImageLoader = ({ src, width, quality }: ImageLoaderProps) => {
|
|
21
|
-
const url = new URL(src);
|
|
22
|
-
url.searchParams.set("w", String(width));
|
|
23
|
-
url.searchParams.set("q", String(quality || 90));
|
|
24
|
-
url.searchParams.set("fm", "webp");
|
|
25
|
-
return url.toString();
|
|
26
|
-
};
|
|
27
|
-
|
|
28
|
-
export const NextImage = forwardRef<HTMLImageElement, NextImageComponentProps>(
|
|
29
|
-
({ className, ...props }, ref) => {
|
|
30
|
-
const srcString = typeof props.src === "string" ? props.src : "";
|
|
31
|
-
const urlWithoutParams = srcString.toLowerCase().split("?")[0] || "";
|
|
32
|
-
const isContentfulImage = srcString.includes("images.ctfassets.net");
|
|
33
|
-
const isSvgFromContentful =
|
|
34
|
-
isContentfulImage && urlWithoutParams.endsWith(".svg");
|
|
35
|
-
|
|
36
|
-
// Use Contentful's Image API for non-SVG Contentful images;
|
|
37
|
-
// skip optimization entirely for SVGs.
|
|
38
|
-
const loaderProps =
|
|
39
|
-
isContentfulImage && !isSvgFromContentful
|
|
40
|
-
? { loader: contentfulImageLoader, unoptimized: false }
|
|
41
|
-
: { unoptimized: isSvgFromContentful };
|
|
42
|
-
|
|
43
|
-
return (
|
|
44
|
-
<NextJsImage
|
|
45
|
-
ref={ref}
|
|
46
|
-
className={cx(className)}
|
|
47
|
-
{...props}
|
|
48
|
-
{...loaderProps}
|
|
49
|
-
/>
|
|
50
|
-
);
|
|
51
|
-
}
|
|
52
|
-
);
|
|
53
|
-
|
|
54
|
-
NextImage.displayName = "NextImage";
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import { forwardRef } from "react";
|
|
4
|
+
import NextJsImage, {
|
|
5
|
+
type ImageLoaderProps,
|
|
6
|
+
type ImageProps as NextImageProps,
|
|
7
|
+
} from "next/image";
|
|
8
|
+
|
|
9
|
+
import { cx } from "@shared/utils";
|
|
10
|
+
|
|
11
|
+
export interface NextImageComponentProps extends NextImageProps {
|
|
12
|
+
className?: string;
|
|
13
|
+
}
|
|
14
|
+
|
|
15
|
+
/**
|
|
16
|
+
* Image loader that uses Contentful's Image API to serve optimized WebP images
|
|
17
|
+
* at the requested width and quality, avoiding an extra round-trip through
|
|
18
|
+
* the Next.js image optimization server.
|
|
19
|
+
*/
|
|
20
|
+
const contentfulImageLoader = ({ src, width, quality }: ImageLoaderProps) => {
|
|
21
|
+
const url = new URL(src);
|
|
22
|
+
url.searchParams.set("w", String(width));
|
|
23
|
+
url.searchParams.set("q", String(quality || 90));
|
|
24
|
+
url.searchParams.set("fm", "webp");
|
|
25
|
+
return url.toString();
|
|
26
|
+
};
|
|
27
|
+
|
|
28
|
+
export const NextImage = forwardRef<HTMLImageElement, NextImageComponentProps>(
|
|
29
|
+
({ className, ...props }, ref) => {
|
|
30
|
+
const srcString = typeof props.src === "string" ? props.src : "";
|
|
31
|
+
const urlWithoutParams = srcString.toLowerCase().split("?")[0] || "";
|
|
32
|
+
const isContentfulImage = srcString.includes("images.ctfassets.net");
|
|
33
|
+
const isSvgFromContentful =
|
|
34
|
+
isContentfulImage && urlWithoutParams.endsWith(".svg");
|
|
35
|
+
|
|
36
|
+
// Use Contentful's Image API for non-SVG Contentful images;
|
|
37
|
+
// skip optimization entirely for SVGs.
|
|
38
|
+
const loaderProps =
|
|
39
|
+
isContentfulImage && !isSvgFromContentful
|
|
40
|
+
? { loader: contentfulImageLoader, unoptimized: false }
|
|
41
|
+
: { unoptimized: isSvgFromContentful };
|
|
42
|
+
|
|
43
|
+
return (
|
|
44
|
+
<NextJsImage
|
|
45
|
+
ref={ref}
|
|
46
|
+
className={cx(className)}
|
|
47
|
+
{...props}
|
|
48
|
+
{...loaderProps}
|
|
49
|
+
/>
|
|
50
|
+
);
|
|
51
|
+
}
|
|
52
|
+
);
|
|
53
|
+
|
|
54
|
+
NextImage.displayName = "NextImage";
|
|
@@ -1 +1 @@
|
|
|
1
|
-
export type { NextImageComponentProps } from "./index";
|
|
1
|
+
export type { NextImageComponentProps } from "./index";
|
|
@@ -1,100 +1,100 @@
|
|
|
1
|
-
"use client";
|
|
2
|
-
|
|
3
|
-
import React from "react";
|
|
4
|
-
import { PaginationProps } from "./types";
|
|
5
|
-
|
|
6
|
-
import { Button } from "@shared/components/button";
|
|
7
|
-
import { MaterialIcon } from "@shared/components/material-icon";
|
|
8
|
-
|
|
9
|
-
export const Pagination: React.FC<PaginationProps> = ({
|
|
10
|
-
currentPage,
|
|
11
|
-
totalPages,
|
|
12
|
-
onPageChange,
|
|
13
|
-
ariaLabel = "Pagination",
|
|
14
|
-
}: PaginationProps) => {
|
|
15
|
-
const pageItems = buildPageItems(currentPage, totalPages);
|
|
16
|
-
|
|
17
|
-
const navBtnBase =
|
|
18
|
-
"inline-flex items-center justify-center w-9 h-9 rounded-full bg-transparent cursor-pointer transition-colors duration-150 hover:border-text-brand hover:text-text-brand disabled:opacity-40 disabled:cursor-not-allowed";
|
|
19
|
-
|
|
20
|
-
const pageBtnBase =
|
|
21
|
-
"inline-flex items-center justify-center w-9 h-9 rounded-lg text-body2 text-gray-700 bg-transparent border-none cursor-pointer transition-colors duration-150 hover:bg-gray-100 hover:text-gray-900";
|
|
22
|
-
|
|
23
|
-
const pageBtnActive = "bg-bg-surface-active font-bold hover:bg-bg-gray-300";
|
|
24
|
-
|
|
25
|
-
return (
|
|
26
|
-
<nav
|
|
27
|
-
className="flex items-center justify-center gap-1 px-6 pb-12"
|
|
28
|
-
aria-label={ariaLabel}
|
|
29
|
-
>
|
|
30
|
-
{/* Previous */}
|
|
31
|
-
<Button
|
|
32
|
-
className={navBtnBase}
|
|
33
|
-
onClick={() => onPageChange(currentPage - 1)}
|
|
34
|
-
disabled={currentPage === 1}
|
|
35
|
-
aria-label="Go to previous page"
|
|
36
|
-
>
|
|
37
|
-
<MaterialIcon name="chevron_left" size={32} aria-hidden="true" />
|
|
38
|
-
</Button>
|
|
39
|
-
|
|
40
|
-
{/* Page numbers */}
|
|
41
|
-
{pageItems.map((item, idx) =>
|
|
42
|
-
item === "..." ? (
|
|
43
|
-
<span
|
|
44
|
-
key={`ellipsis-${idx}`}
|
|
45
|
-
className="inline-flex h-9 w-9 cursor-default items-center justify-center text-body2 text-gray-500"
|
|
46
|
-
aria-hidden="true"
|
|
47
|
-
>
|
|
48
|
-
…
|
|
49
|
-
</span>
|
|
50
|
-
) : (
|
|
51
|
-
<Button
|
|
52
|
-
key={item}
|
|
53
|
-
className={`${pageBtnBase} ${item === currentPage ? pageBtnActive : ""}`}
|
|
54
|
-
onClick={() => onPageChange(item)}
|
|
55
|
-
aria-label={`Go to page ${item}`}
|
|
56
|
-
aria-current={item === currentPage ? "page" : undefined}
|
|
57
|
-
>
|
|
58
|
-
{item}
|
|
59
|
-
</Button>
|
|
60
|
-
)
|
|
61
|
-
)}
|
|
62
|
-
|
|
63
|
-
{/* Next */}
|
|
64
|
-
<Button
|
|
65
|
-
className={navBtnBase}
|
|
66
|
-
onClick={() => onPageChange(currentPage + 1)}
|
|
67
|
-
disabled={currentPage === totalPages}
|
|
68
|
-
aria-label="Go to next page"
|
|
69
|
-
>
|
|
70
|
-
<MaterialIcon name="chevron_right" size={32} aria-hidden="true" />
|
|
71
|
-
</Button>
|
|
72
|
-
</nav>
|
|
73
|
-
);
|
|
74
|
-
};
|
|
75
|
-
|
|
76
|
-
/** Returns an array of page numbers and ellipsis to render. */
|
|
77
|
-
function buildPageItems(current: number, total: number): (number | "...")[] {
|
|
78
|
-
if (total <= 7) {
|
|
79
|
-
return Array.from({ length: total }, (_, i) => i + 1);
|
|
80
|
-
}
|
|
81
|
-
const items: (number | "...")[] = [];
|
|
82
|
-
items.push(1);
|
|
83
|
-
|
|
84
|
-
if (current > 3) {
|
|
85
|
-
items.push("...");
|
|
86
|
-
}
|
|
87
|
-
const rangeStart = Math.max(2, current - 1);
|
|
88
|
-
const rangeEnd = Math.min(total - 1, current + 1);
|
|
89
|
-
|
|
90
|
-
for (let i = rangeStart; i <= rangeEnd; i++) {
|
|
91
|
-
items.push(i);
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
if (current < total - 2) {
|
|
95
|
-
items.push("...");
|
|
96
|
-
}
|
|
97
|
-
|
|
98
|
-
items.push(total);
|
|
99
|
-
return items;
|
|
100
|
-
}
|
|
1
|
+
"use client";
|
|
2
|
+
|
|
3
|
+
import React from "react";
|
|
4
|
+
import { PaginationProps } from "./types";
|
|
5
|
+
|
|
6
|
+
import { Button } from "@shared/components/button";
|
|
7
|
+
import { MaterialIcon } from "@shared/components/material-icon";
|
|
8
|
+
|
|
9
|
+
export const Pagination: React.FC<PaginationProps> = ({
|
|
10
|
+
currentPage,
|
|
11
|
+
totalPages,
|
|
12
|
+
onPageChange,
|
|
13
|
+
ariaLabel = "Pagination",
|
|
14
|
+
}: PaginationProps) => {
|
|
15
|
+
const pageItems = buildPageItems(currentPage, totalPages);
|
|
16
|
+
|
|
17
|
+
const navBtnBase =
|
|
18
|
+
"inline-flex items-center justify-center w-9 h-9 rounded-full bg-transparent cursor-pointer transition-colors duration-150 hover:border-text-brand hover:text-text-brand disabled:opacity-40 disabled:cursor-not-allowed";
|
|
19
|
+
|
|
20
|
+
const pageBtnBase =
|
|
21
|
+
"inline-flex items-center justify-center w-9 h-9 rounded-lg text-body2 text-gray-700 bg-transparent border-none cursor-pointer transition-colors duration-150 hover:bg-gray-100 hover:text-gray-900";
|
|
22
|
+
|
|
23
|
+
const pageBtnActive = "bg-bg-surface-active font-bold hover:bg-bg-gray-300";
|
|
24
|
+
|
|
25
|
+
return (
|
|
26
|
+
<nav
|
|
27
|
+
className="flex items-center justify-center gap-1 px-6 pb-12"
|
|
28
|
+
aria-label={ariaLabel}
|
|
29
|
+
>
|
|
30
|
+
{/* Previous */}
|
|
31
|
+
<Button
|
|
32
|
+
className={navBtnBase}
|
|
33
|
+
onClick={() => onPageChange(currentPage - 1)}
|
|
34
|
+
disabled={currentPage === 1}
|
|
35
|
+
aria-label="Go to previous page"
|
|
36
|
+
>
|
|
37
|
+
<MaterialIcon name="chevron_left" size={32} aria-hidden="true" />
|
|
38
|
+
</Button>
|
|
39
|
+
|
|
40
|
+
{/* Page numbers */}
|
|
41
|
+
{pageItems.map((item, idx) =>
|
|
42
|
+
item === "..." ? (
|
|
43
|
+
<span
|
|
44
|
+
key={`ellipsis-${idx}`}
|
|
45
|
+
className="inline-flex h-9 w-9 cursor-default items-center justify-center text-body2 text-gray-500"
|
|
46
|
+
aria-hidden="true"
|
|
47
|
+
>
|
|
48
|
+
…
|
|
49
|
+
</span>
|
|
50
|
+
) : (
|
|
51
|
+
<Button
|
|
52
|
+
key={item}
|
|
53
|
+
className={`${pageBtnBase} ${item === currentPage ? pageBtnActive : ""}`}
|
|
54
|
+
onClick={() => onPageChange(item)}
|
|
55
|
+
aria-label={`Go to page ${item}`}
|
|
56
|
+
aria-current={item === currentPage ? "page" : undefined}
|
|
57
|
+
>
|
|
58
|
+
{item}
|
|
59
|
+
</Button>
|
|
60
|
+
)
|
|
61
|
+
)}
|
|
62
|
+
|
|
63
|
+
{/* Next */}
|
|
64
|
+
<Button
|
|
65
|
+
className={navBtnBase}
|
|
66
|
+
onClick={() => onPageChange(currentPage + 1)}
|
|
67
|
+
disabled={currentPage === totalPages}
|
|
68
|
+
aria-label="Go to next page"
|
|
69
|
+
>
|
|
70
|
+
<MaterialIcon name="chevron_right" size={32} aria-hidden="true" />
|
|
71
|
+
</Button>
|
|
72
|
+
</nav>
|
|
73
|
+
);
|
|
74
|
+
};
|
|
75
|
+
|
|
76
|
+
/** Returns an array of page numbers and ellipsis to render. */
|
|
77
|
+
function buildPageItems(current: number, total: number): (number | "...")[] {
|
|
78
|
+
if (total <= 7) {
|
|
79
|
+
return Array.from({ length: total }, (_, i) => i + 1);
|
|
80
|
+
}
|
|
81
|
+
const items: (number | "...")[] = [];
|
|
82
|
+
items.push(1);
|
|
83
|
+
|
|
84
|
+
if (current > 3) {
|
|
85
|
+
items.push("...");
|
|
86
|
+
}
|
|
87
|
+
const rangeStart = Math.max(2, current - 1);
|
|
88
|
+
const rangeEnd = Math.min(total - 1, current + 1);
|
|
89
|
+
|
|
90
|
+
for (let i = rangeStart; i <= rangeEnd; i++) {
|
|
91
|
+
items.push(i);
|
|
92
|
+
}
|
|
93
|
+
|
|
94
|
+
if (current < total - 2) {
|
|
95
|
+
items.push("...");
|
|
96
|
+
}
|
|
97
|
+
|
|
98
|
+
items.push(total);
|
|
99
|
+
return items;
|
|
100
|
+
}
|
|
@@ -1,6 +1,6 @@
|
|
|
1
|
-
export interface PaginationProps {
|
|
2
|
-
currentPage: number;
|
|
3
|
-
totalPages: number;
|
|
4
|
-
onPageChange: (page: number) => void;
|
|
5
|
-
ariaLabel?: string;
|
|
6
|
-
}
|
|
1
|
+
export interface PaginationProps {
|
|
2
|
+
currentPage: number;
|
|
3
|
+
totalPages: number;
|
|
4
|
+
onPageChange: (page: number) => void;
|
|
5
|
+
ariaLabel?: string;
|
|
6
|
+
}
|