@internetarchive/bookreader 5.0.0-6-multiple-files → 5.0.0-60

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (303) hide show
  1. package/.eslintrc.js +17 -15
  2. package/.github/workflows/node.js.yml +77 -6
  3. package/.github/workflows/npm-publish.yml +6 -20
  4. package/.testcaferc.js +10 -0
  5. package/BookReader/BookReader.css +241 -377
  6. package/BookReader/BookReader.js +2 -21564
  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/1up.svg +1 -12
  13. package/BookReader/icons/2up.svg +1 -15
  14. package/BookReader/icons/advance.svg +3 -26
  15. package/BookReader/icons/chevron-right.svg +1 -1
  16. package/BookReader/icons/close-circle-dark.svg +1 -0
  17. package/BookReader/icons/close-circle.svg +1 -1
  18. package/BookReader/icons/fullscreen.svg +1 -17
  19. package/BookReader/icons/fullscreen_exit.svg +1 -17
  20. package/BookReader/icons/hamburger.svg +1 -15
  21. package/BookReader/icons/left-arrow.svg +1 -12
  22. package/BookReader/icons/magnify-minus.svg +1 -16
  23. package/BookReader/icons/magnify-plus.svg +1 -17
  24. package/BookReader/icons/magnify.svg +1 -15
  25. package/BookReader/icons/pause.svg +1 -23
  26. package/BookReader/icons/play.svg +1 -22
  27. package/BookReader/icons/playback-speed.svg +1 -34
  28. package/BookReader/icons/read-aloud.svg +1 -22
  29. package/BookReader/icons/review.svg +3 -22
  30. package/BookReader/icons/thumbnails.svg +1 -17
  31. package/BookReader/icons/voice.svg +1 -0
  32. package/BookReader/icons/volume-full.svg +1 -22
  33. package/BookReader/images/BRicons.svg +5 -94
  34. package/BookReader/images/books_graphic.svg +1 -177
  35. package/BookReader/images/icon_book.svg +1 -12
  36. package/BookReader/images/icon_bookmark.svg +1 -12
  37. package/BookReader/images/icon_gear.svg +1 -14
  38. package/BookReader/images/icon_hamburger.svg +1 -20
  39. package/BookReader/images/icon_home.svg +1 -21
  40. package/BookReader/images/icon_info.svg +1 -11
  41. package/BookReader/images/icon_one_page.svg +1 -8
  42. package/BookReader/images/icon_pause.svg +1 -1
  43. package/BookReader/images/icon_play.svg +1 -1
  44. package/BookReader/images/icon_playback-rate.svg +1 -15
  45. package/BookReader/images/icon_search_button.svg +1 -8
  46. package/BookReader/images/icon_share.svg +1 -9
  47. package/BookReader/images/icon_skip-ahead.svg +1 -6
  48. package/BookReader/images/icon_skip-back.svg +2 -13
  49. package/BookReader/images/icon_speaker.svg +1 -18
  50. package/BookReader/images/icon_speaker_open.svg +1 -10
  51. package/BookReader/images/icon_thumbnails.svg +1 -12
  52. package/BookReader/images/icon_toc.svg +1 -5
  53. package/BookReader/images/icon_two_pages.svg +1 -9
  54. package/BookReader/images/marker_chap-off.svg +1 -11
  55. package/BookReader/images/marker_chap-on.svg +1 -11
  56. package/BookReader/images/marker_srch-on.svg +1 -11
  57. package/BookReader/jquery-3.js +2 -0
  58. package/BookReader/jquery-3.js.LICENSE.txt +24 -0
  59. package/BookReader/plugins/plugin.archive_analytics.js +1 -172
  60. package/BookReader/plugins/plugin.archive_analytics.js.map +1 -1
  61. package/BookReader/plugins/plugin.autoplay.js +1 -165
  62. package/BookReader/plugins/plugin.autoplay.js.map +1 -1
  63. package/BookReader/plugins/plugin.chapters.js +1 -304
  64. package/BookReader/plugins/plugin.chapters.js.map +1 -1
  65. package/BookReader/plugins/plugin.iframe.js +1 -74
  66. package/BookReader/plugins/plugin.iframe.js.map +1 -1
  67. package/BookReader/plugins/plugin.mobile_nav.js +1 -334
  68. package/BookReader/plugins/plugin.mobile_nav.js.map +1 -1
  69. package/BookReader/plugins/plugin.resume.js +1 -368
  70. package/BookReader/plugins/plugin.resume.js.map +1 -1
  71. package/BookReader/plugins/plugin.search.js +1 -1420
  72. package/BookReader/plugins/plugin.search.js.map +1 -1
  73. package/BookReader/plugins/plugin.text_selection.js +1 -1080
  74. package/BookReader/plugins/plugin.text_selection.js.map +1 -1
  75. package/BookReader/plugins/plugin.tts.js +2 -9193
  76. package/BookReader/plugins/plugin.tts.js.map +1 -1
  77. package/BookReader/plugins/plugin.url.js +1 -269
  78. package/BookReader/plugins/plugin.url.js.map +1 -1
  79. package/BookReader/plugins/plugin.vendor-fullscreen.js +1 -379
  80. package/BookReader/plugins/plugin.vendor-fullscreen.js.map +1 -1
  81. package/BookReader/webcomponents-bundle.js +3 -0
  82. package/BookReader/webcomponents-bundle.js.LICENSE.txt +9 -0
  83. package/BookReader/webcomponents-bundle.js.map +1 -0
  84. package/BookReaderDemo/BookReaderDemo.css +14 -1
  85. package/BookReaderDemo/BookReaderJSAutoplay.js +4 -1
  86. package/BookReaderDemo/BookReaderJSSimple.js +1 -0
  87. package/BookReaderDemo/IADemoBr.js +147 -0
  88. package/BookReaderDemo/demo-advanced.html +2 -2
  89. package/BookReaderDemo/demo-autoplay.html +2 -1
  90. package/BookReaderDemo/demo-embed-iframe-src.html +2 -1
  91. package/BookReaderDemo/demo-fullscreen-mobile.html +2 -1
  92. package/BookReaderDemo/demo-fullscreen.html +2 -1
  93. package/BookReaderDemo/demo-iiif.html +2 -1
  94. package/BookReaderDemo/demo-internetarchive.html +84 -17
  95. package/BookReaderDemo/demo-multiple.html +2 -1
  96. package/BookReaderDemo/demo-preview-pages.html +2 -1
  97. package/BookReaderDemo/demo-simple.html +2 -1
  98. package/BookReaderDemo/demo-vendor-fullscreen.html +2 -1
  99. package/BookReaderDemo/ia-multiple-volumes-manifest.js +170 -0
  100. package/BookReaderDemo/immersion-1up.html +2 -1
  101. package/BookReaderDemo/immersion-mode.html +2 -1
  102. package/BookReaderDemo/toggle_controls.html +2 -1
  103. package/BookReaderDemo/view_mode.html +2 -1
  104. package/BookReaderDemo/viewmode-cycle.html +2 -3
  105. package/CHANGELOG.md +246 -0
  106. package/README.md +14 -1
  107. package/babel.config.js +19 -0
  108. package/codecov.yml +6 -0
  109. package/index.html +3 -0
  110. package/jsconfig.json +19 -0
  111. package/netlify.toml +5 -0
  112. package/package.json +70 -59
  113. package/renovate.json +52 -0
  114. package/scripts/preversion.js +4 -1
  115. package/src/BookNavigator/assets/bookmark-colors.js +1 -1
  116. package/src/BookNavigator/assets/button-base.js +9 -2
  117. package/src/BookNavigator/assets/ia-logo.js +17 -0
  118. package/src/BookNavigator/assets/icon_checkmark.js +1 -1
  119. package/src/BookNavigator/assets/icon_close.js +1 -1
  120. package/src/BookNavigator/assets/icon_sort_asc.js +5 -0
  121. package/src/BookNavigator/assets/{icon_sort_ascending.js → icon_sort_desc.js} +2 -2
  122. package/src/BookNavigator/assets/icon_sort_neutral.js +5 -0
  123. package/src/BookNavigator/assets/icon_volumes.js +11 -0
  124. package/src/BookNavigator/book-navigator.js +585 -0
  125. package/src/BookNavigator/bookmarks/bookmark-button.js +3 -2
  126. package/src/BookNavigator/bookmarks/bookmark-edit.js +3 -4
  127. package/src/BookNavigator/bookmarks/bookmarks-list.js +2 -3
  128. package/src/BookNavigator/bookmarks/bookmarks-loginCTA.js +3 -8
  129. package/src/BookNavigator/bookmarks/bookmarks-provider.js +27 -17
  130. package/src/BookNavigator/bookmarks/ia-bookmarks.js +116 -67
  131. package/src/BookNavigator/delete-modal-actions.js +1 -1
  132. package/src/BookNavigator/downloads/downloads-provider.js +36 -21
  133. package/src/BookNavigator/downloads/downloads.js +41 -25
  134. package/src/BookNavigator/search/search-provider.js +80 -28
  135. package/src/BookNavigator/search/search-results.js +34 -25
  136. package/src/BookNavigator/sharing.js +27 -0
  137. package/src/BookNavigator/visual-adjustments/visual-adjustments-provider.js +11 -10
  138. package/src/BookNavigator/visual-adjustments/visual-adjustments.js +3 -3
  139. package/src/BookNavigator/volumes/volumes-provider.js +88 -53
  140. package/src/BookNavigator/volumes/volumes.js +41 -14
  141. package/src/BookReader/BookModel.js +59 -30
  142. package/src/BookReader/DebugConsole.js +3 -3
  143. package/src/BookReader/DragScrollable.js +233 -0
  144. package/src/BookReader/Mode1Up.js +56 -351
  145. package/src/BookReader/Mode1UpLit.js +391 -0
  146. package/src/BookReader/Mode2Up.js +73 -1318
  147. package/src/BookReader/Mode2UpLit.js +781 -0
  148. package/src/BookReader/ModeCoordinateSpace.js +29 -0
  149. package/src/BookReader/ModeSmoothZoom.js +211 -0
  150. package/src/BookReader/ModeThumb.js +17 -11
  151. package/src/BookReader/Navbar/Navbar.js +10 -36
  152. package/src/BookReader/PageContainer.js +69 -6
  153. package/src/BookReader/ReduceSet.js +1 -1
  154. package/src/BookReader/Toolbar/Toolbar.js +10 -37
  155. package/src/BookReader/events.js +2 -0
  156. package/src/BookReader/options.js +24 -2
  157. package/src/BookReader/utils/HTMLDimensionsCacher.js +44 -0
  158. package/src/BookReader/utils/ScrollClassAdder.js +31 -0
  159. package/src/BookReader/utils.js +108 -13
  160. package/src/BookReader.js +480 -825
  161. package/src/assets/icons/close-circle-dark.svg +1 -0
  162. package/src/assets/icons/magnify-minus.svg +3 -7
  163. package/src/assets/icons/magnify-plus.svg +3 -7
  164. package/src/assets/icons/voice.svg +1 -0
  165. package/src/css/BookReader.scss +0 -12
  166. package/src/css/_BRBookmarks.scss +1 -1
  167. package/src/css/_BRComponent.scss +1 -1
  168. package/src/css/_BRmain.scss +33 -24
  169. package/src/css/_BRnav.scss +4 -26
  170. package/src/css/_BRpages.scss +147 -40
  171. package/src/css/_BRsearch.scss +25 -216
  172. package/src/css/_TextSelection.scss +16 -17
  173. package/src/css/_colorbox.scss +2 -2
  174. package/src/css/_controls.scss +17 -5
  175. package/src/css/_icons.scss +7 -1
  176. package/src/ia-bookreader/ia-bookreader.js +224 -0
  177. package/src/plugins/plugin.archive_analytics.js +3 -3
  178. package/src/plugins/plugin.autoplay.js +4 -9
  179. package/src/plugins/plugin.chapters.js +28 -35
  180. package/src/plugins/plugin.mobile_nav.js +11 -10
  181. package/src/plugins/plugin.resume.js +3 -3
  182. package/src/plugins/plugin.text_selection.js +32 -41
  183. package/src/plugins/plugin.vendor-fullscreen.js +4 -4
  184. package/src/plugins/search/plugin.search.js +179 -116
  185. package/src/plugins/search/view.js +63 -179
  186. package/src/plugins/tts/AbstractTTSEngine.js +46 -37
  187. package/src/plugins/tts/FestivalTTSEngine.js +13 -14
  188. package/src/plugins/tts/PageChunk.js +15 -21
  189. package/src/plugins/tts/PageChunkIterator.js +8 -12
  190. package/src/plugins/tts/WebTTSEngine.js +87 -71
  191. package/src/plugins/tts/plugin.tts.js +94 -125
  192. package/src/plugins/tts/utils.js +0 -25
  193. package/src/plugins/url/UrlPlugin.js +193 -0
  194. package/src/plugins/{plugin.url.js → url/plugin.url.js} +45 -16
  195. package/src/util/docCookies.js +21 -2
  196. package/tests/e2e/README.md +37 -0
  197. package/tests/e2e/autoplay.test.js +2 -2
  198. package/tests/e2e/base.test.js +7 -7
  199. package/tests/e2e/helpers/base.js +28 -23
  200. package/tests/e2e/helpers/debug.js +1 -1
  201. package/tests/e2e/helpers/desktopSearch.js +14 -13
  202. package/tests/e2e/helpers/mobileSearch.js +3 -3
  203. package/tests/e2e/helpers/params.js +17 -0
  204. package/tests/e2e/helpers/rightToLeft.js +4 -10
  205. package/tests/e2e/models/Navigation.js +13 -4
  206. package/tests/e2e/rightToLeft.test.js +4 -5
  207. package/tests/e2e/viewmode.test.js +40 -33
  208. package/tests/jest/BookNavigator/book-navigator.test.js +658 -0
  209. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  210. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  211. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  212. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  213. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  214. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  215. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  216. package/tests/{karma/BookNavigator → jest/BookNavigator/search}/search-results.test.js +104 -60
  217. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  218. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  219. package/tests/jest/BookNavigator/volumes/volumes-provider.test.js +184 -0
  220. package/tests/jest/BookNavigator/volumes/volumes.test.js +97 -0
  221. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +59 -14
  222. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +193 -0
  223. package/tests/{BookReader → jest/BookReader}/DebugConsole.test.js +1 -1
  224. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  225. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  226. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  227. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  228. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  229. package/tests/jest/BookReader/ModeSmoothZoom.test.js +175 -0
  230. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  231. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +10 -10
  232. package/tests/{BookReader → jest/BookReader}/PageContainer.test.js +88 -6
  233. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  234. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +2 -2
  235. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  236. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  237. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  238. package/tests/jest/BookReader/utils.test.js +217 -0
  239. package/tests/jest/BookReader.keyboard.test.js +190 -0
  240. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +9 -1
  241. package/tests/{BookReader.test.js → jest/BookReader.test.js} +26 -37
  242. package/tests/{plugins → jest/plugins}/plugin.archive_analytics.test.js +2 -2
  243. package/tests/{plugins → jest/plugins}/plugin.autoplay.test.js +4 -4
  244. package/tests/{plugins → jest/plugins}/plugin.chapters.test.js +10 -11
  245. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +2 -2
  246. package/tests/{plugins → jest/plugins}/plugin.mobile_nav.test.js +5 -5
  247. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +3 -3
  248. package/tests/{plugins → jest/plugins}/plugin.text_selection.test.js +39 -47
  249. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  250. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +63 -47
  251. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +35 -6
  252. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +9 -9
  253. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  254. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  255. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  256. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +47 -1
  257. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  258. package/tests/jest/plugins/url/UrlPlugin.test.js +190 -0
  259. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +53 -14
  260. package/tests/jest/setup.js +3 -0
  261. package/tests/{util → jest/util}/browserSniffing.test.js +1 -1
  262. package/tests/jest/util/docCookies.test.js +24 -0
  263. package/tests/{util → jest/util}/strings.test.js +1 -1
  264. package/tests/{utils.js → jest/utils.js} +38 -0
  265. package/webpack.config.js +11 -5
  266. package/.babelrc +0 -12
  267. package/.dependabot/config.yml +0 -6
  268. package/.testcaferc.json +0 -5
  269. package/BookReader/bookreader-component-bundle.js +0 -14312
  270. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  271. package/BookReader/bookreader-component-bundle.js.map +0 -1
  272. package/BookReader/icons/sort-ascending.svg +0 -1
  273. package/BookReader/icons/sort-descending.svg +0 -1
  274. package/BookReader/jquery-1.10.1.js +0 -108
  275. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  276. package/BookReader/plugins/plugin.menu_toggle.js +0 -369
  277. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  278. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  279. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  280. package/karma.conf.js +0 -23
  281. package/src/BookNavigator/BookModel.js +0 -14
  282. package/src/BookNavigator/BookNavigator.js +0 -452
  283. package/src/BookNavigator/assets/book-loader.js +0 -27
  284. package/src/BookNavigator/assets/icon_sort_descending.js +0 -5
  285. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  286. package/src/BookNavigator/search/a-search-result.js +0 -55
  287. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  288. package/src/ItemNavigator/ItemNavigator.js +0 -372
  289. package/src/ItemNavigator/providers/sharing.js +0 -29
  290. package/src/assets/icons/sort-ascending.svg +0 -1
  291. package/src/assets/icons/sort-descending.svg +0 -1
  292. package/src/dragscrollable-br.js +0 -261
  293. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  294. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  295. package/tests/BookReader/Mode1Up.test.js +0 -164
  296. package/tests/BookReader/Mode2Up.test.js +0 -247
  297. package/tests/BookReader/utils.test.js +0 -109
  298. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  299. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  300. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  301. package/tests/karma/BookNavigator/volumes.test.js +0 -101
  302. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  303. package/tests/util/docCookies.test.js +0 -15
