2012-01-05 17:04:03 -08:00
|
|
|
/* -*- Mode: Java; tab-width: 2; indent-tabs-mode: nil; c-basic-offset: 2 -*- /
|
|
|
|
/* vim: set shiftwidth=2 tabstop=2 autoindent cindent expandtab: */
|
2012-02-08 04:37:00 -08:00
|
|
|
/* 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/. */
|
2011-12-06 15:52:26 -08:00
|
|
|
|
2012-02-28 14:01:48 -08:00
|
|
|
Cu.import('resource://gre/modules/ContactService.jsm');
|
2012-06-04 16:12:24 -07:00
|
|
|
Cu.import('resource://gre/modules/SettingsChangeNotifier.jsm');
|
2013-10-02 10:27:15 -07:00
|
|
|
Cu.import('resource://gre/modules/DataStoreChangeNotifier.jsm');
|
2012-06-22 00:39:07 -07:00
|
|
|
Cu.import('resource://gre/modules/AlarmService.jsm');
|
2012-07-20 08:41:30 -07:00
|
|
|
Cu.import('resource://gre/modules/ActivitiesService.jsm');
|
2012-08-19 12:00:19 -07:00
|
|
|
Cu.import('resource://gre/modules/PermissionPromptHelper.jsm');
|
2013-10-02 18:27:53 -07:00
|
|
|
Cu.import('resource://gre/modules/NotificationDB.jsm');
|
2012-08-29 14:41:35 -07:00
|
|
|
Cu.import('resource://gre/modules/Payment.jsm');
|
2012-10-02 22:38:03 -07:00
|
|
|
Cu.import("resource://gre/modules/AppsUtils.jsm");
|
2012-10-03 13:27:46 -07:00
|
|
|
Cu.import('resource://gre/modules/UserAgentOverrides.jsm');
|
2012-11-07 03:53:24 -08:00
|
|
|
Cu.import('resource://gre/modules/Keyboard.jsm');
|
2013-03-04 18:19:00 -08:00
|
|
|
Cu.import('resource://gre/modules/ErrorPage.jsm');
|
2013-10-29 01:12:45 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2012-10-04 09:42:41 -07:00
|
|
|
Cu.import('resource://gre/modules/NetworkStatsService.jsm');
|
2012-10-04 14:19:05 -07:00
|
|
|
#endif
|
2012-02-01 16:46:50 -08:00
|
|
|
|
2013-12-13 03:25:26 -08:00
|
|
|
// Identity
|
2012-11-12 16:46:35 -08:00
|
|
|
Cu.import('resource://gre/modules/SignInToWebsite.jsm');
|
|
|
|
SignInToWebsiteController.init();
|
2014-03-17 09:58:16 -07:00
|
|
|
|
|
|
|
#ifdef MOZ_SERVICES_FXACCOUNTS
|
2013-12-13 03:25:26 -08:00
|
|
|
Cu.import('resource://gre/modules/FxAccountsMgmtService.jsm');
|
2014-03-17 09:58:16 -07:00
|
|
|
#endif
|
2012-11-12 16:46:35 -08:00
|
|
|
|
2013-11-27 23:18:08 -08:00
|
|
|
Cu.import('resource://gre/modules/DownloadsAPI.jsm');
|
|
|
|
|
2014-04-07 06:59:48 -07:00
|
|
|
XPCOMUtils.defineLazyModuleGetter(this, "SystemAppProxy",
|
|
|
|
"resource://gre/modules/SystemAppProxy.jsm");
|
|
|
|
|
2014-01-22 10:06:29 -08:00
|
|
|
Cu.import('resource://gre/modules/Webapps.jsm');
|
|
|
|
DOMApplicationRegistry.allAppsLaunchable = true;
|
|
|
|
|
2012-05-04 11:02:05 -07:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(Services, 'env',
|
|
|
|
'@mozilla.org/process/environment;1',
|
|
|
|
'nsIEnvironment');
|
2012-02-01 16:46:50 -08:00
|
|
|
|
2012-05-04 11:02:05 -07:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(Services, 'ss',
|
|
|
|
'@mozilla.org/content/style-sheet-service;1',
|
|
|
|
'nsIStyleSheetService');
|
2012-02-17 13:37:53 -08:00
|
|
|
|
2012-11-06 17:45:54 -08:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(this, 'gSystemMessenger',
|
|
|
|
'@mozilla.org/system-message-internal;1',
|
|
|
|
'nsISystemMessagesInternal');
|
|
|
|
|
2012-05-04 11:02:05 -07:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(Services, 'fm',
|
|
|
|
'@mozilla.org/focus-manager;1',
|
|
|
|
'nsIFocusManager');
|
2012-03-02 15:32:46 -08:00
|
|
|
|
2012-03-16 09:50:45 -07:00
|
|
|
XPCOMUtils.defineLazyGetter(this, 'DebuggerServer', function() {
|
|
|
|
Cu.import('resource://gre/modules/devtools/dbg-server.jsm');
|
|
|
|
return DebuggerServer;
|
|
|
|
});
|
|
|
|
|
2012-08-15 09:00:02 -07:00
|
|
|
XPCOMUtils.defineLazyGetter(this, "ppmm", function() {
|
|
|
|
return Cc["@mozilla.org/parentprocessmessagemanager;1"]
|
2012-08-27 07:13:02 -07:00
|
|
|
.getService(Ci.nsIMessageListenerManager);
|
2012-08-15 09:00:02 -07:00
|
|
|
});
|
|
|
|
|
2012-10-04 10:29:43 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
XPCOMUtils.defineLazyGetter(this, "libcutils", function () {
|
|
|
|
Cu.import("resource://gre/modules/systemlibs.js");
|
|
|
|
return libcutils;
|
|
|
|
});
|
|
|
|
#endif
|
|
|
|
|
2013-02-15 14:26:58 -08:00
|
|
|
#ifdef MOZ_CAPTIVEDETECT
|
2012-11-14 19:26:30 -08:00
|
|
|
XPCOMUtils.defineLazyServiceGetter(Services, 'captivePortalDetector',
|
2013-02-15 14:26:58 -08:00
|
|
|
'@mozilla.org/toolkit/captive-detector;1',
|
2012-11-14 19:26:30 -08:00
|
|
|
'nsICaptivePortalDetector');
|
|
|
|
#endif
|
|
|
|
|
2012-07-12 19:28:19 -07:00
|
|
|
function getContentWindow() {
|
|
|
|
return shell.contentBrowser.contentWindow;
|
|
|
|
}
|
|
|
|
|
2012-12-03 06:42:36 -08:00
|
|
|
function debug(str) {
|
|
|
|
dump(' -*- Shell.js: ' + str + '\n');
|
|
|
|
}
|
|
|
|
|
2013-04-22 13:09:15 -07:00
|
|
|
#ifdef MOZ_CRASHREPORTER
|
|
|
|
function debugCrashReport(aStr) {
|
|
|
|
dump('Crash reporter : ' + aStr);
|
|
|
|
}
|
|
|
|
#else
|
|
|
|
function debugCrashReport(aStr) {}
|
|
|
|
#endif
|
|
|
|
|
2011-12-06 15:52:26 -08:00
|
|
|
var shell = {
|
2012-07-18 14:40:24 -07:00
|
|
|
|
|
|
|
get CrashSubmit() {
|
|
|
|
delete this.CrashSubmit;
|
2013-01-04 14:30:28 -08:00
|
|
|
#ifdef MOZ_CRASHREPORTER
|
2012-07-18 14:40:24 -07:00
|
|
|
Cu.import("resource://gre/modules/CrashSubmit.jsm", this);
|
|
|
|
return this.CrashSubmit;
|
2013-01-04 14:30:28 -08:00
|
|
|
#else
|
2013-04-22 13:09:15 -07:00
|
|
|
dump('Crash reporter : disabled at build time.');
|
2013-01-04 14:30:28 -08:00
|
|
|
return this.CrashSubmit = null;
|
|
|
|
#endif
|
2012-07-18 14:40:24 -07:00
|
|
|
},
|
|
|
|
|
2012-11-26 15:29:20 -08:00
|
|
|
onlineForCrashReport: function shell_onlineForCrashReport() {
|
|
|
|
let wifiManager = navigator.mozWifiManager;
|
|
|
|
let onWifi = (wifiManager &&
|
|
|
|
(wifiManager.connection.status == 'connected'));
|
|
|
|
return !Services.io.offline && onWifi;
|
|
|
|
},
|
|
|
|
|
2012-10-29 17:27:09 -07:00
|
|
|
reportCrash: function shell_reportCrash(isChrome, aCrashID) {
|
2012-10-11 07:18:00 -07:00
|
|
|
let crashID = aCrashID;
|
2012-08-21 15:29:46 -07:00
|
|
|
try {
|
2012-10-29 17:27:09 -07:00
|
|
|
// For chrome crashes, we want to report the lastRunCrashID.
|
|
|
|
if (isChrome) {
|
2012-10-11 07:18:00 -07:00
|
|
|
crashID = Cc["@mozilla.org/xre/app-info;1"]
|
|
|
|
.getService(Ci.nsIXULRuntime).lastRunCrashID;
|
2012-10-29 17:27:09 -07:00
|
|
|
}
|
2013-04-22 13:09:15 -07:00
|
|
|
} catch(e) {
|
|
|
|
debugCrashReport('Failed to fetch crash id. Crash ID is "' + crashID
|
|
|
|
+ '" Exception: ' + e);
|
|
|
|
}
|
2012-07-18 17:19:41 -07:00
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
// Bail if there isn't a valid crashID.
|
2013-01-04 14:30:28 -08:00
|
|
|
if (!this.CrashSubmit || !crashID && !this.CrashSubmit.pendingIDs().length) {
|
2012-10-29 11:18:50 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-12-11 14:06:48 -08:00
|
|
|
// purge the queue.
|
|
|
|
this.CrashSubmit.pruneSavedDumps();
|
|
|
|
|
2014-02-27 13:36:27 -08:00
|
|
|
// check for environment affecting crash reporting
|
|
|
|
let env = Cc["@mozilla.org/process/environment;1"]
|
|
|
|
.getService(Ci.nsIEnvironment);
|
|
|
|
let shutdown = env.get("MOZ_CRASHREPORTER_SHUTDOWN");
|
|
|
|
if (shutdown) {
|
|
|
|
let appStartup = Cc["@mozilla.org/toolkit/app-startup;1"]
|
|
|
|
.getService(Ci.nsIAppStartup);
|
|
|
|
appStartup.quit(Ci.nsIAppStartup.eForceQuit);
|
|
|
|
}
|
|
|
|
|
|
|
|
let noReport = env.get("MOZ_CRASHREPORTER_NO_REPORT");
|
|
|
|
if (noReport) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
try {
|
2012-10-29 17:27:09 -07:00
|
|
|
// Check if we should automatically submit this crash.
|
2013-04-22 13:09:15 -07:00
|
|
|
if (Services.prefs.getBoolPref('app.reportCrashes')) {
|
2012-10-29 11:18:50 -07:00
|
|
|
this.submitCrash(crashID);
|
2013-04-22 13:09:15 -07:00
|
|
|
} else {
|
2013-05-02 09:57:41 -07:00
|
|
|
this.deleteCrash(crashID);
|
2012-10-29 11:18:50 -07:00
|
|
|
}
|
2013-04-22 13:09:15 -07:00
|
|
|
} catch (e) {
|
|
|
|
debugCrashReport('Can\'t fetch app.reportCrashes. Exception: ' + e);
|
|
|
|
}
|
2012-10-29 17:27:09 -07:00
|
|
|
|
2013-01-11 04:53:45 -08:00
|
|
|
// We can get here if we're just submitting old pending crashes.
|
|
|
|
// Check that there's a valid crashID so that we only notify the
|
|
|
|
// user if a crash just happened and not when we OOM. Bug 829477
|
|
|
|
if (crashID) {
|
|
|
|
this.sendChromeEvent({
|
|
|
|
type: "handle-crash",
|
|
|
|
crashID: crashID,
|
|
|
|
chrome: isChrome
|
|
|
|
});
|
|
|
|
}
|
2012-07-18 14:40:24 -07:00
|
|
|
},
|
|
|
|
|
2013-05-02 09:57:41 -07:00
|
|
|
deleteCrash: function shell_deleteCrash(aCrashID) {
|
|
|
|
if (aCrashID) {
|
|
|
|
debugCrashReport('Deleting pending crash: ' + aCrashID);
|
|
|
|
shell.CrashSubmit.delete(aCrashID);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-12-25 15:06:13 -08:00
|
|
|
// this function submit the pending crashes.
|
|
|
|
// make sure you are online.
|
|
|
|
submitQueuedCrashes: function shell_submitQueuedCrashes() {
|
|
|
|
// submit the pending queue.
|
|
|
|
let pending = shell.CrashSubmit.pendingIDs();
|
|
|
|
for (let crashid of pending) {
|
2013-04-22 13:09:15 -07:00
|
|
|
debugCrashReport('Submitting crash: ' + crashid);
|
2012-12-25 15:06:13 -08:00
|
|
|
shell.CrashSubmit.submit(crashid);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
// This function submits a crash when we're online.
|
|
|
|
submitCrash: function shell_submitCrash(aCrashID) {
|
2012-11-26 15:29:20 -08:00
|
|
|
if (this.onlineForCrashReport()) {
|
2012-12-25 15:06:13 -08:00
|
|
|
this.submitQueuedCrashes();
|
2012-11-15 08:42:00 -08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-26 15:29:20 -08:00
|
|
|
|
2013-04-22 13:09:15 -07:00
|
|
|
debugCrashReport('Not online, postponing.');
|
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
Services.obs.addObserver(function observer(subject, topic, state) {
|
2012-11-26 15:29:20 -08:00
|
|
|
let network = subject.QueryInterface(Ci.nsINetworkInterface);
|
|
|
|
if (network.state == Ci.nsINetworkInterface.NETWORK_STATE_CONNECTED
|
|
|
|
&& network.type == Ci.nsINetworkInterface.NETWORK_TYPE_WIFI) {
|
2012-12-25 15:06:13 -08:00
|
|
|
shell.submitQueuedCrashes();
|
2012-12-06 13:27:49 -08:00
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
Services.obs.removeObserver(observer, topic);
|
|
|
|
}
|
2014-03-09 20:36:42 -07:00
|
|
|
}, "network-connection-state-changed", false);
|
2012-10-29 11:18:50 -07:00
|
|
|
},
|
|
|
|
|
2012-02-01 16:46:50 -08:00
|
|
|
get contentBrowser() {
|
|
|
|
delete this.contentBrowser;
|
2013-08-28 14:35:34 -07:00
|
|
|
return this.contentBrowser = document.getElementById('systemapp');
|
2011-12-06 15:52:26 -08:00
|
|
|
},
|
|
|
|
|
2011-12-08 17:16:03 -08:00
|
|
|
get homeURL() {
|
2011-12-06 15:52:26 -08:00
|
|
|
try {
|
2011-12-08 17:16:03 -08:00
|
|
|
let homeSrc = Services.env.get('B2G_HOMESCREEN');
|
2011-12-06 15:52:26 -08:00
|
|
|
if (homeSrc)
|
|
|
|
return homeSrc;
|
|
|
|
} catch (e) {}
|
|
|
|
|
2012-03-23 16:39:15 -07:00
|
|
|
return Services.prefs.getCharPref('browser.homescreenURL');
|
2011-12-06 15:52:26 -08:00
|
|
|
},
|
|
|
|
|
2012-06-20 12:25:23 -07:00
|
|
|
get manifestURL() {
|
|
|
|
return Services.prefs.getCharPref('browser.manifestURL');
|
2012-11-27 11:28:56 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
_started: false,
|
|
|
|
hasStarted: function shell_hasStarted() {
|
|
|
|
return this._started;
|
|
|
|
},
|
2012-06-20 12:25:23 -07:00
|
|
|
|
2012-04-11 10:05:35 -07:00
|
|
|
start: function shell_start() {
|
2012-11-27 11:28:56 -08:00
|
|
|
this._started = true;
|
|
|
|
|
2012-11-13 09:40:00 -08:00
|
|
|
// This forces the initialization of the cookie service before we hit the
|
|
|
|
// network.
|
|
|
|
// See bug 810209
|
|
|
|
let cookies = Cc["@mozilla.org/cookieService;1"];
|
|
|
|
|
2012-09-26 15:49:46 -07:00
|
|
|
try {
|
2012-10-05 12:16:13 -07:00
|
|
|
let cr = Cc["@mozilla.org/xre/app-info;1"]
|
|
|
|
.getService(Ci.nsICrashReporter);
|
2012-10-04 21:10:45 -07:00
|
|
|
// Dogfood id. We might want to remove it in the future.
|
|
|
|
// see bug 789466
|
|
|
|
try {
|
|
|
|
let dogfoodId = Services.prefs.getCharPref('prerelease.dogfood.id');
|
|
|
|
if (dogfoodId != "") {
|
|
|
|
cr.annotateCrashReport("Email", dogfoodId);
|
|
|
|
}
|
2012-09-26 15:49:46 -07:00
|
|
|
}
|
2012-10-04 21:10:45 -07:00
|
|
|
catch (e) { }
|
2012-09-26 15:49:46 -07:00
|
|
|
|
2012-10-04 10:29:43 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2012-10-04 21:10:45 -07:00
|
|
|
// Annotate crash report
|
|
|
|
let annotations = [ [ "Android_Hardware", "ro.hardware" ],
|
|
|
|
[ "Android_Device", "ro.product.device" ],
|
|
|
|
[ "Android_CPU_ABI2", "ro.product.cpu.abi2" ],
|
|
|
|
[ "Android_CPU_ABI", "ro.product.cpu.abi" ],
|
|
|
|
[ "Android_Manufacturer", "ro.product.manufacturer" ],
|
|
|
|
[ "Android_Brand", "ro.product.brand" ],
|
|
|
|
[ "Android_Model", "ro.product.model" ],
|
|
|
|
[ "Android_Board", "ro.product.board" ],
|
|
|
|
];
|
|
|
|
|
|
|
|
annotations.forEach(function (element) {
|
|
|
|
cr.annotateCrashReport(element[0], libcutils.property_get(element[1]));
|
|
|
|
});
|
|
|
|
|
|
|
|
let androidVersion = libcutils.property_get("ro.build.version.sdk") +
|
|
|
|
"(" + libcutils.property_get("ro.build.version.codename") + ")";
|
|
|
|
cr.annotateCrashReport("Android_Version", androidVersion);
|
2013-01-10 15:47:32 -08:00
|
|
|
|
|
|
|
SettingsListener.observe("deviceinfo.os", "", function(value) {
|
|
|
|
try {
|
|
|
|
let cr = Cc["@mozilla.org/xre/app-info;1"]
|
|
|
|
.getService(Ci.nsICrashReporter);
|
|
|
|
cr.annotateCrashReport("B2G_OS_Version", value);
|
2013-01-11 05:18:34 -08:00
|
|
|
} catch(e) { }
|
2013-01-10 15:47:32 -08:00
|
|
|
});
|
2012-10-04 10:29:43 -07:00
|
|
|
#endif
|
2013-01-10 15:47:32 -08:00
|
|
|
} catch(e) {
|
2013-04-22 13:09:15 -07:00
|
|
|
debugCrashReport('exception: ' + e);
|
2013-01-10 15:47:32 -08:00
|
|
|
}
|
2012-10-04 10:29:43 -07:00
|
|
|
|
2011-12-08 17:16:03 -08:00
|
|
|
let homeURL = this.homeURL;
|
|
|
|
if (!homeURL) {
|
2012-03-21 15:50:53 -07:00
|
|
|
let msg = 'Fatal error during startup: No homescreen found: try setting B2G_HOMESCREEN';
|
2012-04-11 10:05:35 -07:00
|
|
|
alert(msg);
|
|
|
|
return;
|
2011-12-08 17:16:03 -08:00
|
|
|
}
|
2012-06-20 12:25:23 -07:00
|
|
|
let manifestURL = this.manifestURL;
|
2013-08-28 14:35:34 -07:00
|
|
|
// <html:iframe id="systemapp"
|
2012-10-29 15:02:24 -07:00
|
|
|
// mozbrowser="true" allowfullscreen="true"
|
2013-08-28 14:35:34 -07:00
|
|
|
// style="overflow: hidden; height: 100%; width: 100%; border: none;"
|
2012-06-20 12:25:23 -07:00
|
|
|
// src="data:text/html;charset=utf-8,%3C!DOCTYPE html>%3Cbody style='background:black;'>"/>
|
2013-08-28 14:35:34 -07:00
|
|
|
let systemAppFrame =
|
2012-06-20 12:25:23 -07:00
|
|
|
document.createElementNS('http://www.w3.org/1999/xhtml', 'html:iframe');
|
2013-08-28 14:35:34 -07:00
|
|
|
systemAppFrame.setAttribute('id', 'systemapp');
|
|
|
|
systemAppFrame.setAttribute('mozbrowser', 'true');
|
|
|
|
systemAppFrame.setAttribute('mozapp', manifestURL);
|
|
|
|
systemAppFrame.setAttribute('allowfullscreen', 'true');
|
|
|
|
systemAppFrame.setAttribute('style', "overflow: hidden; height: 100%; width: 100%; border: none;");
|
|
|
|
systemAppFrame.setAttribute('src', "data:text/html;charset=utf-8,%3C!DOCTYPE html>%3Cbody style='background:black;");
|
|
|
|
let container = document.getElementById('container');
|
|
|
|
#ifdef MOZ_WIDGET_COCOA
|
2014-03-19 05:38:59 -07:00
|
|
|
// See shell.html
|
|
|
|
let hotfix = document.getElementById('placeholder');
|
|
|
|
if (hotfix) {
|
|
|
|
container.removeChild(hotfix);
|
|
|
|
}
|
2013-08-28 14:35:34 -07:00
|
|
|
#endif
|
|
|
|
container.appendChild(systemAppFrame);
|
|
|
|
|
|
|
|
systemAppFrame.contentWindow
|
|
|
|
.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
|
|
.getInterface(Ci.nsIWebNavigation)
|
|
|
|
.sessionHistory = Cc["@mozilla.org/browser/shistory;1"]
|
|
|
|
.createInstance(Ci.nsISHistory);
|
2012-06-26 08:18:10 -07:00
|
|
|
|
2014-01-22 11:06:46 -08:00
|
|
|
// On firefox mulet, shell.html is loaded in a tab
|
|
|
|
// and we have to listen on the chrome event handler
|
|
|
|
// to catch key events
|
|
|
|
let chromeEventHandler = window.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
|
|
.getInterface(Ci.nsIWebNavigation)
|
|
|
|
.QueryInterface(Ci.nsIDocShell)
|
|
|
|
.chromeEventHandler || window;
|
2012-07-17 23:20:19 -07:00
|
|
|
// Capture all key events so we can filter out hardware buttons
|
|
|
|
// And send them to Gaia via mozChromeEvents.
|
|
|
|
// Ideally, hardware buttons wouldn't generate key events at all, or
|
|
|
|
// if they did, they would use keycodes that conform to DOM 3 Events.
|
|
|
|
// See discussion in https://bugzilla.mozilla.org/show_bug.cgi?id=762362
|
2014-01-22 11:06:46 -08:00
|
|
|
chromeEventHandler.addEventListener('keydown', this, true);
|
|
|
|
chromeEventHandler.addEventListener('keypress', this, true);
|
|
|
|
chromeEventHandler.addEventListener('keyup', this, true);
|
|
|
|
|
2011-12-14 02:17:51 -08:00
|
|
|
window.addEventListener('MozApplicationManifest', this);
|
2012-02-17 13:37:53 -08:00
|
|
|
window.addEventListener('mozfullscreenchange', this);
|
2013-09-03 05:50:43 -07:00
|
|
|
window.addEventListener('MozAfterPaint', this);
|
2012-04-04 23:03:00 -07:00
|
|
|
window.addEventListener('sizemodechange', this);
|
2014-02-19 08:00:34 -08:00
|
|
|
window.addEventListener('unload', this);
|
2012-06-20 12:25:23 -07:00
|
|
|
this.contentBrowser.addEventListener('mozbrowserloadstart', this, true);
|
2011-12-06 15:52:26 -08:00
|
|
|
|
2014-04-07 06:59:48 -07:00
|
|
|
SystemAppProxy.registerFrame(this.contentBrowser);
|
|
|
|
|
2012-03-16 15:23:28 -07:00
|
|
|
CustomEventManager.init();
|
2012-03-16 15:23:28 -07:00
|
|
|
WebappsHelper.init();
|
2012-10-03 13:27:46 -07:00
|
|
|
UserAgentOverrides.init();
|
2013-01-09 08:24:56 -08:00
|
|
|
IndexedDBPromptHelper.init();
|
2012-11-14 19:26:30 -08:00
|
|
|
CaptivePortalLoginHelper.init();
|
2012-03-16 15:23:28 -07:00
|
|
|
|
2012-06-20 12:25:23 -07:00
|
|
|
this.contentBrowser.src = homeURL;
|
2012-09-29 10:57:18 -07:00
|
|
|
this.isHomeLoaded = false;
|
2012-08-15 09:00:02 -07:00
|
|
|
|
|
|
|
ppmm.addMessageListener("content-handler", this);
|
2012-10-01 11:28:01 -07:00
|
|
|
ppmm.addMessageListener("dial-handler", this);
|
|
|
|
ppmm.addMessageListener("sms-handler", this);
|
|
|
|
ppmm.addMessageListener("mail-handler", this);
|
2012-11-28 22:36:15 -08:00
|
|
|
ppmm.addMessageListener("app-notification-send", AlertsHelper);
|
2013-03-19 10:49:16 -07:00
|
|
|
ppmm.addMessageListener("file-picker", this);
|
2013-10-31 23:35:58 -07:00
|
|
|
ppmm.addMessageListener("getProfD", function(message) {
|
|
|
|
return Services.dirsvc.get("ProfD", Ci.nsIFile).path;
|
|
|
|
});
|
2011-12-06 15:52:26 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
stop: function shell_stop() {
|
2014-02-19 08:00:34 -08:00
|
|
|
window.removeEventListener('unload', this);
|
2012-07-17 23:20:19 -07:00
|
|
|
window.removeEventListener('keydown', this, true);
|
|
|
|
window.removeEventListener('keypress', this, true);
|
|
|
|
window.removeEventListener('keyup', this, true);
|
2011-12-14 02:17:51 -08:00
|
|
|
window.removeEventListener('MozApplicationManifest', this);
|
2012-03-05 10:09:05 -08:00
|
|
|
window.removeEventListener('mozfullscreenchange', this);
|
2012-04-04 23:03:00 -07:00
|
|
|
window.removeEventListener('sizemodechange', this);
|
2012-06-20 12:25:23 -07:00
|
|
|
this.contentBrowser.removeEventListener('mozbrowserloadstart', this, true);
|
2012-08-15 09:00:02 -07:00
|
|
|
ppmm.removeMessageListener("content-handler", this);
|
2012-08-29 19:46:35 -07:00
|
|
|
if (this.timer) {
|
|
|
|
this.timer.cancel();
|
|
|
|
this.timer = null;
|
|
|
|
}
|
2012-05-04 11:02:05 -07:00
|
|
|
|
2012-10-03 13:27:46 -07:00
|
|
|
UserAgentOverrides.uninit();
|
2013-01-09 08:24:56 -08:00
|
|
|
IndexedDBPromptHelper.uninit();
|
2011-12-06 15:52:26 -08:00
|
|
|
},
|
2012-06-30 18:42:10 -07:00
|
|
|
|
2012-07-17 23:20:19 -07:00
|
|
|
// If this key event actually represents a hardware button, filter it here
|
|
|
|
// and send a mozChromeEvent with detail.type set to xxx-button-press or
|
|
|
|
// xxx-button-release instead.
|
|
|
|
filterHardwareKeys: function shell_filterHardwareKeys(evt) {
|
|
|
|
var type;
|
|
|
|
switch (evt.keyCode) {
|
|
|
|
case evt.DOM_VK_HOME: // Home button
|
|
|
|
type = 'home-button';
|
|
|
|
break;
|
|
|
|
case evt.DOM_VK_SLEEP: // Sleep button
|
2012-07-27 16:19:30 -07:00
|
|
|
case evt.DOM_VK_END: // On desktop we don't have a sleep button
|
2012-07-17 23:20:19 -07:00
|
|
|
type = 'sleep-button';
|
|
|
|
break;
|
|
|
|
case evt.DOM_VK_PAGE_UP: // Volume up button
|
|
|
|
type = 'volume-up-button';
|
|
|
|
break;
|
|
|
|
case evt.DOM_VK_PAGE_DOWN: // Volume down button
|
|
|
|
type = 'volume-down-button';
|
|
|
|
break;
|
|
|
|
case evt.DOM_VK_ESCAPE: // Back button (should be disabled)
|
|
|
|
type = 'back-button';
|
|
|
|
break;
|
|
|
|
case evt.DOM_VK_CONTEXT_MENU: // Menu button
|
|
|
|
type = 'menu-button';
|
|
|
|
break;
|
2012-11-06 17:45:54 -08:00
|
|
|
case evt.DOM_VK_F1: // headset button
|
|
|
|
type = 'headset-button';
|
|
|
|
break;
|
2013-06-13 20:58:09 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
let mediaKeys = {
|
|
|
|
'MediaNextTrack': 'media-next-track-button',
|
|
|
|
'MediaPreviousTrack': 'media-previous-track-button',
|
|
|
|
'MediaPause': 'media-pause-button',
|
|
|
|
'MediaPlay': 'media-play-button',
|
|
|
|
'MediaPlayPause': 'media-play-pause-button',
|
|
|
|
'MediaStop': 'media-stop-button',
|
|
|
|
'MediaRewind': 'media-rewind-button',
|
|
|
|
'FastFwd': 'media-fast-forward-button'
|
|
|
|
};
|
|
|
|
|
|
|
|
let isMediaKey = false;
|
|
|
|
if (mediaKeys[evt.key]) {
|
|
|
|
isMediaKey = true;
|
|
|
|
type = mediaKeys[evt.key];
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!type) {
|
|
|
|
return;
|
2012-07-17 23:20:19 -07:00
|
|
|
}
|
2012-03-05 10:09:05 -08:00
|
|
|
|
2012-07-17 23:20:19 -07:00
|
|
|
// If we didn't return, then the key event represents a hardware key
|
|
|
|
// and we need to prevent it from propagating to Gaia
|
|
|
|
evt.stopImmediatePropagation();
|
|
|
|
evt.preventDefault(); // Prevent keypress events (when #501496 is fixed).
|
|
|
|
|
|
|
|
// If it is a key down or key up event, we send a chrome event to Gaia.
|
|
|
|
// If it is a keypress event we just ignore it.
|
|
|
|
switch (evt.type) {
|
|
|
|
case 'keydown':
|
|
|
|
type = type + '-press';
|
|
|
|
break;
|
|
|
|
case 'keyup':
|
|
|
|
type = type + '-release';
|
|
|
|
break;
|
|
|
|
case 'keypress':
|
|
|
|
return;
|
|
|
|
}
|
2012-09-11 18:17:01 -07:00
|
|
|
|
2012-11-06 17:45:54 -08:00
|
|
|
// Let applications receive the headset button key press/release event.
|
|
|
|
if (evt.keyCode == evt.DOM_VK_F1 && type !== this.lastHardwareButtonEventType) {
|
|
|
|
this.lastHardwareButtonEventType = type;
|
|
|
|
gSystemMessenger.broadcastMessage('headset-button', type);
|
2013-06-13 20:58:09 -07:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isMediaKey) {
|
|
|
|
this.lastHardwareButtonEventType = type;
|
|
|
|
gSystemMessenger.broadcastMessage('media-button', type);
|
2012-11-06 17:45:54 -08:00
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-17 23:20:19 -07:00
|
|
|
// On my device, the physical hardware buttons (sleep and volume)
|
|
|
|
// send multiple events (press press release release), but the
|
|
|
|
// soft home button just sends one. This hack is to manually
|
|
|
|
// "debounce" the keys. If the type of this event is the same as
|
|
|
|
// the type of the last one, then don't send it. We'll never send
|
|
|
|
// two presses or two releases in a row.
|
|
|
|
// FIXME: https://bugzilla.mozilla.org/show_bug.cgi?id=761067
|
|
|
|
if (type !== this.lastHardwareButtonEventType) {
|
|
|
|
this.lastHardwareButtonEventType = type;
|
|
|
|
this.sendChromeEvent({type: type});
|
|
|
|
}
|
2012-03-05 10:09:05 -08:00
|
|
|
},
|
2012-08-29 19:46:35 -07:00
|
|
|
|
2012-07-17 23:20:19 -07:00
|
|
|
lastHardwareButtonEventType: null, // property for the hack above
|
2013-05-24 01:17:54 -07:00
|
|
|
needBufferOpenAppReq: true,
|
|
|
|
bufferedOpenAppReqs: [],
|
2012-08-29 19:46:35 -07:00
|
|
|
timer: null,
|
2013-03-26 21:04:32 -07:00
|
|
|
visibleNormalAudioActive: false,
|
2012-03-05 10:09:05 -08:00
|
|
|
|
2011-12-06 15:52:26 -08:00
|
|
|
handleEvent: function shell_handleEvent(evt) {
|
2012-06-20 12:25:23 -07:00
|
|
|
let content = this.contentBrowser.contentWindow;
|
2011-12-06 15:52:26 -08:00
|
|
|
switch (evt.type) {
|
2012-03-05 10:09:05 -08:00
|
|
|
case 'keydown':
|
|
|
|
case 'keyup':
|
2011-12-06 15:52:26 -08:00
|
|
|
case 'keypress':
|
2012-07-17 23:20:19 -07:00
|
|
|
this.filterHardwareKeys(evt);
|
2012-02-07 21:22:34 -08:00
|
|
|
break;
|
2012-02-17 13:37:53 -08:00
|
|
|
case 'mozfullscreenchange':
|
|
|
|
// When the screen goes fullscreen make sure to set the focus to the
|
|
|
|
// main window so noboby can prevent the ESC key to get out fullscreen
|
|
|
|
// mode
|
|
|
|
if (document.mozFullScreen)
|
|
|
|
Services.fm.focusedWindow = window;
|
|
|
|
break;
|
2012-04-04 23:03:00 -07:00
|
|
|
case 'sizemodechange':
|
2013-03-26 21:04:32 -07:00
|
|
|
if (window.windowState == window.STATE_MINIMIZED && !this.visibleNormalAudioActive) {
|
2012-06-20 12:25:23 -07:00
|
|
|
this.contentBrowser.setVisible(false);
|
2012-04-04 23:03:00 -07:00
|
|
|
} else {
|
2012-06-20 12:25:23 -07:00
|
|
|
this.contentBrowser.setVisible(true);
|
2012-04-04 23:03:00 -07:00
|
|
|
}
|
|
|
|
break;
|
2012-06-20 12:25:23 -07:00
|
|
|
case 'mozbrowserloadstart':
|
2013-08-28 04:58:27 -07:00
|
|
|
if (content.document.location == 'about:blank') {
|
|
|
|
this.contentBrowser.addEventListener('mozbrowserlocationchange', this, true);
|
2012-06-25 23:42:45 -07:00
|
|
|
return;
|
2013-08-28 04:58:27 -07:00
|
|
|
}
|
2012-06-25 23:42:45 -07:00
|
|
|
|
2013-08-28 04:58:27 -07:00
|
|
|
this.notifyContentStart();
|
|
|
|
break;
|
|
|
|
case 'mozbrowserlocationchange':
|
|
|
|
if (content.document.location == 'about:blank') {
|
|
|
|
return;
|
|
|
|
}
|
2012-11-28 10:43:19 -08:00
|
|
|
|
2013-08-28 04:58:27 -07:00
|
|
|
this.notifyContentStart();
|
|
|
|
break;
|
2012-09-29 10:57:18 -07:00
|
|
|
|
2011-12-14 02:17:51 -08:00
|
|
|
case 'MozApplicationManifest':
|
|
|
|
try {
|
2011-12-14 12:02:48 -08:00
|
|
|
if (!Services.prefs.getBoolPref('browser.cache.offline.enable'))
|
|
|
|
return;
|
|
|
|
|
2011-12-14 02:17:51 -08:00
|
|
|
let contentWindow = evt.originalTarget.defaultView;
|
|
|
|
let documentElement = contentWindow.document.documentElement;
|
|
|
|
if (!documentElement)
|
|
|
|
return;
|
|
|
|
|
2012-02-01 16:46:49 -08:00
|
|
|
let manifest = documentElement.getAttribute('manifest');
|
2011-12-14 02:17:51 -08:00
|
|
|
if (!manifest)
|
|
|
|
return;
|
|
|
|
|
2012-07-20 18:06:35 -07:00
|
|
|
let principal = contentWindow.document.nodePrincipal;
|
|
|
|
if (Services.perms.testPermissionFromPrincipal(principal, 'offline-app') == Ci.nsIPermissionManager.UNKNOWN_ACTION) {
|
2011-12-14 12:02:48 -08:00
|
|
|
if (Services.prefs.getBoolPref('browser.offline-apps.notify')) {
|
|
|
|
// FIXME Bug 710729 - Add a UI for offline cache notifications
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-07-20 18:06:35 -07:00
|
|
|
Services.perms.addFromPrincipal(principal, 'offline-app',
|
|
|
|
Ci.nsIPermissionManager.ALLOW_ACTION);
|
2011-12-14 02:17:51 -08:00
|
|
|
|
2013-11-19 14:15:59 -08:00
|
|
|
let documentURI = Services.io.newURI(contentWindow.document.documentURI,
|
|
|
|
null,
|
|
|
|
null);
|
2011-12-14 12:02:48 -08:00
|
|
|
let manifestURI = Services.io.newURI(manifest, null, documentURI);
|
2011-12-14 02:17:51 -08:00
|
|
|
let updateService = Cc['@mozilla.org/offlinecacheupdate-service;1']
|
|
|
|
.getService(Ci.nsIOfflineCacheUpdateService);
|
|
|
|
updateService.scheduleUpdate(manifestURI, documentURI, window);
|
|
|
|
} catch (e) {
|
|
|
|
dump('Error while creating offline cache: ' + e + '\n');
|
|
|
|
}
|
|
|
|
break;
|
2013-09-03 05:50:43 -07:00
|
|
|
case 'MozAfterPaint':
|
|
|
|
window.removeEventListener('MozAfterPaint', this);
|
|
|
|
this.sendChromeEvent({
|
|
|
|
type: 'system-first-paint'
|
|
|
|
});
|
|
|
|
break;
|
2014-02-19 08:00:34 -08:00
|
|
|
case 'unload':
|
|
|
|
this.stop();
|
|
|
|
break;
|
2011-12-06 15:52:26 -08:00
|
|
|
}
|
|
|
|
},
|
2012-08-15 09:00:02 -07:00
|
|
|
|
2014-03-03 11:50:56 -08:00
|
|
|
// Send an event to a specific window, document or element.
|
|
|
|
sendEvent: function shell_sendEvent(target, type, details) {
|
|
|
|
let doc = target.document || target.ownerDocument || target;
|
|
|
|
let event = doc.createEvent('CustomEvent');
|
2011-12-06 15:52:26 -08:00
|
|
|
event.initCustomEvent(type, true, true, details ? details : {});
|
2014-03-03 11:50:56 -08:00
|
|
|
target.dispatchEvent(event);
|
2012-07-17 23:20:19 -07:00
|
|
|
},
|
2012-08-15 09:00:02 -07:00
|
|
|
|
2013-10-14 08:59:28 -07:00
|
|
|
sendCustomEvent: function shell_sendCustomEvent(type, details) {
|
2014-03-03 11:50:56 -08:00
|
|
|
let target = getContentWindow();
|
|
|
|
let payload = details ? Cu.cloneInto(details, target) : {};
|
|
|
|
this.sendEvent(target, type, payload);
|
2013-10-14 08:59:28 -07:00
|
|
|
},
|
|
|
|
|
2012-07-17 23:20:19 -07:00
|
|
|
sendChromeEvent: function shell_sendChromeEvent(details) {
|
2012-09-29 10:57:18 -07:00
|
|
|
if (!this.isHomeLoaded) {
|
|
|
|
if (!('pendingChromeEvents' in this)) {
|
|
|
|
this.pendingChromeEvents = [];
|
|
|
|
}
|
|
|
|
|
|
|
|
this.pendingChromeEvents.push(details);
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2012-08-19 12:00:19 -07:00
|
|
|
this.sendEvent(getContentWindow(), "mozChromeEvent",
|
2014-02-01 11:06:59 -08:00
|
|
|
Cu.cloneInto(details, getContentWindow()));
|
2012-08-15 09:00:02 -07:00
|
|
|
},
|
|
|
|
|
2013-05-24 01:17:54 -07:00
|
|
|
openAppForSystemMessage: function shell_openAppForSystemMessage(msg) {
|
2013-10-14 08:59:28 -07:00
|
|
|
let payload = {
|
2014-03-31 01:04:36 -07:00
|
|
|
url: msg.pageURL,
|
2014-03-31 01:04:29 -07:00
|
|
|
manifestURL: msg.manifestURL,
|
2012-08-29 19:46:35 -07:00
|
|
|
isActivity: (msg.type == 'activity'),
|
2013-07-25 05:36:15 -07:00
|
|
|
onlyShowApp: msg.onlyShowApp,
|
|
|
|
showApp: msg.showApp,
|
2013-02-14 12:41:30 -08:00
|
|
|
target: msg.target,
|
2013-07-25 10:18:24 -07:00
|
|
|
expectingSystemMessage: true,
|
|
|
|
extra: msg.extra
|
2013-10-14 08:59:28 -07:00
|
|
|
}
|
|
|
|
this.sendCustomEvent('open-app', payload);
|
2012-08-29 19:46:35 -07:00
|
|
|
},
|
|
|
|
|
2012-08-15 09:00:02 -07:00
|
|
|
receiveMessage: function shell_receiveMessage(message) {
|
2013-03-19 10:49:16 -07:00
|
|
|
var activities = { 'content-handler': { name: 'view', response: null },
|
|
|
|
'dial-handler': { name: 'dial', response: null },
|
|
|
|
'mail-handler': { name: 'new', response: null },
|
|
|
|
'sms-handler': { name: 'new', response: null },
|
|
|
|
'file-picker': { name: 'pick', response: 'file-picked' } };
|
2012-10-01 11:28:01 -07:00
|
|
|
|
2013-03-19 10:49:16 -07:00
|
|
|
if (!(message.name in activities))
|
2012-08-15 09:00:02 -07:00
|
|
|
return;
|
2012-10-01 11:28:01 -07:00
|
|
|
|
|
|
|
let data = message.data;
|
2013-03-19 10:49:16 -07:00
|
|
|
let activity = activities[message.name];
|
|
|
|
|
|
|
|
let a = new MozActivity({
|
|
|
|
name: activity.name,
|
2012-10-01 11:28:01 -07:00
|
|
|
data: data
|
2012-08-15 09:00:02 -07:00
|
|
|
});
|
2013-03-19 10:49:16 -07:00
|
|
|
|
|
|
|
if (activity.response) {
|
|
|
|
a.onsuccess = function() {
|
|
|
|
let sender = message.target.QueryInterface(Ci.nsIMessageSender);
|
|
|
|
sender.sendAsyncMessage(activity.response, { success: true,
|
|
|
|
result: a.result });
|
|
|
|
}
|
|
|
|
a.onerror = function() {
|
|
|
|
let sender = message.target.QueryInterface(Ci.nsIMessageSender);
|
|
|
|
sender.sendAsyncMessage(activity.response, { success: false });
|
|
|
|
}
|
|
|
|
}
|
2013-08-28 04:58:27 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
notifyContentStart: function shell_notifyContentStart() {
|
|
|
|
this.contentBrowser.removeEventListener('mozbrowserloadstart', this, true);
|
|
|
|
this.contentBrowser.removeEventListener('mozbrowserlocationchange', this, true);
|
|
|
|
|
|
|
|
let content = this.contentBrowser.contentWindow;
|
|
|
|
|
|
|
|
this.reportCrash(true);
|
|
|
|
|
|
|
|
this.sendEvent(window, 'ContentStart');
|
|
|
|
|
2013-12-13 03:25:26 -08:00
|
|
|
Services.obs.notifyObservers(null, 'content-start', null);
|
|
|
|
|
2013-10-29 01:12:45 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2013-07-31 12:34:19 -07:00
|
|
|
Cu.import('resource://gre/modules/OperatorApps.jsm');
|
|
|
|
#endif
|
|
|
|
|
2013-08-28 04:58:27 -07:00
|
|
|
content.addEventListener('load', function shell_homeLoaded() {
|
|
|
|
content.removeEventListener('load', shell_homeLoaded);
|
|
|
|
shell.isHomeLoaded = true;
|
|
|
|
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
libcutils.property_set('sys.boot_completed', '1');
|
|
|
|
#endif
|
|
|
|
|
|
|
|
Services.obs.notifyObservers(null, "browser-ui-startup-complete", "");
|
|
|
|
|
2014-04-07 06:59:48 -07:00
|
|
|
SystemAppProxy.setIsReady();
|
2013-08-28 04:58:27 -07:00
|
|
|
if ('pendingChromeEvents' in shell) {
|
|
|
|
shell.pendingChromeEvents.forEach((shell.sendChromeEvent).bind(shell));
|
|
|
|
}
|
|
|
|
delete shell.pendingChromeEvents;
|
|
|
|
});
|
2012-02-01 16:46:49 -08:00
|
|
|
}
|
2011-12-06 15:52:26 -08:00
|
|
|
};
|
|
|
|
|
2013-05-24 01:17:54 -07:00
|
|
|
// Listen for the request of opening app and relay them to Gaia.
|
|
|
|
Services.obs.addObserver(function onSystemMessageOpenApp(subject, topic, data) {
|
2012-07-20 08:41:30 -07:00
|
|
|
let msg = JSON.parse(data);
|
2013-05-24 01:17:54 -07:00
|
|
|
// Buffer non-activity request until content starts to load for 10 seconds.
|
|
|
|
// We'll revisit this later if new kind of requests don't need to be cached.
|
|
|
|
if (shell.needBufferOpenAppReq && msg.type !== 'activity') {
|
|
|
|
shell.bufferedOpenAppReqs.push(msg);
|
2012-08-29 19:46:35 -07:00
|
|
|
return;
|
|
|
|
}
|
2013-05-24 01:17:54 -07:00
|
|
|
shell.openAppForSystemMessage(msg);
|
2012-07-20 08:41:30 -07:00
|
|
|
}, 'system-messages-open-app', false);
|
2012-07-02 17:16:55 -07:00
|
|
|
|
2013-08-16 02:49:15 -07:00
|
|
|
Services.obs.addObserver(function onInterAppCommConnect(subject, topic, data) {
|
|
|
|
data = JSON.parse(data);
|
|
|
|
shell.sendChromeEvent({ type: "inter-app-comm-permission",
|
|
|
|
chromeEventID: data.callerID,
|
|
|
|
manifestURL: data.manifestURL,
|
|
|
|
keyword: data.keyword,
|
|
|
|
peers: data.appsToSelect });
|
|
|
|
}, 'inter-app-comm-select-app', false);
|
|
|
|
|
|
|
|
Services.obs.addObserver(function onFullscreenOriginChange(subject, topic, data) {
|
2012-08-19 12:00:19 -07:00
|
|
|
shell.sendChromeEvent({ type: "fullscreenoriginchange",
|
2013-08-16 02:49:15 -07:00
|
|
|
fullscreenorigin: data });
|
2012-07-30 19:09:31 -07:00
|
|
|
}, "fullscreen-origin-change", false);
|
|
|
|
|
2014-01-22 10:06:29 -08:00
|
|
|
DOMApplicationRegistry.registryStarted.then(function () {
|
2012-10-16 21:30:43 -07:00
|
|
|
shell.sendChromeEvent({ type: 'webapps-registry-start' });
|
2014-01-22 10:06:29 -08:00
|
|
|
});
|
|
|
|
DOMApplicationRegistry.registryReady.then(function () {
|
2012-09-11 18:17:01 -07:00
|
|
|
shell.sendChromeEvent({ type: 'webapps-registry-ready' });
|
2014-01-22 10:06:29 -08:00
|
|
|
});
|
2012-09-11 18:17:01 -07:00
|
|
|
|
2012-09-26 19:24:39 -07:00
|
|
|
Services.obs.addObserver(function onBluetoothVolumeChange(subject, topic, data) {
|
2012-10-25 03:50:46 -07:00
|
|
|
shell.sendChromeEvent({
|
2012-11-07 10:12:41 -08:00
|
|
|
type: "bluetooth-volumeset",
|
2012-10-25 03:50:46 -07:00
|
|
|
value: data
|
|
|
|
});
|
2012-09-26 19:24:39 -07:00
|
|
|
}, 'bluetooth-volume-change', false);
|
|
|
|
|
2014-01-14 21:32:19 -08:00
|
|
|
Services.obs.addObserver(function(subject, topic, data) {
|
|
|
|
shell.sendCustomEvent('mozmemorypressure');
|
|
|
|
}, 'memory-pressure', false);
|
|
|
|
|
2012-04-11 10:05:35 -07:00
|
|
|
var CustomEventManager = {
|
2012-03-16 15:23:28 -07:00
|
|
|
init: function custevt_init() {
|
|
|
|
window.addEventListener("ContentStart", (function(evt) {
|
2012-06-20 12:25:23 -07:00
|
|
|
let content = shell.contentBrowser.contentWindow;
|
2012-03-16 15:23:28 -07:00
|
|
|
content.addEventListener("mozContentEvent", this, false, true);
|
2012-08-29 19:46:35 -07:00
|
|
|
|
|
|
|
// After content starts to load for 10 seconds, send and
|
2013-05-24 01:17:54 -07:00
|
|
|
// clean up the buffered open-app requests if there is any.
|
|
|
|
//
|
|
|
|
// TODO: Bug 793420 - Remove the waiting timer for the 'open-app'
|
|
|
|
// mozChromeEvents requested by System Message
|
2012-08-29 19:46:35 -07:00
|
|
|
shell.timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
|
|
|
|
shell.timer.initWithCallback(function timerCallback() {
|
2013-05-24 01:17:54 -07:00
|
|
|
shell.bufferedOpenAppReqs.forEach(function bufferOpenAppReq(msg) {
|
|
|
|
shell.openAppForSystemMessage(msg);
|
2012-08-29 19:46:35 -07:00
|
|
|
});
|
2013-05-24 01:17:54 -07:00
|
|
|
shell.bufferedOpenAppReqs.length = 0;
|
|
|
|
shell.needBufferOpenAppReq = false;
|
2012-08-29 19:46:35 -07:00
|
|
|
shell.timer = null;
|
|
|
|
}, 10000, Ci.nsITimer.TYPE_ONE_SHOT);
|
2012-03-16 15:23:28 -07:00
|
|
|
}).bind(this), false);
|
|
|
|
},
|
|
|
|
|
|
|
|
handleEvent: function custevt_handleEvent(evt) {
|
|
|
|
let detail = evt.detail;
|
2012-07-02 17:16:55 -07:00
|
|
|
dump('XXX FIXME : Got a mozContentEvent: ' + detail.type + "\n");
|
2012-03-16 15:23:28 -07:00
|
|
|
|
|
|
|
switch(detail.type) {
|
2012-07-12 08:45:38 -07:00
|
|
|
case 'desktop-notification-show':
|
2012-06-06 05:19:33 -07:00
|
|
|
case 'desktop-notification-click':
|
|
|
|
case 'desktop-notification-close':
|
2012-03-16 15:23:28 -07:00
|
|
|
AlertsHelper.handleEvent(detail);
|
|
|
|
break;
|
2012-06-06 05:19:33 -07:00
|
|
|
case 'webapps-install-granted':
|
|
|
|
case 'webapps-install-denied':
|
2012-03-20 14:25:37 -07:00
|
|
|
WebappsHelper.handleEvent(detail);
|
|
|
|
break;
|
2012-06-06 05:19:33 -07:00
|
|
|
case 'select-choicechange':
|
|
|
|
FormsHelper.handleEvent(detail);
|
|
|
|
break;
|
2012-09-05 13:00:06 -07:00
|
|
|
case 'system-message-listener-ready':
|
|
|
|
Services.obs.notifyObservers(null, 'system-message-listener-ready', null);
|
|
|
|
break;
|
2013-01-07 23:08:00 -08:00
|
|
|
case 'remote-debugger-prompt':
|
|
|
|
RemoteDebugger.handleEvent(detail);
|
|
|
|
break;
|
2012-11-14 19:26:30 -08:00
|
|
|
case 'captive-portal-login-cancel':
|
|
|
|
CaptivePortalLoginHelper.handleEvent(detail);
|
|
|
|
break;
|
2013-08-16 02:49:15 -07:00
|
|
|
case 'inter-app-comm-permission':
|
|
|
|
Services.obs.notifyObservers(null, 'inter-app-comm-select-app-result',
|
|
|
|
JSON.stringify({ callerID: detail.chromeEventID,
|
|
|
|
keyword: detail.keyword,
|
|
|
|
manifestURL: detail.manifestURL,
|
|
|
|
selectedApps: detail.peers }));
|
|
|
|
break;
|
2013-09-12 10:02:36 -07:00
|
|
|
case 'inputmethod-update-layouts':
|
|
|
|
KeyboardHelper.handleEvent(detail);
|
|
|
|
break;
|
2012-03-16 15:23:28 -07:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-04-11 10:05:35 -07:00
|
|
|
var AlertsHelper = {
|
2012-03-16 15:23:28 -07:00
|
|
|
_listeners: {},
|
|
|
|
_count: 0,
|
|
|
|
|
|
|
|
handleEvent: function alert_handleEvent(detail) {
|
|
|
|
if (!detail || !detail.id)
|
|
|
|
return;
|
|
|
|
|
2012-12-03 06:42:36 -08:00
|
|
|
let uid = detail.id;
|
|
|
|
let listener = this._listeners[uid];
|
2012-11-28 22:36:15 -08:00
|
|
|
if (!listener)
|
|
|
|
return;
|
|
|
|
|
2012-07-12 08:45:38 -07:00
|
|
|
let topic;
|
|
|
|
if (detail.type == "desktop-notification-click") {
|
|
|
|
topic = "alertclickcallback";
|
|
|
|
} else if (detail.type == "desktop-notification-show") {
|
|
|
|
topic = "alertshow";
|
|
|
|
} else {
|
|
|
|
/* desktop-notification-close */
|
|
|
|
topic = "alertfinished";
|
|
|
|
}
|
|
|
|
|
2013-09-10 07:49:42 -07:00
|
|
|
if (uid.startsWith("alert")) {
|
|
|
|
try {
|
|
|
|
listener.observer.observe(null, topic, listener.cookie);
|
|
|
|
} catch (e) { }
|
|
|
|
} else {
|
2012-12-06 00:22:53 -08:00
|
|
|
try {
|
|
|
|
listener.mm.sendAsyncMessage("app-notification-return", {
|
|
|
|
uid: uid,
|
|
|
|
topic: topic,
|
|
|
|
target: listener.target
|
|
|
|
});
|
2013-09-10 07:49:42 -07:00
|
|
|
} catch (e) {
|
2013-02-07 04:54:32 -08:00
|
|
|
// we get an exception if the app is not launched yet
|
2012-12-06 00:22:53 -08:00
|
|
|
gSystemMessenger.sendMessage("notification", {
|
2013-02-07 04:54:32 -08:00
|
|
|
clicked: (detail.type === "desktop-notification-click"),
|
|
|
|
title: listener.title,
|
|
|
|
body: listener.text,
|
2014-01-09 07:55:44 -08:00
|
|
|
imageURL: listener.imageURL,
|
|
|
|
lang: listener.lang,
|
|
|
|
dir: listener.dir,
|
|
|
|
id: listener.id,
|
|
|
|
tag: listener.tag
|
2013-02-07 04:54:32 -08:00
|
|
|
},
|
|
|
|
Services.io.newURI(listener.target, null, null),
|
|
|
|
Services.io.newURI(listener.manifestURL, null, null)
|
|
|
|
);
|
2012-12-06 00:22:53 -08:00
|
|
|
}
|
2012-11-28 22:36:15 -08:00
|
|
|
}
|
2012-03-16 15:23:28 -07:00
|
|
|
|
|
|
|
// we're done with this notification
|
2012-12-03 06:42:36 -08:00
|
|
|
if (topic === "alertfinished") {
|
|
|
|
delete this._listeners[uid];
|
|
|
|
}
|
2012-03-16 15:23:28 -07:00
|
|
|
},
|
|
|
|
|
2012-07-12 08:45:38 -07:00
|
|
|
registerListener: function alert_registerListener(alertId, cookie, alertListener) {
|
|
|
|
this._listeners[alertId] = { observer: alertListener, cookie: cookie };
|
2012-03-16 15:23:28 -07:00
|
|
|
},
|
|
|
|
|
2012-12-03 06:42:36 -08:00
|
|
|
registerAppListener: function alert_registerAppListener(uid, listener) {
|
|
|
|
this._listeners[uid] = listener;
|
|
|
|
|
|
|
|
let app = DOMApplicationRegistry.getAppByManifestURL(listener.manifestURL);
|
2014-01-16 16:26:24 -08:00
|
|
|
DOMApplicationRegistry.getManifestFor(app.manifestURL).then((manifest) => {
|
2012-12-03 06:42:36 -08:00
|
|
|
let helper = new ManifestHelper(manifest, app.origin);
|
|
|
|
let getNotificationURLFor = function(messages) {
|
|
|
|
if (!messages)
|
|
|
|
return null;
|
|
|
|
|
|
|
|
for (let i = 0; i < messages.length; i++) {
|
|
|
|
let message = messages[i];
|
|
|
|
if (message === "notification") {
|
|
|
|
return helper.fullLaunchPath();
|
2012-12-28 10:43:49 -08:00
|
|
|
} else if (typeof message == "object" && "notification" in message) {
|
2012-12-03 06:42:36 -08:00
|
|
|
return helper.resolveFromOrigin(message["notification"]);
|
|
|
|
}
|
|
|
|
}
|
2013-08-27 15:11:16 -07:00
|
|
|
|
|
|
|
// No message found...
|
|
|
|
return null;
|
2012-12-03 06:42:36 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
listener.target = getNotificationURLFor(manifest.messages);
|
|
|
|
|
|
|
|
// Bug 816944 - Support notification messages for entry_points.
|
|
|
|
});
|
2012-11-28 22:36:15 -08:00
|
|
|
},
|
|
|
|
|
2014-03-31 01:04:29 -07:00
|
|
|
showNotification: function alert_showNotification(imageURL,
|
2012-11-28 22:36:15 -08:00
|
|
|
title,
|
|
|
|
text,
|
|
|
|
textClickable,
|
|
|
|
cookie,
|
2012-12-03 06:42:36 -08:00
|
|
|
uid,
|
2012-07-12 08:45:38 -07:00
|
|
|
bidi,
|
|
|
|
lang,
|
2014-03-31 01:04:29 -07:00
|
|
|
manifestURL) {
|
2012-11-28 22:36:15 -08:00
|
|
|
function send(appName, appIcon) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: "desktop-notification",
|
2012-12-03 06:42:36 -08:00
|
|
|
id: uid,
|
2014-03-31 01:04:29 -07:00
|
|
|
icon: imageURL,
|
2012-11-28 22:36:15 -08:00
|
|
|
title: title,
|
|
|
|
text: text,
|
2012-07-12 08:45:38 -07:00
|
|
|
bidi: bidi,
|
|
|
|
lang: lang,
|
2012-11-28 22:36:15 -08:00
|
|
|
appName: appName,
|
2013-01-16 10:22:28 -08:00
|
|
|
appIcon: appIcon,
|
2014-03-31 01:04:29 -07:00
|
|
|
manifestURL: manifestURL
|
2012-11-28 22:36:15 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2014-03-31 01:04:29 -07:00
|
|
|
if (!manifestURL || !manifestURL.length) {
|
2012-11-28 22:36:15 -08:00
|
|
|
send(null, null);
|
2013-08-09 10:02:33 -07:00
|
|
|
return;
|
2012-11-28 22:36:15 -08:00
|
|
|
}
|
2012-12-03 06:42:36 -08:00
|
|
|
|
|
|
|
// If we have a manifest URL, get the icon and title from the manifest
|
|
|
|
// to prevent spoofing.
|
2014-03-31 01:04:29 -07:00
|
|
|
let app = DOMApplicationRegistry.getAppByManifestURL(manifestURL);
|
|
|
|
DOMApplicationRegistry.getManifestFor(manifestURL).then((aManifest) => {
|
2012-12-03 06:42:36 -08:00
|
|
|
let helper = new ManifestHelper(aManifest, app.origin);
|
|
|
|
send(helper.name, helper.iconURLForSize(128));
|
|
|
|
});
|
2012-11-28 22:36:15 -08:00
|
|
|
},
|
|
|
|
|
2014-03-31 01:04:29 -07:00
|
|
|
showAlertNotification: function alert_showAlertNotification(imageURL,
|
2012-07-17 23:20:19 -07:00
|
|
|
title,
|
|
|
|
text,
|
|
|
|
textClickable,
|
|
|
|
cookie,
|
|
|
|
alertListener,
|
2012-07-12 08:45:38 -07:00
|
|
|
name,
|
|
|
|
bidi,
|
|
|
|
lang) {
|
|
|
|
let currentListener = this._listeners[name];
|
|
|
|
if (currentListener) {
|
|
|
|
currentListener.observer.observe(null, "alertfinished", currentListener.cookie);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.registerListener(name, cookie, alertListener);
|
2014-03-31 01:04:29 -07:00
|
|
|
this.showNotification(imageURL, title, text, textClickable, cookie,
|
2012-07-12 08:45:38 -07:00
|
|
|
name, bidi, lang, null);
|
|
|
|
},
|
|
|
|
|
|
|
|
closeAlert: function alert_closeAlert(name) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: "desktop-notification-close",
|
|
|
|
id: name
|
|
|
|
});
|
2012-11-28 22:36:15 -08:00
|
|
|
},
|
|
|
|
|
2012-12-20 11:06:42 -08:00
|
|
|
receiveMessage: function alert_receiveMessage(aMessage) {
|
2013-01-14 02:08:55 -08:00
|
|
|
if (!aMessage.target.assertAppHasPermission("desktop-notification")) {
|
2012-12-20 11:06:42 -08:00
|
|
|
Cu.reportError("Desktop-notification message " + aMessage.name +
|
|
|
|
" from a content process with no desktop-notification privileges.");
|
2013-08-27 15:11:16 -07:00
|
|
|
return;
|
2012-12-20 11:06:42 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
let data = aMessage.data;
|
2013-10-11 11:12:13 -07:00
|
|
|
let details = data.details;
|
2012-12-03 06:42:36 -08:00
|
|
|
let listener = {
|
2012-12-20 11:06:42 -08:00
|
|
|
mm: aMessage.target,
|
2012-12-03 06:42:36 -08:00
|
|
|
title: data.title,
|
|
|
|
text: data.text,
|
2013-10-11 11:12:13 -07:00
|
|
|
manifestURL: details.manifestURL,
|
2014-01-09 07:55:44 -08:00
|
|
|
imageURL: data.imageURL,
|
|
|
|
lang: details.lang || undefined,
|
|
|
|
id: details.id || undefined,
|
|
|
|
dir: details.dir || undefined,
|
|
|
|
tag: details.tag || undefined
|
2013-10-11 11:12:13 -07:00
|
|
|
};
|
2012-12-03 06:42:36 -08:00
|
|
|
this.registerAppListener(data.uid, listener);
|
2012-11-28 22:36:15 -08:00
|
|
|
|
|
|
|
this.showNotification(data.imageURL, data.title, data.text,
|
2013-10-11 11:12:13 -07:00
|
|
|
details.textClickable, null,
|
|
|
|
data.uid, details.dir,
|
|
|
|
details.lang, details.manifestURL);
|
2012-11-28 22:36:15 -08:00
|
|
|
},
|
2012-03-16 15:23:28 -07:00
|
|
|
}
|
2012-03-16 15:23:28 -07:00
|
|
|
|
2012-04-11 10:05:35 -07:00
|
|
|
var WebappsHelper = {
|
2012-03-20 14:25:37 -07:00
|
|
|
_installers: {},
|
|
|
|
_count: 0,
|
|
|
|
|
2012-03-16 15:23:28 -07:00
|
|
|
init: function webapps_init() {
|
|
|
|
Services.obs.addObserver(this, "webapps-launch", false);
|
2012-03-20 14:25:37 -07:00
|
|
|
Services.obs.addObserver(this, "webapps-ask-install", false);
|
2013-05-06 06:51:53 -07:00
|
|
|
Services.obs.addObserver(this, "webapps-close", false);
|
2012-03-20 14:25:37 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
registerInstaller: function webapps_registerInstaller(data) {
|
|
|
|
let id = "installer" + this._count++;
|
|
|
|
this._installers[id] = data;
|
|
|
|
return id;
|
|
|
|
},
|
|
|
|
|
|
|
|
handleEvent: function webapps_handleEvent(detail) {
|
|
|
|
if (!detail || !detail.id)
|
|
|
|
return;
|
|
|
|
|
|
|
|
let installer = this._installers[detail.id];
|
2012-11-29 17:18:28 -08:00
|
|
|
delete this._installers[detail.id];
|
2012-03-20 14:25:37 -07:00
|
|
|
switch (detail.type) {
|
|
|
|
case "webapps-install-granted":
|
|
|
|
DOMApplicationRegistry.confirmInstall(installer);
|
|
|
|
break;
|
|
|
|
case "webapps-install-denied":
|
|
|
|
DOMApplicationRegistry.denyInstall(installer);
|
|
|
|
break;
|
|
|
|
}
|
2012-03-16 15:23:28 -07:00
|
|
|
},
|
|
|
|
|
|
|
|
observe: function webapps_observe(subject, topic, data) {
|
|
|
|
let json = JSON.parse(data);
|
2012-09-18 10:34:55 -07:00
|
|
|
json.mm = subject;
|
|
|
|
|
2012-03-20 14:25:37 -07:00
|
|
|
switch(topic) {
|
|
|
|
case "webapps-launch":
|
2014-01-16 16:26:24 -08:00
|
|
|
DOMApplicationRegistry.getManifestFor(json.manifestURL).then((aManifest) => {
|
2012-03-20 14:25:37 -07:00
|
|
|
if (!aManifest)
|
|
|
|
return;
|
2012-03-16 15:23:28 -07:00
|
|
|
|
2012-10-02 22:38:03 -07:00
|
|
|
let manifest = new ManifestHelper(aManifest, json.origin);
|
2013-10-14 08:59:28 -07:00
|
|
|
let payload = {
|
|
|
|
__exposedProps__: {
|
|
|
|
timestamp: "r",
|
|
|
|
url: "r",
|
|
|
|
manifestURL: "r"
|
|
|
|
},
|
|
|
|
timestamp: json.timestamp,
|
|
|
|
url: manifest.fullLaunchPath(json.startPoint),
|
|
|
|
manifestURL: json.manifestURL
|
|
|
|
}
|
|
|
|
shell.sendEvent(getContentWindow(), "webapps-launch", payload);
|
2012-03-20 14:25:37 -07:00
|
|
|
});
|
|
|
|
break;
|
|
|
|
case "webapps-ask-install":
|
|
|
|
let id = this.registerInstaller(json);
|
2012-07-17 23:20:19 -07:00
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: "webapps-ask-install",
|
|
|
|
id: id,
|
|
|
|
app: json.app
|
|
|
|
});
|
2012-03-20 14:25:37 -07:00
|
|
|
break;
|
2013-05-06 06:51:53 -07:00
|
|
|
case "webapps-close":
|
2013-10-31 19:52:39 -07:00
|
|
|
shell.sendEvent(getContentWindow(), "webapps-close",
|
2013-10-14 08:59:28 -07:00
|
|
|
{
|
|
|
|
__exposedProps__: { "manifestURL": "r" },
|
|
|
|
"manifestURL": json.manifestURL
|
|
|
|
});
|
2013-05-06 06:51:53 -07:00
|
|
|
break;
|
2012-03-20 14:25:37 -07:00
|
|
|
}
|
2012-03-16 15:23:28 -07:00
|
|
|
}
|
|
|
|
}
|
2012-03-16 09:50:45 -07:00
|
|
|
|
2013-01-09 08:24:56 -08:00
|
|
|
let IndexedDBPromptHelper = {
|
|
|
|
_quotaPrompt: "indexedDB-quota-prompt",
|
|
|
|
_quotaResponse: "indexedDB-quota-response",
|
|
|
|
|
|
|
|
init:
|
|
|
|
function IndexedDBPromptHelper_init() {
|
|
|
|
Services.obs.addObserver(this, this._quotaPrompt, false);
|
|
|
|
},
|
|
|
|
|
|
|
|
uninit:
|
|
|
|
function IndexedDBPromptHelper_uninit() {
|
2013-04-06 11:46:34 -07:00
|
|
|
Services.obs.removeObserver(this, this._quotaPrompt);
|
2013-01-09 08:24:56 -08:00
|
|
|
},
|
|
|
|
|
|
|
|
observe:
|
|
|
|
function IndexedDBPromptHelper_observe(subject, topic, data) {
|
|
|
|
if (topic != this._quotaPrompt) {
|
|
|
|
throw new Error("Unexpected topic!");
|
|
|
|
}
|
|
|
|
|
|
|
|
let observer = subject.QueryInterface(Ci.nsIInterfaceRequestor)
|
|
|
|
.getInterface(Ci.nsIObserver);
|
|
|
|
let responseTopic = this._quotaResponse;
|
|
|
|
|
|
|
|
setTimeout(function() {
|
|
|
|
observer.observe(null, responseTopic,
|
|
|
|
Ci.nsIPermissionManager.DENY_ACTION);
|
|
|
|
}, 0);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
let RemoteDebugger = {
|
|
|
|
_promptDone: false,
|
|
|
|
_promptAnswer: false,
|
2014-01-08 17:18:03 -08:00
|
|
|
_running: false,
|
2012-03-16 09:50:45 -07:00
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
prompt: function debugger_prompt() {
|
|
|
|
this._promptDone = false;
|
2012-03-16 09:50:45 -07:00
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
shell.sendChromeEvent({
|
|
|
|
"type": "remote-debugger-prompt"
|
|
|
|
});
|
|
|
|
|
|
|
|
while(!this._promptDone) {
|
|
|
|
Services.tm.currentThread.processNextEvent(true);
|
|
|
|
}
|
|
|
|
|
|
|
|
return this._promptAnswer;
|
|
|
|
},
|
|
|
|
|
|
|
|
handleEvent: function debugger_handleEvent(detail) {
|
|
|
|
this._promptAnswer = detail.value;
|
|
|
|
this._promptDone = true;
|
|
|
|
},
|
|
|
|
|
2014-01-08 17:18:03 -08:00
|
|
|
get isDebugging() {
|
|
|
|
if (!this._running) {
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
return DebuggerServer._connections &&
|
|
|
|
Object.keys(DebuggerServer._connections).length > 0;
|
|
|
|
},
|
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
// Start the debugger server.
|
|
|
|
start: function debugger_start() {
|
2014-01-08 17:18:03 -08:00
|
|
|
if (this._running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
if (!DebuggerServer.initialized) {
|
|
|
|
// Ask for remote connections.
|
|
|
|
DebuggerServer.init(this.prompt.bind(this));
|
2014-01-24 02:11:00 -08:00
|
|
|
|
2014-03-03 11:11:00 -08:00
|
|
|
// /!\ Be careful when adding a new actor, especially global actors.
|
|
|
|
// Any new global actor will be exposed and returned by the root actor.
|
|
|
|
|
2014-01-24 02:11:00 -08:00
|
|
|
// Add Firefox-specific actors, but prevent tab actors to be loaded in
|
|
|
|
// the parent process, unless we enable certified apps debugging.
|
|
|
|
let restrictPrivileges = Services.prefs.getBoolPref("devtools.debugger.forbid-certified-apps");
|
|
|
|
DebuggerServer.addBrowserActors("navigator:browser", restrictPrivileges);
|
2014-03-03 11:11:00 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Construct a root actor appropriate for use in a server running in B2G.
|
|
|
|
* The returned root actor respects the factories registered with
|
|
|
|
* DebuggerServer.addGlobalActor only if certified apps debugging is on,
|
|
|
|
* otherwise we used an explicit limited list of global actors
|
|
|
|
*
|
|
|
|
* * @param connection DebuggerServerConnection
|
|
|
|
* The conection to the client.
|
|
|
|
*/
|
|
|
|
DebuggerServer.createRootActor = function createRootActor(connection)
|
|
|
|
{
|
2014-03-20 03:11:36 -07:00
|
|
|
let { Promise: promise } = Cu.import("resource://gre/modules/Promise.jsm", {});
|
2014-03-03 11:11:00 -08:00
|
|
|
let parameters = {
|
|
|
|
// We do not expose browser tab actors yet,
|
|
|
|
// but we still have to define tabList.getList(),
|
|
|
|
// otherwise, client won't be able to fetch global actors
|
|
|
|
// from listTabs request!
|
|
|
|
tabList: {
|
|
|
|
getList: function() {
|
|
|
|
return promise.resolve([]);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
// Use an explicit global actor list to prevent exposing
|
|
|
|
// unexpected actors
|
|
|
|
globalActorFactories: restrictPrivileges ? {
|
|
|
|
webappsActor: DebuggerServer.globalActorFactories.webappsActor,
|
2014-03-02 17:41:36 -08:00
|
|
|
deviceActor: DebuggerServer.globalActorFactories.deviceActor,
|
2014-03-03 11:11:00 -08:00
|
|
|
} : DebuggerServer.globalActorFactories
|
|
|
|
};
|
|
|
|
let root = new DebuggerServer.RootActor(connection, parameters);
|
|
|
|
root.applicationType = "operating-system";
|
|
|
|
return root;
|
|
|
|
};
|
2013-09-07 16:28:51 -07:00
|
|
|
|
2013-09-17 11:29:38 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
2014-04-17 10:27:03 -07:00
|
|
|
DebuggerServer.on("connectionchange", function() {
|
2013-09-07 16:28:51 -07:00
|
|
|
AdbController.updateState();
|
2014-04-17 10:27:03 -07:00
|
|
|
});
|
2013-09-17 11:29:38 -07:00
|
|
|
#endif
|
2013-01-07 23:08:00 -08:00
|
|
|
}
|
|
|
|
|
2013-09-10 09:52:33 -07:00
|
|
|
let path = Services.prefs.getCharPref("devtools.debugger.unix-domain-socket") ||
|
|
|
|
"/data/local/debugger-socket";
|
2013-01-07 23:08:00 -08:00
|
|
|
try {
|
2013-09-10 09:52:33 -07:00
|
|
|
DebuggerServer.openListener(path);
|
2014-01-24 02:11:00 -08:00
|
|
|
// Temporary event, until bug 942756 lands and offers a way to know
|
|
|
|
// when the server is up and running.
|
2014-01-13 05:44:47 -08:00
|
|
|
Services.obs.notifyObservers(null, 'debugger-server-started', null);
|
2014-01-08 17:18:03 -08:00
|
|
|
this._running = true;
|
2013-01-07 23:08:00 -08:00
|
|
|
} catch (e) {
|
|
|
|
dump('Unable to start debugger server: ' + e + '\n');
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
stop: function debugger_stop() {
|
2014-01-08 17:18:03 -08:00
|
|
|
if (!this._running) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
if (!DebuggerServer.initialized) {
|
2014-01-08 17:18:03 -08:00
|
|
|
// Can this really happen if we are running?
|
|
|
|
this._running = false;
|
2013-01-07 23:08:00 -08:00
|
|
|
return;
|
|
|
|
}
|
2012-11-28 02:48:10 -08:00
|
|
|
|
2013-01-07 23:08:00 -08:00
|
|
|
try {
|
|
|
|
DebuggerServer.closeListener();
|
|
|
|
} catch (e) {
|
|
|
|
dump('Unable to stop debugger server: ' + e + '\n');
|
|
|
|
}
|
2014-01-08 17:18:03 -08:00
|
|
|
this._running = false;
|
2012-11-28 02:48:10 -08:00
|
|
|
}
|
|
|
|
}
|
2012-05-04 11:02:05 -07:00
|
|
|
|
2013-09-12 10:02:36 -07:00
|
|
|
let KeyboardHelper = {
|
2013-09-17 08:11:57 -07:00
|
|
|
handleEvent: function keyboard_handleEvent(detail) {
|
|
|
|
Keyboard.setLayouts(detail.layouts);
|
2013-09-12 10:02:36 -07:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2012-06-27 07:47:30 -07:00
|
|
|
// This is the backend for Gaia's screenshot feature. Gaia requests a
|
|
|
|
// screenshot by sending a mozContentEvent with detail.type set to
|
|
|
|
// 'take-screenshot'. Then we take a screenshot and send a
|
|
|
|
// mozChromeEvent with detail.type set to 'take-screenshot-success'
|
|
|
|
// and detail.file set to the an image/png blob
|
2012-06-15 00:20:44 -07:00
|
|
|
window.addEventListener('ContentStart', function ss_onContentStart() {
|
2012-06-20 12:25:23 -07:00
|
|
|
let content = shell.contentBrowser.contentWindow;
|
2012-06-15 00:20:44 -07:00
|
|
|
content.addEventListener('mozContentEvent', function ss_onMozContentEvent(e) {
|
2012-06-27 07:47:30 -07:00
|
|
|
if (e.detail.type !== 'take-screenshot')
|
2012-06-15 00:20:44 -07:00
|
|
|
return;
|
|
|
|
|
|
|
|
try {
|
|
|
|
var canvas = document.createElementNS('http://www.w3.org/1999/xhtml',
|
|
|
|
'canvas');
|
|
|
|
var width = window.innerWidth;
|
|
|
|
var height = window.innerHeight;
|
2014-01-12 18:24:36 -08:00
|
|
|
var scale = window.devicePixelRatio;
|
|
|
|
canvas.setAttribute('width', width * scale);
|
|
|
|
canvas.setAttribute('height', height * scale);
|
2012-06-15 00:20:44 -07:00
|
|
|
|
|
|
|
var context = canvas.getContext('2d');
|
|
|
|
var flags =
|
|
|
|
context.DRAWWINDOW_DRAW_CARET |
|
|
|
|
context.DRAWWINDOW_DRAW_VIEW |
|
|
|
|
context.DRAWWINDOW_USE_WIDGET_LAYERS;
|
2014-01-12 18:24:36 -08:00
|
|
|
context.scale(scale, scale);
|
2012-06-15 00:20:44 -07:00
|
|
|
context.drawWindow(window, 0, 0, width, height,
|
|
|
|
'rgb(255,255,255)', flags);
|
|
|
|
|
2012-08-21 15:12:37 -07:00
|
|
|
// I can't use sendChromeEvent() here because it doesn't wrap
|
|
|
|
// the blob in the detail object correctly. So I use __exposedProps__
|
|
|
|
// instead to safely send the chrome detail object to content.
|
|
|
|
shell.sendEvent(getContentWindow(), 'mozChromeEvent', {
|
|
|
|
__exposedProps__: { type: 'r', file: 'r' },
|
2012-07-17 23:20:19 -07:00
|
|
|
type: 'take-screenshot-success',
|
|
|
|
file: canvas.mozGetAsFile('screenshot', 'image/png')
|
2012-06-27 07:47:30 -07:00
|
|
|
});
|
|
|
|
} catch (e) {
|
|
|
|
dump('exception while creating screenshot: ' + e + '\n');
|
2012-07-17 23:20:19 -07:00
|
|
|
shell.sendChromeEvent({
|
2012-06-27 07:47:30 -07:00
|
|
|
type: 'take-screenshot-error',
|
2012-06-15 00:20:44 -07:00
|
|
|
error: String(e)
|
|
|
|
});
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2012-07-24 09:34:12 -07:00
|
|
|
|
2012-10-11 07:18:00 -07:00
|
|
|
(function contentCrashTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
let props = aSubject.QueryInterface(Ci.nsIPropertyBag2);
|
|
|
|
if (props.hasKey("abnormal") && props.hasKey("dumpID")) {
|
2012-10-29 17:27:09 -07:00
|
|
|
shell.reportCrash(false, props.getProperty("dumpID"));
|
2012-10-11 07:18:00 -07:00
|
|
|
}
|
|
|
|
},
|
|
|
|
"ipc:content-shutdown", false);
|
|
|
|
})();
|
|
|
|
|
2012-11-14 19:26:30 -08:00
|
|
|
var CaptivePortalLoginHelper = {
|
|
|
|
init: function init() {
|
|
|
|
Services.obs.addObserver(this, 'captive-portal-login', false);
|
|
|
|
Services.obs.addObserver(this, 'captive-portal-login-abort', false);
|
|
|
|
},
|
|
|
|
handleEvent: function handleEvent(detail) {
|
|
|
|
Services.captivePortalDetector.cancelLogin(detail.id);
|
|
|
|
},
|
|
|
|
observe: function observe(subject, topic, data) {
|
|
|
|
shell.sendChromeEvent(JSON.parse(data));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2012-10-29 11:18:50 -07:00
|
|
|
// Listen for crashes submitted through the crash reporter UI.
|
|
|
|
window.addEventListener('ContentStart', function cr_onContentStart() {
|
|
|
|
let content = shell.contentBrowser.contentWindow;
|
|
|
|
content.addEventListener("mozContentEvent", function cr_onMozContentEvent(e) {
|
2012-10-29 17:27:09 -07:00
|
|
|
if (e.detail.type == "submit-crash" && e.detail.crashID) {
|
2013-05-10 14:39:15 -07:00
|
|
|
debugCrashReport("submitting crash at user request ", e.detail.crashID);
|
2012-10-29 11:18:50 -07:00
|
|
|
shell.submitCrash(e.detail.crashID);
|
2013-05-10 14:39:15 -07:00
|
|
|
} else if (e.detail.type == "delete-crash" && e.detail.crashID) {
|
|
|
|
debugCrashReport("deleting crash at user request ", e.detail.crashID);
|
|
|
|
shell.deleteCrash(e.detail.crashID);
|
2012-10-29 11:18:50 -07:00
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
2012-10-13 11:11:09 -07:00
|
|
|
window.addEventListener('ContentStart', function update_onContentStart() {
|
2013-09-11 05:00:48 -07:00
|
|
|
Cu.import('resource://gre/modules/WebappsUpdater.jsm');
|
|
|
|
WebappsUpdater.handleContentStart(shell);
|
|
|
|
|
2013-08-16 11:08:03 -07:00
|
|
|
let promptCc = Cc["@mozilla.org/updates/update-prompt;1"];
|
|
|
|
if (!promptCc) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
let updatePrompt = promptCc.createInstance(Ci.nsIUpdatePrompt);
|
2013-01-14 23:40:51 -08:00
|
|
|
if (!updatePrompt) {
|
|
|
|
return;
|
|
|
|
}
|
2012-10-13 11:11:09 -07:00
|
|
|
|
2013-01-14 23:40:51 -08:00
|
|
|
updatePrompt.wrappedJSObject.handleContentStart(shell);
|
2012-10-13 11:11:09 -07:00
|
|
|
});
|
|
|
|
|
2012-07-25 10:00:18 -07:00
|
|
|
(function geolocationStatusTracker() {
|
2013-02-08 08:34:47 -08:00
|
|
|
let gGeolocationActive = false;
|
2012-07-25 10:00:18 -07:00
|
|
|
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
2013-02-08 08:34:47 -08:00
|
|
|
let oldState = gGeolocationActive;
|
2012-07-25 10:00:18 -07:00
|
|
|
if (aData == "starting") {
|
2013-02-08 08:34:47 -08:00
|
|
|
gGeolocationActive = true;
|
2012-07-25 10:00:18 -07:00
|
|
|
} else if (aData == "shutdown") {
|
2013-02-08 08:34:47 -08:00
|
|
|
gGeolocationActive = false;
|
2012-07-25 10:00:18 -07:00
|
|
|
}
|
|
|
|
|
2013-02-08 08:34:47 -08:00
|
|
|
if (gGeolocationActive != oldState) {
|
2012-07-25 10:00:18 -07:00
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'geolocation-status',
|
2013-02-08 08:34:47 -08:00
|
|
|
active: gGeolocationActive
|
2012-07-25 10:00:18 -07:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}, "geolocation-device-events", false);
|
|
|
|
})();
|
2012-08-15 13:56:00 -07:00
|
|
|
|
2012-08-13 07:59:30 -07:00
|
|
|
(function headphonesStatusTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
shell.sendChromeEvent({
|
2012-09-06 13:47:58 -07:00
|
|
|
type: 'headphones-status-changed',
|
2012-08-13 07:59:30 -07:00
|
|
|
state: aData
|
|
|
|
});
|
2012-09-06 13:47:58 -07:00
|
|
|
}, "headphones-status-changed", false);
|
2012-08-13 07:59:30 -07:00
|
|
|
})();
|
|
|
|
|
2012-12-05 17:20:59 -08:00
|
|
|
(function audioChannelChangedTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'audio-channel-changed',
|
|
|
|
channel: aData
|
|
|
|
});
|
|
|
|
}, "audio-channel-changed", false);
|
|
|
|
})();
|
|
|
|
|
2013-09-12 05:26:03 -07:00
|
|
|
(function defaultVolumeChannelChangedTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'default-volume-channel-changed',
|
|
|
|
channel: aData
|
|
|
|
});
|
|
|
|
}, "default-volume-channel-changed", false);
|
|
|
|
})();
|
|
|
|
|
2013-01-28 03:47:18 -08:00
|
|
|
(function visibleAudioChannelChangedTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'visible-audio-channel-changed',
|
|
|
|
channel: aData
|
|
|
|
});
|
2013-03-26 21:04:32 -07:00
|
|
|
shell.visibleNormalAudioActive = (aData == 'normal');
|
2013-01-28 03:47:18 -08:00
|
|
|
}, "visible-audio-channel-changed", false);
|
|
|
|
})();
|
|
|
|
|
2012-08-15 13:56:00 -07:00
|
|
|
(function recordingStatusTracker() {
|
2013-10-25 17:04:57 -07:00
|
|
|
// Recording status is tracked per process with following data structure:
|
2013-11-25 22:22:23 -08:00
|
|
|
// {<processId>: {<requestURL>: {isApp: <isApp>,
|
|
|
|
// count: <N>,
|
|
|
|
// audioCount: <N>,
|
|
|
|
// videoCount: <N>}}
|
2013-09-30 05:09:55 -07:00
|
|
|
let gRecordingActiveProcesses = {};
|
2012-08-15 13:56:00 -07:00
|
|
|
|
2013-09-30 05:09:55 -07:00
|
|
|
let recordingHandler = function(aSubject, aTopic, aData) {
|
2013-10-25 17:04:57 -07:00
|
|
|
let props = aSubject.QueryInterface(Ci.nsIPropertyBag2);
|
|
|
|
let processId = (props.hasKey('childID')) ? props.get('childID')
|
|
|
|
: 'main';
|
2013-09-30 05:09:55 -07:00
|
|
|
if (processId && !gRecordingActiveProcesses.hasOwnProperty(processId)) {
|
2013-11-25 22:22:23 -08:00
|
|
|
gRecordingActiveProcesses[processId] = {};
|
2013-09-30 05:09:55 -07:00
|
|
|
}
|
|
|
|
|
2013-11-25 22:22:23 -08:00
|
|
|
let commandHandler = function (requestURL, command) {
|
|
|
|
let currentProcess = gRecordingActiveProcesses[processId];
|
|
|
|
let currentActive = currentProcess[requestURL];
|
|
|
|
let wasActive = (currentActive['count'] > 0);
|
|
|
|
let wasAudioActive = (currentActive['audioCount'] > 0);
|
|
|
|
let wasVideoActive = (currentActive['videoCount'] > 0);
|
|
|
|
|
|
|
|
switch (command.type) {
|
|
|
|
case 'starting':
|
|
|
|
currentActive['count']++;
|
|
|
|
currentActive['audioCount'] += (command.isAudio) ? 1 : 0;
|
|
|
|
currentActive['videoCount'] += (command.isVideo) ? 1 : 0;
|
|
|
|
break;
|
|
|
|
case 'shutdown':
|
|
|
|
currentActive['count']--;
|
|
|
|
currentActive['audioCount'] -= (command.isAudio) ? 1 : 0;
|
|
|
|
currentActive['videoCount'] -= (command.isVideo) ? 1 : 0;
|
|
|
|
break;
|
|
|
|
case 'content-shutdown':
|
|
|
|
currentActive['count'] = 0;
|
|
|
|
currentActive['audioCount'] = 0;
|
|
|
|
currentActive['videoCount'] = 0;
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
|
|
|
if (currentActive['count'] > 0) {
|
|
|
|
currentProcess[requestURL] = currentActive;
|
|
|
|
} else {
|
|
|
|
delete currentProcess[requestURL];
|
|
|
|
}
|
|
|
|
|
|
|
|
// We need to track changes if any active state is changed.
|
|
|
|
let isActive = (currentActive['count'] > 0);
|
|
|
|
let isAudioActive = (currentActive['audioCount'] > 0);
|
|
|
|
let isVideoActive = (currentActive['videoCount'] > 0);
|
|
|
|
if ((isActive != wasActive) ||
|
|
|
|
(isAudioActive != wasAudioActive) ||
|
|
|
|
(isVideoActive != wasVideoActive)) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'recording-status',
|
|
|
|
active: isActive,
|
|
|
|
requestURL: requestURL,
|
|
|
|
isApp: currentActive['isApp'],
|
|
|
|
isAudio: isAudioActive,
|
|
|
|
isVideo: isVideoActive
|
|
|
|
});
|
|
|
|
}
|
|
|
|
};
|
2013-10-25 17:04:57 -07:00
|
|
|
|
2013-09-30 05:09:55 -07:00
|
|
|
switch (aData) {
|
|
|
|
case 'starting':
|
|
|
|
case 'shutdown':
|
2013-11-25 22:22:23 -08:00
|
|
|
// create page record if it is not existed yet.
|
|
|
|
let requestURL = props.get('requestURL');
|
|
|
|
if (requestURL &&
|
|
|
|
!gRecordingActiveProcesses[processId].hasOwnProperty(requestURL)) {
|
|
|
|
gRecordingActiveProcesses[processId][requestURL] = {isApp: props.get('isApp'),
|
|
|
|
count: 0,
|
|
|
|
audioCount: 0,
|
|
|
|
videoCount: 0};
|
|
|
|
}
|
|
|
|
commandHandler(requestURL, { type: aData,
|
|
|
|
isAudio: props.get('isAudio'),
|
|
|
|
isVideo: props.get('isVideo')});
|
2013-09-30 05:09:55 -07:00
|
|
|
break;
|
|
|
|
case 'content-shutdown':
|
2013-11-25 22:22:23 -08:00
|
|
|
// iterate through all the existing active processes
|
2013-12-05 09:08:26 -08:00
|
|
|
Object.keys(gRecordingActiveProcesses[processId]).forEach(function(requestURL) {
|
2013-11-25 22:22:23 -08:00
|
|
|
commandHandler(requestURL, { type: aData,
|
|
|
|
isAudio: true,
|
|
|
|
isVideo: true});
|
|
|
|
});
|
2013-09-30 05:09:55 -07:00
|
|
|
break;
|
|
|
|
}
|
|
|
|
|
2013-11-25 22:22:23 -08:00
|
|
|
// clean up process record if no page record in it.
|
|
|
|
if (Object.keys(gRecordingActiveProcesses[processId]).length == 0) {
|
2013-09-30 05:09:55 -07:00
|
|
|
delete gRecordingActiveProcesses[processId];
|
2012-08-15 13:56:00 -07:00
|
|
|
}
|
2013-09-30 05:09:55 -07:00
|
|
|
};
|
|
|
|
Services.obs.addObserver(recordingHandler, 'recording-device-events', false);
|
|
|
|
Services.obs.addObserver(recordingHandler, 'recording-device-ipc-events', false);
|
|
|
|
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
// send additional recording events if content process is being killed
|
2013-10-25 17:04:57 -07:00
|
|
|
let processId = aSubject.QueryInterface(Ci.nsIPropertyBag2).get('childID');
|
|
|
|
if (gRecordingActiveProcesses.hasOwnProperty(processId)) {
|
2013-09-30 05:09:55 -07:00
|
|
|
Services.obs.notifyObservers(aSubject, 'recording-device-ipc-events', 'content-shutdown');
|
|
|
|
}
|
|
|
|
}, 'ipc:content-shutdown', false);
|
2012-08-15 13:56:00 -07:00
|
|
|
})();
|
2012-09-04 08:51:46 -07:00
|
|
|
|
|
|
|
(function volumeStateTracker() {
|
|
|
|
Services.obs.addObserver(function(aSubject, aTopic, aData) {
|
|
|
|
shell.sendChromeEvent({
|
|
|
|
type: 'volume-state-changed',
|
|
|
|
active: (aData == 'Shared')
|
|
|
|
});
|
|
|
|
}, 'volume-state-changed', false);
|
|
|
|
})();
|
2012-10-15 12:26:49 -07:00
|
|
|
|
2013-05-17 15:57:10 -07:00
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
// Devices don't have all the same partition size for /cache where we
|
|
|
|
// store the http cache.
|
|
|
|
(function setHTTPCacheSize() {
|
|
|
|
let path = Services.prefs.getCharPref("browser.cache.disk.parent_directory");
|
|
|
|
let volumeService = Cc["@mozilla.org/telephony/volume-service;1"]
|
|
|
|
.getService(Ci.nsIVolumeService);
|
|
|
|
|
|
|
|
let stats = volumeService.createOrGetVolumeByPath(path).getStats();
|
|
|
|
|
|
|
|
// We must set the size in KB, and keep a bit of free space.
|
|
|
|
let size = Math.floor(stats.totalBytes / 1024) - 1024;
|
|
|
|
Services.prefs.setIntPref("browser.cache.disk.capacity", size);
|
|
|
|
}) ()
|
|
|
|
#endif
|
2013-08-03 03:50:18 -07:00
|
|
|
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
let SensorsListener = {
|
|
|
|
sensorsListenerDevices: ['crespo'],
|
|
|
|
device: libcutils.property_get("ro.product.device"),
|
|
|
|
|
|
|
|
deviceNeedsWorkaround: function SensorsListener_deviceNeedsWorkaround() {
|
|
|
|
return (this.sensorsListenerDevices.indexOf(this.device) != -1);
|
|
|
|
},
|
|
|
|
|
|
|
|
handleEvent: function SensorsListener_handleEvent(evt) {
|
|
|
|
switch(evt.type) {
|
|
|
|
case 'devicemotion':
|
|
|
|
// Listener that does nothing, we need this to have the sensor being
|
|
|
|
// able to report correct values, as explained in bug 753245, comment 6
|
|
|
|
// and in bug 871916
|
|
|
|
break;
|
|
|
|
|
|
|
|
default:
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
observe: function SensorsListener_observe(subject, topic, data) {
|
|
|
|
// We remove the listener when the screen is off, otherwise sensor will
|
|
|
|
// continue to bother us with data and we won't be able to get the
|
|
|
|
// system into suspend state, thus draining battery.
|
|
|
|
if (data === 'on') {
|
|
|
|
window.addEventListener('devicemotion', this);
|
|
|
|
} else {
|
|
|
|
window.removeEventListener('devicemotion', this);
|
|
|
|
}
|
|
|
|
},
|
|
|
|
|
|
|
|
init: function SensorsListener_init() {
|
|
|
|
if (this.deviceNeedsWorkaround()) {
|
|
|
|
// On boot, enable the listener, screen will be on.
|
|
|
|
window.addEventListener('devicemotion', this);
|
|
|
|
|
|
|
|
// Then listen for further screen state changes
|
|
|
|
Services.obs.addObserver(this, 'screen-state-changed', false);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
SensorsListener.init();
|
|
|
|
#endif
|
2013-09-18 10:42:06 -07:00
|
|
|
|
|
|
|
// Calling this observer will cause a shutdown an a profile reset.
|
|
|
|
// Use eg. : Services.obs.notifyObservers(null, 'b2g-reset-profile', null);
|
|
|
|
Services.obs.addObserver(function resetProfile(subject, topic, data) {
|
|
|
|
Services.obs.removeObserver(resetProfile, topic);
|
|
|
|
|
|
|
|
// Listening for 'profile-before-change2' which is late in the shutdown
|
|
|
|
// sequence, but still has xpcom access.
|
|
|
|
Services.obs.addObserver(function clearProfile(subject, topic, data) {
|
|
|
|
Services.obs.removeObserver(clearProfile, topic);
|
|
|
|
#ifdef MOZ_WIDGET_GONK
|
|
|
|
let json = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsIFile);
|
|
|
|
json.initWithPath('/system/b2g/webapps/webapps.json');
|
|
|
|
let toRemove = json.exists()
|
|
|
|
// This is a user build, just rm -r /data/local /data/b2g/mozilla
|
|
|
|
? ['/data/local', '/data/b2g/mozilla']
|
|
|
|
// This is an eng build. We clear the profile and a set of files
|
|
|
|
// under /data/local.
|
|
|
|
: ['/data/b2g/mozilla',
|
|
|
|
'/data/local/permissions.sqlite',
|
|
|
|
'/data/local/storage',
|
|
|
|
'/data/local/OfflineCache'];
|
|
|
|
|
|
|
|
toRemove.forEach(function(dir) {
|
|
|
|
try {
|
|
|
|
let file = Cc['@mozilla.org/file/local;1'].createInstance(Ci.nsIFile);
|
|
|
|
file.initWithPath(dir);
|
|
|
|
file.remove(true);
|
|
|
|
} catch(e) { dump(e); }
|
|
|
|
});
|
|
|
|
#else
|
|
|
|
// Desktop builds.
|
|
|
|
let profile = Services.dirsvc.get('ProfD', Ci.nsIFile);
|
|
|
|
|
|
|
|
// We don't want to remove everything from the profile, since this
|
|
|
|
// would prevent us from starting up.
|
|
|
|
let whitelist = ['defaults', 'extensions', 'settings.json',
|
|
|
|
'user.js', 'webapps'];
|
|
|
|
let enumerator = profile.directoryEntries;
|
|
|
|
while (enumerator.hasMoreElements()) {
|
|
|
|
let file = enumerator.getNext().QueryInterface(Ci.nsIFile);
|
|
|
|
if (whitelist.indexOf(file.leafName) == -1) {
|
|
|
|
file.remove(true);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
#endif
|
|
|
|
},
|
|
|
|
'profile-before-change2', false);
|
|
|
|
|
|
|
|
let appStartup = Cc['@mozilla.org/toolkit/app-startup;1']
|
|
|
|
.getService(Ci.nsIAppStartup);
|
|
|
|
appStartup.quit(Ci.nsIAppStartup.eForceQuit);
|
|
|
|
}, 'b2g-reset-profile', false);
|
2013-11-27 23:18:08 -08:00
|
|
|
|
|
|
|
/**
|
|
|
|
* CID of our implementation of nsIDownloadManagerUI.
|
|
|
|
*/
|
|
|
|
const kTransferCid = Components.ID("{1b4c85df-cbdd-4bb6-b04e-613caece083c}");
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Contract ID of the service implementing nsITransfer.
|
|
|
|
*/
|
|
|
|
const kTransferContractId = "@mozilla.org/transfer;1";
|
|
|
|
|
|
|
|
// Override Toolkit's nsITransfer implementation with the one from the
|
|
|
|
// JavaScript API for downloads. This will eventually be removed when
|
|
|
|
// nsIDownloadManager will not be available anymore (bug 851471). The
|
|
|
|
// old code in this module will be removed in bug 899110.
|
|
|
|
Components.manager.QueryInterface(Ci.nsIComponentRegistrar)
|
|
|
|
.registerFactory(kTransferCid, "",
|
|
|
|
kTransferContractId, null);
|