/* Copyright 2017 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. */ 'use strict'; var uiUtils = require('./ui_utils.js'); var overlayManager = require('./overlay_manager.js'); var app = require('./app.js'); var pdfjsLib = require('./pdfjs.js'); var mozL10n = uiUtils.mozL10n; var CSS_UNITS = uiUtils.CSS_UNITS; var PDFPrintServiceFactory = app.PDFPrintServiceFactory; var OverlayManager = overlayManager.OverlayManager; var activeService = null; function renderPage(activeServiceOnEntry, pdfDocument, pageNumber, size) { var scratchCanvas = activeService.scratchCanvas; var PRINT_RESOLUTION = 150; var PRINT_UNITS = PRINT_RESOLUTION / 72.0; scratchCanvas.width = Math.floor(size.width * PRINT_UNITS); scratchCanvas.height = Math.floor(size.height * PRINT_UNITS); var width = Math.floor(size.width * CSS_UNITS) + 'px'; var height = Math.floor(size.height * CSS_UNITS) + 'px'; var ctx = scratchCanvas.getContext('2d'); ctx.save(); ctx.fillStyle = 'rgb(255, 255, 255)'; ctx.fillRect(0, 0, scratchCanvas.width, scratchCanvas.height); ctx.restore(); return pdfDocument.getPage(pageNumber).then(function (pdfPage) { var renderContext = { canvasContext: ctx, transform: [ PRINT_UNITS, 0, 0, PRINT_UNITS, 0, 0 ], viewport: pdfPage.getViewport(1, size.rotation), intent: 'print' }; return pdfPage.render(renderContext).promise; }).then(function () { return { width: width, height: height }; }); } function PDFPrintService(pdfDocument, pagesOverview, printContainer) { this.pdfDocument = pdfDocument; this.pagesOverview = pagesOverview; this.printContainer = printContainer; this.currentPage = -1; this.scratchCanvas = document.createElement('canvas'); } PDFPrintService.prototype = { layout: function () { this.throwIfInactive(); var body = document.querySelector('body'); body.setAttribute('data-pdfjsprinting', true); var 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!'); } this.pageStyleSheet = document.createElement('style'); var pageSize = this.pagesOverview[0]; this.pageStyleSheet.textContent = '@supports ((size:A4) and (size:1pt 1pt)) {' + '@page { size: ' + pageSize.width + 'pt ' + pageSize.height + 'pt;}' + '}'; body.appendChild(this.pageStyleSheet); }, destroy: function () { if (activeService !== this) { return; } this.printContainer.textContent = ''; if (this.pageStyleSheet && this.pageStyleSheet.parentNode) { this.pageStyleSheet.parentNode.removeChild(this.pageStyleSheet); this.pageStyleSheet = null; } this.scratchCanvas.width = this.scratchCanvas.height = 0; this.scratchCanvas = null; activeService = null; ensureOverlay().then(function () { if (OverlayManager.active !== 'printServiceOverlay') { return; } OverlayManager.close('printServiceOverlay'); }); }, renderPages: function () { var pageCount = this.pagesOverview.length; var renderNextPage = function (resolve, reject) { this.throwIfInactive(); if (++this.currentPage >= pageCount) { renderProgress(pageCount, pageCount); resolve(); return; } var index = this.currentPage; renderProgress(index, pageCount); renderPage(this, this.pdfDocument, index + 1, this.pagesOverview[index]).then(this.useRenderedPage.bind(this)).then(function () { renderNextPage(resolve, reject); }, reject); }.bind(this); return new Promise(renderNextPage); }, useRenderedPage: function (printItem) { this.throwIfInactive(); var img = document.createElement('img'); img.style.width = printItem.width; img.style.height = printItem.height; var scratchCanvas = this.scratchCanvas; if ('toBlob' in scratchCanvas && !pdfjsLib.PDFJS.disableCreateObjectURL) { scratchCanvas.toBlob(function (blob) { img.src = URL.createObjectURL(blob); }); } else { img.src = scratchCanvas.toDataURL(); } var wrapper = document.createElement('div'); wrapper.appendChild(img); this.printContainer.appendChild(wrapper); return new Promise(function (resolve, reject) { img.onload = resolve; img.onerror = reject; }); }, performPrint: function () { this.throwIfInactive(); return new Promise(function (resolve) { setTimeout(function () { if (!this.active) { resolve(); return; } print.call(window); setTimeout(resolve, 20); }.bind(this), 0); }.bind(this)); }, get active() { return this === activeService; }, throwIfInactive: function () { if (!this.active) { throw new Error('This print request was cancelled or completed.'); } } }; var print = window.print; window.print = function print() { if (activeService) { console.warn('Ignored window.print() because of a pending print job.'); return; } ensureOverlay().then(function () { if (activeService) { OverlayManager.open('printServiceOverlay'); } }); try { dispatchEvent('beforeprint'); } finally { if (!activeService) { console.error('Expected print service to be initialized.'); if (OverlayManager.active === 'printServiceOverlay') { OverlayManager.close('printServiceOverlay'); } return; } var activeServiceOnEntry = activeService; activeService.renderPages().then(function () { return activeServiceOnEntry.performPrint(); }).catch(function () { }).then(function () { if (activeServiceOnEntry.active) { abort(); } }); } }; function dispatchEvent(eventType) { var 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) { var progressContainer = document.getElementById('printServiceOverlay'); var progress = Math.round(100 * index / total); var progressBar = progressContainer.querySelector('progress'); var progressPerc = progressContainer.querySelector('.relative-progress'); progressBar.value = progress; progressPerc.textContent = mozL10n.get('print_progress_percent', { progress: progress }, progress + '%'); } var hasAttachEvent = !!document.attachEvent; window.addEventListener('keydown', function (event) { if (event.keyCode === 80 && (event.ctrlKey || event.metaKey) && !event.altKey && (!event.shiftKey || window.chrome || window.opera)) { window.print(); if (hasAttachEvent) { return; } event.preventDefault(); if (event.stopImmediatePropagation) { event.stopImmediatePropagation(); } else { event.stopPropagation(); } return; } }, true); if (hasAttachEvent) { document.attachEvent('onkeydown', function (event) { event = event || window.event; if (event.keyCode === 80 && event.ctrlKey) { event.keyCode = 0; return false; } }); } if ('onbeforeprint' in window) { var stopPropagationIfNeeded = function (event) { if (event.detail !== 'custom' && event.stopImmediatePropagation) { event.stopImmediatePropagation(); } }; window.addEventListener('beforeprint', stopPropagationIfNeeded); window.addEventListener('afterprint', stopPropagationIfNeeded); } var overlayPromise; function ensureOverlay() { if (!overlayPromise) { overlayPromise = OverlayManager.register('printServiceOverlay', document.getElementById('printServiceOverlay'), abort, true); document.getElementById('printCancel').onclick = abort; } return overlayPromise; } PDFPrintServiceFactory.instance = { supportsPrinting: true, createPrintService: function (pdfDocument, pagesOverview, printContainer) { if (activeService) { throw new Error('The print service is created and active.'); } activeService = new PDFPrintService(pdfDocument, pagesOverview, printContainer); return activeService; } }; exports.PDFPrintService = PDFPrintService;