@internetarchive/bookreader 5.0.0-6-14 → 5.0.0-60

Sign up to get free protection for your applications and to get access to all the features.
Files changed (271) hide show
  1. package/.eslintrc.js +17 -15
  2. package/.github/workflows/node.js.yml +72 -10
  3. package/.github/workflows/npm-publish.yml +6 -20
  4. package/.testcaferc.js +10 -0
  5. package/BookReader/BookReader.css +241 -140
  6. package/BookReader/BookReader.js +1 -1
  7. package/BookReader/BookReader.js.LICENSE.txt +24 -20
  8. package/BookReader/BookReader.js.map +1 -1
  9. package/BookReader/ia-bookreader-bundle.js +1519 -0
  10. package/BookReader/ia-bookreader-bundle.js.LICENSE.txt +17 -0
  11. package/BookReader/ia-bookreader-bundle.js.map +1 -0
  12. package/BookReader/icons/close-circle-dark.svg +1 -0
  13. package/BookReader/icons/magnify-minus.svg +1 -1
  14. package/BookReader/icons/magnify-plus.svg +1 -1
  15. package/BookReader/icons/pause.svg +1 -1
  16. package/BookReader/icons/playback-speed.svg +1 -1
  17. package/BookReader/icons/read-aloud.svg +1 -1
  18. package/BookReader/icons/voice.svg +1 -0
  19. package/BookReader/images/BRicons.svg +2 -2
  20. package/BookReader/images/books_graphic.svg +1 -1
  21. package/BookReader/images/icon_book.svg +1 -1
  22. package/BookReader/images/icon_gear.svg +1 -1
  23. package/BookReader/images/icon_info.svg +1 -1
  24. package/BookReader/images/icon_playback-rate.svg +1 -1
  25. package/BookReader/images/icon_search_button.svg +1 -1
  26. package/BookReader/images/icon_share.svg +1 -1
  27. package/BookReader/images/icon_speaker.svg +1 -1
  28. package/BookReader/images/icon_speaker_open.svg +1 -1
  29. package/BookReader/images/marker_chap-off.svg +1 -1
  30. package/BookReader/images/marker_chap-on.svg +1 -1
  31. package/BookReader/images/marker_srch-on.svg +1 -1
  32. package/BookReader/jquery-3.js +2 -0
  33. package/BookReader/jquery-3.js.LICENSE.txt +24 -0
  34. package/BookReader/plugins/plugin.archive_analytics.js +1 -1
  35. package/BookReader/plugins/plugin.archive_analytics.js.map +1 -1
  36. package/BookReader/plugins/plugin.autoplay.js +1 -1
  37. package/BookReader/plugins/plugin.autoplay.js.map +1 -1
  38. package/BookReader/plugins/plugin.chapters.js +1 -1
  39. package/BookReader/plugins/plugin.chapters.js.map +1 -1
  40. package/BookReader/plugins/plugin.iframe.js +1 -1
  41. package/BookReader/plugins/plugin.iframe.js.map +1 -1
  42. package/BookReader/plugins/plugin.mobile_nav.js +1 -1
  43. package/BookReader/plugins/plugin.mobile_nav.js.map +1 -1
  44. package/BookReader/plugins/plugin.resume.js +1 -1
  45. package/BookReader/plugins/plugin.resume.js.map +1 -1
  46. package/BookReader/plugins/plugin.search.js +1 -1
  47. package/BookReader/plugins/plugin.search.js.map +1 -1
  48. package/BookReader/plugins/plugin.text_selection.js +1 -1
  49. package/BookReader/plugins/plugin.text_selection.js.map +1 -1
  50. package/BookReader/plugins/plugin.tts.js +1 -1
  51. package/BookReader/plugins/plugin.tts.js.map +1 -1
  52. package/BookReader/plugins/plugin.url.js +1 -1
  53. package/BookReader/plugins/plugin.url.js.map +1 -1
  54. package/BookReader/plugins/plugin.vendor-fullscreen.js +1 -1
  55. package/BookReader/plugins/plugin.vendor-fullscreen.js.map +1 -1
  56. package/BookReader/webcomponents-bundle.js +3 -0
  57. package/BookReader/webcomponents-bundle.js.LICENSE.txt +9 -0
  58. package/BookReader/webcomponents-bundle.js.map +1 -0
  59. package/BookReaderDemo/BookReaderDemo.css +14 -1
  60. package/BookReaderDemo/BookReaderJSAutoplay.js +4 -1
  61. package/BookReaderDemo/BookReaderJSSimple.js +1 -0
  62. package/BookReaderDemo/IADemoBr.js +147 -0
  63. package/BookReaderDemo/demo-advanced.html +2 -2
  64. package/BookReaderDemo/demo-autoplay.html +2 -1
  65. package/BookReaderDemo/demo-embed-iframe-src.html +2 -1
  66. package/BookReaderDemo/demo-fullscreen-mobile.html +2 -1
  67. package/BookReaderDemo/demo-fullscreen.html +2 -1
  68. package/BookReaderDemo/demo-iiif.html +2 -1
  69. package/BookReaderDemo/demo-internetarchive.html +84 -17
  70. package/BookReaderDemo/demo-multiple.html +2 -1
  71. package/BookReaderDemo/demo-preview-pages.html +2 -1
  72. package/BookReaderDemo/demo-simple.html +2 -1
  73. package/BookReaderDemo/demo-vendor-fullscreen.html +2 -1
  74. package/BookReaderDemo/ia-multiple-volumes-manifest.js +170 -0
  75. package/BookReaderDemo/immersion-1up.html +2 -1
  76. package/BookReaderDemo/immersion-mode.html +2 -1
  77. package/BookReaderDemo/toggle_controls.html +2 -1
  78. package/BookReaderDemo/view_mode.html +2 -1
  79. package/BookReaderDemo/viewmode-cycle.html +2 -3
  80. package/CHANGELOG.md +244 -0
  81. package/README.md +14 -1
  82. package/babel.config.js +19 -0
  83. package/codecov.yml +6 -0
  84. package/index.html +3 -0
  85. package/jsconfig.json +19 -0
  86. package/netlify.toml +5 -0
  87. package/package.json +70 -59
  88. package/renovate.json +52 -0
  89. package/scripts/preversion.js +4 -1
  90. package/src/BookNavigator/assets/bookmark-colors.js +1 -1
  91. package/src/BookNavigator/assets/button-base.js +9 -2
  92. package/src/BookNavigator/assets/ia-logo.js +17 -0
  93. package/src/BookNavigator/assets/icon_checkmark.js +1 -1
  94. package/src/BookNavigator/assets/icon_close.js +1 -1
  95. package/src/BookNavigator/assets/icon_sort_asc.js +5 -0
  96. package/src/BookNavigator/assets/icon_sort_desc.js +5 -0
  97. package/src/BookNavigator/assets/icon_sort_neutral.js +5 -0
  98. package/src/BookNavigator/assets/icon_volumes.js +11 -0
  99. package/src/BookNavigator/book-navigator.js +585 -0
  100. package/src/BookNavigator/bookmarks/bookmark-button.js +3 -2
  101. package/src/BookNavigator/bookmarks/bookmark-edit.js +3 -4
  102. package/src/BookNavigator/bookmarks/bookmarks-list.js +2 -3
  103. package/src/BookNavigator/bookmarks/bookmarks-loginCTA.js +3 -8
  104. package/src/BookNavigator/bookmarks/bookmarks-provider.js +27 -17
  105. package/src/BookNavigator/bookmarks/ia-bookmarks.js +116 -67
  106. package/src/BookNavigator/delete-modal-actions.js +1 -1
  107. package/src/BookNavigator/downloads/downloads-provider.js +36 -21
  108. package/src/BookNavigator/downloads/downloads.js +41 -25
  109. package/src/BookNavigator/search/search-provider.js +80 -28
  110. package/src/BookNavigator/search/search-results.js +34 -25
  111. package/src/BookNavigator/sharing.js +27 -0
  112. package/src/BookNavigator/visual-adjustments/visual-adjustments-provider.js +11 -10
  113. package/src/BookNavigator/visual-adjustments/visual-adjustments.js +3 -3
  114. package/src/BookNavigator/volumes/volumes-provider.js +111 -0
  115. package/src/BookNavigator/volumes/volumes.js +188 -0
  116. package/src/BookReader/BookModel.js +59 -30
  117. package/src/BookReader/DebugConsole.js +3 -3
  118. package/src/BookReader/DragScrollable.js +233 -0
  119. package/src/BookReader/Mode1Up.js +56 -351
  120. package/src/BookReader/Mode1UpLit.js +391 -0
  121. package/src/BookReader/Mode2Up.js +73 -1318
  122. package/src/BookReader/Mode2UpLit.js +781 -0
  123. package/src/BookReader/ModeCoordinateSpace.js +29 -0
  124. package/src/BookReader/ModeSmoothZoom.js +211 -0
  125. package/src/BookReader/ModeThumb.js +17 -11
  126. package/src/BookReader/Navbar/Navbar.js +10 -36
  127. package/src/BookReader/PageContainer.js +69 -6
  128. package/src/BookReader/ReduceSet.js +1 -1
  129. package/src/BookReader/Toolbar/Toolbar.js +10 -37
  130. package/src/BookReader/events.js +2 -0
  131. package/src/BookReader/options.js +24 -2
  132. package/src/BookReader/utils/HTMLDimensionsCacher.js +44 -0
  133. package/src/BookReader/utils/ScrollClassAdder.js +31 -0
  134. package/src/BookReader/utils.js +108 -13
  135. package/src/BookReader.js +481 -828
  136. package/src/assets/icons/close-circle-dark.svg +1 -0
  137. package/src/assets/icons/magnify-minus.svg +3 -7
  138. package/src/assets/icons/magnify-plus.svg +3 -7
  139. package/src/assets/icons/voice.svg +1 -0
  140. package/src/css/_BRBookmarks.scss +1 -1
  141. package/src/css/_BRComponent.scss +1 -1
  142. package/src/css/_BRmain.scss +33 -0
  143. package/src/css/_BRnav.scss +4 -26
  144. package/src/css/_BRpages.scss +147 -40
  145. package/src/css/_BRsearch.scss +25 -11
  146. package/src/css/_TextSelection.scss +16 -17
  147. package/src/css/_colorbox.scss +2 -2
  148. package/src/css/_controls.scss +17 -5
  149. package/src/css/_icons.scss +7 -1
  150. package/src/ia-bookreader/ia-bookreader.js +224 -0
  151. package/src/plugins/plugin.archive_analytics.js +3 -3
  152. package/src/plugins/plugin.autoplay.js +4 -9
  153. package/src/plugins/plugin.chapters.js +28 -35
  154. package/src/plugins/plugin.mobile_nav.js +11 -10
  155. package/src/plugins/plugin.resume.js +3 -3
  156. package/src/plugins/plugin.text_selection.js +32 -41
  157. package/src/plugins/plugin.vendor-fullscreen.js +4 -4
  158. package/src/plugins/search/plugin.search.js +179 -103
  159. package/src/plugins/search/view.js +59 -44
  160. package/src/plugins/tts/AbstractTTSEngine.js +46 -37
  161. package/src/plugins/tts/FestivalTTSEngine.js +13 -14
  162. package/src/plugins/tts/PageChunk.js +15 -21
  163. package/src/plugins/tts/PageChunkIterator.js +8 -12
  164. package/src/plugins/tts/WebTTSEngine.js +87 -71
  165. package/src/plugins/tts/plugin.tts.js +94 -125
  166. package/src/plugins/tts/utils.js +0 -25
  167. package/src/plugins/url/UrlPlugin.js +193 -0
  168. package/src/plugins/{plugin.url.js → url/plugin.url.js} +45 -16
  169. package/src/util/docCookies.js +21 -2
  170. package/tests/e2e/README.md +37 -0
  171. package/tests/e2e/autoplay.test.js +2 -2
  172. package/tests/e2e/base.test.js +7 -7
  173. package/tests/e2e/helpers/base.js +28 -23
  174. package/tests/e2e/helpers/debug.js +1 -1
  175. package/tests/e2e/helpers/desktopSearch.js +14 -13
  176. package/tests/e2e/helpers/mobileSearch.js +3 -3
  177. package/tests/e2e/helpers/params.js +17 -0
  178. package/tests/e2e/helpers/rightToLeft.js +4 -10
  179. package/tests/e2e/models/Navigation.js +13 -4
  180. package/tests/e2e/rightToLeft.test.js +4 -5
  181. package/tests/e2e/viewmode.test.js +40 -33
  182. package/tests/jest/BookNavigator/book-navigator.test.js +658 -0
  183. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  184. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  185. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  186. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  187. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  188. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  189. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  190. package/tests/{karma/BookNavigator → jest/BookNavigator/search}/search-results.test.js +104 -60
  191. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  192. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  193. package/tests/jest/BookNavigator/volumes/volumes-provider.test.js +184 -0
  194. package/tests/jest/BookNavigator/volumes/volumes.test.js +97 -0
  195. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +59 -14
  196. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +193 -0
  197. package/tests/{BookReader → jest/BookReader}/DebugConsole.test.js +1 -1
  198. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  199. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  200. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  201. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  202. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  203. package/tests/jest/BookReader/ModeSmoothZoom.test.js +175 -0
  204. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  205. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +10 -10
  206. package/tests/{BookReader → jest/BookReader}/PageContainer.test.js +88 -6
  207. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  208. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +2 -2
  209. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  210. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  211. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  212. package/tests/jest/BookReader/utils.test.js +217 -0
  213. package/tests/jest/BookReader.keyboard.test.js +190 -0
  214. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +9 -1
  215. package/tests/{BookReader.test.js → jest/BookReader.test.js} +26 -37
  216. package/tests/{plugins → jest/plugins}/plugin.archive_analytics.test.js +2 -2
  217. package/tests/{plugins → jest/plugins}/plugin.autoplay.test.js +4 -4
  218. package/tests/{plugins → jest/plugins}/plugin.chapters.test.js +10 -11
  219. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +2 -2
  220. package/tests/{plugins → jest/plugins}/plugin.mobile_nav.test.js +5 -5
  221. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +3 -3
  222. package/tests/{plugins → jest/plugins}/plugin.text_selection.test.js +39 -47
  223. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  224. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +57 -47
  225. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +35 -6
  226. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +9 -9
  227. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  228. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  229. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  230. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +47 -1
  231. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  232. package/tests/jest/plugins/url/UrlPlugin.test.js +190 -0
  233. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +53 -14
  234. package/tests/jest/setup.js +3 -0
  235. package/tests/{util → jest/util}/browserSniffing.test.js +1 -1
  236. package/tests/jest/util/docCookies.test.js +24 -0
  237. package/tests/{util → jest/util}/strings.test.js +1 -1
  238. package/tests/{utils.js → jest/utils.js} +38 -0
  239. package/webpack.config.js +11 -5
  240. package/.babelrc +0 -12
  241. package/.dependabot/config.yml +0 -6
  242. package/.testcaferc.json +0 -5
  243. package/BookReader/bookreader-component-bundle.js +0 -1450
  244. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  245. package/BookReader/bookreader-component-bundle.js.map +0 -1
  246. package/BookReader/jquery-1.10.1.js +0 -2
  247. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  248. package/BookReader/plugins/plugin.menu_toggle.js +0 -2
  249. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  250. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  251. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  252. package/karma.conf.js +0 -23
  253. package/src/BookNavigator/BookModel.js +0 -14
  254. package/src/BookNavigator/BookNavigator.js +0 -438
  255. package/src/BookNavigator/assets/book-loader.js +0 -27
  256. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  257. package/src/BookNavigator/search/a-search-result.js +0 -55
  258. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  259. package/src/ItemNavigator/ItemNavigator.js +0 -372
  260. package/src/ItemNavigator/providers/sharing.js +0 -29
  261. package/src/dragscrollable-br.js +0 -261
  262. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  263. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  264. package/tests/BookReader/Mode1Up.test.js +0 -164
  265. package/tests/BookReader/Mode2Up.test.js +0 -247
  266. package/tests/BookReader/utils.test.js +0 -109
  267. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  268. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  269. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  270. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  271. package/tests/util/docCookies.test.js +0 -15
