summaryrefslogtreecommitdiffstats
path: root/mobile/android/components/extensions/ext-tabs.js
diff options
context:
space:
mode:
Diffstat (limited to 'mobile/android/components/extensions/ext-tabs.js')
-rw-r--r--mobile/android/components/extensions/ext-tabs.js519
1 files changed, 519 insertions, 0 deletions
diff --git a/mobile/android/components/extensions/ext-tabs.js b/mobile/android/components/extensions/ext-tabs.js
new file mode 100644
index 0000000000..b8f3b0716b
--- /dev/null
+++ b/mobile/android/components/extensions/ext-tabs.js
@@ -0,0 +1,519 @@
+/* -*- Mode: indent-tabs-mode: nil; js-indent-level: 2 -*- */
+/* vim: set sts=2 sw=2 et tw=80: */
+"use strict";
+
+ChromeUtils.defineModuleGetter(
+ this,
+ "PromiseUtils",
+ "resource://gre/modules/PromiseUtils.jsm"
+);
+
+ChromeUtils.defineModuleGetter(
+ this,
+ "GeckoViewTabBridge",
+ "resource://gre/modules/GeckoViewTab.jsm"
+);
+
+ChromeUtils.defineModuleGetter(
+ this,
+ "mobileWindowTracker",
+ "resource://gre/modules/GeckoViewWebExtension.jsm"
+);
+
+const getBrowserWindow = window => {
+ return window.browsingContext.topChromeWindow;
+};
+
+const tabListener = {
+ tabReadyInitialized: false,
+ tabReadyPromises: new WeakMap(),
+ initializingTabs: new WeakSet(),
+
+ initTabReady() {
+ if (!this.tabReadyInitialized) {
+ windowTracker.addListener("progress", this);
+
+ this.tabReadyInitialized = true;
+ }
+ },
+
+ onLocationChange(browser, webProgress, request, locationURI, flags) {
+ if (webProgress.isTopLevel) {
+ const { tab } = browser.ownerGlobal;
+
+ // Ignore initial about:blank
+ if (!request && this.initializingTabs.has(tab)) {
+ return;
+ }
+
+ // Now we are certain that the first page in the tab was loaded.
+ this.initializingTabs.delete(tab);
+
+ // browser.innerWindowID is now set, resolve the promises if any.
+ const deferred = this.tabReadyPromises.get(tab);
+ if (deferred) {
+ deferred.resolve(tab);
+ this.tabReadyPromises.delete(tab);
+ }
+ }
+ },
+
+ /**
+ * Returns a promise that resolves when the tab is ready.
+ * Tabs created via the `tabs.create` method are "ready" once the location
+ * changes to the requested URL. Other tabs are assumed to be ready once their
+ * inner window ID is known.
+ *
+ * @param {NativeTab} nativeTab The native tab object.
+ * @returns {Promise} Resolves with the given tab once ready.
+ */
+ awaitTabReady(nativeTab) {
+ let deferred = this.tabReadyPromises.get(nativeTab);
+ if (!deferred) {
+ deferred = PromiseUtils.defer();
+ if (
+ !this.initializingTabs.has(nativeTab) &&
+ (nativeTab.browser.innerWindowID ||
+ nativeTab.browser.currentURI.spec === "about:blank")
+ ) {
+ deferred.resolve(nativeTab);
+ } else {
+ this.initTabReady();
+ this.tabReadyPromises.set(nativeTab, deferred);
+ }
+ }
+ return deferred.promise;
+ },
+};
+
+this.tabs = class extends ExtensionAPI {
+ getAPI(context) {
+ const { extension } = context;
+
+ const { tabManager } = extension;
+
+ function getTabOrActive(tabId) {
+ if (tabId !== null) {
+ return tabTracker.getTab(tabId);
+ }
+ return tabTracker.activeTab;
+ }
+
+ async function promiseTabWhenReady(tabId) {
+ let tab;
+ if (tabId !== null) {
+ tab = tabManager.get(tabId);
+ } else {
+ tab = tabManager.getWrapper(tabTracker.activeTab);
+ }
+ if (!tab) {
+ throw new ExtensionError(
+ tabId == null ? "Cannot access activeTab" : `Invalid tab ID: ${tabId}`
+ );
+ }
+
+ await tabListener.awaitTabReady(tab.nativeTab);
+
+ return tab;
+ }
+
+ const self = {
+ tabs: {
+ onActivated: new EventManager({
+ context,
+ name: "tabs.onActivated",
+ register: fire => {
+ const listener = (eventName, event) => {
+ const { windowId, tabId, isPrivate } = event;
+ if (isPrivate && !context.privateBrowsingAllowed) {
+ return;
+ }
+ fire.async({
+ windowId,
+ tabId,
+ // In GeckoView each window has only one tab, so previousTabId is omitted.
+ });
+ };
+
+ mobileWindowTracker.on("tab-activated", listener);
+ return () => {
+ mobileWindowTracker.off("tab-activated", listener);
+ };
+ },
+ }).api(),
+
+ onCreated: new EventManager({
+ context,
+ name: "tabs.onCreated",
+ register: fire => {
+ const listener = (eventName, event) => {
+ fire.async(tabManager.convert(event.nativeTab));
+ };
+
+ tabTracker.on("tab-created", listener);
+ return () => {
+ tabTracker.off("tab-created", listener);
+ };
+ },
+ }).api(),
+
+ /**
+ * Since multiple tabs currently can't be highlighted, onHighlighted
+ * essentially acts an alias for self.tabs.onActivated but returns
+ * the tabId in an array to match the API.
+ * @see https://developer.mozilla.org/en-US/Add-ons/WebExtensions/API/Tabs/onHighlighted
+ */
+ onHighlighted: makeGlobalEvent(
+ context,
+ "tabs.onHighlighted",
+ "Tab:Selected",
+ (fire, data) => {
+ const tab = tabManager.get(data.id);
+
+ fire.async({ tabIds: [tab.id], windowId: tab.windowId });
+ }
+ ),
+
+ onAttached: new EventManager({
+ context,
+ name: "tabs.onAttached",
+ register: fire => {
+ return () => {};
+ },
+ }).api(),
+
+ onDetached: new EventManager({
+ context,
+ name: "tabs.onDetached",
+ register: fire => {
+ return () => {};
+ },
+ }).api(),
+
+ onRemoved: new EventManager({
+ context,
+ name: "tabs.onRemoved",
+ register: fire => {
+ const listener = (eventName, event) => {
+ fire.async(event.tabId, {
+ windowId: event.windowId,
+ isWindowClosing: event.isWindowClosing,
+ });
+ };
+
+ tabTracker.on("tab-removed", listener);
+ return () => {
+ tabTracker.off("tab-removed", listener);
+ };
+ },
+ }).api(),
+
+ onReplaced: new EventManager({
+ context,
+ name: "tabs.onReplaced",
+ register: fire => {
+ return () => {};
+ },
+ }).api(),
+
+ onMoved: new EventManager({
+ context,
+ name: "tabs.onMoved",
+ register: fire => {
+ return () => {};
+ },
+ }).api(),
+
+ onUpdated: new EventManager({
+ context,
+ name: "tabs.onUpdated",
+ register: fire => {
+ const restricted = ["url", "favIconUrl", "title"];
+
+ function sanitize(tab, changeInfo) {
+ const result = {};
+ let nonempty = false;
+ for (const prop in changeInfo) {
+ // In practice, changeInfo contains at most one property from
+ // restricted. Therefore it is not necessary to cache the value
+ // of tab.hasTabPermission outside the loop.
+ if (!restricted.includes(prop) || tab.hasTabPermission) {
+ nonempty = true;
+ result[prop] = changeInfo[prop];
+ }
+ }
+ return [nonempty, result];
+ }
+
+ const fireForTab = (tab, changed) => {
+ const [needed, changeInfo] = sanitize(tab, changed);
+ if (needed) {
+ fire.async(tab.id, changeInfo, tab.convert());
+ }
+ };
+
+ const listener = event => {
+ const needed = [];
+ let nativeTab;
+ switch (event.type) {
+ case "pagetitlechanged": {
+ const window = getBrowserWindow(event.target.ownerGlobal);
+ nativeTab = window.tab;
+
+ needed.push("title");
+ break;
+ }
+
+ case "DOMAudioPlaybackStarted":
+ case "DOMAudioPlaybackStopped": {
+ const window = event.target.ownerGlobal;
+ nativeTab = window.tab;
+ needed.push("audible");
+ break;
+ }
+ }
+
+ if (!nativeTab) {
+ return;
+ }
+
+ const tab = tabManager.getWrapper(nativeTab);
+ const changeInfo = {};
+ for (const prop of needed) {
+ changeInfo[prop] = tab[prop];
+ }
+
+ fireForTab(tab, changeInfo);
+ };
+
+ const statusListener = ({ browser, status, url }) => {
+ const { tab } = browser.ownerGlobal;
+ if (tab) {
+ const changed = { status };
+ if (url) {
+ changed.url = url;
+ }
+
+ fireForTab(tabManager.wrapTab(tab), changed);
+ }
+ };
+
+ windowTracker.addListener("status", statusListener);
+ windowTracker.addListener("pagetitlechanged", listener);
+ return () => {
+ windowTracker.removeListener("status", statusListener);
+ windowTracker.removeListener("pagetitlechanged", listener);
+ };
+ },
+ }).api(),
+
+ async create({
+ active,
+ cookieStoreId,
+ discarded,
+ index,
+ openInReaderMode,
+ pinned,
+ title,
+ url,
+ } = {}) {
+ if (active === null) {
+ active = true;
+ }
+
+ tabListener.initTabReady();
+
+ if (url !== null) {
+ url = context.uri.resolve(url);
+
+ if (!context.checkLoadURL(url, { dontReportErrors: true })) {
+ return Promise.reject({ message: `Illegal URL: ${url}` });
+ }
+ }
+
+ const nativeTab = await GeckoViewTabBridge.createNewTab({
+ extensionId: context.extension.id,
+ createProperties: {
+ active,
+ cookieStoreId,
+ discarded,
+ index,
+ openInReaderMode,
+ pinned,
+ url,
+ },
+ });
+
+ const { browser } = nativeTab;
+
+ let flags = Ci.nsIWebNavigation.LOAD_FLAGS_NONE;
+
+ // Make sure things like about:blank URIs never inherit,
+ // and instead always get a NullPrincipal.
+ if (url !== null) {
+ tabListener.initializingTabs.add(nativeTab);
+ } else {
+ url = "about:blank";
+ flags |= Ci.nsIWebNavigation.LOAD_FLAGS_DISALLOW_INHERIT_PRINCIPAL;
+ }
+
+ browser.loadURI(url, {
+ flags,
+ // GeckoView doesn't support about:newtab so we don't need to worry
+ // about using the system principal here.
+ triggeringPrincipal: context.principal,
+ });
+
+ if (active) {
+ const newWindow = browser.ownerGlobal;
+ mobileWindowTracker.setTabActive(newWindow, true);
+ }
+
+ return tabManager.convert(nativeTab);
+ },
+
+ async remove(tabs) {
+ if (!Array.isArray(tabs)) {
+ tabs = [tabs];
+ }
+
+ await Promise.all(
+ tabs.map(async tabId => {
+ const windowId = GeckoViewTabBridge.tabIdToWindowId(tabId);
+ const window = windowTracker.getWindow(windowId, context, false);
+ if (!window) {
+ throw new ExtensionError(`Invalid tab ID ${tabId}`);
+ }
+ await GeckoViewTabBridge.closeTab({
+ window,
+ extensionId: context.extension.id,
+ });
+ })
+ );
+ },
+
+ async update(
+ tabId,
+ { active, autoDiscardable, highlighted, muted, pinned, url } = {}
+ ) {
+ const nativeTab = getTabOrActive(tabId);
+ const window = nativeTab.browser.ownerGlobal;
+
+ if (url !== null) {
+ url = context.uri.resolve(url);
+
+ if (!context.checkLoadURL(url, { dontReportErrors: true })) {
+ return Promise.reject({ message: `Illegal URL: ${url}` });
+ }
+ }
+
+ await GeckoViewTabBridge.updateTab({
+ window,
+ extensionId: context.extension.id,
+ updateProperties: {
+ active,
+ autoDiscardable,
+ highlighted,
+ muted,
+ pinned,
+ url,
+ },
+ });
+
+ if (url !== null) {
+ nativeTab.browser.loadURI(url, {
+ triggeringPrincipal: context.principal,
+ });
+ }
+
+ // FIXME: openerTabId, successorTabId
+ if (active) {
+ mobileWindowTracker.setTabActive(window, true);
+ }
+
+ return tabManager.convert(nativeTab);
+ },
+
+ async reload(tabId, reloadProperties) {
+ const nativeTab = getTabOrActive(tabId);
+
+ let flags = Ci.nsIWebNavigation.LOAD_FLAGS_NONE;
+ if (reloadProperties && reloadProperties.bypassCache) {
+ flags |= Ci.nsIWebNavigation.LOAD_FLAGS_BYPASS_CACHE;
+ }
+ nativeTab.browser.reloadWithFlags(flags);
+ },
+
+ async get(tabId) {
+ return tabManager.get(tabId).convert();
+ },
+
+ async getCurrent() {
+ if (context.tabId) {
+ return tabManager.get(context.tabId).convert();
+ }
+ },
+
+ async query(queryInfo) {
+ return Array.from(tabManager.query(queryInfo, context), tab =>
+ tab.convert()
+ );
+ },
+
+ async captureTab(tabId, options) {
+ const nativeTab = getTabOrActive(tabId);
+ await tabListener.awaitTabReady(nativeTab);
+
+ const { browser } = nativeTab;
+ const window = browser.ownerGlobal;
+ const zoom = window.windowUtils.fullZoom;
+
+ const tab = tabManager.wrapTab(nativeTab);
+ return tab.capture(context, zoom, options);
+ },
+
+ async captureVisibleTab(windowId, options) {
+ const window =
+ windowId == null
+ ? windowTracker.topWindow
+ : windowTracker.getWindow(windowId, context);
+
+ const tab = tabManager.wrapTab(window.tab);
+ await tabListener.awaitTabReady(tab.nativeTab);
+ const zoom = window.windowUtils.fullZoom;
+
+ return tab.capture(context, zoom, options);
+ },
+
+ async executeScript(tabId, details) {
+ const tab = await promiseTabWhenReady(tabId);
+
+ return tab.executeScript(context, details);
+ },
+
+ async insertCSS(tabId, details) {
+ const tab = await promiseTabWhenReady(tabId);
+
+ return tab.insertCSS(context, details);
+ },
+
+ async removeCSS(tabId, details) {
+ const tab = await promiseTabWhenReady(tabId);
+
+ return tab.removeCSS(context, details);
+ },
+
+ goForward(tabId) {
+ const { browser } = getTabOrActive(tabId);
+ browser.goForward();
+ },
+
+ goBack(tabId) {
+ const { browser } = getTabOrActive(tabId);
+ browser.goBack();
+ },
+ },
+ };
+ return self;
+ }
+};