@internetarchive/bookreader 5.0.0-7 → 5.0.0-70

Sign up to get free protection for your applications and to get access to all the features.
Files changed (313) 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 +1509 -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 +283 -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 +71 -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/assets/icon_sort_asc.js +5 -0
  125. package/src/BookNavigator/assets/icon_sort_desc.js +5 -0
  126. package/src/BookNavigator/assets/icon_sort_neutral.js +5 -0
  127. package/src/BookNavigator/assets/icon_volumes.js +11 -0
  128. package/src/BookNavigator/book-navigator.js +586 -0
  129. package/src/BookNavigator/bookmarks/bookmark-button.js +3 -2
  130. package/src/BookNavigator/bookmarks/bookmark-edit.js +3 -4
  131. package/src/BookNavigator/bookmarks/bookmarks-list.js +2 -3
  132. package/src/BookNavigator/bookmarks/bookmarks-loginCTA.js +3 -8
  133. package/src/BookNavigator/bookmarks/bookmarks-provider.js +27 -17
  134. package/src/BookNavigator/bookmarks/ia-bookmarks.js +116 -67
  135. package/src/BookNavigator/delete-modal-actions.js +1 -1
  136. package/src/BookNavigator/downloads/downloads-provider.js +36 -21
  137. package/src/BookNavigator/downloads/downloads.js +41 -25
  138. package/src/BookNavigator/search/search-provider.js +49 -27
  139. package/src/BookNavigator/search/search-results.js +23 -9
  140. package/src/BookNavigator/sharing.js +27 -0
  141. package/src/BookNavigator/visual-adjustments/visual-adjustments-provider.js +11 -10
  142. package/src/BookNavigator/visual-adjustments/visual-adjustments.js +3 -3
  143. package/src/BookNavigator/volumes/volumes-provider.js +111 -0
  144. package/src/BookNavigator/volumes/volumes.js +188 -0
  145. package/src/BookReader/BookModel.js +64 -34
  146. package/src/BookReader/DragScrollable.js +233 -0
  147. package/src/BookReader/Mode1Up.js +56 -351
  148. package/src/BookReader/Mode1UpLit.js +388 -0
  149. package/src/BookReader/Mode2Up.js +73 -1318
  150. package/src/BookReader/Mode2UpLit.js +776 -0
  151. package/src/BookReader/ModeCoordinateSpace.js +29 -0
  152. package/src/BookReader/ModeSmoothZoom.js +312 -0
  153. package/src/BookReader/ModeThumb.js +18 -12
  154. package/src/BookReader/Navbar/Navbar.js +12 -38
  155. package/src/BookReader/PageContainer.js +81 -6
  156. package/src/BookReader/ReduceSet.js +1 -1
  157. package/src/BookReader/Toolbar/Toolbar.js +10 -37
  158. package/src/BookReader/events.js +2 -3
  159. package/src/BookReader/options.js +24 -2
  160. package/src/BookReader/utils/HTMLDimensionsCacher.js +44 -0
  161. package/src/BookReader/utils/ScrollClassAdder.js +31 -0
  162. package/src/BookReader/utils/SelectionObserver.js +43 -0
  163. package/src/BookReader/utils.js +118 -13
  164. package/src/BookReader.js +427 -1061
  165. package/src/assets/icons/magnify-minus.svg +3 -7
  166. package/src/assets/icons/magnify-plus.svg +3 -7
  167. package/src/assets/icons/voice.svg +1 -0
  168. package/src/css/BookReader.scss +1 -5
  169. package/src/css/_BRBookmarks.scss +1 -1
  170. package/src/css/_BRComponent.scss +1 -1
  171. package/src/css/_BRmain.scss +16 -0
  172. package/src/css/_BRnav.scss +11 -38
  173. package/src/css/_BRpages.scss +149 -40
  174. package/src/css/_BRsearch.scss +67 -21
  175. package/src/css/_TextSelection.scss +87 -27
  176. package/src/css/_colorbox.scss +2 -2
  177. package/src/css/_controls.scss +20 -7
  178. package/src/css/_icons.scss +1 -1
  179. package/src/ia-bookreader/ia-bookreader.js +224 -0
  180. package/src/plugins/plugin.archive_analytics.js +3 -3
  181. package/src/plugins/plugin.autoplay.js +5 -11
  182. package/src/plugins/plugin.chapters.js +211 -186
  183. package/src/plugins/plugin.resume.js +3 -3
  184. package/src/plugins/plugin.text_selection.js +464 -134
  185. package/src/plugins/plugin.vendor-fullscreen.js +4 -4
  186. package/src/plugins/search/plugin.search.js +142 -125
  187. package/src/plugins/search/utils.js +43 -0
  188. package/src/plugins/search/view.js +33 -58
  189. package/src/plugins/tts/AbstractTTSEngine.js +68 -40
  190. package/src/plugins/tts/FestivalTTSEngine.js +13 -14
  191. package/src/plugins/tts/PageChunk.js +15 -21
  192. package/src/plugins/tts/PageChunkIterator.js +8 -12
  193. package/src/plugins/tts/WebTTSEngine.js +87 -71
  194. package/src/plugins/tts/plugin.tts.js +95 -126
  195. package/src/plugins/tts/utils.js +0 -25
  196. package/src/plugins/url/UrlPlugin.js +191 -0
  197. package/src/plugins/{plugin.url.js → url/plugin.url.js} +45 -16
  198. package/src/util/browserSniffing.js +22 -0
  199. package/src/util/docCookies.js +21 -2
  200. package/tests/e2e/README.md +37 -0
  201. package/tests/e2e/autoplay.test.js +2 -2
  202. package/tests/e2e/base.test.js +8 -16
  203. package/tests/e2e/helpers/base.js +53 -48
  204. package/tests/e2e/helpers/debug.js +1 -1
  205. package/tests/e2e/helpers/params.js +17 -0
  206. package/tests/e2e/helpers/rightToLeft.js +8 -14
  207. package/tests/e2e/helpers/search.js +73 -0
  208. package/tests/e2e/models/Navigation.js +20 -37
  209. package/tests/e2e/rightToLeft.test.js +4 -5
  210. package/tests/e2e/viewmode.test.js +40 -33
  211. package/tests/jest/BookNavigator/book-navigator.test.js +658 -0
  212. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  213. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  214. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  215. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  216. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  217. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  218. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  219. package/tests/{karma → jest}/BookNavigator/search/search-results.test.js +109 -60
  220. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  221. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  222. package/tests/jest/BookNavigator/volumes/volumes-provider.test.js +184 -0
  223. package/tests/jest/BookNavigator/volumes/volumes.test.js +97 -0
  224. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +74 -14
  225. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +193 -0
  226. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  227. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  228. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  229. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  230. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  231. package/tests/jest/BookReader/ModeSmoothZoom.test.js +218 -0
  232. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  233. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +10 -10
  234. package/tests/{BookReader → jest/BookReader}/PageContainer.test.js +88 -6
  235. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  236. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +2 -2
  237. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  238. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  239. package/tests/jest/BookReader/utils/SelectionObserver.test.js +43 -0
  240. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  241. package/tests/jest/BookReader/utils.test.js +229 -0
  242. package/tests/jest/BookReader.keyboard.test.js +190 -0
  243. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +9 -1
  244. package/tests/{BookReader.test.js → jest/BookReader.test.js} +26 -37
  245. package/tests/{plugins → jest/plugins}/plugin.archive_analytics.test.js +2 -2
  246. package/tests/{plugins → jest/plugins}/plugin.autoplay.test.js +4 -4
  247. package/tests/jest/plugins/plugin.chapters.test.js +145 -0
  248. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +2 -2
  249. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +3 -3
  250. package/tests/jest/plugins/plugin.text_selection.test.js +317 -0
  251. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  252. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +25 -47
  253. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +39 -6
  254. package/tests/jest/plugins/search/utils.js +25 -0
  255. package/tests/jest/plugins/search/utils.test.js +29 -0
  256. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +29 -9
  257. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  258. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  259. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  260. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +47 -1
  261. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  262. package/tests/jest/plugins/url/UrlPlugin.test.js +198 -0
  263. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +53 -14
  264. package/tests/jest/setup.js +3 -0
  265. package/tests/{util → jest/util}/browserSniffing.test.js +1 -1
  266. package/tests/jest/util/docCookies.test.js +24 -0
  267. package/tests/{util → jest/util}/strings.test.js +1 -1
  268. package/tests/{utils.js → jest/utils.js} +38 -0
  269. package/webpack.config.js +11 -6
  270. package/.babelrc +0 -12
  271. package/.dependabot/config.yml +0 -6
  272. package/.testcaferc.json +0 -5
  273. package/BookReader/bookreader-component-bundle.js +0 -1450
  274. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  275. package/BookReader/bookreader-component-bundle.js.map +0 -1
  276. package/BookReader/jquery-1.10.1.js +0 -2
  277. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  278. package/BookReader/plugins/plugin.menu_toggle.js +0 -2
  279. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  280. package/BookReader/plugins/plugin.mobile_nav.js +0 -2
  281. package/BookReader/plugins/plugin.mobile_nav.js.map +0 -1
  282. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  283. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  284. package/karma.conf.js +0 -23
  285. package/src/BookNavigator/BookModel.js +0 -14
  286. package/src/BookNavigator/BookNavigator.js +0 -446
  287. package/src/BookNavigator/assets/book-loader.js +0 -27
  288. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  289. package/src/BookNavigator/search/a-search-result.js +0 -55
  290. package/src/BookReader/DebugConsole.js +0 -54
  291. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  292. package/src/ItemNavigator/ItemNavigator.js +0 -376
  293. package/src/ItemNavigator/providers/sharing.js +0 -29
  294. package/src/css/_MobileNav.scss +0 -194
  295. package/src/dragscrollable-br.js +0 -261
  296. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  297. package/src/plugins/plugin.mobile_nav.js +0 -287
  298. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  299. package/tests/BookReader/DebugConsole.test.js +0 -25
  300. package/tests/BookReader/Mode1Up.test.js +0 -164
  301. package/tests/BookReader/Mode2Up.test.js +0 -247
  302. package/tests/BookReader/utils.test.js +0 -109
  303. package/tests/e2e/helpers/desktopSearch.js +0 -72
  304. package/tests/e2e/helpers/mobileSearch.js +0 -85
  305. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  306. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  307. package/tests/karma/BookNavigator/search/search-provider.test.js +0 -23
  308. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  309. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  310. package/tests/plugins/plugin.chapters.test.js +0 -130
  311. package/tests/plugins/plugin.mobile_nav.test.js +0 -66
  312. package/tests/plugins/plugin.text_selection.test.js +0 -203
  313. 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() {