This method is currently called from `PDFViewer._scrollUpdate` on *every* scroll event in the viewer. However, I cannot see why this code is now necessary (assuming that it once was), since text-selection and searching still works *exactly* the same way with this patch as with the current `master`. When `PDFPageView.updatePosition` is called, the page can be in either of these states: 1. The page hasn't been rendered, in which case the `textLayer` property doesn't exist yet. 2. The page is currently rendering, meaning that the `textLayer` property exists. Given that the `textContent` won't be fetched until the page has been successfully rendered, `TextLayerBuilder.render` will return immediately and effectively be a no-op (since there's nothing to render yet). 3. The has been been rendered, and the `textLayer` is currently rendering. 4. The page, and its `textLayer`, has been completely rendered. In this case, `TextLayerBuilder.render` will return immediately and effectively be a no-op. Here, only the *third* case seem to require any further analysis: When scrolling occurs while the `textLayer` is rendering, `TextLayerBuilder.render` will via a helper method call `TextLayerRenderTask.cancel` (in src/display/text_layer.js) to stop processing. However, due to the run-to-completion nature of JavaScript, once `TextLayerRenderTask._render` has been invoked `appendText` will always run.[1] So even though we cancel rendering of pending `textLayer`s during scrolling, via the repeated `TextLayerBuilder.render` calls from within the `PDFPageView.updatePosition` method, that does *not* prevent us from running the code inside of `TextLayerRenderTask._render` over and over for the *same* page; which all seems *very* inefficient to me.[2] All this will thus have the effect of delaying the *actual* rendering of a `textLayer` ever so slightly while scrolling in the viewer. However, it does so at the expense of potentially hundreds of unnecessary `appendText` calls.[3] Hence it seems to me that it's less resource intensive overall to simply let rendering of the `textLayer` complete, once it has started. Obviously, we still abort all rendering of a page, and its `textLayer`, when it's being destroyed (e.g. by being evicted from the page cache). In case that there's any worry that the patch could affect e.g. highlighting of search results, please note that the existing code in `TextLayerBuilder.render` already calls `updateMatches` when the `TextLayerTask` resolves successfully. *I'm sorry that this became quite long, but to try and summarize:* `PDFPageView.updatePosition` doesn't actually do anything in *most* cases. In the one case where it matters, it seems that it's actually doing more harm than good; which is why I'm proposing that we just remove it. --- [1] Although we may be able to skip the `render` call, provided that it happens *after* a `timeout` (as is the case in the default viewer). [2] With current work being done to support streaming of `TextContent`, we'd also need to add just as many duplicate API calls to `PDFPageView.updatePosition`. [3] The number of duplicate `appendText` calls is directly proportional not only to the scroll speed, but also to the number of pages in the document.
644 lines
20 KiB
JavaScript
644 lines
20 KiB
JavaScript
/* Copyright 2012 Mozilla Foundation
|
|
*
|
|
* Licensed under the Apache License, Version 2.0 (the "License");
|
|
* you may not use this file except in compliance with the License.
|
|
* You may obtain a copy of the License at
|
|
*
|
|
* http://www.apache.org/licenses/LICENSE-2.0
|
|
*
|
|
* Unless required by applicable law or agreed to in writing, software
|
|
* distributed under the License is distributed on an "AS IS" BASIS,
|
|
* WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
* See the License for the specific language governing permissions and
|
|
* limitations under the License.
|
|
*/
|
|
|
|
import {
|
|
approximateFraction, CSS_UNITS, DEFAULT_SCALE, getOutputScale, NullL10n,
|
|
RendererType, roundToDivide
|
|
} from './ui_utils';
|
|
import {
|
|
createPromiseCapability, CustomStyle, PDFJS, RenderingCancelledException,
|
|
SVGGraphics
|
|
} from 'pdfjs-lib';
|
|
import { getGlobalEventBus } from './dom_events';
|
|
import { RenderingStates } from './pdf_rendering_queue';
|
|
|
|
const TEXT_LAYER_RENDER_DELAY = 200; // ms
|
|
|
|
/**
|
|
* @typedef {Object} PDFPageViewOptions
|
|
* @property {HTMLDivElement} container - The viewer element.
|
|
* @property {EventBus} eventBus - The application event bus.
|
|
* @property {number} id - The page unique ID (normally its number).
|
|
* @property {number} scale - The page scale display.
|
|
* @property {PageViewport} defaultViewport - The page viewport.
|
|
* @property {PDFRenderingQueue} renderingQueue - The rendering queue object.
|
|
* @property {IPDFTextLayerFactory} textLayerFactory
|
|
* @property {IPDFAnnotationLayerFactory} annotationLayerFactory
|
|
* @property {boolean} enhanceTextSelection - Turns on the text selection
|
|
* enhancement. The default is `false`.
|
|
* @property {boolean} renderInteractiveForms - Turns on rendering of
|
|
* interactive form elements. The default is `false`.
|
|
* @property {string} renderer - 'canvas' or 'svg'. The default is 'canvas'.
|
|
* @property {IL10n} l10n - Localization service.
|
|
*/
|
|
|
|
/**
|
|
* @implements {IRenderableView}
|
|
*/
|
|
class PDFPageView {
|
|
/**
|
|
* @param {PDFPageViewOptions} options
|
|
*/
|
|
constructor(options) {
|
|
let container = options.container;
|
|
let defaultViewport = options.defaultViewport;
|
|
|
|
this.id = options.id;
|
|
this.renderingId = 'page' + this.id;
|
|
|
|
this.pageLabel = null;
|
|
this.rotation = 0;
|
|
this.scale = options.scale || DEFAULT_SCALE;
|
|
this.viewport = defaultViewport;
|
|
this.pdfPageRotate = defaultViewport.rotation;
|
|
this.hasRestrictedScaling = false;
|
|
this.enhanceTextSelection = options.enhanceTextSelection || false;
|
|
this.renderInteractiveForms = options.renderInteractiveForms || false;
|
|
|
|
this.eventBus = options.eventBus || getGlobalEventBus();
|
|
this.renderingQueue = options.renderingQueue;
|
|
this.textLayerFactory = options.textLayerFactory;
|
|
this.annotationLayerFactory = options.annotationLayerFactory;
|
|
this.renderer = options.renderer || RendererType.CANVAS;
|
|
this.l10n = options.l10n || NullL10n;
|
|
|
|
this.paintTask = null;
|
|
this.paintedViewportMap = new WeakMap();
|
|
this.renderingState = RenderingStates.INITIAL;
|
|
this.resume = null;
|
|
this.error = null;
|
|
|
|
this.onBeforeDraw = null;
|
|
this.onAfterDraw = null;
|
|
|
|
this.annotationLayer = null;
|
|
this.textLayer = null;
|
|
this.zoomLayer = null;
|
|
|
|
let div = document.createElement('div');
|
|
div.className = 'page';
|
|
div.style.width = Math.floor(this.viewport.width) + 'px';
|
|
div.style.height = Math.floor(this.viewport.height) + 'px';
|
|
div.setAttribute('data-page-number', this.id);
|
|
this.div = div;
|
|
|
|
container.appendChild(div);
|
|
}
|
|
|
|
setPdfPage(pdfPage) {
|
|
this.pdfPage = pdfPage;
|
|
this.pdfPageRotate = pdfPage.rotate;
|
|
|
|
let totalRotation = (this.rotation + this.pdfPageRotate) % 360;
|
|
this.viewport = pdfPage.getViewport(this.scale * CSS_UNITS,
|
|
totalRotation);
|
|
this.stats = pdfPage.stats;
|
|
this.reset();
|
|
}
|
|
|
|
destroy() {
|
|
this.reset();
|
|
if (this.pdfPage) {
|
|
this.pdfPage.cleanup();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* @private
|
|
*/
|
|
_resetZoomLayer(removeFromDOM = false) {
|
|
if (!this.zoomLayer) {
|
|
return;
|
|
}
|
|
let zoomLayerCanvas = this.zoomLayer.firstChild;
|
|
this.paintedViewportMap.delete(zoomLayerCanvas);
|
|
// Zeroing the width and height causes Firefox to release graphics
|
|
// resources immediately, which can greatly reduce memory consumption.
|
|
zoomLayerCanvas.width = 0;
|
|
zoomLayerCanvas.height = 0;
|
|
|
|
if (removeFromDOM) {
|
|
// Note: `ChildNode.remove` doesn't throw if the parent node is undefined.
|
|
this.zoomLayer.remove();
|
|
}
|
|
this.zoomLayer = null;
|
|
}
|
|
|
|
reset(keepZoomLayer = false, keepAnnotations = false) {
|
|
this.cancelRendering();
|
|
|
|
let div = this.div;
|
|
div.style.width = Math.floor(this.viewport.width) + 'px';
|
|
div.style.height = Math.floor(this.viewport.height) + 'px';
|
|
|
|
let childNodes = div.childNodes;
|
|
let currentZoomLayerNode = (keepZoomLayer && this.zoomLayer) || null;
|
|
let currentAnnotationNode = (keepAnnotations && this.annotationLayer &&
|
|
this.annotationLayer.div) || null;
|
|
for (let i = childNodes.length - 1; i >= 0; i--) {
|
|
let node = childNodes[i];
|
|
if (currentZoomLayerNode === node || currentAnnotationNode === node) {
|
|
continue;
|
|
}
|
|
div.removeChild(node);
|
|
}
|
|
div.removeAttribute('data-loaded');
|
|
|
|
if (currentAnnotationNode) {
|
|
// Hide the annotation layer until all elements are resized
|
|
// so they are not displayed on the already resized page.
|
|
this.annotationLayer.hide();
|
|
} else {
|
|
this.annotationLayer = null;
|
|
}
|
|
|
|
if (!currentZoomLayerNode) {
|
|
if (this.canvas) {
|
|
this.paintedViewportMap.delete(this.canvas);
|
|
// Zeroing the width and height causes Firefox to release graphics
|
|
// resources immediately, which can greatly reduce memory consumption.
|
|
this.canvas.width = 0;
|
|
this.canvas.height = 0;
|
|
delete this.canvas;
|
|
}
|
|
this._resetZoomLayer();
|
|
}
|
|
if (this.svg) {
|
|
this.paintedViewportMap.delete(this.svg);
|
|
delete this.svg;
|
|
}
|
|
|
|
this.loadingIconDiv = document.createElement('div');
|
|
this.loadingIconDiv.className = 'loadingIcon';
|
|
div.appendChild(this.loadingIconDiv);
|
|
}
|
|
|
|
update(scale, rotation) {
|
|
this.scale = scale || this.scale;
|
|
if (typeof rotation !== 'undefined') { // The rotation may be zero.
|
|
this.rotation = rotation;
|
|
}
|
|
|
|
let totalRotation = (this.rotation + this.pdfPageRotate) % 360;
|
|
this.viewport = this.viewport.clone({
|
|
scale: this.scale * CSS_UNITS,
|
|
rotation: totalRotation,
|
|
});
|
|
|
|
if (this.svg) {
|
|
this.cssTransform(this.svg, true);
|
|
|
|
this.eventBus.dispatch('pagerendered', {
|
|
source: this,
|
|
pageNumber: this.id,
|
|
cssTransform: true,
|
|
});
|
|
return;
|
|
}
|
|
|
|
let isScalingRestricted = false;
|
|
if (this.canvas && PDFJS.maxCanvasPixels > 0) {
|
|
let outputScale = this.outputScale;
|
|
if (((Math.floor(this.viewport.width) * outputScale.sx) | 0) *
|
|
((Math.floor(this.viewport.height) * outputScale.sy) | 0) >
|
|
PDFJS.maxCanvasPixels) {
|
|
isScalingRestricted = true;
|
|
}
|
|
}
|
|
|
|
if (this.canvas) {
|
|
if (PDFJS.useOnlyCssZoom ||
|
|
(this.hasRestrictedScaling && isScalingRestricted)) {
|
|
this.cssTransform(this.canvas, true);
|
|
|
|
this.eventBus.dispatch('pagerendered', {
|
|
source: this,
|
|
pageNumber: this.id,
|
|
cssTransform: true,
|
|
});
|
|
return;
|
|
}
|
|
if (!this.zoomLayer) {
|
|
this.zoomLayer = this.canvas.parentNode;
|
|
this.zoomLayer.style.position = 'absolute';
|
|
}
|
|
}
|
|
if (this.zoomLayer) {
|
|
this.cssTransform(this.zoomLayer.firstChild);
|
|
}
|
|
this.reset(/* keepZoomLayer = */ true, /* keepAnnotations = */ true);
|
|
}
|
|
|
|
cancelRendering() {
|
|
if (this.paintTask) {
|
|
this.paintTask.cancel();
|
|
this.paintTask = null;
|
|
}
|
|
this.renderingState = RenderingStates.INITIAL;
|
|
this.resume = null;
|
|
|
|
if (this.textLayer) {
|
|
this.textLayer.cancel();
|
|
this.textLayer = null;
|
|
}
|
|
}
|
|
|
|
cssTransform(target, redrawAnnotations = false) {
|
|
// Scale target (canvas or svg), its wrapper and page container.
|
|
let width = this.viewport.width;
|
|
let height = this.viewport.height;
|
|
let div = this.div;
|
|
target.style.width = target.parentNode.style.width = div.style.width =
|
|
Math.floor(width) + 'px';
|
|
target.style.height = target.parentNode.style.height = div.style.height =
|
|
Math.floor(height) + 'px';
|
|
// The canvas may have been originally rotated; rotate relative to that.
|
|
let relativeRotation = this.viewport.rotation -
|
|
this.paintedViewportMap.get(target).rotation;
|
|
let absRotation = Math.abs(relativeRotation);
|
|
let scaleX = 1, scaleY = 1;
|
|
if (absRotation === 90 || absRotation === 270) {
|
|
// Scale x and y because of the rotation.
|
|
scaleX = height / width;
|
|
scaleY = width / height;
|
|
}
|
|
let cssTransform = 'rotate(' + relativeRotation + 'deg) ' +
|
|
'scale(' + scaleX + ',' + scaleY + ')';
|
|
CustomStyle.setProp('transform', target, cssTransform);
|
|
|
|
if (this.textLayer) {
|
|
// Rotating the text layer is more complicated since the divs inside the
|
|
// the text layer are rotated.
|
|
// TODO: This could probably be simplified by drawing the text layer in
|
|
// one orientation and then rotating overall.
|
|
let textLayerViewport = this.textLayer.viewport;
|
|
let textRelativeRotation = this.viewport.rotation -
|
|
textLayerViewport.rotation;
|
|
let textAbsRotation = Math.abs(textRelativeRotation);
|
|
let scale = width / textLayerViewport.width;
|
|
if (textAbsRotation === 90 || textAbsRotation === 270) {
|
|
scale = width / textLayerViewport.height;
|
|
}
|
|
let textLayerDiv = this.textLayer.textLayerDiv;
|
|
let transX, transY;
|
|
switch (textAbsRotation) {
|
|
case 0:
|
|
transX = transY = 0;
|
|
break;
|
|
case 90:
|
|
transX = 0;
|
|
transY = '-' + textLayerDiv.style.height;
|
|
break;
|
|
case 180:
|
|
transX = '-' + textLayerDiv.style.width;
|
|
transY = '-' + textLayerDiv.style.height;
|
|
break;
|
|
case 270:
|
|
transX = '-' + textLayerDiv.style.width;
|
|
transY = 0;
|
|
break;
|
|
default:
|
|
console.error('Bad rotation value.');
|
|
break;
|
|
}
|
|
CustomStyle.setProp('transform', textLayerDiv,
|
|
'rotate(' + textAbsRotation + 'deg) ' +
|
|
'scale(' + scale + ', ' + scale + ') ' +
|
|
'translate(' + transX + ', ' + transY + ')');
|
|
CustomStyle.setProp('transformOrigin', textLayerDiv, '0% 0%');
|
|
}
|
|
|
|
if (redrawAnnotations && this.annotationLayer) {
|
|
this.annotationLayer.render(this.viewport, 'display');
|
|
}
|
|
}
|
|
|
|
get width() {
|
|
return this.viewport.width;
|
|
}
|
|
|
|
get height() {
|
|
return this.viewport.height;
|
|
}
|
|
|
|
getPagePoint(x, y) {
|
|
return this.viewport.convertToPdfPoint(x, y);
|
|
}
|
|
|
|
draw() {
|
|
if (this.renderingState !== RenderingStates.INITIAL) {
|
|
console.error('Must be in new state before drawing');
|
|
this.reset(); // Ensure that we reset all state to prevent issues.
|
|
}
|
|
|
|
this.renderingState = RenderingStates.RUNNING;
|
|
|
|
let pdfPage = this.pdfPage;
|
|
let div = this.div;
|
|
// Wrap the canvas so that if it has a CSS transform for high DPI the
|
|
// overflow will be hidden in Firefox.
|
|
let canvasWrapper = document.createElement('div');
|
|
canvasWrapper.style.width = div.style.width;
|
|
canvasWrapper.style.height = div.style.height;
|
|
canvasWrapper.classList.add('canvasWrapper');
|
|
|
|
if (this.annotationLayer && this.annotationLayer.div) {
|
|
// The annotation layer needs to stay on top.
|
|
div.insertBefore(canvasWrapper, this.annotationLayer.div);
|
|
} else {
|
|
div.appendChild(canvasWrapper);
|
|
}
|
|
|
|
let textLayer = null;
|
|
if (this.textLayerFactory) {
|
|
let textLayerDiv = document.createElement('div');
|
|
textLayerDiv.className = 'textLayer';
|
|
textLayerDiv.style.width = canvasWrapper.style.width;
|
|
textLayerDiv.style.height = canvasWrapper.style.height;
|
|
if (this.annotationLayer && this.annotationLayer.div) {
|
|
// The annotation layer needs to stay on top.
|
|
div.insertBefore(textLayerDiv, this.annotationLayer.div);
|
|
} else {
|
|
div.appendChild(textLayerDiv);
|
|
}
|
|
|
|
textLayer = this.textLayerFactory.
|
|
createTextLayerBuilder(textLayerDiv, this.id - 1, this.viewport,
|
|
this.enhanceTextSelection);
|
|
}
|
|
this.textLayer = textLayer;
|
|
|
|
let renderContinueCallback = null;
|
|
if (this.renderingQueue) {
|
|
renderContinueCallback = (cont) => {
|
|
if (!this.renderingQueue.isHighestPriority(this)) {
|
|
this.renderingState = RenderingStates.PAUSED;
|
|
this.resume = () => {
|
|
this.renderingState = RenderingStates.RUNNING;
|
|
cont();
|
|
};
|
|
return;
|
|
}
|
|
cont();
|
|
};
|
|
}
|
|
|
|
let finishPaintTask = (error) => {
|
|
// The paintTask may have been replaced by a new one, so only remove
|
|
// the reference to the paintTask if it matches the one that is
|
|
// triggering this callback.
|
|
if (paintTask === this.paintTask) {
|
|
this.paintTask = null;
|
|
}
|
|
|
|
if (((typeof PDFJSDev === 'undefined' ||
|
|
!PDFJSDev.test('PDFJS_NEXT')) && error === 'cancelled') ||
|
|
error instanceof RenderingCancelledException) {
|
|
this.error = null;
|
|
return Promise.resolve(undefined);
|
|
}
|
|
|
|
this.renderingState = RenderingStates.FINISHED;
|
|
|
|
if (this.loadingIconDiv) {
|
|
div.removeChild(this.loadingIconDiv);
|
|
delete this.loadingIconDiv;
|
|
}
|
|
this._resetZoomLayer(/* removeFromDOM = */ true);
|
|
|
|
this.error = error;
|
|
this.stats = pdfPage.stats;
|
|
if (this.onAfterDraw) {
|
|
this.onAfterDraw();
|
|
}
|
|
this.eventBus.dispatch('pagerendered', {
|
|
source: this,
|
|
pageNumber: this.id,
|
|
cssTransform: false,
|
|
});
|
|
|
|
if (error) {
|
|
return Promise.reject(error);
|
|
}
|
|
return Promise.resolve(undefined);
|
|
};
|
|
|
|
let paintTask = this.renderer === RendererType.SVG ?
|
|
this.paintOnSvg(canvasWrapper) :
|
|
this.paintOnCanvas(canvasWrapper);
|
|
paintTask.onRenderContinue = renderContinueCallback;
|
|
this.paintTask = paintTask;
|
|
|
|
let resultPromise = paintTask.promise.then(function() {
|
|
return finishPaintTask(null).then(function () {
|
|
if (textLayer) {
|
|
pdfPage.getTextContent({
|
|
normalizeWhitespace: true,
|
|
}).then(function textContentResolved(textContent) {
|
|
textLayer.setTextContent(textContent);
|
|
textLayer.render(TEXT_LAYER_RENDER_DELAY);
|
|
});
|
|
}
|
|
});
|
|
}, function(reason) {
|
|
return finishPaintTask(reason);
|
|
});
|
|
|
|
if (this.annotationLayerFactory) {
|
|
if (!this.annotationLayer) {
|
|
this.annotationLayer = this.annotationLayerFactory.
|
|
createAnnotationLayerBuilder(div, pdfPage,
|
|
this.renderInteractiveForms, this.l10n);
|
|
}
|
|
this.annotationLayer.render(this.viewport, 'display');
|
|
}
|
|
div.setAttribute('data-loaded', true);
|
|
|
|
if (this.onBeforeDraw) {
|
|
this.onBeforeDraw();
|
|
}
|
|
return resultPromise;
|
|
}
|
|
|
|
paintOnCanvas(canvasWrapper) {
|
|
let renderCapability = createPromiseCapability();
|
|
let result = {
|
|
promise: renderCapability.promise,
|
|
onRenderContinue(cont) {
|
|
cont();
|
|
},
|
|
cancel() {
|
|
renderTask.cancel();
|
|
},
|
|
};
|
|
|
|
let viewport = this.viewport;
|
|
let canvas = document.createElement('canvas');
|
|
canvas.id = this.renderingId;
|
|
|
|
// Keep the canvas hidden until the first draw callback, or until drawing
|
|
// is complete when `!this.renderingQueue`, to prevent black flickering.
|
|
canvas.setAttribute('hidden', 'hidden');
|
|
let isCanvasHidden = true;
|
|
let showCanvas = function () {
|
|
if (isCanvasHidden) {
|
|
canvas.removeAttribute('hidden');
|
|
isCanvasHidden = false;
|
|
}
|
|
};
|
|
|
|
canvasWrapper.appendChild(canvas);
|
|
this.canvas = canvas;
|
|
|
|
if (typeof PDFJSDev === 'undefined' ||
|
|
PDFJSDev.test('MOZCENTRAL || FIREFOX || GENERIC')) {
|
|
canvas.mozOpaque = true;
|
|
}
|
|
|
|
let ctx = canvas.getContext('2d', { alpha: false, });
|
|
let outputScale = getOutputScale(ctx);
|
|
this.outputScale = outputScale;
|
|
|
|
if (PDFJS.useOnlyCssZoom) {
|
|
let actualSizeViewport = viewport.clone({ scale: CSS_UNITS, });
|
|
// Use a scale that makes the canvas have the originally intended size
|
|
// of the page.
|
|
outputScale.sx *= actualSizeViewport.width / viewport.width;
|
|
outputScale.sy *= actualSizeViewport.height / viewport.height;
|
|
outputScale.scaled = true;
|
|
}
|
|
|
|
if (PDFJS.maxCanvasPixels > 0) {
|
|
let pixelsInViewport = viewport.width * viewport.height;
|
|
let maxScale = Math.sqrt(PDFJS.maxCanvasPixels / pixelsInViewport);
|
|
if (outputScale.sx > maxScale || outputScale.sy > maxScale) {
|
|
outputScale.sx = maxScale;
|
|
outputScale.sy = maxScale;
|
|
outputScale.scaled = true;
|
|
this.hasRestrictedScaling = true;
|
|
} else {
|
|
this.hasRestrictedScaling = false;
|
|
}
|
|
}
|
|
|
|
let sfx = approximateFraction(outputScale.sx);
|
|
let sfy = approximateFraction(outputScale.sy);
|
|
canvas.width = roundToDivide(viewport.width * outputScale.sx, sfx[0]);
|
|
canvas.height = roundToDivide(viewport.height * outputScale.sy, sfy[0]);
|
|
canvas.style.width = roundToDivide(viewport.width, sfx[1]) + 'px';
|
|
canvas.style.height = roundToDivide(viewport.height, sfy[1]) + 'px';
|
|
// Add the viewport so it's known what it was originally drawn with.
|
|
this.paintedViewportMap.set(canvas, viewport);
|
|
|
|
// Rendering area
|
|
let transform = !outputScale.scaled ? null :
|
|
[outputScale.sx, 0, 0, outputScale.sy, 0, 0];
|
|
let renderContext = {
|
|
canvasContext: ctx,
|
|
transform,
|
|
viewport: this.viewport,
|
|
renderInteractiveForms: this.renderInteractiveForms,
|
|
};
|
|
let renderTask = this.pdfPage.render(renderContext);
|
|
renderTask.onContinue = function (cont) {
|
|
showCanvas();
|
|
if (result.onRenderContinue) {
|
|
result.onRenderContinue(cont);
|
|
} else {
|
|
cont();
|
|
}
|
|
};
|
|
|
|
renderTask.promise.then(function() {
|
|
showCanvas();
|
|
renderCapability.resolve(undefined);
|
|
}, function(error) {
|
|
showCanvas();
|
|
renderCapability.reject(error);
|
|
});
|
|
return result;
|
|
}
|
|
|
|
paintOnSvg(wrapper) {
|
|
if (typeof PDFJSDev !== 'undefined' &&
|
|
PDFJSDev.test('FIREFOX || MOZCENTRAL || CHROME')) {
|
|
// Return a mock object, to prevent errors such as e.g.
|
|
// "TypeError: paintTask.promise is undefined".
|
|
return {
|
|
promise: Promise.reject(new Error('SVG rendering is not supported.')),
|
|
onRenderContinue(cont) { },
|
|
cancel() { },
|
|
};
|
|
}
|
|
|
|
let cancelled = false;
|
|
let ensureNotCancelled = () => {
|
|
if (cancelled) {
|
|
if ((typeof PDFJSDev !== 'undefined' &&
|
|
PDFJSDev.test('PDFJS_NEXT')) || PDFJS.pdfjsNext) {
|
|
throw new RenderingCancelledException(
|
|
'Rendering cancelled, page ' + this.id, 'svg');
|
|
} else {
|
|
throw 'cancelled'; // eslint-disable-line no-throw-literal
|
|
}
|
|
}
|
|
};
|
|
|
|
let pdfPage = this.pdfPage;
|
|
let actualSizeViewport = this.viewport.clone({ scale: CSS_UNITS, });
|
|
let promise = pdfPage.getOperatorList().then((opList) => {
|
|
ensureNotCancelled();
|
|
let svgGfx = new SVGGraphics(pdfPage.commonObjs, pdfPage.objs);
|
|
return svgGfx.getSVG(opList, actualSizeViewport).then((svg) => {
|
|
ensureNotCancelled();
|
|
this.svg = svg;
|
|
this.paintedViewportMap.set(svg, actualSizeViewport);
|
|
|
|
svg.style.width = wrapper.style.width;
|
|
svg.style.height = wrapper.style.height;
|
|
this.renderingState = RenderingStates.FINISHED;
|
|
wrapper.appendChild(svg);
|
|
});
|
|
});
|
|
|
|
return {
|
|
promise,
|
|
onRenderContinue(cont) {
|
|
cont();
|
|
},
|
|
cancel() {
|
|
cancelled = true;
|
|
},
|
|
};
|
|
}
|
|
|
|
/**
|
|
* @param {string|null} label
|
|
*/
|
|
setPageLabel(label) {
|
|
this.pageLabel = (typeof label === 'string' ? label : null);
|
|
|
|
if (this.pageLabel !== null) {
|
|
this.div.setAttribute('data-page-label', this.pageLabel);
|
|
} else {
|
|
this.div.removeAttribute('data-page-label');
|
|
}
|
|
}
|
|
}
|
|
|
|
export {
|
|
PDFPageView,
|
|
};
|