@@ -5,125 +5,126 @@ export function buildPaginationModel(
5
5
marginPageCount : number ,
6
6
surroundingPageCount : number ,
7
7
) {
8
- const pages = [ ]
8
+ const prev : PageType = { type : 'PREV' , num : currentPage - 1 , disabled : currentPage === 1 }
9
+ const next : PageType = { type : 'NEXT' , num : currentPage + 1 , disabled : currentPage === pageCount }
10
+ if ( ! showPages ) {
11
+ return [ prev , next ]
12
+ }
9
13
10
- if ( showPages ) {
11
- const pageNums : Array < number > = [ ]
12
- const addPage = ( n : number ) => {
13
- if ( n >= 1 && n <= pageCount ) {
14
- pageNums . push ( n )
15
- }
16
- }
14
+ if ( pageCount <= 0 ) {
15
+ return [ prev , { ...next , disabled : true } ]
16
+ }
17
17
18
- // Start by defining the window of pages to show around the current page.
19
- // If the window goes off either edge, shift it until it fits.
20
- let extentLeft = currentPage - surroundingPageCount
21
- let extentRight = currentPage + surroundingPageCount
22
- if ( extentLeft < 1 && extentRight > pageCount ) {
23
- // Our window is larger than the entire range,
24
- // so simply display every page.
25
- extentLeft = 1
26
- extentRight = pageCount
27
- } else if ( extentLeft < 1 ) {
28
- while ( extentLeft < 1 ) {
29
- extentLeft ++
30
- extentRight ++
31
- }
32
- } else if ( extentRight > pageCount ) {
33
- while ( extentRight > pageCount ) {
34
- extentLeft --
35
- extentRight --
36
- }
37
- }
18
+ const pages : PageType [ ] = [ ]
38
19
39
- // Next, include the pages in the margins.
40
- // If a margin page is already covered in the window,
41
- // extend the window to the other direction.
42
- for ( let i = 1 ; i <= marginPageCount ; i ++ ) {
43
- const leftPage = i
44
- const rightPage = pageCount - ( i - 1 )
45
- if ( leftPage >= extentLeft ) {
46
- extentRight ++
47
- } else {
48
- addPage ( leftPage )
49
- }
50
- if ( rightPage <= extentRight ) {
51
- extentLeft --
52
- } else {
53
- addPage ( rightPage )
54
- }
55
- }
20
+ // number of pages shown on each side of the current page
21
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
22
+ // standardGap: 3
23
+ const standardGap = surroundingPageCount + marginPageCount
56
24
57
- for ( let i = extentLeft ; i <= extentRight ; i ++ ) {
58
- addPage ( i )
59
- }
25
+ // the maximum number of pages that can be shown at a given time
26
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
27
+ // maxVisiblePages: 7
28
+ const maxVisiblePages = standardGap + standardGap + 1
60
29
61
- const sorted = pageNums
62
- . slice ( )
63
- . sort ( ( a , b ) => a - b )
64
- . filter ( ( item , idx , ary ) => ! idx || item !== ary [ idx - 1 ] )
65
- for ( let idx = 0 ; idx < sorted . length ; idx ++ ) {
66
- const num = sorted [ idx ]
67
- const selected = num === currentPage
68
- const last = sorted [ idx - 1 ]
69
- const next = sorted [ idx + 1 ]
70
- const lastDelta = num - last
71
- const nextDelta = num - next
72
- const precedesBreak = nextDelta !== - 1
73
-
74
- if ( idx === 0 ) {
75
- if ( num !== 1 ) {
76
- // If the first page isn't page one,
77
- // we need to add a break
78
- pages . push ( {
79
- type : 'BREAK' ,
80
- num : 1 ,
81
- } )
82
- }
83
- pages . push ( {
84
- type : 'NUM' ,
85
- num,
86
- selected,
87
- precedesBreak,
88
- } )
89
- } else {
90
- if ( lastDelta === 1 ) {
91
- pages . push ( {
92
- type : 'NUM' ,
93
- num,
94
- selected,
95
- precedesBreak,
96
- } )
97
- } else {
98
- // We skipped some, so add a break
99
- pages . push ( {
100
- type : 'BREAK' ,
101
- num : num - 1 ,
102
- } )
103
- pages . push ( {
104
- type : 'NUM' ,
105
- num,
106
- selected,
107
- precedesBreak : false ,
108
- } )
109
- }
110
- }
111
- }
30
+ // if the number of pages is less than the maximum number of pages that can be shown just return all of them
31
+ if ( pageCount <= maxVisiblePages ) {
32
+ addPages ( 1 , pageCount , false )
33
+ return [ prev , ...pages , next ]
34
+ }
35
+
36
+ // startGap is the number of pages hidden by the start ellipsis
37
+ // startOffset is the number of pages to offset at the start to compensate
38
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
39
+ // startGap: 5
40
+ // startOffset: 0
41
+ // when the margin and the surrounding windows overlap.
42
+ // [1, _2_, 3, 4, 5, 6, ..., 15]
43
+ // startGap = 0
44
+ // startOffset: -3 <--
45
+ let startGap = 0
46
+ let startOffset = 0
112
47
113
- const lastPage = pages [ pages . length - 1 ]
114
- if ( lastPage . type === 'NUM' && lastPage . num !== pageCount ) {
115
- // The last page we rendered wasn't the actual last page,
116
- // so we need an additional break
48
+ // When there is overlap
49
+ if ( currentPage - standardGap - 1 <= 1 ) {
50
+ startOffset = currentPage - standardGap - 2
51
+ } else {
52
+ startGap = currentPage - standardGap - 1
53
+ }
54
+
55
+ // These are equivalent to startGap and startOffset but at the end of the list
56
+ let endGap = 0
57
+ let endOffset = 0
58
+
59
+ // When there is overlap
60
+ if ( pageCount - currentPage - standardGap <= 1 ) {
61
+ endOffset = pageCount - currentPage - standardGap - 1
62
+ } else {
63
+ endGap = pageCount - currentPage - standardGap
64
+ }
65
+
66
+ const hasStartEllipsis = startGap > 0
67
+ const hasEndEllipsis = endGap > 0
68
+
69
+ // add pages "before" the start ellipsis (if any)
70
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
71
+ // marginPageCount: 1
72
+ // addPages(1, 1, true)
73
+ addPages ( 1 , marginPageCount , hasStartEllipsis )
74
+
75
+ if ( hasStartEllipsis ) {
76
+ addEllipsis ( marginPageCount )
77
+ }
78
+
79
+ // add middle pages
80
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
81
+ // marginPageCount: 1
82
+ // surroundingPageCount: 2
83
+ // startGap: 5
84
+ // startOffset: 0
85
+ // endGap: 3
86
+ // endOffset: 0
87
+ // addPages(7, 11, true)
88
+ addPages (
89
+ marginPageCount + startGap + endOffset + 1 ,
90
+ pageCount - startOffset - endGap - marginPageCount ,
91
+ hasEndEllipsis ,
92
+ )
93
+
94
+ if ( hasEndEllipsis ) {
95
+ addEllipsis ( pageCount - startOffset - endGap - marginPageCount )
96
+ }
97
+
98
+ // add pages "after" the start ellipsis (if any)
99
+ // [1, ..., 7, 8, _9_, 10, 11, ..., 15]
100
+ // marginPageCount: 1
101
+ // surroundingPageCount: 2
102
+ // startGap: 5
103
+ // startOffset: 0
104
+ // endGap: 3
105
+ // endOffset: 0
106
+ // addPages(15, 15)
107
+ addPages ( pageCount - marginPageCount + 1 , pageCount )
108
+
109
+ return [ prev , ...pages , next ]
110
+
111
+ function addEllipsis ( previousPage : number ) : void {
112
+ pages . push ( {
113
+ type : 'BREAK' ,
114
+ num : previousPage + 1 ,
115
+ } )
116
+ }
117
+
118
+ function addPages ( start : number , end : number , precedesBreak : boolean = false ) : void {
119
+ for ( let i = start ; i <= end ; i ++ ) {
117
120
pages . push ( {
118
- type : 'BREAK' ,
119
- num : pageCount ,
121
+ type : 'NUM' ,
122
+ num : i ,
123
+ selected : i === currentPage ,
124
+ precedesBreak : i === end && precedesBreak ,
120
125
} )
121
126
}
122
127
}
123
-
124
- const prev = { type : 'PREV' , num : currentPage - 1 , disabled : currentPage === 1 }
125
- const next = { type : 'NEXT' , num : currentPage + 1 , disabled : currentPage === pageCount }
126
- return [ prev , ...pages , next ]
127
128
}
128
129
129
130
type PageType = {
@@ -148,7 +149,7 @@ export function buildComponentData(
148
149
key = 'page-prev'
149
150
content = 'Previous'
150
151
if ( page . disabled ) {
151
- Object . assign ( props , { 'aria-hidden' : 'true' } )
152
+ Object . assign ( props , { rel : 'prev' , 'aria-hidden' : 'true' } )
152
153
} else {
153
154
Object . assign ( props , {
154
155
rel : 'prev' ,
@@ -163,7 +164,7 @@ export function buildComponentData(
163
164
key = 'page-next'
164
165
content = 'Next'
165
166
if ( page . disabled ) {
166
- Object . assign ( props , { 'aria-hidden' : 'true' } )
167
+ Object . assign ( props , { rel : 'next' , 'aria-hidden' : 'true' } )
167
168
} else {
168
169
Object . assign ( props , {
169
170
rel : 'next' ,
0 commit comments