@internetarchive/bookreader 5.0.0-7 → 5.0.0-70-a1

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 (307) hide show
  1. package/.eslintrc.js +17 -15
  2. package/.github/workflows/node.js.yml +73 -10
  3. package/.github/workflows/npm-publish.yml +6 -20
  4. package/.testcaferc.js +10 -0
  5. package/BookReader/BookReader.css +396 -1129
  6. package/BookReader/BookReader.js +1 -1
  7. package/BookReader/BookReader.js.LICENSE.txt +20 -20
  8. package/BookReader/BookReader.js.map +1 -1
  9. package/BookReader/ia-bookreader-bundle.js +1788 -0
  10. package/BookReader/ia-bookreader-bundle.js.LICENSE.txt +19 -0
  11. package/BookReader/ia-bookreader-bundle.js.map +1 -0
  12. package/BookReader/icons/1up.svg +1 -1
  13. package/BookReader/icons/2up.svg +1 -1
  14. package/BookReader/icons/advance.svg +1 -1
  15. package/BookReader/icons/chevron-right.svg +1 -1
  16. package/BookReader/icons/close-circle-dark.svg +1 -1
  17. package/BookReader/icons/close-circle.svg +1 -1
  18. package/BookReader/icons/fullscreen.svg +1 -1
  19. package/BookReader/icons/fullscreen_exit.svg +1 -1
  20. package/BookReader/icons/hamburger.svg +1 -1
  21. package/BookReader/icons/left-arrow.svg +1 -1
  22. package/BookReader/icons/magnify-minus.svg +1 -1
  23. package/BookReader/icons/magnify-plus.svg +1 -1
  24. package/BookReader/icons/magnify.svg +1 -1
  25. package/BookReader/icons/pause.svg +1 -1
  26. package/BookReader/icons/play.svg +1 -1
  27. package/BookReader/icons/playback-speed.svg +1 -1
  28. package/BookReader/icons/read-aloud.svg +1 -1
  29. package/BookReader/icons/review.svg +1 -1
  30. package/BookReader/icons/thumbnails.svg +1 -1
  31. package/BookReader/icons/voice.svg +1 -0
  32. package/BookReader/icons/volume-full.svg +1 -1
  33. package/BookReader/images/BRicons.svg +3 -3
  34. package/BookReader/images/books_graphic.svg +1 -1
  35. package/BookReader/images/icon_book.svg +1 -1
  36. package/BookReader/images/icon_bookmark.svg +1 -1
  37. package/BookReader/images/icon_gear.svg +1 -1
  38. package/BookReader/images/icon_hamburger.svg +1 -1
  39. package/BookReader/images/icon_home.svg +1 -1
  40. package/BookReader/images/icon_info.svg +1 -1
  41. package/BookReader/images/icon_one_page.svg +1 -1
  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 -1
  45. package/BookReader/images/icon_search_button.svg +1 -1
  46. package/BookReader/images/icon_share.svg +1 -1
  47. package/BookReader/images/icon_skip-ahead.svg +1 -1
  48. package/BookReader/images/icon_skip-back.svg +1 -1
  49. package/BookReader/images/icon_speaker.svg +1 -1
  50. package/BookReader/images/icon_speaker_open.svg +1 -1
  51. package/BookReader/images/icon_thumbnails.svg +1 -1
  52. package/BookReader/images/icon_toc.svg +1 -1
  53. package/BookReader/images/icon_two_pages.svg +1 -1
  54. package/BookReader/images/marker_chap-off.svg +1 -1
  55. package/BookReader/images/marker_chap-on.svg +1 -1
  56. package/BookReader/images/marker_srch-on.svg +1 -1
  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 -1
  60. package/BookReader/plugins/plugin.archive_analytics.js.map +1 -1
  61. package/BookReader/plugins/plugin.autoplay.js +1 -1
  62. package/BookReader/plugins/plugin.autoplay.js.map +1 -1
  63. package/BookReader/plugins/plugin.chapters.js +25 -1
  64. package/BookReader/plugins/plugin.chapters.js.LICENSE.txt +1 -0
  65. package/BookReader/plugins/plugin.chapters.js.map +1 -1
  66. package/BookReader/plugins/plugin.iframe.js +1 -1
  67. package/BookReader/plugins/plugin.iframe.js.map +1 -1
  68. package/BookReader/plugins/plugin.resume.js +1 -1
  69. package/BookReader/plugins/plugin.resume.js.map +1 -1
  70. package/BookReader/plugins/plugin.search.js +2 -1
  71. package/BookReader/plugins/plugin.search.js.LICENSE.txt +1 -0
  72. package/BookReader/plugins/plugin.search.js.map +1 -1
  73. package/BookReader/plugins/plugin.text_selection.js +2 -1
  74. package/BookReader/plugins/plugin.text_selection.js.LICENSE.txt +1 -0
  75. package/BookReader/plugins/plugin.text_selection.js.map +1 -1
  76. package/BookReader/plugins/plugin.tts.js +1 -1
  77. package/BookReader/plugins/plugin.tts.js.LICENSE.txt +2 -0
  78. package/BookReader/plugins/plugin.tts.js.map +1 -1
  79. package/BookReader/plugins/plugin.url.js +1 -1
  80. package/BookReader/plugins/plugin.url.js.map +1 -1
  81. package/BookReader/plugins/plugin.vendor-fullscreen.js +1 -1
  82. package/BookReader/plugins/plugin.vendor-fullscreen.js.map +1 -1
  83. package/BookReader/webcomponents-bundle.js +3 -0
  84. package/BookReader/webcomponents-bundle.js.LICENSE.txt +9 -0
  85. package/BookReader/webcomponents-bundle.js.map +1 -0
  86. package/BookReaderDemo/BookReaderDemo.css +16 -19
  87. package/BookReaderDemo/BookReaderJSAdvanced.js +0 -3
  88. package/BookReaderDemo/BookReaderJSAutoplay.js +4 -1
  89. package/BookReaderDemo/BookReaderJSSimple.js +1 -0
  90. package/BookReaderDemo/IADemoBr.js +147 -0
  91. package/BookReaderDemo/demo-advanced.html +2 -2
  92. package/BookReaderDemo/demo-autoplay.html +2 -3
  93. package/BookReaderDemo/demo-embed-iframe-src.html +2 -1
  94. package/BookReaderDemo/demo-fullscreen-mobile.html +3 -5
  95. package/BookReaderDemo/demo-fullscreen.html +2 -4
  96. package/BookReaderDemo/demo-iiif.html +2 -1
  97. package/BookReaderDemo/demo-iiif.js +0 -1
  98. package/BookReaderDemo/demo-internetarchive.html +213 -17
  99. package/BookReaderDemo/demo-multiple.html +2 -1
  100. package/BookReaderDemo/demo-preview-pages.html +2 -1
  101. package/BookReaderDemo/demo-simple.html +2 -1
  102. package/BookReaderDemo/demo-vendor-fullscreen.html +2 -4
  103. package/BookReaderDemo/ia-multiple-volumes-manifest.js +170 -0
  104. package/BookReaderDemo/immersion-1up.html +2 -2
  105. package/BookReaderDemo/immersion-mode.html +2 -4
  106. package/BookReaderDemo/toggle_controls.html +3 -2
  107. package/BookReaderDemo/view_mode.html +2 -1
  108. package/BookReaderDemo/viewmode-cycle.html +2 -3
  109. package/CHANGELOG.md +279 -0
  110. package/README.md +14 -1
  111. package/babel.config.js +20 -0
  112. package/codecov.yml +6 -0
  113. package/index.html +4 -1
  114. package/jsconfig.json +19 -0
  115. package/netlify.toml +9 -0
  116. package/package.json +69 -60
  117. package/renovate.json +52 -0
  118. package/scripts/preversion.js +4 -1
  119. package/src/BookNavigator/assets/bookmark-colors.js +1 -1
  120. package/src/BookNavigator/assets/button-base.js +4 -2
  121. package/src/BookNavigator/assets/ia-logo.js +17 -0
  122. package/src/BookNavigator/assets/icon_checkmark.js +1 -1
  123. package/src/BookNavigator/assets/icon_close.js +1 -1
  124. package/src/BookNavigator/book-navigator.js +586 -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 +49 -27
  135. package/src/BookNavigator/search/search-results.js +23 -9
  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 +95 -0
  140. package/src/BookReader/BookModel.js +64 -34
  141. package/src/BookReader/DragScrollable.js +233 -0
  142. package/src/BookReader/Mode1Up.js +56 -351
  143. package/src/BookReader/Mode1UpLit.js +388 -0
  144. package/src/BookReader/Mode2Up.js +73 -1318
  145. package/src/BookReader/Mode2UpLit.js +776 -0
  146. package/src/BookReader/ModeCoordinateSpace.js +29 -0
  147. package/src/BookReader/ModeSmoothZoom.js +312 -0
  148. package/src/BookReader/ModeThumb.js +18 -12
  149. package/src/BookReader/Navbar/Navbar.js +12 -38
  150. package/src/BookReader/PageContainer.js +81 -6
  151. package/src/BookReader/ReduceSet.js +1 -1
  152. package/src/BookReader/Toolbar/Toolbar.js +10 -37
  153. package/src/BookReader/events.js +2 -3
  154. package/src/BookReader/options.js +24 -2
  155. package/src/BookReader/utils/HTMLDimensionsCacher.js +44 -0
  156. package/src/BookReader/utils/ScrollClassAdder.js +31 -0
  157. package/src/BookReader/utils/SelectionObserver.js +43 -0
  158. package/src/BookReader/utils.js +118 -13
  159. package/src/BookReader.js +423 -1056
  160. package/src/assets/icons/magnify-minus.svg +3 -7
  161. package/src/assets/icons/magnify-plus.svg +3 -7
  162. package/src/assets/icons/voice.svg +1 -0
  163. package/src/css/BookReader.scss +1 -5
  164. package/src/css/_BRBookmarks.scss +1 -1
  165. package/src/css/_BRComponent.scss +1 -1
  166. package/src/css/_BRmain.scss +16 -0
  167. package/src/css/_BRnav.scss +11 -38
  168. package/src/css/_BRpages.scss +149 -40
  169. package/src/css/_BRsearch.scss +67 -21
  170. package/src/css/_TextSelection.scss +87 -27
  171. package/src/css/_colorbox.scss +2 -2
  172. package/src/css/_controls.scss +20 -7
  173. package/src/css/_icons.scss +1 -1
  174. package/src/ia-bookreader/ia-bookreader.js +224 -0
  175. package/src/plugins/plugin.archive_analytics.js +3 -3
  176. package/src/plugins/plugin.autoplay.js +5 -11
  177. package/src/plugins/plugin.chapters.js +211 -186
  178. package/src/plugins/plugin.resume.js +3 -3
  179. package/src/plugins/plugin.text_selection.js +464 -134
  180. package/src/plugins/plugin.vendor-fullscreen.js +4 -4
  181. package/src/plugins/search/plugin.search.js +142 -125
  182. package/src/plugins/search/utils.js +43 -0
  183. package/src/plugins/search/view.js +33 -58
  184. package/src/plugins/tts/AbstractTTSEngine.js +46 -37
  185. package/src/plugins/tts/FestivalTTSEngine.js +13 -14
  186. package/src/plugins/tts/PageChunk.js +15 -21
  187. package/src/plugins/tts/PageChunkIterator.js +8 -12
  188. package/src/plugins/tts/WebTTSEngine.js +87 -71
  189. package/src/plugins/tts/plugin.tts.js +95 -126
  190. package/src/plugins/tts/utils.js +0 -25
  191. package/src/plugins/url/UrlPlugin.js +193 -0
  192. package/src/plugins/{plugin.url.js → url/plugin.url.js} +45 -16
  193. package/src/util/browserSniffing.js +22 -0
  194. package/src/util/docCookies.js +21 -2
  195. package/tests/e2e/README.md +37 -0
  196. package/tests/e2e/autoplay.test.js +2 -2
  197. package/tests/e2e/base.test.js +8 -16
  198. package/tests/e2e/helpers/base.js +53 -48
  199. package/tests/e2e/helpers/debug.js +1 -1
  200. package/tests/e2e/helpers/params.js +17 -0
  201. package/tests/e2e/helpers/rightToLeft.js +8 -14
  202. package/tests/e2e/helpers/search.js +73 -0
  203. package/tests/e2e/models/Navigation.js +20 -37
  204. package/tests/e2e/rightToLeft.test.js +4 -5
  205. package/tests/e2e/viewmode.test.js +40 -33
  206. package/tests/jest/BookNavigator/book-navigator.test.js +658 -0
  207. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  208. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  209. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  210. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  211. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  212. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  213. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  214. package/tests/{karma → jest}/BookNavigator/search/search-results.test.js +109 -60
  215. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  216. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  217. package/tests/jest/BookNavigator/volumes/volumes-provider.test.js +80 -0
  218. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +74 -14
  219. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +193 -0
  220. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  221. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  222. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  223. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  224. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  225. package/tests/jest/BookReader/ModeSmoothZoom.test.js +218 -0
  226. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  227. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +10 -10
  228. package/tests/{BookReader → jest/BookReader}/PageContainer.test.js +88 -6
  229. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  230. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +2 -2
  231. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  232. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  233. package/tests/jest/BookReader/utils/SelectionObserver.test.js +43 -0
  234. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  235. package/tests/jest/BookReader/utils.test.js +229 -0
  236. package/tests/jest/BookReader.keyboard.test.js +190 -0
  237. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +9 -1
  238. package/tests/{BookReader.test.js → jest/BookReader.test.js} +26 -37
  239. package/tests/{plugins → jest/plugins}/plugin.archive_analytics.test.js +2 -2
  240. package/tests/{plugins → jest/plugins}/plugin.autoplay.test.js +4 -4
  241. package/tests/jest/plugins/plugin.chapters.test.js +145 -0
  242. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +2 -2
  243. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +3 -3
  244. package/tests/jest/plugins/plugin.text_selection.test.js +317 -0
  245. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  246. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +25 -47
  247. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +39 -6
  248. package/tests/jest/plugins/search/utils.js +25 -0
  249. package/tests/jest/plugins/search/utils.test.js +29 -0
  250. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +9 -9
  251. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  252. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  253. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  254. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +47 -1
  255. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  256. package/tests/jest/plugins/url/UrlPlugin.test.js +190 -0
  257. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +53 -14
  258. package/tests/jest/setup.js +3 -0
  259. package/tests/{util → jest/util}/browserSniffing.test.js +1 -1
  260. package/tests/jest/util/docCookies.test.js +24 -0
  261. package/tests/{util → jest/util}/strings.test.js +1 -1
  262. package/tests/{utils.js → jest/utils.js} +38 -0
  263. package/webpack.config.js +11 -6
  264. package/.babelrc +0 -12
  265. package/.dependabot/config.yml +0 -6
  266. package/.testcaferc.json +0 -5
  267. package/BookReader/bookreader-component-bundle.js +0 -1450
  268. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  269. package/BookReader/bookreader-component-bundle.js.map +0 -1
  270. package/BookReader/jquery-1.10.1.js +0 -2
  271. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  272. package/BookReader/plugins/plugin.menu_toggle.js +0 -2
  273. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  274. package/BookReader/plugins/plugin.mobile_nav.js +0 -2
  275. package/BookReader/plugins/plugin.mobile_nav.js.map +0 -1
  276. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  277. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  278. package/karma.conf.js +0 -23
  279. package/src/BookNavigator/BookModel.js +0 -14
  280. package/src/BookNavigator/BookNavigator.js +0 -446
  281. package/src/BookNavigator/assets/book-loader.js +0 -27
  282. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  283. package/src/BookNavigator/search/a-search-result.js +0 -55
  284. package/src/BookReader/DebugConsole.js +0 -54
  285. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  286. package/src/ItemNavigator/ItemNavigator.js +0 -376
  287. package/src/ItemNavigator/providers/sharing.js +0 -29
  288. package/src/css/_MobileNav.scss +0 -194
  289. package/src/dragscrollable-br.js +0 -261
  290. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  291. package/src/plugins/plugin.mobile_nav.js +0 -287
  292. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  293. package/tests/BookReader/DebugConsole.test.js +0 -25
  294. package/tests/BookReader/Mode1Up.test.js +0 -164
  295. package/tests/BookReader/Mode2Up.test.js +0 -247
  296. package/tests/BookReader/utils.test.js +0 -109
  297. package/tests/e2e/helpers/desktopSearch.js +0 -72
  298. package/tests/e2e/helpers/mobileSearch.js +0 -85
  299. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  300. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  301. package/tests/karma/BookNavigator/search/search-provider.test.js +0 -23
  302. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  303. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  304. package/tests/plugins/plugin.chapters.test.js +0 -130
  305. package/tests/plugins/plugin.mobile_nav.test.js +0 -66
  306. package/tests/plugins/plugin.text_selection.test.js +0 -203
  307. package/tests/util/docCookies.test.js +0 -15
