/* 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/. */ Components.utils.import("resource://gre/modules/XPCOMUtils.jsm"); const Cc = Components.classes; const Ci = Components.interfaces; const CATEGORY_WAKEUP_REQUEST = "wakeup-request"; function MessageWakeupService() { }; MessageWakeupService.prototype = { classID: Components.ID("{f9798742-4f7b-4188-86ba-48b116412b29}"), QueryInterface: XPCOMUtils.generateQI([Ci.nsIObserver]), messagesData: [], get messageManager() { if (!this._messageManager) this._messageManager = Cc["@mozilla.org/parentprocessmessagemanager;1"]. getService(Ci.nsIMessageListenerManager); return this._messageManager; }, requestWakeup: function(aMessageName, aCid, aIid, aMethod) { this.messagesData[aMessageName] = { cid: aCid, iid: aIid, method: aMethod, }; this.messageManager.addMessageListener(aMessageName, this); }, receiveMessage: function(aMessage) { let data = this.messagesData[aMessage.name]; // TODO: When bug 593407 is ready, stop doing the wrappedJSObject hack // and use this line instead: // QueryInterface(Ci.nsIMessageListener); let service = Cc[data.cid][data.method](Ci[data.iid]). wrappedJSObject; // The receiveMessage() call itself may spin an event loop, and we // do not want to swap listeners in that - it would cause the current // message to be answered by two listeners. So, we call that first, // then queue the swap for the next event loop let ret = service.receiveMessage(aMessage); if (data.timer) { // Handle the case of two such messages happening in quick succession data.timer.cancel(); data.timer = null; } data.timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer); let self = this; data.timer.initWithCallback(function() { self.messageManager.addMessageListener(aMessage.name, service); self.messageManager.removeMessageListener(aMessage.name, self); delete self.messagesData[aMessage.name]; }, 0, Ci.nsITimer.TYPE_ONE_SHOT); return ret; }, observe: function TM_observe(aSubject, aTopic, aData) { switch (aTopic) { case "profile-after-change": { var catMan = Cc["@mozilla.org/categorymanager;1"]. getService(Ci.nsICategoryManager); var entries = catMan.enumerateCategory(CATEGORY_WAKEUP_REQUEST); while (entries.hasMoreElements()) { var entry = entries.getNext().QueryInterface(Ci.nsISupportsCString).data; var value = catMan.getCategoryEntry(CATEGORY_WAKEUP_REQUEST, entry); var parts = value.split(","); var cid = parts[0]; var iid = parts[1]; var method = parts[2]; var messages = parts.slice(3); messages.forEach(function(messageName) { this.requestWakeup(messageName, cid, iid, method); }, this); } } break; } }, }; var components = [MessageWakeupService]; this.NSGetFactory = XPCOMUtils.generateNSGetFactory(components);