mirror of
https://gitlab.winehq.org/wine/wine-gecko.git
synced 2024-09-13 09:24:08 -07:00
36e9d9804d
--HG-- extra : rebase_source : e2c4038f36338af70fa6842d0146a63335002a9b
279 lines
9.4 KiB
JavaScript
279 lines
9.4 KiB
JavaScript
/* This Source Code Form is subject to the terms of the Mozilla Public
|
|
* License, v. 2.0. If a copy of the MPL was not distributed with this file,
|
|
* You can obtain one at http://mozilla.org/MPL/2.0/. */
|
|
|
|
"use strict";
|
|
|
|
let Cu = Components.utils;
|
|
let Ci = Components.interfaces;
|
|
let Cc = Components.classes;
|
|
|
|
Cu.import("resource://gre/modules/Services.jsm");
|
|
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
|
|
|
|
const NS_PREFBRANCH_PREFCHANGE_TOPIC_ID = "nsPref:changed";
|
|
const BROWSER_FRAMES_ENABLED_PREF = "dom.mozBrowserFramesEnabled";
|
|
|
|
function debug(msg) {
|
|
//dump("BrowserElementParent - " + msg + "\n");
|
|
}
|
|
|
|
function sendAsyncMsg(frameElement, msg, data) {
|
|
let mm = frameElement.QueryInterface(Ci.nsIFrameLoaderOwner)
|
|
.frameLoader
|
|
.messageManager;
|
|
|
|
mm.sendAsyncMessage('browser-element-api:' + msg, data);
|
|
}
|
|
|
|
/**
|
|
* The BrowserElementParent implements one half of <iframe mozbrowser>.
|
|
* (The other half is, unsurprisingly, BrowserElementChild.)
|
|
*
|
|
* We detect windows and docshells contained inside <iframe mozbrowser>s and
|
|
* inject script to listen for certain events in the child. We then listen to
|
|
* messages from the child script and take appropriate action here.
|
|
*/
|
|
|
|
function BrowserElementParent() {}
|
|
BrowserElementParent.prototype = {
|
|
classID: Components.ID("{ddeafdac-cb39-47c4-9cb8-c9027ee36d26}"),
|
|
QueryInterface: XPCOMUtils.generateQI([Ci.nsIObserver,
|
|
Ci.nsISupportsWeakReference]),
|
|
|
|
/**
|
|
* Called on app startup, and also when the browser frames enabled pref is
|
|
* changed.
|
|
*/
|
|
_init: function() {
|
|
if (this._initialized) {
|
|
return;
|
|
}
|
|
|
|
// If the pref is disabled, do nothing except wait for the pref to change.
|
|
// (This is important for tests, if nothing else.)
|
|
if (!this._browserFramesPrefEnabled()) {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefBranch);
|
|
prefs.addObserver(BROWSER_FRAMES_ENABLED_PREF, this, /* ownsWeak = */ true);
|
|
return;
|
|
}
|
|
|
|
debug("_init");
|
|
this._initialized = true;
|
|
|
|
this._screenshotListeners = {};
|
|
this._screenshotReqCounter = 0;
|
|
|
|
var os = Cc["@mozilla.org/observer-service;1"].getService(Ci.nsIObserverService);
|
|
os.addObserver(this, 'remote-browser-frame-shown', /* ownsWeak = */ true);
|
|
os.addObserver(this, 'in-process-browser-frame-shown', /* ownsWeak = */ true);
|
|
},
|
|
|
|
_browserFramesPrefEnabled: function() {
|
|
var prefs = Cc["@mozilla.org/preferences-service;1"].getService(Ci.nsIPrefBranch);
|
|
try {
|
|
return prefs.getBoolPref(BROWSER_FRAMES_ENABLED_PREF);
|
|
}
|
|
catch(e) {
|
|
return false;
|
|
}
|
|
},
|
|
|
|
_observeInProcessBrowserFrameShown: function(frameLoader) {
|
|
debug("In-process browser frame shown " + frameLoader);
|
|
this._setUpMessageManagerListeners(frameLoader);
|
|
},
|
|
|
|
_observeRemoteBrowserFrameShown: function(frameLoader) {
|
|
debug("Remote browser frame shown " + frameLoader);
|
|
this._setUpMessageManagerListeners(frameLoader);
|
|
},
|
|
|
|
_setUpMessageManagerListeners: function(frameLoader) {
|
|
let frameElement = frameLoader.QueryInterface(Ci.nsIFrameLoader).ownerElement;
|
|
if (!frameElement) {
|
|
debug("No frame element?");
|
|
return;
|
|
}
|
|
|
|
let mm = frameLoader.messageManager;
|
|
|
|
// Messages we receive are handled by functions with parameters
|
|
// (frameElement, data), where |data| is the message manager's data object.
|
|
|
|
let self = this;
|
|
function addMessageListener(msg, handler) {
|
|
mm.addMessageListener('browser-element-api:' + msg, handler.bind(self, frameElement));
|
|
}
|
|
|
|
addMessageListener("hello", this._recvHello);
|
|
addMessageListener("locationchange", this._fireEventFromMsg);
|
|
addMessageListener("loadstart", this._fireEventFromMsg);
|
|
addMessageListener("loadend", this._fireEventFromMsg);
|
|
addMessageListener("titlechange", this._fireEventFromMsg);
|
|
addMessageListener("iconchange", this._fireEventFromMsg);
|
|
addMessageListener("close", this._fireEventFromMsg);
|
|
addMessageListener("get-mozapp-manifest-url", this._sendMozAppManifestURL);
|
|
addMessageListener("keyevent", this._fireKeyEvent);
|
|
addMessageListener("showmodalprompt", this._handleShowModalPrompt);
|
|
mm.addMessageListener('browser-element-api:got-screenshot',
|
|
this._recvGotScreenshot.bind(this));
|
|
|
|
XPCNativeWrapper.unwrap(frameElement).getScreenshot =
|
|
this._getScreenshot.bind(this, mm, frameElement);
|
|
|
|
XPCNativeWrapper.unwrap(frameElement).setVisible =
|
|
this._setVisible.bind(this, mm, frameElement);
|
|
|
|
mm.loadFrameScript("chrome://global/content/BrowserElementChild.js",
|
|
/* allowDelayedLoad = */ true);
|
|
},
|
|
|
|
_recvHello: function(frameElement, data) {
|
|
debug("recvHello " + frameElement);
|
|
},
|
|
|
|
/**
|
|
* Fire either a vanilla or a custom event, depending on the contents of
|
|
* |data|.
|
|
*/
|
|
_fireEventFromMsg: function(frameElement, data) {
|
|
let name = data.name.substring('browser-element-api:'.length);
|
|
let detail = data.json;
|
|
|
|
debug('fireEventFromMsg: ' + name + ', ' + detail);
|
|
let evt = this._createEvent(frameElement, name, detail,
|
|
/* cancelable = */ false);
|
|
frameElement.dispatchEvent(evt);
|
|
},
|
|
|
|
_handleShowModalPrompt: function(frameElement, data) {
|
|
// Fire a showmodalprmopt event on the iframe. When this method is called,
|
|
// the child is spinning in a nested event loop waiting for an
|
|
// unblock-modal-prompt message.
|
|
//
|
|
// If the embedder calls preventDefault() on the showmodalprompt event,
|
|
// we'll block the child until event.detail.unblock() is called.
|
|
//
|
|
// Otherwise, if preventDefault() is not called, we'll send the
|
|
// unblock-modal-prompt message to the child as soon as the event is done
|
|
// dispatching.
|
|
|
|
let detail = data.json;
|
|
debug('handleShowPrompt ' + JSON.stringify(detail));
|
|
|
|
// Strip off the windowID property from the object we send along in the
|
|
// event.
|
|
let windowID = detail.windowID;
|
|
delete detail.windowID;
|
|
debug("Event will have detail: " + JSON.stringify(detail));
|
|
let evt = this._createEvent(frameElement, 'showmodalprompt', detail,
|
|
/* cancelable = */ true);
|
|
|
|
let unblockMsgSent = false;
|
|
function sendUnblockMsg() {
|
|
if (unblockMsgSent) {
|
|
return;
|
|
}
|
|
unblockMsgSent = true;
|
|
|
|
// We don't need to sanitize evt.detail.returnValue (e.g. converting the
|
|
// return value of confirm() to a boolean); Gecko does that for us.
|
|
|
|
let data = { windowID: windowID,
|
|
returnValue: evt.detail.returnValue };
|
|
sendAsyncMsg(frameElement, 'unblock-modal-prompt', data);
|
|
}
|
|
|
|
XPCNativeWrapper.unwrap(evt.detail).unblock = function() {
|
|
sendUnblockMsg();
|
|
};
|
|
|
|
frameElement.dispatchEvent(evt);
|
|
|
|
if (!evt.defaultPrevented) {
|
|
// Unblock the inner frame immediately. Otherwise we'll unblock upon
|
|
// evt.detail.unblock().
|
|
sendUnblockMsg();
|
|
}
|
|
},
|
|
|
|
_createEvent: function(frameElement, evtName, detail, cancelable) {
|
|
let win = frameElement.ownerDocument.defaultView;
|
|
let evt;
|
|
|
|
// This will have to change if we ever want to send a CustomEvent with null
|
|
// detail. For now, it's OK.
|
|
if (detail !== undefined && detail !== null) {
|
|
evt = new win.CustomEvent('mozbrowser' + evtName,
|
|
{bubbles: true, cancelable: cancelable,
|
|
detail: detail});
|
|
}
|
|
else {
|
|
evt = new win.Event('mozbrowser' + evtName,
|
|
{bubbles: true, cancelable: cancelable});
|
|
}
|
|
|
|
return evt;
|
|
},
|
|
|
|
_sendMozAppManifestURL: function(frameElement, data) {
|
|
return frameElement.getAttribute('mozapp');
|
|
},
|
|
|
|
_recvGotScreenshot: function(data) {
|
|
var req = this._screenshotListeners[data.json.id];
|
|
delete this._screenshotListeners[data.json.id];
|
|
Services.DOMRequest.fireSuccess(req, data.json.screenshot);
|
|
},
|
|
|
|
_getScreenshot: function(mm, frameElement) {
|
|
let id = 'req_' + this._screenshotReqCounter++;
|
|
let req = Services.DOMRequest
|
|
.createRequest(frameElement.ownerDocument.defaultView);
|
|
this._screenshotListeners[id] = req;
|
|
mm.sendAsyncMessage('browser-element-api:get-screenshot', {id: id});
|
|
return req;
|
|
},
|
|
|
|
_setVisible: function(mm, frameElement, visible) {
|
|
mm.sendAsyncMessage('browser-element-api:set-visible', {visible: visible});
|
|
},
|
|
|
|
_fireKeyEvent: function(frameElement, data) {
|
|
let win = frameElement.ownerDocument.defaultView;
|
|
let evt = frameElement.ownerDocument.createEvent("KeyboardEvent");
|
|
|
|
evt.initKeyEvent(data.json.type, true, true, win,
|
|
false, false, false, false, // modifiers
|
|
data.json.keyCode,
|
|
data.json.charCode);
|
|
|
|
frameElement.dispatchEvent(evt);
|
|
},
|
|
|
|
observe: function(subject, topic, data) {
|
|
switch(topic) {
|
|
case 'app-startup':
|
|
this._init();
|
|
break;
|
|
case NS_PREFBRANCH_PREFCHANGE_TOPIC_ID:
|
|
if (data == BROWSER_FRAMES_ENABLED_PREF) {
|
|
this._init();
|
|
}
|
|
break;
|
|
case 'remote-browser-frame-shown':
|
|
this._observeRemoteBrowserFrameShown(subject);
|
|
break;
|
|
case 'in-process-browser-frame-shown':
|
|
this._observeInProcessBrowserFrameShown(subject);
|
|
break;
|
|
case 'content-document-global-created':
|
|
this._observeContentGlobalCreated(subject);
|
|
break;
|
|
}
|
|
},
|
|
};
|
|
|
|
var NSGetFactory = XPCOMUtils.generateNSGetFactory([BrowserElementParent]);
|