@internetarchive/bookreader 5.0.0-9 → 5.0.0-91

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 (325) hide show
  1. package/.eslintrc.js +21 -19
  2. package/.github/workflows/node.js.yml +76 -11
  3. package/.github/workflows/npm-publish.yml +6 -20
  4. package/.testcaferc.js +10 -0
  5. package/BookReader/BookReader.css +404 -1125
  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 +1782 -0
  10. package/BookReader/ia-bookreader-bundle.js.LICENSE.txt +7 -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/images/unviewable_page.png +0 -0
  58. package/BookReader/jquery-3.js +2 -0
  59. package/BookReader/jquery-3.js.LICENSE.txt +24 -0
  60. package/BookReader/plugins/plugin.archive_analytics.js +1 -1
  61. package/BookReader/plugins/plugin.archive_analytics.js.map +1 -1
  62. package/BookReader/plugins/plugin.autoplay.js +1 -1
  63. package/BookReader/plugins/plugin.autoplay.js.map +1 -1
  64. package/BookReader/plugins/plugin.chapters.js +25 -1
  65. package/BookReader/plugins/plugin.chapters.js.LICENSE.txt +1 -0
  66. package/BookReader/plugins/plugin.chapters.js.map +1 -1
  67. package/BookReader/plugins/plugin.iframe.js +1 -1
  68. package/BookReader/plugins/plugin.iframe.js.map +1 -1
  69. package/BookReader/plugins/plugin.iiif.js +2 -0
  70. package/BookReader/plugins/plugin.iiif.js.map +1 -0
  71. package/BookReader/plugins/plugin.resume.js +1 -1
  72. package/BookReader/plugins/plugin.resume.js.map +1 -1
  73. package/BookReader/plugins/plugin.search.js +2 -1
  74. package/BookReader/plugins/plugin.search.js.LICENSE.txt +1 -0
  75. package/BookReader/plugins/plugin.search.js.map +1 -1
  76. package/BookReader/plugins/plugin.text_selection.js +2 -1
  77. package/BookReader/plugins/plugin.text_selection.js.LICENSE.txt +1 -0
  78. package/BookReader/plugins/plugin.text_selection.js.map +1 -1
  79. package/BookReader/plugins/plugin.tts.js +1 -1
  80. package/BookReader/plugins/plugin.tts.js.LICENSE.txt +2 -0
  81. package/BookReader/plugins/plugin.tts.js.map +1 -1
  82. package/BookReader/plugins/plugin.url.js +1 -1
  83. package/BookReader/plugins/plugin.url.js.map +1 -1
  84. package/BookReader/plugins/plugin.vendor-fullscreen.js +1 -1
  85. package/BookReader/plugins/plugin.vendor-fullscreen.js.map +1 -1
  86. package/BookReader/webcomponents-bundle.js +3 -0
  87. package/BookReader/webcomponents-bundle.js.LICENSE.txt +9 -0
  88. package/BookReader/webcomponents-bundle.js.map +1 -0
  89. package/BookReaderDemo/BookReaderDemo.css +18 -19
  90. package/BookReaderDemo/BookReaderJSAdvanced.js +0 -3
  91. package/BookReaderDemo/BookReaderJSSimple.js +1 -0
  92. package/BookReaderDemo/IADemoBr.js +144 -0
  93. package/BookReaderDemo/demo-advanced.html +2 -2
  94. package/BookReaderDemo/demo-embed-iframe-src.html +2 -1
  95. package/BookReaderDemo/demo-fullscreen-mobile.html +3 -5
  96. package/BookReaderDemo/demo-fullscreen.html +2 -4
  97. package/BookReaderDemo/demo-iiif.html +99 -12
  98. package/BookReaderDemo/demo-internetarchive.html +214 -18
  99. package/BookReaderDemo/demo-multiple.html +2 -1
  100. package/BookReaderDemo/demo-preview-pages.html +526 -525
  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 +600 -33
  110. package/README.md +14 -1
  111. package/babel.config.js +20 -0
  112. package/codecov.yml +6 -0
  113. package/index.html +5 -2
  114. package/jsconfig.json +19 -0
  115. package/netlify.toml +9 -0
  116. package/package.json +70 -62
  117. package/renovate.json +52 -0
  118. package/scripts/preversion.js +0 -1
  119. package/src/BookNavigator/assets/bookmark-colors.js +1 -1
  120. package/src/BookNavigator/assets/button-base.js +5 -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 +590 -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 +4 -9
  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 +29 -25
  134. package/src/BookNavigator/search/search-provider.js +50 -28
  135. package/src/BookNavigator/search/search-results.js +24 -10
  136. package/src/BookNavigator/sharing.js +27 -0
  137. package/src/BookNavigator/viewable-files.js +95 -0
  138. package/src/BookNavigator/visual-adjustments/visual-adjustments-provider.js +13 -12
  139. package/src/BookNavigator/visual-adjustments/visual-adjustments.js +7 -7
  140. package/src/BookReader/BookModel.js +79 -43
  141. package/src/BookReader/DragScrollable.js +233 -0
  142. package/src/BookReader/ImageCache.js +48 -15
  143. package/src/BookReader/Mode1Up.js +56 -351
  144. package/src/BookReader/Mode1UpLit.js +388 -0
  145. package/src/BookReader/Mode2Up.js +73 -1318
  146. package/src/BookReader/Mode2UpLit.js +777 -0
  147. package/src/BookReader/ModeCoordinateSpace.js +29 -0
  148. package/src/BookReader/ModeSmoothZoom.js +312 -0
  149. package/src/BookReader/ModeThumb.js +19 -13
  150. package/src/BookReader/Navbar/Navbar.js +70 -54
  151. package/src/BookReader/PageContainer.js +116 -22
  152. package/src/BookReader/ReduceSet.js +3 -3
  153. package/src/BookReader/Toolbar/Toolbar.js +14 -41
  154. package/src/BookReader/events.js +2 -3
  155. package/src/BookReader/options.js +75 -15
  156. package/src/BookReader/utils/HTMLDimensionsCacher.js +44 -0
  157. package/src/BookReader/utils/ScrollClassAdder.js +31 -0
  158. package/src/BookReader/utils/SelectionObserver.js +45 -0
  159. package/src/BookReader/utils/classes.js +1 -1
  160. package/src/BookReader/utils.js +128 -13
  161. package/src/BookReader.js +559 -1083
  162. package/src/BookReaderPlugin.js +44 -0
  163. package/src/assets/icons/magnify-minus.svg +3 -7
  164. package/src/assets/icons/magnify-plus.svg +3 -7
  165. package/src/assets/icons/voice.svg +1 -0
  166. package/src/assets/images/unviewable_page.png +0 -0
  167. package/src/css/BookReader.scss +1 -5
  168. package/src/css/_BRBookmarks.scss +1 -1
  169. package/src/css/_BRComponent.scss +1 -1
  170. package/src/css/_BRicon.scss +8 -2
  171. package/src/css/_BRmain.scss +16 -3
  172. package/src/css/_BRnav.scss +12 -42
  173. package/src/css/_BRpages.scss +170 -42
  174. package/src/css/_BRsearch.scss +68 -25
  175. package/src/css/_BRtoolbar.scss +5 -5
  176. package/src/css/_TextSelection.scss +87 -27
  177. package/src/css/_colorbox.scss +2 -2
  178. package/src/css/_controls.scss +24 -7
  179. package/src/css/_icons.scss +1 -1
  180. package/src/ia-bookreader/ia-bookreader.js +224 -0
  181. package/src/plugins/plugin.archive_analytics.js +84 -78
  182. package/src/plugins/plugin.autoplay.js +99 -104
  183. package/src/plugins/plugin.chapters.js +310 -201
  184. package/src/plugins/plugin.iframe.js +1 -1
  185. package/src/plugins/plugin.iiif.js +141 -0
  186. package/src/plugins/plugin.resume.js +53 -50
  187. package/src/plugins/plugin.text_selection.js +519 -175
  188. package/src/plugins/plugin.vendor-fullscreen.js +7 -7
  189. package/src/plugins/search/plugin.search.js +151 -127
  190. package/src/plugins/search/utils.js +43 -0
  191. package/src/plugins/search/view.js +37 -59
  192. package/src/plugins/tts/AbstractTTSEngine.js +75 -45
  193. package/src/plugins/tts/FestivalTTSEngine.js +21 -31
  194. package/src/plugins/tts/PageChunk.js +16 -23
  195. package/src/plugins/tts/PageChunkIterator.js +11 -17
  196. package/src/plugins/tts/WebTTSEngine.js +126 -84
  197. package/src/plugins/tts/plugin.tts.js +308 -350
  198. package/src/plugins/tts/utils.js +29 -26
  199. package/src/plugins/url/UrlPlugin.js +191 -0
  200. package/src/plugins/{plugin.url.js → url/plugin.url.js} +47 -18
  201. package/src/util/browserSniffing.js +33 -1
  202. package/src/util/docCookies.js +21 -2
  203. package/src/util/strings.js +1 -0
  204. package/tests/e2e/README.md +37 -0
  205. package/tests/e2e/autoplay.test.js +9 -6
  206. package/tests/e2e/base.test.js +8 -16
  207. package/tests/e2e/helpers/base.js +55 -50
  208. package/tests/e2e/helpers/debug.js +1 -1
  209. package/tests/e2e/helpers/mockSearch.js +19 -22
  210. package/tests/e2e/helpers/params.js +17 -0
  211. package/tests/e2e/helpers/rightToLeft.js +8 -14
  212. package/tests/e2e/helpers/search.js +73 -0
  213. package/tests/e2e/models/Navigation.js +20 -37
  214. package/tests/e2e/rightToLeft.test.js +4 -5
  215. package/tests/e2e/viewmode.test.js +40 -33
  216. package/tests/jest/BookNavigator/book-navigator.test.js +661 -0
  217. package/tests/jest/BookNavigator/bookmarks/bookmark-button.test.js +43 -0
  218. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmark-edit.test.js +25 -26
  219. package/tests/{karma → jest}/BookNavigator/bookmarks/bookmarks-list.test.js +41 -42
  220. package/tests/jest/BookNavigator/bookmarks/ia-bookmarks.test.js +45 -0
  221. package/tests/jest/BookNavigator/downloads/downloads-provider.test.js +67 -0
  222. package/tests/jest/BookNavigator/downloads/downloads.test.js +53 -0
  223. package/tests/jest/BookNavigator/search/search-provider.test.js +167 -0
  224. package/tests/{karma → jest}/BookNavigator/search/search-results.test.js +109 -60
  225. package/tests/jest/BookNavigator/sharing/sharing-provider.test.js +49 -0
  226. package/tests/jest/BookNavigator/viewable-files/viewable-files-provider.test.js +80 -0
  227. package/tests/jest/BookNavigator/visual-adjustments.test.js +200 -0
  228. package/tests/{BookReader → jest/BookReader}/BookModel.test.js +74 -14
  229. package/tests/jest/BookReader/BookReaderPublicFunctions.test.js +263 -0
  230. package/tests/{BookReader → jest/BookReader}/ImageCache.test.js +4 -4
  231. package/tests/jest/BookReader/Mode1UpLit.test.js +73 -0
  232. package/tests/jest/BookReader/Mode2Up.test.js +98 -0
  233. package/tests/jest/BookReader/Mode2UpLit.test.js +190 -0
  234. package/tests/jest/BookReader/ModeCoordinateSpace.test.js +16 -0
  235. package/tests/jest/BookReader/ModeSmoothZoom.test.js +218 -0
  236. package/tests/jest/BookReader/ModeThumb.test.js +71 -0
  237. package/tests/{BookReader → jest/BookReader}/Navbar/Navbar.test.js +42 -29
  238. package/tests/jest/BookReader/PageContainer.test.js +238 -0
  239. package/tests/{BookReader → jest/BookReader}/ReduceSet.test.js +1 -1
  240. package/tests/{BookReader → jest/BookReader}/Toolbar/Toolbar.test.js +3 -3
  241. package/tests/jest/BookReader/utils/HTMLDimensionsCacher.test.js +59 -0
  242. package/tests/jest/BookReader/utils/ScrollClassAdder.test.js +49 -0
  243. package/tests/jest/BookReader/utils/SelectionObserver.test.js +57 -0
  244. package/tests/{BookReader → jest/BookReader}/utils/classes.test.js +1 -1
  245. package/tests/jest/BookReader/utils.test.js +250 -0
  246. package/tests/jest/BookReader.keyboard.test.js +190 -0
  247. package/tests/{BookReader.options.test.js → jest/BookReader.options.test.js} +10 -2
  248. package/tests/{BookReader.test.js → jest/BookReader.test.js} +43 -53
  249. package/tests/jest/plugins/plugin.archive_analytics.test.js +20 -0
  250. package/tests/jest/plugins/plugin.autoplay.test.js +35 -0
  251. package/tests/jest/plugins/plugin.chapters.test.js +193 -0
  252. package/tests/{plugins → jest/plugins}/plugin.iframe.test.js +4 -4
  253. package/tests/{plugins → jest/plugins}/plugin.resume.test.js +22 -35
  254. package/tests/jest/plugins/plugin.text_selection.test.js +316 -0
  255. package/tests/{plugins → jest/plugins}/plugin.vendor-fullscreen.test.js +2 -2
  256. package/tests/{plugins → jest/plugins}/search/plugin.search.test.js +19 -47
  257. package/tests/{plugins → jest/plugins}/search/plugin.search.view.test.js +42 -9
  258. package/tests/jest/plugins/search/utils.js +25 -0
  259. package/tests/jest/plugins/search/utils.test.js +29 -0
  260. package/tests/{plugins → jest/plugins}/tts/AbstractTTSEngine.test.js +30 -10
  261. package/tests/{plugins → jest/plugins}/tts/FestivalTTSEngine.test.js +4 -4
  262. package/tests/{plugins → jest/plugins}/tts/PageChunk.test.js +1 -1
  263. package/tests/{plugins → jest/plugins}/tts/PageChunkIterator.test.js +3 -3
  264. package/tests/{plugins → jest/plugins}/tts/WebTTSEngine.test.js +65 -13
  265. package/tests/{plugins → jest/plugins}/tts/utils.test.js +1 -60
  266. package/tests/jest/plugins/url/UrlPlugin.test.js +198 -0
  267. package/tests/{plugins → jest/plugins/url}/plugin.url.test.js +57 -18
  268. package/tests/jest/setup.js +3 -0
  269. package/tests/{util → jest/util}/browserSniffing.test.js +10 -4
  270. package/tests/jest/util/docCookies.test.js +24 -0
  271. package/tests/{util → jest/util}/strings.test.js +1 -1
  272. package/tests/jest/utils.js +83 -0
  273. package/webpack.config.js +16 -10
  274. package/.babelrc +0 -12
  275. package/.dependabot/config.yml +0 -6
  276. package/.testcaferc.json +0 -5
  277. package/BookReader/bookreader-component-bundle.js +0 -1450
  278. package/BookReader/bookreader-component-bundle.js.LICENSE.txt +0 -38
  279. package/BookReader/bookreader-component-bundle.js.map +0 -1
  280. package/BookReader/jquery-1.10.1.js +0 -2
  281. package/BookReader/jquery-1.10.1.js.LICENSE.txt +0 -24
  282. package/BookReader/plugins/plugin.menu_toggle.js +0 -2
  283. package/BookReader/plugins/plugin.menu_toggle.js.map +0 -1
  284. package/BookReader/plugins/plugin.mobile_nav.js +0 -2
  285. package/BookReader/plugins/plugin.mobile_nav.js.map +0 -1
  286. package/BookReaderDemo/BookReaderJSAutoplay.js +0 -56
  287. package/BookReaderDemo/IIIFBookReader.js +0 -207
  288. package/BookReaderDemo/bookreader-template-bundle.js +0 -7178
  289. package/BookReaderDemo/demo-autoplay.html +0 -38
  290. package/BookReaderDemo/demo-iiif.js +0 -26
  291. package/BookReaderDemo/demo-plugin-menu-toggle.html +0 -34
  292. package/karma.conf.js +0 -23
  293. package/src/BookNavigator/BookModel.js +0 -14
  294. package/src/BookNavigator/BookNavigator.js +0 -446
  295. package/src/BookNavigator/assets/book-loader.js +0 -27
  296. package/src/BookNavigator/br-fullscreen-mgr.js +0 -83
  297. package/src/BookNavigator/search/a-search-result.js +0 -55
  298. package/src/BookReader/DebugConsole.js +0 -54
  299. package/src/BookReaderComponent/BookReaderComponent.js +0 -112
  300. package/src/ItemNavigator/ItemNavigator.js +0 -376
  301. package/src/ItemNavigator/providers/sharing.js +0 -29
  302. package/src/css/_MobileNav.scss +0 -194
  303. package/src/dragscrollable-br.js +0 -261
  304. package/src/plugins/menu_toggle/plugin.menu_toggle.js +0 -324
  305. package/src/plugins/plugin.mobile_nav.js +0 -287
  306. package/tests/BookReader/BookReaderPublicFunctions.test.js +0 -171
  307. package/tests/BookReader/DebugConsole.test.js +0 -25
  308. package/tests/BookReader/Mode1Up.test.js +0 -164
  309. package/tests/BookReader/Mode2Up.test.js +0 -247
  310. package/tests/BookReader/PageContainer.test.js +0 -115
  311. package/tests/BookReader/utils.test.js +0 -109
  312. package/tests/e2e/helpers/desktopSearch.js +0 -72
  313. package/tests/e2e/helpers/mobileSearch.js +0 -85
  314. package/tests/e2e/ia-production/ia-prod-base.js +0 -17
  315. package/tests/karma/BookNavigator/book-navigator.test.js +0 -132
  316. package/tests/karma/BookNavigator/search/search-provider.test.js +0 -23
  317. package/tests/karma/BookNavigator/visual-adjustments.test.js +0 -201
  318. package/tests/plugins/menu_toggle/plugin.menu_toggle.test.js +0 -68
  319. package/tests/plugins/plugin.archive_analytics.test.js +0 -23
  320. package/tests/plugins/plugin.autoplay.test.js +0 -52
  321. package/tests/plugins/plugin.chapters.test.js +0 -130
  322. package/tests/plugins/plugin.mobile_nav.test.js +0 -66
  323. package/tests/plugins/plugin.text_selection.test.js +0 -203
  324. package/tests/util/docCookies.test.js +0 -15
  325. package/tests/utils.js +0 -42
