@vaadin-component-factory/vcf-pdf-viewer 0.9.0 → 1.0.0

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 (172) hide show
  1. package/README.md +1 -1
  2. package/package.json +42 -26
  3. package/{src/display → pdfjs/dist}/display_utils.js +344 -139
  4. package/{src/display → pdfjs/dist}/fetch_stream.js +115 -97
  5. package/pdfjs/dist/l10n_utils.js +140 -0
  6. package/{src/shared → pdfjs/dist}/message_handler.js +243 -259
  7. package/{src/display → pdfjs/dist}/network.js +149 -87
  8. package/{src/display/content_disposition.js → pdfjs/dist/network_utils.js} +167 -55
  9. package/{src/display → pdfjs/dist}/node_stream.js +133 -98
  10. package/pdfjs/dist/pdf.js +12778 -0
  11. package/pdfjs/dist/pdf_link_service.js +638 -0
  12. package/pdfjs/dist/pdf_rendering_queue.js +199 -0
  13. package/pdfjs/dist/pdf_thumbnail_viewer.js +819 -0
  14. package/pdfjs/dist/pdf_viewer.js +3598 -0
  15. package/pdfjs/dist/ui_utils.js +1033 -0
  16. package/{src/shared → pdfjs/dist}/util.js +301 -287
  17. package/pdfjs/dist/worker.js +62813 -0
  18. package/src/vcf-pdf-viewer.js +98 -46
  19. package/theme/lumo/vcf-pdf-viewer-styles.js +4 -4
  20. package/theme/material/vcf-pdf-viewer-styles.js +4 -4
  21. package/theme/material/vcf-pdf-viewer.js +2 -2
  22. package/src/core/.eslintrc +0 -13
  23. package/src/core/annotation.js +0 -2948
  24. package/src/core/arithmetic_decoder.js +0 -182
  25. package/src/core/ascii_85_stream.js +0 -98
  26. package/src/core/ascii_hex_stream.js +0 -79
  27. package/src/core/base_stream.js +0 -110
  28. package/src/core/bidi.js +0 -438
  29. package/src/core/calibri_factors.js +0 -308
  30. package/src/core/catalog.js +0 -1459
  31. package/src/core/ccitt.js +0 -1062
  32. package/src/core/ccitt_stream.js +0 -60
  33. package/src/core/cff_font.js +0 -116
  34. package/src/core/cff_parser.js +0 -1949
  35. package/src/core/charsets.js +0 -119
  36. package/src/core/chunked_stream.js +0 -557
  37. package/src/core/cmap.js +0 -1039
  38. package/src/core/colorspace.js +0 -1533
  39. package/src/core/core_utils.js +0 -464
  40. package/src/core/crypto.js +0 -1900
  41. package/src/core/decode_stream.js +0 -170
  42. package/src/core/decrypt_stream.js +0 -59
  43. package/src/core/default_appearance.js +0 -99
  44. package/src/core/document.js +0 -1456
  45. package/src/core/encodings.js +0 -301
  46. package/src/core/evaluator.js +0 -4601
  47. package/src/core/file_spec.js +0 -108
  48. package/src/core/flate_stream.js +0 -402
  49. package/src/core/font_renderer.js +0 -882
  50. package/src/core/fonts.js +0 -3260
  51. package/src/core/fonts_utils.js +0 -221
  52. package/src/core/function.js +0 -1257
  53. package/src/core/glyf.js +0 -706
  54. package/src/core/glyphlist.js +0 -4558
  55. package/src/core/helvetica_factors.js +0 -353
  56. package/src/core/image.js +0 -802
  57. package/src/core/image_utils.js +0 -291
  58. package/src/core/jbig2.js +0 -2572
  59. package/src/core/jbig2_stream.js +0 -73
  60. package/src/core/jpeg_stream.js +0 -105
  61. package/src/core/jpg.js +0 -1416
  62. package/src/core/jpx.js +0 -2343
  63. package/src/core/jpx_stream.js +0 -87
  64. package/src/core/liberationsans_widths.js +0 -221
  65. package/src/core/lzw_stream.js +0 -150
  66. package/src/core/metadata_parser.js +0 -146
  67. package/src/core/metrics.js +0 -2970
  68. package/src/core/murmurhash3.js +0 -139
  69. package/src/core/myriadpro_factors.js +0 -290
  70. package/src/core/name_number_tree.js +0 -153
  71. package/src/core/object_loader.js +0 -149
  72. package/src/core/opentype_file_builder.js +0 -154
  73. package/src/core/operator_list.js +0 -734
  74. package/src/core/parser.js +0 -1416
  75. package/src/core/pattern.js +0 -985
  76. package/src/core/pdf_manager.js +0 -217
  77. package/src/core/predictor_stream.js +0 -238
  78. package/src/core/primitives.js +0 -402
  79. package/src/core/ps_parser.js +0 -272
  80. package/src/core/run_length_stream.js +0 -61
  81. package/src/core/segoeui_factors.js +0 -308
  82. package/src/core/standard_fonts.js +0 -817
  83. package/src/core/stream.js +0 -103
  84. package/src/core/struct_tree.js +0 -335
  85. package/src/core/to_unicode_map.js +0 -103
  86. package/src/core/type1_font.js +0 -421
  87. package/src/core/type1_parser.js +0 -776
  88. package/src/core/unicode.js +0 -1649
  89. package/src/core/worker.js +0 -848
  90. package/src/core/worker_stream.js +0 -135
  91. package/src/core/writer.js +0 -278
  92. package/src/core/xfa/bind.js +0 -652
  93. package/src/core/xfa/builder.js +0 -207
  94. package/src/core/xfa/config.js +0 -1926
  95. package/src/core/xfa/connection_set.js +0 -202
  96. package/src/core/xfa/data.js +0 -82
  97. package/src/core/xfa/datasets.js +0 -76
  98. package/src/core/xfa/factory.js +0 -111
  99. package/src/core/xfa/fonts.js +0 -181
  100. package/src/core/xfa/formcalc_lexer.js +0 -385
  101. package/src/core/xfa/formcalc_parser.js +0 -1340
  102. package/src/core/xfa/html_utils.js +0 -639
  103. package/src/core/xfa/layout.js +0 -383
  104. package/src/core/xfa/locale_set.js +0 -345
  105. package/src/core/xfa/namespaces.js +0 -81
  106. package/src/core/xfa/parser.js +0 -184
  107. package/src/core/xfa/setup.js +0 -38
  108. package/src/core/xfa/signature.js +0 -40
  109. package/src/core/xfa/som.js +0 -338
  110. package/src/core/xfa/stylesheet.js +0 -40
  111. package/src/core/xfa/template.js +0 -6260
  112. package/src/core/xfa/text.js +0 -290
  113. package/src/core/xfa/unknown.js +0 -29
  114. package/src/core/xfa/utils.js +0 -217
  115. package/src/core/xfa/xdp.js +0 -59
  116. package/src/core/xfa/xfa_object.js +0 -1130
  117. package/src/core/xfa/xhtml.js +0 -543
  118. package/src/core/xfa_fonts.js +0 -208
  119. package/src/core/xml_parser.js +0 -507
  120. package/src/core/xref.js +0 -899
  121. package/src/display/annotation_layer.js +0 -2107
  122. package/src/display/annotation_storage.js +0 -113
  123. package/src/display/api.js +0 -3292
  124. package/src/display/base_factory.js +0 -180
  125. package/src/display/canvas.js +0 -2828
  126. package/src/display/font_loader.js +0 -484
  127. package/src/display/metadata.js +0 -41
  128. package/src/display/network_utils.js +0 -100
  129. package/src/display/node_utils.js +0 -83
  130. package/src/display/optional_content_config.js +0 -189
  131. package/src/display/pattern_helper.js +0 -659
  132. package/src/display/svg.js +0 -1709
  133. package/src/display/text_layer.js +0 -847
  134. package/src/display/transport_stream.js +0 -303
  135. package/src/display/worker_options.js +0 -40
  136. package/src/display/xfa_layer.js +0 -204
  137. package/src/doc_helper.js +0 -25
  138. package/src/images/logo.svg +0 -41
  139. package/src/interfaces.js +0 -169
  140. package/src/license_header.js +0 -14
  141. package/src/license_header_libre.js +0 -21
  142. package/src/pdf.image_decoders.js +0 -46
  143. package/src/pdf.js +0 -146
  144. package/src/pdf.sandbox.external.js +0 -181
  145. package/src/pdf.sandbox.js +0 -151
  146. package/src/pdf.scripting.js +0 -25
  147. package/src/pdf.worker.entry.js +0 -19
  148. package/src/pdf.worker.js +0 -23
  149. package/src/scripting_api/aform.js +0 -608
  150. package/src/scripting_api/app.js +0 -621
  151. package/src/scripting_api/color.js +0 -129
  152. package/src/scripting_api/common.js +0 -58
  153. package/src/scripting_api/console.js +0 -38
  154. package/src/scripting_api/constants.js +0 -208
  155. package/src/scripting_api/doc.js +0 -1195
  156. package/src/scripting_api/error.js +0 -23
  157. package/src/scripting_api/event.js +0 -232
  158. package/src/scripting_api/field.js +0 -620
  159. package/src/scripting_api/fullscreen.js +0 -145
  160. package/src/scripting_api/initialization.js +0 -223
  161. package/src/scripting_api/pdf_object.js +0 -24
  162. package/src/scripting_api/print_params.js +0 -146
  163. package/src/scripting_api/proxy.js +0 -139
  164. package/src/scripting_api/thermometer.js +0 -69
  165. package/src/scripting_api/util.js +0 -581
  166. package/src/shared/.eslintrc +0 -13
  167. package/src/shared/cffStandardStrings.js +0 -311
  168. package/src/shared/compatibility.js +0 -114
  169. package/src/shared/fonts_utils.js +0 -429
  170. package/src/shared/is_node.js +0 -27
  171. package/src/shared/scripting_utils.js +0 -85
  172. package/src/worker_loader.js +0 -32
