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

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 (174) hide show
  1. package/README.md +1 -1
  2. package/package.json +33 -18
  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/get.js +1857 -0
  6. package/pdfjs/dist/index.js +767 -0
  7. package/pdfjs/dist/l10n_utils.js +140 -0
  8. package/{src/shared → pdfjs/dist}/message_handler.js +243 -259
  9. package/{src/display → pdfjs/dist}/network.js +149 -87
  10. package/{src/display/content_disposition.js → pdfjs/dist/network_utils.js} +167 -55
  11. package/{src/display → pdfjs/dist}/node_stream.js +133 -98
  12. package/pdfjs/dist/pdf.js +12778 -0
  13. package/pdfjs/dist/pdf_link_service.js +638 -0
  14. package/pdfjs/dist/pdf_rendering_queue.js +199 -0
  15. package/pdfjs/dist/pdf_thumbnail_viewer.js +819 -0
  16. package/pdfjs/dist/pdf_viewer.js +3598 -0
  17. package/pdfjs/dist/typeof.js +100 -0
  18. package/pdfjs/dist/ui_utils.js +1033 -0
  19. package/{src/shared → pdfjs/dist}/util.js +301 -287
  20. package/pdfjs/dist/worker.js +62813 -0
  21. package/src/vcf-pdf-viewer.js +77 -27
  22. package/theme/lumo/vcf-pdf-viewer-styles.js +1 -1
  23. package/theme/material/vcf-pdf-viewer.js +2 -2
  24. package/src/core/.eslintrc +0 -13
  25. package/src/core/annotation.js +0 -2948
  26. package/src/core/arithmetic_decoder.js +0 -182
  27. package/src/core/ascii_85_stream.js +0 -98
  28. package/src/core/ascii_hex_stream.js +0 -79
  29. package/src/core/base_stream.js +0 -110
  30. package/src/core/bidi.js +0 -438
  31. package/src/core/calibri_factors.js +0 -308
  32. package/src/core/catalog.js +0 -1459
  33. package/src/core/ccitt.js +0 -1062
  34. package/src/core/ccitt_stream.js +0 -60
  35. package/src/core/cff_font.js +0 -116
  36. package/src/core/cff_parser.js +0 -1949
  37. package/src/core/charsets.js +0 -119
  38. package/src/core/chunked_stream.js +0 -557
  39. package/src/core/cmap.js +0 -1039
  40. package/src/core/colorspace.js +0 -1533
  41. package/src/core/core_utils.js +0 -464
  42. package/src/core/crypto.js +0 -1900
  43. package/src/core/decode_stream.js +0 -170
  44. package/src/core/decrypt_stream.js +0 -59
  45. package/src/core/default_appearance.js +0 -99
  46. package/src/core/document.js +0 -1456
  47. package/src/core/encodings.js +0 -301
  48. package/src/core/evaluator.js +0 -4601
  49. package/src/core/file_spec.js +0 -108
  50. package/src/core/flate_stream.js +0 -402
  51. package/src/core/font_renderer.js +0 -882
  52. package/src/core/fonts.js +0 -3260
  53. package/src/core/fonts_utils.js +0 -221
  54. package/src/core/function.js +0 -1257
  55. package/src/core/glyf.js +0 -706
  56. package/src/core/glyphlist.js +0 -4558
  57. package/src/core/helvetica_factors.js +0 -353
  58. package/src/core/image.js +0 -802
  59. package/src/core/image_utils.js +0 -291
  60. package/src/core/jbig2.js +0 -2572
  61. package/src/core/jbig2_stream.js +0 -73
  62. package/src/core/jpeg_stream.js +0 -105
  63. package/src/core/jpg.js +0 -1416
  64. package/src/core/jpx.js +0 -2343
  65. package/src/core/jpx_stream.js +0 -87
  66. package/src/core/liberationsans_widths.js +0 -221
  67. package/src/core/lzw_stream.js +0 -150
  68. package/src/core/metadata_parser.js +0 -146
  69. package/src/core/metrics.js +0 -2970
  70. package/src/core/murmurhash3.js +0 -139
  71. package/src/core/myriadpro_factors.js +0 -290
  72. package/src/core/name_number_tree.js +0 -153
  73. package/src/core/object_loader.js +0 -149
  74. package/src/core/opentype_file_builder.js +0 -154
  75. package/src/core/operator_list.js +0 -734
  76. package/src/core/parser.js +0 -1416
  77. package/src/core/pattern.js +0 -985
  78. package/src/core/pdf_manager.js +0 -217
  79. package/src/core/predictor_stream.js +0 -238
  80. package/src/core/primitives.js +0 -402
  81. package/src/core/ps_parser.js +0 -272
  82. package/src/core/run_length_stream.js +0 -61
  83. package/src/core/segoeui_factors.js +0 -308
  84. package/src/core/standard_fonts.js +0 -817
  85. package/src/core/stream.js +0 -103
  86. package/src/core/struct_tree.js +0 -335
  87. package/src/core/to_unicode_map.js +0 -103
  88. package/src/core/type1_font.js +0 -421
  89. package/src/core/type1_parser.js +0 -776
  90. package/src/core/unicode.js +0 -1649
  91. package/src/core/worker.js +0 -848
  92. package/src/core/worker_stream.js +0 -135
  93. package/src/core/writer.js +0 -278
  94. package/src/core/xfa/bind.js +0 -652
  95. package/src/core/xfa/builder.js +0 -207
  96. package/src/core/xfa/config.js +0 -1926
  97. package/src/core/xfa/connection_set.js +0 -202
  98. package/src/core/xfa/data.js +0 -82
  99. package/src/core/xfa/datasets.js +0 -76
  100. package/src/core/xfa/factory.js +0 -111
  101. package/src/core/xfa/fonts.js +0 -181
  102. package/src/core/xfa/formcalc_lexer.js +0 -385
  103. package/src/core/xfa/formcalc_parser.js +0 -1340
  104. package/src/core/xfa/html_utils.js +0 -639
  105. package/src/core/xfa/layout.js +0 -383
  106. package/src/core/xfa/locale_set.js +0 -345
  107. package/src/core/xfa/namespaces.js +0 -81
  108. package/src/core/xfa/parser.js +0 -184
  109. package/src/core/xfa/setup.js +0 -38
  110. package/src/core/xfa/signature.js +0 -40
  111. package/src/core/xfa/som.js +0 -338
  112. package/src/core/xfa/stylesheet.js +0 -40
  113. package/src/core/xfa/template.js +0 -6260
  114. package/src/core/xfa/text.js +0 -290
  115. package/src/core/xfa/unknown.js +0 -29
  116. package/src/core/xfa/utils.js +0 -217
  117. package/src/core/xfa/xdp.js +0 -59
  118. package/src/core/xfa/xfa_object.js +0 -1130
  119. package/src/core/xfa/xhtml.js +0 -543
  120. package/src/core/xfa_fonts.js +0 -208
  121. package/src/core/xml_parser.js +0 -507
  122. package/src/core/xref.js +0 -899
  123. package/src/display/annotation_layer.js +0 -2107
  124. package/src/display/annotation_storage.js +0 -113
  125. package/src/display/api.js +0 -3292
  126. package/src/display/base_factory.js +0 -180
  127. package/src/display/canvas.js +0 -2828
  128. package/src/display/font_loader.js +0 -484
  129. package/src/display/metadata.js +0 -41
  130. package/src/display/network_utils.js +0 -100
  131. package/src/display/node_utils.js +0 -83
  132. package/src/display/optional_content_config.js +0 -189
  133. package/src/display/pattern_helper.js +0 -659
  134. package/src/display/svg.js +0 -1709
  135. package/src/display/text_layer.js +0 -847
  136. package/src/display/transport_stream.js +0 -303
  137. package/src/display/worker_options.js +0 -40
  138. package/src/display/xfa_layer.js +0 -204
  139. package/src/doc_helper.js +0 -25
  140. package/src/images/logo.svg +0 -41
  141. package/src/interfaces.js +0 -169
  142. package/src/license_header.js +0 -14
  143. package/src/license_header_libre.js +0 -21
  144. package/src/pdf.image_decoders.js +0 -46
  145. package/src/pdf.js +0 -146
  146. package/src/pdf.sandbox.external.js +0 -181
  147. package/src/pdf.sandbox.js +0 -151
  148. package/src/pdf.scripting.js +0 -25
  149. package/src/pdf.worker.entry.js +0 -19
  150. package/src/pdf.worker.js +0 -23
  151. package/src/scripting_api/aform.js +0 -608
  152. package/src/scripting_api/app.js +0 -621
  153. package/src/scripting_api/color.js +0 -129
  154. package/src/scripting_api/common.js +0 -58
  155. package/src/scripting_api/console.js +0 -38
  156. package/src/scripting_api/constants.js +0 -208
  157. package/src/scripting_api/doc.js +0 -1195
  158. package/src/scripting_api/error.js +0 -23
  159. package/src/scripting_api/event.js +0 -232
  160. package/src/scripting_api/field.js +0 -620
  161. package/src/scripting_api/fullscreen.js +0 -145
  162. package/src/scripting_api/initialization.js +0 -223
  163. package/src/scripting_api/pdf_object.js +0 -24
  164. package/src/scripting_api/print_params.js +0 -146
  165. package/src/scripting_api/proxy.js +0 -139
  166. package/src/scripting_api/thermometer.js +0 -69
  167. package/src/scripting_api/util.js +0 -581
  168. package/src/shared/.eslintrc +0 -13
  169. package/src/shared/cffStandardStrings.js +0 -311
  170. package/src/shared/compatibility.js +0 -114
  171. package/src/shared/fonts_utils.js +0 -429
  172. package/src/shared/is_node.js +0 -27
  173. package/src/shared/scripting_utils.js +0 -85
  174. 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 };