@@ -1,19 +1,13 @@
1
1
  //@ts-check
2
- import { isFirefox, isSafari } from '../util/browserSniffing.js';
2
+ import { createDIVPageLayer } from '../BookReader/PageContainer.js';
3
+ import { SelectionObserver } from '../BookReader/utils/SelectionObserver.js';
4
+ import { BookReaderPlugin } from '../BookReaderPlugin.js';
3
5
  import { applyVariables } from '../util/strings.js';
4
6
  /** @typedef {import('../util/strings.js').StringWithVars} StringWithVars */
7
+ /** @typedef {import('../BookReader/PageContainer.js').PageContainer} PageContainer */
5
8
 
6
9
  const BookReader = /** @type {typeof import('../BookReader').default} */(window.BookReader);
7
10
 
8
- export const DEFAULT_OPTIONS = {
9
- enabled: true,
10
- /** @type {StringWithVars} The URL to fetch the entire DJVU xml. Supports options.vars */
11
- fullDjvuXmlUrl: null,
12
- /** @type {StringWithVars} The URL to fetch a single page of the DJVU xml. Supports options.vars. Also has {{pageIndex}} */
13
- singlePageDjvuXmlUrl: null,
14
- };
15
- /** @typedef {typeof DEFAULT_OPTIONS} TextSelectionPluginOptions */
16
-
17
11
  /**
18
12
  * @template T
19
13
  */
