425 lines
12 KiB
JavaScript
425 lines
12 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 { isPdfFile, PDFDataRangeTransport } from "pdfjs-lib";
|
|
import { BaseExternalServices } from "./external_services.js";
|
|
import { BasePreferences } from "./preferences.js";
|
|
import { DEFAULT_SCALE_VALUE } from "./ui_utils.js";
|
|
import { L10n } from "./l10n.js";
|
|
|
|
if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) {
|
|
throw new Error(
|
|
'Module "./firefoxcom.js" shall not be used outside MOZCENTRAL builds.'
|
|
);
|
|
}
|
|
|
|
let viewerApp = { initialized: false };
|
|
function initCom(app) {
|
|
viewerApp = app;
|
|
}
|
|
|
|
class FirefoxCom {
|
|
/**
|
|
* Creates an event that the extension is listening for and will
|
|
* asynchronously respond to.
|
|
* @param {string} action - The action to trigger.
|
|
* @param {Object|string} [data] - The data to send.
|
|
* @returns {Promise<any>} A promise that is resolved with the response data.
|
|
*/
|
|
static requestAsync(action, data) {
|
|
return new Promise(resolve => {
|
|
this.request(action, data, resolve);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Creates an event that the extension is listening for and will, optionally,
|
|
* asynchronously respond to.
|
|
* @param {string} action - The action to trigger.
|
|
* @param {Object|string} [data] - The data to send.
|
|
*/
|
|
static request(action, data, callback = null) {
|
|
const request = document.createTextNode("");
|
|
if (callback) {
|
|
request.addEventListener(
|
|
"pdf.js.response",
|
|
event => {
|
|
const response = event.detail.response;
|
|
event.target.remove();
|
|
|
|
callback(response);
|
|
},
|
|
{ once: true }
|
|
);
|
|
}
|
|
document.documentElement.append(request);
|
|
|
|
const sender = new CustomEvent("pdf.js.message", {
|
|
bubbles: true,
|
|
cancelable: false,
|
|
detail: {
|
|
action,
|
|
data,
|
|
responseExpected: !!callback,
|
|
},
|
|
});
|
|
request.dispatchEvent(sender);
|
|
}
|
|
}
|
|
|
|
class DownloadManager {
|
|
#openBlobUrls = new WeakMap();
|
|
|
|
downloadUrl(url, filename, options = {}) {
|
|
FirefoxCom.request("download", {
|
|
originalUrl: url,
|
|
filename,
|
|
options,
|
|
});
|
|
}
|
|
|
|
downloadData(data, filename, contentType) {
|
|
const blobUrl = URL.createObjectURL(
|
|
new Blob([data], { type: contentType })
|
|
);
|
|
|
|
FirefoxCom.request("download", {
|
|
blobUrl,
|
|
originalUrl: blobUrl,
|
|
filename,
|
|
isAttachment: true,
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @returns {boolean} Indicating if the data was opened.
|
|
*/
|
|
openOrDownloadData(data, filename, dest = null) {
|
|
const isPdfData = isPdfFile(filename);
|
|
const contentType = isPdfData ? "application/pdf" : "";
|
|
|
|
if (isPdfData) {
|
|
let blobUrl = this.#openBlobUrls.get(data);
|
|
if (!blobUrl) {
|
|
blobUrl = URL.createObjectURL(new Blob([data], { type: contentType }));
|
|
this.#openBlobUrls.set(data, blobUrl);
|
|
}
|
|
// Let Firefox's content handler catch the URL and display the PDF.
|
|
// NOTE: This cannot use a query string for the filename, see
|
|
// https://bugzilla.mozilla.org/show_bug.cgi?id=1632644#c5
|
|
let viewerUrl = blobUrl + "#filename=" + encodeURIComponent(filename);
|
|
if (dest) {
|
|
viewerUrl += `&filedest=${escape(dest)}`;
|
|
}
|
|
|
|
try {
|
|
window.open(viewerUrl);
|
|
return true;
|
|
} catch (ex) {
|
|
console.error(`openOrDownloadData: ${ex}`);
|
|
// Release the `blobUrl`, since opening it failed, and fallback to
|
|
// downloading the PDF file.
|
|
URL.revokeObjectURL(blobUrl);
|
|
this.#openBlobUrls.delete(data);
|
|
}
|
|
}
|
|
|
|
this.downloadData(data, filename, contentType);
|
|
return false;
|
|
}
|
|
|
|
download(blob, url, filename, options = {}) {
|
|
const blobUrl = URL.createObjectURL(blob);
|
|
|
|
FirefoxCom.request("download", {
|
|
blobUrl,
|
|
originalUrl: url,
|
|
filename,
|
|
options,
|
|
});
|
|
}
|
|
}
|
|
|
|
class Preferences extends BasePreferences {
|
|
async _readFromStorage(prefObj) {
|
|
return FirefoxCom.requestAsync("getPreferences", prefObj);
|
|
}
|
|
}
|
|
|
|
(function listenFindEvents() {
|
|
const events = [
|
|
"find",
|
|
"findagain",
|
|
"findhighlightallchange",
|
|
"findcasesensitivitychange",
|
|
"findentirewordchange",
|
|
"findbarclose",
|
|
"finddiacriticmatchingchange",
|
|
];
|
|
const findLen = "find".length;
|
|
|
|
const handleEvent = function ({ type, detail }) {
|
|
if (!viewerApp.initialized) {
|
|
return;
|
|
}
|
|
if (type === "findbarclose") {
|
|
viewerApp.eventBus.dispatch(type, { source: window });
|
|
return;
|
|
}
|
|
viewerApp.eventBus.dispatch("find", {
|
|
source: window,
|
|
type: type.substring(findLen),
|
|
query: detail.query,
|
|
caseSensitive: !!detail.caseSensitive,
|
|
entireWord: !!detail.entireWord,
|
|
highlightAll: !!detail.highlightAll,
|
|
findPrevious: !!detail.findPrevious,
|
|
matchDiacritics: !!detail.matchDiacritics,
|
|
});
|
|
};
|
|
|
|
for (const event of events) {
|
|
window.addEventListener(event, handleEvent);
|
|
}
|
|
})();
|
|
|
|
(function listenZoomEvents() {
|
|
const events = ["zoomin", "zoomout", "zoomreset"];
|
|
const handleEvent = function ({ type, detail }) {
|
|
if (!viewerApp.initialized) {
|
|
return;
|
|
}
|
|
// Avoid attempting to needlessly reset the zoom level *twice* in a row,
|
|
// when using the `Ctrl + 0` keyboard shortcut.
|
|
if (
|
|
type === "zoomreset" &&
|
|
viewerApp.pdfViewer.currentScaleValue === DEFAULT_SCALE_VALUE
|
|
) {
|
|
return;
|
|
}
|
|
viewerApp.eventBus.dispatch(type, { source: window });
|
|
};
|
|
|
|
for (const event of events) {
|
|
window.addEventListener(event, handleEvent);
|
|
}
|
|
})();
|
|
|
|
(function listenSaveEvent() {
|
|
const handleEvent = function ({ type, detail }) {
|
|
if (!viewerApp.initialized) {
|
|
return;
|
|
}
|
|
viewerApp.eventBus.dispatch("download", { source: window });
|
|
};
|
|
|
|
window.addEventListener("save", handleEvent);
|
|
})();
|
|
|
|
(function listenEditingEvent() {
|
|
const handleEvent = function ({ detail }) {
|
|
if (!viewerApp.initialized) {
|
|
return;
|
|
}
|
|
viewerApp.eventBus.dispatch("editingaction", {
|
|
source: window,
|
|
name: detail.name,
|
|
});
|
|
};
|
|
|
|
window.addEventListener("editingaction", handleEvent);
|
|
})();
|
|
|
|
if (PDFJSDev.test("GECKOVIEW")) {
|
|
(function listenQueryEvents() {
|
|
window.addEventListener("pdf.js.query", async ({ detail: { queryId } }) => {
|
|
let result = null;
|
|
if (viewerApp.initialized && queryId === "canDownloadInsteadOfPrint") {
|
|
result = false;
|
|
const { pdfDocument, pdfViewer } = viewerApp;
|
|
if (pdfDocument) {
|
|
try {
|
|
const hasUnchangedAnnotations =
|
|
pdfDocument.annotationStorage.size === 0;
|
|
// WillPrint is called just before printing the document and could
|
|
// lead to have modified annotations.
|
|
const hasWillPrint =
|
|
pdfViewer.enableScripting &&
|
|
!!(await pdfDocument.getJSActions())?.WillPrint;
|
|
const hasUnchangedOptionalContent = (
|
|
await pdfViewer.optionalContentConfigPromise
|
|
).hasInitialVisibility;
|
|
|
|
result =
|
|
hasUnchangedAnnotations &&
|
|
!hasWillPrint &&
|
|
hasUnchangedOptionalContent;
|
|
} catch {
|
|
console.warn("Unable to check if the document can be downloaded.");
|
|
}
|
|
}
|
|
}
|
|
|
|
window.dispatchEvent(
|
|
new CustomEvent("pdf.js.query.answer", {
|
|
bubbles: true,
|
|
cancelable: false,
|
|
detail: {
|
|
queryId,
|
|
value: result,
|
|
},
|
|
})
|
|
);
|
|
});
|
|
})();
|
|
}
|
|
|
|
class FirefoxComDataRangeTransport extends PDFDataRangeTransport {
|
|
requestDataRange(begin, end) {
|
|
FirefoxCom.request("requestDataRange", { begin, end });
|
|
}
|
|
|
|
// NOTE: This method is currently not invoked in the Firefox PDF Viewer.
|
|
abort() {
|
|
FirefoxCom.request("abortLoading", null);
|
|
}
|
|
}
|
|
|
|
class FirefoxScripting {
|
|
static async createSandbox(data) {
|
|
const success = await FirefoxCom.requestAsync("createSandbox", data);
|
|
if (!success) {
|
|
throw new Error("Cannot create sandbox.");
|
|
}
|
|
}
|
|
|
|
static async dispatchEventInSandbox(event) {
|
|
FirefoxCom.request("dispatchEventInSandbox", event);
|
|
}
|
|
|
|
static async destroySandbox() {
|
|
FirefoxCom.request("destroySandbox", null);
|
|
}
|
|
}
|
|
|
|
class MLManager {
|
|
guess(data) {
|
|
return FirefoxCom.requestAsync("mlGuess", data);
|
|
}
|
|
}
|
|
|
|
class ExternalServices extends BaseExternalServices {
|
|
updateFindControlState(data) {
|
|
FirefoxCom.request("updateFindControlState", data);
|
|
}
|
|
|
|
updateFindMatchesCount(data) {
|
|
FirefoxCom.request("updateFindMatchesCount", data);
|
|
}
|
|
|
|
initPassiveLoading(callbacks) {
|
|
let pdfDataRangeTransport;
|
|
|
|
window.addEventListener("message", function windowMessage(e) {
|
|
if (e.source !== null) {
|
|
// The message MUST originate from Chrome code.
|
|
console.warn("Rejected untrusted message from " + e.origin);
|
|
return;
|
|
}
|
|
const args = e.data;
|
|
|
|
if (typeof args !== "object" || !("pdfjsLoadAction" in args)) {
|
|
return;
|
|
}
|
|
switch (args.pdfjsLoadAction) {
|
|
case "supportsRangedLoading":
|
|
if (args.done && !args.data) {
|
|
callbacks.onError();
|
|
break;
|
|
}
|
|
pdfDataRangeTransport = new FirefoxComDataRangeTransport(
|
|
args.length,
|
|
args.data,
|
|
args.done,
|
|
args.filename
|
|
);
|
|
|
|
callbacks.onOpenWithTransport(pdfDataRangeTransport);
|
|
break;
|
|
case "range":
|
|
pdfDataRangeTransport.onDataRange(args.begin, args.chunk);
|
|
break;
|
|
case "rangeProgress":
|
|
pdfDataRangeTransport.onDataProgress(args.loaded);
|
|
break;
|
|
case "progressiveRead":
|
|
pdfDataRangeTransport.onDataProgressiveRead(args.chunk);
|
|
|
|
// Don't forget to report loading progress as well, since otherwise
|
|
// the loadingBar won't update when `disableRange=true` is set.
|
|
pdfDataRangeTransport.onDataProgress(args.loaded, args.total);
|
|
break;
|
|
case "progressiveDone":
|
|
pdfDataRangeTransport?.onDataProgressiveDone();
|
|
break;
|
|
case "progress":
|
|
callbacks.onProgress(args.loaded, args.total);
|
|
break;
|
|
case "complete":
|
|
if (!args.data) {
|
|
callbacks.onError(args.errorCode);
|
|
break;
|
|
}
|
|
callbacks.onOpenWithData(args.data, args.filename);
|
|
break;
|
|
}
|
|
});
|
|
FirefoxCom.request("initPassiveLoading", null);
|
|
}
|
|
|
|
reportTelemetry(data) {
|
|
FirefoxCom.request("reportTelemetry", data);
|
|
}
|
|
|
|
updateEditorStates(data) {
|
|
FirefoxCom.request("updateEditorStates", data);
|
|
}
|
|
|
|
async createL10n() {
|
|
const [localeProperties] = await Promise.all([
|
|
FirefoxCom.requestAsync("getLocaleProperties", null),
|
|
document.l10n.ready,
|
|
]);
|
|
return new L10n(localeProperties, document.l10n);
|
|
}
|
|
|
|
createScripting() {
|
|
return FirefoxScripting;
|
|
}
|
|
|
|
async getNimbusExperimentData() {
|
|
if (!PDFJSDev.test("GECKOVIEW")) {
|
|
return null;
|
|
}
|
|
const nimbusData = await FirefoxCom.requestAsync(
|
|
"getNimbusExperimentData",
|
|
null
|
|
);
|
|
return nimbusData && JSON.parse(nimbusData);
|
|
}
|
|
}
|
|
|
|
export { DownloadManager, ExternalServices, initCom, MLManager, Preferences };
|