@@ -0,0 +1,819 @@
1
+ import { getOutputScale, watchScroll, getVisibleElements, scrollIntoView, isValidRotation } from './ui_utils.js';
2
+ import './pdf.js';
3
+ import { RenderingStates } from './pdf_rendering_queue.js';
4
+ import { R as RenderingCancelledException } from './display_utils.js';
5
+ import './util.js';
6
+ import './message_handler.js';
7
+
8
+ /* Copyright 2012 Mozilla Foundation
9
+ *
10
+ * Licensed under the Apache License, Version 2.0 (the "License");
11
+ * you may not use this file except in compliance with the License.
12
+ * You may obtain a copy of the License at
13
+ *
14
+ * http://www.apache.org/licenses/LICENSE-2.0
15
+ *
16
+ * Unless required by applicable law or agreed to in writing, software
17
+ * distributed under the License is distributed on an "AS IS" BASIS,
18
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
19
+ * See the License for the specific language governing permissions and
20
+ * limitations under the License.
21
+ */
22
+ const DRAW_UPSCALE_FACTOR = 2; // See comment in `PDFThumbnailView.draw` below.
23
+
24
+ const MAX_NUM_SCALING_STEPS = 3;
25
+ const THUMBNAIL_CANVAS_BORDER_WIDTH = 1; // px
26
+
27
+ const THUMBNAIL_WIDTH = 98; // px
28
+
29
+ /**
30
+ * @typedef {Object} PDFThumbnailViewOptions
31
+ * @property {HTMLDivElement} container - The viewer element.
32
+ * @property {number} id - The thumbnail's unique ID (normally its number).
33
+ * @property {PageViewport} defaultViewport - The page viewport.
34
+ * @property {Promise<OptionalContentConfig>} [optionalContentConfigPromise] -
35
+ * A promise that is resolved with an {@link OptionalContentConfig} instance.
36
+ * The default value is `null`.
37
+ * @property {IPDFLinkService} linkService - The navigation/linking service.
38
+ * @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
39
+ * @property {function} checkSetImageDisabled
40
+ * @property {IL10n} l10n - Localization service.
41
+ */
42
+
43
+ const TempImageFactory = function TempImageFactoryClosure() {
44
+ let tempCanvasCache = null;
45
+ return {
46
+ getCanvas(width, height) {
47
+ let tempCanvas = tempCanvasCache;
48
+
49
+ if (!tempCanvas) {
50
+ tempCanvas = document.createElement("canvas");
51
+ tempCanvasCache = tempCanvas;
52
+ }
53
+
54
+ tempCanvas.width = width;
55
+ tempCanvas.height = height; // Since this is a temporary canvas, we need to fill it with a white
56
+ // background ourselves. `_getPageDrawContext` uses CSS rules for this.
57
+
58
+ if (typeof PDFJSDev === "undefined" || PDFJSDev.test("MOZCENTRAL || GENERIC")) {
59
+ tempCanvas.mozOpaque = true;
60
+ }
61
+
62
+ const ctx = tempCanvas.getContext("2d", {
63
+ alpha: false
64
+ });
65
+ ctx.save();
66
+ ctx.fillStyle = "rgb(255, 255, 255)";
67
+ ctx.fillRect(0, 0, width, height);
68
+ ctx.restore();
69
+ return [tempCanvas, tempCanvas.getContext("2d")];
70
+ },
71
+
72
+ destroyCanvas() {
73
+ const tempCanvas = tempCanvasCache;
74
+
75
+ if (tempCanvas) {
76
+ // Zeroing the width and height causes Firefox to release graphics
77
+ // resources immediately, which can greatly reduce memory consumption.
78
+ tempCanvas.width = 0;
79
+ tempCanvas.height = 0;
80
+ }
81
+
82
+ tempCanvasCache = null;
83
+ }
84
+
85
+ };
86
+ }();
87
+ /**
88
+ * @implements {IRenderableView}
89
+ */
90
+
91
+
92
+ class PDFThumbnailView {
93
+ /**
94
+ * @param {PDFThumbnailViewOptions} options
95
+ */
96
+ constructor({
97
+ container,
98
+ id,
99
+ defaultViewport,
100
+ optionalContentConfigPromise,
101
+ linkService,
102
+ renderingQueue,
103
+ checkSetImageDisabled,
104
+ l10n
105
+ }) {
106
+ this.id = id;
107
+ this.renderingId = "thumbnail" + id;
108
+ this.pageLabel = null;
109
+ this.pdfPage = null;
110
+ this.rotation = 0;
111
+ this.viewport = defaultViewport;
112
+ this.pdfPageRotate = defaultViewport.rotation;
113
+ this._optionalContentConfigPromise = optionalContentConfigPromise || null;
114
+ this.linkService = linkService;
115
+ this.renderingQueue = renderingQueue;
116
+ this.renderTask = null;
117
+ this.renderingState = RenderingStates.INITIAL;
118
+ this.resume = null;
119
+
120
+ this._checkSetImageDisabled = checkSetImageDisabled || function () {
121
+ return false;
122
+ };
123
+
124
+ const pageWidth = this.viewport.width,
125
+ pageHeight = this.viewport.height,
126
+ pageRatio = pageWidth / pageHeight;
127
+ this.canvasWidth = THUMBNAIL_WIDTH;
128
+ this.canvasHeight = this.canvasWidth / pageRatio | 0;
129
+ this.scale = this.canvasWidth / pageWidth;
130
+ this.l10n = l10n;
131
+ const anchor = document.createElement("a");
132
+ anchor.href = linkService.getAnchorUrl("#page=" + id);
133
+
134
+ this._thumbPageTitle.then(msg => {
135
+ anchor.title = msg;
136
+ });
137
+
138
+ anchor.onclick = function () {
139
+ linkService.goToPage(id);
140
+ return false;
141
+ };
142
+
143
+ this.anchor = anchor;
144
+ const div = document.createElement("div");
145
+ div.className = "thumbnail";
146
+ div.setAttribute("data-page-number", this.id);
147
+ this.div = div;
148
+ const ring = document.createElement("div");
149
+ ring.className = "thumbnailSelectionRing";
150
+ const borderAdjustment = 2 * THUMBNAIL_CANVAS_BORDER_WIDTH;
151
+ ring.style.width = this.canvasWidth + borderAdjustment + "px";
152
+ ring.style.height = this.canvasHeight + borderAdjustment + "px";
153
+ this.ring = ring;
154
+ div.appendChild(ring);
155
+ anchor.appendChild(div);
156
+ container.appendChild(anchor);
157
+ }
158
+
159
+ setPdfPage(pdfPage) {
160
+ this.pdfPage = pdfPage;
161
+ this.pdfPageRotate = pdfPage.rotate;
162
+ const totalRotation = (this.rotation + this.pdfPageRotate) % 360;
163
+ this.viewport = pdfPage.getViewport({
164
+ scale: 1,
165
+ rotation: totalRotation
166
+ });
167
+ this.reset();
168
+ }
169
+
170
+ reset() {
171
+ this.cancelRendering();
172
+ this.renderingState = RenderingStates.INITIAL;
173
+ const pageWidth = this.viewport.width,
174
+ pageHeight = this.viewport.height,
175
+ pageRatio = pageWidth / pageHeight;
176
+ this.canvasHeight = this.canvasWidth / pageRatio | 0;
177
+ this.scale = this.canvasWidth / pageWidth;
178
+ this.div.removeAttribute("data-loaded");
179
+ const ring = this.ring;
180
+ ring.textContent = ""; // Remove the thumbnail from the DOM.
181
+
182
+ const borderAdjustment = 2 * THUMBNAIL_CANVAS_BORDER_WIDTH;
183
+ ring.style.width = this.canvasWidth + borderAdjustment + "px";
184
+ ring.style.height = this.canvasHeight + borderAdjustment + "px";
185
+
186
+ if (this.canvas) {
187
+ // Zeroing the width and height causes Firefox to release graphics
188
+ // resources immediately, which can greatly reduce memory consumption.
189
+ this.canvas.width = 0;
190
+ this.canvas.height = 0;
191
+ delete this.canvas;
192
+ }
193
+
194
+ if (this.image) {
195
+ this.image.removeAttribute("src");
196
+ delete this.image;
197
+ }
198
+ }
199
+
200
+ update(rotation) {
201
+ if (typeof rotation !== "undefined") {
202
+ this.rotation = rotation;
203
+ }
204
+
205
+ const totalRotation = (this.rotation + this.pdfPageRotate) % 360;
206
+ this.viewport = this.viewport.clone({
207
+ scale: 1,
208
+ rotation: totalRotation
209
+ });
210
+ this.reset();
211
+ }
212
+ /**
213
+ * PLEASE NOTE: Most likely you want to use the `this.reset()` method,
214
+ * rather than calling this one directly.
215
+ */
216
+
217
+
218
+ cancelRendering() {
219
+ if (this.renderTask) {
220
+ this.renderTask.cancel();
221
+ this.renderTask = null;
222
+ }
223
+
224
+ this.resume = null;
225
+ }
226
+ /**
227
+ * @private
228
+ */
229
+
230
+
231
+ _getPageDrawContext(upscaleFactor = 1) {
232
+ // Keep the no-thumbnail outline visible, i.e. `data-loaded === false`,
233
+ // until rendering/image conversion is complete, to avoid display issues.
234
+ const canvas = document.createElement("canvas");
235
+
236
+ if (typeof PDFJSDev === "undefined" || PDFJSDev.test("MOZCENTRAL || GENERIC")) {
237
+ canvas.mozOpaque = true;
238
+ }
239
+
240
+ const ctx = canvas.getContext("2d", {
241
+ alpha: false
242
+ });
243
+ const outputScale = getOutputScale(ctx);
244
+ canvas.width = upscaleFactor * this.canvasWidth * outputScale.sx | 0;
245
+ canvas.height = upscaleFactor * this.canvasHeight * outputScale.sy | 0;
246
+ const transform = outputScale.scaled ? [outputScale.sx, 0, 0, outputScale.sy, 0, 0] : null;
247
+ return {
248
+ ctx,
249
+ canvas,
250
+ transform
251
+ };
252
+ }
253
+ /**
254
+ * @private
255
+ */
256
+
257
+
258
+ _convertCanvasToImage(canvas) {
259
+ if (this.renderingState !== RenderingStates.FINISHED) {
260
+ throw new Error("_convertCanvasToImage: Rendering has not finished.");
261
+ }
262
+
263
+ const reducedCanvas = this._reduceImage(canvas);
264
+
265
+ const image = document.createElement("img");
266
+ image.className = "thumbnailImage";
267
+
268
+ this._thumbPageCanvas.then(msg => {
269
+ image.setAttribute("aria-label", msg);
270
+ });
271
+
272
+ image.style.width = this.canvasWidth + "px";
273
+ image.style.height = this.canvasHeight + "px";
274
+ image.src = reducedCanvas.toDataURL();
275
+ this.image = image;
276
+ this.div.setAttribute("data-loaded", true);
277
+ this.ring.appendChild(image); // Zeroing the width and height causes Firefox to release graphics
278
+ // resources immediately, which can greatly reduce memory consumption.
279
+
280
+ reducedCanvas.width = 0;
281
+ reducedCanvas.height = 0;
282
+ }
283
+
284
+ draw() {
285
+ if (this.renderingState !== RenderingStates.INITIAL) {
286
+ console.error("Must be in new state before drawing");
287
+ return Promise.resolve(undefined);
288
+ }
289
+
290
+ const {
291
+ pdfPage
292
+ } = this;
293
+
294
+ if (!pdfPage) {
295
+ this.renderingState = RenderingStates.FINISHED;
296
+ return Promise.reject(new Error("pdfPage is not loaded"));
297
+ }
298
+
299
+ this.renderingState = RenderingStates.RUNNING;
300
+
301
+ const finishRenderTask = async (error = null) => {
302
+ // The renderTask may have been replaced by a new one, so only remove
303
+ // the reference to the renderTask if it matches the one that is
304
+ // triggering this callback.
305
+ if (renderTask === this.renderTask) {
306
+ this.renderTask = null;
307
+ }
308
+
309
+ if (error instanceof RenderingCancelledException) {
310
+ return;
311
+ }
312
+
313
+ this.renderingState = RenderingStates.FINISHED;
314
+
315
+ this._convertCanvasToImage(canvas);
316
+
317
+ if (error) {
318
+ throw error;
319
+ }
320
+ }; // Render the thumbnail at a larger size and downsize the canvas (similar
321
+ // to `setImage`), to improve consistency between thumbnails created by
322
+ // the `draw` and `setImage` methods (fixes issue 8233).
323
+ // NOTE: To primarily avoid increasing memory usage too much, but also to
324
+ // reduce downsizing overhead, we purposely limit the up-scaling factor.
325
+
326
+
327
+ const {
328
+ ctx,
329
+ canvas,
330
+ transform
331
+ } = this._getPageDrawContext(DRAW_UPSCALE_FACTOR);
332
+
333
+ const drawViewport = this.viewport.clone({
334
+ scale: DRAW_UPSCALE_FACTOR * this.scale
335
+ });
336
+
337
+ const renderContinueCallback = cont => {
338
+ if (!this.renderingQueue.isHighestPriority(this)) {
339
+ this.renderingState = RenderingStates.PAUSED;
340
+
341
+ this.resume = () => {
342
+ this.renderingState = RenderingStates.RUNNING;
343
+ cont();
344
+ };
345
+
346
+ return;
347
+ }
348
+
349
+ cont();
350
+ };
351
+
352
+ const renderContext = {
353
+ canvasContext: ctx,
354
+ transform,
355
+ viewport: drawViewport,
356
+ optionalContentConfigPromise: this._optionalContentConfigPromise
357
+ };
358
+ const renderTask = this.renderTask = pdfPage.render(renderContext);
359
+ renderTask.onContinue = renderContinueCallback;
360
+ const resultPromise = renderTask.promise.then(function () {
361
+ return finishRenderTask(null);
362
+ }, function (error) {
363
+ return finishRenderTask(error);
364
+ });
365
+ resultPromise.finally(() => {
366
+ // Zeroing the width and height causes Firefox to release graphics
367
+ // resources immediately, which can greatly reduce memory consumption.
368
+ canvas.width = 0;
369
+ canvas.height = 0; // Only trigger cleanup, once rendering has finished, when the current
370
+ // pageView is *not* cached on the `BaseViewer`-instance.
371
+
372
+ const pageCached = this.linkService.isPageCached(this.id);
373
+
374
+ if (!pageCached) {
375
+ var _this$pdfPage;
376
+
377
+ (_this$pdfPage = this.pdfPage) === null || _this$pdfPage === void 0 ? void 0 : _this$pdfPage.cleanup();
378
+ }
379
+ });
380
+ return resultPromise;
381
+ }
382
+
383
+ setImage(pageView) {
384
+ if (this._checkSetImageDisabled()) {
385
+ return;
386
+ }
387
+
388
+ if (this.renderingState !== RenderingStates.INITIAL) {
389
+ return;
390
+ }
391
+
392
+ const {
393
+ canvas,
394
+ pdfPage
395
+ } = pageView;
396
+
397
+ if (!canvas) {
398
+ return;
399
+ }
400
+
401
+ if (!this.pdfPage) {
402
+ this.setPdfPage(pdfPage);
403
+ }
404
+
405
+ this.renderingState = RenderingStates.FINISHED;
406
+
407
+ this._convertCanvasToImage(canvas);
408
+ }
409
+ /**
410
+ * @private
411
+ */
412
+
413
+
414
+ _reduceImage(img) {
415
+ const {
416
+ ctx,
417
+ canvas
418
+ } = this._getPageDrawContext();
419
+
420
+ if (img.width <= 2 * canvas.width) {
421
+ ctx.drawImage(img, 0, 0, img.width, img.height, 0, 0, canvas.width, canvas.height);
422
+ return canvas;
423
+ } // drawImage does an awful job of rescaling the image, doing it gradually.
424
+
425
+
426
+ let reducedWidth = canvas.width << MAX_NUM_SCALING_STEPS;
427
+ let reducedHeight = canvas.height << MAX_NUM_SCALING_STEPS;
428
+ const [reducedImage, reducedImageCtx] = TempImageFactory.getCanvas(reducedWidth, reducedHeight);
429
+
430
+ while (reducedWidth > img.width || reducedHeight > img.height) {
431
+ reducedWidth >>= 1;
432
+ reducedHeight >>= 1;
433
+ }
434
+
435
+ reducedImageCtx.drawImage(img, 0, 0, img.width, img.height, 0, 0, reducedWidth, reducedHeight);
436
+
437
+ while (reducedWidth > 2 * canvas.width) {
438
+ reducedImageCtx.drawImage(reducedImage, 0, 0, reducedWidth, reducedHeight, 0, 0, reducedWidth >> 1, reducedHeight >> 1);
439
+ reducedWidth >>= 1;
440
+ reducedHeight >>= 1;
441
+ }
442
+
443
+ ctx.drawImage(reducedImage, 0, 0, reducedWidth, reducedHeight, 0, 0, canvas.width, canvas.height);
444
+ return canvas;
445
+ }
446
+
447
+ get _thumbPageTitle() {
448
+ var _this$pageLabel;
449
+
450
+ return this.l10n.get("thumb_page_title", {
451
+ page: (_this$pageLabel = this.pageLabel) !== null && _this$pageLabel !== void 0 ? _this$pageLabel : this.id
452
+ });
453
+ }
454
+
455
+ get _thumbPageCanvas() {
456
+ var _this$pageLabel2;
457
+
458
+ return this.l10n.get("thumb_page_canvas", {
459
+ page: (_this$pageLabel2 = this.pageLabel) !== null && _this$pageLabel2 !== void 0 ? _this$pageLabel2 : this.id
460
+ });
461
+ }
462
+ /**
463
+ * @param {string|null} label
464
+ */
465
+
466
+
467
+ setPageLabel(label) {
468
+ this.pageLabel = typeof label === "string" ? label : null;
469
+
470
+ this._thumbPageTitle.then(msg => {
471
+ this.anchor.title = msg;
472
+ });
473
+
474
+ if (this.renderingState !== RenderingStates.FINISHED) {
475
+ return;
476
+ }
477
+
478
+ this._thumbPageCanvas.then(msg => {
479
+ var _this$image;
480
+
481
+ (_this$image = this.image) === null || _this$image === void 0 ? void 0 : _this$image.setAttribute("aria-label", msg);
482
+ });
483
+ }
484
+
485
+ }
486
+
487
+ /* Copyright 2012 Mozilla Foundation
488
+ *
489
+ * Licensed under the Apache License, Version 2.0 (the "License");
490
+ * you may not use this file except in compliance with the License.
491
+ * You may obtain a copy of the License at
492
+ *
493
+ * http://www.apache.org/licenses/LICENSE-2.0
494
+ *
495
+ * Unless required by applicable law or agreed to in writing, software
496
+ * distributed under the License is distributed on an "AS IS" BASIS,
497
+ * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
498
+ * See the License for the specific language governing permissions and
499
+ * limitations under the License.
500
+ */
501
+ const THUMBNAIL_SCROLL_MARGIN = -19;
502
+ const THUMBNAIL_SELECTED_CLASS = "selected";
503
+ /**
504
+ * @typedef {Object} PDFThumbnailViewerOptions
505
+ * @property {HTMLDivElement} container - The container for the thumbnail
506
+ * elements.
507
+ * @property {EventBus} eventBus - The application event bus.
508
+ * @property {IPDFLinkService} linkService - The navigation/linking service.
509
+ * @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
510
+ * @property {IL10n} l10n - Localization service.
511
+ */
512
+
513
+ /**
514
+ * Viewer control to display thumbnails for pages in a PDF document.
515
+ *
516
+ * @implements {IRenderableView}
517
+ */
518
+
519
+ class PDFThumbnailViewer {
520
+ /**
521
+ * @param {PDFThumbnailViewerOptions} options
522
+ */
523
+ constructor({
524
+ container,
525
+ eventBus,
526
+ linkService,
527
+ renderingQueue,
528
+ l10n
529
+ }) {
530
+ this.container = container;
531
+ this.linkService = linkService;
532
+ this.renderingQueue = renderingQueue;
533
+ this.l10n = l10n;
534
+ this.scroll = watchScroll(this.container, this._scrollUpdated.bind(this));
535
+
536
+ this._resetView();
537
+
538
+ eventBus._on("optionalcontentconfigchanged", () => {
539
+ // Ensure that the thumbnails always render with the *default* optional
540
+ // content configuration.
541
+ this._setImageDisabled = true;
542
+ });
543
+ }
544
+ /**
545
+ * @private
546
+ */
547
+
548
+
549
+ _scrollUpdated() {
550
+ this.renderingQueue.renderHighestPriority();
551
+ }
552
+
553
+ getThumbnail(index) {
554
+ return this._thumbnails[index];
555
+ }
556
+ /**
557
+ * @private
558
+ */
559
+
560
+
561
+ _getVisibleThumbs() {
562
+ return getVisibleElements({
563
+ scrollEl: this.container,
564
+ views: this._thumbnails
565
+ });
566
+ }
567
+
568
+ scrollThumbnailIntoView(pageNumber) {
569
+ if (!this.pdfDocument) {
570
+ return;
571
+ }
572
+
573
+ const thumbnailView = this._thumbnails[pageNumber - 1];
574
+
575
+ if (!thumbnailView) {
576
+ console.error('scrollThumbnailIntoView: Invalid "pageNumber" parameter.');
577
+ return;
578
+ }
579
+
580
+ if (pageNumber !== this._currentPageNumber) {
581
+ const prevThumbnailView = this._thumbnails[this._currentPageNumber - 1]; // Remove the highlight from the previous thumbnail...
582
+
583
+ prevThumbnailView.div.classList.remove(THUMBNAIL_SELECTED_CLASS); // ... and add the highlight to the new thumbnail.
584
+
585
+ thumbnailView.div.classList.add(THUMBNAIL_SELECTED_CLASS);
586
+ }
587
+
588
+ const visibleThumbs = this._getVisibleThumbs();
589
+
590
+ const numVisibleThumbs = visibleThumbs.views.length; // If the thumbnail isn't currently visible, scroll it into view.
591
+
592
+ if (numVisibleThumbs > 0) {
593
+ const first = visibleThumbs.first.id; // Account for only one thumbnail being visible.
594
+
595
+ const last = numVisibleThumbs > 1 ? visibleThumbs.last.id : first;
596
+ let shouldScroll = false;
597
+
598
+ if (pageNumber <= first || pageNumber >= last) {
599
+ shouldScroll = true;
600
+ } else {
601
+ visibleThumbs.views.some(function (view) {
602
+ if (view.id !== pageNumber) {
603
+ return false;
604
+ }
605
+
606
+ shouldScroll = view.percent < 100;
607
+ return true;
608
+ });
609
+ }
610
+
611
+ if (shouldScroll) {
612
+ scrollIntoView(thumbnailView.div, {
613
+ top: THUMBNAIL_SCROLL_MARGIN
614
+ });
615
+ }
616
+ }
617
+
618
+ this._currentPageNumber = pageNumber;
619
+ }
620
+
621
+ get pagesRotation() {
622
+ return this._pagesRotation;
623
+ }
624
+
625
+ set pagesRotation(rotation) {
626
+ if (!isValidRotation(rotation)) {
627
+ throw new Error("Invalid thumbnails rotation angle.");
628
+ }
629
+
630
+ if (!this.pdfDocument) {
631
+ return;
632
+ }
633
+
634
+ if (this._pagesRotation === rotation) {
635
+ return; // The rotation didn't change.
636
+ }
637
+
638
+ this._pagesRotation = rotation;
639
+
640
+ for (let i = 0, ii = this._thumbnails.length; i < ii; i++) {
641
+ this._thumbnails[i].update(rotation);
642
+ }
643
+ }
644
+
645
+ cleanup() {
646
+ for (let i = 0, ii = this._thumbnails.length; i < ii; i++) {
647
+ if (this._thumbnails[i] && this._thumbnails[i].renderingState !== RenderingStates.FINISHED) {
648
+ this._thumbnails[i].reset();
649
+ }
650
+ }
651
+
652
+ TempImageFactory.destroyCanvas();
653
+ }
654
+ /**
655
+ * @private
656
+ */
657
+
658
+
659
+ _resetView() {
660
+ this._thumbnails = [];
661
+ this._currentPageNumber = 1;
662
+ this._pageLabels = null;
663
+ this._pagesRotation = 0;
664
+ this._optionalContentConfigPromise = null;
665
+ this._pagesRequests = new WeakMap();
666
+ this._setImageDisabled = false; // Remove the thumbnails from the DOM.
667
+
668
+ this.container.textContent = "";
669
+ }
670
+
671
+ setDocument(pdfDocument) {
672
+ if (this.pdfDocument) {
673
+ this._cancelRendering();
674
+
675
+ this._resetView();
676
+ }
677
+
678
+ this.pdfDocument = pdfDocument;
679
+
680
+ if (!pdfDocument) {
681
+ return;
682
+ }
683
+
684
+ const firstPagePromise = pdfDocument.getPage(1);
685
+ const optionalContentConfigPromise = pdfDocument.getOptionalContentConfig();
686
+ firstPagePromise.then(firstPdfPage => {
687
+ this._optionalContentConfigPromise = optionalContentConfigPromise;
688
+ const pagesCount = pdfDocument.numPages;
689
+ const viewport = firstPdfPage.getViewport({
690
+ scale: 1
691
+ });
692
+
693
+ const checkSetImageDisabled = () => {
694
+ return this._setImageDisabled;
695
+ };
696
+
697
+ for (let pageNum = 1; pageNum <= pagesCount; ++pageNum) {
698
+ const thumbnail = new PDFThumbnailView({
699
+ container: this.container,
700
+ id: pageNum,
701
+ defaultViewport: viewport.clone(),
702
+ optionalContentConfigPromise,
703
+ linkService: this.linkService,
704
+ renderingQueue: this.renderingQueue,
705
+ checkSetImageDisabled,
706
+ l10n: this.l10n
707
+ });
708
+
709
+ this._thumbnails.push(thumbnail);
710
+ } // Set the first `pdfPage` immediately, since it's already loaded,
711
+ // rather than having to repeat the `PDFDocumentProxy.getPage` call in
712
+ // the `this._ensurePdfPageLoaded` method before rendering can start.
713
+
714
+
715
+ const firstThumbnailView = this._thumbnails[0];
716
+
717
+ if (firstThumbnailView) {
718
+ firstThumbnailView.setPdfPage(firstPdfPage);
719
+ } // Ensure that the current thumbnail is always highlighted on load.
720
+
721
+
722
+ const thumbnailView = this._thumbnails[this._currentPageNumber - 1];
723
+ thumbnailView.div.classList.add(THUMBNAIL_SELECTED_CLASS);
724
+ }).catch(reason => {
725
+ console.error("Unable to initialize thumbnail viewer", reason);
726
+ });
727
+ }
728
+ /**
729
+ * @private
730
+ */
731
+
732
+
733
+ _cancelRendering() {
734
+ for (let i = 0, ii = this._thumbnails.length; i < ii; i++) {
735
+ if (this._thumbnails[i]) {
736
+ this._thumbnails[i].cancelRendering();
737
+ }
738
+ }
739
+ }
740
+ /**
741
+ * @param {Array|null} labels
742
+ */
743
+
744
+
745
+ setPageLabels(labels) {
746
+ if (!this.pdfDocument) {
747
+ return;
748
+ }
749
+
750
+ if (!labels) {
751
+ this._pageLabels = null;
752
+ } else if (!(Array.isArray(labels) && this.pdfDocument.numPages === labels.length)) {
753
+ this._pageLabels = null;
754
+ console.error("PDFThumbnailViewer_setPageLabels: Invalid page labels.");
755
+ } else {
756
+ this._pageLabels = labels;
757
+ } // Update all the `PDFThumbnailView` instances.
758
+
759
+
760
+ for (let i = 0, ii = this._thumbnails.length; i < ii; i++) {
761
+ var _this$_pageLabels$i, _this$_pageLabels;
762
+
763
+ this._thumbnails[i].setPageLabel((_this$_pageLabels$i = (_this$_pageLabels = this._pageLabels) === null || _this$_pageLabels === void 0 ? void 0 : _this$_pageLabels[i]) !== null && _this$_pageLabels$i !== void 0 ? _this$_pageLabels$i : null);
764
+ }
765
+ }
766
+ /**
767
+ * @param {PDFThumbnailView} thumbView
768
+ * @returns {PDFPage}
769
+ * @private
770
+ */
771
+
772
+
773
+ _ensurePdfPageLoaded(thumbView) {
774
+ if (thumbView.pdfPage) {
775
+ return Promise.resolve(thumbView.pdfPage);
776
+ }
777
+
778
+ if (this._pagesRequests.has(thumbView)) {
779
+ return this._pagesRequests.get(thumbView);
780
+ }
781
+
782
+ const promise = this.pdfDocument.getPage(thumbView.id).then(pdfPage => {
783
+ if (!thumbView.pdfPage) {
784
+ thumbView.setPdfPage(pdfPage);
785
+ }
786
+
787
+ this._pagesRequests.delete(thumbView);
788
+
789
+ return pdfPage;
790
+ }).catch(reason => {
791
+ console.error("Unable to get page for thumb view", reason); // Page error -- there is nothing that can be done.
792
+
793
+ this._pagesRequests.delete(thumbView);
794
+ });
795
+
796
+ this._pagesRequests.set(thumbView, promise);
797
+
798
+ return promise;
799
+ }
800
+
801
+ forceRendering() {
802
+ const visibleThumbs = this._getVisibleThumbs();
803
+
804
+ const thumbView = this.renderingQueue.getHighestPriority(visibleThumbs, this._thumbnails, this.scroll.down);
805
+
806
+ if (thumbView) {
807
+ this._ensurePdfPageLoaded(thumbView).then(() => {
808
+ this.renderingQueue.renderView(thumbView);
809
+ });
810
+
811
+ return true;
812
+ }
813
+
814
+ return false;
815
+ }
816
+
817
+ }
818
+
819
+ export { PDFThumbnailViewer };