145 lines
4.6 KiB
JavaScript
145 lines
4.6 KiB
JavaScript
|
const Cc = Components.classes;
|
||
|
const Ci = Components.interfaces;
|
||
|
const Cr = Components.results;
|
||
|
const Cu = Components.utils;
|
||
|
|
||
|
const PDF_CONTENT_TYPE = "application/pdf";
|
||
|
|
||
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
||
|
Cu.import("resource://gre/modules/Services.jsm");
|
||
|
|
||
|
// TODO
|
||
|
// Add some download progress event
|
||
|
|
||
|
function log(aMsg) {
|
||
|
let msg = "pdfContentHandler.js: " + (aMsg.join ? aMsg.join("") : aMsg);
|
||
|
Cc["@mozilla.org/consoleservice;1"].getService(Ci.nsIConsoleService)
|
||
|
.logStringMessage(msg);
|
||
|
dump(msg + "\n");
|
||
|
};
|
||
|
|
||
|
function loadDocument(aWindow, aDocumentUrl) {
|
||
|
let xhr = Cc["@mozilla.org/xmlextras/xmlhttprequest;1"]
|
||
|
.createInstance(Ci.nsIXMLHttpRequest);
|
||
|
xhr.open("GET", aDocumentUrl);
|
||
|
xhr.mozResponseType = xhr.responseType = "arraybuffer";
|
||
|
xhr.onreadystatechange = function() {
|
||
|
if (xhr.readyState == 4 && xhr.status == 200) {
|
||
|
let data = (xhr.mozResponseArrayBuffer || xhr.mozResponse ||
|
||
|
xhr.responseArrayBuffer || xhr.response);
|
||
|
try {
|
||
|
var view = new Uint8Array(data);
|
||
|
|
||
|
// I think accessing aWindow.wrappedJSObject returns a
|
||
|
// XPCSafeJSObjectWrapper and so it is safe but mrbkap can confirm that
|
||
|
let window = aWindow.wrappedJSObject;
|
||
|
var arrayBuffer = new window.ArrayBuffer(data.byteLength);
|
||
|
var view2 = new window.Uint8Array(arrayBuffer);
|
||
|
view2.set(view);
|
||
|
|
||
|
let evt = window.document.createEvent("CustomEvent");
|
||
|
evt.initCustomEvent("pdfloaded", false, false, arrayBuffer);
|
||
|
window.document.dispatchEvent(evt);
|
||
|
} catch(e) {
|
||
|
log("Error - " + e);
|
||
|
}
|
||
|
}
|
||
|
};
|
||
|
xhr.send(null);
|
||
|
};
|
||
|
|
||
|
let WebProgressListener = {
|
||
|
init: function(aWindow, aUrl) {
|
||
|
this._locationHasChanged = false;
|
||
|
this._documentUrl = aUrl;
|
||
|
|
||
|
let flags = Ci.nsIWebProgress.NOTIFY_LOCATION |
|
||
|
Ci.nsIWebProgress.NOTIFY_STATE_NETWORK |
|
||
|
Ci.nsIWebProgress.NOTIFY_STATE_DOCUMENT;
|
||
|
|
||
|
let docShell = aWindow.QueryInterface(Ci.nsIInterfaceRequestor)
|
||
|
.getInterface(Ci.nsIWebNavigation)
|
||
|
.QueryInterface(Ci.nsIDocShell);
|
||
|
let webProgress = docShell.QueryInterface(Ci.nsIInterfaceRequestor)
|
||
|
.getInterface(Ci.nsIWebProgress);
|
||
|
try {
|
||
|
webProgress.removeProgressListener(this);
|
||
|
} catch(e) {}
|
||
|
webProgress.addProgressListener(this, flags);
|
||
|
},
|
||
|
|
||
|
onStateChange: function onStateChange(aWebProgress, aRequest, aStateFlags, aStatus) {
|
||
|
const complete = Ci.nsIWebProgressListener.STATE_IS_WINDOW +
|
||
|
Ci.nsIWebProgressListener.STATE_STOP;
|
||
|
if ((aStateFlags & complete) == complete && this._locationHasChanged) {
|
||
|
aWebProgress.removeProgressListener(this);
|
||
|
loadDocument(aWebProgress.DOMWindow, this._documentUrl);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
onProgressChange: function onProgressChange(aWebProgress, aRequest, aCurSelf, aMaxSelf, aCurTotal, aMaxTotal) {
|
||
|
},
|
||
|
|
||
|
onLocationChange: function onLocationChange(aWebProgress, aRequest, aLocationURI) {
|
||
|
this._locationHasChanged = true;
|
||
|
},
|
||
|
|
||
|
onStatusChange: function onStatusChange(aWebProgress, aRequest, aStatus, aMessage) {
|
||
|
},
|
||
|
|
||
|
onSecurityChange: function onSecurityChange(aWebProgress, aRequest, aState) {
|
||
|
},
|
||
|
|
||
|
QueryInterface: function QueryInterface(aIID) {
|
||
|
if (aIID.equals(Ci.nsIWebProgressListener) ||
|
||
|
aIID.equals(Ci.nsISupportsWeakReference) ||
|
||
|
aIID.equals(Ci.nsISupports)) {
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
throw Components.results.NS_ERROR_NO_INTERFACE;
|
||
|
}
|
||
|
};
|
||
|
|
||
|
|
||
|
function pdfContentHandler() {
|
||
|
}
|
||
|
|
||
|
pdfContentHandler.prototype = {
|
||
|
handleContent: function handleContent(aMimetype, aContext, aRequest) {
|
||
|
if (aMimetype != PDF_CONTENT_TYPE)
|
||
|
throw Cr.NS_ERROR_WONT_HANDLE_CONTENT;
|
||
|
|
||
|
if (!(aRequest instanceof Ci.nsIChannel))
|
||
|
throw Cr.NS_ERROR_WONT_HANDLE_CONTENT;
|
||
|
|
||
|
let window = null;
|
||
|
let callbacks = aRequest.notificationCallbacks ?
|
||
|
aRequest.notificationCallbacks :
|
||
|
aRequest.loadGroup.notificationCallbacks;
|
||
|
if (!callbacks)
|
||
|
return;
|
||
|
|
||
|
aRequest.cancel(Cr.NS_BINDING_ABORTED);
|
||
|
let uri = aRequest.URI;
|
||
|
|
||
|
window = callbacks.getInterface(Ci.nsIDOMWindow);
|
||
|
WebProgressListener.init(window, uri.spec);
|
||
|
|
||
|
try {
|
||
|
let url = Services.prefs.getCharPref("extensions.pdf.js.url");
|
||
|
url = url.replace("%s", uri.spec);
|
||
|
window.location = url;
|
||
|
} catch(e) {
|
||
|
log("Error - " + e);
|
||
|
}
|
||
|
},
|
||
|
|
||
|
classID: Components.ID("{2278dfd0-b75c-11e0-8257-1ba3d93c9f1a}"),
|
||
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIContentHandler]),
|
||
|
};
|
||
|
|
||
|
var NSGetFactory = XPCOMUtils.generateNSGetFactory([pdfContentHandler]);
|
||
|
|
||
|
|