@internetarchive/bookreader 5.0.0-7-multiple-files → 5.0.0-70

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 (319) hide show
  1. package/.eslintrc.js +17 -15
  2. package/.github/workflows/node.js.yml +78 -6
  3. package/.github/workflows/npm-publish.yml +6 -20
  4. package/.testcaferc.js +10 -0
  5. package/BookReader/BookReader.css +416 -1365
  6. package/BookReader/BookReader.js +2 -21564
  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 -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 +22 -301
  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 -74
  67. package/BookReader/plugins/plugin.iframe.js.map +1 -1
  68. package/BookReader/plugins/plugin.resume.js +1 -368
  69. package/BookReader/plugins/plugin.resume.js.map +1 -1
  70. package/BookReader/plugins/plugin.search.js +2 -1420
  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 -1080
  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 +2 -9193
  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 -269
  80. package/BookReader/plugins/plugin.url.js.map +1 -1
  81. package/BookReader/plugins/plugin.vendor-fullscreen.js +1 -379
  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 +289 -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 +9 -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_ascending.js → icon_sort_desc.js} +2 -2
  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 +80 -28
  139. package/src/BookNavigator/search/search-results.js +28 -25
  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 +88 -53
  144. package/src/BookNavigator/volumes/volumes.js +41 -14
  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 +446 -1062
  165. package/src/assets/icons/close-circle-dark.svg +1 -0
  166. package/src/assets/icons/magnify-minus.svg +3 -7
  167. package/src/assets/icons/magnify-plus.svg +3 -7
  168. package/src/assets/icons/voice.svg +1 -0
  169. package/src/css/BookReader.scss +1 -17
  170. package/src/css/_BRBookmarks.scss +1 -1
  171. package/src/css/_BRComponent.scss +1 -1
  172. package/src/css/_BRmain.scss +33 -24
  173. package/src/css/_BRnav.scss +11 -38
  174. package/src/css/_BRpages.scss +149 -40
  175. package/src/css/_BRsearch.scss +67 -226
  176. package/src/css/_TextSelection.scss +87 -27
  177. package/src/css/_colorbox.scss +2 -2
  178. package/src/css/_controls.scss +20 -7
  179. package/src/css/_icons.scss +7 -1
  180. package/src/ia-bookreader/ia-bookreader.js +224 -0
  181. package/src/plugins/plugin.archive_analytics.js +3 -3
  182. package/src/plugins/plugin.autoplay.js +5 -11
  183. package/src/plugins/plugin.chapters.js +211 -186
  184. package/src/plugins/plugin.resume.js +3 -3
  185. package/src/plugins/plugin.text_selection.js +464 -134
  186. package/src/plugins/plugin.vendor-fullscreen.js +4 -4
  187. package/src/plugins/search/plugin.search.js +175 -120
  188. package/src/plugins/search/utils.js +43 -0
  189. package/src/plugins/search/view.js +64 -202
  190. package/src/plugins/tts/AbstractTTSEngine.js +68 -40
  191. package/src/plugins/tts/FestivalTTSEngine.js +13 -14
  192. package/src/plugins/tts/PageChunk.js +15 -21
  193. package/src/plugins/tts/PageChunkIterator.js +8 -12
  194. package/src/plugins/tts/WebTTSEngine.js +87 -71
  195. package/src/plugins/tts/plugin.tts.js +95 -126
  196. package/src/plugins/tts/utils.js +0 -25
  197. package/src/plugins/url/UrlPlugin.js +191 -0
  198. package/src/plugins/{plugin.url.js → url/plugin.url.js} +45 -16
  199. package/src/util/browserSniffing.js +22 -0
  200. package/src/util/docCookies.js +21 -2
  201. package/tests/e2e/README.md +37 -0
  202. package/tests/e2e/autoplay.test.js +2 -2
  203. package/tests/e2e/base.test.js +8 -16
  204. package/tests/e2e/helpers/base.js +53 -48
  205. package/tests/e2e/helpers/debug.js +1 -1
  206. package/tests/e2e/helpers/params.js +17 -0
  207. package/tests/e2e/helpers/rightToLeft.js +8 -14
  208. package/tests/e2e/helpers/search.js +73 -0
  209. package/tests/e2e/models/Navigation.js +20 -37
  210. package/tests/e2e/rightToLeft.test.js +4 -5
  211. package/tests/e2e/viewmode.test.js +40 -33
  212. package/tests/jest/BookNavigator/book-navigator.test.js +658 -0
  213. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  214. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  215. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  216. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  217. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  218. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  219. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  220. package/tests/{karma/BookNavigator → jest/BookNavigator/search}/search-results.test.js +109 -60
  221. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  222. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  223. package/tests/jest/BookNavigator/volumes/volumes-provider.test.js +184 -0
  224. package/tests/jest/BookNavigator/volumes/volumes.test.js +97 -0
  225. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +74 -14
  226. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +193 -0
  227. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  228. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  229. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  230. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  231. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  232. package/tests/jest/BookReader/ModeSmoothZoom.test.js +218 -0
  233. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  234. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +10 -10
  235. package/tests/{BookReader → jest/BookReader}/PageContainer.test.js +88 -6
  236. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  237. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +2 -2
  238. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  239. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  240. package/tests/jest/BookReader/utils/SelectionObserver.test.js +43 -0
  241. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  242. package/tests/jest/BookReader/utils.test.js +229 -0
  243. package/tests/jest/BookReader.keyboard.test.js +190 -0
  244. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +9 -1
  245. package/tests/{BookReader.test.js → jest/BookReader.test.js} +26 -37
  246. package/tests/{plugins → jest/plugins}/plugin.archive_analytics.test.js +2 -2
  247. package/tests/{plugins → jest/plugins}/plugin.autoplay.test.js +4 -4
  248. package/tests/jest/plugins/plugin.chapters.test.js +145 -0
  249. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +2 -2
  250. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +3 -3
  251. package/tests/jest/plugins/plugin.text_selection.test.js +317 -0
  252. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  253. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +26 -47
  254. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +39 -6
  255. package/tests/jest/plugins/search/utils.js +25 -0
  256. package/tests/jest/plugins/search/utils.test.js +29 -0
  257. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +29 -9
  258. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  259. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  260. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  261. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +47 -1
  262. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  263. package/tests/jest/plugins/url/UrlPlugin.test.js +198 -0
  264. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +53 -14
  265. package/tests/jest/setup.js +3 -0
  266. package/tests/{util → jest/util}/browserSniffing.test.js +1 -1
  267. package/tests/jest/util/docCookies.test.js +24 -0
  268. package/tests/{util → jest/util}/strings.test.js +1 -1
  269. package/tests/{utils.js → jest/utils.js} +38 -0
  270. package/webpack.config.js +11 -6
  271. package/.babelrc +0 -12
  272. package/.dependabot/config.yml +0 -6
  273. package/.testcaferc.json +0 -5
  274. package/BookReader/bookreader-component-bundle.js +0 -14311
  275. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  276. package/BookReader/bookreader-component-bundle.js.map +0 -1
  277. package/BookReader/icons/sort-ascending.svg +0 -1
  278. package/BookReader/icons/sort-descending.svg +0 -1
  279. package/BookReader/jquery-1.10.1.js +0 -108
  280. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  281. package/BookReader/plugins/plugin.menu_toggle.js +0 -369
  282. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  283. package/BookReader/plugins/plugin.mobile_nav.js +0 -335
  284. package/BookReader/plugins/plugin.mobile_nav.js.map +0 -1
  285. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  286. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  287. package/karma.conf.js +0 -23
  288. package/src/BookNavigator/BookModel.js +0 -14
  289. package/src/BookNavigator/BookNavigator.js +0 -452
  290. package/src/BookNavigator/assets/book-loader.js +0 -27
  291. package/src/BookNavigator/assets/icon_sort_descending.js +0 -5
  292. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  293. package/src/BookNavigator/search/a-search-result.js +0 -55
  294. package/src/BookReader/DebugConsole.js +0 -54
  295. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  296. package/src/ItemNavigator/ItemNavigator.js +0 -372
  297. package/src/ItemNavigator/providers/sharing.js +0 -29
  298. package/src/assets/icons/sort-ascending.svg +0 -1
  299. package/src/assets/icons/sort-descending.svg +0 -1
  300. package/src/css/_MobileNav.scss +0 -194
  301. package/src/dragscrollable-br.js +0 -261
  302. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  303. package/src/plugins/plugin.mobile_nav.js +0 -287
  304. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  305. package/tests/BookReader/DebugConsole.test.js +0 -25
  306. package/tests/BookReader/Mode1Up.test.js +0 -164
  307. package/tests/BookReader/Mode2Up.test.js +0 -247
  308. package/tests/BookReader/utils.test.js +0 -109
  309. package/tests/e2e/helpers/desktopSearch.js +0 -72
  310. package/tests/e2e/helpers/mobileSearch.js +0 -85
  311. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  312. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  313. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  314. package/tests/karma/BookNavigator/volumes.test.js +0 -101
  315. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  316. package/tests/plugins/plugin.chapters.test.js +0 -130
  317. package/tests/plugins/plugin.mobile_nav.test.js +0 -66
  318. package/tests/plugins/plugin.text_selection.test.js +0 -203
  319. package/tests/util/docCookies.test.js +0 -15
