From 5b0d10ffba3d11290948a5d09cd9d057a3743414 Mon Sep 17 00:00:00 2001 From: Domizio Demichelis Date: Sun, 29 Dec 2024 20:54:41 +0700 Subject: [PATCH] Refactor Frontend: - Improve reliability of URLs and javascript placeholders - Remove the *_TOKEN constants - Reduce the size of 'data-pagy' and compiled javascript files --- docs/api/console.md | 2 +- docs/extras/metadata.md | 6 +- e2e/snapshots.js | 300 ++++++++++----------- gem/javascripts/pagy.js | 43 ++- gem/javascripts/pagy.js.map | 6 +- gem/javascripts/pagy.min.js | 2 +- gem/javascripts/pagy.mjs | 41 ++- gem/lib/pagy/extras/bootstrap.rb | 10 +- gem/lib/pagy/extras/bulma.rb | 12 +- gem/lib/pagy/extras/calendar.rb | 5 +- gem/lib/pagy/extras/headers.rb | 12 +- gem/lib/pagy/extras/js_tools.rb | 6 + gem/lib/pagy/extras/keyset_for_ui.rb | 2 +- gem/lib/pagy/extras/limit.rb | 8 +- gem/lib/pagy/extras/metadata.rb | 14 +- gem/lib/pagy/extras/pagy.rb | 9 +- gem/lib/pagy/frontend.rb | 11 +- src/pagy.ts | 112 +++++--- test/pagy/extras/bootstrap_test.rb.yaml | 135 +++++----- test/pagy/extras/bulma_test.rb.yaml | 155 ++++++----- test/pagy/extras/js_tools_calendar_test.rb | 6 - test/pagy/extras/limit_test.rb.yaml | 4 +- test/pagy/extras/metadata_test.rb.yaml | 62 ++--- test/pagy/extras/pagy_test.rb.yaml | 250 ++++++++--------- 24 files changed, 616 insertions(+), 597 deletions(-) diff --git a/docs/api/console.md b/docs/api/console.md index 9fc2a5fcc..1593faf96 100644 --- a/docs/api/console.md +++ b/docs/api/console.md @@ -33,7 +33,7 @@ pagy_nav(pagy) pagy_metadata(pagy) => -{ :scaffold_url => "http://www.example.com/subdir?page=__pagy_page__", +{ :scaffold_url => "http://www.example.com/subdir?page=P ", :first_url => "http://www.example.com/subdir?page=1", :prev_url => "http://www.example.com/subdir?page=2", :page_url => "http://www.example.com/subdir?page=3", diff --git a/docs/extras/metadata.md b/docs/extras/metadata.md index f75170d37..dfa97f65a 100644 --- a/docs/extras/metadata.md +++ b/docs/extras/metadata.md @@ -41,8 +41,8 @@ the default. This is a special url string that you can use as the scaffold to build real page urls in your frontend (instead of producing them on the backend). -It is a pagination url/path (complete with all the params) containing the `__pagy_page__` placeholder in place of the page -number (e.g. `'/foo?page=__pagy_page__&bar=baz'`) +It is a pagination url/path (complete with all the params) containing the `P ` placeholder in place of the page +number (e.g. `'/foo?page=P &bar=baz'`) You can generate all the actual links on the frontend by simply replacing the placeholder with the actual page number you want to link to. @@ -50,7 +50,7 @@ link to. In javascript you can do something like: ```js -page_url = scaffold_url.replace(/__pagy_page__/, page_number) +pageUrl = scaffoldUrl.replace("P ", pageNumber) ``` This is particularly useful when you want to build some dynamic pagination UI (e.g. similar to what the `pagy_*combo_js` diff --git a/e2e/snapshots.js b/e2e/snapshots.js index 2b26e5007..b653bda50 100644 --- a/e2e/snapshots.js +++ b/e2e/snapshots.js @@ -14,61 +14,61 @@ module.exports = { }, "[demo] Test #nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "" + "10": "" }, "[demo] Test #nav-js-responsive": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "12": "", + "12": "", "13": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "14": "", + "14": "", "15": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "20": "", + "20": "", "21": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "22": "", + "22": "", "23": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "24": "", + "24": "", "25": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "26": "", + "26": "", "27": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "28": "", + "28": "", "29": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "30": "" + "30": "" }, "[demo] Test #combo-nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records:\n 941,942,943,944,945,946,947,948,949,950,951,952,953,954,955,956,957,958,959,960\n

", - "12": "" + "12": "" }, "[demo] Test #pagy-info": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", @@ -80,29 +80,29 @@ module.exports = { }, "[demo] Test #limit-selector-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records: 1,2,3,4,5,6,7,8,9,10

", - "4": "", + "4": "", "5": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17

", - "6": "", + "6": "", "7": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18

", - "8": "", + "8": "", "9": "

@records:\n 701,702,703,704,705,706,707,708,709,710,711,712,713,714,715,716,717,718,719,720\n

", - "10": "", + "10": "", "11": "

@records: 701,702,703,704,705,706,707,708,709,710

", - "12": "", + "12": "", "13": "

@records:\n 698,699,700,701,702,703,704,705,706,707,708,709,710,711,712,713,714

", - "14": "", + "14": "", "15": "

@records:\n 685,686,687,688,689,690,691,692,693,694,695,696,697,698,699,700,701,702

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records: 981,982,983,984,985,986,987,988,989,990

", - "20": "", + "20": "", "21": "

@records:\n 970,971,972,973,974,975,976,977,978,979,980,981,982,983,984,985,986

", - "22": "", + "22": "", "23": "

@records:\n 955,956,957,958,959,960,961,962,963,964,965,966,967,968,969,970,971,972

", - "24": "" + "24": "" } }, "[demo] Test /bootstrap helpers": { @@ -120,61 +120,61 @@ module.exports = { }, "[demo] Test #nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "" + "10": "" }, "[demo] Test #nav-js-responsive": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "12": "", + "12": "", "13": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "14": "", + "14": "", "15": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "20": "", + "20": "", "21": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "22": "", + "22": "", "23": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "24": "", + "24": "", "25": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "26": "", + "26": "", "27": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "28": "", + "28": "", "29": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "30": "" + "30": "" }, "[demo] Test #combo-nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records:\n 941,942,943,944,945,946,947,948,949,950,951,952,953,954,955,956,957,958,959,960\n

", - "12": "" + "12": "" }, "[demo] Test #pagy-info": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", @@ -200,61 +200,61 @@ module.exports = { }, "[demo] Test #nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "" + "10": "" }, "[demo] Test #nav-js-responsive": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "12": "", + "12": "", "13": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "14": "", + "14": "", "15": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "20": "", + "20": "", "21": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "22": "", + "22": "", "23": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "24": "", + "24": "", "25": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "26": "", + "26": "", "27": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "28": "", + "28": "", "29": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "30": "" + "30": "" }, "[demo] Test #combo-nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records:\n 941,942,943,944,945,946,947,948,949,950,951,952,953,954,955,956,957,958,959,960\n

", - "12": "" + "12": "" }, "[demo] Test #pagy-info": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", @@ -281,61 +281,61 @@ module.exports = { }, "[repro] Test #nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "" + "10": "" }, "[repro] Test #nav-js-responsive": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "12": "", + "12": "", "13": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "14": "", + "14": "", "15": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "20": "", + "20": "", "21": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "22": "", + "22": "", "23": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "24": "", + "24": "", "25": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "26": "", + "26": "", "27": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "28": "", + "28": "", "29": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "30": "" + "30": "" }, "[repro] Test #combo-nav-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records:\n 21,22,23,24,25,26,27,28,29,30,31,32,33,34,35,36,37,38,39,40

", - "4": "", + "4": "", "5": "

@records:\n 41,42,43,44,45,46,47,48,49,50,51,52,53,54,55,56,57,58,59,60

", - "6": "", + "6": "", "7": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "8": "", + "8": "", "9": "

@records:\n 961,962,963,964,965,966,967,968,969,970,971,972,973,974,975,976,977,978,979,980\n

", - "10": "", + "10": "", "11": "

@records:\n 941,942,943,944,945,946,947,948,949,950,951,952,953,954,955,956,957,958,959,960\n

", - "12": "" + "12": "" }, "[repro] Test #pagy-info": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", @@ -347,29 +347,29 @@ module.exports = { }, "[repro] Test #limit-selector-js": { "1": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18,19,20

", - "2": "", + "2": "", "3": "

@records: 1,2,3,4,5,6,7,8,9,10

", - "4": "", + "4": "", "5": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17

", - "6": "", + "6": "", "7": "

@records: 1,2,3,4,5,6,7,8,9,10,11,12,13,14,15,16,17,18

", - "8": "", + "8": "", "9": "

@records:\n 701,702,703,704,705,706,707,708,709,710,711,712,713,714,715,716,717,718,719,720\n

", - "10": "", + "10": "", "11": "

@records: 701,702,703,704,705,706,707,708,709,710

", - "12": "", + "12": "", "13": "

@records:\n 698,699,700,701,702,703,704,705,706,707,708,709,710,711,712,713,714

", - "14": "", + "14": "", "15": "

@records:\n 685,686,687,688,689,690,691,692,693,694,695,696,697,698,699,700,701,702

", - "16": "", + "16": "", "17": "

@records:\n 981,982,983,984,985,986,987,988,989,990,991,992,993,994,995,996,997,998,999,1000\n

", - "18": "", + "18": "", "19": "

@records: 981,982,983,984,985,986,987,988,989,990

", - "20": "", + "20": "", "21": "

@records:\n 970,971,972,973,974,975,976,977,978,979,980,981,982,983,984,985,986

", - "22": "", + "22": "", "23": "

@records:\n 955,956,957,958,959,960,961,962,963,964,965,966,967,968,969,970,971,972

", - "24": "" + "24": "" } }, "[rails] Test helpers": { @@ -385,53 +385,53 @@ module.exports = { }, "[rails] Test #nav-js": { "1": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "2": "", + "2": "", "3": "
\n

Comment 1 to Post 6

\n

Comment 2 to Post 6

\n

Comment 1 to Post 7

\n

Comment 2 to Post 7

\n

Comment 1 to Post 8

\n

Comment 2 to Post 8

\n

Comment 1 to Post 9

\n

Comment 2 to Post 9

\n

Comment 1 to Post 10

\n

Comment 2 to Post 10

\n
", - "4": "", + "4": "", "5": "
\n

Comment 1 to Post 11

\n

Comment 2 to Post 11

\n
", - "6": "", + "6": "", "7": "
\n

Comment 1 to Post 6

\n

Comment 2 to Post 6

\n

Comment 1 to Post 7

\n

Comment 2 to Post 7

\n

Comment 1 to Post 8

\n

Comment 2 to Post 8

\n

Comment 1 to Post 9

\n

Comment 2 to Post 9

\n

Comment 1 to Post 10

\n

Comment 2 to Post 10

\n
", - "8": "" + "8": "" }, "[rails] Test #combo-nav-js": { "1": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "2": "", + "2": "", "3": "
\n

Comment 1 to Post 6

\n

Comment 2 to Post 6

\n

Comment 1 to Post 7

\n

Comment 2 to Post 7

\n

Comment 1 to Post 8

\n

Comment 2 to Post 8

\n

Comment 1 to Post 9

\n

Comment 2 to Post 9

\n

Comment 1 to Post 10

\n

Comment 2 to Post 10

\n
", - "4": "", + "4": "", "5": "
\n

Comment 1 to Post 11

\n

Comment 2 to Post 11

\n
", - "6": "", + "6": "", "7": "
\n

Comment 1 to Post 11

\n

Comment 2 to Post 11

\n
", - "8": "", + "8": "", "9": "
\n

Comment 1 to Post 6

\n

Comment 2 to Post 6

\n

Comment 1 to Post 7

\n

Comment 2 to Post 7

\n

Comment 1 to Post 8

\n

Comment 2 to Post 8

\n

Comment 1 to Post 9

\n

Comment 2 to Post 9

\n

Comment 1 to Post 10

\n

Comment 2 to Post 10

\n
", - "10": "", + "10": "", "11": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "12": "" + "12": "" }, "[rails] Test #limit-selector-js": { "1": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "2": "", + "2": "", "3": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "4": "", + "4": "", "5": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", - "6": "", + "6": "", "7": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n

Comment 1 to Post 6

\n
", - "8": "", + "8": "", "9": "
\n
", - "10": "", + "10": "", "11": "
\n
", - "12": "", + "12": "", "13": "
\n
", - "14": "", + "14": "", "15": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n

Comment 1 to Post 6

\n
", - "16": "", + "16": "", "17": "
\n
", - "18": "", + "18": "", "19": "
\n
", - "20": "", + "20": "", "21": "
\n
", - "22": "", + "22": "", "23": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n

Comment 1 to Post 6

\n
", - "24": "" + "24": "" }, "[rails] Test #pagy-info": { "1": "
\n

Comment 1 to Post 1

\n

Comment 2 to Post 1

\n

Comment 1 to Post 2

\n

Comment 2 to Post 2

\n

Comment 1 to Post 3

\n

Comment 2 to Post 3

\n

Comment 1 to Post 4

\n

Comment 2 to Post 4

\n

Comment 1 to Post 5

\n

Comment 2 to Post 5

\n
", diff --git a/gem/javascripts/pagy.js b/gem/javascripts/pagy.js index 1ca5715e2..5db985c69 100644 --- a/gem/javascripts/pagy.js +++ b/gem/javascripts/pagy.js @@ -1,6 +1,6 @@ // pagy.ts window.Pagy = (() => { - const storageSupport = "sessionStorage" in window && "BroadcastChannel" in window; + const storageSupport = "sessionStorage" in window && "BroadcastChannel" in window, pageRe = "P "; let storage, sync, tabId; if (storageSupport) { storage = sessionStorage; @@ -10,17 +10,17 @@ window.Pagy = (() => { if (e.data.from) { const cutoffs = storage.getItem(e.data.key); if (cutoffs) { - sync.postMessage({ to: e.data.from, key: e.data.key, cutoffs }); + sync.postMessage({ to: e.data.from, key: e.data.key, str: cutoffs }); } } else if (e.data.to) { if (e.data.to == tabId) { - storage.setItem(e.data.key, e.data.cutoffs); + storage.setItem(e.data.key, e.data.str); } } }); } const rjsObserver = new ResizeObserver((entries) => entries.forEach((e) => { - e.target.querySelectorAll(".pagy-rjs").forEach((el) => el.pagyRender()); + e.target.querySelectorAll(".pagy-rjs").forEach((el) => el.init()); })); const B64Encode = (unicode) => btoa(String.fromCharCode(...new TextEncoder().encode(unicode))), B64Safe = (unsafe) => unsafe.replace(/=/g, "").replace(/[+/]/g, (match) => match == "+" ? "-" : "_"), B64SafeEncode = (unicode) => B64Safe(B64Encode(unicode)), B64Decode = (base64) => new TextDecoder().decode(Uint8Array.from(atob(base64), (c) => c.charCodeAt(0))); const initCutoffs = async (el, [pageSym, [storageKey, spliceArgs]]) => { @@ -53,30 +53,29 @@ window.Pagy = (() => { a.href = url.replace(re, `${pageSym}=${value}`); } }; - const initNavJs = (el, [tokens, sequels, labelSequels, cutoffArgs]) => { - const container = el.parentElement ?? el, widths = Object.keys(sequels).map((w) => parseInt(w)).sort((a, b) => b - a); + const initNavJs = (el, [[before, a, current, gap, after], sequels, labelSequels, cutoffArgs]) => { + const container = el.parentElement ?? el, widths = Object.keys(sequels).map((w) => parseInt(w)).sort((a2, b) => b - a2), fillIn = (a2, page, label) => a2.replace(pageRe, page).replace("T<", label + "<"); let lastWidth = -1; - const fillIn = (a, page, label) => a.replace(/__pagy_page__/g, page).replace(/__pagy_label__/g, label); - (el.pagyRender = () => { + (el.init = () => { const width = widths.find((w) => w < container.clientWidth) || 0; if (width === lastWidth) { return; } - let html = tokens.before; + let html = before; const series = sequels[width.toString()], labels = labelSequels?.[width.toString()] ?? series.map((l) => l.toString()); series.forEach((item, i) => { const label = labels[i]; let filled; if (typeof item == "number") { - filled = fillIn(tokens.a, item.toString(), label); + filled = fillIn(a, item.toString(), label); } else if (item == "gap") { - filled = tokens.gap; + filled = gap; } else { - filled = fillIn(tokens.current, item, label); + filled = fillIn(current, item, label); } html += filled; }); - html += tokens.after; + html += after; el.innerHTML = ""; el.insertAdjacentHTML("afterbegin", html); lastWidth = width; @@ -88,11 +87,11 @@ window.Pagy = (() => { rjsObserver.observe(container); } }; - const initComboJs = (el, [url_token]) => initInput(el, (inputValue) => url_token.replace(/__pagy_page__/, inputValue)); + const initComboJs = (el, [url_token]) => initInput(el, (inputValue) => url_token.replace(pageRe, inputValue)); const initSelectorJs = (el, [from, url_token]) => { initInput(el, (inputValue) => { const page = Math.max(Math.ceil(from / parseInt(inputValue)), 1).toString(); - return url_token.replace(/__pagy_page__/, page).replace(/__pagy_limit__/, inputValue); + return url_token.replace(pageRe, page).replace("L ", inputValue); }); }; const initInput = (el, getUrl) => { @@ -123,23 +122,23 @@ window.Pagy = (() => { const target = arg instanceof Element ? arg : document, elements = target.querySelectorAll("[data-pagy]"); for (const el of elements) { try { - const [keyword, ...args] = JSON.parse(B64Decode(el.getAttribute("data-pagy"))); - if (keyword == "nav") { + const [key, ...args] = JSON.parse(B64Decode(el.getAttribute("data-pagy"))); + if (key == "n") { initCutoffs(el, args); - } else if (keyword == "nav_js") { + } else if (key == "nj") { initNavJs(el, args); - } else if (keyword == "combo_js") { + } else if (key == "cj") { initComboJs(el, args); - } else if (keyword == "selector_js") { + } else if (key == "sj") { initSelectorJs(el, args); } } catch (err) { - console.warn("Pagy.init() error: %o\n%s", el, err); + console.warn("Pagy.init: %o\n%s", el, err); } } } }; })(); -//# debugId=0974C14BAD5BAB2A64756E2164756E21 +//# debugId=2417F8E030EB240E64756E2164756E21 //# sourceMappingURL=pagy.js.map diff --git a/gem/javascripts/pagy.js.map b/gem/javascripts/pagy.js.map index 39936d011..9f7e89c36 100644 --- a/gem/javascripts/pagy.js.map +++ b/gem/javascripts/pagy.js.map @@ -2,9 +2,9 @@ "version": 3, "sources": ["../../src/pagy.ts"], "sourcesContent": [ - "type InitArgs = [\"nav\", CutoffArgs] | [\"nav_js\", NavJsArgs] | [\"combo_js\", ComboJsArgs] | [\"selector_js\", SelectorJsArgs]\ntype CutoffArgs = readonly [pageSym:string, update:[storageKey:string, splice?:SpliceArgs]]\ntype SpliceArgs = readonly [start:number, deleteCount:number, ...items:Cutoff[]]\ntype Cutoff = readonly (string | number | boolean)[]\ntype CutoffsParams = [pagyId:string, storageKey:string, pageNumber:number, pages:number, prevCutoff?:Cutoff, pageCutoff?:Cutoff]\ntype NavJsArgs = readonly [Tokens, Sequels, LabelSequels, CutoffArgs]\ntype ComboJsArgs = readonly [urlToken:string]\ntype SelectorJsArgs = readonly [from:number, urlToken:string]\ninterface SyncData {\n from?: number\n to?: number\n key: string\n cutoffs?: string\n}\ninterface Tokens {\n readonly before: string\n readonly a: string\n readonly current: string\n readonly gap: string\n readonly after: string\n}\ninterface Sequels {readonly [width:string]:(string | number)[]}\ninterface LabelSequels {readonly [width:string]:string[]}\ninterface NavJsElement extends Element {pagyRender():void}\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nconst Pagy = (() => {\n const storageSupport = 'sessionStorage' in window && 'BroadcastChannel' in window;\n let storage: Storage, sync: BroadcastChannel, tabId: number;\n if (storageSupport) {\n storage = sessionStorage; // shorten the .min.js\n sync = new BroadcastChannel(\"pagy\");\n tabId = Date.now();\n // Sync the sessionStorage keys for the cutoffs opened in a new tab/window\n sync.addEventListener(\"message\", (e:MessageEvent) => {\n if (e.data.from) { // request cutoffs\n const cutoffs = storage.getItem(e.data.key);\n if (cutoffs) {\n sync.postMessage({to: e.data.from, key: e.data.key, cutoffs: cutoffs});\n } // send response\n } else if (e.data.to) { // receive cutoffs\n if (e.data.to == tabId) {\n storage.setItem(e.data.key, e.data.cutoffs);\n }\n }\n });\n }\n // The observer instance for responsive navs\n const rjsObserver = new ResizeObserver(\n entries => entries.forEach(e => {\n e.target.querySelectorAll(\".pagy-rjs\").forEach(el => el.pagyRender());\n }));\n\n const B64Encode = (unicode:string) => btoa(String.fromCharCode(...(new TextEncoder).encode(unicode))),\n B64Safe = (unsafe:string) => unsafe.replace(/=/g, \"\").replace(/[+/]/g, (match) => match == \"+\" ? \"-\" : \"_\"),\n B64SafeEncode = (unicode:string) => B64Safe(B64Encode(unicode)),\n B64Decode = (base64:string) => (new TextDecoder()).decode(Uint8Array.from(atob(base64), c => c.charCodeAt(0)));\n // B64Unsafe = (safe:string) => safe.replace(/[-_]/g, (match) => match == \"-\" ? \"+\" : \"/\"),\n // B64SafeDecode = (base64:string) => B64Decode(B64Unsafe(base64))\n\n // Init the Cutoffs features\n const initCutoffs = async (el:Element, [pageSym, [storageKey, spliceArgs]]:CutoffArgs) => {\n if (!storageSupport) { return }\n\n let browserId = document.cookie.split(/;\\s+/) // it works even if malformed\n .find((row) => row.startsWith(\"pagy=\"))\n ?.split(\"=\")[1];\n if (!browserId) {\n document.cookie = \"pagy=\" + (browserId = Math.floor(Math.random() * 36 ** 3).toString(36));\n }\n if (storageKey && !(storageKey in storage)) {\n // Sync the sessiongStorage from other tabs/windows (e.g. open page in new tab/window)\n sync.postMessage({ from: tabId, key: storageKey });\n // Wait for the listener to copy the cutoffs in the current sessionStorage\n await new Promise((resolve) => setTimeout(() => resolve(\"\"), 100));\n }\n storageKey ||= \"pagy-\" + Date.now().toString(36);\n const data = storage.getItem(storageKey),\n cutoffs = (data ? JSON.parse(data) : [undefined]);\n if (spliceArgs) {\n cutoffs.splice(...spliceArgs);\n storage.setItem(storageKey, JSON.stringify(cutoffs));\n }\n // Augment the page param of each url\n for (const a of el.querySelectorAll('a[href]')) {\n const url = a.href,\n re = new RegExp(`(?<=\\\\?.*)\\\\b${pageSym}=([\\\\d]+)`), // find the numeric page\n pageNum = parseInt(url.match(re)?.[1]), // page=\\d+ is always in href\n value = B64SafeEncode(JSON.stringify([browserId,\n storageKey,\n pageNum,\n cutoffs.length, // pages\n cutoffs[pageNum - 1], // prevCutoff\n cutoffs[pageNum]])); // pageCutoff\n a.href = url.replace(re, `${pageSym}=${value}`);\n }\n };\n\n // Init the *_nav_js helpers\n const initNavJs = (el:NavJsElement, [tokens, sequels, labelSequels, cutoffArgs]:NavJsArgs) => {\n const container = el.parentElement ?? el,\n widths = Object.keys(sequels).map(w => parseInt(w)).sort((a, b) => b - a);\n let lastWidth = -1;\n const fillIn = (a:string, page:string, label:string) =>\n a.replace(/__pagy_page__/g, page).replace(/__pagy_label__/g, label);\n (el.pagyRender = () => {\n const width = widths.find(w => w < container.clientWidth) || 0;\n if (width === lastWidth) { return } // no change: abort\n\n let html = tokens.before;\n const series = sequels[width.toString()],\n labels = labelSequels?.[width.toString()] ?? series.map(l => l.toString());\n series.forEach((item, i) => {\n const label = labels[i];\n let filled;\n if (typeof item == \"number\") {\n filled = fillIn(tokens.a, item.toString(), label);\n } else if (item == \"gap\") {\n filled = tokens.gap;\n } else { // active page\n filled = fillIn(tokens.current, item, label);\n }\n html += filled;\n });\n html += tokens.after;\n\n el.innerHTML = \"\";\n el.insertAdjacentHTML(\"afterbegin\", html);\n lastWidth = width;\n if (cutoffArgs) { void initCutoffs(el, cutoffArgs) }\n })();\n if (el.classList.contains(\"pagy-rjs\")) { rjsObserver.observe(container) }\n };\n\n // Init the *_combo_nav_js helpers\n const initComboJs = (el:Element, [url_token]:ComboJsArgs) =>\n initInput(el, inputValue => url_token.replace(/__pagy_page__/, inputValue));\n\n // Init the limit_selector_js helper\n const initSelectorJs = (el:Element, [from, url_token]:SelectorJsArgs) => {\n initInput(el, inputValue => {\n const page = Math.max(Math.ceil(from / parseInt(inputValue)), 1).toString();\n return url_token.replace(/__pagy_page__/, page).replace(/__pagy_limit__/, inputValue);\n });\n };\n\n // Init the input element\n const initInput = (el:Element, getUrl:(v:string) => string) => {\n const input = el.querySelector(\"input\"),\n link = el.querySelector(\"a\"),\n initial = input.value,\n action = () => {\n if (input.value === initial) { return } // not changed\n const [min, val, max] = [input.min, input.value, input.max].map(n => parseInt(n) || 0);\n if (val < min || val > max) { // reset invalid/out-of-range\n input.value = initial;\n input.select();\n return;\n }\n link.href = getUrl(input.value);\n link.click();\n };\n [\"change\", \"focus\"].forEach(e => input.addEventListener(e, () => input.select())); // auto-select\n input.addEventListener(\"focusout\", action); // trigger action\n input.addEventListener(\"keypress\", e => { if (e.key == \"Enter\") { action() } }); // trigger action\n };\n\n // Public interface\n return {\n version: \"9.3.3\",\n\n // Scan for elements with a \"data-pagy\" attribute and call their init functions with the decoded args\n init(arg?:Element) {\n const target = arg instanceof Element ? arg : document,\n elements = target.querySelectorAll(\"[data-pagy]\");\n for (const el of elements) {\n try {\n const [keyword, ...args] = JSON.parse(B64Decode(el.getAttribute(\"data-pagy\")));\n if (keyword == \"nav\") {\n void initCutoffs(el, args);\n } else if (keyword == \"nav_js\") {\n initNavJs(el, args);\n } else if (keyword == \"combo_js\") {\n initComboJs(el, args);\n } else if (keyword == \"selector_js\") {\n initSelectorJs(el, args);\n }\n //else { console.warn(\"Pagy.init() error: %o\\nUnknown keyword '%s'\", el, keyword) }\n } catch (err) { console.warn(\"Pagy.init() error: %o\\n%s\", el, err) }\n }\n }\n };\n})();\n" + "interface SyncData {\n from?: number\n to?: number\n key: string\n str?: string\n}\ntype InitArgs = [\"n\", CutoffArgs] |\n [\"nj\", NavJsArgs] |\n [\"cj\", ComboJsArgs] |\n [\"sj\", SelectorJsArgs]\n\ntype CutoffArgs = readonly [pageSym: string,\n update: [storageKey: string, splice?: SpliceArgs]]\n\ntype SpliceArgs = readonly [start: number,\n deleteCount: number, // it would be optional, but ts complains\n ...items: Cutoff[]]\n\ntype Cutoff = readonly (string | number | boolean)[]\n\ntype CutoffsParams = [browserId: string,\n storageKey: string,\n pageNumber: number,\n pages: number,\n prevCutoff?: Cutoff,\n pageCutoff?: Cutoff]\n\ntype NavJsArgs = readonly [Tokens, Sequels, LabelSequels, CutoffArgs]\n\ntype ComboJsArgs = readonly [urlToken:string]\n\ntype SelectorJsArgs = readonly [from: number,\n urlToken: string]\n\ntype Tokens = readonly [before: string,\n a: string,\n current: string,\n gap: string,\n after: string]\ninterface Sequels {\n readonly [width:string]: (string | number)[]\n}\ninterface LabelSequels {\n readonly [width:string]: string[]\n}\ninterface NavJsElement extends Element {\n init(): void\n}\n\n// eslint-disable-next-line @typescript-eslint/no-unused-vars\nconst Pagy = (() => {\n const storageSupport = 'sessionStorage' in window && 'BroadcastChannel' in window,\n pageRe = \"P \"; // shorten the compiled size\n let storage: Storage, sync: BroadcastChannel, tabId: number;\n if (storageSupport) {\n storage = sessionStorage; // shorten the compiled size\n sync = new BroadcastChannel(\"pagy\");\n tabId = Date.now();\n // Sync the sessionStorage keys for the cutoffs opened in a new tab/window\n sync.addEventListener(\"message\", (e:MessageEvent) => {\n if (e.data.from) { // request cutoffs\n const cutoffs = storage.getItem(e.data.key);\n if (cutoffs) {\n sync.postMessage({to: e.data.from, key: e.data.key, str: cutoffs});\n } // send response\n } else if (e.data.to) { // receive cutoffs\n if (e.data.to == tabId) {\n storage.setItem(e.data.key, e.data.str);\n }\n }\n });\n }\n // The observer instance for responsive navs\n const rjsObserver = new ResizeObserver(\n entries => entries.forEach(e => {\n e.target.querySelectorAll(\".pagy-rjs\").forEach(el => el.init());\n }));\n\n const B64Encode = (unicode:string) => btoa(String.fromCharCode(...(new TextEncoder).encode(unicode))),\n B64Safe = (unsafe:string) => unsafe.replace(/=/g, \"\").replace(/[+/]/g, (match) => match == \"+\" ? \"-\" : \"_\"),\n B64SafeEncode = (unicode:string) => B64Safe(B64Encode(unicode)),\n B64Decode = (base64:string) => (new TextDecoder()).decode(Uint8Array.from(atob(base64), c => c.charCodeAt(0)));\n // B64Unsafe = (safe:string) => safe.replace(/[-_]/g, (match) => match == \"-\" ? \"+\" : \"/\"),\n // B64SafeDecode = (base64:string) => B64Decode(B64Unsafe(base64))\n\n // Init the Cutoffs features\n const initCutoffs = async (el:Element, [pageSym, [storageKey, spliceArgs]]:CutoffArgs) => {\n if (!storageSupport) { return }\n\n let browserId = document.cookie.split(/;\\s+/) // it works even if malformed\n .find((row) => row.startsWith(\"pagy=\"))\n ?.split(\"=\")[1];\n if (!browserId) {\n document.cookie = \"pagy=\" + (browserId = Math.floor(Math.random() * 36 ** 3).toString(36));\n }\n if (storageKey && !(storageKey in storage)) {\n // Sync the sessiongStorage from other tabs/windows (e.g. open page in new tab/window)\n sync.postMessage({ from: tabId, key: storageKey });\n // Wait for the listener to copy the cutoffs in the current sessionStorage\n await new Promise((resolve) => setTimeout(() => resolve(\"\"), 100));\n }\n storageKey ||= \"pagy-\" + Date.now().toString(36);\n const data = storage.getItem(storageKey),\n cutoffs = (data ? JSON.parse(data) : [undefined]);\n if (spliceArgs) {\n cutoffs.splice(...spliceArgs);\n storage.setItem(storageKey, JSON.stringify(cutoffs));\n }\n // Augment the page param of each url\n for (const a of el.querySelectorAll('a[href]')) {\n const url = a.href,\n re = new RegExp(`(?<=\\\\?.*)\\\\b${pageSym}=([\\\\d]+)`), // find the numeric page\n pageNum = parseInt(url.match(re)?.[1]), // page=\\d+ is always in href\n value = B64SafeEncode(JSON.stringify([browserId,\n storageKey,\n pageNum,\n cutoffs.length, // pages/last\n cutoffs[pageNum - 1], // prevCutoff\n cutoffs[pageNum]])); // pageCutoff\n a.href = url.replace(re, `${pageSym}=${value}`);\n }\n };\n\n // Init the *_nav_js helpers\n const initNavJs = (el:NavJsElement, [[before, a, current, gap, after], sequels, labelSequels, cutoffArgs]:NavJsArgs) => {\n const container = el.parentElement ?? el,\n widths = Object.keys(sequels).map(w => parseInt(w)).sort((a, b) => b - a),\n fillIn = (a:string, page:string, label:string) => a.replace(pageRe, page).replace(\"T<\", label + \"<\");\n let lastWidth = -1;\n (el.init = () => {\n const width = widths.find(w => w < container.clientWidth) || 0;\n if (width === lastWidth) { return } // no change: abort\n\n let html = before;\n const series = sequels[width.toString()],\n labels = labelSequels?.[width.toString()] ?? series.map(l => l.toString());\n series.forEach((item, i) => {\n const label = labels[i];\n let filled;\n if (typeof item == \"number\") {\n filled = fillIn(a, item.toString(), label);\n } else if (item == \"gap\") {\n filled = gap;\n } else { // active page\n filled = fillIn(current, item, label);\n }\n html += filled;\n });\n html += after;\n\n el.innerHTML = \"\";\n el.insertAdjacentHTML(\"afterbegin\", html);\n lastWidth = width;\n if (cutoffArgs) { void initCutoffs(el, cutoffArgs) }\n })();\n if (el.classList.contains(\"pagy-rjs\")) { rjsObserver.observe(container) }\n };\n\n // Init the *_combo_nav_js helpers\n const initComboJs = (el:Element, [url_token]:ComboJsArgs) =>\n initInput(el, inputValue => url_token.replace(pageRe, inputValue));\n\n // Init the limit_selector_js helper\n const initSelectorJs = (el:Element, [from, url_token]:SelectorJsArgs) => {\n initInput(el, inputValue => {\n const page = Math.max(Math.ceil(from / parseInt(inputValue)), 1).toString();\n return url_token.replace(pageRe, page).replace('L ', inputValue);\n });\n };\n\n // Init the input element\n const initInput = (el:Element, getUrl:(v:string) => string) => {\n const input = el.querySelector(\"input\"),\n link = el.querySelector(\"a\"),\n initial = input.value,\n action = () => {\n if (input.value === initial) { return } // not changed\n const [min, val, max] = [input.min, input.value, input.max].map(n => parseInt(n) || 0);\n if (val < min || val > max) { // reset invalid/out-of-range\n input.value = initial;\n input.select();\n return;\n }\n link.href = getUrl(input.value);\n link.click();\n };\n [\"change\", \"focus\"].forEach(e => input.addEventListener(e, () => input.select())); // auto-select\n input.addEventListener(\"focusout\", action); // trigger action\n input.addEventListener(\"keypress\", e => { if (e.key == \"Enter\") { action() } }); // trigger action\n };\n\n // Public interface\n return {\n version: \"9.3.3\",\n\n // Scan for elements with a \"data-pagy\" attribute and call their init functions with the decoded args\n init(arg?:Element) {\n const target = arg instanceof Element ? arg : document,\n elements = target.querySelectorAll(\"[data-pagy]\");\n for (const el of elements) {\n try {\n const [key, ...args] = JSON.parse(B64Decode(el.getAttribute(\"data-pagy\")));\n if (key == \"n\") {\n void initCutoffs(el, args);\n } else if (key == \"nj\") {\n initNavJs(el, args);\n } else if (key == \"cj\") {\n initComboJs(el, args);\n } else if (key == \"sj\") {\n initSelectorJs(el, args);\n }\n //else { console.warn(\"Pagy.init: %o\\nUnknown key '%s'\", el, key) }\n } catch (err) { console.warn(\"Pagy.init: %o\\n%s\", el, err) }\n }\n }\n };\n})();\n" ], - "mappings": "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", - "debugId": "0974C14BAD5BAB2A64756E2164756E21", + "mappings": "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", + "debugId": "2417F8E030EB240E64756E2164756E21", "names": [] } \ No newline at end of file diff --git a/gem/javascripts/pagy.min.js b/gem/javascripts/pagy.min.js index 61e892d35..c6ebcb6cd 100644 --- a/gem/javascripts/pagy.min.js +++ b/gem/javascripts/pagy.min.js @@ -1 +1 @@ -window.Pagy=(()=>{const P="sessionStorage"in window&&"BroadcastChannel"in window;let $,x,O;if(P)$=sessionStorage,x=new BroadcastChannel("pagy"),O=Date.now(),x.addEventListener("message",(z)=>{if(z.data.from){const G=$.getItem(z.data.key);if(G)x.postMessage({to:z.data.from,key:z.data.key,cutoffs:G})}else if(z.data.to){if(z.data.to==O)$.setItem(z.data.key,z.data.cutoffs)}});const J=new ResizeObserver((z)=>z.forEach((G)=>{G.target.querySelectorAll(".pagy-rjs").forEach((F)=>F.pagyRender())})),U=(z)=>btoa(String.fromCharCode(...new TextEncoder().encode(z))),B=(z)=>z.replace(/=/g,"").replace(/[+/]/g,(G)=>G=="+"?"-":"_"),N=(z)=>B(U(z)),_=(z)=>new TextDecoder().decode(Uint8Array.from(atob(z),(G)=>G.charCodeAt(0))),E=async(z,[G,[F,L]])=>{if(!P)return;let H=document.cookie.split(/;\s+/).find((T)=>T.startsWith("pagy="))?.split("=")[1];if(!H)document.cookie="pagy="+(H=Math.floor(Math.random()*46656).toString(36));if(F&&!(F in $))x.postMessage({from:O,key:F}),await new Promise((T)=>setTimeout(()=>T(""),100));F||="pagy-"+Date.now().toString(36);const Q=$.getItem(F),R=Q?JSON.parse(Q):[void 0];if(L)R.splice(...L),$.setItem(F,JSON.stringify(R));for(let T of z.querySelectorAll("a[href]")){const Z=T.href,M=new RegExp(`(?<=\\?.*)\\b${G}=([\\d]+)`),X=parseInt(Z.match(M)?.[1]),j=N(JSON.stringify([H,F,X,R.length,R[X-1],R[X]]));T.href=Z.replace(M,`${G}=${j}`)}},q=(z,[G,F,L,H])=>{const Q=z.parentElement??z,R=Object.keys(F).map((M)=>parseInt(M)).sort((M,X)=>X-M);let T=-1;const Z=(M,X,j)=>M.replace(/__pagy_page__/g,X).replace(/__pagy_label__/g,j);if((z.pagyRender=()=>{const M=R.find((Y)=>YY.toString());if(j.forEach((Y,I)=>{const C=A[I];let D;if(typeof Y=="number")D=Z(G.a,Y.toString(),C);else if(Y=="gap")D=G.gap;else D=Z(G.current,Y,C);X+=D}),X+=G.after,z.innerHTML="",z.insertAdjacentHTML("afterbegin",X),T=M,H)E(z,H)})(),z.classList.contains("pagy-rjs"))J.observe(Q)},V=(z,[G])=>W(z,(F)=>G.replace(/__pagy_page__/,F)),v=(z,[G,F])=>{W(z,(L)=>{const H=Math.max(Math.ceil(G/parseInt(L)),1).toString();return F.replace(/__pagy_page__/,H).replace(/__pagy_limit__/,L)})},W=(z,G)=>{const F=z.querySelector("input"),L=z.querySelector("a"),H=F.value,Q=()=>{if(F.value===H)return;const[R,T,Z]=[F.min,F.value,F.max].map((M)=>parseInt(M)||0);if(TZ){F.value=H,F.select();return}L.href=G(F.value),L.click()};["change","focus"].forEach((R)=>F.addEventListener(R,()=>F.select())),F.addEventListener("focusout",Q),F.addEventListener("keypress",(R)=>{if(R.key=="Enter")Q()})};return{version:"9.3.3",init(z){const G=z instanceof Element?z:document,F=G.querySelectorAll("[data-pagy]");for(let L of F)try{const[H,...Q]=JSON.parse(_(L.getAttribute("data-pagy")));if(H=="nav")E(L,Q);else if(H=="nav_js")q(L,Q);else if(H=="combo_js")V(L,Q);else if(H=="selector_js")v(L,Q)}catch(H){console.warn("Pagy.init() error: %o\n%s",L,H)}}}})(); +window.Pagy=(()=>{const C="sessionStorage"in window&&"BroadcastChannel"in window;let j,D,W;if(C)j=sessionStorage,D=new BroadcastChannel("pagy"),W=Date.now(),D.addEventListener("message",(z)=>{if(z.data.from){const G=j.getItem(z.data.key);if(G)D.postMessage({to:z.data.from,key:z.data.key,str:G})}else if(z.data.to){if(z.data.to==W)j.setItem(z.data.key,z.data.str)}});const _=new ResizeObserver((z)=>z.forEach((G)=>{G.target.querySelectorAll(".pagy-rjs").forEach((F)=>F.init())})),q=(z)=>btoa(String.fromCharCode(...new TextEncoder().encode(z))),V=(z)=>z.replace(/=/g,"").replace(/[+/]/g,(G)=>G=="+"?"-":"_"),v=(z)=>V(q(z)),A=(z)=>new TextDecoder().decode(Uint8Array.from(atob(z),(G)=>G.charCodeAt(0))),J=async(z,[G,[F,H]])=>{if(!C)return;let L=document.cookie.split(/;\s+/).find((R)=>R.startsWith("pagy="))?.split("=")[1];if(!L)document.cookie="pagy="+(L=Math.floor(Math.random()*46656).toString(36));if(F&&!(F in j))D.postMessage({from:W,key:F}),await new Promise((R)=>setTimeout(()=>R(""),100));F||="pagy-"+Date.now().toString(36);const Q=j.getItem(F),M=Q?JSON.parse(Q):[void 0];if(H)M.splice(...H),j.setItem(F,JSON.stringify(M));for(let R of z.querySelectorAll("a[href]")){const Y=R.href,Z=new RegExp(`(?<=\\?.*)\\b${G}=([\\d]+)`),x=parseInt(Y.match(Z)?.[1]),O=v(JSON.stringify([L,F,x,M.length,M[x-1],M[x]]));R.href=Y.replace(Z,`${G}=${O}`)}},I=(z,[[G,F,H,L,Q],M,R,Y])=>{const Z=z.parentElement??z,x=Object.keys(M).map((T)=>parseInt(T)).sort((T,$)=>$-T),O=(T,$,P)=>T.replace("P ",$).replace("T<",P+"<");let B=-1;if((z.init=()=>{const T=x.find((X)=>XX.toString());if(P.forEach((X,K)=>{const N=k[K];let E;if(typeof X=="number")E=O(F,X.toString(),N);else if(X=="gap")E=L;else E=O(H,X,N);$+=E}),$+=Q,z.innerHTML="",z.insertAdjacentHTML("afterbegin",$),B=T,Y)J(z,Y)})(),z.classList.contains("pagy-rjs"))_.observe(Z)},S=(z,[G])=>U(z,(F)=>G.replace("P ",F)),h=(z,[G,F])=>{U(z,(H)=>{const L=Math.max(Math.ceil(G/parseInt(H)),1).toString();return F.replace("P ",L).replace("L ",H)})},U=(z,G)=>{const F=z.querySelector("input"),H=z.querySelector("a"),L=F.value,Q=()=>{if(F.value===L)return;const[M,R,Y]=[F.min,F.value,F.max].map((Z)=>parseInt(Z)||0);if(RY){F.value=L,F.select();return}H.href=G(F.value),H.click()};["change","focus"].forEach((M)=>F.addEventListener(M,()=>F.select())),F.addEventListener("focusout",Q),F.addEventListener("keypress",(M)=>{if(M.key=="Enter")Q()})};return{version:"9.3.3",init(z){const G=z instanceof Element?z:document,F=G.querySelectorAll("[data-pagy]");for(let H of F)try{const[L,...Q]=JSON.parse(A(H.getAttribute("data-pagy")));if(L=="n")J(H,Q);else if(L=="nj")I(H,Q);else if(L=="cj")S(H,Q);else if(L=="sj")h(H,Q)}catch(L){console.warn("Pagy.init: %o\n%s",H,L)}}}})(); diff --git a/gem/javascripts/pagy.mjs b/gem/javascripts/pagy.mjs index 3b8924d98..de775a29a 100644 --- a/gem/javascripts/pagy.mjs +++ b/gem/javascripts/pagy.mjs @@ -1,5 +1,5 @@ const Pagy = (() => { - const storageSupport = "sessionStorage" in window && "BroadcastChannel" in window; + const storageSupport = "sessionStorage" in window && "BroadcastChannel" in window, pageRe = "P "; let storage, sync, tabId; if (storageSupport) { storage = sessionStorage; @@ -9,17 +9,17 @@ const Pagy = (() => { if (e.data.from) { const cutoffs = storage.getItem(e.data.key); if (cutoffs) { - sync.postMessage({ to: e.data.from, key: e.data.key, cutoffs }); + sync.postMessage({ to: e.data.from, key: e.data.key, str: cutoffs }); } } else if (e.data.to) { if (e.data.to == tabId) { - storage.setItem(e.data.key, e.data.cutoffs); + storage.setItem(e.data.key, e.data.str); } } }); } const rjsObserver = new ResizeObserver((entries) => entries.forEach((e) => { - e.target.querySelectorAll(".pagy-rjs").forEach((el) => el.pagyRender()); + e.target.querySelectorAll(".pagy-rjs").forEach((el) => el.init()); })); const B64Encode = (unicode) => btoa(String.fromCharCode(...new TextEncoder().encode(unicode))), B64Safe = (unsafe) => unsafe.replace(/=/g, "").replace(/[+/]/g, (match) => match == "+" ? "-" : "_"), B64SafeEncode = (unicode) => B64Safe(B64Encode(unicode)), B64Decode = (base64) => new TextDecoder().decode(Uint8Array.from(atob(base64), (c) => c.charCodeAt(0))); const initCutoffs = async (el, [pageSym, [storageKey, spliceArgs]]) => { @@ -52,30 +52,29 @@ const Pagy = (() => { a.href = url.replace(re, `${pageSym}=${value}`); } }; - const initNavJs = (el, [tokens, sequels, labelSequels, cutoffArgs]) => { - const container = el.parentElement ?? el, widths = Object.keys(sequels).map((w) => parseInt(w)).sort((a, b) => b - a); + const initNavJs = (el, [[before, a, current, gap, after], sequels, labelSequels, cutoffArgs]) => { + const container = el.parentElement ?? el, widths = Object.keys(sequels).map((w) => parseInt(w)).sort((a, b) => b - a), fillIn = (a, page, label) => a.replace(pageRe, page).replace("T<", label + "<"); let lastWidth = -1; - const fillIn = (a, page, label) => a.replace(/__pagy_page__/g, page).replace(/__pagy_label__/g, label); - (el.pagyRender = () => { + (el.init = () => { const width = widths.find((w) => w < container.clientWidth) || 0; if (width === lastWidth) { return; } - let html = tokens.before; + let html = before; const series = sequels[width.toString()], labels = labelSequels?.[width.toString()] ?? series.map((l) => l.toString()); series.forEach((item, i) => { const label = labels[i]; let filled; if (typeof item == "number") { - filled = fillIn(tokens.a, item.toString(), label); + filled = fillIn(a, item.toString(), label); } else if (item == "gap") { - filled = tokens.gap; + filled = gap; } else { - filled = fillIn(tokens.current, item, label); + filled = fillIn(current, item, label); } html += filled; }); - html += tokens.after; + html += after; el.innerHTML = ""; el.insertAdjacentHTML("afterbegin", html); lastWidth = width; @@ -87,11 +86,11 @@ const Pagy = (() => { rjsObserver.observe(container); } }; - const initComboJs = (el, [url_token]) => initInput(el, (inputValue) => url_token.replace(/__pagy_page__/, inputValue)); + const initComboJs = (el, [url_token]) => initInput(el, (inputValue) => url_token.replace(pageRe, inputValue)); const initSelectorJs = (el, [from, url_token]) => { initInput(el, (inputValue) => { const page = Math.max(Math.ceil(from / parseInt(inputValue)), 1).toString(); - return url_token.replace(/__pagy_page__/, page).replace(/__pagy_limit__/, inputValue); + return url_token.replace(pageRe, page).replace("L ", inputValue); }); }; const initInput = (el, getUrl) => { @@ -122,18 +121,18 @@ const Pagy = (() => { const target = arg instanceof Element ? arg : document, elements = target.querySelectorAll("[data-pagy]"); for (const el of elements) { try { - const [keyword, ...args] = JSON.parse(B64Decode(el.getAttribute("data-pagy"))); - if (keyword == "nav") { + const [key, ...args] = JSON.parse(B64Decode(el.getAttribute("data-pagy"))); + if (key == "n") { initCutoffs(el, args); - } else if (keyword == "nav_js") { + } else if (key == "nj") { initNavJs(el, args); - } else if (keyword == "combo_js") { + } else if (key == "cj") { initComboJs(el, args); - } else if (keyword == "selector_js") { + } else if (key == "sj") { initSelectorJs(el, args); } } catch (err) { - console.warn("Pagy.init() error: %o\n%s", el, err); + console.warn("Pagy.init: %o\n%s", el, err); } } } diff --git a/gem/lib/pagy/extras/bootstrap.rb b/gem/lib/pagy/extras/bootstrap.rb index b23bd0af5..cac56fe63 100644 --- a/gem/lib/pagy/extras/bootstrap.rb +++ b/gem/lib/pagy/extras/bootstrap.rb @@ -11,7 +11,7 @@ module BootstrapExtra def pagy_bootstrap_nav(pagy, id: nil, classes: 'pagination', aria_label: nil, **vars) id = %( id="#{id}") if id a = pagy_anchor(pagy, **vars) - data = %( #{pagy_data(pagy, :nav)}) if defined?(::Pagy::KeysetForUI) && pagy.is_a?(KeysetForUI) + data = %( #{pagy_data(pagy, :n)}) if defined?(::Pagy::KeysetForUI) && pagy.is_a?(KeysetForUI) html = %(
    #{bootstrap_prev_html(pagy, a)}) @@ -37,16 +37,16 @@ def pagy_bootstrap_nav_js(pagy, id: nil, classes: 'pagination', aria_label: nil, id = %( id="#{id}") if id a = pagy_anchor(pagy, **vars) tokens = { before: %(
      #{bootstrap_prev_html(pagy, a)}), - a: %(
    • #{a.(PAGE_TOKEN, LABEL_TOKEN, classes: 'page-link')}
    • ), + a: %(
    • #{JST.token(a.('P', 'T', classes: 'page-link'), pagy.vars[:page_sym])}
    • ), current: %(
    • #{LABEL_TOKEN}
    • ), + %(aria-current="page" aria-disabled="true">T), gap: %(
    • #{ pagy_t('pagy.gap')}
    • ), after: %(#{bootstrap_next_html pagy, a}
    ) } %() end @@ -62,7 +62,7 @@ def pagy_bootstrap_combo_nav_js(pagy, id: nil, classes: 'pagination', aria_label %(
      #{ bootstrap_prev_html(pagy, a) }