@@ -0,0 +1,391 @@
1
+ // @ts-check
2
+ import { customElement, property, query } from 'lit/decorators.js';
3
+ import {LitElement, html} from 'lit';
4
+ import { styleMap } from 'lit/directives/style-map.js';
5
+ import { ModeSmoothZoom } from './ModeSmoothZoom';
6
+ import { arrChanged, genToArray, sum, throttle } from './utils';
7
+ import { HTMLDimensionsCacher } from "./utils/HTMLDimensionsCacher";
8
+ import { ScrollClassAdder } from './utils/ScrollClassAdder';
9
+ import { ModeCoordinateSpace } from './ModeCoordinateSpace';
10
+ /** @typedef {import('./BookModel').BookModel} BookModel */
11
+ /** @typedef {import('./BookModel').PageIndex} PageIndex */
12
+ /** @typedef {import('./BookModel').PageModel} PageModel */
13
+ /** @typedef {import('./ModeSmoothZoom').SmoothZoomable} SmoothZoomable */
14
+ /** @typedef {import('./PageContainer').PageContainer} PageContainer */
15
+ /** @typedef {import('../BookReader').default} BookReader */
16
+
17
+ // I _have_ to make this globally public, otherwise it won't let me call
18
+ // it's constructor :/
19
+ /** @implements {SmoothZoomable} */
20
+ @customElement('br-mode-1up')
21
+ export class Mode1UpLit extends LitElement {
22
+ /****************************************/
23
+ /************** PROPERTIES **************/
24
+ /****************************************/
25
+
26
+ /** @type {BookReader} */
27
+ br;
28
+
29
+ /************** BOOK-RELATED PROPERTIES **************/
30
+
31
+ /** @type {BookModel} */
32
+ @property({ type: Object })
33
+ book;
34
+
35
+ /** @type {PageModel[]} */
36
+ @property({ type: Array })
37
+ pages = [];
38
+
39
+ /** @type {Record<PageIndex, number>} in world coordinates (inches) */
40
+ @property({ type: Object })
41
+ pageTops = {};
42
+
43
+ /************** SCALE-RELATED PROPERTIES **************/
44
+
45
+ /** @type {ModeCoordinateSpace} Manage conversion between coordinates */
46
+ coordSpace = new ModeCoordinateSpace(this);
47
+
48
+ @property({ type: Number })
49
+ scale = 1;
50
+ /** Position (in unit-less, [0, 1] coordinates) in client to scale around */
51
+ @property({ type: Object })
52
+ scaleCenter = { x: 0.5, y: 0.5 };
53
+
54
+ /************** VIRTUAL-SCROLLING PROPERTIES **************/
55
+
56
+ /** in world coordinates (inches) */
57
+ @property({ type: Object })
58
+ visibleRegion = {
59
+ top: 0,
60
+ left: 0,
61
+ width: 100,
62
+ height: 100,
63
+ };
64
+
65
+ /** @type {PageModel[]} */
66
+ @property({ type: Array, hasChanged: arrChanged })
67
+ visiblePages = [];
68
+
69
+ /** @type {PageModel[]} */
70
+ @property({ type: Array })
71
+ renderedPages = [];
72
+
73
+ /** @type {Record<PageIndex, PageContainer>} position in inches */
74
+ pageContainerCache = {};
75
+
76
+ /************** WORLD-RELATED PROPERTIES **************/
77
+ /**
78
+ * The world is an imaginary giant document that contains all the pages.
79
+ * The "world"'s size is used to determine how long the scroll bar should
80
+ * be, for example.
81
+ */
82
+
83
+ /** @type {HTMLElement} */
84
+ @query('.br-mode-1up__world')
85
+ $world;
86
+
87
+ worldDimensions = { width: 100, height: 100 };
88
+
89
+ get worldStyle() {
90
+ const wToR = this.coordSpace.worldUnitsToRenderedPixels;
91
+ return {
92
+ width: wToR(this.worldDimensions.width) + "px",
93
+ height: wToR(this.worldDimensions.height) + "px",
94
+ };
95
+ }
96
+
97
+ /** @type {HTMLElement} */
98
+ get $container() { return this; }
99
+
100
+ /** @type {HTMLElement} */
101
+ @query('.br-mode-1up__visible-world')
102
+ $visibleWorld;
103
+
104
+ /************** DOM-RELATED PROPERTIES **************/
105
+
106
+ /** @type {HTMLDimensionsCacher} Cache things like clientWidth to reduce repaints */
107
+ htmlDimensionsCacher = new HTMLDimensionsCacher(this);
108
+
109
+ smoothZoomer = new ModeSmoothZoom(this);
110
+
111
+ scrollClassAdder = new ScrollClassAdder(this, 'BRscrolling-active');
112
+
113
+ /************** CONSTANT PROPERTIES **************/
114
+
115
+ /** Vertical space between/around the pages in inches */
116
+ SPACING_IN = 0.2;
117
+
118
+ /** How much to zoom when zoom button pressed */
119
+ ZOOM_FACTOR = 1.1;
120
+
121
+ /****************************************/
122
+ /************** PUBLIC API **************/
123
+ /****************************************/
124
+
125
+ /************** MAIN PUBLIC METHODS **************/
126
+
127
+ /**
128
+ * @param {PageIndex} index
129
+ */
130
+ jumpToIndex(index, { smooth = false } = {}) {
131
+ if (smooth) {
132
+ this.style.scrollBehavior = 'smooth';
133
+ }
134
+ this.scrollTop = this.coordSpace.worldUnitsToVisiblePixels(this.pageTops[index] - this.SPACING_IN / 2);
135
+ // TODO: Also h center?
136
+ if (smooth) {
137
+ setTimeout(() => this.style.scrollBehavior = '', 100);
138
+ }
139
+ }
140
+
141
+ zoomIn() {
142
+ this.scale *= this.ZOOM_FACTOR;
143
+ }
144
+
145
+ zoomOut() {
146
+ this.scale *= 1 / this.ZOOM_FACTOR;
147
+ }
148
+
149
+ /********************************************/
150
+ /************** INTERNAL STUFF **************/
151
+ /********************************************/
152
+
153
+ /************** LIFE CYCLE **************/
154
+
155
+ /**
156
+ * @param {BookModel} book
157
+ * @param {BookReader} br
158
+ */
159
+ constructor(book, br) {
160
+ super();
161
+ this.book = book;
162
+
163
+ /** @type {BookReader} */
164
+ this.br = br;
165
+ }
166
+
167
+ /** @override */
168
+ firstUpdated(changedProps) {
169
+ super.firstUpdated(changedProps);
170
+ this.htmlDimensionsCacher.updateClientSizes();
171
+ this.smoothZoomer.attach();
172
+ }
173
+
174
+ /**
175
+ * @param {PageIndex} startIndex
176
+ */
177
+ initFirstRender(startIndex) {
178
+ const page = this.book.getPage(startIndex);
179
+ this.scale = this.computeDefaultScale(page);
180
+ }
181
+
182
+ /** @override */
183
+ updated(changedProps) {
184
+ // this.X is the new value
185
+ // changedProps.get('X') is the old value
186
+ if (changedProps.has('book')) {
187
+ this.updatePages();
188
+ }
189
+ if (changedProps.has('pages')) {
190
+ this.worldDimensions = this.computeWorldDimensions();
191
+ this.pageTops = this.computePageTops(this.pages, this.SPACING_IN);
192
+ }
193
+ if (changedProps.has('visibleRegion')) {
194
+ this.visiblePages = this.computeVisiblePages();
195
+ }
196
+ if (changedProps.has('visiblePages')) {
197
+ this.throttledUpdateRenderedPages();
198
+ if (this.visiblePages.length) {
199
+ // unclear why this is ever really happening
200
+ this.br.displayedIndices = this.visiblePages.map(p => p.index);
201
+ this.br.updateFirstIndex(this.br.displayedIndices[0]);
202
+ this.br._components.navbar.updateNavIndexThrottled();
203
+ }
204
+ }
205
+ if (changedProps.has('scale')) {
206
+ const oldVal = changedProps.get('scale');
207
+ // Need to set this scale to actually scale the pages
208
+ this.$visibleWorld.style.transform = `scale(${this.scale})`;
209
+ this.smoothZoomer.updateViewportOnZoom(this.scale, oldVal);
210
+ this.updateVisibleRegion();
211
+ // Need to set this scale to update the world size, so the scrollbar gets the correct size
212
+ this.$world.style.transform = `scale(${this.scale})`;
213
+ }
214
+ }
215
+
216
+ updatePages() {
217
+ this.pages = genToArray(this.book.pagesIterator({ combineConsecutiveUnviewables: true }));
218
+ }
219
+
220
+ /** @override */
221
+ connectedCallback() {
222
+ super.connectedCallback();
223
+ this.htmlDimensionsCacher.attachResizeListener();
224
+ this.attachScrollListeners();
225
+ this.smoothZoomer.attach();
226
+ }
227
+
228
+ /** @override */
229
+ disconnectedCallback() {
230
+ this.htmlDimensionsCacher.detachResizeListener();
231
+ this.detachScrollListeners();
232
+ this.smoothZoomer.detach();
233
+ super.disconnectedCallback();
234
+ }
235
+
236
+ /************** LIT CONFIGS **************/
237
+
238
+ /** @override */
239
+ createRenderRoot() {
240
+ // Disable shadow DOM; that would require a huge rejiggering of CSS
241
+ return this;
242
+ }
243
+
244
+ /************** RENDERING **************/
245
+
246
+ /** @override */
247
+ render() {
248
+ return html`
249
+ <div class="br-mode-1up__world" style=${styleMap(this.worldStyle)}></div>
250
+ <div class="br-mode-1up__visible-world">
251
+ ${this.renderedPages.map(p => this.renderPage(p))}
252
+ </div>`;
253
+ }
254
+
255
+ /** @param {PageModel} page */
256
+ createPageContainer = (page) => {
257
+ return this.pageContainerCache[page.index] || (
258
+ this.pageContainerCache[page.index] = (
259
+ // @ts-ignore I know it's protected, TS! But Mode1Up and BookReader are friends.
260
+ this.br._createPageContainer(page.index)
261
+ )
262
+ );
263
+ }
264
+
265
+ /** @param {PageModel} page */
266
+ renderPage = (page) => {
267
+ const wToR = this.coordSpace.worldUnitsToRenderedPixels;
268
+ const wToV = this.coordSpace.worldUnitsToVisiblePixels;
269
+ const containerWidth = this.coordSpace.visiblePixelsToWorldUnits(this.htmlDimensionsCacher.clientWidth);
270
+
271
+ const width = wToR(page.widthInches);
272
+ const height = wToR(page.heightInches);
273
+ const left = Math.max(this.SPACING_IN, (containerWidth - page.widthInches) / 2);
274
+ const top = this.pageTops[page.index];
275
+
276
+ const transform = `translate(${wToR(left)}px, ${wToR(top)}px)`;
277
+ const pageContainerEl = this.createPageContainer(page)
278
+ .update({
279
+ dimensions: {
280
+ width,
281
+ height,
282
+ top: 0,
283
+ left: 0,
284
+ },
285
+ reduce: page.width / wToV(page.widthInches),
286
+ }).$container[0];
287
+
288
+ pageContainerEl.style.transform = transform;
289
+ pageContainerEl.classList.toggle('BRpage-visible', this.visiblePages.includes(page));
290
+ return pageContainerEl;
291
+ }
292
+
293
+ /************** VIRTUAL SCROLLING LOGIC **************/
294
+
295
+ updateVisibleRegion = () => {
296
+ const { scrollTop, scrollLeft } = this;
297
+ // clientHeight excludes scrollbars, which is good.
298
+ const clientWidth = this.htmlDimensionsCacher.clientWidth;
299
+ const clientHeight = this.htmlDimensionsCacher.clientHeight;
300
+
301
+ // Note: scrollTop, and clientWidth all are in visible space;
302
+ // i.e. they are affects by the CSS transforms.
303
+
304
+ const vToW = this.coordSpace.visiblePixelsToWorldUnits;
305
+ this.visibleRegion = {
306
+ top: vToW(scrollTop),
307
+ height: vToW(clientHeight),
308
+ // TODO: These are very likely wrong
309
+ left: vToW(scrollLeft),
310
+ width: vToW(clientWidth),
311
+ };
312
+ }
313
+
314
+ /**
315
+ * @returns {PageModel[]}
316
+ */
317
+ computeRenderedPages() {
318
+ // Also render 1 page before/after
319
+ // @ts-ignore TS doesn't understand the filtering out of null values
320
+ return [
321
+ this.visiblePages[0]?.prev,
322
+ ...this.visiblePages,
323
+ this.visiblePages[this.visiblePages.length - 1]?.next,
324
+ ]
325
+ .filter(p => p)
326
+ // Never render more than 10 pages! Usually means something is wrong
327
+ .slice(0, 10);
328
+ }
329
+
330
+ throttledUpdateRenderedPages = throttle(() => {
331
+ this.renderedPages = this.computeRenderedPages();
332
+ this.requestUpdate();
333
+ }, 100, null)
334
+
335
+ /**
336
+ * @param {PageModel[]} pages
337
+ * @param {number} spacing
338
+ */
339
+ computePageTops(pages, spacing) {
340
+ /** @type {{ [pageIndex: string]: number }} */
341
+ const result = {};
342
+ let top = spacing;
343
+ for (const page of pages) {
344
+ result[page.index] = top;
345
+ top += page.heightInches + spacing;
346
+ }
347
+ return result;
348
+ }
349
+
350
+ /**
351
+ * @param {PageModel} page
352
+ * @returns {number}
353
+ */
354
+ computeDefaultScale(page) {
355
+ // Default to real size if it fits, otherwise default to full width
356
+ const containerWidthIn = this.coordSpace.visiblePixelsToWorldUnits(this.htmlDimensionsCacher.clientWidth);
357
+ return Math.min(1, containerWidthIn / (page.widthInches + 2 * this.SPACING_IN)) || 1;
358
+ }
359
+
360
+ computeWorldDimensions() {
361
+ return {
362
+ width: Math.max(...this.pages.map(p => p.widthInches)) + 2 * this.SPACING_IN,
363
+ height:
364
+ sum(this.pages.map(p => p.heightInches)) +
365
+ (this.pages.length + 1) * this.SPACING_IN,
366
+ };
367
+ }
368
+
369
+ computeVisiblePages() {
370
+ return this.pages.filter(page => {
371
+ const PT = this.pageTops[page.index];
372
+ const PB = PT + page.heightInches;
373
+
374
+ const VT = this.visibleRegion.top;
375
+ const VB = VT + this.visibleRegion.height;
376
+ return PT <= VB && PB >= VT;
377
+ });
378
+ }
379
+
380
+ /************** INPUT HANDLERS **************/
381
+
382
+ attachScrollListeners = () => {
383
+ this.addEventListener("scroll", this.updateVisibleRegion);
384
+ this.scrollClassAdder.attach();
385
+ }
386
+
387
+ detachScrollListeners = () => {
388
+ this.removeEventListener("scroll", this.updateVisibleRegion);
389
+ this.scrollClassAdder.detach();
390
+ }
391
+ }