import {
ChevronLeftIcon,
ChevronRightIcon,
MoreHorizontalIcon,
} from "lucide-react";
import type { ComponentProps } from "react";
import { Button } from "@/components/ui/button";
import { cn } from "@/lib/utils";
function Pagination({ className, ...props }: ComponentProps<"nav">) {
return (
<nav
aria-label="pagination"
className={cn("mx-auto flex w-full justify-center", className)}
data-slot="pagination"
{...props}
/>
);
}
function PaginationContent({ className, ...props }: ComponentProps<"ul">) {
return (
<ul
className={cn("flex items-center gap-0.5", className)}
data-slot="pagination-content"
{...props}
/>
);
}
function PaginationItem({ ...props }: ComponentProps<"li">) {
return <li data-slot="pagination-item" {...props} />;
}
type PaginationLinkProps = {
isActive?: boolean;
} & Pick<ComponentProps<typeof Button>, "size"> &
ComponentProps<"a">;
function PaginationLink({
className,
isActive,
size = "icon",
...props
}: PaginationLinkProps) {
return (
<Button
className={cn(className)}
nativeButton={false}
render={
<a
aria-current={isActive ? "page" : undefined}
data-active={isActive}
data-slot="pagination-link"
{...props}
/>
}
size={size}
variant={isActive ? "outline" : "ghost"}
/>
);
}
function PaginationPrevious({
className,
text = "Previous",
...props
}: ComponentProps<typeof PaginationLink> & { text?: string }) {
return (
<PaginationLink
aria-label="Go to previous page"
className={cn("ps-2!", className)}
size="default"
{...props}
>
<ChevronLeftIcon className="rtl:rotate-180" data-icon="inline-start" />
<span className="hidden sm:block">{text}</span>
</PaginationLink>
);
}
function PaginationNext({
className,
text = "Next",
...props
}: ComponentProps<typeof PaginationLink> & { text?: string }) {
return (
<PaginationLink
aria-label="Go to next page"
className={cn("pe-2!", className)}
size="default"
{...props}
>
<span className="hidden sm:block">{text}</span>
<ChevronRightIcon className="rtl:rotate-180" data-icon="inline-end" />
</PaginationLink>
);
}
function PaginationEllipsis({ className, ...props }: ComponentProps<"span">) {
return (
<span
aria-hidden
className={cn(
"flex size-7 items-center justify-center [&_svg:not([class*='size-'])]:size-3.5",
className
)}
data-slot="pagination-ellipsis"
{...props}
>
<MoreHorizontalIcon />
<span className="sr-only">More pages</span>
</span>
);
}
export {
Pagination,
PaginationContent,
PaginationEllipsis,
PaginationItem,
PaginationLink,
PaginationNext,
PaginationPrevious,
};