@@ -35,45 +29,117 @@ export class Cache {
35
29
  }
36
30
  }
37
31
 
38
- export class TextSelectionPlugin {
39
-
40
- constructor(options = DEFAULT_OPTIONS, optionVariables, avoidTspans = isFirefox(), pointerEventsOnParagraph = isSafari()) {
41
- this.options = options;
42
- this.optionVariables = optionVariables;
43
- /**@type {PromiseLike<JQuery<HTMLElement>|undefined>} */
44
- this.djvuPagesPromise = null;
45
- // Using text elements instead of tspans for words because Firefox does not allow svg tspan stretch.
46
- // Tspans are necessary on Chrome because they prevent newline character after every word when copying
47
- this.svgParagraphElement = "text";
48
- this.svgWordElement = "tspan";
49
- this.insertNewlines = avoidTspans
50
- // Safari has a bug where `pointer-events` doesn't work on `<tspans>`. So
51
- // there we will set `pointer-events: all` on the paragraph element. We don't
52
- // do this everywhere, because it's a worse experience. Thanks Safari :/
53
- this.pointerEventsOnParagraph = pointerEventsOnParagraph;
54
- if (avoidTspans) {
55
- this.svgParagraphElement = "g";
56
- this.svgWordElement = "text";
57
- }
32
+ export class TextSelectionPlugin extends BookReaderPlugin {
33
+ options = {
34
+ enabled: true,
35
+ /** @type {StringWithVars} The URL to fetch the entire DJVU xml. Supports options.vars */
36
+ fullDjvuXmlUrl: null,
37
+ /** @type {StringWithVars} The URL to fetch a single page of the DJVU xml. Supports options.vars. Also has {{pageIndex}} */
38
+ singlePageDjvuXmlUrl: null,
39
+ /** Whether to fetch the XML as a jsonp */
40
+ jsonp: false,
41
+ }
58
42
 
59
- /** @type {Cache<{index: number, response: any}>} */
60
- this.pageTextCache = new Cache();
43
+ /**@type {PromiseLike<JQuery<HTMLElement>|undefined>} */
44
+ djvuPagesPromise = null;
61
45
 
62
- /**
63
- * Sometimes there are too many words on a page, and the browser becomes near
64
- * unusable. For now don't render text layer for pages with too many words.
65
- */
66
- this.maxWordRendered = 2500;
46
+ /** @type {Cache<{index: number, response: any}>} */
47
+ pageTextCache = new Cache();
48
+
49
+ /**
50
+ * Sometimes there are too many words on a page, and the browser becomes near
51
+ * unusable. For now don't render text layer for pages with too many words.
52
+ */
53
+ maxWordRendered = 2500;
54
+
55
+ /**
56
+ * @param {import('../BookReader.js').default} br
57
+ */
58
+ constructor(br) {
59
+ super(br);
60
+ // In the future this should be in the ocr file
61
+ // since a book being right to left doesn't mean the ocr is right to left. But for
62
+ // now we do make that assumption.
63
+ /** Whether the book is right-to-left */
64
+ this.rtl = this.br.pageProgression === 'rl';
65
+ this.selectionObserver = new SelectionObserver('.BRtextLayer', this._onSelectionChange);
67
66
  }
68
67
 
68
+ /** @override */
69
69
  init() {
70
+ if (!this.options.enabled) return;
71
+
72
+ this.loadData();
73
+
74
+ this.selectionObserver.attach();
75
+ new SelectionObserver('.BRtextLayer', (selectEvent) => {
76
+ // Track how often selection is used
77
+ if (selectEvent == 'started') {
78
+ this.br._plugins.archiveAnalytics?.sendEvent('BookReader', 'SelectStart');
79
+
80
+ // Set a class on the page to avoid hiding it when zooming/etc
81
+ this.br.refs.$br.find('.BRpagecontainer--hasSelection').removeClass('BRpagecontainer--hasSelection');
82
+ $(window.getSelection().anchorNode).closest('.BRpagecontainer').addClass('BRpagecontainer--hasSelection');
83
+ }
84
+ }).attach();
85
+
86
+ if (this.br.protected) {
87
+ // Prevent right clicking when selected text
88
+ $(document.body).on('contextmenu dragstart copy', (e) => {
89
+ const selection = document.getSelection();
90
+ if (selection?.toString()) {
91
+ const intersectsTextLayer = $('.BRtextLayer')
92
+ .toArray()
93
+ .some(el => selection.containsNode(el, true));
94
+ if (intersectsTextLayer) {
95
+ e.preventDefault();
96
+ return false;
97
+ }
98
+ }
99
+ });
100
+ }
101
+ }
102
+
103
+ /**
104
+ * @override
105
+ * @param {PageContainer} pageContainer
106
+ * @returns {PageContainer}
107
+ */
108
+ _configurePageContainer(pageContainer) {
109
+ // Disable if thumb mode; it's too janky
110
+ // .page can be null for "pre-cover" region
111
+ if (this.br.mode !== this.br.constModeThumb && pageContainer.page) {
112
+ this.createTextLayer(pageContainer);
113
+ }
114
+ return pageContainer;
115
+ }
116
+
117
+ /**
118
+ * @param {'started' | 'cleared'} type
119
+ * @param {HTMLElement} target
120
+ */
121
+ _onSelectionChange = (type, target) => {
122
+ if (type === 'started') {
123
+ this.textSelectingMode(target);
124
+ } else if (type === 'cleared') {
125
+ this.defaultMode(target);
126
+ } else {
127
+ throw new Error(`Unknown type ${type}`);
128
+ }
129
+ }
130
+
131
+ loadData() {
70
132
  // Only fetch the full djvu xml if the single page url isn't there
71
133
  if (this.options.singlePageDjvuXmlUrl) return;
72
134
  this.djvuPagesPromise = $.ajax({
73
135
  type: "GET",
74
- url: applyVariables(this.options.fullDjvuXmlUrl, this.optionVariables),
75
- dataType: "html",
76
- error: (e) => undefined
136
+ url: applyVariables(this.options.fullDjvuXmlUrl, this.br.options.vars),
137
+ dataType: this.options.jsonp ? "jsonp" : "html",
138
+ cache: true,
139
+ xhrFields: {
140
+ withCredentials: this.br.protected,
141
+ },
142
+ error: (e) => undefined,
77
143
  }).then((res) => {
78
144
  try {
79
145
  const xmlMap = $.parseXML(res);
@@ -94,21 +160,24 @@ export class TextSelectionPlugin {
94
160
  if (cachedEntry) {
95
161
  return cachedEntry.response;
96
162
  }
97
- return $.ajax({
163
+ const res = await $.ajax({
98
164
  type: "GET",
99
- url: applyVariables(this.options.singlePageDjvuXmlUrl, this.optionVariables, { pageIndex: index }),
100
- dataType: "html",
165
+ url: applyVariables(this.options.singlePageDjvuXmlUrl, this.br.options.vars, { pageIndex: index }),
166
+ dataType: this.options.jsonp ? "jsonp" : "html",
167
+ cache: true,
168
+ xhrFields: {
169
+ withCredentials: this.br.protected,
170
+ },
101
171
  error: (e) => undefined,
102
- }).then((res) => {
103
- try {
104
- const xmlDoc = $.parseXML(res);
105
- const result = xmlDoc && $(xmlDoc).find("OBJECT")[0];
106
- this.pageTextCache.add({ index, response: result });
107
- return result;
108
- } catch (e) {
109
- return undefined;
110
- }
111
172
  });
173
+ try {
174
+ const xmlDoc = $.parseXML(res);
175
+ const result = xmlDoc && $(xmlDoc).find("OBJECT")[0];
176
+ this.pageTextCache.add({ index, response: result });
177
+ return result;
178
+ } catch (e) {
179
+ return undefined;
180
+ }
112
181
  } else {
113
182
  const XMLpagesArr = await this.djvuPagesPromise;
114
183
  if (XMLpagesArr) return XMLpagesArr[index];
@@ -129,67 +198,87 @@ export class TextSelectionPlugin {
129
198
 
130
199
  /**
131
200
  * Applies mouse events when in default mode
132
- * @param {SVGElement} svg
201
+ * @param {HTMLElement} textLayer
133
202
  */
134
- defaultMode(svg) {
135
- svg.classList.remove("selectingSVG");
136
- $(svg).on("mousedown.textSelectPluginHandler", (event) => {
137
- if (!$(event.target).is(".BRwordElement")) return;
138
- event.stopPropagation();
139
- svg.classList.add("selectingSVG");
140
- $(svg).one("mouseup.textSelectPluginHandler", (event) => {
141
- if (window.getSelection().toString() != "") {
142
- event.stopPropagation();
143
- $(svg).off(".textSelectPluginHandler");
144
- this.textSelectingMode(svg);
145
- }
146
- else svg.classList.remove("selectingSVG");
147
- })
148
- })
203
+ defaultMode(textLayer) {
204
+ const $pageContainer = $(textLayer).closest('.BRpagecontainer');
205
+ textLayer.style.pointerEvents = "none";
206
+ $pageContainer.find("img").css("pointer-events", "auto");
207
+
208
+ $(textLayer).off(".textSelectPluginHandler");
209
+ const startedMouseDown = this.mouseIsDown;
210
+ let skipNextMouseup = this.mouseIsDown;
211
+ if (startedMouseDown) {
212
+ textLayer.style.pointerEvents = "auto";
213
+ }
214
+
215
+ // Need to stop propagation to prevent DragScrollable from
216
+ // blocking selection
217
+ $(textLayer).on("mousedown.textSelectPluginHandler", (event) => {
218
+ this.mouseIsDown = true;
219
+ if ($(event.target).is(".BRwordElement, .BRspace")) {
220
+ event.stopPropagation();
221
+ }
222
+ });
223
+
224
+ $(textLayer).on("mouseup.textSelectPluginHandler", (event) => {
225
+ this.mouseIsDown = false;
226
+ textLayer.style.pointerEvents = "none";
227
+ if (skipNextMouseup) {
228
+ skipNextMouseup = false;
229
+ event.stopPropagation();
230
+ }
231
+ });
149
232
  }
150
233
 
151
234
  /**
152
- * Applies mouse events when in textSelecting mode
153
- * @param {SVGElement} svg
235
+ * This mode is active while there is a selection on the given textLayer
236
+ * @param {HTMLElement} textLayer
154
237
  */
155
- textSelectingMode(svg) {
156
- $(svg).on('mousedown.textSelectPluginHandler', (event) => {
157
- if (!$(event.target).is(".BRwordElement")) {
158
- if (window.getSelection().toString() != "") window.getSelection().removeAllRanges();
159
- }
238
+ textSelectingMode(textLayer) {
239
+ const $pageContainer = $(textLayer).closest('.BRpagecontainer');
240
+ // Make text layer consume all events
241
+ textLayer.style.pointerEvents = "all";
242
+ // Block img from getting long-press to save while selecting
243
+ $pageContainer.find("img").css("pointer-events", "none");
244
+
245
+ $(textLayer).off(".textSelectPluginHandler");
246
+
247
+ $(textLayer).on('mousedown.textSelectPluginHandler', (event) => {
248
+ this.mouseIsDown = true;
160
249
  event.stopPropagation();
161
- })
162
- $(svg).on('mouseup.textSelectPluginHandler', (event) => {
250
+ });
251
+
252
+ // Prevent page flip on click
253
+ $(textLayer).on('mouseup.textSelectPluginHandler', (event) => {
254
+ this.mouseIsDown = false;
163
255
  event.stopPropagation();
164
- if (window.getSelection().toString() == "") {
165
- $(svg).off(".textSelectPluginHandler");
166
- this.defaultMode(svg); }
167
- })
256
+ });
168
257
  }
169
258
 
170
259
  /**
171
- * Initializes text selection modes if there is an svg on the page
260
+ * Initializes text selection modes if there is a text layer on the page
172
261
  * @param {JQuery} $container
173
262
  */
174
263
  stopPageFlip($container) {
175
- /** @type {JQuery<SVGElement>} */
176
- const $svg = $container.find('svg.textSelectionSVG');
177
- if (!$svg.length) return;
178
- $svg.each((i, s) => this.defaultMode(s))
264
+ /** @type {JQuery<HTMLElement>} */
265
+ const $textLayer = $container.find('.BRtextLayer');
266
+ if (!$textLayer.length) return;
267
+ $textLayer.each((i, s) => this.defaultMode(s));
179
268
  this.interceptCopy($container);
180
269
  }
181
270
 
182
271
  /**
183
- * @param {number} pageIndex
184
- * @param {JQuery} $container
272
+ * @param {PageContainer} pageContainer
185
273
  */
186
- async createTextLayer(pageIndex, $container) {
187
- const $svgLayers = $container.find('.textSelectionSVG');
188
- if ($svgLayers.length) return;
274
+ async createTextLayer(pageContainer) {
275
+ const pageIndex = pageContainer.page.index;
276
+ const $container = pageContainer.$container;
277
+ const $textLayers = $container.find('.BRtextLayer');
278
+ if ($textLayers.length) return;
189
279
  const XMLpage = await this.getPageText(pageIndex);
190
280
  if (!XMLpage) return;
191
- const XMLwidth = $(XMLpage).attr("width");
192
- const XMLheight = $(XMLpage).attr("height");
281
+ recursivelyAddCoords(XMLpage);
193
282
 
194
283
  const totalWords = $(XMLpage).find("WORD").length;
195
284
  if (totalWords > this.maxWordRendered) {
@@ -197,107 +286,362 @@ export class TextSelectionPlugin {
197
286
  return;
198
287
  }
199
288
 
200
- const svg = document.createElementNS("http://www.w3.org/2000/svg", "svg");
201
- svg.setAttribute("xmlns", "http://www.w3.org/2000/svg");
202
- svg.setAttribute("viewBox", `0 0 ${XMLwidth} ${XMLheight}`);
203
- $container.append(svg);
204
- svg.setAttribute('class', 'textSelectionSVG');
205
- svg.setAttribute('preserveAspectRatio', 'none');
206
- $(svg).css({
207
- "width": "100%",
208
- "position": "absolute",
209
- "height": "100%",
210
- "top": "0",
211
- "left": "0",
289
+ const textLayer = createDIVPageLayer(pageContainer.page, 'BRtextLayer');
290
+ const ratioW = parseFloat(pageContainer.$container[0].style.width) / pageContainer.page.width;
291
+ const ratioH = parseFloat(pageContainer.$container[0].style.height) / pageContainer.page.height;
292
+ textLayer.style.transform = `scale(${ratioW}, ${ratioH})`;
293
+ textLayer.setAttribute("dir", this.rtl ? "rtl" : "ltr");
294
+
295
+ const ocrParagraphs = $(XMLpage).find("PARAGRAPH[coords]").toArray();
296
+ const paragEls = ocrParagraphs.map(p => {
297
+ const el = this.renderParagraph(p);
298
+ textLayer.appendChild(el);
299
+ return el;
212
300
  });
213
301
 
214
- $(XMLpage).find("PARAGRAPH").each((i, paragraph) => {
215
- // Adding text element for each paragraph in the page
216
- const words = $(paragraph).find("WORD");
217
- if (!words.length) return;
218
- const paragSvg = document.createElementNS("http://www.w3.org/2000/svg", this.svgParagraphElement);
219
- paragSvg.setAttribute("class", "BRparagElement");
220
- if (this.pointerEventsOnParagraph) {
221
- paragSvg.style.pointerEvents = "all";
222
- }
302
+ // Fix up paragraph positions
303
+ const paragraphRects = determineRealRects(textLayer, '.BRparagraphElement');
304
+ let yAdded = 0;
305
+ for (const [ocrParagraph, paragEl] of zip(ocrParagraphs, paragEls)) {
306
+ const ocrParagBounds = $(ocrParagraph).attr("coords").split(",").map(parseFloat);
307
+ const realRect = paragraphRects.get(paragEl);
308
+ const [ocrLeft, , ocrRight, ocrTop] = ocrParagBounds;
309
+ const newStartMargin = this.rtl ? (realRect.right - ocrRight) : (ocrLeft - realRect.left);
310
+ const newTop = ocrTop - (realRect.top + yAdded);
311
+
312
+ paragEl.style[this.rtl ? 'marginRight' : 'marginLeft'] = `${newStartMargin}px`;
313
+ paragEl.style.marginTop = `${newTop}px`;
314
+ yAdded += newTop;
315
+ textLayer.appendChild(paragEl);
316
+ }
317
+ $container.append(textLayer);
318
+ this.stopPageFlip($container);
319
+ }
320
+
321
+ /**
322
+ * @param {HTMLElement} ocrParagraph
323
+ * @returns {HTMLParagraphElement}
324
+ */
325
+ renderParagraph(ocrParagraph) {
326
+ const paragEl = document.createElement('p');
327
+ paragEl.classList.add('BRparagraphElement');
328
+ const [paragLeft, paragBottom, paragRight, paragTop] = $(ocrParagraph).attr("coords").split(",").map(parseFloat);
329
+ const wordHeightArr = [];
330
+ const lines = $(ocrParagraph).find("LINE[coords]").toArray();
331
+ if (!lines.length) return paragEl;
223
332
 
224
- const wordHeightArr = [];
225
333
 
226
- for (let i = 0; i < words.length; i++) {
227
- // Adding tspan for each word in paragraph
228
- const currWord = words[i];
229
- // eslint-disable-next-line no-unused-vars
230
- const [left, bottom, right, top] = $(currWord).attr("coords").split(',').map(parseFloat);
334
+ for (const [prevLine, line, nextLine] of lookAroundWindow(genMap(lines, augmentLine))) {
335
+ const isLastLineOfParagraph = line.ocrElement == lines[lines.length - 1];
336
+ const lineEl = document.createElement('span');
337
+ lineEl.classList.add('BRlineElement');
338
+
339
+ for (const [wordIndex, currWord] of line.words.entries()) {
340
+ const [, bottom, right, top] = $(currWord).attr("coords").split(',').map(parseFloat);
231
341
  const wordHeight = bottom - top;
232
342
  wordHeightArr.push(wordHeight);
233
343
 
234
- const wordTspan = document.createElementNS("http://www.w3.org/2000/svg", this.svgWordElement);
235
- wordTspan.setAttribute("class", "BRwordElement");
236
- wordTspan.setAttribute("x", left.toString());
237
- wordTspan.setAttribute("y", bottom.toString());
238
- wordTspan.setAttribute("textLength", (right - left).toString());
239
- wordTspan.setAttribute("lengthAdjust", "spacingAndGlyphs");
240
- wordTspan.textContent = currWord.textContent;
241
- paragSvg.appendChild(wordTspan);
242
-
243
- // Adding spaces after words except at the end of the paragraph
244
- // TODO: assumes left-to-right text
245
- if (i < words.length - 1) {
246
- const nextWord = words[i + 1];
247
- // eslint-disable-next-line no-unused-vars
248
- const [leftNext, bottomNext, rightNext, topNext] = $(nextWord).attr("coords").split(',').map(parseFloat);
249
- const spaceTspan = document.createElementNS("http://www.w3.org/2000/svg", this.svgWordElement);
250
- spaceTspan.setAttribute("class", "BRwordElement");
251
- spaceTspan.setAttribute("x", right.toString());
252
- spaceTspan.setAttribute("y", bottom.toString());
253
- if ((leftNext - right) > 0) spaceTspan.setAttribute("textLength", (leftNext - right).toString());
254
- spaceTspan.setAttribute("lengthAdjust", "spacingAndGlyphs");
255
- spaceTspan.textContent = " ";
256
- paragSvg.appendChild(spaceTspan);
344
+ if (wordIndex == 0 && prevLine?.lastWord.textContent.trim().endsWith('-')) {
345
+ // ideally prefer the next line to determine the left position,
346
+ // since the previous line could be the first line of the paragraph
347
+ // and hence have an incorrectly indented first word.
348
+ // E.g. https://archive.org/details/driitaleofdaring00bachuoft/page/360/mode/2up
349
+ const [newLeft, , , ] = $((nextLine || prevLine).firstWord).attr("coords").split(',').map(parseFloat);
350
+ $(currWord).attr("coords", `${newLeft},${bottom},${right},${top}`);
257
351
  }
258
352
 
259
- // Adds newline at the end of paragraph on Firefox
260
- if ((i == words.length - 1 && (this.insertNewlines))) {
261
- paragSvg.appendChild(document.createTextNode("\n"));
353
+ const wordEl = document.createElement('span');
354
+ wordEl.setAttribute("class", "BRwordElement");
355
+ wordEl.textContent = currWord.textContent.trim();
356
+
357
+ if (wordIndex > 0) {
358
+ const space = document.createElement('span');
359
+ space.classList.add('BRspace');
360
+ space.textContent = ' ';
361
+ lineEl.append(space);
362
+
363
+ // Edge ignores empty elements (like BRspace), so add another
364
+ // space to ensure Edge's ReadAloud works correctly.
365
+ lineEl.appendChild(document.createTextNode(' '));
262
366
  }
367
+
368
+ lineEl.appendChild(wordEl);
263
369
  }
264
370
 
265
- wordHeightArr.sort();
266
- const paragWordHeight = wordHeightArr[Math.floor(wordHeightArr.length * 0.85)];
267
- paragSvg.setAttribute("font-size", paragWordHeight.toString());
268
- svg.appendChild(paragSvg);
269
- })
270
- this.stopPageFlip($container);
371
+ const hasHyphen = line.lastWord.textContent.trim().endsWith('-');
372
+ const lastWordEl = lineEl.children[lineEl.children.length - 1];
373
+ if (hasHyphen && !isLastLineOfParagraph) {
374
+ lastWordEl.textContent = lastWordEl.textContent.trim().slice(0, -1);
375
+ lastWordEl.classList.add('BRwordElement--hyphen');
376
+ }
377
+
378
+ paragEl.appendChild(lineEl);
379
+ if (!isLastLineOfParagraph && !hasHyphen) {
380
+ // Edge does not correctly have spaces between the lines.
381
+ paragEl.appendChild(document.createTextNode(' '));
382
+ }
383
+ }
384
+
385
+ wordHeightArr.sort((a, b) => a - b);
386
+ const paragWordHeight = wordHeightArr[Math.floor(wordHeightArr.length * 0.85)] + 4;
387
+ paragEl.style.left = `${paragLeft}px`;
388
+ paragEl.style.top = `${paragTop}px`;
389
+ paragEl.style.width = `${paragRight - paragLeft}px`;
390
+ paragEl.style.height = `${paragBottom - paragTop}px`;
391
+ paragEl.style.fontSize = `${paragWordHeight}px`;
392
+
393
+ // Fix up sizes - stretch/crush words as necessary using letter spacing
394
+ let wordRects = determineRealRects(paragEl, '.BRwordElement');
395
+ const ocrWords = $(ocrParagraph).find("WORD").toArray();
396
+ const wordEls = paragEl.querySelectorAll('.BRwordElement');
397
+ for (const [ocrWord, wordEl] of zip(ocrWords, wordEls)) {
398
+ const realRect = wordRects.get(wordEl);
399
+ const [left, , right ] = $(ocrWord).attr("coords").split(',').map(parseFloat);
400
+ let ocrWidth = right - left;
401
+ // Some books (eg theworksofplato01platiala) have a space _inside_ the <WORD>
402
+ // element. That makes it impossible to determine the correct positining
403
+ // of everything, but to avoid the BRspace's being width 0, which makes selection
404
+ // janky on Chrome Android, assume the space is the same width as one of the
405
+ // letters.
406
+ if (ocrWord.textContent.endsWith(' ')) {
407
+ ocrWidth = ocrWidth * (ocrWord.textContent.length - 1) / ocrWord.textContent.length;
408
+ }
409
+ const diff = ocrWidth - realRect.width;
410
+ wordEl.style.letterSpacing = `${diff / (ocrWord.textContent.length - 1)}px`;
411
+ }
412
+
413
+ // Stretch/crush lines as necessary using line spacing
414
+ // Recompute rects after letter spacing
415
+ wordRects = determineRealRects(paragEl, '.BRwordElement');
416
+ const spaceRects = determineRealRects(paragEl, '.BRspace');
417
+
418
+ const ocrLines = $(ocrParagraph).find("LINE[coords]").toArray();
419
+ const lineEls = Array.from(paragEl.querySelectorAll('.BRlineElement'));
420
+
421
+ let ySoFar = paragTop;
422
+ for (const [ocrLine, lineEl] of zip(ocrLines, lineEls)) {
423
+ // shift words using marginLeft to align with the correct x position
424
+ const words = $(ocrLine).find("WORD").toArray();
425
+ // const ocrLineLeft = Math.min(...words.map(w => parseFloat($(w).attr("coords").split(',')[0])));
426
+ let xSoFar = this.rtl ? paragRight : paragLeft;
427
+ for (const [ocrWord, wordEl] of zip(words, lineEl.querySelectorAll('.BRwordElement'))) {
428
+ // start of line, need to compute the offset relative to the OCR words
429
+ const wordRect = wordRects.get(wordEl);
430
+ const [ocrLeft, , ocrRight ] = $(ocrWord).attr("coords").split(',').map(parseFloat);
431
+ const diff = (this.rtl ? -(ocrRight - xSoFar) : ocrLeft - xSoFar);
432
+
433
+ if (wordEl.previousElementSibling) {
434
+ const space = wordEl.previousElementSibling;
435
+ space.style.letterSpacing = `${diff - spaceRects.get(space).width}px`;
436
+ } else {
437
+ wordEl.style[this.rtl ? 'paddingRight' : 'paddingLeft'] = `${diff}px`;
438
+ }
439
+ if (this.rtl) xSoFar -= diff + wordRect.width;
440
+ else xSoFar += diff + wordRect.width;
441
+ }
442
+ // And also fix y position
443
+ const ocrLineTop = Math.min(...words.map(w => parseFloat($(w).attr("coords").split(',')[3])));
444
+ const diff = ocrLineTop - ySoFar;
445
+ if (lineEl.previousElementSibling) {
446
+ lineEl.previousElementSibling.style.lineHeight = `${diff}px`;
447
+ ySoFar += diff;
448
+ }
449
+ }
450
+
451
+ // The last line will have a line height subtracting from the paragraph height
452
+ lineEls[lineEls.length - 1].style.lineHeight = `${paragBottom - ySoFar}px`;
453
+
454
+ // Edge does not include a newline for some reason when copying/pasting the <p> els
455
+ paragEl.appendChild(document.createElement('br'));
456
+ return paragEl;
271
457
  }
272
458
  }
273
459
 
274
- export class BookreaderWithTextSelection extends BookReader {
275
- init() {
276
- const options = Object.assign({}, DEFAULT_OPTIONS, this.options.plugins.textSelection);
277
- if (options.enabled) {
278
- this.textSelectionPlugin = new TextSelectionPlugin(options, this.options.vars);
279
- // Write this back; this way the plugin is the source of truth, and BR just
280
- // contains a reference to it.
281
- this.options.plugins.textSelection = options;
282
- this.textSelectionPlugin.init();
460
+ BookReader?.registerPlugin('textSelection', TextSelectionPlugin);
461
+
462
+
463
+ /**
464
+ * @param {HTMLElement} parentEl
465
+ * @param {string} selector
466
+ * @returns {Map<Element, Rect>}
467
+ */
468
+ function determineRealRects(parentEl, selector) {
469
+ const initals = {
470
+ position: parentEl.style.position,
471
+ visibility: parentEl.style.visibility,
472
+ top: parentEl.style.top,
473
+ left: parentEl.style.left,
474
+ transform: parentEl.style.transform,
475
+ };
476
+ parentEl.style.position = 'absolute';
477
+ parentEl.style.visibility = 'hidden';
478
+ parentEl.style.top = '0';
479
+ parentEl.style.left = '0';
480
+ parentEl.style.transform = 'none';
481
+ document.body.appendChild(parentEl);
482
+ const rects = new Map(
483
+ Array.from(parentEl.querySelectorAll(selector))
484
+ .map(wordEl => {
485
+ const origRect = wordEl.getBoundingClientRect();
486
+ return [wordEl, new Rect(
487
+ origRect.left + window.scrollX,
488
+ origRect.top + window.scrollY,
489
+ origRect.width,
490
+ origRect.height,
491
+ )];
492
+ }),
493
+ );
494
+ document.body.removeChild(parentEl);
495
+ Object.assign(parentEl.style, initals);
496
+ return rects;
497
+ }
498
+
499
+ /**
500
+ * @param {HTMLElement} line
501
+ */
502
+ function augmentLine(line) {
503
+ const words = $(line).find("WORD").toArray();
504
+ return {
505
+ ocrElement: line,
506
+ words,
507
+ firstWord: words[0],
508
+ lastWord: words[words.length - 1],
509
+ };
510
+ }
511
+
512
+ /**
513
+ * @template TFrom, TTo
514
+ * Generator version of map
515
+ * @param {Iterable<TFrom>} gen
516
+ * @param {function(TFrom): TTo} fn
517
+ * @returns {Iterable<TTo>}
518
+ */
519
+ export function* genMap(gen, fn) {
520
+ for (const x of gen) yield fn(x);
521
+ }
522
+
523
+ /**
524
+ * @template T
525
+ * Generator that provides a sliding window of 3 elements,
526
+ * prev, current, and next.
527
+ * @param {Iterable<T>} gen
528
+ * @returns {Iterable<[T | undefined, T, T | undefined]>}
529
+ */
530
+ export function* lookAroundWindow(gen) {
531
+ let prev = undefined;
532
+ let cur = undefined;
533
+ let next = undefined;
534
+ for (const x of gen) {
535
+ if (typeof cur !== 'undefined') {
536
+ next = x;
537
+ yield [prev, cur, next];
283
538
  }
284
- super.init();
539
+ prev = cur;
540
+ cur = x;
541
+ next = undefined;
285
542
  }
286
543
 
544
+ if (typeof cur !== 'undefined') {
545
+ yield [prev, cur, next];
546
+ }
547
+ }
548
+
549
+ /**
550
+ * @template T1, T2
551
+ * Lazy zip implementation to avoid importing lodash
552
+ * Expects iterators to be of the same length
553
+ * @param {Iterable<T1>} gen1
554
+ * @param {Iterable<T2>} gen2
555
+ * @returns {Iterable<[T1, T2]>}
556
+ */
557
+ export function* zip(gen1, gen2) {
558
+ const it1 = gen1[Symbol.iterator]();
559
+ const it2 = gen2[Symbol.iterator]();
560
+ while (true) {
561
+ const r1 = it1.next();
562
+ const r2 = it2.next();
563
+ if (r1.done && r2.done) {
564
+ return;
565
+ }
566
+ if (r1.done || r2.done) {
567
+ throw new Error('zip: one of the iterators is done');
568
+ }
569
+ yield [r1.value, r2.value];
570
+ }
571
+ }
572
+
573
+ /**
574
+ * [left, bottom, right, top]
575
+ * @param {Array<[number, number, number, number]>} bounds
576
+ * @returns {[number, number, number, number]}
577
+ */
578
+ function determineBounds(bounds) {
579
+ let leftMost = Infinity;
580
+ let bottomMost = -Infinity;
581
+ let rightMost = -Infinity;
582
+ let topMost = Infinity;
583
+
584
+ for (const [left, bottom, right, top] of bounds) {
585
+ leftMost = Math.min(leftMost, left);
586
+ bottomMost = Math.max(bottomMost, bottom);
587
+ rightMost = Math.max(rightMost, right);
588
+ topMost = Math.min(topMost, top);
589
+ }
590
+
591
+ return [leftMost, bottomMost, rightMost, topMost];
592
+ }
593
+
594
+ /**
595
+ * Recursively traverses the XML tree and adds coords
596
+ * which are the bounding box of all child coords
597
+ * @param {Element} xmlEl
598
+ */
599
+ function recursivelyAddCoords(xmlEl) {
600
+ if ($(xmlEl).attr('coords') || !xmlEl.children) {
601
+ return;
602
+ }
603
+
604
+ const children = $(xmlEl).children().toArray();
605
+ if (children.length === 0) {
606
+ return;
607
+ }
608
+
609
+ for (const child of children) {
610
+ recursivelyAddCoords(child);
611
+ }
612
+
613
+ const childCoords = [];
614
+
615
+ for (const child of children) {
616
+ if (!$(child).attr('coords')) continue;
617
+ childCoords.push($(child).attr('coords').split(',').map(parseFloat));
618
+ }
619
+
620
+ const boundingCoords = determineBounds(childCoords);
621
+ if (Math.abs(boundingCoords[0]) != Infinity) {
622
+ $(xmlEl).attr('coords', boundingCoords.join(','));
623
+ }
624
+ }
625
+
626
+ /**
627
+ * Basically a polyfill for the native DOMRect class
628
+ */
629
+ class Rect {
287
630
  /**
288
- * @param {number} index
631
+ * @param {number} x
632
+ * @param {number} y
633
+ * @param {number} width
634
+ * @param {number} height
289
635
  */
290
- _createPageContainer(index) {
291
- const pageContainer = super._createPageContainer(index);
292
- // Disable if thumb mode; it's too janky
293
- // index can be -1 for "pre-cover" region
294
- // Added checking of lastPageIndex to avoid loop around index value
295
- const lastPageIndex = this.getNumLeafs() - 1;
296
- if (this.mode !== this.constModeThumb && (index >= 0 && index <= lastPageIndex)) {
297
- this.textSelectionPlugin?.createTextLayer(index, pageContainer.$container);
298
- }
299
- return pageContainer;
636
+ constructor(x, y, width, height) {
637
+ this.x = x;
638
+ this.y = y;
639
+ this.width = width;
640
+ this.height = height;
300
641
  }
642
+
643
+ get right() { return this.x + this.width; }
644
+ get bottom() { return this.y + this.height; }
645
+ get top() { return this.y; }
646
+ get left() { return this.x; }
301
647
  }
302
- window.BookReader = BookreaderWithTextSelection;
303
- export default BookreaderWithTextSelection;