@@ -0,0 +1,586 @@
1
+ // eslint-disable-next-line no-unused-vars
2
+ import { SharedResizeObserver } from '@internetarchive/shared-resize-observer';
3
+ // eslint-disable-next-line no-unused-vars
4
+ import { ModalManager } from '@internetarchive/modal-manager';
5
+ import { css, html, LitElement, nothing } from 'lit';
6
+ import SearchProvider from './search/search-provider.js';
7
+ import DownloadProvider from './downloads/downloads-provider.js';
8
+ import VisualAdjustmentProvider from './visual-adjustments/visual-adjustments-provider.js';
9
+ import BookmarksProvider from './bookmarks/bookmarks-provider.js';
10
+ import SharingProvider from './sharing.js';
11
+ import VolumesProvider from './volumes/volumes-provider.js';
12
+ import iaLogo from './assets/ia-logo.js';
13
+
14
+ const events = {
15
+ menuUpdated: 'menuUpdated',
16
+ updateSideMenu: 'updateSideMenu',
17
+ PostInit: 'PostInit',
18
+ ViewportInFullScreen: 'ViewportInFullScreen',
19
+ };
20
+ export class BookNavigator extends LitElement {
21
+ static get properties() {
22
+ return {
23
+ itemMD: { type: Object },
24
+ bookReaderLoaded: { type: Boolean },
25
+ bookreader: { type: Object },
26
+ bookIsRestricted: { type: Boolean },
27
+ downloadableTypes: { type: Array },
28
+ isAdmin: { type: Boolean },
29
+ lendingInitialized: { type: Boolean },
30
+ lendingStatus: { type: Object },
31
+ menuProviders: { type: Object },
32
+ menuShortcuts: { type: Array },
33
+ signedIn: { type: Boolean },
34
+ loaded: { type: Boolean },
35
+ sharedObserver: { type: Object, attribute: false },
36
+ modal: { type: Object, attribute: false },
37
+ fullscreenBranding: { type: Object },
38
+ };
39
+ }
40
+
41
+ constructor() {
42
+ super();
43
+ this.itemMD = undefined;
44
+ this.loaded = false;
45
+ this.bookReaderCannotLoad = false;
46
+ this.bookReaderLoaded = false;
47
+ this.bookreader = null;
48
+ this.bookIsRestricted = false;
49
+ this.downloadableTypes = [];
50
+ this.isAdmin = false;
51
+ this.lendingInitialized = false;
52
+ this.lendingStatus = {};
53
+ this.menuProviders = {};
54
+ this.menuShortcuts = [];
55
+ this.signedIn = false;
56
+ /** @type {ModalManager} */
57
+ this.modal = undefined;
58
+ /** @type {SharedResizeObserver} */
59
+ this.sharedObserver = undefined;
60
+ this.fullscreenBranding = iaLogo;
61
+ // Untracked properties
62
+ this.sharedObserverHandler = undefined;
63
+ this.brWidth = 0;
64
+ this.brHeight = 0;
65
+ this.shortcutOrder = [
66
+ /**
67
+ * sets exit FS button (`this.fullscreenBranding1)
68
+ * when `br.options.enableFSLogoShortcut`
69
+ */
70
+ 'fullscreen',
71
+ 'volumes',
72
+ 'search',
73
+ 'bookmarks'
74
+ ];
75
+ }
76
+
77
+ disconnectedCallback() {
78
+ this.sharedObserver.removeObserver({
79
+ target: this.mainBRContainer,
80
+ handler: this.sharedObserverHandler
81
+ });
82
+ }
83
+
84
+ firstUpdated() {
85
+ this.bindEventListeners();
86
+ this.emitPostInit();
87
+ this.loaded = true;
88
+ }
89
+
90
+ updated(changed) {
91
+ if (!this.bookreader || !this.itemMD || !this.bookReaderLoaded) {
92
+ return;
93
+ }
94
+
95
+ const reload = changed.has('loaded') && this.loaded;
96
+ if (reload
97
+ || changed.has('itemMD')
98
+ || changed.has('bookreader')
99
+ || changed.has('signedIn')
100
+ || changed.has('isAdmin')
101
+ || changed.has('modal')) {
102
+ this.initializeBookSubmenus();
103
+ }
104
+
105
+ if (changed.has('sharedObserver') && this.bookreader) {
106
+ this.loadSharedObserver();
107
+ this.initializeBookSubmenus();
108
+ }
109
+
110
+ if (changed.has('downloadableTypes')) {
111
+ this.initializeBookSubmenus();
112
+ }
113
+ }
114
+
115
+ /**
116
+ * Global event emitter for when Book Navigator loads
117
+ */
118
+ emitPostInit() {
119
+ // emit global event when book nav has loaded with current bookreader selector
120
+ this.dispatchEvent(new CustomEvent(`BrBookNav:${events.PostInit}`, {
121
+ detail: { brSelector: this.bookreader?.el },
122
+ bubbles: true,
123
+ composed: true,
124
+ }));
125
+ }
126
+
127
+ /**
128
+ * @typedef {{
129
+ * baseHost: string,
130
+ * modal: ModalManager,
131
+ * sharedObserver: SharedResizeObserver,
132
+ * bookreader: BookReader,
133
+ * item: Item,
134
+ * signedIn: boolean,
135
+ * isAdmin: boolean,
136
+ * onProviderChange: (BookReader, object) => void,
137
+ * }} baseProviderConfig
138
+ *
139
+ * @return {baseProviderConfig}
140
+ */
141
+ get baseProviderConfig() {
142
+ return {
143
+ baseHost: this.baseHost,
144
+ modal: this.modal,
145
+ sharedObserver: this.sharedObserver,
146
+ bookreader: this.bookreader,
147
+ item: this.itemMD,
148
+ signedIn: this.signedIn,
149
+ isAdmin: this.isAdmin,
150
+ onProviderChange: () => {}
151
+ };
152
+ }
153
+
154
+ get isWideEnoughToOpenMenu() {
155
+ return this.brWidth >= 640;
156
+ }
157
+ /**
158
+ * Instantiates books submenus & their update callbacks
159
+ *
160
+ * NOTE: we are doing our best to scope bookreader's instance.
161
+ * If your submenu provider uses a bookreader instance to read, manually
162
+ * manipulate BookReader, please update the navigator's instance of it
163
+ * to keep it in sync.
164
+ */
165
+ initializeBookSubmenus() {
166
+ const providers = {
167
+ share: new SharingProvider(this.baseProviderConfig),
168
+ visualAdjustments: new VisualAdjustmentProvider({
169
+ ...this.baseProviderConfig,
170
+ /** Update menu contents */
171
+ onProviderChange: () => {
172
+ this.updateMenuContents();
173
+ },
174
+ }),
175
+ };
176
+
177
+ if (this.shouldShowDownloadsMenu()) {
178
+ providers.downloads = new DownloadProvider(this.baseProviderConfig);
179
+ }
180
+
181
+ if (this.bookreader.options.enableSearch) {
182
+ providers.search = new SearchProvider({
183
+ ...this.baseProviderConfig,
184
+ /**
185
+ * Search specific menu updates
186
+ * @param {BookReader} brInstance
187
+ * @param {{ searchCanceled: boolean }} searchUpdates
188
+ */
189
+ onProviderChange: (brInstance = null, searchUpdates = {}) => {
190
+ if (brInstance) {
191
+ /* refresh br instance reference */
192
+ this.bookreader = brInstance;
193
+ }
194
+
195
+ this.updateMenuContents();
196
+
197
+ if (searchUpdates.openMenu === false) {
198
+ return;
199
+ }
200
+
201
+ if (this.isWideEnoughToOpenMenu && !searchUpdates?.searchCanceled) {
202
+ /* open side search menu */
203
+ setTimeout(() => {
204
+ this.updateSideMenu('search', 'open');
205
+ }, 0);
206
+ }
207
+ },
208
+ });
209
+ }
210
+
211
+ if (this.bookreader.options.enableBookmarks) {
212
+ providers.bookmarks = new BookmarksProvider({
213
+ ...this.baseProviderConfig,
214
+ onProviderChange: (bookmarks) => {
215
+ const method = Object.keys(bookmarks).length ? 'add' : 'remove';
216
+ this[`${method}MenuShortcut`]('bookmarks');
217
+ this.updateMenuContents();
218
+ }
219
+ });
220
+ }
221
+
222
+ // add shortcut for volumes if multipleBooksList exists
223
+ if (this.bookreader.options.enableMultipleBooks) {
224
+ providers.volumes = new VolumesProvider({
225
+ ...this.baseProviderConfig,
226
+ onProviderChange: (brInstance = null, volumesUpdates = {}) => {
227
+ if (brInstance) {
228
+ /* refresh br instance reference */
229
+ this.bookreader = brInstance;
230
+ }
231
+ this.updateMenuContents();
232
+ if (this.isWideEnoughToOpenMenu) {
233
+ /* open side search menu */
234
+ setTimeout(() => {
235
+ this.updateSideMenu('volumes', 'open');
236
+ });
237
+ }
238
+ }
239
+ });
240
+ }
241
+
242
+ Object.assign(this.menuProviders, providers);
243
+ this.addMenuShortcut('search');
244
+ this.addMenuShortcut('volumes');
245
+ this.updateMenuContents();
246
+ }
247
+
248
+ /** gets element that houses the bookreader in light dom */
249
+ get mainBRContainer() {
250
+ return document.querySelector(this.bookreader?.el);
251
+ }
252
+
253
+ /** Fullscreen Shortcut */
254
+ addFullscreenShortcut() {
255
+ const closeFS = {
256
+ icon: this.fullscreenShortcut,
257
+ id: 'fullscreen',
258
+ };
259
+ this.menuShortcuts.push(closeFS);
260
+ this.sortMenuShortcuts();
261
+ this.emitMenuShortcutsUpdated();
262
+ }
263
+
264
+ deleteFullscreenShortcut() {
265
+ const updatedShortcuts = this.menuShortcuts.filter(({ id }) => {
266
+ return id !== 'fullscreen';
267
+ });
268
+ this.menuShortcuts = updatedShortcuts;
269
+ this.sortMenuShortcuts();
270
+ this.emitMenuShortcutsUpdated();
271
+ }
272
+
273
+ closeFullscreen() {
274
+ this.bookreader.exitFullScreen();
275
+ }
276
+
277
+ get fullscreenShortcut() {
278
+ return html`
279
+ <button
280
+ @click=${() => this.closeFullscreen()}
281
+ title="Exit fullscreen view"
282
+ >${this.fullscreenBranding}</button>
283
+ `;
284
+ }
285
+ /** End Fullscreen Shortcut */
286
+
287
+ /**
288
+ * Open side menu
289
+ * @param {string} menuId
290
+ * @param {('open'|'close'|'toggle')} action
291
+ */
292
+ updateSideMenu(menuId = '', action = 'open') {
293
+ if (!menuId) {
294
+ return;
295
+ }
296
+ const event = new CustomEvent(
297
+ events.updateSideMenu, {
298
+ detail: { menuId, action },
299
+ },
300
+ );
301
+ this.dispatchEvent(event);
302
+ }
303
+
304
+ /**
305
+ * Sets order of menu and emits custom event when done
306
+ */
307
+ updateMenuContents() {
308
+ const {
309
+ search, downloads, visualAdjustments, share, bookmarks, volumes, chapters
310
+ } = this.menuProviders;
311
+ const availableMenus = [volumes, chapters, search, bookmarks, visualAdjustments, share].filter((menu) => !!menu);
312
+
313
+ if (this.shouldShowDownloadsMenu()) {
314
+ downloads?.update(this.downloadableTypes);
315
+ availableMenus.splice(1, 0, downloads);
316
+ }
317
+
318
+ const event = new CustomEvent(
319
+ events.menuUpdated, {
320
+ detail: availableMenus,
321
+ },
322
+ );
323
+ this.dispatchEvent(event);
324
+ }
325
+
326
+ /**
327
+ * Confirms if we should show the downloads menu
328
+ * @returns {bool}
329
+ */
330
+ shouldShowDownloadsMenu() {
331
+ if (!this.downloadableTypes.length) { return false; }
332
+ if (this.bookIsRestricted === false) { return true; }
333
+ if (this.isAdmin) { return true; }
334
+ const { user_loan_record = {} } = this.lendingStatus;
335
+ const hasNoLoanRecord = Array.isArray(user_loan_record); /* (bc PHP assoc. arrays) */
336
+
337
+ if (hasNoLoanRecord) { return false; }
338
+
339
+ const hasValidLoan = user_loan_record.type && (user_loan_record.type !== 'SESSION_LOAN');
340
+ return hasValidLoan;
341
+ }
342
+
343
+ /**
344
+ * Adds a provider object to the menuShortcuts array property if it isn't
345
+ * already added. menuShortcuts are then sorted by shortcutOrder and
346
+ * a menuShortcutsUpdated event is emitted.
347
+ *
348
+ * @param {string} menuId - a string matching the id property of a provider
349
+ */
350
+ addMenuShortcut(menuId) {
351
+ if (this.menuShortcuts.find((m) => m.id === menuId)) {
352
+ // menu is already there
353
+ return;
354
+ }
355
+
356
+ if (!this.menuProviders[menuId]) {
357
+ // no provider for this menu
358
+ return;
359
+ }
360
+
361
+ this.menuShortcuts.push(this.menuProviders[menuId]);
362
+
363
+ this.sortMenuShortcuts();
364
+ this.emitMenuShortcutsUpdated();
365
+ }
366
+
367
+ /**
368
+ * Removes a provider object from the menuShortcuts array and emits a
369
+ * menuShortcutsUpdated event.
370
+ *
371
+ * @param {string} menuId - a string matching the id property of a provider
372
+ */
373
+ removeMenuShortcut(menuId) {
374
+ this.menuShortcuts = this.menuShortcuts.filter((m) => m.id !== menuId);
375
+ this.emitMenuShortcutsUpdated();
376
+ }
377
+
378
+ /**
379
+ * Sorts the menuShortcuts property by comparing each provider's id to
380
+ * the id in each iteration over the shortcutOrder array.
381
+ */
382
+ sortMenuShortcuts() {
383
+ this.menuShortcuts = this.shortcutOrder.reduce((shortcuts, id) => {
384
+ const menu = this.menuShortcuts.find((m) => m.id === id);
385
+ if (menu) { shortcuts.push(menu); }
386
+ return shortcuts;
387
+ }, []);
388
+ }
389
+
390
+ emitMenuShortcutsUpdated() {
391
+ const event = new CustomEvent('menuShortcutsUpdated', {
392
+ detail: this.menuShortcuts,
393
+ });
394
+ this.dispatchEvent(event);
395
+ }
396
+
397
+ emitLoadingStatusUpdate(loaded) {
398
+ const event = new CustomEvent('loadingStateUpdated', {
399
+ detail: { loaded },
400
+ });
401
+ this.dispatchEvent(event);
402
+ }
403
+
404
+ /**
405
+ * Core bookreader event handler registry
406
+ *
407
+ * NOTE: we are trying to keep bookreader's instance in scope
408
+ * Please update Book Navigator's instance reference of it to keep it current
409
+ */
410
+ bindEventListeners() {
411
+ window.addEventListener('BookReader:PostInit', (e) => {
412
+ this.bookreader = e.detail.props;
413
+ this.bookreader.shell = this;
414
+ this.bookReaderLoaded = true;
415
+ this.bookReaderCannotLoad = false;
416
+ this.emitLoadingStatusUpdate(true);
417
+ this.loadSharedObserver();
418
+ setTimeout(() => {
419
+ this.bookreader.resize();
420
+ }, 0);
421
+ });
422
+ window.addEventListener('BookReader:fullscreenToggled', (event) => {
423
+ const { detail: { props: brInstance = null } } = event;
424
+ if (brInstance) {
425
+ this.bookreader = brInstance;
426
+ }
427
+ this.manageFullScreenBehavior();
428
+ }, { passive: true });
429
+ window.addEventListener('BookReader:ToggleSearchMenu', (event) => {
430
+ this.dispatchEvent(new CustomEvent(events.updateSideMenu, {
431
+ detail: { menuId: 'search', action: 'toggle' },
432
+ }));
433
+ });
434
+ window.addEventListener('LendingFlow:PostInit', ({ detail }) => {
435
+ const {
436
+ downloadTypesAvailable, lendingStatus, isAdmin, previewType,
437
+ } = detail;
438
+ this.lendingInitialized = true;
439
+ this.downloadableTypes = downloadTypesAvailable;
440
+ this.lendingStatus = lendingStatus;
441
+ this.isAdmin = isAdmin;
442
+ this.bookReaderCannotLoad = previewType === 'singlePagePreview';
443
+ this.emitLoadingStatusUpdate(true);
444
+ });
445
+ window.addEventListener('BRJSIA:PostInit', ({ detail }) => {
446
+ const { isRestricted, downloadURLs } = detail;
447
+ this.bookReaderLoaded = true;
448
+ this.downloadableTypes = downloadURLs;
449
+ this.bookIsRestricted = isRestricted;
450
+ });
451
+ window.addEventListener('contextmenu', (e) => this.manageContextMenuVisibility(e), { capture: true });
452
+ }
453
+
454
+ /** Display an element's context menu */
455
+ manageContextMenuVisibility(e) {
456
+ window.archive_analytics?.send_event(
457
+ 'BookReader',
458
+ `contextmenu-${this.bookIsRestricted ? 'restricted' : 'unrestricted'}`,
459
+ e.target?.classList?.value
460
+ );
461
+ if (!this.bookIsRestricted) {
462
+ return;
463
+ }
464
+
465
+ const imagePane = e.target.classList.value.match(/BRscreen|BRpageimage/g);
466
+ if (!imagePane) {
467
+ return;
468
+ }
469
+
470
+ e.preventDefault();
471
+ return false;
472
+ }
473
+
474
+ loadSharedObserver() {
475
+ this.sharedObserverHandler = { handleResize: this.handleResize.bind(this) };
476
+ this.sharedObserver?.addObserver({
477
+ target: this.mainBRContainer,
478
+ handler: this.sharedObserverHandler
479
+ });
480
+ }
481
+
482
+ /**
483
+ * Uses resize observer to fire BookReader's `resize` functionality
484
+ * We do not want to trigger resize IF:
485
+ * - book animation is happening
486
+ * - book is in fullscreen (fullscreen is handled separately)
487
+ *
488
+ * @param { target: HTMLElement, contentRect: DOMRectReadOnly } entry
489
+ */
490
+ handleResize({ contentRect, target }) {
491
+ const startBrWidth = this.brWidth;
492
+ const startBrHeight = this.brHeight;
493
+ const { animating } = this.bookreader;
494
+
495
+ if (target === this.mainBRContainer) {
496
+ this.brWidth = contentRect.width;
497
+ this.brHeight = contentRect.height;
498
+ }
499
+
500
+ if (!startBrWidth && this.brWidth) {
501
+ // loading up, let's update side menus
502
+ this.initializeBookSubmenus();
503
+ }
504
+
505
+ const widthChange = startBrWidth !== this.brWidth;
506
+ const heightChange = startBrHeight !== this.brHeight;
507
+
508
+ if (!animating && (widthChange || heightChange)) {
509
+ this.bookreader?.resize();
510
+ }
511
+ }
512
+
513
+ /**
514
+ * Manages Fullscreen behavior
515
+ * This makes sure that controls are _always_ in view
516
+ * We need this to accommodate LOAN BAR during fullscreen
517
+ */
518
+ manageFullScreenBehavior() {
519
+ this.emitFullScreenState();
520
+
521
+ if (!this.bookreader.options.enableFSLogoShortcut) {
522
+ return;
523
+ }
524
+
525
+ const isFullScreen = this.bookreader.isFullscreen();
526
+ if (isFullScreen) {
527
+ this.addFullscreenShortcut();
528
+ } else {
529
+ this.deleteFullscreenShortcut();
530
+ }
531
+ }
532
+
533
+ /**
534
+ * Relays fullscreen toggle events
535
+ */
536
+ emitFullScreenState() {
537
+ const isFullScreen = this.bookreader.isFullscreen();
538
+ const event = new CustomEvent('ViewportInFullScreen', {
539
+ detail: { isFullScreen },
540
+ });
541
+ this.dispatchEvent(event);
542
+ }
543
+
544
+ get itemImage() {
545
+ const identifier = this.itemMD?.metadata.identifier;
546
+ const url = `https://${this.baseHost}/services/img/${identifier}`;
547
+ return html`<img class="cover-img" src=${url} alt="cover image for ${identifier}">`;
548
+ }
549
+
550
+ get placeholder() {
551
+ return html`<div class="placeholder">${this.itemImage}</div>`;
552
+ }
553
+
554
+ render() {
555
+ return html`<div id="book-navigator__root">
556
+ ${this.bookReaderCannotLoad ? this.placeholder : nothing}
557
+ ${!this.bookReaderCannotLoad ? html`<slot name="main"></slot>` : nothing}
558
+ </div>
559
+ `;
560
+ }
561
+
562
+ static get styles() {
563
+ return css`
564
+ :host,
565
+ #book-navigator__root,
566
+ slot,
567
+ slot > * {
568
+ display: block;
569
+ height: inherit;
570
+ width: inherit;
571
+ }
572
+ .placeholder {
573
+ display: flex;
574
+ align-items: center;
575
+ justify-content: center;
576
+ flex-direction: column;
577
+ margin: 5%;
578
+ }
579
+ .cover-img {
580
+ max-height: 300px;
581
+ }
582
+ `;
583
+ }
584
+ }
585
+
586
+ customElements.define('book-navigator', BookNavigator);
@@ -1,4 +1,4 @@
1
- import { LitElement, html, css } from 'lit-element';
1
+ import { LitElement, html, css } from 'lit';
2
2
 