@@ -0,0 +1,175 @@
1
+ import sinon from 'sinon';
2
+ import { EventTargetSpy } from '../utils.js';
3
+ import { ModeSmoothZoom } from '@/src/BookReader/ModeSmoothZoom.js';
4
+ /** @typedef {import('@/src/BookReader/ModeSmoothZoom.js').SmoothZoomable} SmoothZoomable */
5
+
6
+ /**
7
+ * @param {Partial<SmoothZoomable>} overrides
8
+ * @returns {SmoothZoomable}
9
+ */
10
+ function dummy_mode(overrides = {}) {
11
+ return {
12
+ $container: document.createElement('div'),
13
+ $visibleWorld: document.createElement('div'),
14
+ scale: 1,
15
+ htmlDimensionsCacher: {
16
+ clientWidth: 100,
17
+ clientHeight: 100,
18
+ boundingClientRect: { left: 0, top: 0 },
19
+ },
20
+ scaleCenter: {x: 0.5, y: 0.5},
21
+ ...overrides
22
+ };
23
+ }
24
+
25
+ afterEach(() => sinon.restore());
26
+
27
+ describe('ModeSmoothZoom', () => {
28
+ test('preventsDefault on iOS-only gesture events', () => {
29
+ const mode = dummy_mode();
30
+ const msz = new ModeSmoothZoom(mode);
31
+ msz.attach();
32
+ for (const event_name of ['gesturestart', 'gesturechange', 'gestureend']) {
33
+ const ev = new Event(event_name, {});
34
+ const prevDefaultSpy = sinon.spy(ev, 'preventDefault');
35
+ mode.$container.dispatchEvent(ev);
36
+ expect(prevDefaultSpy.callCount).toBe(1);
37
+ }
38
+ });
39
+
40
+ test('pinchCancel alias for pinchEnd', () => {
41
+ const mode = dummy_mode();
42
+ const msz = new ModeSmoothZoom(mode);
43
+ const pinchEndSpy = sinon.spy(msz, '_pinchEnd');
44
+ msz._pinchStart();
45
+ msz._pinchCancel();
46
+ expect(pinchEndSpy.callCount).toBe(1);
47
+ });
48
+
49
+ test('sets will-change', async () => {
50
+ const mode = dummy_mode();
51
+ const msz = new ModeSmoothZoom(mode);
52
+ expect(mode.$visibleWorld.style.willChange).toBeFalsy();
53
+ msz._pinchStart();
54
+ expect(mode.$visibleWorld.style.willChange).toBe('transform');
55
+ await msz._pinchEnd();
56
+ expect(mode.$visibleWorld.style.willChange).toBe('auto');
57
+ });
58
+
59
+ test('pinch move updates scale', () => {
60
+ const mode = dummy_mode();
61
+ const msz = new ModeSmoothZoom(mode);
62
+ // disable buffering
63
+ msz.bufferFn = (callback) => callback();
64
+ msz._pinchStart();
65
+ expect(mode.scale).toBe(1);
66
+ msz._pinchMove({ scale: 2, center: { x: 0, y: 0 }});
67
+ expect(mode.scale).toBe(2);
68
+ });
69
+
70
+ test('updateScaleCenter sets scaleCenter in unitless coordinates', () => {
71
+ const mode = dummy_mode({
72
+ htmlDimensionsCacher: {
73
+ clientWidth: 200,
74
+ clientHeight: 100,
75
+ boundingClientRect: {
76
+ left: 5,
77
+ top: 50
78
+ }
79
+ }
80
+ });
81
+ const msz = new ModeSmoothZoom(mode);
82
+ expect(mode.scaleCenter).toEqual({ x: 0.5, y: 0.5 });
83
+ msz.updateScaleCenter({ clientX: 85, clientY: 110 });
84
+ expect(mode.scaleCenter).toEqual({ x: 0.4, y: 0.6 });
85
+ });
86
+
87
+ test('detaches all listeners', () => {
88
+ const mode = dummy_mode();
89
+ const msz = new ModeSmoothZoom(mode);
90
+ const containerEventSpy = EventTargetSpy.wrap(mode.$container);
91
+ const visibleWorldSpy = EventTargetSpy.wrap(mode.$visibleWorld);
92
+ const hammerEventSpy = new EventTargetSpy();
93
+ msz.hammer.on = hammerEventSpy.addEventListener.bind(hammerEventSpy);
94
+ msz.hammer.off = hammerEventSpy.removeEventListener.bind(hammerEventSpy);
95
+
96
+ msz.attach();
97
+ expect(containerEventSpy._totalListenerCount).toBeGreaterThan(0);
98
+ expect(hammerEventSpy._totalListenerCount).toBeGreaterThan(0);
99
+
100
+ msz.detach();
101
+ expect(containerEventSpy._totalListenerCount).toBe(0);
102
+ expect(visibleWorldSpy._totalListenerCount).toBe(0);
103
+ expect(hammerEventSpy._totalListenerCount).toBe(0);
104
+ });
105
+
106
+ test('attach can be called twice without double attachments', () => {
107
+ const mode = dummy_mode();
108
+ const msz = new ModeSmoothZoom(mode);
109
+ const containerEventSpy = EventTargetSpy.wrap(mode.$container);
110
+ const visibleWorldSpy = EventTargetSpy.wrap(mode.$visibleWorld);
111
+ const hammerEventSpy = new EventTargetSpy();
112
+ msz.hammer.on = hammerEventSpy.addEventListener.bind(hammerEventSpy);
113
+ msz.hammer.off = hammerEventSpy.removeEventListener.bind(hammerEventSpy);
114
+ msz.attach();
115
+
116
+ const containerListenersCount = containerEventSpy._totalListenerCount;
117
+ const visibleWorldListenersCount = visibleWorldSpy._totalListenerCount;
118
+ const hammerListenersCount = hammerEventSpy._totalListenerCount;
119
+
120
+ msz.attach();
121
+ expect(containerEventSpy._totalListenerCount).toBe(containerListenersCount);
122
+ expect(visibleWorldSpy._totalListenerCount).toBe(visibleWorldListenersCount);
123
+ expect(hammerEventSpy._totalListenerCount).toBe(hammerListenersCount);
124
+ });
125
+
126
+ describe('_handleCtrlWheel', () => {
127
+ test('non-ctrl wheel events ignored', () => {
128
+ const mode = dummy_mode();
129
+ const msz = new ModeSmoothZoom(mode);
130
+ expect(mode.scale).toBe(1);
131
+ const ev = new WheelEvent('wheel', { ctrlKey: false, deltaY: 20 });
132
+ const preventDefaultSpy = sinon.spy(ev, 'preventDefault');
133
+ msz._handleCtrlWheel(ev);
134
+ expect(preventDefaultSpy.callCount).toBe(0);
135
+ expect(mode.scale).toBe(1);
136
+ });
137
+
138
+ test('ctrl-wheel events update scale', () => {
139
+ const mode = dummy_mode();
140
+ const msz = new ModeSmoothZoom(mode);
141
+ expect(mode.scale).toBe(1);
142
+ const ev = new WheelEvent('wheel', { ctrlKey: true, deltaY: 20 });
143
+ const preventDefaultSpy = sinon.spy(ev, 'preventDefault');
144
+ msz._handleCtrlWheel(ev);
145
+ expect(preventDefaultSpy.callCount).toBe(1);
146
+ expect(mode.scale).not.toBe(1);
147
+ });
148
+ });
149
+
150
+ describe("updateViewportOnZoom", () => {
151
+ test("adjusts scroll position when zooming in", () => {
152
+ const mode = dummy_mode();
153
+ const msz = new ModeSmoothZoom(mode);
154
+ mode.$container.scrollTop = 100;
155
+ mode.$container.scrollLeft = 100;
156
+
157
+ msz.updateViewportOnZoom(2, 1);
158
+
159
+ expect(mode.$container.scrollTop).toBeGreaterThan(100);
160
+ expect(mode.$container.scrollLeft).toBeGreaterThan(100);
161
+ });
162
+
163
+ test("updates scroll position when zooming out", () => {
164
+ const mode = dummy_mode();
165
+ const msz = new ModeSmoothZoom(mode);
166
+ mode.$container.scrollTop = 100;
167
+ mode.$container.scrollLeft = 100;
168
+
169
+ msz.updateViewportOnZoom(0.5, 1);
170
+
171
+ expect(mode.$container.scrollTop).toBeLessThan(100);
172
+ expect(mode.$container.scrollLeft).toBeLessThan(100);
173
+ });
174
+ });
175
+ });
@@ -0,0 +1,71 @@
1
+
2
+ import sinon from 'sinon';
3
+ import BookReader from '@/src/BookReader.js';
4
+ /** @typedef {import('@/src/BookReader/options.js').BookReaderOptions} BookReaderOptions */
5
+
6
+ beforeAll(() => {
7
+ global.alert = jest.fn();
8
+ });
9
+ afterEach(() => {
10
+ jest.restoreAllMocks();
11
+ sinon.restore();
12
+ });
13
+
14
+ /** @type {BookReaderOptions['data']} */
15
+ const SAMPLE_DATA = [
16
+ [
17
+ { width: 123, height: 123, uri: 'https://archive.org/image0.jpg', pageNum: '1' },
18
+ ],
19
+ [
20
+ { width: 123, height: 123, uri: 'https://archive.org/image1.jpg', pageNum: '2' },
21
+ { width: 123, height: 123, uri: 'https://archive.org/image2.jpg', pageNum: '3' },
22
+ ],
23
+ [
24
+ { width: 123, height: 123, uri: 'https://archive.org/image3.jpg', pageNum: '4' },
25
+ { width: 123, height: 123, uri: 'https://archive.org/image4.jpg', pageNum: '5' },
26
+ ],
27
+ [
28
+ { width: 123, height: 123, uri: 'https://archive.org/image5.jpg', pageNum: '6' },
29
+ ],
30
+ ];
31
+
32
+ describe('zoom', () => {
33
+ const br = new BookReader({ data: SAMPLE_DATA });
34
+ br.init();
35
+
36
+ test('initializes with default columns', () => {
37
+ expect(br.thumbColumns).toBe(br.options.thumbColumns);
38
+ });
39
+
40
+ test('removes column and redraws zooming in', () => {
41
+ const prepare = sinon.spy(br._modes.modeThumb, 'prepare');
42
+ const startColumns = br.thumbColumns;
43
+ br._modes.modeThumb.zoom('in');
44
+ expect(br.thumbColumns).toBe(startColumns - 1);
45
+ expect(prepare.callCount).toBe(1);
46
+ });
47
+
48
+ test('adds column and redraws zooming out', () => {
49
+ const prepare = sinon.spy(br._modes.modeThumb, 'prepare');
50
+ const startColumns = br.thumbColumns;
51
+ br._modes.modeThumb.zoom('out');
52
+ expect(br.thumbColumns).toBe(startColumns + 1);
53
+ expect(prepare.callCount).toBe(1);
54
+ });
55
+
56
+ test('keeps columns and no redraw at zooming in limit', () => {
57
+ const prepare = sinon.spy(br._modes.modeThumb, 'prepare');
58
+ br.thumbColumns = br.options.thumbMinZoomColumns;
59
+ br._modes.modeThumb.zoom('in');
60
+ expect(br.thumbColumns).toBe(br.options.thumbMinZoomColumns);
61
+ expect(prepare.callCount).toBe(0);
62
+ });
63
+
64
+ test('keeps columns and no redraw at zooming out limit', () => {
65
+ const prepare = sinon.spy(br._modes.modeThumb, 'prepare');
66
+ br.thumbColumns = br.options.thumbMaxZoomColumns;
67
+ br._modes.modeThumb.zoom('out');
68
+ expect(br.thumbColumns).toBe(br.options.thumbMaxZoomColumns);
69
+ expect(prepare.callCount).toBe(0);
70
+ });
71
+ });
@@ -1,19 +1,19 @@
1
1
  import sinon from 'sinon';
