350 lines
11 KiB
JavaScript
350 lines
11 KiB
JavaScript
|
/* Copyright 2014 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.
|
||
|
*/
|
||
|
|
||
|
/* globals PDFJS, Promise */
|
||
|
|
||
|
'use strict';
|
||
|
|
||
|
PDFJS.useOnlyCssZoom = true;
|
||
|
PDFJS.disableTextLayer = true;
|
||
|
PDFJS.maxImageSize = 1024 * 1024;
|
||
|
PDFJS.workerSrc = '../pdfjs-components/build/pdf.worker.js';
|
||
|
PDFJS.cMapUrl = '../pdfjs-components/cmaps/';
|
||
|
PDFJS.cMapPacked = true;
|
||
|
|
||
|
var DEFAULT_SCALE_DELTA = 1.1;
|
||
|
var MIN_SCALE = 0.25;
|
||
|
var MAX_SCALE = 10.0;
|
||
|
var DEFAULT_SCALE_VALUE = 'auto';
|
||
|
|
||
|
var PDFViewerApplication = {
|
||
|
pdfDocument: null,
|
||
|
pdfViewer: null,
|
||
|
pdfHistory: null,
|
||
|
pdfLinkService: null,
|
||
|
|
||
|
open: function (params) {
|
||
|
var url = params.url, originalUrl = params.originalUrl;
|
||
|
var self = this;
|
||
|
this.setTitleUsingUrl(originalUrl);
|
||
|
|
||
|
// Loading document.
|
||
|
var loadingTask = PDFJS.getDocument(url);
|
||
|
loadingTask.onProgress = function (progressData) {
|
||
|
self.progress(progressData.loaded / progressData.total);
|
||
|
};
|
||
|
loadingTask.then(function (pdfDocument) {
|
||
|
// Document loaded, specifying document for the viewer.
|
||
|
this.pdfDocument = pdfDocument;
|
||
|
this.pdfViewer.setDocument(pdfDocument);
|
||
|
this.pdfLinkService.setDocument(pdfDocument);
|
||
|
this.pdfHistory.initialize(pdfDocument.fingerprint);
|
||
|
|
||
|
this.loadingBar.hide();
|
||
|
this.setTitleUsingMetadata(pdfDocument);
|
||
|
}.bind(this), function (exception) {
|
||
|
var message = exception && exception.message;
|
||
|
var loadingErrorMessage = mozL10n.get('loading_error', null,
|
||
|
'An error occurred while loading the PDF.');
|
||
|
|
||
|
if (exception instanceof PDFJS.InvalidPDFException) {
|
||
|
// change error message also for other builds
|
||
|
loadingErrorMessage = mozL10n.get('invalid_file_error', null,
|
||
|
'Invalid or corrupted PDF file.');
|
||
|
} else if (exception instanceof PDFJS.MissingPDFException) {
|
||
|
// special message for missing PDFs
|
||
|
loadingErrorMessage = mozL10n.get('missing_file_error', null,
|
||
|
'Missing PDF file.');
|
||
|
} else if (exception instanceof PDFJS.UnexpectedResponseException) {
|
||
|
loadingErrorMessage = mozL10n.get('unexpected_response_error', null,
|
||
|
'Unexpected server response.');
|
||
|
}
|
||
|
|
||
|
var moreInfo = {
|
||
|
message: message
|
||
|
};
|
||
|
self.error(loadingErrorMessage, moreInfo);
|
||
|
self.loadingBar.hide();
|
||
|
});
|
||
|
},
|
||
|
|
||
|
get loadingBar() {
|
||
|
var bar = new PDFJS.ProgressBar('#loadingBar', {});
|
||
|
|
||
|
return PDFJS.shadow(this, 'loadingBar', bar);
|
||
|
},
|
||
|
|
||
|
setTitleUsingUrl: function pdfViewSetTitleUsingUrl(url) {
|
||
|
this.url = url;
|
||
|
var title = PDFJS.getFileName(url) || url;
|
||
|
try {
|
||
|
title = decodeURIComponent(title);
|
||
|
} catch (e) {
|
||
|
// decodeURIComponent may throw URIError,
|
||
|
// fall back to using the unprocessed url in that case
|
||
|
}
|
||
|
this.setTitle(title);
|
||
|
},
|
||
|
|
||
|
setTitleUsingMetadata: function (pdfDocument) {
|
||
|
var self = this;
|
||
|
pdfDocument.getMetadata().then(function(data) {
|
||
|
var info = data.info, metadata = data.metadata;
|
||
|
self.documentInfo = info;
|
||
|
self.metadata = metadata;
|
||
|
|
||
|
// Provides some basic debug information
|
||
|
console.log('PDF ' + pdfDocument.fingerprint + ' [' +
|
||
|
info.PDFFormatVersion + ' ' + (info.Producer || '-').trim() +
|
||
|
' / ' + (info.Creator || '-').trim() + ']' +
|
||
|
' (PDF.js: ' + (PDFJS.version || '-') +
|
||
|
(!PDFJS.disableWebGL ? ' [WebGL]' : '') + ')');
|
||
|
|
||
|
var pdfTitle;
|
||
|
if (metadata && metadata.has('dc:title')) {
|
||
|
var title = metadata.get('dc:title');
|
||
|
// Ghostscript sometimes returns 'Untitled', so prevent setting the
|
||
|
// title to 'Untitled.
|
||
|
if (title !== 'Untitled') {
|
||
|
pdfTitle = title;
|
||
|
}
|
||
|
}
|
||
|
|
||
|
if (!pdfTitle && info && info['Title']) {
|
||
|
pdfTitle = info['Title'];
|
||
|
}
|
||
|
|
||
|
if (pdfTitle) {
|
||
|
self.setTitle(pdfTitle + ' - ' + document.title);
|
||
|
}
|
||
|
});
|
||
|
},
|
||
|
|
||
|
setTitle: function pdfViewSetTitle(title) {
|
||
|
document.title = title;
|
||
|
document.getElementById('activityTitle').textContent = title;
|
||
|
},
|
||
|
|
||
|
error: function pdfViewError(message, moreInfo) {
|
||
|
var moreInfoText = mozL10n.get('error_version_info',
|
||
|
{version: PDFJS.version || '?', build: PDFJS.build || '?'},
|
||
|
'PDF.js v{{version}} (build: {{build}})') + '\n';
|
||
|
|
||
|
if (moreInfo) {
|
||
|
moreInfoText +=
|
||
|
mozL10n.get('error_message', {message: moreInfo.message},
|
||
|
'Message: {{message}}');
|
||
|
if (moreInfo.stack) {
|
||
|
moreInfoText += '\n' +
|
||
|
mozL10n.get('error_stack', {stack: moreInfo.stack},
|
||
|
'Stack: {{stack}}');
|
||
|
} else {
|
||
|
if (moreInfo.filename) {
|
||
|
moreInfoText += '\n' +
|
||
|
mozL10n.get('error_file', {file: moreInfo.filename},
|
||
|
'File: {{file}}');
|
||
|
}
|
||
|
if (moreInfo.lineNumber) {
|
||
|
moreInfoText += '\n' +
|
||
|
mozL10n.get('error_line', {line: moreInfo.lineNumber},
|
||
|
'Line: {{line}}');
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
var errorWrapper = document.getElementById('errorWrapper');
|
||
|
errorWrapper.removeAttribute('hidden');
|
||
|
|
||
|
var errorMessage = document.getElementById('errorMessage');
|
||
|
errorMessage.textContent = message;
|
||
|
|
||
|
var closeButton = document.getElementById('errorClose');
|
||
|
closeButton.onclick = function() {
|
||
|
errorWrapper.setAttribute('hidden', 'true');
|
||
|
};
|
||
|
|
||
|
var errorMoreInfo = document.getElementById('errorMoreInfo');
|
||
|
var moreInfoButton = document.getElementById('errorShowMore');
|
||
|
var lessInfoButton = document.getElementById('errorShowLess');
|
||
|
moreInfoButton.onclick = function() {
|
||
|
errorMoreInfo.removeAttribute('hidden');
|
||
|
moreInfoButton.setAttribute('hidden', 'true');
|
||
|
lessInfoButton.removeAttribute('hidden');
|
||
|
errorMoreInfo.style.height = errorMoreInfo.scrollHeight + 'px';
|
||
|
};
|
||
|
lessInfoButton.onclick = function() {
|
||
|
errorMoreInfo.setAttribute('hidden', 'true');
|
||
|
moreInfoButton.removeAttribute('hidden');
|
||
|
lessInfoButton.setAttribute('hidden', 'true');
|
||
|
};
|
||
|
moreInfoButton.removeAttribute('hidden');
|
||
|
lessInfoButton.setAttribute('hidden', 'true');
|
||
|
errorMoreInfo.value = moreInfoText;
|
||
|
},
|
||
|
|
||
|
progress: function pdfViewProgress(level) {
|
||
|
var percent = Math.round(level * 100);
|
||
|
// Updating the bar if value increases.
|
||
|
if (percent > this.loadingBar.percent || isNaN(percent)) {
|
||
|
this.loadingBar.percent = percent;
|
||
|
}
|
||
|
},
|
||
|
|
||
|
get pagesCount() {
|
||
|
return this.pdfDocument.numPages;
|
||
|
},
|
||
|
|
||
|
set page(val) {
|
||
|
this.pdfViewer.currentPageNumber = val;
|
||
|
},
|
||
|
|
||
|
get page() {
|
||
|
return this.pdfViewer.currentPageNumber;
|
||
|
},
|
||
|
|
||
|
zoomIn: function pdfViewZoomIn(ticks) {
|
||
|
var newScale = this.pdfViewer.currentScale;
|
||
|
do {
|
||
|
newScale = (newScale * DEFAULT_SCALE_DELTA).toFixed(2);
|
||
|
newScale = Math.ceil(newScale * 10) / 10;
|
||
|
newScale = Math.min(MAX_SCALE, newScale);
|
||
|
} while (--ticks && newScale < MAX_SCALE);
|
||
|
this.pdfViewer.currentScaleValue = newScale;
|
||
|
},
|
||
|
|
||
|
zoomOut: function pdfViewZoomOut(ticks) {
|
||
|
var newScale = this.pdfViewer.currentScale;
|
||
|
do {
|
||
|
newScale = (newScale / DEFAULT_SCALE_DELTA).toFixed(2);
|
||
|
newScale = Math.floor(newScale * 10) / 10;
|
||
|
newScale = Math.max(MIN_SCALE, newScale);
|
||
|
} while (--ticks && newScale > MIN_SCALE);
|
||
|
this.pdfViewer.currentScaleValue = newScale;
|
||
|
},
|
||
|
|
||
|
initUI: function pdfViewInitUI() {
|
||
|
var linkService = new PDFJS.PDFLinkService();
|
||
|
this.pdfLinkService = linkService;
|
||
|
|
||
|
var container = document.getElementById('viewerContainer');
|
||
|
var pdfViewer = new PDFJS.PDFViewer({
|
||
|
container: container,
|
||
|
linkService: linkService
|
||
|
});
|
||
|
this.pdfViewer = pdfViewer;
|
||
|
linkService.setViewer(pdfViewer);
|
||
|
|
||
|
this.pdfHistory = new PDFJS.PDFHistory({
|
||
|
linkService: linkService
|
||
|
});
|
||
|
linkService.setHistory(this.pdfHistory);
|
||
|
|
||
|
document.getElementById('previous').addEventListener('click', function() {
|
||
|
PDFViewerApplication.page--;
|
||
|
});
|
||
|
|
||
|
document.getElementById('next').addEventListener('click', function() {
|
||
|
PDFViewerApplication.page++;
|
||
|
});
|
||
|
|
||
|
document.getElementById('zoomIn').addEventListener('click', function() {
|
||
|
PDFViewerApplication.zoomIn();
|
||
|
});
|
||
|
|
||
|
document.getElementById('zoomOut').addEventListener('click', function() {
|
||
|
PDFViewerApplication.zoomOut();
|
||
|
});
|
||
|
|
||
|
document.getElementById('pageNumber').addEventListener('click', function() {
|
||
|
this.select();
|
||
|
});
|
||
|
|
||
|
document.getElementById('pageNumber').addEventListener('change',
|
||
|
function() {
|
||
|
// Handle the user inputting a floating point number.
|
||
|
PDFViewerApplication.page = (this.value | 0);
|
||
|
|
||
|
if (this.value !== (this.value | 0).toString()) {
|
||
|
this.value = PDFViewerApplication.page;
|
||
|
}
|
||
|
});
|
||
|
|
||
|
container.addEventListener('pagesinit', function () {
|
||
|
// We can use pdfViewer now, e.g. let's change default scale.
|
||
|
pdfViewer.currentScaleValue = DEFAULT_SCALE_VALUE;
|
||
|
});
|
||
|
|
||
|
container.addEventListener('pagechange', function (evt) {
|
||
|
var page = evt.pageNumber;
|
||
|
if (evt.previousPageNumber !== page) {
|
||
|
document.getElementById('pageNumber').value = page;
|
||
|
}
|
||
|
var numPages = PDFViewerApplication.pagesCount;
|
||
|
|
||
|
document.getElementById('previous').disabled = (page <= 1);
|
||
|
document.getElementById('next').disabled = (page >= numPages);
|
||
|
}, true);
|
||
|
}
|
||
|
};
|
||
|
|
||
|
document.addEventListener('DOMContentLoaded', function () {
|
||
|
PDFViewerApplication.initUI();
|
||
|
}, true);
|
||
|
|
||
|
(function animationStartedClosure() {
|
||
|
// The offsetParent is not set until the PDF.js iframe or object is visible.
|
||
|
// Waiting for first animation.
|
||
|
PDFViewerApplication.animationStartedPromise = new Promise(
|
||
|
function (resolve) {
|
||
|
window.requestAnimationFrame(resolve);
|
||
|
});
|
||
|
})();
|
||
|
|
||
|
// Support of the new version of navigator.mozL10n -- in PDF.js older/custom
|
||
|
// version is used.
|
||
|
var mozL10n = {
|
||
|
get: function (id, args, fallback) {
|
||
|
var s = (navigator.mozL10n && navigator.mozL10n.get(id)) || fallback;
|
||
|
s = s.replace(/\{\{\s*(\w+)\s*\}\}/g, function (all, key) {
|
||
|
return args[key] || '';
|
||
|
});
|
||
|
return s;
|
||
|
},
|
||
|
|
||
|
translate: function (fragment) {
|
||
|
if (navigator.mozL10n) {
|
||
|
navigator.mozL10n.translateFragment(fragment);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
|
||
|
window.navigator.mozSetMessageHandler('activity', function(activity) {
|
||
|
var blob = activity.source.data.blob;
|
||
|
var fileURL = activity.source.data.url ||
|
||
|
activity.source.data.filename ||
|
||
|
' '; // if no url or filename, use a non-empty string
|
||
|
|
||
|
var url = URL.createObjectURL(blob);
|
||
|
// We need to delay opening until all HTML is loaded.
|
||
|
PDFViewerApplication.animationStartedPromise.then(function () {
|
||
|
PDFViewerApplication.open({url: url, originalUrl: fileURL});
|
||
|
|
||
|
var header = document.getElementById('header');
|
||
|
header.addEventListener('action', function() {
|
||
|
activity.postResult('close');
|
||
|
});
|
||
|
});
|
||
|
});
|