-
Notifications
You must be signed in to change notification settings - Fork 0
/
pagination-controls.client.tsx
69 lines (66 loc) · 1.84 KB
/
pagination-controls.client.tsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
'use client'
import {
Pagination,
PaginationButton,
PaginationContent,
PaginationItem,
PaginationNext,
PaginationPrevious
} from '@/src/components/ui/pagination'
import { cn } from '@/src/lib/utils'
import { useQueryState } from 'nuqs'
import React from 'react'
import { searchParams } from './searchParams'
type PaginationControlsProps = {
numPages: number
}
// Use client-side hooks to update the page number
// and observe the loading state
export function ClientPaginationControls({
numPages
}: PaginationControlsProps) {
const [isLoading, startTransition] = React.useTransition()
const [page, setPage] = useQueryState(
'page',
searchParams.page.withOptions({
startTransition,
shallow: false // Send updates to the server
})
)
return (
<Pagination className="not-prose items-center gap-2">
<PaginationContent>
<PaginationItem>
<PaginationPrevious
disabled={page === 1}
onClick={() => setPage(p => Math.max(1, p - 1))}
/>
</PaginationItem>
{Array.from({ length: numPages }, (_, i) => (
<PaginationItem key={i}>
<PaginationButton
isActive={page === i + 1}
onClick={() => setPage(i + 1)}
>
{i + 1}
</PaginationButton>
</PaginationItem>
))}
<PaginationItem>
<PaginationNext
disabled={page === numPages}
onClick={() => setPage(p => Math.min(numPages, p + 1))}
/>
</PaginationItem>
</PaginationContent>
<div
aria-label={isLoading ? 'Loading' : 'Idle'}
aria-live={isLoading ? 'polite' : undefined}
className={cn(
'h-2 w-2 rounded-full bg-green-500',
isLoading && 'animate-pulse bg-amber-500'
)}
/>
</Pagination>
)
}