2
- import { getNavPageNumHtml } from '../../../src/BookReader/Navbar/Navbar.js';
3
- import BookReader from '../../../src/BookReader.js';
2
+ import { getNavPageNumHtml } from '@/src/BookReader/Navbar/Navbar.js';
3
+ import BookReader from '@/src/BookReader.js';
4
4
 
5
5
  describe('getNavPageNumHtml', () => {
6
6
  const f = getNavPageNumHtml;
7
7
  test('handle n-prefixed page numbers', () => {
8
- expect(f(3, 40, 'n3', '', 40)).toBe('Page (4 of 40)');
8
+ expect(f(3, 40, 'n3', '', 40)).toBe('(4 of 40)');
9
9
  });
10
10
 
11
11
  test('handle regular page numbers', () => {
12
- expect(f(3, 40, '14', '', 40)).toBe('Page 14 of 40');
12
+ expect(f(3, 40, '14', '', 40)).toBe('14 of 40');
13
13
  });
14
14
 
15
15
  test('handle no max page', () => {
16
- expect(f(3, 40, '14', '', null)).toBe('Page 14');
16
+ expect(f(3, 40, '14', '', null)).toBe('14');
17
17
  });
18
18
  });
19
19
 
@@ -63,14 +63,14 @@ describe('Navbar slider', () => {
63
63
 
64
64
  test('on slide change, actual page changes', () => {
65
65
  const $slider = navbar.$root.find('.BRpager');
66
- const jumpToIndexSpy = sinon.spy(br, 'jumpToIndex');
66
+ const jumpToIndexStub = sinon.stub(br, 'jumpToIndex');
67
67
  expect(br.currentIndex()).toBe(0);
68
68
 
69
69
  $slider.trigger('slidechange', { value: 3 });
70
70
 
71
71
  expect(navbar.$root.find('.BRcurrentpage').text().includes('3'));
72
- expect(jumpToIndexSpy.callCount).toBe(1);
73
- expect(jumpToIndexSpy.args[0][0]).toBe(3);
72
+ expect(jumpToIndexStub.callCount).toBe(1);
73
+ expect(jumpToIndexStub.args[0][0]).toBe(3);
74
74
  });
75
75
  });
