summaryrefslogtreecommitdiffstats
path: root/mobile/android/modules
diff options
context:
space:
mode:
Diffstat (limited to 'mobile/android/modules')
-rw-r--r--mobile/android/modules/Accounts.jsm79
-rw-r--r--mobile/android/modules/AndroidLog.jsm92
-rw-r--r--mobile/android/modules/DelayedInit.jsm177
-rw-r--r--mobile/android/modules/DownloadNotifications.jsm291
-rw-r--r--mobile/android/modules/HelperApps.jsm229
-rw-r--r--mobile/android/modules/Home.jsm487
-rw-r--r--mobile/android/modules/HomeProvider.jsm407
-rw-r--r--mobile/android/modules/JNI.jsm1167
-rw-r--r--mobile/android/modules/JavaAddonManager.jsm115
-rw-r--r--mobile/android/modules/LightweightThemeConsumer.jsm44
-rw-r--r--mobile/android/modules/MediaPlayerApp.jsm166
-rw-r--r--mobile/android/modules/Messaging.jsm183
-rw-r--r--mobile/android/modules/NetErrorHelper.jsm175
-rw-r--r--mobile/android/modules/Notifications.jsm259
-rw-r--r--mobile/android/modules/PageActions.jsm113
-rw-r--r--mobile/android/modules/Prompt.jsm234
-rw-r--r--mobile/android/modules/RuntimePermissions.jsm41
-rw-r--r--mobile/android/modules/SSLExceptions.jsm118
-rw-r--r--mobile/android/modules/Sanitizer.jsm303
-rw-r--r--mobile/android/modules/SharedPreferences.jsm254
-rw-r--r--mobile/android/modules/Snackbars.jsm72
-rw-r--r--mobile/android/modules/TabMirror.jsm153
-rw-r--r--mobile/android/modules/WebsiteMetadata.jsm475
-rw-r--r--mobile/android/modules/dbg-browser-actors.js77
-rw-r--r--mobile/android/modules/moz.build32
25 files changed, 0 insertions, 5743 deletions
diff --git a/mobile/android/modules/Accounts.jsm b/mobile/android/modules/Accounts.jsm
deleted file mode 100644
index ee0fc0c5b..000000000
--- a/mobile/android/modules/Accounts.jsm
+++ /dev/null
@@ -1,79 +0,0 @@
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["Accounts"];
-
-const { utils: Cu } = Components;
-
-Cu.import("resource://gre/modules/Deprecated.jsm"); /*global Deprecated */
-Cu.import("resource://gre/modules/Messaging.jsm"); /*global Messaging */
-Cu.import("resource://gre/modules/Promise.jsm"); /*global Promise */
-Cu.import("resource://gre/modules/Services.jsm"); /*global Services */
-
-/**
- * A promise-based API for querying the existence of Sync accounts,
- * and accessing the Sync setup wizard.
- *
- * Usage:
- *
- * Cu.import("resource://gre/modules/Accounts.jsm");
- * Accounts.anySyncAccountsExist().then(
- * (exist) => {
- * console.log("Accounts exist? " + exist);
- * if (!exist) {
- * Accounts.launchSetup();
- * }
- * },
- * (err) => {
- * console.log("We failed so hard.");
- * }
- * );
- */
-var Accounts = Object.freeze({
- _accountsExist: function (kind) {
- return Messaging.sendRequestForResult({
- type: "Accounts:Exist",
- kind: kind
- }).then(data => data.exists);
- },
-
- syncAccountsExist: function () {
- Deprecated.warning("The legacy Sync account type has been removed from Firefox for Android.");
- return Promise.resolve(false);
- },
-
- anySyncAccountsExist: function () {
- return this._accountsExist("any");
- },
-
- /**
- * Fire-and-forget: open the Firefox accounts activity, which
- * will be the Getting Started screen if FxA isn't yet set up.
- *
- * Optional extras are passed, as a JSON string, to the Firefox
- * Account Getting Started activity in the extras bundle of the
- * activity launch intent, under the key "extras".
- *
- * There is no return value from this method.
- */
- launchSetup: function (extras) {
- Messaging.sendRequest({
- type: "Accounts:Create",
- extras: extras
- });
- },
-
- _addDefaultEndpoints: function (json) {
- // Empty without FxA
- let newData = Cu.cloneInto(json, {}, { cloneFunctions: false });
- return newData;
- },
-
- showSyncPreferences: function () {
- // Only show Sync preferences of an existing Android Account.
- throw new Error("Can't show Sync preferences without accounts!");
- }
-});
diff --git a/mobile/android/modules/AndroidLog.jsm b/mobile/android/modules/AndroidLog.jsm
deleted file mode 100644
index be6cca4e8..000000000
--- a/mobile/android/modules/AndroidLog.jsm
+++ /dev/null
@@ -1,92 +0,0 @@
-/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
- * 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";
-
-/**
- * Native Android logging for JavaScript. Lets you specify a priority and tag
- * in addition to the message being logged. Resembles the android.util.Log API
- * <http://developer.android.com/reference/android/util/Log.html>.
- *
- * // Import it as a JSM:
- * let Log = Cu.import("resource://gre/modules/AndroidLog.jsm", {}).AndroidLog;
- *
- * // Or require it in a chrome worker:
- * importScripts("resource://gre/modules/workers/require.js");
- * let Log = require("resource://gre/modules/AndroidLog.jsm");
- *
- * // Use Log.i, Log.v, Log.d, Log.w, and Log.e to log verbose, debug, info,
- * // warning, and error messages, respectively.
- * Log.v("MyModule", "This is a verbose message.");
- * Log.d("MyModule", "This is a debug message.");
- * Log.i("MyModule", "This is an info message.");
- * Log.w("MyModule", "This is a warning message.");
- * Log.e("MyModule", "This is an error message.");
- *
- * // Bind a function with a tag to replace a bespoke dump/log/debug function:
- * let debug = Log.d.bind(null, "MyModule");
- * debug("This is a debug message.");
- * // Outputs "D/GeckoMyModule(#####): This is a debug message."
- *
- * // Or "bind" the module object to a tag to automatically tag messages:
- * Log = Log.bind("MyModule");
- * Log.d("This is a debug message.");
- * // Outputs "D/GeckoMyModule(#####): This is a debug message."
- *
- * Note: the module automatically prepends "Gecko" to the tag you specify,
- * since all tags used by Fennec code should start with that string; and it
- * truncates tags longer than MAX_TAG_LENGTH characters (not including "Gecko").
- */
-
-if (typeof Components != "undefined") {
- // Specify exported symbols for JSM module loader.
- this.EXPORTED_SYMBOLS = ["AndroidLog"];
- Components.utils.import("resource://gre/modules/ctypes.jsm");
-}
-
-// From <https://android.googlesource.com/platform/system/core/+/master/include/android/log.h>.
-const ANDROID_LOG_VERBOSE = 2;
-const ANDROID_LOG_DEBUG = 3;
-const ANDROID_LOG_INFO = 4;
-const ANDROID_LOG_WARN = 5;
-const ANDROID_LOG_ERROR = 6;
-
-// android.util.Log.isLoggable throws IllegalArgumentException if a tag length
-// exceeds 23 characters, and we prepend five characters ("Gecko") to every tag,
-// so we truncate tags exceeding 18 characters (although __android_log_write
-// itself and other android.util.Log methods don't seem to mind longer tags).
-const MAX_TAG_LENGTH = 18;
-
-var liblog = ctypes.open("liblog.so"); // /system/lib/liblog.so
-var __android_log_write = liblog.declare("__android_log_write",
- ctypes.default_abi,
- ctypes.int, // return value: num bytes logged
- ctypes.int, // priority (ANDROID_LOG_* constant)
- ctypes.char.ptr, // tag
- ctypes.char.ptr); // message
-
-var AndroidLog = {
- MAX_TAG_LENGTH: MAX_TAG_LENGTH,
- v: (tag, msg) => __android_log_write(ANDROID_LOG_VERBOSE, "Gecko" + tag.substring(0, MAX_TAG_LENGTH), msg),
- d: (tag, msg) => __android_log_write(ANDROID_LOG_DEBUG, "Gecko" + tag.substring(0, MAX_TAG_LENGTH), msg),
- i: (tag, msg) => __android_log_write(ANDROID_LOG_INFO, "Gecko" + tag.substring(0, MAX_TAG_LENGTH), msg),
- w: (tag, msg) => __android_log_write(ANDROID_LOG_WARN, "Gecko" + tag.substring(0, MAX_TAG_LENGTH), msg),
- e: (tag, msg) => __android_log_write(ANDROID_LOG_ERROR, "Gecko" + tag.substring(0, MAX_TAG_LENGTH), msg),
-
- bind: function(tag) {
- return {
- MAX_TAG_LENGTH: MAX_TAG_LENGTH,
- v: AndroidLog.v.bind(null, tag),
- d: AndroidLog.d.bind(null, tag),
- i: AndroidLog.i.bind(null, tag),
- w: AndroidLog.w.bind(null, tag),
- e: AndroidLog.e.bind(null, tag),
- };
- },
-};
-
-if (typeof Components == "undefined") {
- // Specify exported symbols for require.js module loader.
- module.exports = AndroidLog;
-}
diff --git a/mobile/android/modules/DelayedInit.jsm b/mobile/android/modules/DelayedInit.jsm
deleted file mode 100644
index 7c33a3ede..000000000
--- a/mobile/android/modules/DelayedInit.jsm
+++ /dev/null
@@ -1,177 +0,0 @@
-/* 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"
-
-/*globals MessageLoop */
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-
-this.EXPORTED_SYMBOLS = ["DelayedInit"];
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-XPCOMUtils.defineLazyServiceGetter(this, "MessageLoop",
- "@mozilla.org/message-loop;1",
- "nsIMessageLoop");
-
-/**
- * Use DelayedInit to schedule initializers to run some time after startup.
- * Initializers are added to a list of pending inits. Whenever the main thread
- * message loop is idle, DelayedInit will start running initializers from the
- * pending list. To prevent monopolizing the message loop, every idling period
- * has a maximum duration. When that's reached, we give up the message loop and
- * wait for the next idle.
- *
- * DelayedInit is compatible with lazy getters like those from XPCOMUtils. When
- * the lazy getter is first accessed, its corresponding initializer is run
- * automatically if it hasn't been run already. Each initializer also has a
- * maximum wait parameter that specifies a mandatory timeout; when the timeout
- * is reached, the initializer is forced to run.
- *
- * DelayedInit.schedule(() => Foo.init(), null, null, 5000);
- *
- * In the example above, Foo.init will run automatically when the message loop
- * becomes idle, or when 5000ms has elapsed, whichever comes first.
- *
- * DelayedInit.schedule(() => Foo.init(), this, "Foo", 5000);
- *
- * In the example above, Foo.init will run automatically when the message loop
- * becomes idle, when |this.Foo| is accessed, or when 5000ms has elapsed,
- * whichever comes first.
- *
- * It may be simpler to have a wrapper for DelayedInit.schedule. For example,
- *
- * function InitLater(fn, obj, name) {
- * return DelayedInit.schedule(fn, obj, name, 5000); // constant max wait
- * }
- * InitLater(() => Foo.init());
- * InitLater(() => Bar.init(), this, "Bar");
- */
-var DelayedInit = {
- schedule: function (fn, object, name, maxWait) {
- return Impl.scheduleInit(fn, object, name, maxWait);
- },
-};
-
-// Maximum duration for each idling period. Pending inits are run until this
-// duration is exceeded; then we wait for next idling period.
-const MAX_IDLE_RUN_MS = 50;
-
-var Impl = {
- pendingInits: [],
-
- onIdle: function () {
- let startTime = Cu.now();
- let time = startTime;
- let nextDue;
-
- // Go through all the pending inits. Even if we don't run them,
- // we still need to find out when the next timeout should be.
- for (let init of this.pendingInits) {
- if (init.complete) {
- continue;
- }
-
- if (time - startTime < MAX_IDLE_RUN_MS) {
- init.maybeInit();
- time = Cu.now();
- } else {
- // We ran out of time; find when the next closest due time is.
- nextDue = nextDue ? Math.min(nextDue, init.due) : init.due;
- }
- }
-
- // Get rid of completed ones.
- this.pendingInits = this.pendingInits.filter((init) => !init.complete);
-
- if (nextDue !== undefined) {
- // Schedule the next idle, if we still have pending inits.
- MessageLoop.postIdleTask(() => this.onIdle(),
- Math.max(0, nextDue - time));
- }
- },
-
- addPendingInit: function (fn, wait) {
- let init = {
- fn: fn,
- due: Cu.now() + wait,
- complete: false,
- maybeInit: function () {
- if (this.complete) {
- return false;
- }
- this.complete = true;
- this.fn.call();
- this.fn = null;
- return true;
- },
- };
-
- if (!this.pendingInits.length) {
- // Schedule for the first idle.
- MessageLoop.postIdleTask(() => this.onIdle(), wait);
- }
- this.pendingInits.push(init);
- return init;
- },
-
- scheduleInit: function (fn, object, name, wait) {
- let init = this.addPendingInit(fn, wait);
-
- if (!object || !name) {
- // No lazy getter needed.
- return;
- }
-
- // Get any existing information about the property.
- let prop = Object.getOwnPropertyDescriptor(object, name) ||
- { configurable: true, enumerable: true, writable: true };
-
- if (!prop.configurable) {
- // Object.defineProperty won't work, so just perform init here.
- init.maybeInit();
- return;
- }
-
- // Define proxy getter/setter that will call first initializer first,
- // before delegating the get/set to the original target.
- Object.defineProperty(object, name, {
- get: function proxy_getter() {
- init.maybeInit();
-
- // If the initializer actually ran, it may have replaced our proxy
- // property with a real one, so we need to reload he property.
- let newProp = Object.getOwnPropertyDescriptor(object, name);
- if (newProp.get !== proxy_getter) {
- // Set prop if newProp doesn't refer to our proxy property.
- prop = newProp;
- } else {
- // Otherwise, reset to the original property.
- Object.defineProperty(object, name, prop);
- }
-
- if (prop.get) {
- return prop.get.call(object);
- }
- return prop.value;
- },
- set: function (newVal) {
- init.maybeInit();
-
- // Since our initializer already ran,
- // we can get rid of our proxy property.
- if (prop.get || prop.set) {
- Object.defineProperty(object, name, prop);
- return prop.set.call(object);
- }
-
- prop.value = newVal;
- Object.defineProperty(object, name, prop);
- return newVal;
- },
- configurable: true,
- enumerable: true,
- });
- }
-};
diff --git a/mobile/android/modules/DownloadNotifications.jsm b/mobile/android/modules/DownloadNotifications.jsm
deleted file mode 100644
index 39b520979..000000000
--- a/mobile/android/modules/DownloadNotifications.jsm
+++ /dev/null
@@ -1,291 +0,0 @@
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["DownloadNotifications"];
-
-const { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components;
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Downloads", "resource://gre/modules/Downloads.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "FileUtils", "resource://gre/modules/FileUtils.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "Notifications", "resource://gre/modules/Notifications.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "OS", "resource://gre/modules/osfile.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "Services", "resource://gre/modules/Services.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "Snackbars", "resource://gre/modules/Snackbars.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "UITelemetry", "resource://gre/modules/UITelemetry.jsm");
-
-XPCOMUtils.defineLazyServiceGetter(this, "ParentalControls",
- "@mozilla.org/parental-controls-service;1", "nsIParentalControlsService");
-
-var Log = Cu.import("resource://gre/modules/AndroidLog.jsm", {}).AndroidLog.i.bind(null, "DownloadNotifications");
-
-XPCOMUtils.defineLazyGetter(this, "strings",
- () => Services.strings.createBundle("chrome://browser/locale/browser.properties"));
-
-Object.defineProperty(this, "window",
- { get: () => Services.wm.getMostRecentWindow("navigator:browser") });
-
-const kButtons = {
- PAUSE: new DownloadNotificationButton("pause",
- "drawable://pause",
- "alertDownloadsPause"),
- RESUME: new DownloadNotificationButton("resume",
- "drawable://play",
- "alertDownloadsResume"),
- CANCEL: new DownloadNotificationButton("cancel",
- "drawable://close",
- "alertDownloadsCancel")
-};
-
-var notifications = new Map();
-
-var DownloadNotifications = {
- _notificationKey: "downloads",
-
- init: function () {
- Downloads.getList(Downloads.ALL)
- .then(list => list.addView(this))
- .then(() => this._viewAdded = true, Cu.reportError);
-
- // All click, cancel, and button presses will be handled by this handler as part of the Notifications callback API.
- Notifications.registerHandler(this._notificationKey, this);
- },
-
- onDownloadAdded: function (download) {
- // Don't create notifications for pre-existing succeeded downloads.
- // We still add notifications for canceled downloads in case the
- // user decides to retry the download.
- if (download.succeeded && !this._viewAdded) {
- return;
- }
-
- if (!ParentalControls.isAllowed(ParentalControls.DOWNLOAD)) {
- download.cancel().catch(Cu.reportError);
- download.removePartialData().catch(Cu.reportError);
- Snackbars.show(strings.GetStringFromName("downloads.disabledInGuest"), Snackbars.LENGTH_LONG);
- return;
- }
-
- let notification = new DownloadNotification(download);
- notifications.set(download, notification);
- notification.showOrUpdate();
-
- // If this is a new download, show a snackbar as well.
- if (this._viewAdded) {
- Snackbars.show(strings.GetStringFromName("alertDownloadsToast"), Snackbars.LENGTH_LONG);
- }
- },
-
- onDownloadChanged: function (download) {
- let notification = notifications.get(download);
-
- if (download.succeeded) {
- let file = new FileUtils.File(download.target.path);
-
- Snackbars.show(strings.formatStringFromName("alertDownloadSucceeded", [file.leafName], 1), Snackbars.LENGTH_LONG, {
- action: {
- label: strings.GetStringFromName("helperapps.open"),
- callback: () => {
- UITelemetry.addEvent("launch.1", "toast", null, "downloads");
- try {
- file.launch();
- } catch (ex) {
- this.showInAboutDownloads(download);
- }
- if (notification) {
- notification.hide();
- }
- }
- }});
- }
-
- if (notification) {
- notification.showOrUpdate();
- }
- },
-
- onDownloadRemoved: function (download) {
- let notification = notifications.get(download);
- if (!notification) {
- Cu.reportError("Download doesn't have a notification.");
- return;
- }
-
- notification.hide();
- notifications.delete(download);
- },
-
- _findDownloadForCookie: function(cookie) {
- return Downloads.getList(Downloads.ALL)
- .then(list => list.getAll())
- .then((downloads) => {
- for (let download of downloads) {
- let cookie2 = getCookieFromDownload(download);
- if (cookie2 === cookie) {
- return download;
- }
- }
-
- throw "Couldn't find download for " + cookie;
- });
- },
-
- onCancel: function(cookie) {
- // TODO: I'm not sure what we do here...
- },
-
- showInAboutDownloads: function (download) {
- let hash = "#" + window.encodeURIComponent(download.target.path);
-
- // Force using string equality to find a tab
- window.BrowserApp.selectOrAddTab("about:downloads" + hash, null, { startsWith: true });
- },
-
- onClick: function(cookie) {
- this._findDownloadForCookie(cookie).then((download) => {
- if (download.succeeded) {
- // We don't call Download.launch(), because there's (currently) no way to
- // tell if the file was actually launched or not, and we want to show
- // about:downloads if the launch failed.
- let file = new FileUtils.File(download.target.path);
- try {
- file.launch();
- } catch (ex) {
- this.showInAboutDownloads(download);
- }
- } else {
- ConfirmCancelPrompt.show(download);
- }
- }).catch(Cu.reportError);
- },
-
- onButtonClick: function(button, cookie) {
- this._findDownloadForCookie(cookie).then((download) => {
- if (button === kButtons.PAUSE.buttonId) {
- download.cancel().catch(Cu.reportError);
- } else if (button === kButtons.RESUME.buttonId) {
- download.start().catch(Cu.reportError);
- } else if (button === kButtons.CANCEL.buttonId) {
- download.cancel().catch(Cu.reportError);
- download.removePartialData().catch(Cu.reportError);
- }
- }).catch(Cu.reportError);
- },
-};
-
-function getCookieFromDownload(download) {
- return download.target.path +
- download.source.url +
- download.startTime;
-}
-
-function DownloadNotification(download) {
- this.download = download;
- this._fileName = OS.Path.basename(download.target.path);
-
- this.id = null;
-}
-
-DownloadNotification.prototype = {
- _updateFromDownload: function () {
- this._downloading = !this.download.stopped;
- this._paused = this.download.canceled && this.download.hasPartialData;
- this._succeeded = this.download.succeeded;
-
- this._show = this._downloading || this._paused || this._succeeded;
- },
-
- get options() {
- if (!this._show) {
- return null;
- }
-
- let options = {
- icon: "drawable://alert_download",
- cookie: getCookieFromDownload(this.download),
- handlerKey: DownloadNotifications._notificationKey
- };
-
- if (this._downloading) {
- options.icon = "drawable://alert_download_animation";
- if (this.download.currentBytes == 0) {
- this._updateOptionsForStatic(options, "alertDownloadsStart2");
- } else {
- let buttons = this.download.hasPartialData ? [kButtons.PAUSE, kButtons.CANCEL] :
- [kButtons.CANCEL]
- this._updateOptionsForOngoing(options, buttons);
- }
- } else if (this._paused) {
- this._updateOptionsForOngoing(options, [kButtons.RESUME, kButtons.CANCEL]);
- } else if (this._succeeded) {
- options.persistent = false;
- this._updateOptionsForStatic(options, "alertDownloadsDone2");
- }
-
- return options;
- },
-
- _updateOptionsForStatic : function (options, titleName) {
- options.title = strings.GetStringFromName(titleName);
- options.message = this._fileName;
- },
-
- _updateOptionsForOngoing: function (options, buttons) {
- options.title = this._fileName;
- options.message = this.download.progress + "%";
- options.buttons = buttons;
- options.ongoing = true;
- options.progress = this.download.progress;
- options.persistent = true;
- },
-
- showOrUpdate: function () {
- this._updateFromDownload();
-
- if (this._show) {
- if (!this.id) {
- this.id = Notifications.create(this.options);
- } else if (!this.options.ongoing) {
- // We need to explictly cancel ongoing notifications,
- // since updating them to be non-ongoing doesn't seem
- // to work. See bug 1130834.
- Notifications.cancel(this.id);
- this.id = Notifications.create(this.options);
- } else {
- Notifications.update(this.id, this.options);
- }
- } else {
- this.hide();
- }
- },
-
- hide: function () {
- if (this.id) {
- Notifications.cancel(this.id);
- this.id = null;
- }
- },
-};
-
-var ConfirmCancelPrompt = {
- show: function (download) {
- // Open a prompt that offers a choice to cancel the download
- let title = strings.GetStringFromName("downloadCancelPromptTitle1");
- let message = strings.GetStringFromName("downloadCancelPromptMessage1");
-
- if (Services.prompt.confirm(null, title, message)) {
- download.cancel().catch(Cu.reportError);
- download.removePartialData().catch(Cu.reportError);
- }
- }
-};
-
-function DownloadNotificationButton(buttonId, iconUrl, titleStringName, onClicked) {
- this.buttonId = buttonId;
- this.title = strings.GetStringFromName(titleStringName);
- this.icon = iconUrl;
-}
diff --git a/mobile/android/modules/HelperApps.jsm b/mobile/android/modules/HelperApps.jsm
deleted file mode 100644
index 0ac478da0..000000000
--- a/mobile/android/modules/HelperApps.jsm
+++ /dev/null
@@ -1,229 +0,0 @@
-/* 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";
-
-/* globals ContentAreaUtils */
-
-const { classes: Cc, interfaces: Ci, utils: Cu, results: Cr } = Components;
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Services.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Prompt",
- "resource://gre/modules/Prompt.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Messaging",
- "resource://gre/modules/Messaging.jsm");
-
-XPCOMUtils.defineLazyGetter(this, "ContentAreaUtils", function() {
- let ContentAreaUtils = {};
- Services.scriptloader.loadSubScript("chrome://global/content/contentAreaUtils.js", ContentAreaUtils);
- return ContentAreaUtils;
-});
-
-this.EXPORTED_SYMBOLS = ["App","HelperApps"];
-
-function App(data) {
- this.name = data.name;
- this.isDefault = data.isDefault;
- this.packageName = data.packageName;
- this.activityName = data.activityName;
- this.iconUri = "-moz-icon://" + data.packageName;
-}
-
-App.prototype = {
- // callback will be null if a result is not requested
- launch: function(uri, callback) {
- HelperApps._launchApp(this, uri, callback);
- return false;
- }
-}
-
-var HelperApps = {
- get defaultBrowsers() {
- delete this.defaultBrowsers;
- this.defaultBrowsers = this._getHandlers("http://www.example.com", {
- filterBrowsers: false,
- filterHtml: false
- });
- return this.defaultBrowsers;
- },
-
- // Finds handlers that have registered for text/html pages or urls ending in html. Some apps, like
- // the Samsung Video player will only appear for these urls, while some Browsers (like Link Bubble)
- // won't register here because of the text/html mime type.
- get defaultHtmlHandlers() {
- delete this.defaultHtmlHandlers;
- return this.defaultHtmlHandlers = this._getHandlers("http://www.example.com/index.html", {
- filterBrowsers: false,
- filterHtml: false
- });
- },
-
- _getHandlers: function(url, options) {
- let values = {};
-
- let handlers = this.getAppsForUri(Services.io.newURI(url, null, null), options);
- handlers.forEach(function(app) {
- values[app.name] = app;
- }, this);
-
- return values;
- },
-
- get protoSvc() {
- delete this.protoSvc;
- return this.protoSvc = Cc["@mozilla.org/uriloader/external-protocol-service;1"].getService(Ci.nsIExternalProtocolService);
- },
-
- get urlHandlerService() {
- delete this.urlHandlerService;
- return this.urlHandlerService = Cc["@mozilla.org/uriloader/external-url-handler-service;1"].getService(Ci.nsIExternalURLHandlerService);
- },
-
- prompt: function showPicker(apps, promptOptions, callback) {
- let p = new Prompt(promptOptions).addIconGrid({ items: apps });
- p.show(callback);
- },
-
- getAppsForProtocol: function getAppsForProtocol(scheme) {
- let protoHandlers = this.protoSvc.getProtocolHandlerInfoFromOS(scheme, {}).possibleApplicationHandlers;
-
- let results = {};
- for (let i = 0; i < protoHandlers.length; i++) {
- try {
- let protoApp = protoHandlers.queryElementAt(i, Ci.nsIHandlerApp);
- results[protoApp.name] = new App({
- name: protoApp.name,
- description: protoApp.detailedDescription,
- });
- } catch(e) {}
- }
-
- return results;
- },
-
- getAppsForUri: function getAppsForUri(uri, flags = { }, callback) {
- // Return early for well-known internal schemes
- if (!uri || uri.schemeIs("about") || uri.schemeIs("chrome")) {
- if (callback) {
- callback([]);
- }
- return [];
- }
-
- flags.filterBrowsers = "filterBrowsers" in flags ? flags.filterBrowsers : true;
- flags.filterHtml = "filterHtml" in flags ? flags.filterHtml : true;
-
- // Query for apps that can/can't handle the mimetype
- let msg = this._getMessage("Intent:GetHandlers", uri, flags);
- let parseData = (d) => {
- let apps = []
- if (!d) {
- return apps;
- }
-
- apps = this._parseApps(d.apps);
-
- if (flags.filterBrowsers) {
- apps = apps.filter((app) => {
- return app.name && !this.defaultBrowsers[app.name];
- });
- }
-
- // Some apps will register for html files (the Samsung Video player) but should be shown
- // for non-HTML files (like videos). This filters them only if the page has an htm of html
- // file extension.
- if (flags.filterHtml) {
- // Matches from the first '.' to the end of the string, '?', or '#'
- let ext = /\.([^\?#]*)/.exec(uri.path);
- if (ext && (ext[1] === "html" || ext[1] === "htm")) {
- apps = apps.filter(function(app) {
- return app.name && !this.defaultHtmlHandlers[app.name];
- }, this);
- }
- }
-
- return apps;
- };
-
- if (!callback) {
- let data = this._sendMessageSync(msg);
- return parseData(data);
- } else {
- Messaging.sendRequestForResult(msg).then(function(data) {
- callback(parseData(data));
- });
- }
- },
-
- launchUri: function launchUri(uri) {
- let msg = this._getMessage("Intent:Open", uri);
- Messaging.sendRequest(msg);
- },
-
- _parseApps: function _parseApps(appInfo) {
- // appInfo -> {apps: [app1Label, app1Default, app1PackageName, app1ActivityName, app2Label, app2Defaut, ...]}
- // see GeckoAppShell.java getHandlersForIntent function for details
- const numAttr = 4; // 4 elements per ResolveInfo: label, default, package name, activity name.
-
- let apps = [];
- for (let i = 0; i < appInfo.length; i += numAttr) {
- apps.push(new App({"name" : appInfo[i],
- "isDefault" : appInfo[i+1],
- "packageName" : appInfo[i+2],
- "activityName" : appInfo[i+3]}));
- }
-
- return apps;
- },
-
- _getMessage: function(type, uri, options = {}) {
- let mimeType = options.mimeType;
- if (uri && mimeType == undefined) {
- mimeType = ContentAreaUtils.getMIMETypeForURI(uri) || "";
- }
-
- return {
- type: type,
- mime: mimeType,
- action: options.action || "", // empty action string defaults to android.intent.action.VIEW
- url: uri ? uri.spec : "",
- packageName: options.packageName || "",
- className: options.className || ""
- };
- },
-
- _launchApp: function launchApp(app, uri, callback) {
- if (callback) {
- let msg = this._getMessage("Intent:OpenForResult", uri, {
- packageName: app.packageName,
- className: app.activityName
- });
-
- Messaging.sendRequestForResult(msg).then(callback);
- } else {
- let msg = this._getMessage("Intent:Open", uri, {
- packageName: app.packageName,
- className: app.activityName
- });
-
- Messaging.sendRequest(msg);
- }
- },
-
- _sendMessageSync: function(msg) {
- let res = null;
- Messaging.sendRequestForResult(msg).then(function(data) {
- res = data;
- });
-
- let thread = Services.tm.currentThread;
- while (res == null) {
- thread.processNextEvent(true);
- }
-
- return res;
- },
-};
diff --git a/mobile/android/modules/Home.jsm b/mobile/android/modules/Home.jsm
deleted file mode 100644
index e77d35dbd..000000000
--- a/mobile/android/modules/Home.jsm
+++ /dev/null
@@ -1,487 +0,0 @@
-// -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["Home"];
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/SharedPreferences.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-
-// Keep this in sync with the constant defined in PanelAuthCache.java
-const PREFS_PANEL_AUTH_PREFIX = "home_panels_auth_";
-
-// Default weight for a banner message.
-const DEFAULT_WEIGHT = 100;
-
-// See bug 915424
-function resolveGeckoURI(aURI) {
- if (!aURI)
- throw "Can't resolve an empty uri";
-
- if (aURI.startsWith("chrome://")) {
- let registry = Cc['@mozilla.org/chrome/chrome-registry;1'].getService(Ci["nsIChromeRegistry"]);
- return registry.convertChromeURL(Services.io.newURI(aURI, null, null)).spec;
- } else if (aURI.startsWith("resource://")) {
- let handler = Services.io.getProtocolHandler("resource").QueryInterface(Ci.nsIResProtocolHandler);
- return handler.resolveURI(Services.io.newURI(aURI, null, null));
- }
- return aURI;
-}
-
-function BannerMessage(options) {
- let uuidgen = Cc["@mozilla.org/uuid-generator;1"].getService(Ci.nsIUUIDGenerator);
- this.id = uuidgen.generateUUID().toString();
-
- if ("text" in options && options.text != null)
- this.text = options.text;
-
- if ("icon" in options && options.icon != null)
- this.iconURI = resolveGeckoURI(options.icon);
-
- if ("onshown" in options && typeof options.onshown === "function")
- this.onshown = options.onshown;
-
- if ("onclick" in options && typeof options.onclick === "function")
- this.onclick = options.onclick;
-
- if ("ondismiss" in options && typeof options.ondismiss === "function")
- this.ondismiss = options.ondismiss;
-
- let weight = parseInt(options.weight, 10);
- this.weight = weight > 0 ? weight : DEFAULT_WEIGHT;
-}
-
-// We need this object to have access to the HomeBanner
-// private members without leaking it outside Home.jsm.
-var HomeBannerMessageHandlers;
-
-var HomeBanner = (function () {
- // Whether there is a "HomeBanner:Get" request we couldn't fulfill.
- let _pendingRequest = false;
-
- // Functions used to handle messages sent from Java.
- HomeBannerMessageHandlers = {
- "HomeBanner:Get": function handleBannerGet(data) {
- if (Object.keys(_messages).length > 0) {
- _sendBannerData();
- } else {
- _pendingRequest = true;
- }
- }
- };
-
- // Holds the messages that will rotate through the banner.
- let _messages = {};
-
- // Choose a random message from the set of messages, biasing towards those with higher weight.
- // Weight logic copied from desktop snippets:
- // https://github.com/mozilla/snippets-service/blob/7d80edb8b1cddaed075275c2fc7cdf69a10f4003/snippets/base/templates/base/includes/snippet_js.html#L119
- let _sendBannerData = function() {
- let totalWeight = 0;
- for (let key in _messages) {
- let message = _messages[key];
- totalWeight += message.weight;
- message.totalWeight = totalWeight;
- }
-
- let threshold = Math.random() * totalWeight;
- for (let key in _messages) {
- let message = _messages[key];
- if (threshold < message.totalWeight) {
- Messaging.sendRequest({
- type: "HomeBanner:Data",
- id: message.id,
- text: message.text,
- iconURI: message.iconURI
- });
- return;
- }
- }
- };
-
- let _handleShown = function(id) {
- let message = _messages[id];
- if (message.onshown)
- message.onshown();
- };
-
- let _handleClick = function(id) {
- let message = _messages[id];
- if (message.onclick)
- message.onclick();
- };
-
- let _handleDismiss = function(id) {
- let message = _messages[id];
- if (message.ondismiss)
- message.ondismiss();
- };
-
- return Object.freeze({
- observe: function(subject, topic, data) {
- switch(topic) {
- case "HomeBanner:Shown":
- _handleShown(data);
- break;
-
- case "HomeBanner:Click":
- _handleClick(data);
- break;
-
- case "HomeBanner:Dismiss":
- _handleDismiss(data);
- break;
- }
- },
-
- /**
- * Adds a new banner message to the rotation.
- *
- * @return id Unique identifer for the message.
- */
- add: function(options) {
- let message = new BannerMessage(options);
- _messages[message.id] = message;
-
- // If this is the first message we're adding, add
- // observers to listen for requests from the Java UI.
- if (Object.keys(_messages).length == 1) {
- Services.obs.addObserver(this, "HomeBanner:Shown", false);
- Services.obs.addObserver(this, "HomeBanner:Click", false);
- Services.obs.addObserver(this, "HomeBanner:Dismiss", false);
-
- // Send a message to Java if there's a pending "HomeBanner:Get" request.
- if (_pendingRequest) {
- _pendingRequest = false;
- _sendBannerData();
- }
- }
-
- return message.id;
- },
-
- /**
- * Removes a banner message from the rotation.
- *
- * @param id The id of the message to remove.
- */
- remove: function(id) {
- if (!(id in _messages)) {
- throw "Home.banner: Can't remove message that doesn't exist: id = " + id;
- }
-
- delete _messages[id];
-
- // If there are no more messages, remove the observers.
- if (Object.keys(_messages).length == 0) {
- Services.obs.removeObserver(this, "HomeBanner:Shown");
- Services.obs.removeObserver(this, "HomeBanner:Click");
- Services.obs.removeObserver(this, "HomeBanner:Dismiss");
- }
- }
- });
-})();
-
-// We need this object to have access to the HomePanels
-// private members without leaking it outside Home.jsm.
-var HomePanelsMessageHandlers;
-
-var HomePanels = (function () {
- // Functions used to handle messages sent from Java.
- HomePanelsMessageHandlers = {
-
- "HomePanels:Get": function handlePanelsGet(data) {
- data = JSON.parse(data);
-
- let requestId = data.requestId;
- let ids = data.ids || null;
-
- let panels = [];
- for (let id in _registeredPanels) {
- // Null ids means we want to fetch all available panels
- if (ids == null || ids.indexOf(id) >= 0) {
- try {
- panels.push(_generatePanel(id));
- } catch(e) {
- Cu.reportError("Home.panels: Invalid options, panel.id = " + id + ": " + e);
- }
- }
- }
-
- Messaging.sendRequest({
- type: "HomePanels:Data",
- panels: panels,
- requestId: requestId
- });
- },
-
- "HomePanels:Authenticate": function handlePanelsAuthenticate(id) {
- // Generate panel options to get auth handler.
- let options = _registeredPanels[id]();
- if (!options.auth) {
- throw "Home.panels: Invalid auth for panel.id = " + id;
- }
- if (!options.auth.authenticate || typeof options.auth.authenticate !== "function") {
- throw "Home.panels: Invalid auth authenticate function: panel.id = " + this.id;
- }
- options.auth.authenticate();
- },
-
- "HomePanels:RefreshView": function handlePanelsRefreshView(data) {
- data = JSON.parse(data);
-
- let options = _registeredPanels[data.panelId]();
- let view = options.views[data.viewIndex];
-
- if (!view) {
- throw "Home.panels: Invalid view for panel.id = " + data.panelId
- + ", view.index = " + data.viewIndex;
- }
-
- if (!view.onrefresh || typeof view.onrefresh !== "function") {
- throw "Home.panels: Invalid onrefresh for panel.id = " + data.panelId
- + ", view.index = " + data.viewIndex;
- }
-
- view.onrefresh();
- },
-
- "HomePanels:Installed": function handlePanelsInstalled(id) {
- _assertPanelExists(id);
-
- let options = _registeredPanels[id]();
- if (!options.oninstall) {
- return;
- }
- if (typeof options.oninstall !== "function") {
- throw "Home.panels: Invalid oninstall function: panel.id = " + this.id;
- }
- options.oninstall();
- },
-
- "HomePanels:Uninstalled": function handlePanelsUninstalled(id) {
- _assertPanelExists(id);
-
- let options = _registeredPanels[id]();
- if (!options.onuninstall) {
- return;
- }
- if (typeof options.onuninstall !== "function") {
- throw "Home.panels: Invalid onuninstall function: panel.id = " + this.id;
- }
- options.onuninstall();
- }
- };
-
- // Holds the current set of registered panels that can be
- // installed, updated, uninstalled, or unregistered. It maps
- // panel ids with the functions that dynamically generate
- // their respective panel options. This is used to retrieve
- // the current list of available panels in the system.
- // See HomePanels:Get handler.
- let _registeredPanels = {};
-
- // Valid layouts for a panel.
- let Layout = Object.freeze({
- FRAME: "frame"
- });
-
- // Valid types of views for a dataset.
- let View = Object.freeze({
- LIST: "list",
- GRID: "grid"
- });
-
- // Valid item types for a panel view.
- let Item = Object.freeze({
- ARTICLE: "article",
- IMAGE: "image",
- ICON: "icon"
- });
-
- // Valid item handlers for a panel view.
- let ItemHandler = Object.freeze({
- BROWSER: "browser",
- INTENT: "intent"
- });
-
- function Panel(id, options) {
- this.id = id;
- this.title = options.title;
- this.layout = options.layout;
- this.views = options.views;
- this.default = !!options.default;
-
- if (!this.id || !this.title) {
- throw "Home.panels: Can't create a home panel without an id and title!";
- }
-
- if (!this.layout) {
- // Use FRAME layout by default
- this.layout = Layout.FRAME;
- } else if (!_valueExists(Layout, this.layout)) {
- throw "Home.panels: Invalid layout for panel: panel.id = " + this.id + ", panel.layout =" + this.layout;
- }
-
- for (let view of this.views) {
- if (!_valueExists(View, view.type)) {
- throw "Home.panels: Invalid view type: panel.id = " + this.id + ", view.type = " + view.type;
- }
-
- if (!view.itemType) {
- if (view.type == View.LIST) {
- // Use ARTICLE item type by default in LIST views
- view.itemType = Item.ARTICLE;
- } else if (view.type == View.GRID) {
- // Use IMAGE item type by default in GRID views
- view.itemType = Item.IMAGE;
- }
- } else if (!_valueExists(Item, view.itemType)) {
- throw "Home.panels: Invalid item type: panel.id = " + this.id + ", view.itemType = " + view.itemType;
- }
-
- if (!view.itemHandler) {
- // Use BROWSER item handler by default
- view.itemHandler = ItemHandler.BROWSER;
- } else if (!_valueExists(ItemHandler, view.itemHandler)) {
- throw "Home.panels: Invalid item handler: panel.id = " + this.id + ", view.itemHandler = " + view.itemHandler;
- }
-
- if (!view.dataset) {
- throw "Home.panels: No dataset provided for view: panel.id = " + this.id + ", view.type = " + view.type;
- }
-
- if (view.onrefresh) {
- view.refreshEnabled = true;
- }
- }
-
- if (options.auth) {
- if (!options.auth.messageText) {
- throw "Home.panels: Invalid auth messageText: panel.id = " + this.id;
- }
- if (!options.auth.buttonText) {
- throw "Home.panels: Invalid auth buttonText: panel.id = " + this.id;
- }
-
- this.authConfig = {
- messageText: options.auth.messageText,
- buttonText: options.auth.buttonText
- };
-
- // Include optional image URL if it is specified.
- if (options.auth.imageUrl) {
- this.authConfig.imageUrl = options.auth.imageUrl;
- }
- }
-
- if (options.position >= 0) {
- this.position = options.position;
- }
- }
-
- let _generatePanel = function(id) {
- let options = _registeredPanels[id]();
- return new Panel(id, options);
- };
-
- // Helper function used to see if a value is in an object.
- let _valueExists = function(obj, value) {
- for (let key in obj) {
- if (obj[key] == value) {
- return true;
- }
- }
- return false;
- };
-
- let _assertPanelExists = function(id) {
- if (!(id in _registeredPanels)) {
- throw "Home.panels: Panel doesn't exist: id = " + id;
- }
- };
-
- return Object.freeze({
- Layout: Layout,
- View: View,
- Item: Item,
- ItemHandler: ItemHandler,
-
- register: function(id, optionsCallback) {
- // Bail if the panel already exists
- if (id in _registeredPanels) {
- throw "Home.panels: Panel already exists: id = " + id;
- }
-
- if (!optionsCallback || typeof optionsCallback !== "function") {
- throw "Home.panels: Panel callback must be a function: id = " + id;
- }
-
- _registeredPanels[id] = optionsCallback;
- },
-
- unregister: function(id) {
- _assertPanelExists(id);
-
- delete _registeredPanels[id];
- },
-
- install: function(id) {
- _assertPanelExists(id);
-
- Messaging.sendRequest({
- type: "HomePanels:Install",
- panel: _generatePanel(id)
- });
- },
-
- uninstall: function(id) {
- _assertPanelExists(id);
-
- Messaging.sendRequest({
- type: "HomePanels:Uninstall",
- id: id
- });
- },
-
- update: function(id) {
- _assertPanelExists(id);
-
- Messaging.sendRequest({
- type: "HomePanels:Update",
- panel: _generatePanel(id)
- });
- },
-
- setAuthenticated: function(id, isAuthenticated) {
- _assertPanelExists(id);
-
- let authKey = PREFS_PANEL_AUTH_PREFIX + id;
- let sharedPrefs = SharedPreferences.forProfile();
- sharedPrefs.setBoolPref(authKey, isAuthenticated);
- }
- });
-})();
-
-// Public API
-this.Home = Object.freeze({
- banner: HomeBanner,
- panels: HomePanels,
-
- // Lazy notification observer registered in browser.js
- observe: function(subject, topic, data) {
- if (topic in HomeBannerMessageHandlers) {
- HomeBannerMessageHandlers[topic](data);
- } else if (topic in HomePanelsMessageHandlers) {
- HomePanelsMessageHandlers[topic](data);
- } else {
- Cu.reportError("Home.observe: message handler not found for topic: " + topic);
- }
- }
-});
diff --git a/mobile/android/modules/HomeProvider.jsm b/mobile/android/modules/HomeProvider.jsm
deleted file mode 100644
index bca8fa526..000000000
--- a/mobile/android/modules/HomeProvider.jsm
+++ /dev/null
@@ -1,407 +0,0 @@
-// -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
-/* 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";
-
-this.EXPORTED_SYMBOLS = [ "HomeProvider" ];
-
-const { utils: Cu, classes: Cc, interfaces: Ci } = Components;
-
-Cu.import("resource://gre/modules/Messaging.jsm");
-Cu.import("resource://gre/modules/osfile.jsm");
-Cu.import("resource://gre/modules/Promise.jsm");
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/Sqlite.jsm");
-Cu.import("resource://gre/modules/Task.jsm");
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-/*
- * SCHEMA_VERSION history:
- * 1: Create HomeProvider (bug 942288)
- * 2: Add filter column to items table (bug 942295/975841)
- * 3: Add background_color and background_url columns (bug 1157539)
- */
-const SCHEMA_VERSION = 3;
-
-// The maximum number of items you can attempt to save at once.
-const MAX_SAVE_COUNT = 100;
-
-XPCOMUtils.defineLazyGetter(this, "DB_PATH", function() {
- return OS.Path.join(OS.Constants.Path.profileDir, "home.sqlite");
-});
-
-const PREF_STORAGE_LAST_SYNC_TIME_PREFIX = "home.storage.lastSyncTime.";
-const PREF_SYNC_UPDATE_MODE = "home.sync.updateMode";
-const PREF_SYNC_CHECK_INTERVAL_SECS = "home.sync.checkIntervalSecs";
-
-XPCOMUtils.defineLazyGetter(this, "gSyncCheckIntervalSecs", function() {
- return Services.prefs.getIntPref(PREF_SYNC_CHECK_INTERVAL_SECS);
-});
-
-XPCOMUtils.defineLazyServiceGetter(this,
- "gUpdateTimerManager", "@mozilla.org/updates/timer-manager;1", "nsIUpdateTimerManager");
-
-/**
- * All SQL statements should be defined here.
- */
-const SQL = {
- createItemsTable:
- "CREATE TABLE items (" +
- "_id INTEGER PRIMARY KEY AUTOINCREMENT, " +
- "dataset_id TEXT NOT NULL, " +
- "url TEXT," +
- "title TEXT," +
- "description TEXT," +
- "image_url TEXT," +
- "background_color TEXT," +
- "background_url TEXT," +
- "filter TEXT," +
- "created INTEGER" +
- ")",
-
- dropItemsTable:
- "DROP TABLE items",
-
- insertItem:
- "INSERT INTO items (dataset_id, url, title, description, image_url, background_color, background_url, filter, created) " +
- "VALUES (:dataset_id, :url, :title, :description, :image_url, :background_color, :background_url, :filter, :created)",
-
- deleteFromDataset:
- "DELETE FROM items WHERE dataset_id = :dataset_id",
-
- addColumnBackgroundColor:
- "ALTER TABLE items ADD COLUMN background_color TEXT",
-
- addColumnBackgroundUrl:
- "ALTER TABLE items ADD COLUMN background_url TEXT",
-}
-
-/**
- * Technically this function checks to see if the user is on a local network,
- * but we express this as "wifi" to the user.
- */
-function isUsingWifi() {
- let network = Cc["@mozilla.org/network/network-link-service;1"].getService(Ci.nsINetworkLinkService);
- return (network.linkType === Ci.nsINetworkLinkService.LINK_TYPE_WIFI || network.linkType === Ci.nsINetworkLinkService.LINK_TYPE_ETHERNET);
-}
-
-function getNowInSeconds() {
- return Math.round(Date.now() / 1000);
-}
-
-function getLastSyncPrefName(datasetId) {
- return PREF_STORAGE_LAST_SYNC_TIME_PREFIX + datasetId;
-}
-
-// Whether or not we've registered an update timer.
-var gTimerRegistered = false;
-
-// Map of datasetId -> { interval: <integer>, callback: <function> }
-var gSyncCallbacks = {};
-
-/**
- * nsITimerCallback implementation. Checks to see if it's time to sync any registered datasets.
- *
- * @param timer The timer which has expired.
- */
-function syncTimerCallback(timer) {
- for (let datasetId in gSyncCallbacks) {
- let lastSyncTime = 0;
- try {
- lastSyncTime = Services.prefs.getIntPref(getLastSyncPrefName(datasetId));
- } catch(e) { }
-
- let now = getNowInSeconds();
- let { interval: interval, callback: callback } = gSyncCallbacks[datasetId];
-
- if (lastSyncTime < now - interval) {
- let success = HomeProvider.requestSync(datasetId, callback);
- if (success) {
- Services.prefs.setIntPref(getLastSyncPrefName(datasetId), now);
- }
- }
- }
-}
-
-this.HomeStorage = function(datasetId) {
- this.datasetId = datasetId;
-};
-
-this.ValidationError = function(message) {
- this.name = "ValidationError";
- this.message = message;
-};
-ValidationError.prototype = new Error();
-ValidationError.prototype.constructor = ValidationError;
-
-this.HomeProvider = Object.freeze({
- ValidationError: ValidationError,
-
- /**
- * Returns a storage associated with a given dataset identifer.
- *
- * @param datasetId
- * (string) Unique identifier for the dataset.
- *
- * @return HomeStorage
- */
- getStorage: function(datasetId) {
- return new HomeStorage(datasetId);
- },
-
- /**
- * Checks to see if it's an appropriate time to sync.
- *
- * @param datasetId Unique identifier for the dataset to sync.
- * @param callback Function to call when it's time to sync, called with datasetId as a parameter.
- *
- * @return boolean Whether or not we were able to sync.
- */
- requestSync: function(datasetId, callback) {
- // Make sure it's a good time to sync.
- if ((Services.prefs.getIntPref(PREF_SYNC_UPDATE_MODE) === 1) && !isUsingWifi()) {
- Cu.reportError("HomeProvider: Failed to sync because device is not on a local network");
- return false;
- }
-
- callback(datasetId);
- return true;
- },
-
- /**
- * Specifies that a sync should be requested for the given dataset and update interval.
- *
- * @param datasetId Unique identifier for the dataset to sync.
- * @param interval Update interval in seconds. By default, this is throttled to 3600 seconds (1 hour).
- * @param callback Function to call when it's time to sync, called with datasetId as a parameter.
- */
- addPeriodicSync: function(datasetId, interval, callback) {
- // Warn developers if they're expecting more frequent notifications that we allow.
- if (interval < gSyncCheckIntervalSecs) {
- Cu.reportError("HomeProvider: Warning for dataset " + datasetId +
- " : Sync notifications are throttled to " + gSyncCheckIntervalSecs + " seconds");
- }
-
- gSyncCallbacks[datasetId] = {
- interval: interval,
- callback: callback
- };
-
- if (!gTimerRegistered) {
- gUpdateTimerManager.registerTimer("home-provider-sync-timer", syncTimerCallback, gSyncCheckIntervalSecs);
- gTimerRegistered = true;
- }
- },
-
- /**
- * Removes a periodic sync timer.
- *
- * @param datasetId Dataset to sync.
- */
- removePeriodicSync: function(datasetId) {
- delete gSyncCallbacks[datasetId];
- Services.prefs.clearUserPref(getLastSyncPrefName(datasetId));
- // You can't unregister a update timer, so we don't try to do that.
- }
-});
-
-var gDatabaseEnsured = false;
-
-/**
- * Creates the database schema.
- */
-function createDatabase(db) {
- return Task.spawn(function create_database_task() {
- yield db.execute(SQL.createItemsTable);
- });
-}
-
-/**
- * Migrates the database schema to a new version.
- */
-function upgradeDatabase(db, oldVersion, newVersion) {
- return Task.spawn(function upgrade_database_task() {
- switch (oldVersion) {
- case 1:
- // Migration from v1 to latest:
- // Recreate the items table discarding any
- // existing data.
- yield db.execute(SQL.dropItemsTable);
- yield db.execute(SQL.createItemsTable);
- break;
-
- case 2:
- // Migration from v2 to latest:
- // Add new columns: background_color, background_url
- yield db.execute(SQL.addColumnBackgroundColor);
- yield db.execute(SQL.addColumnBackgroundUrl);
- break;
- }
- });
-}
-
-/**
- * Opens a database connection and makes sure that the database schema version
- * is correct, performing migrations if necessary. Consumers should be sure
- * to close any database connections they open.
- *
- * @return Promise
- * @resolves Handle on an opened SQLite database.
- */
-function getDatabaseConnection() {
- return Task.spawn(function get_database_connection_task() {
- let db = yield Sqlite.openConnection({ path: DB_PATH });
- if (gDatabaseEnsured) {
- throw new Task.Result(db);
- }
-
- try {
- // Check to see if we need to perform any migrations.
- let dbVersion = parseInt(yield db.getSchemaVersion());
-
- // getSchemaVersion() returns a 0 int if the schema
- // version is undefined.
- if (dbVersion === 0) {
- yield createDatabase(db);
- } else if (dbVersion < SCHEMA_VERSION) {
- yield upgradeDatabase(db, dbVersion, SCHEMA_VERSION);
- }
-
- yield db.setSchemaVersion(SCHEMA_VERSION);
- } catch(e) {
- // Close the DB connection before passing the exception to the consumer.
- yield db.close();
- throw e;
- }
-
- gDatabaseEnsured = true;
- throw new Task.Result(db);
- });
-}
-
-/**
- * Validates an item to be saved to the DB.
- *
- * @param item
- * (object) item object to be validated.
- */
-function validateItem(datasetId, item) {
- if (!item.url) {
- throw new ValidationError('HomeStorage: All rows must have an URL: datasetId = ' +
- datasetId);
- }
-
- if (!item.image_url && !item.title && !item.description) {
- throw new ValidationError('HomeStorage: All rows must have at least an image URL, ' +
- 'or a title or a description: datasetId = ' + datasetId);
- }
-}
-
-var gRefreshTimers = {};
-
-/**
- * Sends a message to Java to refresh the given dataset. Delays sending
- * messages to avoid successive refreshes, which can result in flashing views.
- */
-function refreshDataset(datasetId) {
- // Bail if there's already a refresh timer waiting to fire
- if (gRefreshTimers[datasetId]) {
- return;
- }
-
- let timer = Cc["@mozilla.org/timer;1"].createInstance(Ci.nsITimer);
- timer.initWithCallback(function(timer) {
- delete gRefreshTimers[datasetId];
-
- Messaging.sendRequest({
- type: "HomePanels:RefreshDataset",
- datasetId: datasetId
- });
- }, 100, Ci.nsITimer.TYPE_ONE_SHOT);
-
- gRefreshTimers[datasetId] = timer;
-}
-
-HomeStorage.prototype = {
- /**
- * Saves data rows to the DB.
- *
- * @param data
- * An array of JS objects represnting row items to save.
- * Each object may have the following properties:
- * - url (string)
- * - title (string)
- * - description (string)
- * - image_url (string)
- * - filter (string)
- * @param options
- * A JS object holding additional cofiguration properties.
- * The following properties are currently supported:
- * - replace (boolean): Whether or not to replace existing items.
- *
- * @return Promise
- * @resolves When the operation has completed.
- */
- save: function(data, options) {
- if (data && data.length > MAX_SAVE_COUNT) {
- throw "save failed for dataset = " + this.datasetId +
- ": you cannot save more than " + MAX_SAVE_COUNT + " items at once";
- }
-
- return Task.spawn(function save_task() {
- let db = yield getDatabaseConnection();
- try {
- yield db.executeTransaction(function save_transaction() {
- if (options && options.replace) {
- yield db.executeCached(SQL.deleteFromDataset, { dataset_id: this.datasetId });
- }
-
- // Insert data into DB.
- for (let item of data) {
- validateItem(this.datasetId, item);
-
- // XXX: Directly pass item as params? More validation for item?
- let params = {
- dataset_id: this.datasetId,
- url: item.url,
- title: item.title,
- description: item.description,
- image_url: item.image_url,
- background_color: item.background_color,
- background_url: item.background_url,
- filter: item.filter,
- created: Date.now()
- };
- yield db.executeCached(SQL.insertItem, params);
- }
- }.bind(this));
- } finally {
- yield db.close();
- }
-
- refreshDataset(this.datasetId);
- }.bind(this));
- },
-
- /**
- * Deletes all rows associated with this storage.
- *
- * @return Promise
- * @resolves When the operation has completed.
- */
- deleteAll: function() {
- return Task.spawn(function delete_all_task() {
- let db = yield getDatabaseConnection();
- try {
- let params = { dataset_id: this.datasetId };
- yield db.executeCached(SQL.deleteFromDataset, params);
- } finally {
- yield db.close();
- }
-
- refreshDataset(this.datasetId);
- }.bind(this));
- }
-};
diff --git a/mobile/android/modules/JNI.jsm b/mobile/android/modules/JNI.jsm
deleted file mode 100644
index 1e10b9cfb..000000000
--- a/mobile/android/modules/JNI.jsm
+++ /dev/null
@@ -1,1167 +0,0 @@
-// JavaScript to Java bridge via the Java Native Interface
-// Allows calling into Android SDK from JavaScript in Firefox Add-On.
-// Released into the public domain.
-// C. Scott Ananian <cscott@laptop.org> (http://cscott.net)
-
-// NOTE: All changes to this file should first be pushed to the repo at:
-// https://github.com/cscott/skeleton-addon-fxandroid/tree/jni
-
-var EXPORTED_SYMBOLS = ["JNI","android_log"];
-
-Components.utils.import("resource://gre/modules/ctypes.jsm")
-
-var liblog = ctypes.open('liblog.so');
-var android_log = liblog.declare("__android_log_write",
- ctypes.default_abi,
- ctypes.int32_t,
- ctypes.int32_t,
- ctypes.char.ptr,
- ctypes.char.ptr);
-
-var libxul = ctypes.open('libxul.so');
-
-var jenvptr = ctypes.voidptr_t;
-var jclass = ctypes.voidptr_t;
-var jobject = ctypes.voidptr_t;
-var jvalue = ctypes.voidptr_t;
-var jmethodid = ctypes.voidptr_t;
-var jfieldid = ctypes.voidptr_t;
-
-var jboolean = ctypes.uint8_t;
-var jbyte = ctypes.int8_t;
-var jchar = ctypes.uint16_t;
-var jshort = ctypes.int16_t;
-var jint = ctypes.int32_t;
-var jlong = ctypes.int64_t;
-var jfloat = ctypes.float32_t;
-var jdouble = ctypes.float64_t;
-
-var jsize = jint;
-var jstring = jobject;
-var jarray = jobject;
-var jthrowable = jobject;
-
-var JNINativeInterface = new ctypes.StructType(
- "JNINativeInterface",
- [{reserved0: ctypes.voidptr_t},
- {reserved1: ctypes.voidptr_t},
- {reserved2: ctypes.voidptr_t},
- {reserved3: ctypes.voidptr_t},
- {GetVersion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.int32_t,
- [ctypes.voidptr_t]).ptr},
- {DefineClass: new ctypes.FunctionType(ctypes.default_abi,
- jclass,
- [jenvptr, ctypes.char.ptr, jobject,
- jbyte.array(), jsize]).ptr},
- {FindClass: new ctypes.FunctionType(ctypes.default_abi,
- jclass,
- [jenvptr,
- ctypes.char.ptr]).ptr},
- {FromReflectedMethod: new ctypes.FunctionType(ctypes.default_abi,
- jmethodid,
- [jenvptr, jobject]).ptr},
- {FromReflectedField: new ctypes.FunctionType(ctypes.default_abi,
- jfieldid,
- [jenvptr, jobject]).ptr},
- {ToReflectedMethod: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jclass,
- jmethodid]).ptr},
- {GetSuperclass: new ctypes.FunctionType(ctypes.default_abi,
- jclass, [jenvptr, jclass]).ptr},
- {IsAssignableFrom: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jclass, jclass]).ptr},
- {ToReflectedField: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {Throw: new ctypes.FunctionType(ctypes.default_abi,
- jint, [jenvptr, jthrowable]).ptr},
- {ThrowNew: new ctypes.FunctionType(ctypes.default_abi,
- jint, [jenvptr, jclass,
- ctypes.char.ptr]).ptr},
- {ExceptionOccurred: new ctypes.FunctionType(ctypes.default_abi,
- jthrowable, [jenvptr]).ptr},
- {ExceptionDescribe: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t, [jenvptr]).ptr},
- {ExceptionClear: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t, [jenvptr]).ptr},
- {FatalError: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr,
- ctypes.char.ptr]).ptr},
- {PushLocalFrame: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr, jint]).ptr},
- {PopLocalFrame: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jobject]).ptr},
- {NewGlobalRef: new ctypes.FunctionType(ctypes.default_abi,
- jobject, [jenvptr, jobject]).ptr},
- {DeleteGlobalRef: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr,
- jobject]).ptr},
- {DeleteLocalRef: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr,
- jobject]).ptr},
- {IsSameObject: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jobject, jobject]).ptr},
- {NewLocalRef: new ctypes.FunctionType(ctypes.default_abi,
- jobject, [jenvptr, jobject]).ptr},
- {EnsureLocalCapacity: new ctypes.FunctionType(ctypes.default_abi,
- jint, [jenvptr, jint]).ptr},
- {AllocObject: new ctypes.FunctionType(ctypes.default_abi,
- jobject, [jenvptr, jclass]).ptr},
- {NewObject: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr,
- jclass,
- jmethodid,
- "..."]).ptr},
- {NewObjectV: ctypes.voidptr_t},
- {NewObjectA: ctypes.voidptr_t},
- {GetObjectClass: new ctypes.FunctionType(ctypes.default_abi,
- jclass,
- [jenvptr, jobject]).ptr},
- {IsInstanceOf: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jobject, jclass]).ptr},
- {GetMethodID: new ctypes.FunctionType(ctypes.default_abi,
- jmethodid,
- [jenvptr,
- jclass,
- ctypes.char.ptr,
- ctypes.char.ptr]).ptr},
- {CallObjectMethod: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jobject, jmethodid,
- "..."]).ptr},
- {CallObjectMethodV: ctypes.voidptr_t},
- {CallObjectMethodA: ctypes.voidptr_t},
- {CallBooleanMethod: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallBooleanMethodV: ctypes.voidptr_t},
- {CallBooleanMethodA: ctypes.voidptr_t},
- {CallByteMethod: new ctypes.FunctionType(ctypes.default_abi,
- jbyte,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallByteMethodV: ctypes.voidptr_t},
- {CallByteMethodA: ctypes.voidptr_t},
- {CallCharMethod: new ctypes.FunctionType(ctypes.default_abi,
- jchar,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallCharMethodV: ctypes.voidptr_t},
- {CallCharMethodA: ctypes.voidptr_t},
- {CallShortMethod: new ctypes.FunctionType(ctypes.default_abi,
- jshort,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallShortMethodV: ctypes.voidptr_t},
- {CallShortMethodA: ctypes.voidptr_t},
- {CallIntMethod: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallIntMethodV: ctypes.voidptr_t},
- {CallIntMethodA: ctypes.voidptr_t},
- {CallLongMethod: new ctypes.FunctionType(ctypes.default_abi,
- jlong,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallLongMethodV: ctypes.voidptr_t},
- {CallLongMethodA: ctypes.voidptr_t},
- {CallFloatMethod: new ctypes.FunctionType(ctypes.default_abi,
- jfloat,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallFloatMethodV: ctypes.voidptr_t},
- {CallFloatMethodA: ctypes.voidptr_t},
- {CallDoubleMethod: new ctypes.FunctionType(ctypes.default_abi,
- jdouble,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallDoubleMethodV: ctypes.voidptr_t},
- {CallDoubleMethodA: ctypes.voidptr_t},
- {CallVoidMethod: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr,
- jobject,
- jmethodid,
- "..."]).ptr},
- {CallVoidMethodV: ctypes.voidptr_t},
- {CallVoidMethodA: ctypes.voidptr_t},
- {CallNonvirtualObjectMethod: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualObjectMethodV: ctypes.voidptr_t},
- {CallNonvirtualObjectMethodA: ctypes.voidptr_t},
- {CallNonvirtualBooleanMethod: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualBooleanMethodV: ctypes.voidptr_t},
- {CallNonvirtualBooleanMethodA: ctypes.voidptr_t},
- {CallNonvirtualByteMethod: new ctypes.FunctionType(ctypes.default_abi,
- jbyte,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualByteMethodV: ctypes.voidptr_t},
- {CallNonvirtualByteMethodA: ctypes.voidptr_t},
- {CallNonvirtualCharMethod: new ctypes.FunctionType(ctypes.default_abi,
- jchar,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualCharMethodV: ctypes.voidptr_t},
- {CallNonvirtualCharMethodA: ctypes.voidptr_t},
- {CallNonvirtualShortMethod: new ctypes.FunctionType(ctypes.default_abi,
- jshort,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualShortMethodV: ctypes.voidptr_t},
- {CallNonvirtualShortMethodA: ctypes.voidptr_t},
- {CallNonvirtualIntMethod: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualIntMethodV: ctypes.voidptr_t},
- {CallNonvirtualIntMethodA: ctypes.voidptr_t},
- {CallNonvirtualLongMethod: new ctypes.FunctionType(ctypes.default_abi,
- jlong,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualLongMethodV: ctypes.voidptr_t},
- {CallNonvirtualLongMethodA: ctypes.voidptr_t},
- {CallNonvirtualFloatMethod: new ctypes.FunctionType(ctypes.default_abi,
- jfloat,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualFloatMethodV: ctypes.voidptr_t},
- {CallNonvirtualFloatMethodA: ctypes.voidptr_t},
- {CallNonvirtualDoubleMethod: new ctypes.FunctionType(ctypes.default_abi,
- jdouble,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualDoubleMethodV: ctypes.voidptr_t},
- {CallNonvirtualDoubleMethodA: ctypes.voidptr_t},
- {CallNonvirtualVoidMethod: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jclass, jmethodid,
- "..."]).ptr},
- {CallNonvirtualVoidMethodV: ctypes.voidptr_t},
- {CallNonvirtualVoidMethodA: ctypes.voidptr_t},
- {GetFieldID: new ctypes.FunctionType(ctypes.default_abi,
- jfieldid,
- [jenvptr, jclass,
- ctypes.char.ptr,
- ctypes.char.ptr]).ptr},
- {GetObjectField: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetBooleanField: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetByteField: new ctypes.FunctionType(ctypes.default_abi,
- jbyte,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetCharField: new ctypes.FunctionType(ctypes.default_abi,
- jchar,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetShortField: new ctypes.FunctionType(ctypes.default_abi,
- jshort,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetIntField: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetLongField: new ctypes.FunctionType(ctypes.default_abi,
- jlong,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetFloatField: new ctypes.FunctionType(ctypes.default_abi,
- jfloat,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {GetDoubleField: new ctypes.FunctionType(ctypes.default_abi,
- jdouble,
- [jenvptr, jobject,
- jfieldid]).ptr},
- {SetObjectField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jobject]).ptr},
- {SetBooleanField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jboolean]).ptr},
- {SetByteField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jbyte]).ptr},
- {SetCharField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jchar]).ptr},
- {SetShortField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jshort]).ptr},
- {SetIntField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jint]).ptr},
- {SetLongField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jlong]).ptr},
- {SetFloatField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jfloat]).ptr},
- {SetDoubleField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jobject,
- jfieldid, jdouble]).ptr},
- {GetStaticMethodID: new ctypes.FunctionType(ctypes.default_abi,
- jmethodid,
- [jenvptr,
- jclass,
- ctypes.char.ptr,
- ctypes.char.ptr]).ptr},
- {CallStaticObjectMethod: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticObjectMethodV: ctypes.voidptr_t},
- {CallStaticObjectMethodA: ctypes.voidptr_t},
- {CallStaticBooleanMethod: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticBooleanMethodV: ctypes.voidptr_t},
- {CallStaticBooleanMethodA: ctypes.voidptr_t},
- {CallStaticByteMethod: new ctypes.FunctionType(ctypes.default_abi,
- jbyte,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticByteMethodV: ctypes.voidptr_t},
- {CallStaticByteMethodA: ctypes.voidptr_t},
- {CallStaticCharMethod: new ctypes.FunctionType(ctypes.default_abi,
- jchar,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticCharMethodV: ctypes.voidptr_t},
- {CallStaticCharMethodA: ctypes.voidptr_t},
- {CallStaticShortMethod: new ctypes.FunctionType(ctypes.default_abi,
- jshort,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticShortMethodV: ctypes.voidptr_t},
- {CallStaticShortMethodA: ctypes.voidptr_t},
- {CallStaticIntMethod: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticIntMethodV: ctypes.voidptr_t},
- {CallStaticIntMethodA: ctypes.voidptr_t},
- {CallStaticLongMethod: new ctypes.FunctionType(ctypes.default_abi,
- jlong,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticLongMethodV: ctypes.voidptr_t},
- {CallStaticLongMethodA: ctypes.voidptr_t},
- {CallStaticFloatMethod: new ctypes.FunctionType(ctypes.default_abi,
- jfloat,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticFloatMethodV: ctypes.voidptr_t},
- {CallStaticFloatMethodA: ctypes.voidptr_t},
- {CallStaticDoubleMethod: new ctypes.FunctionType(ctypes.default_abi,
- jdouble,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticDoubleMethodV: ctypes.voidptr_t},
- {CallStaticDoubleMethodA: ctypes.voidptr_t},
- {CallStaticVoidMethod: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jmethodid,
- "..."]).ptr},
- {CallStaticVoidMethodV: ctypes.voidptr_t},
- {CallStaticVoidMethodA: ctypes.voidptr_t},
- {GetStaticFieldID: new ctypes.FunctionType(ctypes.default_abi,
- jfieldid,
- [jenvptr, jclass,
- ctypes.char.ptr,
- ctypes.char.ptr]).ptr},
- {GetStaticObjectField: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticBooleanField: new ctypes.FunctionType(ctypes.default_abi,
- jboolean,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticByteField: new ctypes.FunctionType(ctypes.default_abi,
- jbyte,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticCharField: new ctypes.FunctionType(ctypes.default_abi,
- jchar,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticShortField: new ctypes.FunctionType(ctypes.default_abi,
- jshort,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticIntField: new ctypes.FunctionType(ctypes.default_abi,
- jint,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticLongField: new ctypes.FunctionType(ctypes.default_abi,
- jlong,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticFloatField: new ctypes.FunctionType(ctypes.default_abi,
- jfloat,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {GetStaticDoubleField: new ctypes.FunctionType(ctypes.default_abi,
- jdouble,
- [jenvptr, jclass,
- jfieldid]).ptr},
- {SetStaticObjectField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jobject]).ptr},
- {SetStaticBooleanField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jboolean]).ptr},
- {SetStaticByteField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jbyte]).ptr},
- {SetStaticCharField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jchar]).ptr},
- {SetStaticShortField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jshort]).ptr},
- {SetStaticIntField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jint]).ptr},
- {SetStaticLongField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jlong]).ptr},
- {SetStaticFloatField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jfloat]).ptr},
- {SetStaticDoubleField: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jclass,
- jfieldid, jdouble]).ptr},
-
- {NewString: new ctypes.FunctionType(ctypes.default_abi,
- jstring,
- [jenvptr, jchar.ptr, jsize]).ptr},
- {GetStringLength: new ctypes.FunctionType(ctypes.default_abi,
- jsize,
- [jenvptr, jstring]).ptr},
- {GetStringChars: new ctypes.FunctionType(ctypes.default_abi,
- jchar.ptr,
- [jenvptr, jstring,
- jboolean.ptr]).ptr},
- {ReleaseStringChars: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jstring,
- jchar.ptr]).ptr},
-
- {NewStringUTF: new ctypes.FunctionType(ctypes.default_abi,
- jstring,
- [jenvptr,
- ctypes.char.ptr]).ptr},
- {GetStringUTFLength: new ctypes.FunctionType(ctypes.default_abi,
- jsize,
- [jenvptr, jstring]).ptr},
- {GetStringUTFChars: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.char.ptr,
- [jenvptr, jstring,
- jboolean.ptr]).ptr},
- {ReleaseStringUTFChars: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jstring,
- ctypes.char.ptr]).ptr},
- {GetArrayLength: new ctypes.FunctionType(ctypes.default_abi,
- jsize,
- [jenvptr, jarray]).ptr},
- {NewObjectArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize,
- jclass, jobject]).ptr},
- {GetObjectArrayElement: new ctypes.FunctionType(ctypes.default_abi,
- jobject,
- [jenvptr, jarray,
- jsize]).ptr},
- {SetObjectArrayElement: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jobject]).ptr},
- {NewBooleanArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewByteArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewCharArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewShortArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewIntArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewLongArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewFloatArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {NewDoubleArray: new ctypes.FunctionType(ctypes.default_abi,
- jarray,
- [jenvptr, jsize]).ptr},
- {GetBooleanArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jboolean.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetByteArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jbyte.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetCharArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jchar.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetShortArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jshort.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetIntArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jint.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetLongArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jlong.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetFloatArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jfloat.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {GetDoubleArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- jdouble.ptr,
- [jenvptr, jarray,
- jboolean.ptr]).ptr},
- {ReleaseBooleanArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jboolean.ptr,
- jint]).ptr},
- {ReleaseByteArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jbyte.ptr,
- jint]).ptr},
- {ReleaseCharArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jchar.ptr,
- jint]).ptr},
- {ReleaseShortArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jshort.ptr,
- jint]).ptr},
- {ReleaseIntArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jint.ptr,
- jint]).ptr},
- {ReleaseLongArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jlong.ptr,
- jint]).ptr},
- {ReleaseFloatArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jfloat.ptr,
- jint]).ptr},
- {ReleaseDoubleArrayElements: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jdouble.ptr,
- jint]).ptr},
- {GetBooleanArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jboolean.array()]).ptr},
- {GetByteArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jbyte.array()]).ptr},
- {GetCharArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jchar.array()]).ptr},
- {GetShortArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jshort.array()]).ptr},
- {GetIntArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jint.array()]).ptr},
- {GetLongArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jlong.array()]).ptr},
- {GetFloatArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jfloat.array()]).ptr},
- {GetDoubleArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jdouble.array()]).ptr},
- {SetBooleanArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jboolean.array()]).ptr},
- {SetByteArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jbyte.array()]).ptr},
- {SetCharArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jchar.array()]).ptr},
- {SetShortArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jshort.array()]).ptr},
- {SetIntArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jint.array()]).ptr},
- {SetLongArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jlong.array()]).ptr},
- {SetFloatArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jfloat.array()]).ptr},
- {SetDoubleArrayRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jarray,
- jsize, jsize,
- jdouble.array()]).ptr},
- {RegisterNatives: ctypes.voidptr_t},
- {UnregisterNatives: ctypes.voidptr_t},
- {MonitorEnter: new ctypes.FunctionType(ctypes.default_abi,
- jint, [jenvptr, jobject]).ptr},
- {MonitorExit: new ctypes.FunctionType(ctypes.default_abi,
- jint, [jenvptr, jobject]).ptr},
- {GetJavaVM: ctypes.voidptr_t},
- {GetStringRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jstring,
- jsize, jsize,
- jchar.array()]).ptr},
- {GetStringUTFRegion: new ctypes.FunctionType(ctypes.default_abi,
- ctypes.void_t,
- [jenvptr, jstring,
- jsize, jsize,
- ctypes.char.array()]).ptr},
- {GetPrimitiveArrayCritical: ctypes.voidptr_t},
- {ReleasePrimitiveArrayCritical: ctypes.voidptr_t},
- {GetStringCritical: ctypes.voidptr_t},
- {ReleaseStringCritical: ctypes.voidptr_t},
- {NewWeakGlobalRef: ctypes.voidptr_t},
- {DeleteWeakGlobalRef: ctypes.voidptr_t},
- {ExceptionCheck: new ctypes.FunctionType(ctypes.default_abi,
- jboolean, [jenvptr]).ptr},
- {NewDirectByteBuffer: ctypes.voidptr_t},
- {GetDirectBufferAddress: ctypes.voidptr_t},
- {GetDirectBufferCapacity: ctypes.voidptr_t},
- {GetObjectRefType: ctypes.voidptr_t}]
-);
-
-var GetJNIForThread = libxul.declare("GetJNIForThread",
- ctypes.default_abi,
- JNINativeInterface.ptr.ptr);
-
-var registry = Object.create(null);
-var classes = Object.create(null);
-
-function JNIUnloadClasses(jenv) {
- Object.getOwnPropertyNames(registry).forEach(function(classname) {
- var jcls = unwrap(registry[classname]);
- jenv.contents.contents.DeleteGlobalRef(jenv, jcls);
-
- // Purge the registry, so we don't try to reuse stale global references
- // in JNI calls and we garbage-collect the JS global reference objects.
- delete registry[classname];
- });
-
- // The refs also get added to the 'classes' object, so we should purge it too.
- // That object is a hierarchical data structure organized by class path parts,
- // but deleting its own properties should be sufficient to break its refs.
- Object.getOwnPropertyNames(classes).forEach(function(topLevelPart) {
- delete classes[topLevelPart];
- });
-}
-
-var PREFIX = 'js#';
-// this regex matches one component of a type signature:
-// any number of array modifiers, followed by either a
-// primitive type character or L<classname>;
-var sigRegex = () => /\[*([VZBCSIJFD]|L([^.\/;]+(\/[^.\/;]+)*);)/g;
-var ensureSig = function(classname_or_signature) {
- // convert a classname into a signature,
- // leaving unchanged signatures. We assume that
- // anything not a valid signature is a classname.
- var m = sigRegex().exec(classname_or_signature);
- return (m && m[0] === classname_or_signature) ? classname_or_signature :
- 'L' + classname_or_signature.replace(/\./g, '/') + ';';
-};
-var wrap = function(obj, classSig) {
- if (!classSig) { return obj; }
- // don't wrap primitive types.
- if (classSig.charAt(0)!=='L' &&
- classSig.charAt(0)!=='[') { return obj; }
- var proto = registry[classSig][PREFIX+'proto'];
- return new proto(obj);
-};
-var unwrap = function(obj, opt_jenv, opt_ctype) {
- if (obj && typeof(obj)==='object' && (PREFIX+'obj') in obj) {
- return obj[PREFIX+'obj'];
- } else if (opt_jenv && opt_ctype) {
- if (opt_ctype !== jobject)
- return opt_ctype(obj); // cast to given primitive ctype
- if (typeof(obj)==='string')
- return unwrap(JNINewString(opt_jenv, obj)); // create Java String
- }
- return obj;
-};
-var ensureLoaded = function(jenv, classSig) {
- if (!Object.hasOwnProperty.call(registry, classSig)) {
- JNILoadClass(jenv, classSig);
- }
- return registry[classSig];
-};
-
-function JNINewString(jenv, value) {
- var s = jenv.contents.contents.NewStringUTF(jenv, ctypes.char.array()(value));
- ensureLoaded(jenv, "Ljava/lang/String;");
- return wrap(s, "Ljava/lang/String;");
-}
-
-function JNIReadString(jenv, jstring_value) {
- var val = unwrap(jstring_value);
- if ((!val) || val.isNull()) { return null; }
- var chars = jenv.contents.contents.GetStringUTFChars(jenv, val, null);
- var result = chars.readString();
- jenv.contents.contents.ReleaseStringUTFChars(jenv, val, chars);
- return result;
-}
-
-var sigInfo = {
- 'V': { name: 'Void', longName: 'Void', ctype: ctypes.void_t },
- 'Z': { name: 'Boolean', longName: 'Boolean', ctype: jboolean },
- 'B': { name: 'Byte', longName: 'Byte', ctype: jbyte },
- 'C': { name: 'Char', longName: 'Char', ctype: jchar },
- 'S': { name: 'Short', longName: 'Short', ctype: jshort },
- 'I': { name: 'Int', longName: 'Integer', ctype: jint },
- 'J': { name: 'Long', longName: 'Long', ctype: jlong },
- 'F': { name: 'Float', longName: 'Float', ctype: jfloat },
- 'D': { name: 'Double', longName: 'Double', ctype: jdouble },
- 'L': { name: 'Object', longName: 'Object', ctype: jobject },
- '[': { name: 'Object', longName: 'Object', ctype: jarray }
-};
-
-var sig2type = function(sig) { return sigInfo[sig.charAt(0)].name; };
-var sig2ctype = function(sig) { return sigInfo[sig.charAt(0)].ctype; };
-var sig2prim = function(sig) { return sigInfo[sig.charAt(0)].longName; };
-
-// return the class object for a signature string.
-// allocates 1 or 2 local refs
-function JNIClassObj(jenv, classSig) {
- var jenvpp = function() { return jenv.contents.contents; };
- // Deal with funny calling convention of JNI FindClass method.
- // Classes get the leading & trailing chars stripped; primitives
- // have to be looked up via their wrapper type.
- var prim = function(ty) {
- var jcls = jenvpp().FindClass(jenv, "java/lang/"+ty);
- var jfld = jenvpp().GetStaticFieldID(jenv, jcls, "TYPE",
- "Ljava/lang/Class;");
- return jenvpp().GetStaticObjectField(jenv, jcls, jfld);
- };
- switch (classSig.charAt(0)) {
- case '[':
- return jenvpp().FindClass(jenv, classSig);
- case 'L':
- classSig = classSig.substring(1, classSig.indexOf(';'));
- return jenvpp().FindClass(jenv, classSig);
- default:
- return prim(sig2prim(classSig));
- }
-}
-
-// return the signature string for a Class object.
-// allocates 2 local refs
-function JNIClassSig(jenv, jcls) {
- var jenvpp = function() { return jenv.contents.contents; };
- var jclscls = jenvpp().FindClass(jenv, "java/lang/Class");
- var jmtd = jenvpp().GetMethodID(jenv, jclscls,
- "getName", "()Ljava/lang/String;");
- var name = jenvpp().CallObjectMethod(jenv, jcls, jmtd);
- name = JNIReadString(jenv, name);
- // API is weird. Make sure we're using slashes not dots
- name = name.replace(/\./g, '/');
- // special case primitives, arrays
- if (name.charAt(0)==='[') return name;
- switch(name) {
- case 'void': return 'V';
- case 'boolean': return 'Z';
- case 'byte': return 'B';
- case 'char': return 'C';
- case 'short': return 'S';
- case 'int': return 'I';
- case 'long': return 'J';
- case 'float': return 'F';
- case 'double': return 'D';
- default:
- return 'L' + name + ';';
- }
-}
-
-// create dispatch method
-// we resolve overloaded methods only by # of arguments. If you need
-// further resolution, use the 'long form' of the method name, ie:
-// obj['toString()Ljava/lang/String'].call(obj);
-var overloadFunc = function(basename) {
- return function() {
- return this[basename+'('+arguments.length+')'].apply(this, arguments);
- };
-};
-
-// Create appropriate wrapper fields/methods for a Java class.
-function JNILoadClass(jenv, classSig, opt_props) {
- var jenvpp = function() { return jenv.contents.contents; };
- var props = opt_props || {};
-
- // allocate a local reference frame with enough space
- // this class (1 or 2 local refs) plus superclass (3 refs)
- // plus array element class (1 or 2 local refs)
- var numLocals = 7;
- jenvpp().PushLocalFrame(jenv, numLocals);
-
- var jcls;
- if (Object.hasOwnProperty.call(registry, classSig)) {
- jcls = unwrap(registry[classSig]);
- } else {
- jcls = jenvpp().NewGlobalRef(jenv, JNIClassObj(jenv, classSig));
-
- // get name of superclass
- var jsuper = jenvpp().GetSuperclass(jenv, jcls);
- if (jsuper.isNull()) {
- jsuper = null;
- } else {
- jsuper = JNIClassSig(jenv, jsuper);
- }
-
- registry[classSig] = Object.create(jsuper?ensureLoaded(jenv, jsuper):null);
- registry[classSig][PREFIX+'obj'] = jcls; // global ref, persistent.
- registry[classSig][PREFIX+'proto'] =
- function(o) { this[PREFIX+'obj'] = o; };
- registry[classSig][PREFIX+'proto'].prototype =
- Object.create(jsuper ?
- ensureLoaded(jenv, jsuper)[PREFIX+'proto'].prototype :
- null);
- // Add a __cast__ method to the wrapper corresponding to the class
- registry[classSig].__cast__ = function(obj) {
- return wrap(unwrap(obj), classSig);
- };
-
- // make wrapper accessible via the classes object.
- var path = sig2type(classSig).toLowerCase();
- if (classSig.charAt(0)==='L') {
- path = classSig.substring(1, classSig.length-1);
- }
- if (classSig.charAt(0)!=='[') {
- var root = classes, i;
- var parts = path.split('/');
- for (i = 0; i < parts.length-1; i++) {
- if (!Object.hasOwnProperty.call(root, parts[i])) {
- root[parts[i]] = Object.create(null);
- }
- root = root[parts[i]];
- }
- root[parts[parts.length-1]] = registry[classSig];
- }
- }
-
- var r = registry[classSig];
- var rpp = r[PREFIX+'proto'].prototype;
-
- if (classSig.charAt(0)==='[') {
- // add 'length' field for arrays
- Object.defineProperty(rpp, 'length', {
- get: function() {
- return jenvpp().GetArrayLength(jenv, unwrap(this));
- }
- });
- // add 'get' and 'set' methods, 'new' constructor
- var elemSig = classSig.substring(1);
- ensureLoaded(jenv, elemSig);
-
- registry[elemSig].__array__ = r;
- if (!Object.hasOwnProperty.call(registry[elemSig], 'array'))
- registry[elemSig].array = r;
-
- if (elemSig.charAt(0)==='L' || elemSig.charAt(0)==='[') {
- var elemClass = unwrap(registry[elemSig]);
-
- rpp.get = function(idx) {
- return wrap(jenvpp().GetObjectArrayElement(jenv, unwrap(this), idx),
- elemSig);
- };
- rpp.set = function(idx, value) {
- jenvpp().SetObjectArrayElement(jenv, unwrap(this), idx,
- unwrap(value, jenv, jobject));
- };
- rpp.getElements = function(start, len) {
- var i, r=[];
- for (i=0; i<len; i++) { r.push(this.get(start+i)); }
- return r;
- };
- rpp.setElements = function(start, vals) {
- vals.forEach(function(v, i) { this.set(start+i, v); }.bind(this));
- };
- r['new'] = function(length) {
- return wrap(jenvpp().NewObjectArray(jenv, length, elemClass, null),
- classSig);
- };
- } else {
- var ty = sig2type(elemSig), ctype = sig2ctype(elemSig);
- var constructor = "New"+ty+"Array";
- var getter = "Get"+ty+"ArrayRegion";
- var setter = "Set"+ty+"ArrayRegion";
- rpp.get = function(idx) { return this.getElements(idx, 1)[0]; };
- rpp.set = function(idx, val) { this.setElements(idx, [val]); };
- rpp.getElements = function(start, len) {
- var j = jenvpp();
- var buf = new (ctype.array())(len);
- j[getter].call(j, jenv, unwrap(this), start, len, buf);
- return buf;
- };
- rpp.setElements = function(start, vals) {
- var j = jenvpp();
- j[setter].call(j, jenv, unwrap(this), start, vals.length,
- ctype.array()(vals));
- };
- r['new'] = function(length) {
- var j = jenvpp();
- return wrap(j[constructor].call(j, jenv, length), classSig);
- };
- }
- }
-
- (props.static_fields || []).forEach(function(fld) {
- var jfld = jenvpp().GetStaticFieldID(jenv, jcls, fld.name, fld.sig);
- var ty = sig2type(fld.sig), nm = fld.sig;
- var getter = "GetStatic"+ty+"Field", setter = "SetStatic"+ty+"Field";
- ensureLoaded(jenv, nm);
- var props = {
- get: function() {
- var j = jenvpp();
- return wrap(j[getter].call(j, jenv, jcls, jfld), nm);
- },
- set: function(newValue) {
- var j = jenvpp();
- j[setter].call(j, jenv, jcls, jfld, unwrap(newValue));
- }
- };
- Object.defineProperty(r, fld.name, props);
- // add static fields to object instances, too.
- Object.defineProperty(rpp, fld.name, props);
- });
- (props.static_methods || []).forEach(function(mtd) {
- var jmtd = jenvpp().GetStaticMethodID(jenv, jcls, mtd.name, mtd.sig);
- var argctypes = mtd.sig.match(sigRegex()).map(s => sig2ctype(s));
- var returnSig = mtd.sig.substring(mtd.sig.indexOf(')')+1);
- var ty = sig2type(returnSig), nm = returnSig;
- var call = "CallStatic"+ty+"Method";
- ensureLoaded(jenv, nm);
- r[mtd.name] = rpp[mtd.name] = overloadFunc(mtd.name);
- r[mtd.name + mtd.sig] = r[mtd.name+'('+(argctypes.length-1)+')'] =
- // add static methods to object instances, too.
- rpp[mtd.name + mtd.sig] = rpp[mtd.name+'('+(argctypes.length-1)+')'] = function() {
- var i, j = jenvpp();
- var args = [jenv, jcls, jmtd];
- for (i=0; i<arguments.length; i++) {
- args.push(unwrap(arguments[i], jenv, argctypes[i]));
- }
- return wrap(j[call].apply(j, args), nm);
- };
- });
- (props.constructors || []).forEach(function(mtd) {
- mtd.name = "<init>";
- var jmtd = jenvpp().GetMethodID(jenv, jcls, mtd.name, mtd.sig);
- var argctypes = mtd.sig.match(sigRegex()).map(s => sig2ctype(s));
- var returnSig = mtd.sig.substring(mtd.sig.indexOf(')')+1);
-
- r['new'] = overloadFunc('new');
- r['new'+mtd.sig] = r['new('+(argctypes.length-1)+')'] = function() {
- var i, j = jenvpp();
- var args = [jenv, jcls, jmtd];
- for (i=0; i<arguments.length; i++) {
- args.push(unwrap(arguments[i], jenv, argctypes[i]));
- }
- return wrap(j.NewObject.apply(j, args), classSig);
- };
- });
- (props.fields || []).forEach(function(fld) {
- var jfld = jenvpp().GetFieldID(jenv, jcls, fld.name, fld.sig);
- var ty = sig2type(fld.sig), nm = fld.sig;
- var getter = "Get"+ty+"Field", setter = "Set"+ty+"Field";
- ensureLoaded(jenv, nm);
- Object.defineProperty(rpp, fld.name, {
- get: function() {
- var j = jenvpp();
- return wrap(j[getter].call(j, jenv, unwrap(this), jfld), nm);
- },
- set: function(newValue) {
- var j = jenvpp();
- j[setter].call(j, jenv, unwrap(this), jfld, unwrap(newValue));
- }
- });
- });
- (props.methods || []).forEach(function(mtd) {
- var jmtd = jenvpp().GetMethodID(jenv, jcls, mtd.name, mtd.sig);
- var argctypes = mtd.sig.match(sigRegex()).map(s => sig2ctype(s));
- var returnSig = mtd.sig.substring(mtd.sig.indexOf(')')+1);
- var ty = sig2type(returnSig), nm = returnSig;
- var call = "Call"+ty+"Method";
- ensureLoaded(jenv, nm);
- rpp[mtd.name] = overloadFunc(mtd.name);
- rpp[mtd.name + mtd.sig] = rpp[mtd.name+'('+(argctypes.length-1)+')'] = function() {
- var i, j = jenvpp();
- var args = [jenv, unwrap(this), jmtd];
- for (i=0; i<arguments.length; i++) {
- args.push(unwrap(arguments[i], jenv, argctypes[i]));
- }
- return wrap(j[call].apply(j, args), nm);
- };
- });
- jenvpp().PopLocalFrame(jenv, null);
- return r;
-}
-
-// exported object
-var JNI = {
- // primitive types
- jboolean: jboolean,
- jbyte: jbyte,
- jchar: jchar,
- jshort: jshort,
- jint: jint,
- jlong: jlong,
- jfloat: jfloat,
- jdouble: jdouble,
- jsize: jsize,
-
- // class registry
- classes: classes,
-
- // methods
- GetForThread: GetJNIForThread,
- NewString: JNINewString,
- ReadString: JNIReadString,
- LoadClass: function(jenv, classname_or_signature, props) {
- return JNILoadClass(jenv, ensureSig(classname_or_signature), props);
- },
- UnloadClasses: JNIUnloadClasses
-};
diff --git a/mobile/android/modules/JavaAddonManager.jsm b/mobile/android/modules/JavaAddonManager.jsm
deleted file mode 100644
index a24535ede..000000000
--- a/mobile/android/modules/JavaAddonManager.jsm
+++ /dev/null
@@ -1,115 +0,0 @@
-// -*- Mode: js2; tab-width: 2; indent-tabs-mode: nil; js2-basic-offset: 2; js2-skip-preprocessor-directives: t; -*-
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["JavaAddonManager"];
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components; /*global Components */
-
-Cu.import("resource://gre/modules/Messaging.jsm"); /*global Messaging */
-Cu.import("resource://gre/modules/Services.jsm"); /*global Services */
-
-function resolveGeckoURI(uri) {
- if (!uri) {
- throw new Error("Can't resolve an empty uri");
- }
- if (uri.startsWith("chrome://")) {
- let registry = Cc['@mozilla.org/chrome/chrome-registry;1'].getService(Ci["nsIChromeRegistry"]);
- return registry.convertChromeURL(Services.io.newURI(uri, null, null)).spec;
- } else if (uri.startsWith("resource://")) {
- let handler = Services.io.getProtocolHandler("resource").QueryInterface(Ci.nsIResProtocolHandler);
- return handler.resolveURI(Services.io.newURI(uri, null, null));
- }
- return uri;
-}
-
-/**
- * A promise-based API
- */
-var JavaAddonManager = Object.freeze({
- classInstanceFromFile: function(classname, filename) {
- if (!classname) {
- throw new Error("classname cannot be null");
- }
- if (!filename) {
- throw new Error("filename cannot be null");
- }
- return Messaging.sendRequestForResult({
- type: "JavaAddonManagerV1:Load",
- classname: classname,
- filename: resolveGeckoURI(filename)
- })
- .then((guid) => {
- if (!guid) {
- throw new Error("Internal error: guid should not be null");
- }
- return new JavaAddonV1({classname: classname, guid: guid});
- });
- }
-});
-
-function JavaAddonV1(options = {}) {
- if (!(this instanceof JavaAddonV1)) {
- return new JavaAddonV1(options);
- }
- if (!options.classname) {
- throw new Error("options.classname cannot be null");
- }
- if (!options.guid) {
- throw new Error("options.guid cannot be null");
- }
- this._classname = options.classname;
- this._guid = options.guid;
- this._loaded = true;
- this._listeners = {};
-}
-
-JavaAddonV1.prototype = Object.freeze({
- unload: function() {
- if (!this._loaded) {
- return;
- }
-
- Messaging.sendRequestForResult({
- type: "JavaAddonManagerV1:Unload",
- guid: this._guid
- })
- .then(() => {
- this._loaded = false;
- for (let listener of this._listeners) {
- // If we use this.removeListener, we prefix twice.
- Messaging.removeListener(listener);
- }
- this._listeners = {};
- });
- },
-
- _prefix: function(message) {
- let newMessage = Cu.cloneInto(message, {}, { cloneFunctions: false });
- newMessage.type = this._guid + ":" + message.type;
- return newMessage;
- },
-
- sendRequest: function(message) {
- return Messaging.sendRequest(this._prefix(message));
- },
-
- sendRequestForResult: function(message) {
- return Messaging.sendRequestForResult(this._prefix(message));
- },
-
- addListener: function(listener, message) {
- let prefixedMessage = this._guid + ":" + message;
- this._listeners[prefixedMessage] = listener;
- return Messaging.addListener(listener, prefixedMessage);
- },
-
- removeListener: function(message) {
- let prefixedMessage = this._guid + ":" + message;
- delete this._listeners[prefixedMessage];
- return Messaging.removeListener(prefixedMessage);
- }
-});
diff --git a/mobile/android/modules/LightweightThemeConsumer.jsm b/mobile/android/modules/LightweightThemeConsumer.jsm
deleted file mode 100644
index 3d3ca4c0b..000000000
--- a/mobile/android/modules/LightweightThemeConsumer.jsm
+++ /dev/null
@@ -1,44 +0,0 @@
-/* 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/. */
-
-var EXPORTED_SYMBOLS = ["LightweightThemeConsumer"];
-var Cc = Components.classes;
-var Ci = Components.interfaces;
-
-Components.utils.import("resource://gre/modules/Services.jsm");
-Components.utils.import("resource://gre/modules/LightweightThemeManager.jsm");
-
-function LightweightThemeConsumer(aDocument) {
- this._doc = aDocument;
- Services.obs.addObserver(this, "lightweight-theme-styling-update", false);
- Services.obs.addObserver(this, "lightweight-theme-apply", false);
-
- this._update(LightweightThemeManager.currentThemeForDisplay);
-}
-
-LightweightThemeConsumer.prototype = {
- observe: function (aSubject, aTopic, aData) {
- if (aTopic == "lightweight-theme-styling-update")
- this._update(JSON.parse(aData));
- else if (aTopic == "lightweight-theme-apply")
- this._update(LightweightThemeManager.currentThemeForDisplay);
- },
-
- destroy: function () {
- Services.obs.removeObserver(this, "lightweight-theme-styling-update");
- Services.obs.removeObserver(this, "lightweight-theme-apply");
- this._doc = null;
- },
-
- _update: function (aData) {
- if (!aData)
- aData = { headerURL: "", footerURL: "", textcolor: "", accentcolor: "" };
-
- let active = !!aData.headerURL;
-
- let msg = active ? { type: "LightweightTheme:Update", data: aData } :
- { type: "LightweightTheme:Disable" };
- Services.androidBridge.handleGeckoMessage(msg);
- }
-}
diff --git a/mobile/android/modules/MediaPlayerApp.jsm b/mobile/android/modules/MediaPlayerApp.jsm
deleted file mode 100644
index 949863d1f..000000000
--- a/mobile/android/modules/MediaPlayerApp.jsm
+++ /dev/null
@@ -1,166 +0,0 @@
-// -*- Mode: js2; tab-width: 2; indent-tabs-mode: nil; js2-basic-offset: 2; js2-skip-preprocessor-directives: t; -*-
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["MediaPlayerApp"];
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-var log = Cu.import("resource://gre/modules/AndroidLog.jsm", {}).AndroidLog.d.bind(null, "MediaPlayerApp");
-
-// Helper function for sending commands to Java.
-function send(type, data, callback) {
- let msg = {
- type: type
- };
-
- for (let i in data) {
- msg[i] = data[i];
- }
-
- Messaging.sendRequestForResult(msg)
- .then(result => callback(result, null),
- error => callback(null, error));
-}
-
-/* These apps represent players supported natively by the platform. This class will proxy commands
- * to native controls */
-function MediaPlayerApp(service) {
- this.service = service;
- this.location = service.location;
- this.id = service.uuid;
-}
-
-MediaPlayerApp.prototype = {
- start: function start(callback) {
- send("MediaPlayer:Start", { id: this.id }, (result, err) => {
- if (callback) {
- callback(err == null);
- }
- });
- },
-
- stop: function stop(callback) {
- send("MediaPlayer:Stop", { id: this.id }, (result, err) => {
- if (callback) {
- callback(err == null);
- }
- });
- },
-
- remoteMedia: function remoteMedia(callback, listener) {
- if (callback) {
- callback(new RemoteMedia(this.id, listener));
- }
- },
-
- mirror: function mirror(callback) {
- send("MediaPlayer:Mirror", { id: this.id }, (result, err) => {
- if (callback) {
- callback(err == null);
- }
- });
- }
-}
-
-/* RemoteMedia provides a proxy to a native media player session.
- */
-function RemoteMedia(id, listener) {
- this._id = id;
- this._listener = listener;
-
- if ("onRemoteMediaStart" in this._listener) {
- Services.tm.mainThread.dispatch((function() {
- this._listener.onRemoteMediaStart(this);
- }).bind(this), Ci.nsIThread.DISPATCH_NORMAL);
- }
-}
-
-RemoteMedia.prototype = {
- shutdown: function shutdown() {
- Services.obs.removeObserver(this, "MediaPlayer:Playing");
- Services.obs.removeObserver(this, "MediaPlayer:Paused");
-
- this._send("MediaPlayer:End", {}, (result, err) => {
- this._status = "shutdown";
- if ("onRemoteMediaStop" in this._listener) {
- this._listener.onRemoteMediaStop(this);
- }
- });
- },
-
- play: function play() {
- this._send("MediaPlayer:Play", {}, (result, err) => {
- if (err) {
- Cu.reportError("Can't play " + err);
- this.shutdown();
- return;
- }
-
- this._status = "started";
- });
- },
-
- pause: function pause() {
- this._send("MediaPlayer:Pause", {}, (result, err) => {
- if (err) {
- Cu.reportError("Can't pause " + err);
- this.shutdown();
- return;
- }
-
- this._status = "paused";
- });
- },
-
- load: function load(aData) {
- this._send("MediaPlayer:Load", aData, (result, err) => {
- if (err) {
- Cu.reportError("Can't load " + err);
- this.shutdown();
- return;
- }
-
- Services.obs.addObserver(this, "MediaPlayer:Playing", false);
- Services.obs.addObserver(this, "MediaPlayer:Paused", false);
- this._status = "started";
- })
- },
-
- get status() {
- return this._status;
- },
-
- observe: function (aSubject, aTopic, aData) {
- switch (aTopic) {
- case "MediaPlayer:Playing":
- if (this._status !== "started") {
- this._status = "started";
- if ("onRemoteMediaStatus" in this._listener) {
- this._listener.onRemoteMediaStatus(this);
- }
- }
- break;
- case "MediaPlayer:Paused":
- if (this._status !== "paused") {
- this._status = "paused";
- if ("onRemoteMediaStatus" in this._listener) {
- this._listener.onRemoteMediaStatus(this);
- }
- }
- break;
- default:
- break;
- }
- },
-
- _send: function(msg, data, callback) {
- data.id = this._id;
- send(msg, data, callback);
- }
-}
diff --git a/mobile/android/modules/Messaging.jsm b/mobile/android/modules/Messaging.jsm
deleted file mode 100644
index 30b7f5a96..000000000
--- a/mobile/android/modules/Messaging.jsm
+++ /dev/null
@@ -1,183 +0,0 @@
-/* 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/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Task.jsm");
-
-this.EXPORTED_SYMBOLS = ["sendMessageToJava", "Messaging"];
-
-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);
- }
-}
-
-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);
- }
-};
diff --git a/mobile/android/modules/NetErrorHelper.jsm b/mobile/android/modules/NetErrorHelper.jsm
deleted file mode 100644
index 9c74df8fe..000000000
--- a/mobile/android/modules/NetErrorHelper.jsm
+++ /dev/null
@@ -1,175 +0,0 @@
-/* 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/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-Cu.import("resource://gre/modules/UITelemetry.jsm");
-
-this.EXPORTED_SYMBOLS = ["NetErrorHelper"];
-
-const KEY_CODE_ENTER = 13;
-
-/* Handlers is a list of objects that will be notified when an error page is shown
- * or when an event occurs on the page that they are registered to handle. Registration
- * is done by just adding yourself to the dictionary.
- *
- * handlers.myKey = {
- * onPageShown: function(browser) { },
- * handleEvent: function(event) { },
- * }
- *
- * The key that you register yourself with should match the ID of the element you want to
- * watch for click events on.
- */
-
-var handlers = {};
-
-function NetErrorHelper(browser) {
- browser.addEventListener("click", this.handleClick, true);
-
- let listener = () => {
- browser.removeEventListener("click", this.handleClick, true);
- browser.removeEventListener("pagehide", listener, true);
- };
- browser.addEventListener("pagehide", listener, true);
-
- // Handlers may want to customize the page
- for (let id in handlers) {
- if (handlers[id].onPageShown) {
- handlers[id].onPageShown(browser);
- }
- }
-}
-
-NetErrorHelper.attachToBrowser = function(browser) {
- return new NetErrorHelper(browser);
-}
-
-NetErrorHelper.prototype = {
- handleClick: function(event) {
- let node = event.target;
-
- while(node) {
- if (node.id in handlers && handlers[node.id].handleClick) {
- handlers[node.id].handleClick(event);
- return;
- }
-
- node = node.parentNode;
- }
- },
-}
-
-handlers.searchbutton = {
- onPageShown: function(browser) {
- let search = browser.contentDocument.querySelector("#searchbox");
- if (!search) {
- return;
- }
-
- let browserWin = Services.wm.getMostRecentWindow("navigator:browser");
- let tab = browserWin.BrowserApp.getTabForBrowser(browser);
-
- // If there is no stored userRequested, just hide the searchbox
- if (!tab.userRequested) {
- search.style.display = "none";
- } else {
- let text = browser.contentDocument.querySelector("#searchtext");
- text.value = tab.userRequested;
- text.addEventListener("keypress", (event) => {
- if (event.keyCode === KEY_CODE_ENTER) {
- this.doSearch(event.target.value);
- }
- });
- }
- },
-
- handleClick: function(event) {
- let value = event.target.previousElementSibling.value;
- this.doSearch(value);
- },
-
- doSearch: function(value) {
- UITelemetry.addEvent("neterror.1", "button", null, "search");
- let engine = Services.search.defaultEngine;
- let uri = engine.getSubmission(value).uri;
-
- let browserWin = Services.wm.getMostRecentWindow("navigator:browser");
- // Reset the user search to whatever the new search term was
- browserWin.BrowserApp.loadURI(uri.spec, undefined, { isSearch: true, userRequested: value });
- }
-};
-
-handlers.wifi = {
- // This registers itself with the nsIObserverService as a weak ref,
- // so we have to implement GetWeakReference as well.
- QueryInterface: XPCOMUtils.generateQI([Ci.nsIObserver,
- Ci.nsISupportsWeakReference]),
-
- GetWeakReference: function() {
- return Cu.getWeakReference(this);
- },
-
- onPageShown: function(browser) {
- // If we have a connection, don't bother showing the wifi toggle.
- let network = Cc["@mozilla.org/network/network-link-service;1"].getService(Ci.nsINetworkLinkService);
- if (network.isLinkUp && network.linkStatusKnown) {
- let nodes = browser.contentDocument.querySelectorAll("#wifi");
- for (let i = 0; i < nodes.length; i++) {
- nodes[i].style.display = "none";
- }
- }
- },
-
- handleClick: function(event) {
- let node = event.target;
- while(node && node.id !== "wifi") {
- node = node.parentNode;
- }
-
- if (!node) {
- return;
- }
-
- UITelemetry.addEvent("neterror.1", "button", null, "wifitoggle");
- // Show indeterminate progress while we wait for the network.
- node.disabled = true;
- node.classList.add("inProgress");
-
- this.node = Cu.getWeakReference(node);
- Services.obs.addObserver(this, "network:link-status-changed", true);
-
- Messaging.sendRequest({
- type: "Wifi:Enable"
- });
- },
-
- observe: function(subject, topic, data) {
- let node = this.node.get();
- if (!node) {
- return;
- }
-
- // Remove the progress bar
- node.disabled = false;
- node.classList.remove("inProgress");
-
- let network = Cc["@mozilla.org/network/network-link-service;1"].getService(Ci.nsINetworkLinkService);
- if (network.isLinkUp && network.linkStatusKnown) {
- // If everything worked, reload the page
- UITelemetry.addEvent("neterror.1", "button", null, "wifitoggle.reload");
- Services.obs.removeObserver(this, "network:link-status-changed");
-
- // Even at this point, Android sometimes lies about the real state of the network and this reload request fails.
- // Add a 500ms delay before refreshing the page.
- node.ownerDocument.defaultView.setTimeout(function() {
- node.ownerDocument.location.reload(false);
- }, 500);
- }
- }
-}
-
diff --git a/mobile/android/modules/Notifications.jsm b/mobile/android/modules/Notifications.jsm
deleted file mode 100644
index a035bb2e3..000000000
--- a/mobile/android/modules/Notifications.jsm
+++ /dev/null
@@ -1,259 +0,0 @@
-/* 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");
-
-this.EXPORTED_SYMBOLS = ["Notifications"];
-
-function log(msg) {
- // Services.console.logStringMessage(msg);
-}
-
-var _notificationsMap = {};
-var _handlersMap = {};
-
-function Notification(aId, aOptions) {
- this._id = aId;
- this._when = (new Date()).getTime();
- this.fillWithOptions(aOptions);
-}
-
-Notification.prototype = {
- fillWithOptions: function(aOptions) {
- if ("icon" in aOptions && aOptions.icon != null)
- this._icon = aOptions.icon;
- else
- throw "Notification icon is mandatory";
-
- if ("title" in aOptions && aOptions.title != null)
- this._title = aOptions.title;
- else
- throw "Notification title is mandatory";
-
- if ("message" in aOptions && aOptions.message != null)
- this._message = aOptions.message;
- else
- this._message = null;
-
- if ("priority" in aOptions && aOptions.priority != null)
- this._priority = aOptions.priority;
-
- if ("buttons" in aOptions && aOptions.buttons != null) {
- if (aOptions.buttons.length > 3)
- throw "Too many buttons provided. The max number is 3";
-
- this._buttons = {};
- for (let i = 0; i < aOptions.buttons.length; i++) {
- let button_id = aOptions.buttons[i].buttonId;
- this._buttons[button_id] = aOptions.buttons[i];
- }
- } else {
- this._buttons = null;
- }
-
- if ("ongoing" in aOptions && aOptions.ongoing != null)
- this._ongoing = aOptions.ongoing;
- else
- this._ongoing = false;
-
- if ("progress" in aOptions && aOptions.progress != null)
- this._progress = aOptions.progress;
- else
- this._progress = null;
-
- if ("onCancel" in aOptions && aOptions.onCancel != null)
- this._onCancel = aOptions.onCancel;
- else
- this._onCancel = null;
-
- if ("onClick" in aOptions && aOptions.onClick != null)
- this._onClick = aOptions.onClick;
- else
- this._onClick = null;
-
- if ("cookie" in aOptions && aOptions.cookie != null)
- this._cookie = aOptions.cookie;
- else
- this._cookie = null;
-
- if ("handlerKey" in aOptions && aOptions.handlerKey != null)
- this._handlerKey = aOptions.handlerKey;
-
- if ("persistent" in aOptions && aOptions.persistent != null)
- this._persistent = aOptions.persistent;
- else
- this._persistent = false;
- },
-
- show: function() {
- let msg = {
- type: "Notification:Show",
- id: this._id,
- title: this._title,
- smallIcon: this._icon,
- ongoing: this._ongoing,
- when: this._when,
- persistent: this._persistent,
- };
-
- if (this._message)
- msg.text = this._message;
-
- if (this._progress) {
- msg.progress_value = this._progress;
- msg.progress_max = 100;
- msg.progress_indeterminate = false;
- } else if (Number.isNaN(this._progress)) {
- msg.progress_value = 0;
- msg.progress_max = 0;
- msg.progress_indeterminate = true;
- }
-
- if (this._cookie)
- msg.cookie = JSON.stringify(this._cookie);
-
- if (this._priority)
- msg.priority = this._priority;
-
- if (this._buttons) {
- msg.actions = [];
- let buttonName;
- for (buttonName in this._buttons) {
- let button = this._buttons[buttonName];
- let obj = {
- buttonId: button.buttonId,
- title : button.title,
- icon : button.icon
- };
- msg.actions.push(obj);
- }
- }
-
- if (this._light)
- msg.light = this._light;
-
- if (this._handlerKey)
- msg.handlerKey = this._handlerKey;
-
- Services.androidBridge.handleGeckoMessage(msg);
- return this;
- },
-
- cancel: function() {
- let msg = {
- type: "Notification:Hide",
- id: this._id,
- handlerKey: this._handlerKey,
- cookie: JSON.stringify(this._cookie),
- };
- Services.androidBridge.handleGeckoMessage(msg);
- }
-}
-
-var Notifications = {
- get idService() {
- delete this.idService;
- return this.idService = Cc["@mozilla.org/uuid-generator;1"].getService(Ci.nsIUUIDGenerator);
- },
-
- registerHandler: function(key, handler) {
- if (!_handlersMap[key]) {
- _handlersMap[key] = [];
- }
- _handlersMap[key].push(handler);
- },
-
- unregisterHandler: function(key, handler) {
- let h = _handlersMap[key];
- if (!h) {
- return;
- }
- let i = h.indexOf(handler);
- if (i > -1) {
- h.splice(i, 1);
- }
- },
-
- create: function notif_notify(aOptions) {
- let id = this.idService.generateUUID().toString();
-
- let notification = new Notification(id, aOptions);
- _notificationsMap[id] = notification;
- notification.show();
-
- return id;
- },
-
- update: function notif_update(aId, aOptions) {
- let notification = _notificationsMap[aId];
- if (!notification)
- throw "Unknown notification id";
- notification.fillWithOptions(aOptions);
- notification.show();
- },
-
- cancel: function notif_cancel(aId) {
- let notification = _notificationsMap[aId];
- if (notification)
- notification.cancel();
- },
-
- observe: function notif_observe(aSubject, aTopic, aData) {
- Services.console.logStringMessage(aTopic + " " + aData);
-
- let data = JSON.parse(aData);
- let id = data.id;
- let handlerKey = data.handlerKey;
- let cookie = data.cookie ? JSON.parse(data.cookie) : undefined;
- let notification = _notificationsMap[id];
-
- switch (data.eventType) {
- case "notification-clicked":
- if (notification && notification._onClick)
- notification._onClick(id, notification._cookie);
-
- if (handlerKey) {
- _handlersMap[handlerKey].forEach(function(handler) {
- handler.onClick(cookie);
- });
- }
-
- break;
- case "notification-button-clicked":
- if (handlerKey) {
- _handlersMap[handlerKey].forEach(function(handler) {
- handler.onButtonClick(data.buttonId, cookie);
- });
- }
-
- break;
- case "notification-cleared":
- case "notification-closed":
- if (handlerKey) {
- _handlersMap[handlerKey].forEach(function(handler) {
- handler.onCancel(cookie);
- });
- }
-
- if (notification && notification._onCancel)
- notification._onCancel(id, notification._cookie);
- delete _notificationsMap[id]; // since the notification was dismissed, we no longer need to hold a reference.
- break;
- }
- },
-
- QueryInterface: function (aIID) {
- if (!aIID.equals(Ci.nsISupports) &&
- !aIID.equals(Ci.nsIObserver) &&
- !aIID.equals(Ci.nsISupportsWeakReference))
- throw Components.results.NS_ERROR_NO_INTERFACE;
- return this;
- }
-};
-
-Services.obs.addObserver(Notifications, "Notification:Event", false);
diff --git a/mobile/android/modules/PageActions.jsm b/mobile/android/modules/PageActions.jsm
deleted file mode 100644
index a66268f82..000000000
--- a/mobile/android/modules/PageActions.jsm
+++ /dev/null
@@ -1,113 +0,0 @@
-/* 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/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-
-XPCOMUtils.defineLazyServiceGetter(this, "uuidgen",
- "@mozilla.org/uuid-generator;1",
- "nsIUUIDGenerator");
-
-this.EXPORTED_SYMBOLS = ["PageActions"];
-
-// Copied from browser.js
-// TODO: We should move this method to a common importable location
-function resolveGeckoURI(aURI) {
- if (!aURI)
- throw "Can't resolve an empty uri";
-
- if (aURI.startsWith("chrome://")) {
- let registry = Cc['@mozilla.org/chrome/chrome-registry;1'].getService(Ci["nsIChromeRegistry"]);
- return registry.convertChromeURL(Services.io.newURI(aURI, null, null)).spec;
- } else if (aURI.startsWith("resource://")) {
- let handler = Services.io.getProtocolHandler("resource").QueryInterface(Ci.nsIResProtocolHandler);
- return handler.resolveURI(Services.io.newURI(aURI, null, null));
- }
- return aURI;
-}
-
-var PageActions = {
- _items: { },
-
- _inited: false,
-
- _maybeInit: function() {
- if (!this._inited && Object.keys(this._items).length > 0) {
- this._inited = true;
- Services.obs.addObserver(this, "PageActions:Clicked", false);
- Services.obs.addObserver(this, "PageActions:LongClicked", false);
- }
- },
-
- _maybeUninit: function() {
- if (this._inited && Object.keys(this._items).length == 0) {
- this._inited = false;
- Services.obs.removeObserver(this, "PageActions:Clicked");
- Services.obs.removeObserver(this, "PageActions:LongClicked");
- }
- },
-
- observe: function(aSubject, aTopic, aData) {
- let item = this._items[aData];
- if (aTopic == "PageActions:Clicked") {
- if (item.clickCallback) {
- item.clickCallback();
- }
- } else if (aTopic == "PageActions:LongClicked") {
- if (item.longClickCallback) {
- item.longClickCallback();
- }
- }
- },
-
- isShown: function(id) {
- return !!this._items[id];
- },
-
- synthesizeClick: function(id) {
- let item = this._items[id];
- if (item && item.clickCallback) {
- item.clickCallback();
- }
- },
-
- add: function(aOptions) {
- let id = aOptions.id || uuidgen.generateUUID().toString()
-
- Messaging.sendRequest({
- type: "PageActions:Add",
- id: id,
- title: aOptions.title,
- icon: resolveGeckoURI(aOptions.icon),
- important: "important" in aOptions ? aOptions.important : false
- });
-
- this._items[id] = {};
-
- if (aOptions.clickCallback) {
- this._items[id].clickCallback = aOptions.clickCallback;
- }
-
- if (aOptions.longClickCallback) {
- this._items[id].longClickCallback = aOptions.longClickCallback;
- }
-
- this._maybeInit();
- return id;
- },
-
- remove: function(id) {
- Messaging.sendRequest({
- type: "PageActions:Remove",
- id: id
- });
-
- delete this._items[id];
- this._maybeUninit();
- }
-}
diff --git a/mobile/android/modules/Prompt.jsm b/mobile/android/modules/Prompt.jsm
deleted file mode 100644
index 5bed87650..000000000
--- a/mobile/android/modules/Prompt.jsm
+++ /dev/null
@@ -1,234 +0,0 @@
-/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
- * 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"
-
-var Cc = Components.classes;
-var Ci = Components.interfaces;
-
-Components.utils.import("resource://gre/modules/Services.jsm");
-Components.utils.import("resource://gre/modules/Messaging.jsm");
-
-this.EXPORTED_SYMBOLS = ["Prompt"];
-
-function log(msg) {
- Services.console.logStringMessage(msg);
-}
-
-function Prompt(aOptions) {
- this.window = "window" in aOptions ? aOptions.window : null;
-
- this.msg = { async: true };
-
- if (this.window) {
- let window = Services.wm.getMostRecentWindow("navigator:browser");
- var tab = window.BrowserApp.getTabForWindow(this.window);
- if (tab) {
- this.msg.tabId = tab.id;
- }
- }
-
- if (aOptions.priority === 1)
- this.msg.type = "Prompt:ShowTop"
- else
- this.msg.type = "Prompt:Show"
-
- if ("title" in aOptions && aOptions.title != null)
- this.msg.title = aOptions.title;
-
- if ("message" in aOptions && aOptions.message != null)
- this.msg.text = aOptions.message;
-
- if ("buttons" in aOptions && aOptions.buttons != null)
- this.msg.buttons = aOptions.buttons;
-
- if ("doubleTapButton" in aOptions && aOptions.doubleTapButton != null)
- this.msg.doubleTapButton = aOptions.doubleTapButton;
-
- if ("hint" in aOptions && aOptions.hint != null)
- this.msg.hint = aOptions.hint;
-}
-
-Prompt.prototype = {
- setHint: function(aHint) {
- if (!aHint)
- delete this.msg.hint;
- else
- this.msg.hint = aHint;
- return this;
- },
-
- addButton: function(aOptions) {
- if (!this.msg.buttons)
- this.msg.buttons = [];
- this.msg.buttons.push(aOptions.label);
- return this;
- },
-
- _addInput: function(aOptions) {
- let obj = aOptions;
- if (this[aOptions.type + "_count"] === undefined)
- this[aOptions.type + "_count"] = 0;
-
- obj.id = aOptions.id || (aOptions.type + this[aOptions.type + "_count"]);
- this[aOptions.type + "_count"]++;
-
- if (!this.msg.inputs)
- this.msg.inputs = [];
- this.msg.inputs.push(obj);
- return this;
- },
-
- addCheckbox: function(aOptions) {
- return this._addInput({
- type: "checkbox",
- label: aOptions.label,
- checked: aOptions.checked,
- id: aOptions.id
- });
- },
-
- addTextbox: function(aOptions) {
- return this._addInput({
- type: "textbox",
- value: aOptions.value,
- hint: aOptions.hint,
- autofocus: aOptions.autofocus,
- id: aOptions.id
- });
- },
-
- addNumber: function(aOptions) {
- return this._addInput({
- type: "number",
- value: aOptions.value,
- hint: aOptions.hint,
- autofocus: aOptions.autofocus,
- id: aOptions.id
- });
- },
-
- addPassword: function(aOptions) {
- return this._addInput({
- type: "password",
- value: aOptions.value,
- hint: aOptions.hint,
- autofocus: aOptions.autofocus,
- id : aOptions.id
- });
- },
-
- addDatePicker: function(aOptions) {
- return this._addInput({
- type: aOptions.type || "date",
- value: aOptions.value,
- id: aOptions.id,
- max: aOptions.max,
- min: aOptions.min
- });
- },
-
- addColorPicker: function(aOptions) {
- return this._addInput({
- type: "color",
- value: aOptions.value,
- id: aOptions.id
- });
- },
-
- addLabel: function(aOptions) {
- return this._addInput({
- type: "label",
- label: aOptions.label,
- id: aOptions.id
- });
- },
-
- addMenulist: function(aOptions) {
- return this._addInput({
- type: "menulist",
- values: aOptions.values,
- id: aOptions.id
- });
- },
-
- addIconGrid: function(aOptions) {
- return this._addInput({
- type: "icongrid",
- items: aOptions.items,
- id: aOptions.id
- });
- },
-
- addTabs: function(aOptions) {
- return this._addInput({
- type: "tabs",
- items: aOptions.items,
- id: aOptions.id
- });
- },
-
- show: function(callback) {
- this.callback = callback;
- log("Sending message");
- this._innerShow();
- },
-
- _innerShow: function() {
- Messaging.sendRequestForResult(this.msg).then((data) => {
- if (this.callback)
- this.callback(data);
- });
- },
-
- _setListItems: function(aItems) {
- let hasSelected = false;
- this.msg.listitems = [];
-
- aItems.forEach(function(item) {
- let obj = { id: item.id };
-
- obj.label = item.label;
-
- if (item.disabled)
- obj.disabled = true;
-
- if (item.selected) {
- if (!this.msg.choiceMode) {
- this.msg.choiceMode = "single";
- }
- obj.selected = item.selected;
- }
-
- if (item.header)
- obj.isGroup = true;
-
- if (item.menu)
- obj.isParent = true;
-
- if (item.child)
- obj.inGroup = true;
-
- if (item.showAsActions)
- obj.showAsActions = item.showAsActions;
-
- if (item.icon)
- obj.icon = item.icon;
-
- this.msg.listitems.push(obj);
-
- }, this);
- return this;
- },
-
- setSingleChoiceItems: function(aItems) {
- return this._setListItems(aItems);
- },
-
- setMultiChoiceItems: function(aItems) {
- this.msg.choiceMode = "multiple";
- return this._setListItems(aItems);
- },
-
-}
diff --git a/mobile/android/modules/RuntimePermissions.jsm b/mobile/android/modules/RuntimePermissions.jsm
deleted file mode 100644
index 42d8024b1..000000000
--- a/mobile/android/modules/RuntimePermissions.jsm
+++ /dev/null
@@ -1,41 +0,0 @@
-/* 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;
-
-this.EXPORTED_SYMBOLS = ["RuntimePermissions"];
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Messaging", "resource://gre/modules/Messaging.jsm");
-
-// See: http://developer.android.com/reference/android/Manifest.permission.html
-const CAMERA = "android.permission.CAMERA";
-const WRITE_EXTERNAL_STORAGE = "android.permission.WRITE_EXTERNAL_STORAGE";
-const RECORD_AUDIO = "android.permission.RECORD_AUDIO";
-
-var RuntimePermissions = {
- CAMERA: CAMERA,
- RECORD_AUDIO: RECORD_AUDIO,
- WRITE_EXTERNAL_STORAGE: WRITE_EXTERNAL_STORAGE,
-
- /**
- * Check whether the permissions have been granted or not. If needed prompt the user to accept the permissions.
- *
- * @returns A promise resolving to true if all the permissions have been granted or false if any of the
- * permissions have been denied.
- */
- waitForPermissions: function(permission) {
- let permissions = [].concat(permission);
-
- let msg = {
- type: 'RuntimePermissions:Prompt',
- permissions: permissions
- };
-
- return Messaging.sendRequestForResult(msg);
- }
-}; \ No newline at end of file
diff --git a/mobile/android/modules/SSLExceptions.jsm b/mobile/android/modules/SSLExceptions.jsm
deleted file mode 100644
index 48dfe8d92..000000000
--- a/mobile/android/modules/SSLExceptions.jsm
+++ /dev/null
@@ -1,118 +0,0 @@
-/* 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"
-
-var Cc = Components.classes;
-var Ci = Components.interfaces;
-var Cu = Components.utils;
-
-Cu.import("resource://gre/modules/PrivateBrowsingUtils.jsm");
-
-this.EXPORTED_SYMBOLS = ["SSLExceptions"];
-
-/**
- A class to add exceptions to override SSL certificate problems. The functionality
- itself is borrowed from exceptionDialog.js.
-*/
-function SSLExceptions() {
- this._overrideService = Cc["@mozilla.org/security/certoverride;1"]
- .getService(Ci.nsICertOverrideService);
-}
-
-
-SSLExceptions.prototype = {
- _overrideService: null,
- _sslStatus: null,
-
- getInterface: function SSLE_getInterface(aIID) {
- return this.QueryInterface(aIID);
- },
- QueryInterface: function SSLE_QueryInterface(aIID) {
- if (aIID.equals(Ci.nsIBadCertListener2) ||
- aIID.equals(Ci.nsISupports))
- return this;
-
- throw Components.results.NS_ERROR_NO_INTERFACE;
- },
-
- /**
- To collect the SSL status we intercept the certificate error here
- and store the status for later use.
- */
- notifyCertProblem: function SSLE_notifyCertProblem(socketInfo, sslStatus, targetHost) {
- this._sslStatus = sslStatus.QueryInterface(Ci.nsISSLStatus);
- return true; // suppress error UI
- },
-
- /**
- Attempt to download the certificate for the location specified to get the SSLState
- for the certificate and the errors.
- */
- _checkCert: function SSLE_checkCert(aURI) {
- this._sslStatus = null;
-
- let req = Cc["@mozilla.org/xmlextras/xmlhttprequest;1"].createInstance(Ci.nsIXMLHttpRequest);
- try {
- if (aURI) {
- req.open("GET", aURI.prePath, false);
- req.channel.notificationCallbacks = this;
- req.send(null);
- }
- } catch (e) {
- // We *expect* exceptions if there are problems with the certificate
- // presented by the site. Log it, just in case, but we can proceed here,
- // with appropriate sanity checks
- Components.utils.reportError("Attempted to connect to a site with a bad certificate in the add exception dialog. " +
- "This results in a (mostly harmless) exception being thrown. " +
- "Logged for information purposes only: " + e);
- }
-
- return this._sslStatus;
- },
-
- /**
- Internal method to create an override.
- */
- _addOverride: function SSLE_addOverride(aURI, aWindow, aTemporary) {
- let SSLStatus = this._checkCert(aURI);
- let certificate = SSLStatus.serverCert;
-
- let flags = 0;
-
- // in private browsing do not store exceptions permanently ever
- if (PrivateBrowsingUtils.isWindowPrivate(aWindow)) {
- aTemporary = true;
- }
-
- if (SSLStatus.isUntrusted)
- flags |= this._overrideService.ERROR_UNTRUSTED;
- if (SSLStatus.isDomainMismatch)
- flags |= this._overrideService.ERROR_MISMATCH;
- if (SSLStatus.isNotValidAtThisTime)
- flags |= this._overrideService.ERROR_TIME;
-
- this._overrideService.rememberValidityOverride(
- aURI.asciiHost,
- aURI.port,
- certificate,
- flags,
- aTemporary);
- },
-
- /**
- Creates a permanent exception to override all overridable errors for
- the given URL.
- */
- addPermanentException: function SSLE_addPermanentException(aURI, aWindow) {
- this._addOverride(aURI, aWindow, false);
- },
-
- /**
- Creates a temporary exception to override all overridable errors for
- the given URL.
- */
- addTemporaryException: function SSLE_addTemporaryException(aURI, aWindow) {
- this._addOverride(aURI, aWindow, true);
- }
-};
diff --git a/mobile/android/modules/Sanitizer.jsm b/mobile/android/modules/Sanitizer.jsm
deleted file mode 100644
index 014a89688..000000000
--- a/mobile/android/modules/Sanitizer.jsm
+++ /dev/null
@@ -1,303 +0,0 @@
-// -*- indent-tabs-mode: nil; js-indent-level: 4 -*-
-/* 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/. */
-
-/*globals LoadContextInfo, FormHistory, Accounts */
-
-var Cc = Components.classes;
-var Ci = Components.interfaces;
-var Cu = Components.utils;
-
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-Cu.import("resource://gre/modules/LoadContextInfo.jsm");
-Cu.import("resource://gre/modules/FormHistory.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-Cu.import("resource://gre/modules/Task.jsm");
-Cu.import("resource://gre/modules/Downloads.jsm");
-Cu.import("resource://gre/modules/osfile.jsm");
-Cu.import("resource://gre/modules/Accounts.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "DownloadIntegration",
- "resource://gre/modules/DownloadIntegration.jsm");
-
-function dump(a) {
- Services.console.logStringMessage(a);
-}
-
-this.EXPORTED_SYMBOLS = ["Sanitizer"];
-
-function Sanitizer() {}
-Sanitizer.prototype = {
- clearItem: function (aItemName)
- {
- let item = this.items[aItemName];
- let canClear = item.canClear;
- if (typeof canClear == "function") {
- canClear(function clearCallback(aCanClear) {
- if (aCanClear)
- item.clear();
- });
- } else if (canClear) {
- item.clear();
- }
- },
-
- items: {
- cache: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- var cache = Cc["@mozilla.org/netwerk/cache-storage-service;1"].getService(Ci.nsICacheStorageService);
- try {
- cache.clear();
- } catch(er) {}
-
- let imageCache = Cc["@mozilla.org/image/tools;1"].getService(Ci.imgITools)
- .getImgCacheForDocument(null);
- try {
- imageCache.clearCache(false); // true=chrome, false=content
- } catch(er) {}
-
- resolve();
- });
- },
-
- get canClear()
- {
- return true;
- }
- },
-
- cookies: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- Services.cookies.removeAll();
- resolve();
- });
- },
-
- get canClear()
- {
- return true;
- }
- },
-
- siteSettings: {
- clear: Task.async(function* () {
- // Clear site-specific permissions like "Allow this site to open popups"
- Services.perms.removeAll();
-
- // Clear site-specific settings like page-zoom level
- Cc["@mozilla.org/content-pref/service;1"]
- .getService(Ci.nsIContentPrefService2)
- .removeAllDomains(null);
-
- // Clear site security settings
- var sss = Cc["@mozilla.org/ssservice;1"]
- .getService(Ci.nsISiteSecurityService);
- sss.clearAll();
-
- // Clear push subscriptions
- yield new Promise((resolve, reject) => {
- let push = Cc["@mozilla.org/push/Service;1"]
- .getService(Ci.nsIPushService);
- push.clearForDomain("*", status => {
- if (Components.isSuccessCode(status)) {
- resolve();
- } else {
- reject(new Error("Error clearing push subscriptions: " +
- status));
- }
- });
- });
- }),
-
- get canClear()
- {
- return true;
- }
- },
-
- offlineApps: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- var cacheService = Cc["@mozilla.org/netwerk/cache-storage-service;1"].getService(Ci.nsICacheStorageService);
- var appCacheStorage = cacheService.appCacheStorage(LoadContextInfo.default, null);
- try {
- appCacheStorage.asyncEvictStorage(null);
- } catch(er) {}
-
- resolve();
- });
- },
-
- get canClear()
- {
- return true;
- }
- },
-
- history: {
- clear: function ()
- {
- return Messaging.sendRequestForResult({ type: "Sanitize:ClearHistory" })
- .catch(e => Cu.reportError("Java-side history clearing failed: " + e))
- .then(function() {
- try {
- Services.obs.notifyObservers(null, "browser:purge-session-history", "");
- }
- catch (e) { }
-
- try {
- var predictor = Cc["@mozilla.org/network/predictor;1"].getService(Ci.nsINetworkPredictor);
- predictor.reset();
- } catch (e) { }
- });
- },
-
- get canClear()
- {
- // bug 347231: Always allow clearing history due to dependencies on
- // the browser:purge-session-history notification. (like error console)
- return true;
- }
- },
-
- searchHistory: {
- clear: function ()
- {
- return Messaging.sendRequestForResult({ type: "Sanitize:ClearHistory", clearSearchHistory: true })
- .catch(e => Cu.reportError("Java-side search history clearing failed: " + e))
- },
-
- get canClear()
- {
- return true;
- }
- },
-
- formdata: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- FormHistory.update({ op: "remove" });
- resolve();
- });
- },
-
- canClear: function (aCallback)
- {
- let count = 0;
- let countDone = {
- handleResult: function(aResult) { count = aResult; },
- handleError: function(aError) { Cu.reportError(aError); },
- handleCompletion: function(aReason) { aCallback(aReason == 0 && count > 0); }
- };
- FormHistory.count({}, countDone);
- }
- },
-
- downloadFiles: {
- clear: Task.async(function* () {
- let list = yield Downloads.getList(Downloads.ALL);
- let downloads = yield list.getAll();
- var finalizePromises = [];
-
- // Logic copied from DownloadList.removeFinished. Ideally, we would
- // just use that method directly, but we want to be able to remove the
- // downloaded files as well.
- for (let download of downloads) {
- // Remove downloads that have been canceled, even if the cancellation
- // operation hasn't completed yet so we don't check "stopped" here.
- // Failed downloads with partial data are also removed.
- if (download.stopped && (!download.hasPartialData || download.error)) {
- // Remove the download first, so that the views don't get the change
- // notifications that may occur during finalization.
- yield list.remove(download);
- // Ensure that the download is stopped and no partial data is kept.
- // This works even if the download state has changed meanwhile. We
- // don't need to wait for the procedure to be complete before
- // processing the other downloads in the list.
- finalizePromises.push(download.finalize(true).then(() => null, Cu.reportError));
-
- // Delete the downloaded files themselves.
- OS.File.remove(download.target.path).then(() => null, ex => {
- if (!(ex instanceof OS.File.Error && ex.becauseNoSuchFile)) {
- Cu.reportError(ex);
- }
- });
- }
- }
-
- yield Promise.all(finalizePromises);
- yield DownloadIntegration.forceSave();
- }),
-
- get canClear()
- {
- return true;
- }
- },
-
- passwords: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- Services.logins.removeAllLogins();
- resolve();
- });
- },
-
- get canClear()
- {
- let count = Services.logins.countLogins("", "", ""); // count all logins
- return (count > 0);
- }
- },
-
- sessions: {
- clear: function ()
- {
- return new Promise(function(resolve, reject) {
- // clear all auth tokens
- var sdr = Cc["@mozilla.org/security/sdr;1"].getService(Ci.nsISecretDecoderRing);
- sdr.logoutAndTeardown();
-
- // clear FTP and plain HTTP auth sessions
- Services.obs.notifyObservers(null, "net:clear-active-logins", null);
-
- resolve();
- });
- },
-
- get canClear()
- {
- return true;
- }
- },
-
- syncedTabs: {
- clear: function ()
- {
- return Messaging.sendRequestForResult({ type: "Sanitize:ClearSyncedTabs" })
- .catch(e => Cu.reportError("Java-side synced tabs clearing failed: " + e));
- },
-
- canClear: function(aCallback)
- {
- Accounts.anySyncAccountsExist().then(aCallback)
- .catch(function(err) {
- Cu.reportError("Java-side synced tabs clearing failed: " + err)
- aCallback(false);
- });
- }
- }
-
- }
-};
-
-this.Sanitizer = new Sanitizer();
diff --git a/mobile/android/modules/SharedPreferences.jsm b/mobile/android/modules/SharedPreferences.jsm
deleted file mode 100644
index 3f32df6ea..000000000
--- a/mobile/android/modules/SharedPreferences.jsm
+++ /dev/null
@@ -1,254 +0,0 @@
-// -*- indent-tabs-mode: nil; js-indent-level: 2 -*-
-/* 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";
-
-this.EXPORTED_SYMBOLS = ["SharedPreferences"];
-
-const { classes: Cc, interfaces: Ci, utils: Cu } = Components;
-
-// For adding observers.
-Cu.import("resource://gre/modules/Services.jsm");
-Cu.import("resource://gre/modules/Messaging.jsm");
-
-var Scope = Object.freeze({
- APP: "app",
- PROFILE: "profile",
- GLOBAL: "global"
-});
-
-/**
- * Public API to getting a SharedPreferencesImpl instance. These scopes mirror GeckoSharedPrefs.
- */
-var SharedPreferences = {
- forApp: function() {
- return new SharedPreferencesImpl({ scope: Scope.APP });
- },
-
- forProfile: function() {
- return new SharedPreferencesImpl({ scope: Scope.PROFILE });
- },
-
- /**
- * Get SharedPreferences for the named profile; if the profile name is null,
- * returns the preferences for the current profile (just like |forProfile|).
- */
- forProfileName: function(profileName) {
- return new SharedPreferencesImpl({ scope: Scope.PROFILE, profileName: profileName });
- },
-
- /**
- * Get SharedPreferences for the given Android branch; if the branch is null,
- * returns the default preferences branch for the application, which is the
- * output of |PreferenceManager.getDefaultSharedPreferences|.
- */
- forAndroid: function(branch) {
- return new SharedPreferencesImpl({ scope: Scope.GLOBAL, branch: branch });
- }
-};
-
-/**
- * Create an interface to an Android SharedPreferences branch.
- *
- * options {Object} with the following valid keys:
- * - scope {String} (required) specifies the scope of preferences that should be accessed.
- * - branch {String} (only when using Scope.GLOBAL) should be a string describing a preferences branch,
- * like "UpdateService" or "background.data", or null to access the
- * default preferences branch for the application.
- * - profileName {String} (optional, only valid when using Scope.PROFILE)
- */
-function SharedPreferencesImpl(options = {}) {
- if (!(this instanceof SharedPreferencesImpl)) {
- return new SharedPreferencesImpl(options);
- }
-
- if (options.scope == null || options.scope == undefined) {
- throw "Shared Preferences must specifiy a scope.";
- }
-
- this._scope = options.scope;
- this._profileName = options.profileName;
- this._branch = options.branch;
- this._observers = {};
-}
-
-SharedPreferencesImpl.prototype = Object.freeze({
- _set: function _set(prefs) {
- Messaging.sendRequest({
- type: "SharedPreferences:Set",
- preferences: prefs,
- scope: this._scope,
- profileName: this._profileName,
- branch: this._branch,
- });
- },
-
- _setOne: function _setOne(prefName, value, type) {
- let prefs = [];
- prefs.push({
- name: prefName,
- value: value,
- type: type,
- });
- this._set(prefs);
- },
-
- setBoolPref: function setBoolPref(prefName, value) {
- this._setOne(prefName, value, "bool");
- },
-
- setCharPref: function setCharPref(prefName, value) {
- this._setOne(prefName, value, "string");
- },
-
- setIntPref: function setIntPref(prefName, value) {
- this._setOne(prefName, value, "int");
- },
-
- _get: function _get(prefs, callback) {
- let result = null;
- Messaging.sendRequestForResult({
- type: "SharedPreferences:Get",
- preferences: prefs,
- scope: this._scope,
- profileName: this._profileName,
- branch: this._branch,
- }).then((data) => {
- result = data.values;
- });
-
- let thread = Services.tm.currentThread;
- while (result == null)
- thread.processNextEvent(true);
-
- return result;
- },
-
- _getOne: function _getOne(prefName, type) {
- let prefs = [];
- prefs.push({
- name: prefName,
- type: type,
- });
- let values = this._get(prefs);
- if (values.length != 1) {
- throw new Error("Got too many values: " + values.length);
- }
- return values[0].value;
- },
-
- getBoolPref: function getBoolPref(prefName) {
- return this._getOne(prefName, "bool");
- },
-
- getCharPref: function getCharPref(prefName) {
- return this._getOne(prefName, "string");
- },
-
- getIntPref: function getIntPref(prefName) {
- return this._getOne(prefName, "int");
- },
-
- /**
- * Invoke `observer` after a change to the preference `domain` in
- * the current branch.
- *
- * `observer` should implement the nsIObserver.observe interface.
- */
- addObserver: function addObserver(domain, observer, holdWeak) {
- if (!domain)
- throw new Error("domain must not be null");
- if (!observer)
- throw new Error("observer must not be null");
- if (holdWeak)
- throw new Error("Weak references not yet implemented.");
-
- if (!this._observers.hasOwnProperty(domain))
- this._observers[domain] = [];
- if (this._observers[domain].indexOf(observer) > -1)
- return;
-
- this._observers[domain].push(observer);
-
- this._updateAndroidListener();
- },
-
- /**
- * Do not invoke `observer` after a change to the preference
- * `domain` in the current branch.
- */
- removeObserver: function removeObserver(domain, observer) {
- if (!this._observers.hasOwnProperty(domain))
- return;
- let index = this._observers[domain].indexOf(observer);
- if (index < 0)
- return;
-
- this._observers[domain].splice(index, 1);
- if (this._observers[domain].length < 1)
- delete this._observers[domain];
-
- this._updateAndroidListener();
- },
-
- _updateAndroidListener: function _updateAndroidListener() {
- if (this._listening && Object.keys(this._observers).length < 1)
- this._uninstallAndroidListener();
- if (!this._listening && Object.keys(this._observers).length > 0)
- this._installAndroidListener();
- },
-
- _installAndroidListener: function _installAndroidListener() {
- if (this._listening)
- return;
- this._listening = true;
-
- Services.obs.addObserver(this, "SharedPreferences:Changed", false);
- Messaging.sendRequest({
- type: "SharedPreferences:Observe",
- enable: true,
- scope: this._scope,
- profileName: this._profileName,
- branch: this._branch,
- });
- },
-
- observe: function observe(subject, topic, data) {
- if (topic != "SharedPreferences:Changed") {
- return;
- }
-
- let msg = JSON.parse(data);
- if (msg.scope !== this._scope ||
- ((this._scope === Scope.PROFILE) && (msg.profileName !== this._profileName)) ||
- ((this._scope === Scope.GLOBAL) && (msg.branch !== this._branch))) {
- return;
- }
-
- if (!this._observers.hasOwnProperty(msg.key)) {
- return;
- }
-
- let observers = this._observers[msg.key];
- for (let obs of observers) {
- obs.observe(obs, msg.key, msg.value);
- }
- },
-
- _uninstallAndroidListener: function _uninstallAndroidListener() {
- if (!this._listening)
- return;
- this._listening = false;
-
- Services.obs.removeObserver(this, "SharedPreferences:Changed");
- Messaging.sendRequest({
- type: "SharedPreferences:Observe",
- enable: false,
- scope: this._scope,
- profileName: this._profileName,
- branch: this._branch,
- });
- },
-});
diff --git a/mobile/android/modules/Snackbars.jsm b/mobile/android/modules/Snackbars.jsm
deleted file mode 100644
index 066a28c56..000000000
--- a/mobile/android/modules/Snackbars.jsm
+++ /dev/null
@@ -1,72 +0,0 @@
-/* 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;
-
-this.EXPORTED_SYMBOLS = ["Snackbars"];
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Messaging", "resource://gre/modules/Messaging.jsm");
-
-const LENGTH_INDEFINITE = -2;
-const LENGTH_LONG = 0;
-const LENGTH_SHORT = -1;
-
-var Snackbars = {
- LENGTH_INDEFINITE: LENGTH_INDEFINITE,
- LENGTH_LONG: LENGTH_LONG,
- LENGTH_SHORT: LENGTH_SHORT,
-
- show: function(aMessage, aDuration, aOptions) {
-
- // Takes care of the deprecated toast calls
- if (typeof aDuration === "string") {
- [aDuration, aOptions] = migrateToastIfNeeded(aDuration, aOptions);
- }
-
- let msg = {
- type: 'Snackbar:Show',
- message: aMessage,
- duration: aDuration,
- };
-
- if (aOptions && aOptions.backgroundColor) {
- msg.backgroundColor = aOptions.backgroundColor;
- }
-
- if (aOptions && aOptions.action) {
- msg.action = {};
-
- if (aOptions.action.label) {
- msg.action.label = aOptions.action.label;
- }
-
- Messaging.sendRequestForResult(msg).then(result => aOptions.action.callback());
- } else {
- Messaging.sendRequest(msg);
- }
- }
-};
-
-function migrateToastIfNeeded(aDuration, aOptions) {
- let duration;
- if (aDuration === "long") {
- duration = LENGTH_LONG;
- }
- else {
- duration = LENGTH_SHORT;
- }
-
- let options = {};
- if (aOptions && aOptions.button) {
- options.action = {
- label: aOptions.button.label,
- callback: () => aOptions.button.callback(),
- };
- }
- return [duration, options];
-} \ No newline at end of file
diff --git a/mobile/android/modules/TabMirror.jsm b/mobile/android/modules/TabMirror.jsm
deleted file mode 100644
index 72a640ec8..000000000
--- a/mobile/android/modules/TabMirror.jsm
+++ /dev/null
@@ -1,153 +0,0 @@
-/* -*- Mode: js; js-indent-level: 2; indent-tabs-mode: nil; tab-width: 2 -*- */
-/* 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/Messaging.jsm");
-
-const CONFIG = { iceServers: [{ "urls": ["stun:stun.services.mozilla.com"] }] };
-
-var log = Cu.import("resource://gre/modules/AndroidLog.jsm",
- {}).AndroidLog.d.bind(null, "TabMirror");
-
-var failure = function(x) {
- log("ERROR: " + JSON.stringify(x));
-};
-
-var TabMirror = function(deviceId, window) {
-
- this.deviceId = deviceId;
- // Save RTCSessionDescription and RTCIceCandidate for later when the window object is not available.
- this.RTCSessionDescription = window.RTCSessionDescription;
- this.RTCIceCandidate = window.RTCIceCandidate;
-
- Services.obs.addObserver((aSubject, aTopic, aData) => this._processMessage(aData), "MediaPlayer:Response", false);
- this._sendMessage({ start: true });
- this._window = window;
- this._pc = new window.RTCPeerConnection(CONFIG, {});
- if (!this._pc) {
- throw "Failure creating Webrtc object";
- }
-
-};
-
-TabMirror.prototype = {
- _window: null,
- _screenSize: { width: 1280, height: 720 },
- _pc: null,
- _start: function() {
- this._pc.onicecandidate = this._onIceCandidate.bind(this);
-
- let windowId = this._window.BrowserApp.selectedBrowser.contentWindow.QueryInterface(Ci.nsIInterfaceRequestor).getInterface(Ci.nsIDOMWindowUtils).outerWindowID;
- let constraints = {
- video: {
- mediaSource: "browser",
- browserWindow: windowId,
- scrollWithPage: true,
- advanced: [
- { width: { min: 0, max: this._screenSize.width },
- height: { min: 0, max: this._screenSize.height }
- },
- { aspectRatio: this._screenSize.width / this._screenSize.height }
- ]
- }
- };
-
- this._window.navigator.mozGetUserMedia(constraints, this._onGumSuccess.bind(this), this._onGumFailure.bind(this));
- },
-
- _processMessage: function(data) {
- if (!data) {
- return;
- }
-
- let msg = JSON.parse(data);
-
- if (!msg) {
- return;
- }
-
- if (msg.sdp && msg.type === "answer") {
- this._processAnswer(msg);
- } else if (msg.type == "size") {
- if (msg.height) {
- this._screenSize.height = msg.height;
- }
- if (msg.width) {
- this._screenSize.width = msg.width;
- }
- this._start();
- } else if (msg.candidate) {
- this._processIceCandidate(msg);
- } else {
- log("dropping unrecognized message: " + JSON.stringify(msg));
- }
- },
-
- // Signaling methods
- _processAnswer: function(msg) {
- this._pc.setRemoteDescription(new this.RTCSessionDescription(msg),
- this._setRemoteAnswerSuccess.bind(this), failure);
- },
-
- _processIceCandidate: function(msg) {
- // WebRTC generates a warning if the success and fail callbacks are not passed in.
- this._pc.addIceCandidate(new this.RTCIceCandidate(msg), () => log("Ice Candiated added successfuly"), () => log("Failed to add Ice Candidate"));
- },
-
- _setRemoteAnswerSuccess: function() {
- },
-
- _setLocalSuccessOffer: function(sdp) {
- this._sendMessage(sdp);
- },
-
- _createOfferSuccess: function(sdp) {
- this._pc.setLocalDescription(sdp, () => this._setLocalSuccessOffer(sdp), failure);
- },
-
- _onIceCandidate: function (msg) {
- log("NEW Ice Candidate: " + JSON.stringify(msg.candidate));
- this._sendMessage(msg.candidate);
- },
-
- _ready: function() {
- this._pc.createOffer(this._createOfferSuccess.bind(this), failure);
- },
-
- _onGumSuccess: function(stream){
- this._pc.addStream(stream);
- this._ready();
- },
-
- _onGumFailure: function() {
- log("Could not get video stream");
- this._pc.close();
- },
-
- _sendMessage: function(msg) {
- if (this.deviceId) {
- let obj = {
- type: "MediaPlayer:Message",
- id: this.deviceId,
- data: JSON.stringify(msg)
- };
- Messaging.sendRequest(obj);
- }
- },
-
- stop: function() {
- if (this.deviceId) {
- let obj = {
- type: "MediaPlayer:End",
- id: this.deviceId
- };
- Services.androidBridge.handleGeckoMessage(obj);
- }
- },
-};
-
-
-this.EXPORTED_SYMBOLS = ["TabMirror"];
diff --git a/mobile/android/modules/WebsiteMetadata.jsm b/mobile/android/modules/WebsiteMetadata.jsm
deleted file mode 100644
index 39af9ddeb..000000000
--- a/mobile/android/modules/WebsiteMetadata.jsm
+++ /dev/null
@@ -1,475 +0,0 @@
-/* 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;
-
-this.EXPORTED_SYMBOLS = ["WebsiteMetadata"];
-
-Cu.import("resource://gre/modules/XPCOMUtils.jsm");
-
-XPCOMUtils.defineLazyModuleGetter(this, "Messaging", "resource://gre/modules/Messaging.jsm");
-XPCOMUtils.defineLazyModuleGetter(this, "Task", "resource://gre/modules/Task.jsm");
-
-var WebsiteMetadata = {
- /**
- * Asynchronously parse the document extract metadata. A 'Website:Metadata' event with the metadata
- * will be sent.
- */
- parseAsynchronously: function(doc) {
- Task.spawn(function() {
- let metadata = getMetadata(doc, doc.location.href, {
- image_url: metadataRules['image_url']
- });
-
- // No metadata was extracted, so don't bother sending it.
- if (Object.keys(metadata).length === 0) {
- return;
- }
-
- let msg = {
- type: 'Website:Metadata',
- location: doc.location.href,
- metadata: metadata,
- };
-
- Messaging.sendRequest(msg);
- });
- }
-};
-
-// #################################################################################################
-// # Modified version of makeUrlAbsolute() to not import url parser library (and dependencies)
-// #################################################################################################
-
-function makeUrlAbsolute(context, relative) {
- var a = context.doc.createElement('a');
- a.href = relative;
- return a.href;
-}
-
-// #################################################################################################
-// # page-metadata-parser
-// # https://github.com/mozilla/page-metadata-parser/
-// # 61c58cbd0f0bf2153df832a388a79c66b288b98c
-// #################################################################################################
-
-function buildRuleset(name, rules, processors) {
- const reversedRules = Array.from(rules).reverse();
- const builtRuleset = ruleset(...reversedRules.map(([query, handler], order) => rule(
- dom(query),
- node => [{
- score: order,
- flavor: name,
- notes: handler(node),
- }]
- )));
-
- return (doc, context) => {
- const kb = builtRuleset.score(doc);
- const maxNode = kb.max(name);
-
- if (maxNode) {
- let value = maxNode.flavors.get(name);
-
- if (processors) {
- processors.forEach(processor => {
- value = processor(value, context);
- });
- }
-
- if (value) {
- if (value.trim) {
- return value.trim();
- }
- return value;
- }
- }
- };
-}
-
-const metadataRules = {
- description: {
- rules: [
- ['meta[property="og:description"]', node => node.element.getAttribute('content')],
- ['meta[name="description"]', node => node.element.getAttribute('content')],
- ],
- },
-
- icon_url: {
- rules: [
- ['link[rel="apple-touch-icon"]', node => node.element.getAttribute('href')],
- ['link[rel="apple-touch-icon-precomposed"]', node => node.element.getAttribute('href')],
- ['link[rel="icon"]', node => node.element.getAttribute('href')],
- ['link[rel="fluid-icon"]', node => node.element.getAttribute('href')],
- ['link[rel="shortcut icon"]', node => node.element.getAttribute('href')],
- ['link[rel="Shortcut Icon"]', node => node.element.getAttribute('href')],
- ['link[rel="mask-icon"]', node => node.element.getAttribute('href')],
- ],
- processors: [
- (icon_url, context) => makeUrlAbsolute(context, icon_url)
- ]
- },
-
- image_url: {
- rules: [
- ['meta[property="og:image:secure_url"]', node => node.element.getAttribute('content')],
- ['meta[property="og:image:url"]', node => node.element.getAttribute('content')],
- ['meta[property="og:image"]', node => node.element.getAttribute('content')],
- ['meta[property="twitter:image"]', node => node.element.getAttribute('content')],
- ['meta[name="thumbnail"]', node => node.element.getAttribute('content')],
- ],
- processors: [
- (image_url, context) => makeUrlAbsolute(context, image_url)
- ],
- },
-
- keywords: {
- rules: [
- ['meta[name="keywords"]', node => node.element.getAttribute('content')],
- ],
- processors: [
- (keywords) => keywords.split(',').map((keyword) => keyword.trim()),
- ]
- },
-
- title: {
- rules: [
- ['meta[property="og:title"]', node => node.element.getAttribute('content')],
- ['meta[property="twitter:title"]', node => node.element.getAttribute('content')],
- ['meta[name="hdl"]', node => node.element.getAttribute('content')],
- ['title', node => node.element.text],
- ],
- },
-
- type: {
- rules: [
- ['meta[property="og:type"]', node => node.element.getAttribute('content')],
- ],
- },
-
- url: {
- rules: [
- ['meta[property="og:url"]', node => node.element.getAttribute('content')],
- ['link[rel="canonical"]', node => node.element.getAttribute('href')],
- ],
- },
-};
-
-function getMetadata(doc, url, rules) {
- const metadata = {};
- const context = {url,doc};
- const ruleSet = rules || metadataRules;
-
- Object.keys(ruleSet).map(metadataKey => {
- const metadataRule = ruleSet[metadataKey];
-
- if(Array.isArray(metadataRule.rules)) {
- const builtRule = buildRuleset(metadataKey, metadataRule.rules, metadataRule.processors);
- metadata[metadataKey] = builtRule(doc, context);
- } else {
- metadata[metadataKey] = getMetadata(doc, url, metadataRule);
- }
- });
-
- return metadata;
-}
-
-// #################################################################################################
-// # Fathom dependencies resolved
-// #################################################################################################
-
-// const {forEach} = require('wu');
-function forEach(fn, obj) {
- for (let x of obj) {
- fn(x);
- }
-}
-
-function best(iterable, by, isBetter) {
- let bestSoFar, bestKeySoFar;
- let isFirst = true;
- forEach(
- function (item) {
- const key = by(item);
- if (isBetter(key, bestKeySoFar) || isFirst) {
- bestSoFar = item;
- bestKeySoFar = key;
- isFirst = false;
- }
- },
- iterable);
- if (isFirst) {
- throw new Error('Tried to call best() on empty iterable');
- }
- return bestSoFar;
-}
-
-// const {max} = require('./utils');
-function max(iterable, by = identity) {
- return best(iterable, by, (a, b) => a > b);
-}
-
-// #################################################################################################
-// # Fathom
-// # https://github.com/mozilla/fathom
-// # cac59e470816f17fc1efd4a34437b585e3e451cd
-// #################################################################################################
-
-// Get a key of a map, first setting it to a default value if it's missing.
-function getDefault(map, key, defaultMaker) {
- if (map.has(key)) {
- return map.get(key);
- }
- const defaultValue = defaultMaker();
- map.set(key, defaultValue);
- return defaultValue;
-}
-
-
-// Construct a filtration network of rules.
-function ruleset(...rules) {
- const rulesByInputFlavor = new Map(); // [someInputFlavor: [rule, ...]]
-
- // File each rule under its input flavor:
- forEach(rule => getDefault(rulesByInputFlavor, rule.source.inputFlavor, () => []).push(rule),
- rules);
-
- return {
- // Iterate over a DOM tree or subtree, building up a knowledgebase, a
- // data structure holding scores and annotations for interesting
- // elements. Return the knowledgebase.
- //
- // This is the "rank" portion of the rank-and-yank algorithm.
- score: function (tree) {
- const kb = knowledgebase();
-
- // Introduce the whole DOM into the KB as flavor 'dom' to get
- // things started:
- const nonterminals = [[{tree}, 'dom']]; // [[node, flavor], [node, flavor], ...]
-
- // While there are new facts, run the applicable rules over them to
- // generate even newer facts. Repeat until everything's fully
- // digested. Rules run in no particular guaranteed order.
- while (nonterminals.length) {
- const [inNode, inFlavor] = nonterminals.pop();
- for (let rule of getDefault(rulesByInputFlavor, inFlavor, () => [])) {
- const outFacts = resultsOf(rule, inNode, inFlavor, kb);
- for (let fact of outFacts) {
- const outNode = kb.nodeForElement(fact.element);
-
- // No matter whether or not this flavor has been
- // emitted before for this node, we multiply the score.
- // We want to be able to add rules that refine the
- // scoring of a node, without having to rewire the path
- // of flavors that winds through the ruleset.
- //
- // 1 score per Node is plenty. That simplifies our
- // data, our rankers, our flavor system (since we don't
- // need to represent score axes), and our engine. If
- // somebody wants more score axes, they can fake it
- // themselves with notes, thus paying only for what
- // they eat. (We can even provide functions that help
- // with that.) Most rulesets will probably be concerned
- // with scoring only 1 thing at a time anyway. So,
- // rankers return a score multiplier + 0 or more new
- // flavors with optional notes. Facts can never be
- // deleted from the KB by rankers (or order would start
- // to matter); after all, they're *facts*.
- outNode.score *= fact.score;
-
- // Add a new annotation to a node--but only if there
- // wasn't already one of the given flavor already
- // there; otherwise there's no point.
- //
- // You might argue that we might want to modify an
- // existing note here, but that would be a bad
- // idea. Notes of a given flavor should be
- // considered immutable once laid down. Otherwise, the
- // order of execution of same-flavored rules could
- // matter, hurting pluggability. Emit a new flavor and
- // a new note if you want to do that.
- //
- // Also, choosing not to add a new fact to nonterminals
- // when we're not adding a new flavor saves the work of
- // running the rules against it, which would be
- // entirely redundant and perform no new work (unless
- // the rankers were nondeterministic, but don't do
- // that).
- if (!outNode.flavors.has(fact.flavor)) {
- outNode.flavors.set(fact.flavor, fact.notes);
- kb.indexNodeByFlavor(outNode, fact.flavor); // TODO: better encapsulation rather than indexing explicitly
- nonterminals.push([outNode, fact.flavor]);
- }
- }
- }
- }
- return kb;
- }
- };
-}
-
-
-// Construct a container for storing and querying facts, where a fact has a
-// flavor (used to dispatch further rules upon), a corresponding DOM element, a
-// score, and some other arbitrary notes opaque to fathom.
-function knowledgebase() {
- const nodesByFlavor = new Map(); // Map{'texty' -> [NodeA],
- // 'spiffy' -> [NodeA, NodeB]}
- // NodeA = {element: <someElement>,
- //
- // // Global nodewide score. Add
- // // custom ones with notes if
- // // you want.
- // score: 8,
- //
- // // Flavors is a map of flavor names to notes:
- // flavors: Map{'texty' -> {ownText: 'blah',
- // someOtherNote: 'foo',
- // someCustomScore: 10},
- // // This is an empty note:
- // 'fluffy' -> undefined}}
- const nodesByElement = new Map();
-
- return {
- // Return the "node" (our own data structure that we control) that
- // corresponds to a given DOM element, creating one if necessary.
- nodeForElement: function (element) {
- return getDefault(nodesByElement,
- element,
- () => ({element,
- score: 1,
- flavors: new Map()}));
- },
-
- // Return the highest-scored node of the given flavor, undefined if
- // there is none.
- max: function (flavor) {
- const nodes = nodesByFlavor.get(flavor);
- return nodes === undefined ? undefined : max(nodes, node => node.score);
- },
-
- // Let the KB know that a new flavor has been added to an element.
- indexNodeByFlavor: function (node, flavor) {
- getDefault(nodesByFlavor, flavor, () => []).push(node);
- },
-
- nodesOfFlavor: function (flavor) {
- return getDefault(nodesByFlavor, flavor, () => []);
- }
- };
-}
-
-
-// Apply a rule (as returned by a call to rule()) to a fact, and return the
-// new facts that result.
-function resultsOf(rule, node, flavor, kb) {
- // If more types of rule pop up someday, do fancier dispatching here.
- return rule.source.flavor === 'flavor' ? resultsOfFlavorRule(rule, node, flavor) : resultsOfDomRule(rule, node, kb);
-}
-
-
-// Pull the DOM tree off the special property of the root "dom" fact, and query
-// against it.
-function *resultsOfDomRule(rule, specialDomNode, kb) {
- // Use the special "tree" property of the special starting node:
- const matches = specialDomNode.tree.querySelectorAll(rule.source.selector);
-
- for (let i = 0; i < matches.length; i++) { // matches is a NodeList, which doesn't conform to iterator protocol
- const element = matches[i];
- const newFacts = explicitFacts(rule.ranker(kb.nodeForElement(element)));
- for (let fact of newFacts) {
- if (fact.element === undefined) {
- fact.element = element;
- }
- if (fact.flavor === undefined) {
- throw new Error('Rankers of dom() rules must return a flavor in each fact. Otherwise, there is no way for that fact to be used later.');
- }
- yield fact;
- }
- }
-}
-
-
-function *resultsOfFlavorRule(rule, node, flavor) {
- const newFacts = explicitFacts(rule.ranker(node));
-
- for (let fact of newFacts) {
- // If the ranker didn't specify a different element, assume it's
- // talking about the one we passed in:
- if (fact.element === undefined) {
- fact.element = node.element;
- }
- if (fact.flavor === undefined) {
- fact.flavor = flavor;
- }
- yield fact;
- }
-}
-
-
-// Take the possibly abbreviated output of a ranker function, and make it
-// explicitly an iterable with a defined score.
-//
-// Rankers can return undefined, which means "no facts", a single fact, or an
-// array of facts.
-function *explicitFacts(rankerResult) {
- const array = (rankerResult === undefined) ? [] : (Array.isArray(rankerResult) ? rankerResult : [rankerResult]);
- for (let fact of array) {
- if (fact.score === undefined) {
- fact.score = 1;
- }
- yield fact;
- }
-}
-
-
-// TODO: For the moment, a lot of responsibility is on the rankers to return a
-// pretty big data structure of up to 4 properties. This is a bit verbose for
-// an arrow function (as I hope we can use most of the time) and the usual case
-// will probably be returning just a score multiplier. Make that case more
-// concise.
-
-// TODO: It is likely that rankers should receive the notes of their input type
-// as a 2nd arg, for brevity.
-
-
-// Return a condition that uses a DOM selector to find its matches from the
-// original DOM tree.
-//
-// For consistency, Nodes will still be delivered to the transformers, but
-// they'll have empty flavors and score = 1.
-//
-// Condition constructors like dom() and flavor() build stupid, introspectable
-// objects that the query engine can read. They don't actually do the query
-// themselves. That way, the query planner can be smarter than them, figuring
-// out which indices to use based on all of them. (We'll probably keep a heap
-// by each dimension's score and a hash by flavor name, for starters.) Someday,
-// fancy things like this may be possible: rule(and(tag('p'), klass('snork')),
-// ...)
-function dom(selector) {
- return {
- flavor: 'dom',
- inputFlavor: 'dom',
- selector
- };
-}
-
-
-// Return a condition that discriminates on nodes of the knowledgebase by flavor.
-function flavor(inputFlavor) {
- return {
- flavor: 'flavor',
- inputFlavor
- };
-}
-
-
-function rule(source, ranker) {
- return {
- source,
- ranker
- };
-}
diff --git a/mobile/android/modules/dbg-browser-actors.js b/mobile/android/modules/dbg-browser-actors.js
deleted file mode 100644
index f96a39151..000000000
--- a/mobile/android/modules/dbg-browser-actors.js
+++ /dev/null
@@ -1,77 +0,0 @@
-/* -*- indent-tabs-mode: nil; js-indent-level: 2 -*- */
-/* 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";
-/**
- * Fennec-specific actors.
- */
-
-const { RootActor } = require("devtools/server/actors/root");
-const { DebuggerServer } = require("devtools/server/main");
-const { BrowserTabList, BrowserAddonList, sendShutdownEvent } =
- require("devtools/server/actors/webbrowser");
-
-/**
- * Construct a root actor appropriate for use in a server running in a
- * browser on Android. The returned root actor:
- * - respects the factories registered with DebuggerServer.addGlobalActor,
- * - uses a MobileTabList to supply tab actors,
- * - sends all navigator:browser window documents a Debugger:Shutdown event
- * when it exits.
- *
- * * @param aConnection DebuggerServerConnection
- * The conection to the client.
- */
-function createRootActor(aConnection)
-{
- let parameters = {
- tabList: new MobileTabList(aConnection),
- addonList: new BrowserAddonList(aConnection),
- globalActorFactories: DebuggerServer.globalActorFactories,
- onShutdown: sendShutdownEvent
- };
- return new RootActor(aConnection, parameters);
-}
-
-/**
- * A live list of BrowserTabActors representing the current browser tabs,
- * to be provided to the root actor to answer 'listTabs' requests.
- *
- * This object also takes care of listening for TabClose events and
- * onCloseWindow notifications, and exiting the BrowserTabActors concerned.
- *
- * (See the documentation for RootActor for the definition of the "live
- * list" interface.)
- *
- * @param aConnection DebuggerServerConnection
- * The connection in which this list's tab actors may participate.
- *
- * @see BrowserTabList for more a extensive description of how tab list objects
- * work.
- */
-function MobileTabList(aConnection)
-{
- BrowserTabList.call(this, aConnection);
-}
-
-MobileTabList.prototype = Object.create(BrowserTabList.prototype);
-
-MobileTabList.prototype.constructor = MobileTabList;
-
-MobileTabList.prototype._getSelectedBrowser = function(aWindow) {
- return aWindow.BrowserApp.selectedBrowser;
-};
-
-MobileTabList.prototype._getChildren = function(aWindow) {
- return aWindow.BrowserApp.tabs.map(tab => tab.browser);
-};
-
-exports.register = function(handle) {
- handle.setRootActor(createRootActor);
-};
-
-exports.unregister = function(handle) {
- handle.setRootActor(null);
-};
diff --git a/mobile/android/modules/moz.build b/mobile/android/modules/moz.build
deleted file mode 100644
index dd62e484a..000000000
--- a/mobile/android/modules/moz.build
+++ /dev/null
@@ -1,32 +0,0 @@
-# -*- Mode: python; indent-tabs-mode: nil; tab-width: 40 -*-
-# vim: set filetype=python:
-# 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/.
-
-EXTRA_JS_MODULES += [
- 'Accounts.jsm',
- 'AndroidLog.jsm',
- 'dbg-browser-actors.js',
- 'DelayedInit.jsm',
- 'DownloadNotifications.jsm',
- 'HelperApps.jsm',
- 'Home.jsm',
- 'HomeProvider.jsm',
- 'JavaAddonManager.jsm',
- 'JNI.jsm',
- 'LightweightThemeConsumer.jsm',
- 'MediaPlayerApp.jsm',
- 'Messaging.jsm',
- 'NetErrorHelper.jsm',
- 'Notifications.jsm',
- 'PageActions.jsm',
- 'Prompt.jsm',
- 'RuntimePermissions.jsm',
- 'Sanitizer.jsm',
- 'SharedPreferences.jsm',
- 'Snackbars.jsm',
- 'SSLExceptions.jsm',
- 'TabMirror.jsm',
- 'WebsiteMetadata.jsm'
-]