123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384 |
- /* Copyright 2016 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 { AnnotationMode, PixelsPerInch } from "pdfjs-lib";
- import { PDFPrintServiceFactory, PDFViewerApplication } from "./app.js";
- import { getXfaHtmlForPrinting } from "./print_utils.js";
- let activeService = null;
- let dialog = null;
- let overlayManager = null;
- // Renders the page to the canvas of the given print service, and returns
- // the suggested dimensions of the output page.
- function renderPage(
- activeServiceOnEntry,
- pdfDocument,
- pageNumber,
- size,
- printResolution,
- optionalContentConfigPromise,
- printAnnotationStoragePromise
- ) {
- const scratchCanvas = activeService.scratchCanvas;
- // The size of the canvas in pixels for printing.
- const PRINT_UNITS = printResolution / PixelsPerInch.PDF;
- scratchCanvas.width = Math.floor(size.width * PRINT_UNITS);
- scratchCanvas.height = Math.floor(size.height * PRINT_UNITS);
- const ctx = scratchCanvas.getContext("2d");
- ctx.save();
- ctx.fillStyle = "rgb(255, 255, 255)";
- ctx.fillRect(0, 0, scratchCanvas.width, scratchCanvas.height);
- ctx.restore();
- return Promise.all([
- pdfDocument.getPage(pageNumber),
- printAnnotationStoragePromise,
- ]).then(function ([pdfPage, printAnnotationStorage]) {
- const renderContext = {
- canvasContext: ctx,
- transform: [PRINT_UNITS, 0, 0, PRINT_UNITS, 0, 0],
- viewport: pdfPage.getViewport({ scale: 1, rotation: size.rotation }),
- intent: "print",
- annotationMode: AnnotationMode.ENABLE_STORAGE,
- optionalContentConfigPromise,
- printAnnotationStorage,
- };
- return pdfPage.render(renderContext).promise;
- });
- }
- function PDFPrintService(
- pdfDocument,
- pagesOverview,
- printContainer,
- printResolution,
- optionalContentConfigPromise = null,
- printAnnotationStoragePromise = null,
- l10n
- ) {
- this.pdfDocument = pdfDocument;
- this.pagesOverview = pagesOverview;
- this.printContainer = printContainer;
- this._printResolution = printResolution || 150;
- this._optionalContentConfigPromise =
- optionalContentConfigPromise || pdfDocument.getOptionalContentConfig();
- this._printAnnotationStoragePromise =
- printAnnotationStoragePromise || Promise.resolve();
- this.l10n = l10n;
- this.currentPage = -1;
- // The temporary canvas where renderPage paints one page at a time.
- this.scratchCanvas = document.createElement("canvas");
- }
- PDFPrintService.prototype = {
- layout() {
- this.throwIfInactive();
- const body = document.querySelector("body");
- body.setAttribute("data-pdfjsprinting", true);
- const hasEqualPageSizes = this.pagesOverview.every(function (size) {
- return (
- size.width === this.pagesOverview[0].width &&
- size.height === this.pagesOverview[0].height
- );
- }, this);
- if (!hasEqualPageSizes) {
- console.warn(
- "Not all pages have the same size. The printed " +
- "result may be incorrect!"
- );
- }
- // Insert a @page + size rule to make sure that the page size is correctly
- // set. Note that we assume that all pages have the same size, because
- // variable-size pages are not supported yet (e.g. in Chrome & Firefox).
- // TODO(robwu): Use named pages when size calculation bugs get resolved
- // (e.g. https://crbug.com/355116) AND when support for named pages is
- // added (http://www.w3.org/TR/css3-page/#using-named-pages).
- // In browsers where @page + size is not supported (such as Firefox,
- // https://bugzil.la/851441), the next stylesheet will be ignored and the
- // user has to select the correct paper size in the UI if wanted.
- this.pageStyleSheet = document.createElement("style");
- const pageSize = this.pagesOverview[0];
- this.pageStyleSheet.textContent =
- "@page { size: " + pageSize.width + "pt " + pageSize.height + "pt;}";
- body.append(this.pageStyleSheet);
- },
- destroy() {
- if (activeService !== this) {
- // |activeService| cannot be replaced without calling destroy() first,
- // so if it differs then an external consumer has a stale reference to us.
- return;
- }
- this.printContainer.textContent = "";
- const body = document.querySelector("body");
- body.removeAttribute("data-pdfjsprinting");
- if (this.pageStyleSheet) {
- this.pageStyleSheet.remove();
- this.pageStyleSheet = null;
- }
- this.scratchCanvas.width = this.scratchCanvas.height = 0;
- this.scratchCanvas = null;
- activeService = null;
- ensureOverlay().then(function () {
- if (overlayManager.active === dialog) {
- overlayManager.close(dialog);
- }
- });
- },
- renderPages() {
- if (this.pdfDocument.isPureXfa) {
- getXfaHtmlForPrinting(this.printContainer, this.pdfDocument);
- return Promise.resolve();
- }
- const pageCount = this.pagesOverview.length;
- const renderNextPage = (resolve, reject) => {
- this.throwIfInactive();
- if (++this.currentPage >= pageCount) {
- renderProgress(pageCount, pageCount, this.l10n);
- resolve();
- return;
- }
- const index = this.currentPage;
- renderProgress(index, pageCount, this.l10n);
- renderPage(
- this,
- this.pdfDocument,
- /* pageNumber = */ index + 1,
- this.pagesOverview[index],
- this._printResolution,
- this._optionalContentConfigPromise,
- this._printAnnotationStoragePromise
- )
- .then(this.useRenderedPage.bind(this))
- .then(function () {
- renderNextPage(resolve, reject);
- }, reject);
- };
- return new Promise(renderNextPage);
- },
- useRenderedPage() {
- this.throwIfInactive();
- const img = document.createElement("img");
- const scratchCanvas = this.scratchCanvas;
- if ("toBlob" in scratchCanvas) {
- scratchCanvas.toBlob(function (blob) {
- img.src = URL.createObjectURL(blob);
- });
- } else {
- img.src = scratchCanvas.toDataURL();
- }
- const wrapper = document.createElement("div");
- wrapper.className = "printedPage";
- wrapper.append(img);
- this.printContainer.append(wrapper);
- return new Promise(function (resolve, reject) {
- img.onload = resolve;
- img.onerror = reject;
- });
- },
- performPrint() {
- this.throwIfInactive();
- return new Promise(resolve => {
- // Push window.print in the macrotask queue to avoid being affected by
- // the deprecation of running print() code in a microtask, see
- // https://github.com/mozilla/pdf.js/issues/7547.
- setTimeout(() => {
- if (!this.active) {
- resolve();
- return;
- }
- print.call(window);
- // Delay promise resolution in case print() was not synchronous.
- setTimeout(resolve, 20); // Tidy-up.
- }, 0);
- });
- },
- get active() {
- return this === activeService;
- },
- throwIfInactive() {
- if (!this.active) {
- throw new Error("This print request was cancelled or completed.");
- }
- },
- };
- const print = window.print;
- window.print = function () {
- if (activeService) {
- console.warn("Ignored window.print() because of a pending print job.");
- return;
- }
- ensureOverlay().then(function () {
- if (activeService) {
- overlayManager.open(dialog);
- }
- });
- try {
- dispatchEvent("beforeprint");
- } finally {
- if (!activeService) {
- console.error("Expected print service to be initialized.");
- ensureOverlay().then(function () {
- if (overlayManager.active === dialog) {
- overlayManager.close(dialog);
- }
- });
- return; // eslint-disable-line no-unsafe-finally
- }
- const activeServiceOnEntry = activeService;
- activeService
- .renderPages()
- .then(function () {
- return activeServiceOnEntry.performPrint();
- })
- .catch(function () {
- // Ignore any error messages.
- })
- .then(function () {
- // aborts acts on the "active" print request, so we need to check
- // whether the print request (activeServiceOnEntry) is still active.
- // Without the check, an unrelated print request (created after aborting
- // this print request while the pages were being generated) would be
- // aborted.
- if (activeServiceOnEntry.active) {
- abort();
- }
- });
- }
- };
- function dispatchEvent(eventType) {
- const event = document.createEvent("CustomEvent");
- event.initCustomEvent(eventType, false, false, "custom");
- window.dispatchEvent(event);
- }
- function abort() {
- if (activeService) {
- activeService.destroy();
- dispatchEvent("afterprint");
- }
- }
- function renderProgress(index, total, l10n) {
- dialog ||= document.getElementById("printServiceDialog");
- const progress = Math.round((100 * index) / total);
- const progressBar = dialog.querySelector("progress");
- const progressPerc = dialog.querySelector(".relative-progress");
- progressBar.value = progress;
- l10n.get("print_progress_percent", { progress }).then(msg => {
- progressPerc.textContent = msg;
- });
- }
- window.addEventListener(
- "keydown",
- function (event) {
- // Intercept Cmd/Ctrl + P in all browsers.
- // Also intercept Cmd/Ctrl + Shift + P in Chrome and Opera
- if (
- event.keyCode === /* P= */ 80 &&
- (event.ctrlKey || event.metaKey) &&
- !event.altKey &&
- (!event.shiftKey || window.chrome || window.opera)
- ) {
- window.print();
- event.preventDefault();
- event.stopImmediatePropagation();
- }
- },
- true
- );
- if ("onbeforeprint" in window) {
- // Do not propagate before/afterprint events when they are not triggered
- // from within this polyfill. (FF / Chrome 63+).
- const stopPropagationIfNeeded = function (event) {
- if (event.detail !== "custom") {
- event.stopImmediatePropagation();
- }
- };
- window.addEventListener("beforeprint", stopPropagationIfNeeded);
- window.addEventListener("afterprint", stopPropagationIfNeeded);
- }
- let overlayPromise;
- function ensureOverlay() {
- if (!overlayPromise) {
- overlayManager = PDFViewerApplication.overlayManager;
- if (!overlayManager) {
- throw new Error("The overlay manager has not yet been initialized.");
- }
- dialog ||= document.getElementById("printServiceDialog");
- overlayPromise = overlayManager.register(
- dialog,
- /* canForceClose = */ true
- );
- document.getElementById("printCancel").onclick = abort;
- dialog.addEventListener("close", abort);
- }
- return overlayPromise;
- }
- PDFPrintServiceFactory.instance = {
- supportsPrinting: true,
- createPrintService(
- pdfDocument,
- pagesOverview,
- printContainer,
- printResolution,
- optionalContentConfigPromise,
- printAnnotationStoragePromise,
- l10n
- ) {
- if (activeService) {
- throw new Error("The print service is created and active.");
- }
- activeService = new PDFPrintService(
- pdfDocument,
- pagesOverview,
- printContainer,
- printResolution,
- optionalContentConfigPromise,
- printAnnotationStoragePromise,
- l10n
- );
- return activeService;
- },
- };
- export { PDFPrintService };
|