Files

349 lines
11 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"
const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
Cu.import("resource://gre/modules/Services.jsm");
Cu.import("resource://gre/modules/Task.jsm");
Cu.import("resource://gre/modules/XPCOMUtils.jsm");
this.EXPORTED_SYMBOLS = ["sendMessageToJava", "Messaging", "EventDispatcher"];
XPCOMUtils.defineLazyServiceGetter(this, "uuidgen",
"@mozilla.org/uuid-generator;1",
"nsIUUIDGenerator");
function sendMessageToJava(aMessage, aCallback) {
Cu.reportError("sendMessageToJava is deprecated. Use Messaging API instead.");
if (aCallback) {
Messaging.sendRequestForResult(aMessage)
.then(result => aCallback(result, null),
error => aCallback(null, error));
} else {
Messaging.sendRequest(aMessage);
}
}
function DispatcherDelegate(dispatcher) {
this._dispatcher = dispatcher;
}
DispatcherDelegate.prototype = {
/**
* Register a listener to be notified of event(s).
*
* @param listener Target listener implementing nsIAndroidEventListener.
* @param events String or array of strings of events to listen to.
*/
registerListener: function (listener, events) {
this._dispatcher.registerListener(listener, events);
},
/**
* Unregister a previously-registered listener.
*
* @param listener Registered listener implementing nsIAndroidEventListener.
* @param events String or array of strings of events to stop listening to.
*/
unregisterListener: function (listener, events) {
this._dispatcher.unregisterListener(listener, events);
},
/**
* Dispatch an event to registered listeners for that event, and pass an
* optional data object and/or a optional callback interface to the
* listeners.
*
* @param event Name of event to dispatch.
* @param data Optional object containing data for the event.
* @param callback Optional callback implementing nsIAndroidEventCallback.
*/
dispatch: function (event, data, callback) {
this._dispatcher.dispatch(event, data, callback);
},
/**
* Implementations of Messaging APIs for backwards compatibility.
*/
/**
* Sends a request to Java.
*
* @param msg Message to send; must be an object with a "type" property
*/
sendRequest: function (msg) {
let type = msg.type;
msg.type = undefined;
this.dispatch(type, msg);
},
/**
* Sends a request to Java, returning a Promise that resolves to the response.
*
* @param msg Message to send; must be an object with a "type" property
* @returns A Promise resolving to the response
*/
sendRequestForResult: function (msg) {
return new Promise((resolve, reject) => {
let type = msg.type;
msg.type = undefined;
this.dispatch(type, msg, {
onSuccess: response => resolve(response),
onError: response => reject(response)
});
});
},
/**
* Add a listener for the given event.
*
* Only one request listener can be registered for a given event.
*
* Example usage:
* // aData is data sent from Java with the request. The return value is
* // used to respond to the request. The return type *must* be an instance
* // of Object.
* let listener = function (aData) {
* if (aData == "foo") {
* return { response: "bar" };
* }
* return {};
* };
* Messaging.addListener(listener, "Demo:Request");
*
* The listener may also be a generator function, useful for performing a
* task asynchronously. For example:
* let listener = function* (aData) {
* // Respond with "bar" after 2 seconds.
* yield new Promise(resolve => setTimeout(resolve, 2000));
* return { response: "bar" };
* };
* Messaging.addListener(listener, "Demo:Request");
*
* @param listener Listener callback taking a single data parameter
* (see example usage above).
* @param event Event name that this listener should observe.
*/
addListener: function (listener, event) {
if (this._requestHandler.listeners[event]) {
throw new Error("Error in addListener: A listener already exists for event " + event);
}
if (typeof listener !== "function") {
throw new Error("Error in addListener: Listener must be a function for event " + event);
}
this._requestHandler.listeners[event] = listener;
this.registerListener(this._requestHandler, event);
},
/**
* Removes a listener for a given event.
*
* @param event The event to stop listening for.
*/
removeListener: function (event) {
if (!this._requestHandler.listeners[event]) {
throw new Error("Error in removeListener: There is no listener for event " + event);
}
this._requestHandler.listeners[event] = undefined;
this.unregisterListener(this._requestHandler, event);
},
_requestHandler: {
listeners: {},
onEvent: Task.async(function* (event, data, callback) {
try {
let response = yield this.listeners[event](data.data);
callback.onSuccess(response);
} catch (e) {
Cu.reportError("Error in Messaging handler for " + event + ": " + e);
callback.onError({
message: e.message || (e && e.toString()),
stack: e.stack || Components.stack.formattedStack,
});
}
}),
},
};
var EventDispatcher = {
instance: new DispatcherDelegate(Services.androidBridge),
for: function (window) {
let view = window && window.arguments && window.arguments[0] &&
window.arguments[0].QueryInterface(Ci.nsIAndroidView);
if (!view) {
throw new Error("window is not a GeckoView-connected window");
}
return new DispatcherDelegate(view);
},
};
// For backwards compatibility.
// const Messaging = EventDispatcher.instance;
// Legacy code
var Messaging = {
/**
* Add a listener for the given message.
*
* Only one request listener can be registered for a given message.
*
* Example usage:
* // aData is data sent from Java with the request. The return value is
* // used to respond to the request. The return type *must* be an instance
* // of Object.
* let listener = function (aData) {
* if (aData == "foo") {
* return { response: "bar" };
* }
* return {};
* };
* Messaging.addListener(listener, "Demo:Request");
*
* The listener may also be a generator function, useful for performing a
* task asynchronously. For example:
* let listener = function* (aData) {
* // Respond with "bar" after 2 seconds.
* yield new Promise(resolve => setTimeout(resolve, 2000));
* return { response: "bar" };
* };
* Messaging.addListener(listener, "Demo:Request");
*
* @param aListener Listener callback taking a single data parameter (see
* example usage above).
* @param aMessage Event name that this listener should observe.
*/
addListener: function (aListener, aMessage) {
requestHandler.addListener(aListener, aMessage);
},
/**
* Removes a listener for a given message.
*
* @param aMessage The event to stop listening for.
*/
removeListener: function (aMessage) {
requestHandler.removeListener(aMessage);
},
/**
* Sends a request to Java.
*
* @param aMessage Message to send; must be an object with a "type" property
*/
sendRequest: function (aMessage) {
Services.androidBridge.handleGeckoMessage(aMessage);
},
/**
* Sends a request to Java, returning a Promise that resolves to the response.
*
* @param aMessage Message to send; must be an object with a "type" property
* @returns A Promise resolving to the response
*/
sendRequestForResult: function (aMessage) {
return new Promise((resolve, reject) => {
let id = uuidgen.generateUUID().toString();
let obs = {
observe: function (aSubject, aTopic, aData) {
let data = JSON.parse(aData);
if (data.__guid__ != id) {
return;
}
Services.obs.removeObserver(obs, aMessage.type + ":Response");
if (data.status === "success") {
resolve(data.response);
} else {
reject(data.response);
}
}
};
aMessage.__guid__ = id;
Services.obs.addObserver(obs, aMessage.type + ":Response", false);
this.sendRequest(aMessage);
});
},
/**
* Handles a request from Java, using the given listener method.
* This is mainly an internal method used by the RequestHandler object, but can be
* used in nsIObserver.observe implmentations that fall outside the normal usage
* patterns.
*
* @param aTopic The string name of the message
* @param aData The data sent to the observe method from Java
* @param aListener A function that takes a JSON data argument and returns a
* response which is sent to Java.
*/
handleRequest: Task.async(function* (aTopic, aData, aListener) {
let wrapper = JSON.parse(aData);
try {
let response = yield aListener(wrapper.data);
if (typeof response !== "object" || response === null) {
throw new Error("Gecko request listener did not return an object");
}
Messaging.sendRequest({
type: "Gecko:Request" + wrapper.id,
response: response
});
} catch (e) {
Cu.reportError("Error in Messaging handler for " + aTopic + ": " + e);
Messaging.sendRequest({
type: "Gecko:Request" + wrapper.id,
error: {
message: e.message || (e && e.toString()),
stack: e.stack || Components.stack.formattedStack,
}
});
}
})
};
var requestHandler = {
_listeners: {},
addListener: function (aListener, aMessage) {
if (aMessage in this._listeners) {
throw new Error("Error in addListener: A listener already exists for message " + aMessage);
}
if (typeof aListener !== "function") {
throw new Error("Error in addListener: Listener must be a function for message " + aMessage);
}
this._listeners[aMessage] = aListener;
Services.obs.addObserver(this, aMessage, false);
},
removeListener: function (aMessage) {
if (!(aMessage in this._listeners)) {
throw new Error("Error in removeListener: There is no listener for message " + aMessage);
}
delete this._listeners[aMessage];
Services.obs.removeObserver(this, aMessage);
},
observe: function(aSubject, aTopic, aData) {
let listener = this._listeners[aTopic];
Messaging.handleRequest(aTopic, aData, listener);
}
};