76
76
 
@@ -97,9 +97,9 @@ describe('Navbar controls overrides', () => {
97
97
  const $viewMode = navbar.$root.find('.viewmode');
98
98
 
99
99
  expect($viewMode.find('.icon-thumb').length).toBe(1);
100
- $viewMode.click();
100
+ $viewMode.trigger("click");
101
101
  expect($viewMode.find('.icon-twopg').length).toBe(1);
102
- $viewMode.click();
102
+ $viewMode.trigger("click");
103
103
  expect($viewMode.find('.icon-thumb').length).toBe(1);
104
104
  });
105
105
 
@@ -1,4 +1,4 @@
1
- import {PageContainer} from '../../src/BookReader/PageContainer.js';
1
+ import {PageContainer, boxToSVGRect, createSVGPageLayer, renderBoxesInPageContainerLayer} from '@/src/BookReader/PageContainer.js';
2
2
 
3
3
  describe('constructor', () => {
4
4
  test('protected books', () => {
@@ -31,7 +31,7 @@ describe('constructor', () => {
31
31
  });
32
32
  });
33
33
 
34
- describe('update', async () => {
34
+ describe('update', () => {
35
35
  test('dimensions sets CSS', () => {
36
36
  const pc = new PageContainer(null, {});
37
37
  pc.update({ dimensions: { left: 20 } });
@@ -62,17 +62,18 @@ describe('update', async () => {
62
62
  expect(pc.$img[0].style.background).toBe('');
63
63
  });
64
64
 
65
- test('removes image between updates', () => {
65
+ test('removes image between updates only if changed', () => {
66
66
  const fakeImageCache = {
67
67
  imageLoaded: () => true,
68
- image: () => $('<img/>'),
68
+ image: (index, reduce) => $(`<img src="page${index}-${reduce}.jpg" />`),
69
69
  };
70
70
  const pc = new PageContainer({index: 12}, {imageCache: fakeImageCache});
71
71
  pc.update({ reduce: 7 });
72
72
  const $im1 = pc.$img;
73
73
  pc.update({ reduce: 7 });
74
- const $im2 = pc.$img;
75
- expect($im1).not.toBe($im2);
74
+ expect(pc.$img).toBe($im1);
75
+ pc.update({ reduce: 16 });
76
+ expect(pc.$img).not.toBe($im1);
76
77
  expect($im1.parent().length).toBe(0);
77
78
  });
78
79
 
@@ -113,3 +114,84 @@ describe('update', async () => {
113
114
  expect(pc.$img.css('background')).toBeFalsy();
114
115
  });
115
116
  });
117
+
118
+ describe('createSVGPageLayer', () => {
119
+ test('Does what it says', () => {
120
+ const svg = createSVGPageLayer({ width: 100, height: 200}, 'myClass');
121
+ expect(svg.getAttribute('viewBox')).toBe('0 0 100 200');
122
+ expect(svg.getAttribute('class')).toContain('myClass');
123
+ });
124
+ });
125
+
126
+ describe('boxToSVGRect', () => {
127
+ test('Does what it says', () => {
128
+ const rect = boxToSVGRect({ l: 100, r: 200, t: 300, b: 500 });
129
+ expect(rect.getAttribute('x')).toBe('100');
130
+ expect(rect.getAttribute('y')).toBe('300');
131
+ expect(rect.getAttribute('width')).toBe('100');
132
+ expect(rect.getAttribute('height')).toBe('200');
133
+ });
134
+ });
135
+
136
+ describe('renderBoxesInPageContainerLayer', () => {
137
+ test('Handles missing layer', () => {
138
+ const container = document.createElement('div');
139
+ const page = { width: 100, height: 200 };
140
+ const boxes = [{l: 1, r: 2, t: 3, b: 4}];
141
+ renderBoxesInPageContainerLayer('foo', boxes, page, container);
142
+ expect(container.querySelector('.foo')).toBeTruthy();
143
+ expect(container.querySelectorAll('.foo rect').length).toBe(1);
144
+ });
145
+
146
+ test('Handles existing layer', () => {
147
+ const container = document.createElement('div');
148
+ const layer = document.createElement('svg');
149
+ layer.classList.add('foo');
150
+ container.append(layer);
151
+
152
+ const page = { width: 100, height: 200 };
153
+ const boxes = [{l: 1, r: 2, t: 3, b: 4}];
154
+ renderBoxesInPageContainerLayer('foo', boxes, page, container);
155
+ expect(container.querySelector('.foo')).toBe(layer);
156
+ expect(container.querySelectorAll('.foo rect').length).toBe(1);
157
+ });
158
+
159
+ test('Adds layer after image if it exists', () => {
160
+ const container = document.createElement('div');
161
+ const img = document.createElement('img');
162
+ img.classList.add('BRpageimage');
163
+ container.append(img);
164
+
165
+ const page = { width: 100, height: 200 };
166
+ const boxes = [{l: 1, r: 2, t: 3, b: 4}];
167
+ renderBoxesInPageContainerLayer('foo', boxes, page, container);
168
+ expect(container.querySelector('.foo')).toBeTruthy();
169
+ expect(container.children[0].getAttribute('class')).toBe('BRpageimage');
170
+ expect(container.children[1].getAttribute('class')).toBe('BRPageLayer foo');
171
+ });
172
+
173
+ test('Renders all boxes', () => {
174
+ const container = document.createElement('div');
175
+ const page = { width: 100, height: 200 };
176
+ const boxes = [{l: 1, r: 2, t: 3, b: 4}, {l: 1, r: 2, t: 3, b: 4}, {l: 1, r: 2, t: 3, b: 4}];
177
+ renderBoxesInPageContainerLayer('foo', boxes, page, container);
178
+ expect(container.querySelectorAll('.foo rect').length).toBe(3);
179
+ });
180
+
181
+ test('Adds optional classes', () => {
182
+ const container = document.createElement('div');
183
+ const page = { width: 100, height: 200 };
184
+ const boxes = [{l: 1, r: 2, t: 3, b: 4}, {l: 1, r: 2, t: 3, b: 4}, {l: 1, r: 2, t: 3, b: 4}];
185
+ renderBoxesInPageContainerLayer('foo', boxes, page, container, ['match-1', 'match-2', 'match-3']);
186
+ const rects = Array.from(container.querySelectorAll('.foo rect'));
187
+ expect(rects.map(r => r.getAttribute('class'))).toEqual(['match-1', 'match-2', 'match-3']);
188
+ });
189
+
190
+ test('Handles no boxes', () => {
191
+ const container = document.createElement('div');
192
+ const page = { width: 100, height: 200 };
193
+ const boxes = [];
194
+ renderBoxesInPageContainerLayer('foo', boxes, page, container);
195
+ expect(container.querySelectorAll('.foo rect').length).toBe(0);
196
+ });
197
+ });
@@ -1,4 +1,4 @@
1
- import {IntegerReduceSet, Pow2ReduceSet} from '../../src/BookReader/ReduceSet.js';
1
+ import {IntegerReduceSet, Pow2ReduceSet} from '@/src/BookReader/ReduceSet.js';
2
2
 
3
3
  describe('IntegerReduceSet', () => {
4
4
  test('floor', () => {
@@ -1,10 +1,10 @@
1
1
  import sinon from 'sinon';
2
2
 
3
- import { createPopup } from '../../../src/BookReader/Toolbar/Toolbar.js';
3
+ import { createPopup } from '@/src/BookReader/Toolbar/Toolbar.js';
4
4
 
5
5
  afterEach(() => {
6
6
  sinon.restore();
7
- })
7
+ });
8
8
 
9
9
  describe('createPopup', () => {
10
10
  test('calls window.open', () => {
@@ -0,0 +1,59 @@
1
+ // @ts-check
2
+ import sinon from 'sinon';
3
+ import { HTMLDimensionsCacher } from '@/src/BookReader/utils/HTMLDimensionsCacher';
4
+
5
+ describe('HTMLDimensionsCacher', () => {
6
+ test('Does not read from element directly', () => {
7
+ const element = document.createElement('div');
8
+ const getBoundingClientRectStub = element.getBoundingClientRect = sinon.stub().returns({ top: 10, left: 20 });
9
+ const clientWidthStub = sinon.stub().returns(300);
10
+ const clientHeightStub = sinon.stub().returns(500);
11
+ Object.defineProperty(element, 'clientWidth', { get: clientWidthStub });
12
+ Object.defineProperty(element, 'clientHeight', { get: clientHeightStub });
13
+
14
+ const hdc = new HTMLDimensionsCacher(element);
15
+ hdc.clientHeight;
16
+ expect(getBoundingClientRectStub.callCount).toBe(0);
17
+ expect(clientWidthStub.callCount).toBe(0);
18
+ expect(clientHeightStub.callCount).toBe(0);
19
+ });
20
+
21
+ test('Read from element when calling update', () => {
22
+ const element = document.createElement('div');
23
+ const getBoundingClientRectStub = element.getBoundingClientRect = sinon.stub().returns({ top: 10, left: 20 });
24
+ const clientWidthStub = sinon.stub().returns(300);
25
+ const clientHeightStub = sinon.stub().returns(500);
26
+ Object.defineProperty(element, 'clientWidth', { get: clientWidthStub });
27
+ Object.defineProperty(element, 'clientHeight', { get: clientHeightStub });
28
+
29
+ const hdc = new HTMLDimensionsCacher(element);
30
+ hdc.updateClientSizes();
31
+ expect(getBoundingClientRectStub.callCount).toBe(1);
32
+ expect(clientWidthStub.callCount).toBe(1);
33
+ expect(clientHeightStub.callCount).toBe(1);
34
+
35
+ expect(hdc.boundingClientRect).toEqual({ top: 10, left: 20 });
36
+ expect(hdc.clientWidth).toBe(300);
37
+ expect(hdc.clientHeight).toBe(500);
38
+ });
39
+
40
+ test('Does not listen for window resizes when not attached', () => {
41
+ const element = document.createElement('div');
42
+ const hdc = new HTMLDimensionsCacher(element);
43
+ const dummyWindow = new EventTarget();
44
+ const debouncedUpdateSpy = sinon.spy(hdc, 'debouncedUpdateClientSizes');
45
+
46
+ dummyWindow.dispatchEvent(new Event('resize', {}));
47
+ expect(debouncedUpdateSpy.callCount).toBe(0);
48
+
49
+ hdc.attachResizeListener(dummyWindow);
50
+
51
+ dummyWindow.dispatchEvent(new Event('resize', {}));
52
+ expect(debouncedUpdateSpy.callCount).toBe(1);
53
+
54
+ hdc.detachResizeListener(dummyWindow);
55
+
56
+ dummyWindow.dispatchEvent(new Event('resize', {}));
57
+ expect(debouncedUpdateSpy.callCount).toBe(1);
58
+ });
59
+ });
@@ -0,0 +1,49 @@
1
+ // @ts-check
2
+ import sinon from 'sinon';
3
+ import { ScrollClassAdder } from '@/src/BookReader/utils/ScrollClassAdder';
4
+
5
+ describe('ScrollClassAdder', () => {
6
+ test('Does not attach during construction', () => {
7
+ const element = document.createElement('div');
8
+ const attachSpy = sinon.spy(ScrollClassAdder.prototype, 'attach');
9
+ new ScrollClassAdder(element, 'foo');
10
+ expect(attachSpy.callCount).toBe(0);
11
+ });
12
+
13
+ test('Attach/detach call correct methods', () => {
14
+ const el = document.createElement('div');
15
+ const addEventListenerSpy = sinon.spy(el, 'addEventListener');
16
+ const removeEventListenerSpy = sinon.spy(el, 'removeEventListener');
17
+ const sca = new ScrollClassAdder(el, 'foo');
18
+ expect(addEventListenerSpy.callCount).toBe(0);
19
+ sca.attach();
20
+ expect(addEventListenerSpy.callCount).toBe(1);
21
+ sca.detach();
22
+ expect(removeEventListenerSpy.callCount).toBe(1);
23
+ });
24
+
25
+ test('onScroll adds class at right time', () => {
26
+ const clock = sinon.useFakeTimers();
27
+ const el = document.createElement('div');
28
+ const sca = new ScrollClassAdder(el, 'foo');
29
+ expect(el.getAttribute('class')).toBeFalsy();
30
+ sca.onScroll();
31
+ expect(el.getAttribute('class')).toBe('foo');
32
+ clock.tick(600);
33
+ expect(el.getAttribute('class')).toBeFalsy();
34
+
35
+ sca.onScroll();
36
+ expect(el.getAttribute('class')).toBe('foo');
37
+ clock.tick(500);
38
+ expect(el.getAttribute('class')).toBe('foo');
39
+ sca.onScroll();
40
+ expect(el.getAttribute('class')).toBe('foo');
41
+ clock.tick(100);
42
+ expect(el.getAttribute('class')).toBe('foo');
43
+ clock.tick(499);
44
+ expect(el.getAttribute('class')).toBe('foo');
45
+ clock.tick(1);
46
+ expect(el.getAttribute('class')).toBeFalsy();
47
+ clock.restore();
48
+ });
49
+ });
@@ -1,4 +1,4 @@
1
- import { exposeOverrideable } from '../../../src/BookReader/utils/classes.js';
1
+ import { exposeOverrideable } from '@/src/BookReader/utils/classes.js';
2
2
 
3
3
  describe('exposeOverrideable', () => {
4
4
  test('adds method to To class', () => {