445 lines
12 KiB
JavaScript
445 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 "../extensions/firefox/tools/l10n.js";
|
|
import { DefaultExternalServices, PDFViewerApplication } from "./app.js";
|
|
import { isPdfFile, PDFDataRangeTransport, shadow } from "pdfjs-lib";
|
|
import { BasePreferences } from "./preferences.js";
|
|
import { DEFAULT_SCALE_VALUE } from "./ui_utils.js";
|
|
import { getL10nFallback } from "./l10n_utils.js";
|
|
|
|
if (typeof PDFJSDev === "undefined" || !PDFJSDev.test("MOZCENTRAL")) {
|
|
throw new Error(
|
|
'Module "./firefoxcom.js" shall not be used outside MOZCENTRAL builds.'
|
|
);
|
|
}
|
|
|
|
class FirefoxCom {
|
|
/**
|
|
* Creates an event that the extension is listening for and will
|
|
* synchronously respond to.
|
|
* NOTE: It is recommended to use requestAsync() instead since one day we may
|
|
* not be able to synchronously reply.
|
|
* @param {string} action - The action to trigger.
|
|
* @param {Object|string} [data] - The data to send.
|
|
* @returns {*} The response.
|
|
*/
|
|
static requestSync(action, data) {
|
|
const request = document.createTextNode("");
|
|
document.documentElement.appendChild(request);
|
|
|
|
const sender = document.createEvent("CustomEvent");
|
|
sender.initCustomEvent("pdf.js.message", true, false, {
|
|
action,
|
|
data,
|
|
sync: true,
|
|
});
|
|
request.dispatchEvent(sender);
|
|
const response = sender.detail.response;
|
|
request.remove();
|
|
|
|
return response;
|
|
}
|
|
|
|
/**
|
|
* 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.appendChild(request);
|
|
|
|
const sender = document.createEvent("CustomEvent");
|
|
sender.initCustomEvent("pdf.js.message", true, false, {
|
|
action,
|
|
data,
|
|
sync: false,
|
|
responseExpected: !!callback,
|
|
});
|
|
request.dispatchEvent(sender);
|
|
}
|
|
}
|
|
|
|
class DownloadManager {
|
|
constructor() {
|
|
this._openBlobUrls = new WeakMap();
|
|
}
|
|
|
|
downloadUrl(url, filename) {
|
|
FirefoxCom.request("download", {
|
|
originalUrl: url,
|
|
filename,
|
|
});
|
|
}
|
|
|
|
downloadData(data, filename, contentType) {
|
|
const blobUrl = URL.createObjectURL(
|
|
new Blob([data], { type: contentType })
|
|
);
|
|
|
|
FirefoxCom.requestAsync("download", {
|
|
blobUrl,
|
|
originalUrl: blobUrl,
|
|
filename,
|
|
isAttachment: true,
|
|
}).then(error => {
|
|
URL.revokeObjectURL(blobUrl);
|
|
});
|
|
}
|
|
|
|
/**
|
|
* @returns {boolean} Indicating if the data was opened.
|
|
*/
|
|
openOrDownloadData(element, data, filename) {
|
|
const isPdfData = isPdfFile(filename);
|
|
const contentType = isPdfData ? "application/pdf" : "";
|
|
|
|
if (isPdfData) {
|
|
let blobUrl = this._openBlobUrls.get(element);
|
|
if (!blobUrl) {
|
|
blobUrl = URL.createObjectURL(new Blob([data], { type: contentType }));
|
|
this._openBlobUrls.set(element, blobUrl);
|
|
}
|
|
// Let Firefox's content handler catch the URL and display the PDF.
|
|
const viewerUrl = blobUrl + "#filename=" + encodeURIComponent(filename);
|
|
|
|
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(element);
|
|
}
|
|
}
|
|
|
|
this.downloadData(data, filename, contentType);
|
|
return false;
|
|
}
|
|
|
|
download(blob, url, filename, sourceEventType = "download") {
|
|
const blobUrl = URL.createObjectURL(blob);
|
|
|
|
FirefoxCom.requestAsync("download", {
|
|
blobUrl,
|
|
originalUrl: url,
|
|
filename,
|
|
sourceEventType,
|
|
}).then(error => {
|
|
if (error) {
|
|
// If downloading failed in `PdfStreamConverter.jsm` it's very unlikely
|
|
// that attempting to fallback and re-download would be helpful here.
|
|
console.error("`ChromeActions.download` failed.");
|
|
}
|
|
URL.revokeObjectURL(blobUrl);
|
|
});
|
|
}
|
|
}
|
|
|
|
class FirefoxPreferences extends BasePreferences {
|
|
async _writeToStorage(prefObj) {
|
|
return FirefoxCom.requestAsync("setPreferences", prefObj);
|
|
}
|
|
|
|
async _readFromStorage(prefObj) {
|
|
const prefStr = await FirefoxCom.requestAsync("getPreferences", prefObj);
|
|
return JSON.parse(prefStr);
|
|
}
|
|
}
|
|
|
|
class MozL10n {
|
|
constructor(mozL10n) {
|
|
this.mozL10n = mozL10n;
|
|
}
|
|
|
|
async getLanguage() {
|
|
return this.mozL10n.getLanguage();
|
|
}
|
|
|
|
async getDirection() {
|
|
return this.mozL10n.getDirection();
|
|
}
|
|
|
|
async get(key, args = null, fallback = getL10nFallback(key, args)) {
|
|
return this.mozL10n.get(key, args, fallback);
|
|
}
|
|
|
|
async translate(element) {
|
|
this.mozL10n.translate(element);
|
|
}
|
|
}
|
|
|
|
(function listenFindEvents() {
|
|
const events = [
|
|
"find",
|
|
"findagain",
|
|
"findhighlightallchange",
|
|
"findcasesensitivitychange",
|
|
"findentirewordchange",
|
|
"findbarclose",
|
|
"finddiacriticmatchingchange",
|
|
];
|
|
const findLen = "find".length;
|
|
|
|
const handleEvent = function ({ type, detail }) {
|
|
if (!PDFViewerApplication.initialized) {
|
|
return;
|
|
}
|
|
if (type === "findbarclose") {
|
|
PDFViewerApplication.eventBus.dispatch(type, { source: window });
|
|
return;
|
|
}
|
|
PDFViewerApplication.eventBus.dispatch("find", {
|
|
source: window,
|
|
type: type.substring(findLen),
|
|
query: detail.query,
|
|
phraseSearch: true,
|
|
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 (!PDFViewerApplication.initialized) {
|
|
return;
|
|
}
|
|
// Avoid attempting to needlessly reset the zoom level *twice* in a row,
|
|
// when using the `Ctrl + 0` keyboard shortcut.
|
|
if (
|
|
type === "zoomreset" &&
|
|
PDFViewerApplication.pdfViewer.currentScaleValue === DEFAULT_SCALE_VALUE
|
|
) {
|
|
return;
|
|
}
|
|
PDFViewerApplication.eventBus.dispatch(type, { source: window });
|
|
};
|
|
|
|
for (const event of events) {
|
|
window.addEventListener(event, handleEvent);
|
|
}
|
|
})();
|
|
|
|
(function listenSaveEvent() {
|
|
const handleEvent = function ({ type, detail }) {
|
|
if (!PDFViewerApplication.initialized) {
|
|
return;
|
|
}
|
|
PDFViewerApplication.eventBus.dispatch(type, { source: window });
|
|
};
|
|
|
|
window.addEventListener("save", handleEvent);
|
|
})();
|
|
|
|
class FirefoxComDataRangeTransport extends PDFDataRangeTransport {
|
|
requestDataRange(begin, end) {
|
|
FirefoxCom.request("requestDataRange", { begin, end });
|
|
}
|
|
|
|
abort() {
|
|
// Sync call to ensure abort is really started.
|
|
FirefoxCom.requestSync("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 FirefoxExternalServices extends DefaultExternalServices {
|
|
static updateFindControlState(data) {
|
|
FirefoxCom.request("updateFindControlState", data);
|
|
}
|
|
|
|
static updateFindMatchesCount(data) {
|
|
FirefoxCom.request("updateFindMatchesCount", data);
|
|
}
|
|
|
|
static 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(
|
|
args.pdfUrl,
|
|
args.length,
|
|
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.requestSync("initPassiveLoading", null);
|
|
}
|
|
|
|
static reportTelemetry(data) {
|
|
FirefoxCom.request("reportTelemetry", JSON.stringify(data));
|
|
}
|
|
|
|
static createDownloadManager(options) {
|
|
return new DownloadManager();
|
|
}
|
|
|
|
static createPreferences() {
|
|
return new FirefoxPreferences();
|
|
}
|
|
|
|
static createL10n(options) {
|
|
const mozL10n = document.mozL10n;
|
|
// TODO refactor mozL10n.setExternalLocalizerServices
|
|
return new MozL10n(mozL10n);
|
|
}
|
|
|
|
static createScripting(options) {
|
|
return FirefoxScripting;
|
|
}
|
|
|
|
static get supportsIntegratedFind() {
|
|
const support = FirefoxCom.requestSync("supportsIntegratedFind");
|
|
return shadow(this, "supportsIntegratedFind", support);
|
|
}
|
|
|
|
static get supportsDocumentFonts() {
|
|
const support = FirefoxCom.requestSync("supportsDocumentFonts");
|
|
return shadow(this, "supportsDocumentFonts", support);
|
|
}
|
|
|
|
static get supportedMouseWheelZoomModifierKeys() {
|
|
const support = FirefoxCom.requestSync(
|
|
"supportedMouseWheelZoomModifierKeys"
|
|
);
|
|
return shadow(this, "supportedMouseWheelZoomModifierKeys", support);
|
|
}
|
|
|
|
static get isInAutomation() {
|
|
// Returns the value of `Cu.isInAutomation`, which is only `true` when e.g.
|
|
// various test-suites are running in mozilla-central.
|
|
const isInAutomation = FirefoxCom.requestSync("isInAutomation");
|
|
return shadow(this, "isInAutomation", isInAutomation);
|
|
}
|
|
}
|
|
PDFViewerApplication.externalServices = FirefoxExternalServices;
|
|
|
|
// l10n.js for Firefox extension expects services to be set.
|
|
document.mozL10n.setExternalLocalizerServices({
|
|
getLocale() {
|
|
return FirefoxCom.requestSync("getLocale", null);
|
|
},
|
|
|
|
getStrings(key) {
|
|
return FirefoxCom.requestSync("getStrings", null);
|
|
},
|
|
});
|
|
|
|
export { DownloadManager, FirefoxCom };
|