3
3
  export default class BookmarkButton extends LitElement {
4
4
  static get styles() {
@@ -12,7 +12,7 @@ export default class BookmarkButton extends LitElement {
12
12
  height: 4rem;
13
13
  width: 4rem;
14
14
  background: transparent;
15
- cursor: url('/images/bookreader/bookmark-add.png'), pointer;
15
+ cursor: url("data:image/svg+xml,%3Csvg xmlns='http://www.w3.org/2000/svg' height='24' viewBox='0 0 16 24' width='16'%3E%3Cg fill='%23333' fill-rule='evenodd'%3E%3Cpath d='m15 0c.5522847 0 1 .44771525 1 1v23l-8-5.4545455-8 5.4545455v-23c0-.55228475.44771525-1 1-1zm-2 2h-10c-.51283584 0-.93550716.38604019-.99327227.88337887l-.00672773.11662113v18l6-4.3181818 6 4.3181818v-18c0-.51283584-.3860402-.93550716-.8833789-.99327227z'/%3E%3Cpath d='m8.75 6v2.25h2.25v1.5h-2.25v2.25h-1.5v-2.25h-2.25v-1.5h2.25v-2.25z' fill-rule='nonzero'/%3E%3C/g%3E%3C/svg%3E"), pointer;
16
16
  position: relative;
17
17
  }
18
18
  button > * {
@@ -40,6 +40,7 @@ export default class BookmarkButton extends LitElement {
40
40
  constructor() {
41
41
  super();
42
42
  this.state = 'hollow';
43
+ this.side = undefined;
43
44
  }
44
45
 
45
46
  handleClick(e) {
@@ -1,6 +1,5 @@
1
- import { nothing } from 'lit-html';
2
- import { repeat } from 'lit-html/directives/repeat.js';
3
- import { css, html, LitElement } from 'lit-element';
1
+ import { repeat } from 'lit/directives/repeat.js';
2
+ import { css, html, LitElement, nothing } from 'lit';
4
3
  import bookmarkColorsCSS from '../assets/bookmark-colors.js';
5
4
  import buttonCSS from '../assets/button-base.js';
6
5
 
@@ -208,7 +207,7 @@ export class IABookmarkEdit extends LitElement {
208
207
  grid-gap: 0 1rem;
209
208
  justify-items: stretch;
210
209
  }
211
- `
210
+ `;
212
211
  return [buttonCSS, bookmarkColorsCSS, bookmarkEditCSS];
213
212
  }
214
213
  }
@@ -1,6 +1,5 @@
1
- import { nothing } from 'lit-html';
2
- import { repeat } from 'lit-html/directives/repeat.js';
3
- import { css, html, LitElement } from 'lit-element';
1
+ import { repeat } from 'lit/directives/repeat.js';
2
+ import { css, html, LitElement, nothing } from 'lit';
4
3
  import './bookmark-edit.js';
5
4
  import '@internetarchive/icon-edit-pencil/icon-edit-pencil.js';
6
5
  import bookmarkColorsCSS from '../assets/bookmark-colors.js';
@@ -1,20 +1,15 @@
1
- import { LitElement, html, css } from 'lit-element';
1
+ import { LitElement, html } from 'lit';
2
2
  import buttonStyles from '../assets/button-base.js';
3
3
 
4
4
  class BookmarksLogin extends LitElement {
5
5
  static get properties() {
6
6
  return {
7
7
  url: { type: String }
8
- }
8
+ };
9
9
  }
10
10
 
11
11
  static get styles() {
12
- const mainCss = css`
13
- a {
14
- text-decoration: none;
15
- }
16
- `;
17
- return [buttonStyles, mainCss];
12
+ return buttonStyles;
18
13
  }
19
14
 
20
15
  constructor() {