@@ -1,7 +1,7 @@
1
- import { html } from 'lit-element';
2
- import { nothing } from 'lit-html';
3
-
1
+ import { html, nothing } from 'lit';
2
+ import '@internetarchive/icon-search/icon-search';
4
3
  import './search-results';
4
+ /** @typedef {import('@/src/plugins/search/plugin.search.js').SearchInsideMatch} SearchInsideMatch */
5
5
 
6
6
  let searchState = {
7
7
  query: '',
@@ -10,8 +10,11 @@ let searchState = {
10
10
  queryInProgress: false,
11
11
  errorMessage: '',
12
12
  };
13
- export default class {
14
- constructor(onSearchChange = () => {}, brInstance) {
13
+ export default class SearchProvider {
14
+ constructor({
15
+ onProviderChange,
16
+ bookreader
17
+ }) {
15
18
  /* search menu events */
16
19
  this.onBookSearchInitiated = this.onBookSearchInitiated.bind(this);
17
20
  /* bookreader search events */
@@ -20,16 +23,18 @@ export default class {
20
23
  this.onSearchResultsClicked = this.onSearchResultsClicked.bind(this);
21
24
  this.onSearchResultsChange = this.onSearchResultsChange.bind(this);
22
25
  this.onSearchResultsCleared = this.onSearchResultsCleared.bind(this);
26
+ this.searchCanceledInMenu = this.searchCanceledInMenu.bind(this);
27
+
23
28
  /* class methods */
24
29
  this.bindEventListeners = this.bindEventListeners.bind(this);
25
30
  this.getMenuDetails = this.getMenuDetails.bind(this);
26
31
  this.getComponent = this.getComponent.bind(this);
27
- this.advanceToPage = this.advanceToPage.bind(this);
28
32
  this.updateMenu = this.updateMenu.bind(this);
29
33
 
30
- this.onSearchChange = onSearchChange;
31
- this.bookreader = brInstance;
32
- this.icon = html`<ia-icon icon="search" style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon>`;
34
+ this.onProviderChange = onProviderChange;
35
+ /** @type {import('@/src/BookReader.js').default} */
36
+ this.bookreader = bookreader;
37
+ this.icon = html`<ia-icon-search style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon-search>`;
33
38
  this.label = 'Search inside';
34
39
  this.menuDetails = this.getMenuDetails();
35
40
  this.id = 'search';
@@ -39,7 +44,7 @@ export default class {
39
44
 
40
45
  getMenuDetails() {
41
46
  const { resultsCount, query, queryInProgress } = searchState;
42
- if (queryInProgress || !query) { return nothing }
47
+ if (queryInProgress || !query) { return nothing; }
43
48
  const unit = resultsCount === 1 ? 'result' : 'results';
44
49
  return html`(${resultsCount} ${unit})`;
45
50
  }
@@ -47,14 +52,40 @@ export default class {
47
52
  bindEventListeners() {
48
53
  window.addEventListener('BookReader:SearchStarted', this.onSearchStarted);
49
54
  window.addEventListener('BookReader:SearchCallback', this.onSearchResultsChange);
50
- window.addEventListener('BookReader:SearchCallbackEmpty', (event) => { this.onSearchRequestError(event, 'noResults') });
51
- window.addEventListener('BookReader:SearchCallbackNotIndexed', (event) => { this.onSearchRequestError(event, 'notIndexed') });
52
- window.addEventListener('BookReader:SearchCallbackError', (event) => { this.onSearchRequestError(event) });
53
- window.addEventListener('BookReader:SearchResultsCleared', () => { this.onSearchResultsCleared() });
55
+ window.addEventListener('BookReader:SearchCallbackEmpty', (event) => { this.onSearchRequestError(event, 'noResults'); });
56
+ window.addEventListener('BookReader:SearchCallbackNotIndexed', (event) => { this.onSearchRequestError(event, 'notIndexed'); });
57
+ window.addEventListener('BookReader:SearchCallbackError', (event) => { this.onSearchRequestError(event); });
58
+ window.addEventListener('BookReader:SearchResultsCleared', () => { this.onSearchResultsCleared(); });
59
+ window.addEventListener('BookReader:SearchCanceled', (e) => { this.onSearchCanceled(e); });
60
+ }
61
+
62
+ /**
63
+ * Cancel search handler
64
+ * resets `searchState`
65
+ */
66
+ onSearchCanceled() {
67
+ searchState = {
68
+ query: '',
69
+ results: [],
70
+ resultsCount: 0,
71
+ queryInProgress: false,
72
+ errorMessage: '',
73
+ };
74
+ const updateMenuFor = {
75
+ searchCanceled: true
76
+ };
77
+ this.updateMenu(updateMenuFor);
78
+
79
+ if (this.bookreader.urlPlugin) {
80
+ this.updateSearchInUrl();
81
+ }
54
82
  }
55
83
 
56
84
  onSearchStarted(e) {
57
- const { term = '' } = e.detail.props;
85
+ const { term = '', instance } = e.detail.props;
86
+ if (instance) {
87
+ this.bookreader = instance;
88
+ }
58
89
  searchState.query = term;
59
90
  searchState.results = [];
60
91
  searchState.resultsCount = 0;
@@ -79,10 +110,11 @@ export default class {
79
110
  noResults: '0 results',
80
111
  notIndexed: `This book hasn't been indexed for searching yet. We've just started indexing it,
81
112
  so search should be available soon. Please try again later. Thanks!`,
82
- default: 'Sorry, there was an error with your search. The text may still be processing.',
113
+ default: 'Sorry, there was an error with your search. Please try again.',
83
114
  };
84
115
 
85
116
  const messageToShow = errorMessages[errorType] ?? errorMessages.default;
117
+ searchState.query = instance?.searchResults?.q || '';
86
118
  searchState.results = [];
87
119
  searchState.resultsCount = 0;
88
120
  searchState.queryInProgress = false;
@@ -104,6 +136,10 @@ export default class {
104
136
  this.updateMenu();
105
137
  }
106
138
 
139
+ searchCanceledInMenu() {
140
+ this.bookreader?.cancelSearchRequest();
141
+ }
142
+
107
143
  onSearchResultsCleared() {
108
144
  searchState = {
109
145
  query: '',
@@ -111,15 +147,34 @@ export default class {
111
147
  resultsCount: 0,
112
148
  queryInProgress: false,
113
149
  errorMessage: '',
150
+ };
151
+ this.updateMenu({ openMenu: false });
152
+ this.bookreader?.searchView?.clearSearchFieldAndResults(false);
153
+ if (this.bookreader.urlPlugin) {
154
+ this.updateSearchInUrl();
155
+ }
156
+ }
157
+
158
+ /** update URL `q=<term>` query param in URL */
159
+ updateSearchInUrl() {
160
+ if (this.bookreader.urlPlugin) {
161
+ this.bookreader.urlPlugin.pullFromAddressBar();
162
+ if (searchState.query) {
163
+ this.bookreader.urlPlugin.setUrlParam('q', searchState.query);
164
+ } else {
165
+ this.bookreader.urlPlugin.removeUrlParam('q');
166
+ }
114
167
  }
115
- this.updateMenu();
116
- this.bookreader?.searchView?.clearSearchFieldAndResults();
117
168
  }
118
169
 
119
- updateMenu() {
170
+ /**
171
+ * Relays how to update side menu given the context of a search update
172
+ @param {{searchCanceled: boolean}} searchUpdates
173
+ */
174
+ updateMenu(searchUpdates = {}) {
120
175
  this.menuDetails = this.getMenuDetails();
121
176
  this.component = this.getComponent();
122
- this.onSearchChange(this.bookreader);
177
+ this.onProviderChange(this.bookreader, searchUpdates);
123
178
  }
124
179
 
125
180
  getComponent() {
@@ -134,18 +189,15 @@ export default class {
134
189
  @resultSelected=${this.onSearchResultsClicked}
135
190
  @bookSearchInitiated=${this.onBookSearchInitiated}
136
191
  @bookSearchResultsCleared=${this.onSearchResultsCleared}
192
+ @bookSearchCanceled=${this.searchCanceledInMenu}
137
193
  ></ia-book-search-results>
138
194
  `;
139
195
  }
140
196
 
197
+ /**
198
+ * @param {{ detail: {match: SearchInsideMatch} }} param0
199
+ */
141
200
  onSearchResultsClicked({ detail }) {
142
- const page = detail.match.par[0].page;
143
- this.advanceToPage(page);
144
- }
145
-
146
- advanceToPage(leaf) {
147
- const page = this.bookreader.leafNumToIndex(leaf);
148
- this.bookreader._searchPluginGoToResult(page);
149
- this.bookreader.updateSearchHilites();
201
+ this.bookreader._searchPluginGoToResult(detail.match.matchIndex);
150
202
  }
151
203
  }
@@ -1,11 +1,11 @@
1
1
  /* eslint-disable class-methods-use-this */
2
- import { nothing } from 'lit-html';
3
- import { css, html, LitElement } from 'lit-element';
2
+ import { unsafeHTML } from 'lit/directives/unsafe-html.js';
3
+ import { css, html, LitElement, nothing } from 'lit';
4
4
  import '@internetarchive/ia-activity-indicator/ia-activity-indicator';
5
- import './a-search-result.js';
6
5
  import checkmarkIcon from '../assets/icon_checkmark.js';
7
6
  import closeIcon from '../assets/icon_close.js';
8
-
7
+ import buttonCSS from '../assets/button-base.js';
8
+ /** @typedef {import('@/src/plugins/search/plugin.search.js').SearchInsideMatch} SearchInsideMatch */
9
9
 
10
10
  export class IABookSearchResults extends LitElement {
11
11
  static get properties() {
@@ -23,6 +23,7 @@ export class IABookSearchResults extends LitElement {
23
23
  constructor() {
24
24
  super();
25
25
 
26
+ /** @type {SearchInsideMatch[]} */
26
27
  this.results = [];
27
28
  this.query = '';
28
29
  this.queryInProgress = false;
@@ -60,6 +61,9 @@ export class IABookSearchResults extends LitElement {
60
61
 
61
62
  setQuery(e) {
62
63
  this.query = e.currentTarget.value;
64
+ if (!this.query) {
65
+ this.cancelSearch();
66
+ }
63
67
  }
64
68
 
65
69
  performSearch(e) {
@@ -77,7 +81,15 @@ export class IABookSearchResults extends LitElement {
77
81
  }));
78
82
  }
79
83
 
80
- selectResult() {
84
+ /**
85
+ * @param {SearchInsideMatch} match
86
+ */
87
+ selectResult(match) {
88
+ this.dispatchEvent(new CustomEvent('resultSelected', {
89
+ bubbles: true,
90
+ composed: true,
91
+ detail: { match },
92
+ }));
81
93
  this.dispatchEvent(new CustomEvent('closeMenu', {
82
94
  bubbles: true,
83
95
  composed: true,
@@ -90,10 +102,7 @@ export class IABookSearchResults extends LitElement {
90
102
  }
91
103
 
92
104
  dispatchSearchCanceled() {
93
- this.dispatchEvent(new CustomEvent('bookSearchCanceled', {
94
- bubbles: true,
95
- composed: true,
96
- }));
105
+ this.dispatchEvent(new Event('bookSearchCanceled'));
97
106
  }
98
107
 
99
108
  get resultsCount() {
@@ -122,6 +131,7 @@ export class IABookSearchResults extends LitElement {
122
131
  <div class="loading">
123
132
  <ia-activity-indicator mode="processing"></ia-activity-indicator>
124
133
  <p>Searching</p>
134
+ <button class="ia-button external cancel-search" @click=${this.cancelSearch}>Cancel</button>
125
135
  </div>
126
136
  `;
127
137
  }
@@ -131,10 +141,12 @@ export class IABookSearchResults extends LitElement {
131
141
  return html`
132
142
  <ul class="results ${resultsClass}">
133
143
  ${this.results.map(match => html`
134
- <book-search-result
135
- .match=${match}
136
- @resultSelected=${this.selectResult}
137
- ></book-search-result>
144
+ <li @click=${this.selectResult.bind(this, match)}>
145
+ ${match.cover ? html`<img src="${match.cover}" />` : nothing}
146
+ <h4>${match.title || nothing}</h4>
147
+ <p class="page-num">Page ${match.displayPageNumber}</p>
148
+ <p>${unsafeHTML(match.html)}</p>
149
+ </li>
138
150
  `)}
139
151
  </ul>
140
152
  `;
@@ -150,6 +162,7 @@ export class IABookSearchResults extends LitElement {
150
162
  name="query"
151
163
  alt="Search inside this book."
152
164
  @keyup=${this.setQuery}
165
+ @search=${this.setQuery}
153
166
  .value=${this.query}
154
167
  />
155
168
  </fieldset>
@@ -188,7 +201,7 @@ export class IABookSearchResults extends LitElement {
188
201
  const searchResultBorder = css`var(--searchResultBorder, #adaedc)`;
189
202
  const activeButtonBg = css`(--tertiaryBGColor, #333)`;
190
203
 
191
- return css`
204
+ const mainCSS = css`
192
205
  :host {
193
206
  display: block;
194
207
  height: 100%;
@@ -348,17 +361,6 @@ export class IABookSearchResults extends LitElement {
348
361
  font-size: 1.2rem;
349
362
  }
350
363
 
351
- .loading button {
352
- -webkit-appearance: none;
353
- appearance: none;
354
- padding: .5rem .7rem;
355
- font: normal 1.4rem "Helvetica Neue", Helvetica, Arial, sans-serif;
356
- border: 1px solid #656565;
357
- border-radius: 3px;
358
- cursor: pointer;
359
- background: transparent;
360
- }
361
-
362
364
  ia-activity-indicator {
363
365
  display: block;
364
366
  width: 40px;
@@ -366,6 +368,7 @@ export class IABookSearchResults extends LitElement {
366
368
  margin: 0 auto;
367
369
  }
368
370
  `;
371
+ return [buttonCSS, mainCSS];
369
372
  }
370
373
  }
371
374
  customElements.define('ia-book-search-results', IABookSearchResults);
@@ -0,0 +1,27 @@
1
+ import { html } from 'lit';
2
+ import '@internetarchive/icon-share/icon-share';
3
+ import '@internetarchive/ia-sharing-options';
4
+
5
+ export default class SharingProvider {
6
+ constructor({
7
+ item,
8
+ baseHost,
9
+ bookreader
10
+ }) {
11
+ const { identifier, creator, title } = item?.metadata;
12
+ const creatorToUse = Array.isArray(creator) ? creator[0] : creator;
13
+ const subPrefix = bookreader.options.subPrefix || '';
14
+ const label = `Share this book`;
15
+ this.icon = html`<ia-icon-share style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon-share>`;
16
+ this.label = label;
17
+ this.id = 'share';
18
+ this.component = html`<ia-sharing-options
19
+ .identifier=${identifier}
20
+ .type=${`book`}
21
+ .creator=${creatorToUse}
22
+ .description=${title}
23
+ .baseHost=${baseHost}
24
+ .fileSubPrefix=${subPrefix}
25
+ ></ia-sharing-options>`;
26
+ }
27
+ }
@@ -1,5 +1,6 @@
1
- import { html } from 'lit-element';
2
- import './visual-adjustments.js';
1
+ import { html } from 'lit';
2
+ import '@internetarchive/icon-visual-adjustment/icon-visual-adjustment';
3
+ import './visual-adjustments';
3
4
 
4
5
  const visualAdjustmentOptions = [{
5
6
  id: 'brightness',
@@ -27,11 +28,11 @@ const visualAdjustmentOptions = [{
27
28
  active: false,
28
29
  }];
29
30
 
30
- export default class {
31
+ export default class VisualAdjustmentsProvider {
31
32
  constructor(options) {
32
- const { onOptionChange = () => {}, bookContainerSelector, bookreader } = options;
33
- this.onOptionChange = onOptionChange;
34
- this.bookContainerSelector = bookContainerSelector;
33
+ const { onProviderChange, bookreader } = options;
34
+ this.onProviderChange = onProviderChange;
35
+ this.bookContainer = bookreader.refs.$brContainer;
35
36
  this.bookreader = bookreader;
36
37
 
37
38
  this.onAdjustmentChange = this.onAdjustmentChange.bind(this);
@@ -41,7 +42,7 @@ export default class {
41
42
  this.onZoomOut = this.onZoomOut.bind(this);
42
43
 
43
44
  this.activeCount = 0;
44
- this.icon = html`<ia-icon icon="visualAdjustment" style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon>`;
45
+ this.icon = html`<ia-icon-visual-adjustment style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon-visual-adjustment>`;
45
46
  this.label = 'Visual Adjustments';
46
47
  this.menuDetails = this.updateOptionsCount();
47
48
  this.id = 'adjustment';
@@ -60,7 +61,7 @@ export default class {
60
61
  }
61
62
 
62
63
  onZoomOut() {
63
- this.bookreader.zoom();
64
+ this.bookreader.zoom(-1);
64
65
  }
65
66
 
66
67
  onAdjustmentChange(event) {
@@ -76,7 +77,7 @@ export default class {
76
77
  return newValue ? [...values, newValue] : values;
77
78
  }, []).join(' ');
78
79
 
79
- document.querySelector(this.bookContainerSelector).style.setProperty('filter', filters);
80
+ this.bookContainer.css('filter', filters);
80
81
 
81
82
  this.optionUpdateComplete(event);
82
83
  }
@@ -84,7 +85,7 @@ export default class {
84
85
  optionUpdateComplete(event) {
85
86
  this.activeCount = event.detail.activeCount;
86
87
  this.updateOptionsCount(event);
87
- this.onOptionChange(event);
88
+ this.onProviderChange();
88
89
  }
89
90
 
90
91
  updateOptionsCount() {
@@ -1,6 +1,6 @@
1
- import { css, html, LitElement } from "lit-element";
2
- import { repeat } from "lit-html/directives/repeat.js";
3
- import { nothing } from "lit-html";
1
+ import { css, html, LitElement } from "lit";
2
+ import { repeat } from "lit/directives/repeat.js";
3
+ import { nothing } from "lit";
4
4
  import checkmarkIcon from '../assets/icon_checkmark.js';
5
5
  import "@internetarchive/icon-magnify-minus/icon-magnify-minus";
6
6
  import "@internetarchive/icon-magnify-plus/icon-magnify-plus";
@@ -1,76 +1,111 @@
1
- import { html } from 'lit-element';
2
- import sortAscendingIcon from '../assets/icon_sort_ascending.js';
3
- import sortDescendingIcon from '../assets/icon_sort_descending.js';
1
+ import { html } from 'lit';
4
2
 
5
- import './volumes.js';
3
+ import sortDescIcon from '../assets/icon_sort_desc.js';
4
+ import sortAscIcon from '../assets/icon_sort_asc.js';
5
+ import sortNeutralIcon from '../assets/icon_sort_neutral.js';
6
+ import volumesIcon from '../assets/icon_volumes.js';
6
7
 
8
+ import './volumes.js';
7
9
 
10
+ const sortType = {
11
+ title_asc: 'title_asc',
12
+ title_desc: 'title_desc',
13
+ default: 'default'
14
+ };
8
15
  export default class VolumesProvider {
16
+ /**
17
+ * @param {import('../../BookReader').default} bookreader
18
+ */
19
+ constructor({ baseHost, bookreader, onProviderChange }) {
20
+ this.onProviderChange = onProviderChange;
21
+ this.component = document.createElement("viewable-files");
9
22
 
10
- constructor(optionChange, volumes, bookreader) {
11
- this.optionChange = optionChange
12
- this.component = document.createElement('viewable-files');
13
-
14
- const files = {}//bookreader.options.multipleBooksList?.by_subprefix
23
+ const files = bookreader.options.multipleBooksList.by_subprefix;
15
24
  this.viewableFiles = Object.keys(files).map(item => files[item]);
16
25
  this.volumeCount = Object.keys(files).length;
17
- this.isFirstLoad = true
18
- this.isSortAscending = false;
19
26
 
20
- this.component.subPrefix = bookreader.options === undefined ? 'theworksofplato01platiala' : bookreader.options.subPrefix;
21
- this.component.hostUrl = volumes.baseHost;
27
+ /** @type {import('../../BookReader').default} */
28
+ this.bookreader = bookreader;
29
+
30
+ this.component.subPrefix = bookreader.options.subPrefix || "";
31
+ this.component.hostUrl = baseHost;
22
32
  this.component.viewableFiles = this.viewableFiles;
23
33
 
24
- this.id = 'volumes';
25
- this.label = `Viewable Files (${this.volumeCount})`
26
- this.icon = html`<ia-icon icon="volumes" style="width: var(--iconWidth); height: var(--iconHeight);"></ia-icon>`;
27
- this.actionButton = this.headerIcon
28
- this.sortVolumes()
29
- }
34
+ this.id = "volumes";
35
+ this.label = `Viewable files (${this.volumeCount})`;
36
+ this.icon = html`${volumesIcon}`;
30
37
 
31
- get sortAscendingIcon() {
32
- const iconStyle = 'background: transparent; border: none; position: absolute; right: 25px';
33
- return html`<button style=${iconStyle} class="sort-asc icon" aria-label="Sort volumes in ascending order" @click=${() => this.sortVolumes()}>${sortAscendingIcon}</button>`;
34
- }
38
+ this.sortOrderBy = sortType.default;
35
39
 
36
- get sortDescendingIcon() {
37
- const iconStyle = 'background: transparent; border: none; position: absolute; right: 25px';
38
- return html`<button style=${iconStyle} class="sort-desc icon" aria-label="Sort volumes in descending order" @click=${() => this.sortVolumes()}>${sortDescendingIcon}</button>`;
40
+ // get sort state from query param
41
+ if (this.bookreader.urlPlugin) {
42
+ this.bookreader.urlPlugin.pullFromAddressBar();
43
+
44
+ const urlSortValue = this.bookreader.urlPlugin.getUrlParam('sort');
45
+ if (urlSortValue === sortType.title_asc || urlSortValue === sortType.title_desc) {
46
+ this.sortOrderBy = urlSortValue;
47
+ }
48
+ }
49
+ this.sortVolumes(this.sortOrderBy);
39
50
  }
40
51
 
41
- get headerIcon() {
42
- return this.isSortAscending ? this.sortAscendingIcon : this.sortDescendingIcon;
52
+ get sortButton() {
53
+ const sortIcons = {
54
+ default: html`
55
+ <button class="sort-by neutral-icon" aria-label="Sort volumes in initial order" @click=${() => this.sortVolumes("title_asc")}>${sortNeutralIcon}</button>
56
+ `,
57
+ title_asc: html`
58
+ <button class="sort-by asc-icon" aria-label="Sort volumes in ascending order" @click=${() => this.sortVolumes("title_desc")}>${sortAscIcon}</button>
59
+ `,
60
+ title_desc: html`
61
+ <button class="sort-by desc-icon" aria-label="Sort volumes in descending order" @click=${() => this.sortVolumes("default")}>${sortDescIcon}</button>
62
+ `,
63
+ };
64
+
65
+ return sortIcons[this.sortOrderBy];
43
66
  }
44
67
 
45
- sortVolumes() {
46
- this.isSortAscending = !this.isSortAscending;
47
- const sortedFiles = this.viewableFiles.sort((a, b) => {
48
- if (this.isSortAscending) return a.title.localeCompare(b.title);
49
- else return b.title.localeCompare(a.title);
50
- })
51
-
52
- this.component.viewableFiles = [...sortedFiles]
53
- if (!this.isFirstLoad) {
54
- this.actionButton = this.headerIcon
55
- this.optionChange(this.bookreader)
56
- } else {
57
- this.isFirstLoad = false
68
+ /**
69
+ * @param {'default' | 'title_asc' | 'title_desc'} sortByType
70
+ */
71
+ sortVolumes(sortByType) {
72
+ let sortedFiles = [];
73
+
74
+ const files = this.viewableFiles;
75
+ sortedFiles = files.sort((a, b) => {
76
+ if (sortByType === sortType.title_asc) return a.title.localeCompare(b.title);
77
+ else if (sortByType === sortType.title_desc) return b.title.localeCompare(a.title);
78
+ else return a.orig_sort - b.orig_sort;
79
+ });
80
+
81
+ this.sortOrderBy = sortByType;
82
+ this.component.sortOrderBy = sortByType;
83
+ this.component.viewableFiles = [...sortedFiles];
84
+ this.actionButton = this.sortButton;
85
+
86
+ if (this.bookreader.urlPlugin) {
87
+ this.bookreader.urlPlugin.pullFromAddressBar();
88
+ if (this.sortOrderBy !== sortType.default) {
89
+ this.bookreader.urlPlugin.setUrlParam('sort', sortByType);
90
+ } else {
91
+ this.bookreader.urlPlugin.removeUrlParam('sort');
92
+ }
58
93
  }
59
- }
60
94
 
61
- bindEvents() {
62
- this.component.addEventListener('click', this.multiplefilesClicked.bind(this));
95
+ this.onProviderChange(this.bookreader);
96
+
97
+ this.multipleFilesClicked(sortByType);
63
98
  }
64
99
 
65
- multiplefilesClicked({ detail }) {
66
- // maybe some analytics?
67
- if (window.archive_analytics) {
68
- window.archive_analytics?.send_event_no_sampling(
69
- 'BookReader',
70
- `VolumesSort`,
71
- window.location.path,
72
- );
73
- }
100
+ /**
101
+ * @param {'default' | 'title_asc' | 'title_desc'} orderBy
102
+ */
103
+ multipleFilesClicked(orderBy) {
104
+ window.archive_analytics?.send_event(
105
+ 'BookReader',
106
+ `VolumesSort|${orderBy}`,
107
+ window.location.path,
108
+ );
74
109
  }
75
110
 
76
111
  }