forked from mirrors/gecko-dev
		
	
		
			
				
	
	
		
			6286 lines
		
	
	
	
		
			211 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			6286 lines
		
	
	
	
		
			211 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
/* This Source Code Form is subject to the terms of the Mozilla Public
 | 
						|
 * License, v. 2.0. If a copy of the MPL was not distributed with this
 | 
						|
 * file, You can obtain one at http://mozilla.org/MPL/2.0/. */
 | 
						|
 | 
						|
import { XPCOMUtils } from "resource://gre/modules/XPCOMUtils.sys.mjs";
 | 
						|
import { AppConstants } from "resource://gre/modules/AppConstants.sys.mjs";
 | 
						|
import { SearchWidgetTracker } from "resource:///modules/SearchWidgetTracker.sys.mjs";
 | 
						|
 | 
						|
const lazy = {};
 | 
						|
 | 
						|
ChromeUtils.defineESModuleGetters(lazy, {
 | 
						|
  AddonManager: "resource://gre/modules/AddonManager.sys.mjs",
 | 
						|
  AddonManagerPrivate: "resource://gre/modules/AddonManager.sys.mjs",
 | 
						|
  BrowserUsageTelemetry: "resource:///modules/BrowserUsageTelemetry.sys.mjs",
 | 
						|
  CustomizableWidgets: "resource:///modules/CustomizableWidgets.sys.mjs",
 | 
						|
  HomePage: "resource:///modules/HomePage.sys.mjs",
 | 
						|
  PanelMultiView: "resource:///modules/PanelMultiView.sys.mjs",
 | 
						|
  PrivateBrowsingUtils: "resource://gre/modules/PrivateBrowsingUtils.sys.mjs",
 | 
						|
  ShortcutUtils: "resource://gre/modules/ShortcutUtils.sys.mjs",
 | 
						|
});
 | 
						|
 | 
						|
ChromeUtils.defineLazyGetter(lazy, "gWidgetsBundle", function () {
 | 
						|
  const kUrl =
 | 
						|
    "chrome://browser/locale/customizableui/customizableWidgets.properties";
 | 
						|
  return Services.strings.createBundle(kUrl);
 | 
						|
});
 | 
						|
 | 
						|
const kDefaultThemeID = "default-theme@mozilla.org";
 | 
						|
 | 
						|
const kSpecialWidgetPfx = "customizableui-special-";
 | 
						|
 | 
						|
const kPrefCustomizationState = "browser.uiCustomization.state";
 | 
						|
const kPrefCustomizationAutoAdd = "browser.uiCustomization.autoAdd";
 | 
						|
const kPrefCustomizationDebug = "browser.uiCustomization.debug";
 | 
						|
const kPrefDrawInTitlebar = "browser.tabs.inTitlebar";
 | 
						|
const kPrefUIDensity = "browser.uidensity";
 | 
						|
const kPrefAutoTouchMode = "browser.touchmode.auto";
 | 
						|
const kPrefAutoHideDownloadsButton = "browser.download.autohideButton";
 | 
						|
const kPrefProtonToolbarVersion = "browser.proton.toolbar.version";
 | 
						|
const kPrefHomeButtonUsed = "browser.engagement.home-button.has-used";
 | 
						|
const kPrefLibraryButtonUsed = "browser.engagement.library-button.has-used";
 | 
						|
const kPrefSidebarButtonUsed = "browser.engagement.sidebar-button.has-used";
 | 
						|
 | 
						|
const kExpectedWindowURL = AppConstants.BROWSER_CHROME_URL;
 | 
						|
 | 
						|
var gDefaultTheme;
 | 
						|
var gSelectedTheme;
 | 
						|
 | 
						|
/**
 | 
						|
 * The keys are the handlers that are fired when the event type (the value)
 | 
						|
 * is fired on the subview. A widget that provides a subview has the option
 | 
						|
 * of providing onViewShowing and onViewHiding event handlers.
 | 
						|
 */
 | 
						|
const kSubviewEvents = ["ViewShowing", "ViewHiding"];
 | 
						|
 | 
						|
/**
 | 
						|
 * The current version. We can use this to auto-add new default widgets as necessary.
 | 
						|
 * (would be const but isn't because of testing purposes)
 | 
						|
 */
 | 
						|
var kVersion = 20;
 | 
						|
 | 
						|
/**
 | 
						|
 * Buttons removed from built-ins by version they were removed. kVersion must be
 | 
						|
 * bumped any time a new id is added to this. Use the button id as key, and
 | 
						|
 * version the button is removed in as the value.  e.g. "pocket-button": 5
 | 
						|
 */
 | 
						|
var ObsoleteBuiltinButtons = {
 | 
						|
  "feed-button": 15,
 | 
						|
};
 | 
						|
 | 
						|
/**
 | 
						|
 * gPalette is a map of every widget that CustomizableUI.sys.mjs knows about, keyed
 | 
						|
 * on their IDs.
 | 
						|
 */
 | 
						|
var gPalette = new Map();
 | 
						|
 | 
						|
/**
 | 
						|
 * gAreas maps area IDs to Sets of properties about those areas. An area is a
 | 
						|
 * place where a widget can be put.
 | 
						|
 */
 | 
						|
var gAreas = new Map();
 | 
						|
 | 
						|
/**
 | 
						|
 * gPlacements maps area IDs to Arrays of widget IDs, indicating that the widgets
 | 
						|
 * are placed within that area (either directly in the area node, or in the
 | 
						|
 * customizationTarget of the node).
 | 
						|
 */
 | 
						|
var gPlacements = new Map();
 | 
						|
 | 
						|
/**
 | 
						|
 * gFuturePlacements represent placements that will happen for areas that have
 | 
						|
 * not yet loaded (due to lazy-loading). This can occur when add-ons register
 | 
						|
 * widgets.
 | 
						|
 */
 | 
						|
var gFuturePlacements = new Map();
 | 
						|
 | 
						|
var gSupportedWidgetTypes = new Set([
 | 
						|
  // A button that does a command.
 | 
						|
  "button",
 | 
						|
 | 
						|
  // A button that opens a view in a panel (or in a subview of the panel).
 | 
						|
  "view",
 | 
						|
 | 
						|
  // A combination of the above, which looks different depending on whether it's
 | 
						|
  // located in the toolbar or in the panel: When located in the toolbar, shown
 | 
						|
  // as a combined item of a button and a dropmarker button. The button triggers
 | 
						|
  // the command and the dropmarker button opens the view. When located in the
 | 
						|
  // panel, shown as one item which opens the view, and the button command
 | 
						|
  // cannot be triggered separately.
 | 
						|
  "button-and-view",
 | 
						|
 | 
						|
  // A custom widget that defines its own markup.
 | 
						|
  "custom",
 | 
						|
]);
 | 
						|
 | 
						|
/**
 | 
						|
 * gPanelsForWindow is a list of known panels in a window which we may need to close
 | 
						|
 * should command events fire which target them.
 | 
						|
 */
 | 
						|
var gPanelsForWindow = new WeakMap();
 | 
						|
 | 
						|
/**
 | 
						|
 * gSeenWidgets remembers which widgets the user has seen for the first time
 | 
						|
 * before. This way, if a new widget is created, and the user has not seen it
 | 
						|
 * before, it can be put in its default location. Otherwise, it remains in the
 | 
						|
 * palette.
 | 
						|
 */
 | 
						|
var gSeenWidgets = new Set();
 | 
						|
 | 
						|
/**
 | 
						|
 * gDirtyAreaCache is a set of area IDs for areas where items have been added,
 | 
						|
 * moved or removed at least once. This set is persisted, and is used to
 | 
						|
 * optimize building of toolbars in the default case where no toolbars should
 | 
						|
 * be "dirty".
 | 
						|
 */
 | 
						|
var gDirtyAreaCache = new Set();
 | 
						|
 | 
						|
/**
 | 
						|
 * gPendingBuildAreas is a map from area IDs to map from build nodes to their
 | 
						|
 * existing children at the time of node registration, that are waiting
 | 
						|
 * for the area to be registered
 | 
						|
 */
 | 
						|
var gPendingBuildAreas = new Map();
 | 
						|
 | 
						|
var gSavedState = null;
 | 
						|
var gRestoring = false;
 | 
						|
var gDirty = false;
 | 
						|
var gInBatchStack = 0;
 | 
						|
var gResetting = false;
 | 
						|
var gUndoResetting = false;
 | 
						|
 | 
						|
/**
 | 
						|
 * gBuildAreas maps area IDs to actual area nodes within browser windows.
 | 
						|
 */
 | 
						|
var gBuildAreas = new Map();
 | 
						|
 | 
						|
/**
 | 
						|
 * gBuildWindows is a map of windows that have registered build areas, mapped
 | 
						|
 * to a Set of known toolboxes in that window.
 | 
						|
 */
 | 
						|
var gBuildWindows = new Map();
 | 
						|
 | 
						|
var gNewElementCount = 0;
 | 
						|
var gGroupWrapperCache = new Map();
 | 
						|
var gSingleWrapperCache = new WeakMap();
 | 
						|
var gListeners = new Set();
 | 
						|
 | 
						|
var gUIStateBeforeReset = {
 | 
						|
  uiCustomizationState: null,
 | 
						|
  drawInTitlebar: null,
 | 
						|
  currentTheme: null,
 | 
						|
  uiDensity: null,
 | 
						|
  autoTouchMode: null,
 | 
						|
};
 | 
						|
 | 
						|
XPCOMUtils.defineLazyPreferenceGetter(
 | 
						|
  lazy,
 | 
						|
  "gDebuggingEnabled",
 | 
						|
  kPrefCustomizationDebug,
 | 
						|
  false,
 | 
						|
  (pref, oldVal, newVal) => {
 | 
						|
    if (typeof lazy.log != "undefined") {
 | 
						|
      lazy.log.maxLogLevel = newVal ? "all" : "log";
 | 
						|
    }
 | 
						|
  }
 | 
						|
);
 | 
						|
 | 
						|
XPCOMUtils.defineLazyPreferenceGetter(
 | 
						|
  lazy,
 | 
						|
  "resetPBMToolbarButtonEnabled",
 | 
						|
  "browser.privatebrowsing.resetPBM.enabled",
 | 
						|
  false
 | 
						|
);
 | 
						|
 | 
						|
ChromeUtils.defineLazyGetter(lazy, "log", () => {
 | 
						|
  let { ConsoleAPI } = ChromeUtils.importESModule(
 | 
						|
    "resource://gre/modules/Console.sys.mjs"
 | 
						|
  );
 | 
						|
  let consoleOptions = {
 | 
						|
    maxLogLevel: lazy.gDebuggingEnabled ? "all" : "log",
 | 
						|
    prefix: "CustomizableUI",
 | 
						|
  };
 | 
						|
  return new ConsoleAPI(consoleOptions);
 | 
						|
});
 | 
						|
 | 
						|
var CustomizableUIInternal = {
 | 
						|
  initialize() {
 | 
						|
    lazy.log.debug("Initializing");
 | 
						|
 | 
						|
    lazy.AddonManagerPrivate.databaseReady.then(async () => {
 | 
						|
      lazy.AddonManager.addAddonListener(this);
 | 
						|
 | 
						|
      let addons = await lazy.AddonManager.getAddonsByTypes(["theme"]);
 | 
						|
      gDefaultTheme = addons.find(addon => addon.id == kDefaultThemeID);
 | 
						|
      gSelectedTheme = addons.find(addon => addon.isActive) || gDefaultTheme;
 | 
						|
    });
 | 
						|
 | 
						|
    this.addListener(this);
 | 
						|
    this._defineBuiltInWidgets();
 | 
						|
    this.loadSavedState();
 | 
						|
    this._updateForNewVersion();
 | 
						|
    this._updateForNewProtonVersion();
 | 
						|
    this._markObsoleteBuiltinButtonsSeen();
 | 
						|
 | 
						|
    this.registerArea(
 | 
						|
      CustomizableUI.AREA_FIXED_OVERFLOW_PANEL,
 | 
						|
      {
 | 
						|
        type: CustomizableUI.TYPE_PANEL,
 | 
						|
        defaultPlacements: [],
 | 
						|
        anchor: "nav-bar-overflow-button",
 | 
						|
      },
 | 
						|
      true
 | 
						|
    );
 | 
						|
 | 
						|
    this.registerArea(
 | 
						|
      CustomizableUI.AREA_ADDONS,
 | 
						|
      {
 | 
						|
        type: CustomizableUI.TYPE_PANEL,
 | 
						|
        defaultPlacements: [],
 | 
						|
        anchor: "unified-extensions-button",
 | 
						|
      },
 | 
						|
      false
 | 
						|
    );
 | 
						|
 | 
						|
    let navbarPlacements = [
 | 
						|
      "back-button",
 | 
						|
      "forward-button",
 | 
						|
      "stop-reload-button",
 | 
						|
      Services.policies.isAllowed("removeHomeButtonByDefault")
 | 
						|
        ? null
 | 
						|
        : "home-button",
 | 
						|
      "spring",
 | 
						|
      "urlbar-container",
 | 
						|
      "spring",
 | 
						|
      "save-to-pocket-button",
 | 
						|
      "downloads-button",
 | 
						|
      AppConstants.MOZ_DEV_EDITION ? "developer-button" : null,
 | 
						|
      "fxa-toolbar-menu-button",
 | 
						|
      lazy.resetPBMToolbarButtonEnabled ? "reset-pbm-toolbar-button" : null,
 | 
						|
    ].filter(name => name);
 | 
						|
 | 
						|
    this.registerArea(
 | 
						|
      CustomizableUI.AREA_NAVBAR,
 | 
						|
      {
 | 
						|
        type: CustomizableUI.TYPE_TOOLBAR,
 | 
						|
        overflowable: true,
 | 
						|
        defaultPlacements: navbarPlacements,
 | 
						|
        defaultCollapsed: false,
 | 
						|
      },
 | 
						|
      true
 | 
						|
    );
 | 
						|
 | 
						|
    if (AppConstants.MENUBAR_CAN_AUTOHIDE) {
 | 
						|
      this.registerArea(
 | 
						|
        CustomizableUI.AREA_MENUBAR,
 | 
						|
        {
 | 
						|
          type: CustomizableUI.TYPE_TOOLBAR,
 | 
						|
          defaultPlacements: ["menubar-items"],
 | 
						|
          defaultCollapsed: true,
 | 
						|
        },
 | 
						|
        true
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    this.registerArea(
 | 
						|
      CustomizableUI.AREA_TABSTRIP,
 | 
						|
      {
 | 
						|
        type: CustomizableUI.TYPE_TOOLBAR,
 | 
						|
        defaultPlacements: [
 | 
						|
          "firefox-view-button",
 | 
						|
          "tabbrowser-tabs",
 | 
						|
          "new-tab-button",
 | 
						|
          "alltabs-button",
 | 
						|
        ],
 | 
						|
        defaultCollapsed: null,
 | 
						|
      },
 | 
						|
      true
 | 
						|
    );
 | 
						|
    this.registerArea(
 | 
						|
      CustomizableUI.AREA_BOOKMARKS,
 | 
						|
      {
 | 
						|
        type: CustomizableUI.TYPE_TOOLBAR,
 | 
						|
        defaultPlacements: ["personal-bookmarks"],
 | 
						|
        defaultCollapsed: "newtab",
 | 
						|
      },
 | 
						|
      true
 | 
						|
    );
 | 
						|
 | 
						|
    SearchWidgetTracker.init();
 | 
						|
 | 
						|
    Services.obs.addObserver(this, "browser-set-toolbar-visibility");
 | 
						|
  },
 | 
						|
 | 
						|
  onEnabled(addon) {
 | 
						|
    if (addon.type == "theme") {
 | 
						|
      gSelectedTheme = addon;
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  get _builtinAreas() {
 | 
						|
    return new Set([
 | 
						|
      ...this._builtinToolbars,
 | 
						|
      CustomizableUI.AREA_FIXED_OVERFLOW_PANEL,
 | 
						|
      CustomizableUI.AREA_ADDONS,
 | 
						|
    ]);
 | 
						|
  },
 | 
						|
 | 
						|
  get _builtinToolbars() {
 | 
						|
    let toolbars = new Set([
 | 
						|
      CustomizableUI.AREA_NAVBAR,
 | 
						|
      CustomizableUI.AREA_BOOKMARKS,
 | 
						|
      CustomizableUI.AREA_TABSTRIP,
 | 
						|
    ]);
 | 
						|
    if (AppConstants.platform != "macosx") {
 | 
						|
      toolbars.add(CustomizableUI.AREA_MENUBAR);
 | 
						|
    }
 | 
						|
    return toolbars;
 | 
						|
  },
 | 
						|
 | 
						|
  _defineBuiltInWidgets() {
 | 
						|
    for (let widgetDefinition of lazy.CustomizableWidgets) {
 | 
						|
      this.createBuiltinWidget(widgetDefinition);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  // eslint-disable-next-line complexity
 | 
						|
  _updateForNewVersion() {
 | 
						|
    // We should still enter even if gSavedState.currentVersion >= kVersion
 | 
						|
    // because the per-widget pref facility is independent of versioning.
 | 
						|
    if (!gSavedState) {
 | 
						|
      // Flip all the prefs so we don't try to re-introduce later:
 | 
						|
      for (let [, widget] of gPalette) {
 | 
						|
        if (widget.defaultArea && widget._introducedInVersion === "pref") {
 | 
						|
          let prefId = "browser.toolbarbuttons.introduced." + widget.id;
 | 
						|
          Services.prefs.setBoolPref(prefId, true);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let currentVersion = gSavedState.currentVersion;
 | 
						|
    for (let [id, widget] of gPalette) {
 | 
						|
      if (widget.defaultArea) {
 | 
						|
        let shouldAdd = false;
 | 
						|
        let shouldSetPref = false;
 | 
						|
        let prefId = "browser.toolbarbuttons.introduced." + widget.id;
 | 
						|
        if (widget._introducedInVersion === "pref") {
 | 
						|
          try {
 | 
						|
            shouldAdd = !Services.prefs.getBoolPref(prefId);
 | 
						|
          } catch (ex) {
 | 
						|
            // Pref doesn't exist:
 | 
						|
            shouldAdd = true;
 | 
						|
          }
 | 
						|
          shouldSetPref = shouldAdd;
 | 
						|
        } else if (widget._introducedInVersion > currentVersion) {
 | 
						|
          shouldAdd = true;
 | 
						|
        }
 | 
						|
 | 
						|
        if (shouldAdd) {
 | 
						|
          let futurePlacements = gFuturePlacements.get(widget.defaultArea);
 | 
						|
          if (futurePlacements) {
 | 
						|
            futurePlacements.add(id);
 | 
						|
          } else {
 | 
						|
            gFuturePlacements.set(widget.defaultArea, new Set([id]));
 | 
						|
          }
 | 
						|
          if (shouldSetPref) {
 | 
						|
            Services.prefs.setBoolPref(prefId, true);
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Nothing to migrate now if we don't have placements.
 | 
						|
    if (!gSavedState.placements) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (
 | 
						|
      currentVersion < 7 &&
 | 
						|
      gSavedState.placements[CustomizableUI.AREA_NAVBAR]
 | 
						|
    ) {
 | 
						|
      let placements = gSavedState.placements[CustomizableUI.AREA_NAVBAR];
 | 
						|
      let newPlacements = [
 | 
						|
        "back-button",
 | 
						|
        "forward-button",
 | 
						|
        "stop-reload-button",
 | 
						|
        "home-button",
 | 
						|
      ];
 | 
						|
      for (let button of placements) {
 | 
						|
        if (!newPlacements.includes(button)) {
 | 
						|
          newPlacements.push(button);
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (!newPlacements.includes("sidebar-button")) {
 | 
						|
        newPlacements.push("sidebar-button");
 | 
						|
      }
 | 
						|
 | 
						|
      gSavedState.placements[CustomizableUI.AREA_NAVBAR] = newPlacements;
 | 
						|
    }
 | 
						|
 | 
						|
    if (currentVersion < 8 && gSavedState.placements["PanelUI-contents"]) {
 | 
						|
      let savedPanelPlacements = gSavedState.placements["PanelUI-contents"];
 | 
						|
      delete gSavedState.placements["PanelUI-contents"];
 | 
						|
      let defaultPlacements = [
 | 
						|
        "edit-controls",
 | 
						|
        "zoom-controls",
 | 
						|
        "new-window-button",
 | 
						|
        "privatebrowsing-button",
 | 
						|
        "save-page-button",
 | 
						|
        "print-button",
 | 
						|
        "history-panelmenu",
 | 
						|
        "fullscreen-button",
 | 
						|
        "find-button",
 | 
						|
        "preferences-button",
 | 
						|
        // This widget no longer exists as of 2023, see Bug 1799009.
 | 
						|
        "add-ons-button",
 | 
						|
        "sync-button",
 | 
						|
      ];
 | 
						|
 | 
						|
      if (!AppConstants.MOZ_DEV_EDITION) {
 | 
						|
        defaultPlacements.splice(-1, 0, "developer-button");
 | 
						|
      }
 | 
						|
 | 
						|
      let showCharacterEncoding = Services.prefs.getComplexValue(
 | 
						|
        "browser.menu.showCharacterEncoding",
 | 
						|
        Ci.nsIPrefLocalizedString
 | 
						|
      ).data;
 | 
						|
      if (showCharacterEncoding == "true") {
 | 
						|
        defaultPlacements.push("characterencoding-button");
 | 
						|
      }
 | 
						|
 | 
						|
      savedPanelPlacements = savedPanelPlacements.filter(
 | 
						|
        id => !defaultPlacements.includes(id)
 | 
						|
      );
 | 
						|
 | 
						|
      if (savedPanelPlacements.length) {
 | 
						|
        gSavedState.placements[CustomizableUI.AREA_FIXED_OVERFLOW_PANEL] =
 | 
						|
          savedPanelPlacements;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (currentVersion < 9 && gSavedState.placements["nav-bar"]) {
 | 
						|
      let placements = gSavedState.placements["nav-bar"];
 | 
						|
      if (placements.includes("urlbar-container")) {
 | 
						|
        let urlbarIndex = placements.indexOf("urlbar-container");
 | 
						|
        let secondSpringIndex = urlbarIndex + 1;
 | 
						|
        // Insert if there isn't already a spring before the urlbar
 | 
						|
        if (
 | 
						|
          urlbarIndex == 0 ||
 | 
						|
          !placements[urlbarIndex - 1].startsWith(kSpecialWidgetPfx + "spring")
 | 
						|
        ) {
 | 
						|
          placements.splice(urlbarIndex, 0, "spring");
 | 
						|
          // The url bar is now 1 index later, so increment the insertion point for
 | 
						|
          // the second spring.
 | 
						|
          secondSpringIndex++;
 | 
						|
        }
 | 
						|
        // If the search container is present, insert after the search container
 | 
						|
        // instead of after the url bar
 | 
						|
        let searchContainerIndex = placements.indexOf("search-container");
 | 
						|
        if (searchContainerIndex != -1) {
 | 
						|
          secondSpringIndex = searchContainerIndex + 1;
 | 
						|
        }
 | 
						|
        if (
 | 
						|
          secondSpringIndex == placements.length ||
 | 
						|
          !placements[secondSpringIndex].startsWith(
 | 
						|
            kSpecialWidgetPfx + "spring"
 | 
						|
          )
 | 
						|
        ) {
 | 
						|
          placements.splice(secondSpringIndex, 0, "spring");
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      // Finally, replace the bookmarks menu button with the library one if present
 | 
						|
      if (placements.includes("bookmarks-menu-button")) {
 | 
						|
        let bmbIndex = placements.indexOf("bookmarks-menu-button");
 | 
						|
        placements.splice(bmbIndex, 1);
 | 
						|
        let downloadButtonIndex = placements.indexOf("downloads-button");
 | 
						|
        let libraryIndex =
 | 
						|
          downloadButtonIndex == -1 ? bmbIndex : downloadButtonIndex + 1;
 | 
						|
        placements.splice(libraryIndex, 0, "library-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (currentVersion < 10) {
 | 
						|
      for (let placements of Object.values(gSavedState.placements)) {
 | 
						|
        if (placements.includes("webcompat-reporter-button")) {
 | 
						|
          placements.splice(placements.indexOf("webcompat-reporter-button"), 1);
 | 
						|
          break;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Move the downloads button to the default position in the navbar if it's
 | 
						|
    // not there already.
 | 
						|
    if (currentVersion < 11) {
 | 
						|
      let navbarPlacements = gSavedState.placements[CustomizableUI.AREA_NAVBAR];
 | 
						|
      // First remove from wherever it currently lives, if anywhere:
 | 
						|
      for (let placements of Object.values(gSavedState.placements)) {
 | 
						|
        let existingIndex = placements.indexOf("downloads-button");
 | 
						|
        if (existingIndex != -1) {
 | 
						|
          placements.splice(existingIndex, 1);
 | 
						|
          break; // It can only be in 1 place, so no point looking elsewhere.
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      // Now put the button in the navbar in the correct spot:
 | 
						|
      if (navbarPlacements) {
 | 
						|
        let insertionPoint = navbarPlacements.indexOf("urlbar-container");
 | 
						|
        // Deliberately iterate to 1 past the end of the array to insert at the
 | 
						|
        // end if need be.
 | 
						|
        while (++insertionPoint < navbarPlacements.length) {
 | 
						|
          let widget = navbarPlacements[insertionPoint];
 | 
						|
          // If we find a non-searchbar, non-spacer node, break out of the loop:
 | 
						|
          if (
 | 
						|
            widget != "search-container" &&
 | 
						|
            !this.matchingSpecials(widget, "spring")
 | 
						|
          ) {
 | 
						|
            break;
 | 
						|
          }
 | 
						|
        }
 | 
						|
        // We either found the right spot, or reached the end of the
 | 
						|
        // placements, so insert here:
 | 
						|
        navbarPlacements.splice(insertionPoint, 0, "downloads-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (currentVersion < 12) {
 | 
						|
      const removedButtons = [
 | 
						|
        "loop-call-button",
 | 
						|
        "loop-button-throttled",
 | 
						|
        "pocket-button",
 | 
						|
      ];
 | 
						|
      for (let placements of Object.values(gSavedState.placements)) {
 | 
						|
        for (let button of removedButtons) {
 | 
						|
          let buttonIndex = placements.indexOf(button);
 | 
						|
          if (buttonIndex != -1) {
 | 
						|
            placements.splice(buttonIndex, 1);
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove the old placements from the now-gone Nightly-only
 | 
						|
    // "New non-e10s window" button.
 | 
						|
    if (currentVersion < 13) {
 | 
						|
      for (let placements of Object.values(gSavedState.placements)) {
 | 
						|
        let buttonIndex = placements.indexOf("e10s-button");
 | 
						|
        if (buttonIndex != -1) {
 | 
						|
          placements.splice(buttonIndex, 1);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove unsupported custom toolbar saved placements
 | 
						|
    if (currentVersion < 14) {
 | 
						|
      for (let area in gSavedState.placements) {
 | 
						|
        if (!this._builtinAreas.has(area)) {
 | 
						|
          delete gSavedState.placements[area];
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Add the FxA toolbar menu as the right most button item
 | 
						|
    if (currentVersion < 16) {
 | 
						|
      let navbarPlacements = gSavedState.placements[CustomizableUI.AREA_NAVBAR];
 | 
						|
      // Place the menu item as the first item to the left of the hamburger menu
 | 
						|
      if (navbarPlacements) {
 | 
						|
        navbarPlacements.push("fxa-toolbar-menu-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Add the save to Pocket button left of downloads button.
 | 
						|
    if (currentVersion < 17) {
 | 
						|
      let navbarPlacements = gSavedState.placements[CustomizableUI.AREA_NAVBAR];
 | 
						|
      let persistedPageActionsPref = Services.prefs.getCharPref(
 | 
						|
        "browser.pageActions.persistedActions",
 | 
						|
        ""
 | 
						|
      );
 | 
						|
      let pocketPreviouslyInUrl = true;
 | 
						|
      try {
 | 
						|
        let persistedPageActionsData = JSON.parse(persistedPageActionsPref);
 | 
						|
        // If Pocket was previously not in the url bar, let's not put it in the toolbar.
 | 
						|
        // It'll still be an option to add from the customization page.
 | 
						|
        pocketPreviouslyInUrl =
 | 
						|
          persistedPageActionsData.idsInUrlbar.includes("pocket");
 | 
						|
      } catch (e) {}
 | 
						|
      if (navbarPlacements && pocketPreviouslyInUrl) {
 | 
						|
        // Pocket's new home is next to the downloads button, or the next best spot.
 | 
						|
        let newPosition =
 | 
						|
          navbarPlacements.indexOf("downloads-button") ??
 | 
						|
          navbarPlacements.indexOf("fxa-toolbar-menu-button") ??
 | 
						|
          navbarPlacements.length;
 | 
						|
 | 
						|
        navbarPlacements.splice(newPosition, 0, "save-to-pocket-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Add firefox-view if not present
 | 
						|
    if (currentVersion < 18) {
 | 
						|
      let tabstripPlacements =
 | 
						|
        gSavedState.placements[CustomizableUI.AREA_TABSTRIP];
 | 
						|
      if (
 | 
						|
        tabstripPlacements &&
 | 
						|
        !tabstripPlacements.includes("firefox-view-button")
 | 
						|
      ) {
 | 
						|
        tabstripPlacements.unshift("firefox-view-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Unified Extensions addon button migration, which puts any browser action
 | 
						|
    // buttons in the overflow menu into the addons panel instead.
 | 
						|
    if (currentVersion < 19) {
 | 
						|
      let overflowPlacements =
 | 
						|
        gSavedState.placements[CustomizableUI.AREA_FIXED_OVERFLOW_PANEL] || [];
 | 
						|
      // The most likely case is that there are no AREA_ADDONS placements, in which case the
 | 
						|
      // array won't exist.
 | 
						|
      let addonsPlacements =
 | 
						|
        gSavedState.placements[CustomizableUI.AREA_ADDONS] || [];
 | 
						|
 | 
						|
      // Migration algorithm for transitioning to Unified Extensions:
 | 
						|
      //
 | 
						|
      // 1. Create two arrays, one for extension widgets, one for built-in widgets.
 | 
						|
      // 2. Iterate all items in the overflow panel, and push them into the
 | 
						|
      //    appropriate array based on whether or not its an extension widget.
 | 
						|
      // 3. Overwrite the overflow panel placements with the built-in widgets array.
 | 
						|
      // 4. Prepend the extension widgets to the addonsPlacements array. Note that this
 | 
						|
      //    does not overwrite this array as a precaution because it's possible
 | 
						|
      //    (though pretty unlikely) that some widgets are already there.
 | 
						|
      //
 | 
						|
      // For extension widgets that were in the palette, they will be appended to the
 | 
						|
      // addons area when they're created within createWidget.
 | 
						|
      let extWidgets = [];
 | 
						|
      let builtInWidgets = [];
 | 
						|
      for (let widgetId of overflowPlacements) {
 | 
						|
        if (CustomizableUI.isWebExtensionWidget(widgetId)) {
 | 
						|
          extWidgets.push(widgetId);
 | 
						|
        } else {
 | 
						|
          builtInWidgets.push(widgetId);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      gSavedState.placements[CustomizableUI.AREA_FIXED_OVERFLOW_PANEL] =
 | 
						|
        builtInWidgets;
 | 
						|
      gSavedState.placements[CustomizableUI.AREA_ADDONS] = [
 | 
						|
        ...extWidgets,
 | 
						|
        ...addonsPlacements,
 | 
						|
      ];
 | 
						|
    }
 | 
						|
 | 
						|
    // Add the PBM reset button as the right most button item
 | 
						|
    if (currentVersion < 20) {
 | 
						|
      let navbarPlacements = gSavedState.placements[CustomizableUI.AREA_NAVBAR];
 | 
						|
      // Place the button as the first item to the left of the hamburger menu
 | 
						|
      if (
 | 
						|
        navbarPlacements &&
 | 
						|
        !navbarPlacements.includes("reset-pbm-toolbar-button")
 | 
						|
      ) {
 | 
						|
        navbarPlacements.push("reset-pbm-toolbar-button");
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _updateForNewProtonVersion() {
 | 
						|
    const VERSION = 3;
 | 
						|
    let currentVersion = Services.prefs.getIntPref(
 | 
						|
      kPrefProtonToolbarVersion,
 | 
						|
      0
 | 
						|
    );
 | 
						|
    if (currentVersion >= VERSION) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let placements = gSavedState?.placements?.[CustomizableUI.AREA_NAVBAR];
 | 
						|
 | 
						|
    if (!placements) {
 | 
						|
      // The profile was created with this version, so no need to migrate.
 | 
						|
      Services.prefs.setIntPref(kPrefProtonToolbarVersion, VERSION);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove the home button if it hasn't been used and is set to about:home
 | 
						|
    if (currentVersion < 1) {
 | 
						|
      let homePage = lazy.HomePage.get();
 | 
						|
      if (
 | 
						|
        placements.includes("home-button") &&
 | 
						|
        !Services.prefs.getBoolPref(kPrefHomeButtonUsed) &&
 | 
						|
        (homePage == "about:home" || homePage == "about:blank") &&
 | 
						|
        Services.policies.isAllowed("removeHomeButtonByDefault")
 | 
						|
      ) {
 | 
						|
        placements.splice(placements.indexOf("home-button"), 1);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove the library button if it hasn't been used
 | 
						|
    if (currentVersion < 2) {
 | 
						|
      if (
 | 
						|
        placements.includes("library-button") &&
 | 
						|
        !Services.prefs.getBoolPref(kPrefLibraryButtonUsed)
 | 
						|
      ) {
 | 
						|
        placements.splice(placements.indexOf("library-button"), 1);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove the library button if it hasn't been used
 | 
						|
    if (currentVersion < 3) {
 | 
						|
      if (
 | 
						|
        placements.includes("sidebar-button") &&
 | 
						|
        !Services.prefs.getBoolPref(kPrefSidebarButtonUsed)
 | 
						|
      ) {
 | 
						|
        placements.splice(placements.indexOf("sidebar-button"), 1);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    Services.prefs.setIntPref(kPrefProtonToolbarVersion, VERSION);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * _markObsoleteBuiltinButtonsSeen
 | 
						|
   * when upgrading, ensure obsoleted buttons are in seen state.
 | 
						|
   */
 | 
						|
  _markObsoleteBuiltinButtonsSeen() {
 | 
						|
    if (!gSavedState) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let currentVersion = gSavedState.currentVersion;
 | 
						|
    if (currentVersion >= kVersion) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    // we're upgrading, update state if necessary
 | 
						|
    for (let id in ObsoleteBuiltinButtons) {
 | 
						|
      let version = ObsoleteBuiltinButtons[id];
 | 
						|
      if (version == kVersion) {
 | 
						|
        gSeenWidgets.add(id);
 | 
						|
        gDirty = true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _placeNewDefaultWidgetsInArea(aArea) {
 | 
						|
    let futurePlacedWidgets = gFuturePlacements.get(aArea);
 | 
						|
    let savedPlacements =
 | 
						|
      gSavedState && gSavedState.placements && gSavedState.placements[aArea];
 | 
						|
    let defaultPlacements = gAreas.get(aArea).get("defaultPlacements");
 | 
						|
    if (
 | 
						|
      !savedPlacements ||
 | 
						|
      !savedPlacements.length ||
 | 
						|
      !futurePlacedWidgets ||
 | 
						|
      !defaultPlacements ||
 | 
						|
      !defaultPlacements.length
 | 
						|
    ) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let defaultWidgetIndex = -1;
 | 
						|
 | 
						|
    for (let widgetId of futurePlacedWidgets) {
 | 
						|
      let widget = gPalette.get(widgetId);
 | 
						|
      if (
 | 
						|
        !widget ||
 | 
						|
        widget.source !== CustomizableUI.SOURCE_BUILTIN ||
 | 
						|
        !widget.defaultArea ||
 | 
						|
        !widget._introducedInVersion ||
 | 
						|
        savedPlacements.includes(widget.id)
 | 
						|
      ) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      defaultWidgetIndex = defaultPlacements.indexOf(widget.id);
 | 
						|
      if (defaultWidgetIndex === -1) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      // Now we know that this widget should be here by default, was newly introduced,
 | 
						|
      // and we have a saved state to insert into, and a default state to work off of.
 | 
						|
      // Try introducing after widgets that come before it in the default placements:
 | 
						|
      for (let i = defaultWidgetIndex; i >= 0; i--) {
 | 
						|
        // Special case: if the defaults list this widget as coming first, insert at the beginning:
 | 
						|
        if (i === 0 && i === defaultWidgetIndex) {
 | 
						|
          savedPlacements.splice(0, 0, widget.id);
 | 
						|
          // Before you ask, yes, deleting things inside a let x of y loop where y is a Set is
 | 
						|
          // safe, and we won't skip any items.
 | 
						|
          futurePlacedWidgets.delete(widget.id);
 | 
						|
          gDirty = true;
 | 
						|
          break;
 | 
						|
        }
 | 
						|
        // Otherwise, if we're somewhere other than the beginning, check if the previous
 | 
						|
        // widget is in the saved placements.
 | 
						|
        if (i) {
 | 
						|
          let previousWidget = defaultPlacements[i - 1];
 | 
						|
          let previousWidgetIndex = savedPlacements.indexOf(previousWidget);
 | 
						|
          if (previousWidgetIndex != -1) {
 | 
						|
            savedPlacements.splice(previousWidgetIndex + 1, 0, widget.id);
 | 
						|
            futurePlacedWidgets.delete(widget.id);
 | 
						|
            gDirty = true;
 | 
						|
            break;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
      // The loop above either inserts the item or doesn't - either way, we can get away
 | 
						|
      // with doing nothing else now; if the item remains in gFuturePlacements, we'll
 | 
						|
      // add it at the end in restoreStateForArea.
 | 
						|
    }
 | 
						|
    this.saveState();
 | 
						|
  },
 | 
						|
 | 
						|
  getCustomizationTarget(aElement) {
 | 
						|
    if (!aElement) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    if (
 | 
						|
      !aElement._customizationTarget &&
 | 
						|
      aElement.hasAttribute("customizable")
 | 
						|
    ) {
 | 
						|
      let id = aElement.getAttribute("customizationtarget");
 | 
						|
      if (id) {
 | 
						|
        aElement._customizationTarget =
 | 
						|
          aElement.ownerDocument.getElementById(id);
 | 
						|
      }
 | 
						|
 | 
						|
      if (!aElement._customizationTarget) {
 | 
						|
        aElement._customizationTarget = aElement;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return aElement._customizationTarget;
 | 
						|
  },
 | 
						|
 | 
						|
  wrapWidget(aWidgetId) {
 | 
						|
    if (gGroupWrapperCache.has(aWidgetId)) {
 | 
						|
      return gGroupWrapperCache.get(aWidgetId);
 | 
						|
    }
 | 
						|
 | 
						|
    let provider = this.getWidgetProvider(aWidgetId);
 | 
						|
    if (!provider) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    if (provider == CustomizableUI.PROVIDER_API) {
 | 
						|
      let widget = gPalette.get(aWidgetId);
 | 
						|
      if (!widget.wrapper) {
 | 
						|
        widget.wrapper = new WidgetGroupWrapper(widget);
 | 
						|
        gGroupWrapperCache.set(aWidgetId, widget.wrapper);
 | 
						|
      }
 | 
						|
      return widget.wrapper;
 | 
						|
    }
 | 
						|
 | 
						|
    // PROVIDER_SPECIAL gets treated the same as PROVIDER_XUL.
 | 
						|
    // XXXgijs: this causes bugs in code that depends on widgetWrapper.provider
 | 
						|
    // giving an accurate answer... filed as bug 1379821
 | 
						|
    let wrapper = new XULWidgetGroupWrapper(aWidgetId);
 | 
						|
    gGroupWrapperCache.set(aWidgetId, wrapper);
 | 
						|
    return wrapper;
 | 
						|
  },
 | 
						|
 | 
						|
  registerArea(aName, aProperties, aInternalCaller) {
 | 
						|
    if (typeof aName != "string" || !/^[a-z0-9-_]{1,}$/i.test(aName)) {
 | 
						|
      throw new Error("Invalid area name");
 | 
						|
    }
 | 
						|
 | 
						|
    let areaIsKnown = gAreas.has(aName);
 | 
						|
    let props = areaIsKnown ? gAreas.get(aName) : new Map();
 | 
						|
    const kImmutableProperties = new Set(["type", "overflowable"]);
 | 
						|
    for (let key in aProperties) {
 | 
						|
      if (
 | 
						|
        areaIsKnown &&
 | 
						|
        kImmutableProperties.has(key) &&
 | 
						|
        props.get(key) != aProperties[key]
 | 
						|
      ) {
 | 
						|
        throw new Error("An area cannot change the property for '" + key + "'");
 | 
						|
      }
 | 
						|
      props.set(key, aProperties[key]);
 | 
						|
    }
 | 
						|
    // Default to a toolbar:
 | 
						|
    if (!props.has("type")) {
 | 
						|
      props.set("type", CustomizableUI.TYPE_TOOLBAR);
 | 
						|
    }
 | 
						|
    if (props.get("type") == CustomizableUI.TYPE_TOOLBAR) {
 | 
						|
      // Check aProperties instead of props because this check is only interested
 | 
						|
      // in the passed arguments, not the state of a potentially pre-existing area.
 | 
						|
      if (!aInternalCaller && aProperties.defaultCollapsed) {
 | 
						|
        throw new Error(
 | 
						|
          "defaultCollapsed is only allowed for default toolbars."
 | 
						|
        );
 | 
						|
      }
 | 
						|
      if (!props.has("defaultCollapsed")) {
 | 
						|
        props.set("defaultCollapsed", true);
 | 
						|
      }
 | 
						|
    } else if (props.has("defaultCollapsed")) {
 | 
						|
      throw new Error("defaultCollapsed only applies for TYPE_TOOLBAR areas.");
 | 
						|
    }
 | 
						|
    // Sanity check type:
 | 
						|
    let allTypes = [CustomizableUI.TYPE_TOOLBAR, CustomizableUI.TYPE_PANEL];
 | 
						|
    if (!allTypes.includes(props.get("type"))) {
 | 
						|
      throw new Error("Invalid area type " + props.get("type"));
 | 
						|
    }
 | 
						|
 | 
						|
    // And to no placements:
 | 
						|
    if (!props.has("defaultPlacements")) {
 | 
						|
      props.set("defaultPlacements", []);
 | 
						|
    }
 | 
						|
    // Sanity check default placements array:
 | 
						|
    if (!Array.isArray(props.get("defaultPlacements"))) {
 | 
						|
      throw new Error("Should provide an array of default placements");
 | 
						|
    }
 | 
						|
 | 
						|
    if (!areaIsKnown) {
 | 
						|
      gAreas.set(aName, props);
 | 
						|
 | 
						|
      // Reconcile new default widgets. Have to do this before we start restoring things.
 | 
						|
      this._placeNewDefaultWidgetsInArea(aName);
 | 
						|
 | 
						|
      if (
 | 
						|
        props.get("type") == CustomizableUI.TYPE_TOOLBAR &&
 | 
						|
        !gPlacements.has(aName)
 | 
						|
      ) {
 | 
						|
        // Guarantee this area exists in gFuturePlacements, to avoid checking it in
 | 
						|
        // various places elsewhere.
 | 
						|
        if (!gFuturePlacements.has(aName)) {
 | 
						|
          gFuturePlacements.set(aName, new Set());
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        this.restoreStateForArea(aName);
 | 
						|
      }
 | 
						|
 | 
						|
      // If we have pending build area nodes, register all of them
 | 
						|
      if (gPendingBuildAreas.has(aName)) {
 | 
						|
        let pendingNodes = gPendingBuildAreas.get(aName);
 | 
						|
        for (let pendingNode of pendingNodes) {
 | 
						|
          this.registerToolbarNode(pendingNode);
 | 
						|
        }
 | 
						|
        gPendingBuildAreas.delete(aName);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  unregisterArea(aName, aDestroyPlacements) {
 | 
						|
    if (typeof aName != "string" || !/^[a-z0-9-_]{1,}$/i.test(aName)) {
 | 
						|
      throw new Error("Invalid area name");
 | 
						|
    }
 | 
						|
    if (!gAreas.has(aName) && !gPlacements.has(aName)) {
 | 
						|
      throw new Error("Area not registered");
 | 
						|
    }
 | 
						|
 | 
						|
    // Move all the widgets out
 | 
						|
    this.beginBatchUpdate();
 | 
						|
    try {
 | 
						|
      let placements = gPlacements.get(aName);
 | 
						|
      if (placements) {
 | 
						|
        // Need to clone this array so removeWidgetFromArea doesn't modify it
 | 
						|
        placements = [...placements];
 | 
						|
        placements.forEach(this.removeWidgetFromArea, this);
 | 
						|
      }
 | 
						|
 | 
						|
      // Delete all remaining traces.
 | 
						|
      gAreas.delete(aName);
 | 
						|
      // Only destroy placements when necessary:
 | 
						|
      if (aDestroyPlacements) {
 | 
						|
        gPlacements.delete(aName);
 | 
						|
      } else {
 | 
						|
        // Otherwise we need to re-set them, as removeFromArea will have emptied
 | 
						|
        // them out:
 | 
						|
        gPlacements.set(aName, placements);
 | 
						|
      }
 | 
						|
      gFuturePlacements.delete(aName);
 | 
						|
      let existingAreaNodes = gBuildAreas.get(aName);
 | 
						|
      if (existingAreaNodes) {
 | 
						|
        for (let areaNode of existingAreaNodes) {
 | 
						|
          this.notifyListeners(
 | 
						|
            "onAreaNodeUnregistered",
 | 
						|
            aName,
 | 
						|
            this.getCustomizationTarget(areaNode),
 | 
						|
            CustomizableUI.REASON_AREA_UNREGISTERED
 | 
						|
          );
 | 
						|
        }
 | 
						|
      }
 | 
						|
      gBuildAreas.delete(aName);
 | 
						|
    } finally {
 | 
						|
      this.endBatchUpdate(true);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  registerToolbarNode(aToolbar) {
 | 
						|
    let area = aToolbar.id;
 | 
						|
    if (gBuildAreas.has(area) && gBuildAreas.get(area).has(aToolbar)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let areaProperties = gAreas.get(area);
 | 
						|
 | 
						|
    // If this area is not registered, try to do it automatically:
 | 
						|
    if (!areaProperties) {
 | 
						|
      if (!gPendingBuildAreas.has(area)) {
 | 
						|
        gPendingBuildAreas.set(area, []);
 | 
						|
      }
 | 
						|
      gPendingBuildAreas.get(area).push(aToolbar);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    this.beginBatchUpdate();
 | 
						|
    try {
 | 
						|
      let placements = gPlacements.get(area);
 | 
						|
      if (
 | 
						|
        !placements &&
 | 
						|
        areaProperties.get("type") == CustomizableUI.TYPE_TOOLBAR
 | 
						|
      ) {
 | 
						|
        this.restoreStateForArea(area);
 | 
						|
        placements = gPlacements.get(area);
 | 
						|
      }
 | 
						|
 | 
						|
      // For toolbars that need it, mark as dirty.
 | 
						|
      let defaultPlacements = areaProperties.get("defaultPlacements");
 | 
						|
      if (
 | 
						|
        !this._builtinToolbars.has(area) ||
 | 
						|
        placements.length != defaultPlacements.length ||
 | 
						|
        !placements.every((id, i) => id == defaultPlacements[i])
 | 
						|
      ) {
 | 
						|
        gDirtyAreaCache.add(area);
 | 
						|
      }
 | 
						|
 | 
						|
      if (areaProperties.get("overflowable")) {
 | 
						|
        aToolbar.overflowable = new OverflowableToolbar(aToolbar);
 | 
						|
      }
 | 
						|
 | 
						|
      this.registerBuildArea(area, aToolbar);
 | 
						|
 | 
						|
      // We only build the toolbar if it's been marked as "dirty". Dirty means
 | 
						|
      // one of the following things:
 | 
						|
      // 1) Items have been added, moved or removed from this toolbar before.
 | 
						|
      // 2) The number of children of the toolbar does not match the length of
 | 
						|
      //    the placements array for that area.
 | 
						|
      //
 | 
						|
      // This notion of being "dirty" is stored in a cache which is persisted
 | 
						|
      // in the saved state.
 | 
						|
      //
 | 
						|
      // Secondly, if the list of placements contains an API-provided widget,
 | 
						|
      // we need to call `buildArea` or it won't be built and put in the toolbar.
 | 
						|
      if (
 | 
						|
        gDirtyAreaCache.has(area) ||
 | 
						|
        placements.some(id => gPalette.has(id))
 | 
						|
      ) {
 | 
						|
        this.buildArea(area, placements, aToolbar);
 | 
						|
      } else {
 | 
						|
        // We must have a builtin toolbar that's in the default state. We need
 | 
						|
        // to only make sure that all the special nodes are correct.
 | 
						|
        let specials = placements.filter(p => this.isSpecialWidget(p));
 | 
						|
        if (specials.length) {
 | 
						|
          this.updateSpecialsForBuiltinToolbar(aToolbar, specials);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      this.notifyListeners(
 | 
						|
        "onAreaNodeRegistered",
 | 
						|
        area,
 | 
						|
        this.getCustomizationTarget(aToolbar)
 | 
						|
      );
 | 
						|
    } finally {
 | 
						|
      this.endBatchUpdate();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  updateSpecialsForBuiltinToolbar(aToolbar, aSpecialIDs) {
 | 
						|
    // Nodes are going to be in the correct order, so we can do this straightforwardly:
 | 
						|
    let { children } = this.getCustomizationTarget(aToolbar);
 | 
						|
    for (let kid of children) {
 | 
						|
      if (
 | 
						|
        this.matchingSpecials(aSpecialIDs[0], kid) &&
 | 
						|
        kid.getAttribute("skipintoolbarset") != "true"
 | 
						|
      ) {
 | 
						|
        kid.id = aSpecialIDs.shift();
 | 
						|
      }
 | 
						|
      if (!aSpecialIDs.length) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  buildArea(aArea, aPlacements, aAreaNode) {
 | 
						|
    let document = aAreaNode.ownerDocument;
 | 
						|
    let window = document.defaultView;
 | 
						|
    let inPrivateWindow = lazy.PrivateBrowsingUtils.isWindowPrivate(window);
 | 
						|
    let container = this.getCustomizationTarget(aAreaNode);
 | 
						|
    let areaIsPanel =
 | 
						|
      gAreas.get(aArea).get("type") == CustomizableUI.TYPE_PANEL;
 | 
						|
 | 
						|
    if (!container) {
 | 
						|
      throw new Error(
 | 
						|
        "Expected area " + aArea + " to have a customizationTarget attribute."
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    // Restore nav-bar visibility since it may have been hidden
 | 
						|
    // through a migration path (bug 938980) or an add-on.
 | 
						|
    if (aArea == CustomizableUI.AREA_NAVBAR) {
 | 
						|
      aAreaNode.collapsed = false;
 | 
						|
    }
 | 
						|
 | 
						|
    this.beginBatchUpdate();
 | 
						|
 | 
						|
    try {
 | 
						|
      let currentNode = container.firstElementChild;
 | 
						|
      let placementsToRemove = new Set();
 | 
						|
      for (let id of aPlacements) {
 | 
						|
        while (
 | 
						|
          currentNode &&
 | 
						|
          currentNode.getAttribute("skipintoolbarset") == "true"
 | 
						|
        ) {
 | 
						|
          currentNode = currentNode.nextElementSibling;
 | 
						|
        }
 | 
						|
 | 
						|
        // Fix ids for specials and continue, for correctly placed specials.
 | 
						|
        if (
 | 
						|
          currentNode &&
 | 
						|
          (!currentNode.id || CustomizableUI.isSpecialWidget(currentNode)) &&
 | 
						|
          this.matchingSpecials(id, currentNode)
 | 
						|
        ) {
 | 
						|
          currentNode.id = id;
 | 
						|
        }
 | 
						|
        if (currentNode && currentNode.id == id) {
 | 
						|
          currentNode = currentNode.nextElementSibling;
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
 | 
						|
        if (this.isSpecialWidget(id) && areaIsPanel) {
 | 
						|
          placementsToRemove.add(id);
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
 | 
						|
        let [provider, node] = this.getWidgetNode(id, window);
 | 
						|
        if (!node) {
 | 
						|
          lazy.log.debug("Unknown widget: " + id);
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
 | 
						|
        let widget = null;
 | 
						|
        // If the placements have items in them which are (now) no longer removable,
 | 
						|
        // we shouldn't be moving them:
 | 
						|
        if (provider == CustomizableUI.PROVIDER_API) {
 | 
						|
          widget = gPalette.get(id);
 | 
						|
          if (!widget.removable && aArea != widget.defaultArea) {
 | 
						|
            placementsToRemove.add(id);
 | 
						|
            continue;
 | 
						|
          }
 | 
						|
        } else if (
 | 
						|
          provider == CustomizableUI.PROVIDER_XUL &&
 | 
						|
          node.parentNode != container &&
 | 
						|
          !this.isWidgetRemovable(node)
 | 
						|
        ) {
 | 
						|
          placementsToRemove.add(id);
 | 
						|
          continue;
 | 
						|
        } // Special widgets are always removable, so no need to check them
 | 
						|
 | 
						|
        if (inPrivateWindow && widget && !widget.showInPrivateBrowsing) {
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
 | 
						|
        this.ensureButtonContextMenu(node, aAreaNode);
 | 
						|
 | 
						|
        // This needs updating in case we're resetting / undoing a reset.
 | 
						|
        if (widget) {
 | 
						|
          widget.currentArea = aArea;
 | 
						|
        }
 | 
						|
        this.insertWidgetBefore(node, currentNode, container, aArea);
 | 
						|
        if (gResetting) {
 | 
						|
          this.notifyListeners("onWidgetReset", node, container);
 | 
						|
        } else if (gUndoResetting) {
 | 
						|
          this.notifyListeners("onWidgetUndoMove", node, container);
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (currentNode) {
 | 
						|
        let palette = window.gNavToolbox ? window.gNavToolbox.palette : null;
 | 
						|
        let limit = currentNode.previousElementSibling;
 | 
						|
        let node = container.lastElementChild;
 | 
						|
        while (node && node != limit) {
 | 
						|
          let previousSibling = node.previousElementSibling;
 | 
						|
          // Nodes opt-in to removability. If they're removable, and we haven't
 | 
						|
          // seen them in the placements array, then we toss them into the palette
 | 
						|
          // if one exists. If no palette exists, we just remove the node. If the
 | 
						|
          // node is not removable, we leave it where it is. However, we can only
 | 
						|
          // safely touch elements that have an ID - both because we depend on
 | 
						|
          // IDs (or are specials), and because such elements are not intended to
 | 
						|
          // be widgets (eg, titlebar-spacer elements).
 | 
						|
          if (
 | 
						|
            (node.id || this.isSpecialWidget(node)) &&
 | 
						|
            node.getAttribute("skipintoolbarset") != "true"
 | 
						|
          ) {
 | 
						|
            if (this.isWidgetRemovable(node)) {
 | 
						|
              if (node.id && (gResetting || gUndoResetting)) {
 | 
						|
                let widget = gPalette.get(node.id);
 | 
						|
                if (widget) {
 | 
						|
                  widget.currentArea = null;
 | 
						|
                }
 | 
						|
              }
 | 
						|
              this.notifyDOMChange(node, null, container, true, () => {
 | 
						|
                if (palette && !this.isSpecialWidget(node.id)) {
 | 
						|
                  palette.appendChild(node);
 | 
						|
                  this.removeLocationAttributes(node);
 | 
						|
                } else {
 | 
						|
                  container.removeChild(node);
 | 
						|
                }
 | 
						|
              });
 | 
						|
            } else {
 | 
						|
              node.setAttribute("removable", false);
 | 
						|
              lazy.log.debug(
 | 
						|
                "Adding non-removable widget to placements of " +
 | 
						|
                  aArea +
 | 
						|
                  ": " +
 | 
						|
                  node.id
 | 
						|
              );
 | 
						|
              gPlacements.get(aArea).push(node.id);
 | 
						|
              gDirty = true;
 | 
						|
            }
 | 
						|
          }
 | 
						|
          node = previousSibling;
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      // If there are placements in here which aren't removable from their original area,
 | 
						|
      // we remove them from this area's placement array. They will (have) be(en) added
 | 
						|
      // to their original area's placements array in the block above this one.
 | 
						|
      if (placementsToRemove.size) {
 | 
						|
        let placementAry = gPlacements.get(aArea);
 | 
						|
        for (let id of placementsToRemove) {
 | 
						|
          let index = placementAry.indexOf(id);
 | 
						|
          placementAry.splice(index, 1);
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (gResetting) {
 | 
						|
        this.notifyListeners("onAreaReset", aArea, container);
 | 
						|
      }
 | 
						|
    } finally {
 | 
						|
      this.endBatchUpdate();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  addPanelCloseListeners(aPanel) {
 | 
						|
    aPanel.addEventListener("click", this, { mozSystemGroup: true });
 | 
						|
    aPanel.addEventListener("keypress", this, { mozSystemGroup: true });
 | 
						|
    let win = aPanel.ownerGlobal;
 | 
						|
    if (!gPanelsForWindow.has(win)) {
 | 
						|
      gPanelsForWindow.set(win, new Set());
 | 
						|
    }
 | 
						|
    gPanelsForWindow.get(win).add(this._getPanelForNode(aPanel));
 | 
						|
  },
 | 
						|
 | 
						|
  removePanelCloseListeners(aPanel) {
 | 
						|
    aPanel.removeEventListener("click", this, { mozSystemGroup: true });
 | 
						|
    aPanel.removeEventListener("keypress", this, { mozSystemGroup: true });
 | 
						|
    let win = aPanel.ownerGlobal;
 | 
						|
    let panels = gPanelsForWindow.get(win);
 | 
						|
    if (panels) {
 | 
						|
      panels.delete(this._getPanelForNode(aPanel));
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  ensureButtonContextMenu(aNode, aAreaNode, forcePanel) {
 | 
						|
    const kPanelItemContextMenu = "customizationPanelItemContextMenu";
 | 
						|
 | 
						|
    let currentContextMenu =
 | 
						|
      aNode.getAttribute("context") || aNode.getAttribute("contextmenu");
 | 
						|
    let contextMenuForPlace;
 | 
						|
 | 
						|
    if (
 | 
						|
      CustomizableUI.isWebExtensionWidget(aNode.id) &&
 | 
						|
      (aAreaNode?.id == CustomizableUI.AREA_ADDONS ||
 | 
						|
        aNode.getAttribute("overflowedItem") == "true")
 | 
						|
    ) {
 | 
						|
      contextMenuForPlace = null;
 | 
						|
    } else {
 | 
						|
      contextMenuForPlace =
 | 
						|
        forcePanel || "panel" == CustomizableUI.getPlaceForItem(aAreaNode)
 | 
						|
          ? kPanelItemContextMenu
 | 
						|
          : null;
 | 
						|
    }
 | 
						|
    if (contextMenuForPlace && !currentContextMenu) {
 | 
						|
      aNode.setAttribute("context", contextMenuForPlace);
 | 
						|
    } else if (
 | 
						|
      currentContextMenu == kPanelItemContextMenu &&
 | 
						|
      contextMenuForPlace != kPanelItemContextMenu
 | 
						|
    ) {
 | 
						|
      aNode.removeAttribute("context");
 | 
						|
      aNode.removeAttribute("contextmenu");
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  getWidgetProvider(aWidgetId) {
 | 
						|
    if (this.isSpecialWidget(aWidgetId)) {
 | 
						|
      return CustomizableUI.PROVIDER_SPECIAL;
 | 
						|
    }
 | 
						|
    if (gPalette.has(aWidgetId)) {
 | 
						|
      return CustomizableUI.PROVIDER_API;
 | 
						|
    }
 | 
						|
    // If this was an API widget that was destroyed, return null:
 | 
						|
    if (gSeenWidgets.has(aWidgetId)) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    // We fall back to the XUL provider, but we don't know for sure (at this
 | 
						|
    // point) whether it exists there either. So the API is technically lying.
 | 
						|
    // Ideally, it would be able to return an error value (or throw an
 | 
						|
    // exception) if it really didn't exist. Our code calling this function
 | 
						|
    // handles that fine, but this is a public API.
 | 
						|
    return CustomizableUI.PROVIDER_XUL;
 | 
						|
  },
 | 
						|
 | 
						|
  getWidgetNode(aWidgetId, aWindow) {
 | 
						|
    let document = aWindow.document;
 | 
						|
 | 
						|
    if (this.isSpecialWidget(aWidgetId)) {
 | 
						|
      let widgetNode =
 | 
						|
        document.getElementById(aWidgetId) ||
 | 
						|
        this.createSpecialWidget(aWidgetId, document);
 | 
						|
      return [CustomizableUI.PROVIDER_SPECIAL, widgetNode];
 | 
						|
    }
 | 
						|
 | 
						|
    let widget = gPalette.get(aWidgetId);
 | 
						|
    if (widget) {
 | 
						|
      // If we have an instance of this widget already, just use that.
 | 
						|
      if (widget.instances.has(document)) {
 | 
						|
        lazy.log.debug(
 | 
						|
          "An instance of widget " +
 | 
						|
            aWidgetId +
 | 
						|
            " already exists in this " +
 | 
						|
            "document. Reusing."
 | 
						|
        );
 | 
						|
        return [CustomizableUI.PROVIDER_API, widget.instances.get(document)];
 | 
						|
      }
 | 
						|
 | 
						|
      return [CustomizableUI.PROVIDER_API, this.buildWidget(document, widget)];
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("Searching for " + aWidgetId + " in toolbox.");
 | 
						|
    let node = this.findWidgetInWindow(aWidgetId, aWindow);
 | 
						|
    if (node) {
 | 
						|
      return [CustomizableUI.PROVIDER_XUL, node];
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("No node for " + aWidgetId + " found.");
 | 
						|
    return [null, null];
 | 
						|
  },
 | 
						|
 | 
						|
  registerPanelNode(aNode, aArea) {
 | 
						|
    if (gBuildAreas.has(aArea) && gBuildAreas.get(aArea).has(aNode)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    aNode._customizationTarget = aNode;
 | 
						|
    this.addPanelCloseListeners(this._getPanelForNode(aNode));
 | 
						|
 | 
						|
    let placements = gPlacements.get(aArea);
 | 
						|
    this.buildArea(aArea, placements, aNode);
 | 
						|
    this.notifyListeners("onAreaNodeRegistered", aArea, aNode);
 | 
						|
 | 
						|
    for (let child of aNode.children) {
 | 
						|
      if (child.localName != "toolbarbutton") {
 | 
						|
        if (child.localName == "toolbaritem") {
 | 
						|
          this.ensureButtonContextMenu(child, aNode, true);
 | 
						|
        }
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      this.ensureButtonContextMenu(child, aNode, true);
 | 
						|
    }
 | 
						|
 | 
						|
    this.registerBuildArea(aArea, aNode);
 | 
						|
  },
 | 
						|
 | 
						|
  onWidgetAdded(aWidgetId, aArea, aPosition) {
 | 
						|
    this.insertNode(aWidgetId, aArea, aPosition, true);
 | 
						|
 | 
						|
    if (!gResetting) {
 | 
						|
      this._clearPreviousUIState();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  onWidgetRemoved(aWidgetId, aArea) {
 | 
						|
    let areaNodes = gBuildAreas.get(aArea);
 | 
						|
    if (!areaNodes) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let area = gAreas.get(aArea);
 | 
						|
    let isToolbar = area.get("type") == CustomizableUI.TYPE_TOOLBAR;
 | 
						|
    let isOverflowable = isToolbar && area.get("overflowable");
 | 
						|
    let showInPrivateBrowsing = gPalette.has(aWidgetId)
 | 
						|
      ? gPalette.get(aWidgetId).showInPrivateBrowsing
 | 
						|
      : true;
 | 
						|
 | 
						|
    for (let areaNode of areaNodes) {
 | 
						|
      let window = areaNode.ownerGlobal;
 | 
						|
      if (
 | 
						|
        !showInPrivateBrowsing &&
 | 
						|
        lazy.PrivateBrowsingUtils.isWindowPrivate(window)
 | 
						|
      ) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
 | 
						|
      let container = this.getCustomizationTarget(areaNode);
 | 
						|
      let widgetNode = window.document.getElementById(aWidgetId);
 | 
						|
      if (widgetNode && isOverflowable) {
 | 
						|
        container = areaNode.overflowable.getContainerFor(widgetNode);
 | 
						|
      }
 | 
						|
 | 
						|
      if (!widgetNode || !container.contains(widgetNode)) {
 | 
						|
        lazy.log.info(
 | 
						|
          "Widget " + aWidgetId + " not found, unable to remove from " + aArea
 | 
						|
        );
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
 | 
						|
      this.notifyDOMChange(widgetNode, null, container, true, () => {
 | 
						|
        // We remove location attributes here to make sure they're gone too when a
 | 
						|
        // widget is removed from a toolbar to the palette. See bug 930950.
 | 
						|
        this.removeLocationAttributes(widgetNode);
 | 
						|
        // We also need to remove the panel context menu if it's there:
 | 
						|
        this.ensureButtonContextMenu(widgetNode);
 | 
						|
        if (gPalette.has(aWidgetId) || this.isSpecialWidget(aWidgetId)) {
 | 
						|
          container.removeChild(widgetNode);
 | 
						|
        } else {
 | 
						|
          window.gNavToolbox.palette.appendChild(widgetNode);
 | 
						|
        }
 | 
						|
      });
 | 
						|
 | 
						|
      let windowCache = gSingleWrapperCache.get(window);
 | 
						|
      if (windowCache) {
 | 
						|
        windowCache.delete(aWidgetId);
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (!gResetting) {
 | 
						|
      this._clearPreviousUIState();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  onWidgetMoved(aWidgetId, aArea, aOldPosition, aNewPosition) {
 | 
						|
    this.insertNode(aWidgetId, aArea, aNewPosition);
 | 
						|
    if (!gResetting) {
 | 
						|
      this._clearPreviousUIState();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  onCustomizeEnd() {
 | 
						|
    this._clearPreviousUIState();
 | 
						|
  },
 | 
						|
 | 
						|
  registerBuildArea(aArea, aNode) {
 | 
						|
    // We ensure that the window is registered to have its customization data
 | 
						|
    // cleaned up when unloading.
 | 
						|
    let window = aNode.ownerGlobal;
 | 
						|
    if (window.closed) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    this.registerBuildWindow(window);
 | 
						|
 | 
						|
    // Also register this build area's toolbox.
 | 
						|
    if (window.gNavToolbox) {
 | 
						|
      gBuildWindows.get(window).add(window.gNavToolbox);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!gBuildAreas.has(aArea)) {
 | 
						|
      gBuildAreas.set(aArea, new Set());
 | 
						|
    }
 | 
						|
 | 
						|
    gBuildAreas.get(aArea).add(aNode);
 | 
						|
 | 
						|
    // Give a class to all customize targets to be used for styling in Customize Mode
 | 
						|
    let customizableNode = this.getCustomizeTargetForArea(aArea, window);
 | 
						|
    customizableNode.classList.add("customization-target");
 | 
						|
  },
 | 
						|
 | 
						|
  registerBuildWindow(aWindow) {
 | 
						|
    if (!gBuildWindows.has(aWindow)) {
 | 
						|
      gBuildWindows.set(aWindow, new Set());
 | 
						|
 | 
						|
      aWindow.addEventListener("unload", this);
 | 
						|
      aWindow.addEventListener("command", this, true);
 | 
						|
 | 
						|
      this.notifyListeners("onWindowOpened", aWindow);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  unregisterBuildWindow(aWindow) {
 | 
						|
    aWindow.removeEventListener("unload", this);
 | 
						|
    aWindow.removeEventListener("command", this, true);
 | 
						|
    gPanelsForWindow.delete(aWindow);
 | 
						|
    gBuildWindows.delete(aWindow);
 | 
						|
    gSingleWrapperCache.delete(aWindow);
 | 
						|
    let document = aWindow.document;
 | 
						|
 | 
						|
    for (let [areaId, areaNodes] of gBuildAreas) {
 | 
						|
      let areaProperties = gAreas.get(areaId);
 | 
						|
      for (let node of areaNodes) {
 | 
						|
        if (node.ownerDocument == document) {
 | 
						|
          this.notifyListeners(
 | 
						|
            "onAreaNodeUnregistered",
 | 
						|
            areaId,
 | 
						|
            this.getCustomizationTarget(node),
 | 
						|
            CustomizableUI.REASON_WINDOW_CLOSED
 | 
						|
          );
 | 
						|
          if (areaProperties.get("overflowable")) {
 | 
						|
            node.overflowable.uninit();
 | 
						|
            node.overflowable = null;
 | 
						|
          }
 | 
						|
          areaNodes.delete(node);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    for (let [, widget] of gPalette) {
 | 
						|
      widget.instances.delete(document);
 | 
						|
      this.notifyListeners("onWidgetInstanceRemoved", widget.id, document);
 | 
						|
    }
 | 
						|
 | 
						|
    for (let [, pendingNodes] of gPendingBuildAreas) {
 | 
						|
      for (let i = pendingNodes.length - 1; i >= 0; i--) {
 | 
						|
        if (pendingNodes[i].ownerDocument == document) {
 | 
						|
          pendingNodes.splice(i, 1);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    this.notifyListeners("onWindowClosed", aWindow);
 | 
						|
  },
 | 
						|
 | 
						|
  setLocationAttributes(aNode, aArea) {
 | 
						|
    let props = gAreas.get(aArea);
 | 
						|
    if (!props) {
 | 
						|
      throw new Error(
 | 
						|
        "Expected area " +
 | 
						|
          aArea +
 | 
						|
          " to have a properties Map " +
 | 
						|
          "associated with it."
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    aNode.setAttribute("cui-areatype", props.get("type") || "");
 | 
						|
    let anchor = props.get("anchor");
 | 
						|
    if (anchor) {
 | 
						|
      aNode.setAttribute("cui-anchorid", anchor);
 | 
						|
    } else {
 | 
						|
      aNode.removeAttribute("cui-anchorid");
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  removeLocationAttributes(aNode) {
 | 
						|
    aNode.removeAttribute("cui-areatype");
 | 
						|
    aNode.removeAttribute("cui-anchorid");
 | 
						|
  },
 | 
						|
 | 
						|
  insertNode(aWidgetId, aArea, aPosition, isNew) {
 | 
						|
    let areaNodes = gBuildAreas.get(aArea);
 | 
						|
    if (!areaNodes) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let placements = gPlacements.get(aArea);
 | 
						|
    if (!placements) {
 | 
						|
      lazy.log.error(
 | 
						|
        "Could not find any placements for " + aArea + " when moving a widget."
 | 
						|
      );
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Go through each of the nodes associated with this area and move the
 | 
						|
    // widget to the requested location.
 | 
						|
    for (let areaNode of areaNodes) {
 | 
						|
      this.insertNodeInWindow(aWidgetId, areaNode, isNew);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  insertNodeInWindow(aWidgetId, aAreaNode, isNew) {
 | 
						|
    let window = aAreaNode.ownerGlobal;
 | 
						|
    let showInPrivateBrowsing = gPalette.has(aWidgetId)
 | 
						|
      ? gPalette.get(aWidgetId).showInPrivateBrowsing
 | 
						|
      : true;
 | 
						|
 | 
						|
    if (
 | 
						|
      !showInPrivateBrowsing &&
 | 
						|
      lazy.PrivateBrowsingUtils.isWindowPrivate(window)
 | 
						|
    ) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let [, widgetNode] = this.getWidgetNode(aWidgetId, window);
 | 
						|
    if (!widgetNode) {
 | 
						|
      lazy.log.error("Widget '" + aWidgetId + "' not found, unable to move");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let areaId = aAreaNode.id;
 | 
						|
    if (isNew) {
 | 
						|
      this.ensureButtonContextMenu(widgetNode, aAreaNode);
 | 
						|
    }
 | 
						|
 | 
						|
    let [insertionContainer, nextNode] = this.findInsertionPoints(
 | 
						|
      widgetNode,
 | 
						|
      aAreaNode
 | 
						|
    );
 | 
						|
    this.insertWidgetBefore(widgetNode, nextNode, insertionContainer, areaId);
 | 
						|
  },
 | 
						|
 | 
						|
  findInsertionPoints(aNode, aAreaNode) {
 | 
						|
    let areaId = aAreaNode.id;
 | 
						|
    let props = gAreas.get(areaId);
 | 
						|
 | 
						|
    // For overflowable toolbars, rely on them (because the work is more complicated):
 | 
						|
    if (
 | 
						|
      props.get("type") == CustomizableUI.TYPE_TOOLBAR &&
 | 
						|
      props.get("overflowable")
 | 
						|
    ) {
 | 
						|
      return aAreaNode.overflowable.findOverflowedInsertionPoints(aNode);
 | 
						|
    }
 | 
						|
 | 
						|
    let container = this.getCustomizationTarget(aAreaNode);
 | 
						|
    let placements = gPlacements.get(areaId);
 | 
						|
    let nodeIndex = placements.indexOf(aNode.id);
 | 
						|
 | 
						|
    while (++nodeIndex < placements.length) {
 | 
						|
      let nextNodeId = placements[nodeIndex];
 | 
						|
      // We use aAreaNode here, because if aNode is in a template, its
 | 
						|
      // `ownerDocument` is *not* going to be the browser.xhtml document,
 | 
						|
      // so we cannot rely on it.
 | 
						|
      let nextNode = aAreaNode.ownerDocument.getElementById(nextNodeId);
 | 
						|
      // If the next placed widget exists, and is a direct child of the
 | 
						|
      // container, or wrapped in a customize mode wrapper (toolbarpaletteitem)
 | 
						|
      // inside the container, insert beside it.
 | 
						|
      // We have to check the parent to avoid errors when the placement ids
 | 
						|
      // are for nodes that are no longer customizable.
 | 
						|
      if (
 | 
						|
        nextNode &&
 | 
						|
        (nextNode.parentNode == container ||
 | 
						|
          (nextNode.parentNode.localName == "toolbarpaletteitem" &&
 | 
						|
            nextNode.parentNode.parentNode == container))
 | 
						|
      ) {
 | 
						|
        return [container, nextNode];
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return [container, null];
 | 
						|
  },
 | 
						|
 | 
						|
  insertWidgetBefore(aNode, aNextNode, aContainer, aArea) {
 | 
						|
    this.notifyDOMChange(aNode, aNextNode, aContainer, false, () => {
 | 
						|
      this.setLocationAttributes(aNode, aArea);
 | 
						|
      aContainer.insertBefore(aNode, aNextNode);
 | 
						|
    });
 | 
						|
  },
 | 
						|
 | 
						|
  notifyDOMChange(aNode, aNextNode, aContainer, aIsRemove, aCallback) {
 | 
						|
    this.notifyListeners(
 | 
						|
      "onWidgetBeforeDOMChange",
 | 
						|
      aNode,
 | 
						|
      aNextNode,
 | 
						|
      aContainer,
 | 
						|
      aIsRemove
 | 
						|
    );
 | 
						|
    aCallback();
 | 
						|
    this.notifyListeners(
 | 
						|
      "onWidgetAfterDOMChange",
 | 
						|
      aNode,
 | 
						|
      aNextNode,
 | 
						|
      aContainer,
 | 
						|
      aIsRemove
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  handleEvent(aEvent) {
 | 
						|
    switch (aEvent.type) {
 | 
						|
      case "command":
 | 
						|
        if (!this._originalEventInPanel(aEvent)) {
 | 
						|
          break;
 | 
						|
        }
 | 
						|
        aEvent = aEvent.sourceEvent;
 | 
						|
      // Fall through
 | 
						|
      case "click":
 | 
						|
      case "keypress":
 | 
						|
        this.maybeAutoHidePanel(aEvent);
 | 
						|
        break;
 | 
						|
      case "unload":
 | 
						|
        this.unregisterBuildWindow(aEvent.currentTarget);
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _originalEventInPanel(aEvent) {
 | 
						|
    let e = aEvent.sourceEvent;
 | 
						|
    if (!e) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    let node = this._getPanelForNode(e.target);
 | 
						|
    if (!node) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    let win = e.view;
 | 
						|
    let panels = gPanelsForWindow.get(win);
 | 
						|
    return !!panels && panels.has(node);
 | 
						|
  },
 | 
						|
 | 
						|
  _getSpecialIdForNode(aNode) {
 | 
						|
    if (typeof aNode == "object" && aNode.localName) {
 | 
						|
      if (aNode.id) {
 | 
						|
        return aNode.id;
 | 
						|
      }
 | 
						|
      if (aNode.localName.startsWith("toolbar")) {
 | 
						|
        return aNode.localName.substring(7);
 | 
						|
      }
 | 
						|
      return "";
 | 
						|
    }
 | 
						|
    return aNode;
 | 
						|
  },
 | 
						|
 | 
						|
  isSpecialWidget(aId) {
 | 
						|
    aId = this._getSpecialIdForNode(aId);
 | 
						|
    return (
 | 
						|
      aId.startsWith(kSpecialWidgetPfx) ||
 | 
						|
      aId.startsWith("separator") ||
 | 
						|
      aId.startsWith("spring") ||
 | 
						|
      aId.startsWith("spacer")
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  matchingSpecials(aId1, aId2) {
 | 
						|
    aId1 = this._getSpecialIdForNode(aId1);
 | 
						|
    aId2 = this._getSpecialIdForNode(aId2);
 | 
						|
 | 
						|
    return (
 | 
						|
      this.isSpecialWidget(aId1) &&
 | 
						|
      this.isSpecialWidget(aId2) &&
 | 
						|
      aId1.match(/spring|spacer|separator/)[0] ==
 | 
						|
        aId2.match(/spring|spacer|separator/)[0]
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  ensureSpecialWidgetId(aId) {
 | 
						|
    let nodeType = aId.match(/spring|spacer|separator/)[0];
 | 
						|
    // If the ID we were passed isn't a generated one, generate one now:
 | 
						|
    if (nodeType == aId) {
 | 
						|
      // Ids are differentiated through a unique count suffix.
 | 
						|
      return kSpecialWidgetPfx + aId + ++gNewElementCount;
 | 
						|
    }
 | 
						|
    return aId;
 | 
						|
  },
 | 
						|
 | 
						|
  createSpecialWidget(aId, aDocument) {
 | 
						|
    let nodeName = "toolbar" + aId.match(/spring|spacer|separator/)[0];
 | 
						|
    let node = aDocument.createXULElement(nodeName);
 | 
						|
    node.className = "chromeclass-toolbar-additional";
 | 
						|
    node.id = this.ensureSpecialWidgetId(aId);
 | 
						|
    return node;
 | 
						|
  },
 | 
						|
 | 
						|
  /* Find a XUL-provided widget in a window. Don't try to use this
 | 
						|
   * for an API-provided widget or a special widget.
 | 
						|
   */
 | 
						|
  findWidgetInWindow(aId, aWindow) {
 | 
						|
    if (!gBuildWindows.has(aWindow)) {
 | 
						|
      throw new Error("Build window not registered");
 | 
						|
    }
 | 
						|
 | 
						|
    if (!aId) {
 | 
						|
      lazy.log.error("findWidgetInWindow was passed an empty string.");
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    let document = aWindow.document;
 | 
						|
 | 
						|
    // look for a node with the same id, as the node may be
 | 
						|
    // in a different toolbar.
 | 
						|
    let node = document.getElementById(aId);
 | 
						|
    if (node) {
 | 
						|
      let parent = node.parentNode;
 | 
						|
      while (
 | 
						|
        parent &&
 | 
						|
        !(
 | 
						|
          this.getCustomizationTarget(parent) ||
 | 
						|
          parent == aWindow.gNavToolbox.palette
 | 
						|
        )
 | 
						|
      ) {
 | 
						|
        parent = parent.parentNode;
 | 
						|
      }
 | 
						|
 | 
						|
      if (parent) {
 | 
						|
        let nodeInArea =
 | 
						|
          node.parentNode.localName == "toolbarpaletteitem"
 | 
						|
            ? node.parentNode
 | 
						|
            : node;
 | 
						|
        // Check if we're in a customization target, or in the palette:
 | 
						|
        if (
 | 
						|
          (this.getCustomizationTarget(parent) == nodeInArea.parentNode &&
 | 
						|
            gBuildWindows.get(aWindow).has(aWindow.gNavToolbox)) ||
 | 
						|
          aWindow.gNavToolbox.palette == nodeInArea.parentNode
 | 
						|
        ) {
 | 
						|
          // Normalize the removable attribute. For backwards compat, if
 | 
						|
          // the widget is not located in a toolbox palette then absence
 | 
						|
          // of the "removable" attribute means it is not removable.
 | 
						|
          if (!node.hasAttribute("removable")) {
 | 
						|
            // If we first see this in customization mode, it may be in the
 | 
						|
            // customization palette instead of the toolbox palette.
 | 
						|
            node.setAttribute(
 | 
						|
              "removable",
 | 
						|
              !this.getCustomizationTarget(parent)
 | 
						|
            );
 | 
						|
          }
 | 
						|
          return node;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    let toolboxes = gBuildWindows.get(aWindow);
 | 
						|
    for (let toolbox of toolboxes) {
 | 
						|
      if (toolbox.palette) {
 | 
						|
        // Attempt to locate an element with a matching ID within
 | 
						|
        // the palette.
 | 
						|
        let element = toolbox.palette.getElementsByAttribute("id", aId)[0];
 | 
						|
        if (element) {
 | 
						|
          // Normalize the removable attribute. For backwards compat, this
 | 
						|
          // is optional if the widget is located in the toolbox palette,
 | 
						|
          // and defaults to *true*, unlike if it was located elsewhere.
 | 
						|
          if (!element.hasAttribute("removable")) {
 | 
						|
            element.setAttribute("removable", true);
 | 
						|
          }
 | 
						|
          return element;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
    return null;
 | 
						|
  },
 | 
						|
 | 
						|
  buildWidget(aDocument, aWidget) {
 | 
						|
    if (aDocument.documentURI != kExpectedWindowURL) {
 | 
						|
      throw new Error("buildWidget was called for a non-browser window!");
 | 
						|
    }
 | 
						|
    if (typeof aWidget == "string") {
 | 
						|
      aWidget = gPalette.get(aWidget);
 | 
						|
    }
 | 
						|
    if (!aWidget) {
 | 
						|
      throw new Error("buildWidget was passed a non-widget to build.");
 | 
						|
    }
 | 
						|
    if (
 | 
						|
      !aWidget.showInPrivateBrowsing &&
 | 
						|
      lazy.PrivateBrowsingUtils.isWindowPrivate(aDocument.defaultView)
 | 
						|
    ) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("Building " + aWidget.id + " of type " + aWidget.type);
 | 
						|
 | 
						|
    let node;
 | 
						|
    let button;
 | 
						|
    if (aWidget.type == "custom") {
 | 
						|
      if (aWidget.onBuild) {
 | 
						|
        node = aWidget.onBuild(aDocument);
 | 
						|
      }
 | 
						|
      if (
 | 
						|
        !node ||
 | 
						|
        !aDocument.defaultView.XULElement.isInstance(node) ||
 | 
						|
        (aWidget.viewId && !node.viewButton)
 | 
						|
      ) {
 | 
						|
        lazy.log.error(
 | 
						|
          "Custom widget with id " +
 | 
						|
            aWidget.id +
 | 
						|
            " does not return a valid node"
 | 
						|
        );
 | 
						|
      }
 | 
						|
      // A custom widget can define a viewId for the panel and a viewButton
 | 
						|
      // property for the panel anchor.  With that, it will be treated as a view
 | 
						|
      // type where necessary to hook up the view panel.
 | 
						|
      if (aWidget.viewId) {
 | 
						|
        button = node.viewButton;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    // Button and view widget types, plus custom widgets that have a viewId and thus a button.
 | 
						|
    if (button || aWidget.type != "custom") {
 | 
						|
      if (
 | 
						|
        aWidget.onBeforeCreated &&
 | 
						|
        aWidget.onBeforeCreated(aDocument) === false
 | 
						|
      ) {
 | 
						|
        return null;
 | 
						|
      }
 | 
						|
 | 
						|
      if (!button) {
 | 
						|
        button = aDocument.createXULElement("toolbarbutton");
 | 
						|
        node = button;
 | 
						|
      }
 | 
						|
      button.classList.add("toolbarbutton-1");
 | 
						|
      button.setAttribute("delegatesanchor", "true");
 | 
						|
 | 
						|
      let viewbutton = null;
 | 
						|
      if (aWidget.type == "button-and-view") {
 | 
						|
        button.setAttribute("id", aWidget.id + "-button");
 | 
						|
        let dropmarker = aDocument.createXULElement("toolbarbutton");
 | 
						|
        dropmarker.setAttribute("id", aWidget.id + "-dropmarker");
 | 
						|
        dropmarker.setAttribute("delegatesanchor", "true");
 | 
						|
        dropmarker.classList.add(
 | 
						|
          "toolbarbutton-1",
 | 
						|
          "toolbarbutton-combined-buttons-dropmarker"
 | 
						|
        );
 | 
						|
        node = aDocument.createXULElement("toolbaritem");
 | 
						|
        node.classList.add("toolbaritem-combined-buttons");
 | 
						|
        node.append(button, dropmarker);
 | 
						|
        viewbutton = dropmarker;
 | 
						|
      } else if (aWidget.viewId) {
 | 
						|
        // Also set viewbutton for anything with a view
 | 
						|
        viewbutton = button;
 | 
						|
      }
 | 
						|
 | 
						|
      node.setAttribute("id", aWidget.id);
 | 
						|
      node.setAttribute("widget-id", aWidget.id);
 | 
						|
      node.setAttribute("widget-type", aWidget.type);
 | 
						|
      if (aWidget.disabled) {
 | 
						|
        node.setAttribute("disabled", true);
 | 
						|
      }
 | 
						|
      node.setAttribute("removable", aWidget.removable);
 | 
						|
      node.setAttribute("overflows", aWidget.overflows);
 | 
						|
      if (aWidget.tabSpecific) {
 | 
						|
        node.setAttribute("tabspecific", aWidget.tabSpecific);
 | 
						|
      }
 | 
						|
      if (aWidget.locationSpecific) {
 | 
						|
        node.setAttribute("locationspecific", aWidget.locationSpecific);
 | 
						|
      }
 | 
						|
      if (aWidget.keepBroadcastAttributesWhenCustomizing) {
 | 
						|
        node.setAttribute(
 | 
						|
          "keepbroadcastattributeswhencustomizing",
 | 
						|
          aWidget.keepBroadcastAttributesWhenCustomizing
 | 
						|
        );
 | 
						|
      }
 | 
						|
 | 
						|
      let shortcut;
 | 
						|
      if (aWidget.shortcutId) {
 | 
						|
        let keyEl = aDocument.getElementById(aWidget.shortcutId);
 | 
						|
        if (keyEl) {
 | 
						|
          shortcut = lazy.ShortcutUtils.prettifyShortcut(keyEl);
 | 
						|
        } else {
 | 
						|
          lazy.log.error(
 | 
						|
            "Key element with id '" +
 | 
						|
              aWidget.shortcutId +
 | 
						|
              "' for widget '" +
 | 
						|
              aWidget.id +
 | 
						|
              "' not found!"
 | 
						|
          );
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (aWidget.l10nId) {
 | 
						|
        aDocument.l10n.setAttributes(node, aWidget.l10nId);
 | 
						|
        if (button != node) {
 | 
						|
          // This is probably a "button-and-view" widget, such as the Profiler
 | 
						|
          // button. In that case, "node" is the "toolbaritem" container, and
 | 
						|
          // "button" the main button (see above).
 | 
						|
          // In this case, the values on the "node" is used in the Customize
 | 
						|
          // view, as well as the tooltips over both buttons; the values on the
 | 
						|
          // "button" are used in the overflow menu.
 | 
						|
          aDocument.l10n.setAttributes(button, aWidget.l10nId);
 | 
						|
        }
 | 
						|
 | 
						|
        if (shortcut) {
 | 
						|
          node.setAttribute("data-l10n-args", JSON.stringify({ shortcut }));
 | 
						|
          if (button != node) {
 | 
						|
            // This is probably a "button-and-view" widget.
 | 
						|
            button.setAttribute("data-l10n-args", JSON.stringify({ shortcut }));
 | 
						|
          }
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        node.setAttribute("label", this.getLocalizedProperty(aWidget, "label"));
 | 
						|
        if (button != node) {
 | 
						|
          // This is probably a "button-and-view" widget.
 | 
						|
          button.setAttribute("label", node.getAttribute("label"));
 | 
						|
        }
 | 
						|
 | 
						|
        let tooltip = this.getLocalizedProperty(
 | 
						|
          aWidget,
 | 
						|
          "tooltiptext",
 | 
						|
          shortcut ? [shortcut] : []
 | 
						|
        );
 | 
						|
        if (tooltip) {
 | 
						|
          node.setAttribute("tooltiptext", tooltip);
 | 
						|
          if (button != node) {
 | 
						|
            // This is probably a "button-and-view" widget.
 | 
						|
            button.setAttribute("tooltiptext", tooltip);
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      let commandHandler = this.handleWidgetCommand.bind(this, aWidget, node);
 | 
						|
      node.addEventListener("command", commandHandler);
 | 
						|
      let clickHandler = this.handleWidgetClick.bind(this, aWidget, node);
 | 
						|
      node.addEventListener("click", clickHandler);
 | 
						|
 | 
						|
      node.classList.add("chromeclass-toolbar-additional");
 | 
						|
 | 
						|
      // If the widget has a view, register a keypress handler because opening
 | 
						|
      // a view with the keyboard has slightly different focus handling than
 | 
						|
      // opening a view with the mouse. (When opened with the keyboard, the
 | 
						|
      // first item in the view should be focused after opening.)
 | 
						|
      if (viewbutton) {
 | 
						|
        lazy.log.debug(
 | 
						|
          "Widget " +
 | 
						|
            aWidget.id +
 | 
						|
            " has a view. Auto-registering event handlers."
 | 
						|
        );
 | 
						|
 | 
						|
        if (aWidget.source == CustomizableUI.SOURCE_BUILTIN) {
 | 
						|
          node.classList.add("subviewbutton-nav");
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      if (aWidget.onCreated) {
 | 
						|
        aWidget.onCreated(node);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    aWidget.instances.set(aDocument, node);
 | 
						|
    return node;
 | 
						|
  },
 | 
						|
 | 
						|
  ensureSubviewListeners(viewNode) {
 | 
						|
    if (viewNode._addedEventListeners) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    let viewId = viewNode.id;
 | 
						|
    let widget = [...gPalette.values()].find(w => w.viewId == viewId);
 | 
						|
    if (!widget) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    for (let eventName of kSubviewEvents) {
 | 
						|
      let handler = "on" + eventName;
 | 
						|
      if (typeof widget[handler] == "function") {
 | 
						|
        viewNode.addEventListener(eventName, widget[handler]);
 | 
						|
      }
 | 
						|
    }
 | 
						|
    viewNode._addedEventListeners = true;
 | 
						|
    lazy.log.debug(
 | 
						|
      "Widget " + widget.id + " showing and hiding event handlers set."
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  getLocalizedProperty(aWidget, aProp, aFormatArgs, aDef) {
 | 
						|
    const kReqStringProps = ["label"];
 | 
						|
 | 
						|
    if (typeof aWidget == "string") {
 | 
						|
      aWidget = gPalette.get(aWidget);
 | 
						|
    }
 | 
						|
    if (!aWidget) {
 | 
						|
      throw new Error(
 | 
						|
        "getLocalizedProperty was passed a non-widget to work with."
 | 
						|
      );
 | 
						|
    }
 | 
						|
    let def, name;
 | 
						|
    // Let widgets pass their own string identifiers or strings, so that
 | 
						|
    // we can use strings which aren't the default (in case string ids change)
 | 
						|
    // and so that non-builtin-widgets can also provide labels, tooltips, etc.
 | 
						|
    if (aWidget[aProp] != null) {
 | 
						|
      name = aWidget[aProp];
 | 
						|
      // By using this as the default, if a widget provides a full string rather
 | 
						|
      // than a string ID for localization, we will fall back to that string
 | 
						|
      // and return that.
 | 
						|
      def = aDef || name;
 | 
						|
    } else {
 | 
						|
      name = aWidget.id + "." + aProp;
 | 
						|
      def = aDef || "";
 | 
						|
    }
 | 
						|
    if (aWidget.localized === false) {
 | 
						|
      return def;
 | 
						|
    }
 | 
						|
    try {
 | 
						|
      if (Array.isArray(aFormatArgs) && aFormatArgs.length) {
 | 
						|
        return (
 | 
						|
          lazy.gWidgetsBundle.formatStringFromName(name, aFormatArgs) || def
 | 
						|
        );
 | 
						|
      }
 | 
						|
      return lazy.gWidgetsBundle.GetStringFromName(name) || def;
 | 
						|
    } catch (ex) {
 | 
						|
      // If an empty string was explicitly passed, treat it as an actual
 | 
						|
      // value rather than a missing property.
 | 
						|
      if (!def && (name != "" || kReqStringProps.includes(aProp))) {
 | 
						|
        lazy.log.error("Could not localize property '" + name + "'.");
 | 
						|
      }
 | 
						|
    }
 | 
						|
    return def;
 | 
						|
  },
 | 
						|
 | 
						|
  addShortcut(aShortcutNode, aTargetNode = aShortcutNode) {
 | 
						|
    // Detect if we've already been here before.
 | 
						|
    if (aTargetNode.hasAttribute("shortcut")) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Use ownerGlobal.document to ensure we get the right doc even for
 | 
						|
    // elements in template tags.
 | 
						|
    let { document } = aShortcutNode.ownerGlobal;
 | 
						|
    let shortcutId = aShortcutNode.getAttribute("key");
 | 
						|
    let shortcut;
 | 
						|
    if (shortcutId) {
 | 
						|
      shortcut = document.getElementById(shortcutId);
 | 
						|
    } else {
 | 
						|
      let commandId = aShortcutNode.getAttribute("command");
 | 
						|
      if (commandId) {
 | 
						|
        shortcut = lazy.ShortcutUtils.findShortcut(
 | 
						|
          document.getElementById(commandId)
 | 
						|
        );
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (!shortcut) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    aTargetNode.setAttribute(
 | 
						|
      "shortcut",
 | 
						|
      lazy.ShortcutUtils.prettifyShortcut(shortcut)
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  doWidgetCommand(aWidget, aNode, aEvent) {
 | 
						|
    if (aWidget.onCommand) {
 | 
						|
      try {
 | 
						|
        aWidget.onCommand.call(null, aEvent);
 | 
						|
      } catch (e) {
 | 
						|
        lazy.log.error(e);
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      // XXXunf Need to think this through more, and formalize.
 | 
						|
      Services.obs.notifyObservers(
 | 
						|
        aNode,
 | 
						|
        "customizedui-widget-command",
 | 
						|
        aWidget.id
 | 
						|
      );
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  showWidgetView(aWidget, aNode, aEvent) {
 | 
						|
    let ownerWindow = aNode.ownerGlobal;
 | 
						|
    let area = this.getPlacementOfWidget(aNode.id).area;
 | 
						|
    let areaType = CustomizableUI.getAreaType(area);
 | 
						|
    let anchor = aNode;
 | 
						|
 | 
						|
    if (
 | 
						|
      aWidget.disallowSubView &&
 | 
						|
      (areaType == CustomizableUI.TYPE_PANEL ||
 | 
						|
        aNode.hasAttribute("overflowedItem"))
 | 
						|
    ) {
 | 
						|
      // Close the containing panel (e.g. overflow), PanelUI will reopen.
 | 
						|
      let wrapper = this.wrapWidget(aWidget.id).forWindow(ownerWindow);
 | 
						|
      if (wrapper?.anchor) {
 | 
						|
        this.hidePanelForNode(aNode);
 | 
						|
        anchor = wrapper.anchor;
 | 
						|
      }
 | 
						|
    } else if (areaType != CustomizableUI.TYPE_PANEL) {
 | 
						|
      let wrapper = this.wrapWidget(aWidget.id).forWindow(ownerWindow);
 | 
						|
 | 
						|
      let hasMultiView = !!aNode.closest("panelmultiview");
 | 
						|
      if (!hasMultiView && wrapper?.anchor) {
 | 
						|
        this.hidePanelForNode(aNode);
 | 
						|
        anchor = wrapper.anchor;
 | 
						|
      }
 | 
						|
    }
 | 
						|
    ownerWindow.PanelUI.showSubView(aWidget.viewId, anchor, aEvent);
 | 
						|
  },
 | 
						|
 | 
						|
  handleWidgetCommand(aWidget, aNode, aEvent) {
 | 
						|
    // Note that aEvent can be a keypress event for widgets of type "view".
 | 
						|
    lazy.log.debug("handleWidgetCommand");
 | 
						|
 | 
						|
    let action;
 | 
						|
    if (aWidget.onBeforeCommand) {
 | 
						|
      try {
 | 
						|
        action = aWidget.onBeforeCommand.call(null, aEvent, aNode);
 | 
						|
      } catch (e) {
 | 
						|
        lazy.log.error(e);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (aWidget.type == "button" || action == "command") {
 | 
						|
      this.doWidgetCommand(aWidget, aNode, aEvent);
 | 
						|
    } else if (aWidget.type == "view" || action == "view") {
 | 
						|
      this.showWidgetView(aWidget, aNode, aEvent);
 | 
						|
    } else if (aWidget.type == "button-and-view") {
 | 
						|
      // Do the command if we're in the toolbar and the button was clicked.
 | 
						|
      // Otherwise, including when we have currently overflowed out of the
 | 
						|
      // toolbar, open the view. There is no way to trigger the command while
 | 
						|
      // the widget is in the panel, by design.
 | 
						|
      let button = aNode.firstElementChild;
 | 
						|
      let area = this.getPlacementOfWidget(aNode.id).area;
 | 
						|
      let areaType = CustomizableUI.getAreaType(area);
 | 
						|
      if (
 | 
						|
        areaType == CustomizableUI.TYPE_TOOLBAR &&
 | 
						|
        button.contains(aEvent.target) &&
 | 
						|
        !aNode.hasAttribute("overflowedItem")
 | 
						|
      ) {
 | 
						|
        this.doWidgetCommand(aWidget, aNode, aEvent);
 | 
						|
      } else {
 | 
						|
        this.showWidgetView(aWidget, aNode, aEvent);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  handleWidgetClick(aWidget, aNode, aEvent) {
 | 
						|
    lazy.log.debug("handleWidgetClick");
 | 
						|
    if (aWidget.onClick) {
 | 
						|
      try {
 | 
						|
        aWidget.onClick.call(null, aEvent);
 | 
						|
      } catch (e) {
 | 
						|
        console.error(e);
 | 
						|
      }
 | 
						|
    } else {
 | 
						|
      // XXXunf Need to think this through more, and formalize.
 | 
						|
      Services.obs.notifyObservers(
 | 
						|
        aNode,
 | 
						|
        "customizedui-widget-click",
 | 
						|
        aWidget.id
 | 
						|
      );
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _getPanelForNode(aNode) {
 | 
						|
    return aNode.closest("panel");
 | 
						|
  },
 | 
						|
 | 
						|
  /*
 | 
						|
   * If people put things in the panel which need more than single-click interaction,
 | 
						|
   * we don't want to close it. Right now we check for text inputs and menu buttons.
 | 
						|
   * We also check for being outside of any toolbaritem/toolbarbutton, ie on a blank
 | 
						|
   * part of the menu, or on another menu (like a context menu inside the panel).
 | 
						|
   */
 | 
						|
  _isOnInteractiveElement(aEvent) {
 | 
						|
    let panel = this._getPanelForNode(aEvent.currentTarget);
 | 
						|
    // This can happen in e.g. customize mode. If there's no panel,
 | 
						|
    // there's clearly nothing for us to close; pretend we're interactive.
 | 
						|
    if (!panel) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
 | 
						|
    function getNextTarget(target) {
 | 
						|
      if (target.nodeType == target.DOCUMENT_NODE) {
 | 
						|
        if (!target.defaultView) {
 | 
						|
          // Err, we're done.
 | 
						|
          return null;
 | 
						|
        }
 | 
						|
        // Find containing browser or iframe element in the parent doc.
 | 
						|
        return target.defaultView.docShell.chromeEventHandler;
 | 
						|
      }
 | 
						|
      // Skip any parent shadow roots
 | 
						|
      return target.parentNode?.host?.parentNode || target.parentNode;
 | 
						|
    }
 | 
						|
 | 
						|
    // While keeping track of that, we go from the original target back up,
 | 
						|
    // to the panel if we have to. We bail as soon as we find an input,
 | 
						|
    // a toolbarbutton/item, or a menuItem.
 | 
						|
    for (
 | 
						|
      let target = aEvent.originalTarget;
 | 
						|
      target && target != panel;
 | 
						|
      target = getNextTarget(target)
 | 
						|
    ) {
 | 
						|
      if (target.nodeType == target.DOCUMENT_NODE) {
 | 
						|
        // Skip out of iframes etc:
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
 | 
						|
      // Break out of the loop immediately for disabled items, as we need to
 | 
						|
      // keep the menu open in that case.
 | 
						|
      if (target.getAttribute("disabled") == "true") {
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
 | 
						|
      let tagName = target.localName;
 | 
						|
      if (tagName == "input" || tagName == "searchbar") {
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
      if (tagName == "toolbaritem" || tagName == "toolbarbutton") {
 | 
						|
        // If we are in a type="menu" toolbarbutton, we'll now interact with
 | 
						|
        // the menu.
 | 
						|
        return target.getAttribute("type") == "menu";
 | 
						|
      }
 | 
						|
      if (tagName == "menuitem") {
 | 
						|
        // If we're in a nested menu we don't need to close this panel.
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // We don't know what we interacted with, assume interactive.
 | 
						|
    return true;
 | 
						|
  },
 | 
						|
 | 
						|
  hidePanelForNode(aNode) {
 | 
						|
    let panel = this._getPanelForNode(aNode);
 | 
						|
    if (panel) {
 | 
						|
      lazy.PanelMultiView.hidePopup(panel);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  maybeAutoHidePanel(aEvent) {
 | 
						|
    let eventType = aEvent.type;
 | 
						|
    if (eventType == "keypress" && aEvent.keyCode != aEvent.DOM_VK_RETURN) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (eventType == "click" && aEvent.button != 0) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // We don't check preventDefault - it makes sense that this was prevented,
 | 
						|
    // but we probably still want to close the panel. If consumers don't want
 | 
						|
    // this to happen, they should specify the closemenu attribute.
 | 
						|
    if (eventType != "command" && this._isOnInteractiveElement(aEvent)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // We can't use event.target because we might have passed an anonymous
 | 
						|
    // content boundary as well, and so target points to the outer element in
 | 
						|
    // that case. Unfortunately, this means we get anonymous child nodes instead
 | 
						|
    // of the real ones, so looking for the 'stoooop, don't close me' attributes
 | 
						|
    // is more involved.
 | 
						|
    let target = aEvent.originalTarget;
 | 
						|
    while (target.parentNode && target.localName != "panel") {
 | 
						|
      if (
 | 
						|
        target.getAttribute("closemenu") == "none" ||
 | 
						|
        target.getAttribute("widget-type") == "view" ||
 | 
						|
        target.getAttribute("widget-type") == "button-and-view" ||
 | 
						|
        target.hasAttribute("view-button-id")
 | 
						|
      ) {
 | 
						|
        return;
 | 
						|
      }
 | 
						|
      target = target.parentNode;
 | 
						|
    }
 | 
						|
 | 
						|
    // If we get here, we can actually hide the popup:
 | 
						|
    this.hidePanelForNode(aEvent.target);
 | 
						|
  },
 | 
						|
 | 
						|
  getUnusedWidgets(aWindowPalette) {
 | 
						|
    let window = aWindowPalette.ownerGlobal;
 | 
						|
    let isWindowPrivate = lazy.PrivateBrowsingUtils.isWindowPrivate(window);
 | 
						|
    // We use a Set because there can be overlap between the widgets in
 | 
						|
    // gPalette and the items in the palette, especially after the first
 | 
						|
    // customization, since programmatically generated widgets will remain
 | 
						|
    // in the toolbox palette.
 | 
						|
    let widgets = new Set();
 | 
						|
 | 
						|
    // It's possible that some widgets have been defined programmatically and
 | 
						|
    // have not been overlayed into the palette. We can find those inside
 | 
						|
    // gPalette.
 | 
						|
    for (let [id, widget] of gPalette) {
 | 
						|
      if (!widget.currentArea) {
 | 
						|
        if (widget.showInPrivateBrowsing || !isWindowPrivate) {
 | 
						|
          widgets.add(id);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("Iterating the actual nodes of the window palette");
 | 
						|
    for (let node of aWindowPalette.children) {
 | 
						|
      lazy.log.debug("In palette children: " + node.id);
 | 
						|
      if (node.id && !this.getPlacementOfWidget(node.id)) {
 | 
						|
        widgets.add(node.id);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return [...widgets];
 | 
						|
  },
 | 
						|
 | 
						|
  getPlacementOfWidget(aWidgetId, aOnlyRegistered, aDeadAreas) {
 | 
						|
    if (aOnlyRegistered && !this.widgetExists(aWidgetId)) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    for (let [area, placements] of gPlacements) {
 | 
						|
      if (!gAreas.has(area) && !aDeadAreas) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      let index = placements.indexOf(aWidgetId);
 | 
						|
      if (index != -1) {
 | 
						|
        return { area, position: index };
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return null;
 | 
						|
  },
 | 
						|
 | 
						|
  widgetExists(aWidgetId) {
 | 
						|
    if (gPalette.has(aWidgetId) || this.isSpecialWidget(aWidgetId)) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
 | 
						|
    // Destroyed API widgets are in gSeenWidgets, but not in gPalette:
 | 
						|
    // The Pocket button is a default API widget that acts like a custom widget.
 | 
						|
    // If it's not in gPalette, it doesn't exist.
 | 
						|
    if (gSeenWidgets.has(aWidgetId) || aWidgetId === "save-to-pocket-button") {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    // We're assuming XUL widgets always exist, as it's much harder to check,
 | 
						|
    // and checking would be much more error prone.
 | 
						|
    return true;
 | 
						|
  },
 | 
						|
 | 
						|
  addWidgetToArea(aWidgetId, aArea, aPosition, aInitialAdd) {
 | 
						|
    if (aArea == CustomizableUI.AREA_NO_AREA) {
 | 
						|
      throw new Error(
 | 
						|
        "AREA_NO_AREA is only used as an argument for " +
 | 
						|
          "canWidgetMoveToArea. Use removeWidgetFromArea instead."
 | 
						|
      );
 | 
						|
    }
 | 
						|
    if (!gAreas.has(aArea)) {
 | 
						|
      throw new Error("Unknown customization area: " + aArea);
 | 
						|
    }
 | 
						|
 | 
						|
    // Hack: don't want special widgets in the panel (need to check here as well
 | 
						|
    // as in canWidgetMoveToArea because the menu panel is lazy):
 | 
						|
    if (
 | 
						|
      gAreas.get(aArea).get("type") == CustomizableUI.TYPE_PANEL &&
 | 
						|
      this.isSpecialWidget(aWidgetId)
 | 
						|
    ) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // If this is a lazy area that hasn't been restored yet, we can't yet modify
 | 
						|
    // it - would would at least like to add to it. So we keep track of it in
 | 
						|
    // gFuturePlacements,  and use that to add it when restoring the area. We
 | 
						|
    // throw away aPosition though, as that can only be bogus if the area hasn't
 | 
						|
    // yet been restorted (caller can't possibly know where its putting the
 | 
						|
    // widget in relation to other widgets).
 | 
						|
    if (this.isAreaLazy(aArea)) {
 | 
						|
      gFuturePlacements.get(aArea).add(aWidgetId);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (this.isSpecialWidget(aWidgetId)) {
 | 
						|
      aWidgetId = this.ensureSpecialWidgetId(aWidgetId);
 | 
						|
    }
 | 
						|
 | 
						|
    let oldPlacement = this.getPlacementOfWidget(aWidgetId, false, true);
 | 
						|
    if (oldPlacement && oldPlacement.area == aArea) {
 | 
						|
      this.moveWidgetWithinArea(aWidgetId, aPosition);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Do nothing if the widget is not allowed to move to the target area.
 | 
						|
    if (!this.canWidgetMoveToArea(aWidgetId, aArea)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (oldPlacement) {
 | 
						|
      this.removeWidgetFromArea(aWidgetId);
 | 
						|
    }
 | 
						|
 | 
						|
    if (!gPlacements.has(aArea)) {
 | 
						|
      gPlacements.set(aArea, [aWidgetId]);
 | 
						|
      aPosition = 0;
 | 
						|
    } else {
 | 
						|
      let placements = gPlacements.get(aArea);
 | 
						|
      if (typeof aPosition != "number") {
 | 
						|
        aPosition = placements.length;
 | 
						|
      }
 | 
						|
      if (aPosition < 0) {
 | 
						|
        aPosition = 0;
 | 
						|
      }
 | 
						|
      placements.splice(aPosition, 0, aWidgetId);
 | 
						|
    }
 | 
						|
 | 
						|
    let widget = gPalette.get(aWidgetId);
 | 
						|
    if (widget) {
 | 
						|
      widget.currentArea = aArea;
 | 
						|
      widget.currentPosition = aPosition;
 | 
						|
    }
 | 
						|
 | 
						|
    // We initially set placements with addWidgetToArea, so in that case
 | 
						|
    // we don't consider the area "dirtied".
 | 
						|
    if (!aInitialAdd) {
 | 
						|
      gDirtyAreaCache.add(aArea);
 | 
						|
    }
 | 
						|
 | 
						|
    gDirty = true;
 | 
						|
    this.saveState();
 | 
						|
 | 
						|
    this.notifyListeners("onWidgetAdded", aWidgetId, aArea, aPosition);
 | 
						|
  },
 | 
						|
 | 
						|
  removeWidgetFromArea(aWidgetId) {
 | 
						|
    let oldPlacement = this.getPlacementOfWidget(aWidgetId, false, true);
 | 
						|
    if (!oldPlacement) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (!this.isWidgetRemovable(aWidgetId)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let placements = gPlacements.get(oldPlacement.area);
 | 
						|
    let position = placements.indexOf(aWidgetId);
 | 
						|
    if (position != -1) {
 | 
						|
      placements.splice(position, 1);
 | 
						|
    }
 | 
						|
 | 
						|
    let widget = gPalette.get(aWidgetId);
 | 
						|
    if (widget) {
 | 
						|
      widget.currentArea = null;
 | 
						|
      widget.currentPosition = null;
 | 
						|
    }
 | 
						|
 | 
						|
    gDirty = true;
 | 
						|
    this.saveState();
 | 
						|
    gDirtyAreaCache.add(oldPlacement.area);
 | 
						|
 | 
						|
    this.notifyListeners("onWidgetRemoved", aWidgetId, oldPlacement.area);
 | 
						|
  },
 | 
						|
 | 
						|
  moveWidgetWithinArea(aWidgetId, aPosition) {
 | 
						|
    let oldPlacement = this.getPlacementOfWidget(aWidgetId);
 | 
						|
    if (!oldPlacement) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let placements = gPlacements.get(oldPlacement.area);
 | 
						|
    if (typeof aPosition != "number") {
 | 
						|
      aPosition = placements.length;
 | 
						|
    } else if (aPosition < 0) {
 | 
						|
      aPosition = 0;
 | 
						|
    } else if (aPosition > placements.length) {
 | 
						|
      aPosition = placements.length;
 | 
						|
    }
 | 
						|
 | 
						|
    let widget = gPalette.get(aWidgetId);
 | 
						|
    if (widget) {
 | 
						|
      widget.currentPosition = aPosition;
 | 
						|
      widget.currentArea = oldPlacement.area;
 | 
						|
    }
 | 
						|
 | 
						|
    if (aPosition == oldPlacement.position) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    placements.splice(oldPlacement.position, 1);
 | 
						|
    // If we just removed the item from *before* where it is now added,
 | 
						|
    // we need to compensate the position offset for that:
 | 
						|
    if (oldPlacement.position < aPosition) {
 | 
						|
      aPosition--;
 | 
						|
    }
 | 
						|
    placements.splice(aPosition, 0, aWidgetId);
 | 
						|
 | 
						|
    gDirty = true;
 | 
						|
    gDirtyAreaCache.add(oldPlacement.area);
 | 
						|
 | 
						|
    this.saveState();
 | 
						|
 | 
						|
    this.notifyListeners(
 | 
						|
      "onWidgetMoved",
 | 
						|
      aWidgetId,
 | 
						|
      oldPlacement.area,
 | 
						|
      oldPlacement.position,
 | 
						|
      aPosition
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  // Note that this does not populate gPlacements, which is done lazily.
 | 
						|
  // The panel area is an exception here.
 | 
						|
  loadSavedState() {
 | 
						|
    let state = Services.prefs.getCharPref(kPrefCustomizationState, "");
 | 
						|
    if (!state) {
 | 
						|
      lazy.log.debug("No saved state found");
 | 
						|
      // Nothing has been customized, so silently fall back to the defaults.
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    try {
 | 
						|
      gSavedState = JSON.parse(state);
 | 
						|
      if (typeof gSavedState != "object" || gSavedState === null) {
 | 
						|
        throw new Error("Invalid saved state");
 | 
						|
      }
 | 
						|
    } catch (e) {
 | 
						|
      Services.prefs.clearUserPref(kPrefCustomizationState);
 | 
						|
      gSavedState = {};
 | 
						|
      lazy.log.debug(
 | 
						|
        "Error loading saved UI customization state, falling back to defaults."
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    if (!("placements" in gSavedState)) {
 | 
						|
      gSavedState.placements = {};
 | 
						|
    }
 | 
						|
 | 
						|
    if (!("currentVersion" in gSavedState)) {
 | 
						|
      gSavedState.currentVersion = 0;
 | 
						|
    }
 | 
						|
 | 
						|
    gSeenWidgets = new Set(gSavedState.seen || []);
 | 
						|
    gDirtyAreaCache = new Set(gSavedState.dirtyAreaCache || []);
 | 
						|
    gNewElementCount = gSavedState.newElementCount || 0;
 | 
						|
  },
 | 
						|
 | 
						|
  restoreStateForArea(aArea) {
 | 
						|
    let placementsPreexisted = gPlacements.has(aArea);
 | 
						|
 | 
						|
    this.beginBatchUpdate();
 | 
						|
    try {
 | 
						|
      gRestoring = true;
 | 
						|
 | 
						|
      let restored = false;
 | 
						|
      if (placementsPreexisted) {
 | 
						|
        lazy.log.debug("Restoring " + aArea + " from pre-existing placements");
 | 
						|
        for (let [position, id] of gPlacements.get(aArea).entries()) {
 | 
						|
          this.moveWidgetWithinArea(id, position);
 | 
						|
        }
 | 
						|
        gDirty = false;
 | 
						|
        restored = true;
 | 
						|
      } else {
 | 
						|
        gPlacements.set(aArea, []);
 | 
						|
      }
 | 
						|
 | 
						|
      if (!restored && gSavedState && aArea in gSavedState.placements) {
 | 
						|
        lazy.log.debug("Restoring " + aArea + " from saved state");
 | 
						|
        let placements = gSavedState.placements[aArea];
 | 
						|
        for (let id of placements) {
 | 
						|
          this.addWidgetToArea(id, aArea);
 | 
						|
        }
 | 
						|
        gDirty = false;
 | 
						|
        restored = true;
 | 
						|
      }
 | 
						|
 | 
						|
      if (!restored) {
 | 
						|
        lazy.log.debug("Restoring " + aArea + " from default state");
 | 
						|
        let defaults = gAreas.get(aArea).get("defaultPlacements");
 | 
						|
        if (defaults) {
 | 
						|
          for (let id of defaults) {
 | 
						|
            this.addWidgetToArea(id, aArea, null, true);
 | 
						|
          }
 | 
						|
        }
 | 
						|
        gDirty = false;
 | 
						|
      }
 | 
						|
 | 
						|
      // Finally, add widgets to the area that were added before the it was able
 | 
						|
      // to be restored. This can occur when add-ons register widgets for a
 | 
						|
      // lazily-restored area before it's been restored.
 | 
						|
      if (gFuturePlacements.has(aArea)) {
 | 
						|
        let areaPlacements = gPlacements.get(aArea);
 | 
						|
        for (let id of gFuturePlacements.get(aArea)) {
 | 
						|
          if (areaPlacements.includes(id)) {
 | 
						|
            continue;
 | 
						|
          }
 | 
						|
          this.addWidgetToArea(id, aArea);
 | 
						|
        }
 | 
						|
        gFuturePlacements.delete(aArea);
 | 
						|
      }
 | 
						|
 | 
						|
      lazy.log.debug(
 | 
						|
        "Placements for " +
 | 
						|
          aArea +
 | 
						|
          ":\n\t" +
 | 
						|
          gPlacements.get(aArea).join("\n\t")
 | 
						|
      );
 | 
						|
 | 
						|
      gRestoring = false;
 | 
						|
    } finally {
 | 
						|
      this.endBatchUpdate();
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  saveState() {
 | 
						|
    if (gInBatchStack || !gDirty) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    // Clone because we want to modify this map:
 | 
						|
    let state = {
 | 
						|
      placements: new Map(gPlacements),
 | 
						|
      seen: gSeenWidgets,
 | 
						|
      dirtyAreaCache: gDirtyAreaCache,
 | 
						|
      currentVersion: kVersion,
 | 
						|
      newElementCount: gNewElementCount,
 | 
						|
    };
 | 
						|
 | 
						|
    // Merge in previously saved areas if not present in gPlacements.
 | 
						|
    // This way, state is still persisted for e.g. temporarily disabled
 | 
						|
    // add-ons - see bug 989338.
 | 
						|
    if (gSavedState && gSavedState.placements) {
 | 
						|
      for (let area of Object.keys(gSavedState.placements)) {
 | 
						|
        if (!state.placements.has(area)) {
 | 
						|
          let placements = gSavedState.placements[area];
 | 
						|
          state.placements.set(area, placements);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("Saving state.");
 | 
						|
    let serialized = JSON.stringify(state, this.serializerHelper);
 | 
						|
    lazy.log.debug("State saved as: " + serialized);
 | 
						|
    Services.prefs.setCharPref(kPrefCustomizationState, serialized);
 | 
						|
    gDirty = false;
 | 
						|
  },
 | 
						|
 | 
						|
  serializerHelper(aKey, aValue) {
 | 
						|
    if (typeof aValue == "object" && aValue.constructor.name == "Map") {
 | 
						|
      let result = {};
 | 
						|
      for (let [mapKey, mapValue] of aValue) {
 | 
						|
        result[mapKey] = mapValue;
 | 
						|
      }
 | 
						|
      return result;
 | 
						|
    }
 | 
						|
 | 
						|
    if (typeof aValue == "object" && aValue.constructor.name == "Set") {
 | 
						|
      return [...aValue];
 | 
						|
    }
 | 
						|
 | 
						|
    return aValue;
 | 
						|
  },
 | 
						|
 | 
						|
  beginBatchUpdate() {
 | 
						|
    gInBatchStack++;
 | 
						|
  },
 | 
						|
 | 
						|
  endBatchUpdate(aForceDirty) {
 | 
						|
    gInBatchStack--;
 | 
						|
    if (aForceDirty === true) {
 | 
						|
      gDirty = true;
 | 
						|
    }
 | 
						|
    if (gInBatchStack == 0) {
 | 
						|
      this.saveState();
 | 
						|
    } else if (gInBatchStack < 0) {
 | 
						|
      throw new Error(
 | 
						|
        "The batch editing stack should never reach a negative number."
 | 
						|
      );
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  addListener(aListener) {
 | 
						|
    gListeners.add(aListener);
 | 
						|
  },
 | 
						|
 | 
						|
  removeListener(aListener) {
 | 
						|
    if (aListener == this) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    gListeners.delete(aListener);
 | 
						|
  },
 | 
						|
 | 
						|
  notifyListeners(aEvent, ...aArgs) {
 | 
						|
    if (gRestoring) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    for (let listener of gListeners) {
 | 
						|
      try {
 | 
						|
        if (typeof listener[aEvent] == "function") {
 | 
						|
          listener[aEvent].apply(listener, aArgs);
 | 
						|
        }
 | 
						|
      } catch (e) {
 | 
						|
        lazy.log.error(e + " -- " + e.fileName + ":" + e.lineNumber);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _dispatchToolboxEventToWindow(aEventType, aDetails, aWindow) {
 | 
						|
    let evt = new aWindow.CustomEvent(aEventType, {
 | 
						|
      bubbles: true,
 | 
						|
      cancelable: true,
 | 
						|
      detail: aDetails,
 | 
						|
    });
 | 
						|
    aWindow.gNavToolbox.dispatchEvent(evt);
 | 
						|
  },
 | 
						|
 | 
						|
  dispatchToolboxEvent(aEventType, aDetails = {}, aWindow = null) {
 | 
						|
    if (aWindow) {
 | 
						|
      this._dispatchToolboxEventToWindow(aEventType, aDetails, aWindow);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    for (let [win] of gBuildWindows) {
 | 
						|
      this._dispatchToolboxEventToWindow(aEventType, aDetails, win);
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  createWidget(aProperties) {
 | 
						|
    let widget = this.normalizeWidget(
 | 
						|
      aProperties,
 | 
						|
      CustomizableUI.SOURCE_EXTERNAL
 | 
						|
    );
 | 
						|
    // XXXunf This should probably throw.
 | 
						|
    if (!widget) {
 | 
						|
      lazy.log.error("unable to normalize widget");
 | 
						|
      return undefined;
 | 
						|
    }
 | 
						|
 | 
						|
    gPalette.set(widget.id, widget);
 | 
						|
 | 
						|
    // Clear our caches:
 | 
						|
    gGroupWrapperCache.delete(widget.id);
 | 
						|
    for (let [win] of gBuildWindows) {
 | 
						|
      let cache = gSingleWrapperCache.get(win);
 | 
						|
      if (cache) {
 | 
						|
        cache.delete(widget.id);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    this.notifyListeners("onWidgetCreated", widget.id);
 | 
						|
 | 
						|
    if (widget.defaultArea) {
 | 
						|
      let addToDefaultPlacements = false;
 | 
						|
      let area = gAreas.get(widget.defaultArea);
 | 
						|
      if (
 | 
						|
        !CustomizableUI.isBuiltinToolbar(widget.defaultArea) &&
 | 
						|
        widget.defaultArea != CustomizableUI.AREA_FIXED_OVERFLOW_PANEL
 | 
						|
      ) {
 | 
						|
        addToDefaultPlacements = true;
 | 
						|
      }
 | 
						|
 | 
						|
      if (addToDefaultPlacements) {
 | 
						|
        if (area.has("defaultPlacements")) {
 | 
						|
          area.get("defaultPlacements").push(widget.id);
 | 
						|
        } else {
 | 
						|
          area.set("defaultPlacements", [widget.id]);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Look through previously saved state to see if we're restoring a widget.
 | 
						|
    let seenAreas = new Set();
 | 
						|
    let widgetMightNeedAutoAdding = true;
 | 
						|
    for (let [area] of gPlacements) {
 | 
						|
      seenAreas.add(area);
 | 
						|
      let areaIsRegistered = gAreas.has(area);
 | 
						|
      let index = gPlacements.get(area).indexOf(widget.id);
 | 
						|
      if (index != -1) {
 | 
						|
        widgetMightNeedAutoAdding = false;
 | 
						|
        if (areaIsRegistered) {
 | 
						|
          widget.currentArea = area;
 | 
						|
          widget.currentPosition = index;
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // Also look at saved state data directly in areas that haven't yet been
 | 
						|
    // restored. Can't rely on this for restored areas, as they may have
 | 
						|
    // changed.
 | 
						|
    if (widgetMightNeedAutoAdding && gSavedState) {
 | 
						|
      for (let area of Object.keys(gSavedState.placements)) {
 | 
						|
        if (seenAreas.has(area)) {
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
 | 
						|
        let areaIsRegistered = gAreas.has(area);
 | 
						|
        let index = gSavedState.placements[area].indexOf(widget.id);
 | 
						|
        if (index != -1) {
 | 
						|
          widgetMightNeedAutoAdding = false;
 | 
						|
          if (areaIsRegistered) {
 | 
						|
            widget.currentArea = area;
 | 
						|
            widget.currentPosition = index;
 | 
						|
          }
 | 
						|
          break;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // If we're restoring the widget to it's old placement, fire off the
 | 
						|
    // onWidgetAdded event - our own handler will take care of adding it to
 | 
						|
    // any build areas.
 | 
						|
    this.beginBatchUpdate();
 | 
						|
    try {
 | 
						|
      if (widget.currentArea) {
 | 
						|
        this.notifyListeners(
 | 
						|
          "onWidgetAdded",
 | 
						|
          widget.id,
 | 
						|
          widget.currentArea,
 | 
						|
          widget.currentPosition
 | 
						|
        );
 | 
						|
      } else if (widgetMightNeedAutoAdding) {
 | 
						|
        let autoAdd = Services.prefs.getBoolPref(
 | 
						|
          kPrefCustomizationAutoAdd,
 | 
						|
          true
 | 
						|
        );
 | 
						|
 | 
						|
        // If the widget doesn't have an existing placement, and it hasn't been
 | 
						|
        // seen before, then add it to its default area so it can be used.
 | 
						|
        // If the widget is not removable, we *have* to add it to its default
 | 
						|
        // area here.
 | 
						|
        let canBeAutoAdded = autoAdd && !gSeenWidgets.has(widget.id);
 | 
						|
        if (!widget.currentArea && (!widget.removable || canBeAutoAdded)) {
 | 
						|
          if (widget.defaultArea) {
 | 
						|
            if (this.isAreaLazy(widget.defaultArea)) {
 | 
						|
              gFuturePlacements.get(widget.defaultArea).add(widget.id);
 | 
						|
            } else {
 | 
						|
              this.addWidgetToArea(widget.id, widget.defaultArea);
 | 
						|
            }
 | 
						|
          }
 | 
						|
        }
 | 
						|
 | 
						|
        // Extension widgets cannot enter the customization palette, so if
 | 
						|
        // at this point, we haven't found an area for them, move them into
 | 
						|
        // AREA_ADDONS.
 | 
						|
        if (
 | 
						|
          !widget.currentArea &&
 | 
						|
          CustomizableUI.isWebExtensionWidget(widget.id)
 | 
						|
        ) {
 | 
						|
          this.addWidgetToArea(widget.id, CustomizableUI.AREA_ADDONS);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    } finally {
 | 
						|
      // Ensure we always have this widget in gSeenWidgets, and save
 | 
						|
      // state in case this needs to be done here.
 | 
						|
      gSeenWidgets.add(widget.id);
 | 
						|
      this.endBatchUpdate(true);
 | 
						|
    }
 | 
						|
 | 
						|
    this.notifyListeners(
 | 
						|
      "onWidgetAfterCreation",
 | 
						|
      widget.id,
 | 
						|
      widget.currentArea
 | 
						|
    );
 | 
						|
    return widget.id;
 | 
						|
  },
 | 
						|
 | 
						|
  createBuiltinWidget(aData) {
 | 
						|
    // This should only ever be called on startup, before any windows are
 | 
						|
    // opened - so we know there's no build areas to handle. Also, builtin
 | 
						|
    // widgets are expected to be (mostly) static, so shouldn't affect the
 | 
						|
    // current placement settings.
 | 
						|
 | 
						|
    // This allows a widget to be both built-in by default but also able to be
 | 
						|
    // destroyed and removed from the area based on criteria that may not be
 | 
						|
    // available when the widget is created -- for example, because some other
 | 
						|
    // feature in the browser supersedes the widget.
 | 
						|
    let conditionalDestroyPromise = aData.conditionalDestroyPromise || null;
 | 
						|
    delete aData.conditionalDestroyPromise;
 | 
						|
 | 
						|
    let widget = this.normalizeWidget(aData, CustomizableUI.SOURCE_BUILTIN);
 | 
						|
    if (!widget) {
 | 
						|
      lazy.log.error("Error creating builtin widget: " + aData.id);
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    lazy.log.debug("Creating built-in widget with id: " + widget.id);
 | 
						|
    gPalette.set(widget.id, widget);
 | 
						|
 | 
						|
    if (conditionalDestroyPromise) {
 | 
						|
      conditionalDestroyPromise.then(
 | 
						|
        shouldDestroy => {
 | 
						|
          if (shouldDestroy) {
 | 
						|
            this.destroyWidget(widget.id);
 | 
						|
            this.removeWidgetFromArea(widget.id);
 | 
						|
          }
 | 
						|
        },
 | 
						|
        err => {
 | 
						|
          console.error(err);
 | 
						|
        }
 | 
						|
      );
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  // Returns true if the area will eventually lazily restore (but hasn't yet).
 | 
						|
  isAreaLazy(aArea) {
 | 
						|
    if (gPlacements.has(aArea)) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    return gAreas.get(aArea).get("type") == CustomizableUI.TYPE_TOOLBAR;
 | 
						|
  },
 | 
						|
 | 
						|
  // XXXunf Log some warnings here, when the data provided isn't up to scratch.
 | 
						|
  normalizeWidget(aData, aSource) {
 | 
						|
    let widget = {
 | 
						|
      implementation: aData,
 | 
						|
      source: aSource || CustomizableUI.SOURCE_EXTERNAL,
 | 
						|
      instances: new Map(),
 | 
						|
      currentArea: null,
 | 
						|
      localized: true,
 | 
						|
      removable: true,
 | 
						|
      overflows: true,
 | 
						|
      defaultArea: null,
 | 
						|
      shortcutId: null,
 | 
						|
      tabSpecific: false,
 | 
						|
      locationSpecific: false,
 | 
						|
      tooltiptext: null,
 | 
						|
      l10nId: null,
 | 
						|
      showInPrivateBrowsing: true,
 | 
						|
      _introducedInVersion: -1,
 | 
						|
      keepBroadcastAttributesWhenCustomizing: false,
 | 
						|
      disallowSubView: false,
 | 
						|
      webExtension: false,
 | 
						|
    };
 | 
						|
 | 
						|
    if (typeof aData.id != "string" || !/^[a-z0-9-_]{1,}$/i.test(aData.id)) {
 | 
						|
      lazy.log.error("Given an illegal id in normalizeWidget: " + aData.id);
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    delete widget.implementation.currentArea;
 | 
						|
    widget.implementation.__defineGetter__(
 | 
						|
      "currentArea",
 | 
						|
      () => widget.currentArea
 | 
						|
    );
 | 
						|
 | 
						|
    const kReqStringProps = ["id"];
 | 
						|
    for (let prop of kReqStringProps) {
 | 
						|
      if (typeof aData[prop] != "string") {
 | 
						|
        lazy.log.error(
 | 
						|
          "Missing required property '" +
 | 
						|
            prop +
 | 
						|
            "' in normalizeWidget: " +
 | 
						|
            aData.id
 | 
						|
        );
 | 
						|
        return null;
 | 
						|
      }
 | 
						|
      widget[prop] = aData[prop];
 | 
						|
    }
 | 
						|
 | 
						|
    const kOptStringProps = ["l10nId", "label", "tooltiptext", "shortcutId"];
 | 
						|
    for (let prop of kOptStringProps) {
 | 
						|
      if (typeof aData[prop] == "string") {
 | 
						|
        widget[prop] = aData[prop];
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    const kOptBoolProps = [
 | 
						|
      "removable",
 | 
						|
      "showInPrivateBrowsing",
 | 
						|
      "overflows",
 | 
						|
      "tabSpecific",
 | 
						|
      "locationSpecific",
 | 
						|
      "localized",
 | 
						|
      "keepBroadcastAttributesWhenCustomizing",
 | 
						|
      "disallowSubView",
 | 
						|
      "webExtension",
 | 
						|
    ];
 | 
						|
    for (let prop of kOptBoolProps) {
 | 
						|
      if (typeof aData[prop] == "boolean") {
 | 
						|
        widget[prop] = aData[prop];
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // When we normalize builtin widgets, areas have not yet been registered:
 | 
						|
    if (
 | 
						|
      aData.defaultArea &&
 | 
						|
      (aSource == CustomizableUI.SOURCE_BUILTIN ||
 | 
						|
        gAreas.has(aData.defaultArea))
 | 
						|
    ) {
 | 
						|
      widget.defaultArea = aData.defaultArea;
 | 
						|
    } else if (!widget.removable) {
 | 
						|
      lazy.log.error(
 | 
						|
        "Widget '" +
 | 
						|
          widget.id +
 | 
						|
          "' is not removable but does not specify " +
 | 
						|
          "a valid defaultArea. That's not possible; it must specify a " +
 | 
						|
          "valid defaultArea as well."
 | 
						|
      );
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    if ("type" in aData && gSupportedWidgetTypes.has(aData.type)) {
 | 
						|
      widget.type = aData.type;
 | 
						|
    } else {
 | 
						|
      widget.type = "button";
 | 
						|
    }
 | 
						|
 | 
						|
    widget.disabled = aData.disabled === true;
 | 
						|
 | 
						|
    if (aSource == CustomizableUI.SOURCE_BUILTIN) {
 | 
						|
      widget._introducedInVersion = aData.introducedInVersion || 0;
 | 
						|
    }
 | 
						|
 | 
						|
    this.wrapWidgetEventHandler("onBeforeCreated", widget);
 | 
						|
    this.wrapWidgetEventHandler("onClick", widget);
 | 
						|
    this.wrapWidgetEventHandler("onCreated", widget);
 | 
						|
    this.wrapWidgetEventHandler("onDestroyed", widget);
 | 
						|
 | 
						|
    if (typeof aData.onBeforeCommand == "function") {
 | 
						|
      widget.onBeforeCommand = aData.onBeforeCommand;
 | 
						|
    }
 | 
						|
 | 
						|
    if (typeof aData.onCommand == "function") {
 | 
						|
      widget.onCommand = aData.onCommand;
 | 
						|
    }
 | 
						|
    if (
 | 
						|
      widget.type == "view" ||
 | 
						|
      widget.type == "button-and-view" ||
 | 
						|
      aData.viewId
 | 
						|
    ) {
 | 
						|
      if (typeof aData.viewId != "string") {
 | 
						|
        lazy.log.error(
 | 
						|
          "Expected a string for widget " +
 | 
						|
            widget.id +
 | 
						|
            " viewId, but got " +
 | 
						|
            aData.viewId
 | 
						|
        );
 | 
						|
        return null;
 | 
						|
      }
 | 
						|
      widget.viewId = aData.viewId;
 | 
						|
 | 
						|
      this.wrapWidgetEventHandler("onViewShowing", widget);
 | 
						|
      this.wrapWidgetEventHandler("onViewHiding", widget);
 | 
						|
    }
 | 
						|
    if (widget.type == "custom") {
 | 
						|
      this.wrapWidgetEventHandler("onBuild", widget);
 | 
						|
    }
 | 
						|
 | 
						|
    if (gPalette.has(widget.id)) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    return widget;
 | 
						|
  },
 | 
						|
 | 
						|
  wrapWidgetEventHandler(aEventName, aWidget) {
 | 
						|
    if (typeof aWidget.implementation[aEventName] != "function") {
 | 
						|
      aWidget[aEventName] = null;
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    aWidget[aEventName] = function (...aArgs) {
 | 
						|
      try {
 | 
						|
        // Don't copy the function to the normalized widget object, instead
 | 
						|
        // keep it on the original object provided to the API so that
 | 
						|
        // additional methods can be implemented and used by the event
 | 
						|
        // handlers.
 | 
						|
        return aWidget.implementation[aEventName].apply(
 | 
						|
          aWidget.implementation,
 | 
						|
          aArgs
 | 
						|
        );
 | 
						|
      } catch (e) {
 | 
						|
        console.error(e);
 | 
						|
        return undefined;
 | 
						|
      }
 | 
						|
    };
 | 
						|
  },
 | 
						|
 | 
						|
  destroyWidget(aWidgetId) {
 | 
						|
    let widget = gPalette.get(aWidgetId);
 | 
						|
    if (!widget) {
 | 
						|
      gGroupWrapperCache.delete(aWidgetId);
 | 
						|
      for (let [window] of gBuildWindows) {
 | 
						|
        let windowCache = gSingleWrapperCache.get(window);
 | 
						|
        if (windowCache) {
 | 
						|
          windowCache.delete(aWidgetId);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    // Remove it from the default placements of an area if it was added there:
 | 
						|
    if (widget.defaultArea) {
 | 
						|
      let area = gAreas.get(widget.defaultArea);
 | 
						|
      if (area) {
 | 
						|
        let defaultPlacements = area.get("defaultPlacements");
 | 
						|
        // We can assume this is present because if a widget has a defaultArea,
 | 
						|
        // we automatically create a defaultPlacements array for that area.
 | 
						|
        let widgetIndex = defaultPlacements.indexOf(aWidgetId);
 | 
						|
        if (widgetIndex != -1) {
 | 
						|
          defaultPlacements.splice(widgetIndex, 1);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // This will not remove the widget from gPlacements - we want to keep the
 | 
						|
    // setting so the widget gets put back in it's old position if/when it
 | 
						|
    // returns.
 | 
						|
    for (let [window] of gBuildWindows) {
 | 
						|
      let windowCache = gSingleWrapperCache.get(window);
 | 
						|
      if (windowCache) {
 | 
						|
        windowCache.delete(aWidgetId);
 | 
						|
      }
 | 
						|
      let widgetNode =
 | 
						|
        window.document.getElementById(aWidgetId) ||
 | 
						|
        window.gNavToolbox.palette.getElementsByAttribute("id", aWidgetId)[0];
 | 
						|
      if (widgetNode) {
 | 
						|
        let container = widgetNode.parentNode;
 | 
						|
        this.notifyListeners(
 | 
						|
          "onWidgetBeforeDOMChange",
 | 
						|
          widgetNode,
 | 
						|
          null,
 | 
						|
          container,
 | 
						|
          true
 | 
						|
        );
 | 
						|
        widgetNode.remove();
 | 
						|
        this.notifyListeners(
 | 
						|
          "onWidgetAfterDOMChange",
 | 
						|
          widgetNode,
 | 
						|
          null,
 | 
						|
          container,
 | 
						|
          true
 | 
						|
        );
 | 
						|
      }
 | 
						|
      if (
 | 
						|
        widget.type == "view" ||
 | 
						|
        widget.type == "button-and-view" ||
 | 
						|
        widget.viewId
 | 
						|
      ) {
 | 
						|
        let viewNode = window.document.getElementById(widget.viewId);
 | 
						|
        if (viewNode) {
 | 
						|
          for (let eventName of kSubviewEvents) {
 | 
						|
            let handler = "on" + eventName;
 | 
						|
            if (typeof widget[handler] == "function") {
 | 
						|
              viewNode.removeEventListener(eventName, widget[handler]);
 | 
						|
            }
 | 
						|
          }
 | 
						|
          viewNode._addedEventListeners = false;
 | 
						|
        }
 | 
						|
      }
 | 
						|
      if (widgetNode && widget.onDestroyed) {
 | 
						|
        widget.onDestroyed(window.document);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    gPalette.delete(aWidgetId);
 | 
						|
    gGroupWrapperCache.delete(aWidgetId);
 | 
						|
 | 
						|
    this.notifyListeners("onWidgetDestroyed", aWidgetId);
 | 
						|
  },
 | 
						|
 | 
						|
  getCustomizeTargetForArea(aArea, aWindow) {
 | 
						|
    let buildAreaNodes = gBuildAreas.get(aArea);
 | 
						|
    if (!buildAreaNodes) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    for (let node of buildAreaNodes) {
 | 
						|
      if (node.ownerGlobal == aWindow) {
 | 
						|
        return this.getCustomizationTarget(node) || node;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return null;
 | 
						|
  },
 | 
						|
 | 
						|
  reset() {
 | 
						|
    gResetting = true;
 | 
						|
    this._resetUIState();
 | 
						|
 | 
						|
    // Rebuild each registered area (across windows) to reflect the state that
 | 
						|
    // was reset above.
 | 
						|
    this._rebuildRegisteredAreas();
 | 
						|
 | 
						|
    for (let [widgetId, widget] of gPalette) {
 | 
						|
      if (widget.source == CustomizableUI.SOURCE_EXTERNAL) {
 | 
						|
        gSeenWidgets.add(widgetId);
 | 
						|
      }
 | 
						|
    }
 | 
						|
    if (gSeenWidgets.size || gNewElementCount) {
 | 
						|
      gDirty = true;
 | 
						|
      this.saveState();
 | 
						|
    }
 | 
						|
 | 
						|
    gResetting = false;
 | 
						|
  },
 | 
						|
 | 
						|
  _resetUIState() {
 | 
						|
    try {
 | 
						|
      gUIStateBeforeReset.drawInTitlebar =
 | 
						|
        Services.prefs.getIntPref(kPrefDrawInTitlebar);
 | 
						|
      gUIStateBeforeReset.uiCustomizationState = Services.prefs.getCharPref(
 | 
						|
        kPrefCustomizationState
 | 
						|
      );
 | 
						|
      gUIStateBeforeReset.uiDensity = Services.prefs.getIntPref(kPrefUIDensity);
 | 
						|
      gUIStateBeforeReset.autoTouchMode =
 | 
						|
        Services.prefs.getBoolPref(kPrefAutoTouchMode);
 | 
						|
      gUIStateBeforeReset.currentTheme = gSelectedTheme;
 | 
						|
      gUIStateBeforeReset.autoHideDownloadsButton = Services.prefs.getBoolPref(
 | 
						|
        kPrefAutoHideDownloadsButton
 | 
						|
      );
 | 
						|
      gUIStateBeforeReset.newElementCount = gNewElementCount;
 | 
						|
    } catch (e) {}
 | 
						|
 | 
						|
    Services.prefs.clearUserPref(kPrefCustomizationState);
 | 
						|
    Services.prefs.clearUserPref(kPrefDrawInTitlebar);
 | 
						|
    Services.prefs.clearUserPref(kPrefUIDensity);
 | 
						|
    Services.prefs.clearUserPref(kPrefAutoTouchMode);
 | 
						|
    Services.prefs.clearUserPref(kPrefAutoHideDownloadsButton);
 | 
						|
    gDefaultTheme.enable();
 | 
						|
    gNewElementCount = 0;
 | 
						|
    lazy.log.debug("State reset");
 | 
						|
 | 
						|
    // Later in the function, we're going to add any area-less extension
 | 
						|
    // buttons to the AREA_ADDONS area. We'll remember the old placements
 | 
						|
    // for that area so that we don't need to re-add widgets that are already
 | 
						|
    // in there in the DOM.
 | 
						|
    let oldAddonPlacements = gPlacements[CustomizableUI.AREA_ADDONS] || [];
 | 
						|
 | 
						|
    // Reset placements to make restoring default placements possible.
 | 
						|
    gPlacements = new Map();
 | 
						|
    gDirtyAreaCache = new Set();
 | 
						|
    gSeenWidgets = new Set();
 | 
						|
    // Clear the saved state to ensure that defaults will be used.
 | 
						|
    gSavedState = null;
 | 
						|
    // Restore the state for each area to its defaults
 | 
						|
    for (let [areaId] of gAreas) {
 | 
						|
      // If the Unified Extensions UI is enabled, we'll be adding any
 | 
						|
      // extension buttons that aren't already in AREA_ADDONS there,
 | 
						|
      // so we can skip restoring the state for it.
 | 
						|
      if (areaId != CustomizableUI.AREA_ADDONS) {
 | 
						|
        this.restoreStateForArea(areaId);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    // restoreStateForArea will have normally set an array for the placements
 | 
						|
    // for each area, but since we skip AREA_ADDONS intentionally, that array
 | 
						|
    // doesn't get set, so we do that manually here.
 | 
						|
    gPlacements.set(CustomizableUI.AREA_ADDONS, []);
 | 
						|
 | 
						|
    for (let [widgetId] of gPalette) {
 | 
						|
      if (
 | 
						|
        CustomizableUI.isWebExtensionWidget(widgetId) &&
 | 
						|
        !oldAddonPlacements.includes(widgetId)
 | 
						|
      ) {
 | 
						|
        this.addWidgetToArea(widgetId, CustomizableUI.AREA_ADDONS);
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  _rebuildRegisteredAreas() {
 | 
						|
    for (let [areaId, areaNodes] of gBuildAreas) {
 | 
						|
      let placements = gPlacements.get(areaId);
 | 
						|
      let isFirstChangedToolbar = true;
 | 
						|
      for (let areaNode of areaNodes) {
 | 
						|
        this.buildArea(areaId, placements, areaNode);
 | 
						|
 | 
						|
        let area = gAreas.get(areaId);
 | 
						|
        if (area.get("type") == CustomizableUI.TYPE_TOOLBAR) {
 | 
						|
          let defaultCollapsed = area.get("defaultCollapsed");
 | 
						|
          let win = areaNode.ownerGlobal;
 | 
						|
          if (defaultCollapsed !== null) {
 | 
						|
            win.setToolbarVisibility(
 | 
						|
              areaNode,
 | 
						|
              typeof defaultCollapsed == "string"
 | 
						|
                ? defaultCollapsed
 | 
						|
                : !defaultCollapsed,
 | 
						|
              isFirstChangedToolbar
 | 
						|
            );
 | 
						|
          }
 | 
						|
        }
 | 
						|
        isFirstChangedToolbar = false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Undoes a previous reset, restoring the state of the UI to the state prior to the reset.
 | 
						|
   */
 | 
						|
  undoReset() {
 | 
						|
    if (
 | 
						|
      gUIStateBeforeReset.uiCustomizationState == null ||
 | 
						|
      gUIStateBeforeReset.drawInTitlebar == null
 | 
						|
    ) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    gUndoResetting = true;
 | 
						|
 | 
						|
    const {
 | 
						|
      uiCustomizationState,
 | 
						|
      drawInTitlebar,
 | 
						|
      currentTheme,
 | 
						|
      uiDensity,
 | 
						|
      autoTouchMode,
 | 
						|
      autoHideDownloadsButton,
 | 
						|
    } = gUIStateBeforeReset;
 | 
						|
    gNewElementCount = gUIStateBeforeReset.newElementCount;
 | 
						|
 | 
						|
    // Need to clear the previous state before setting the prefs
 | 
						|
    // because pref observers may check if there is a previous UI state.
 | 
						|
    this._clearPreviousUIState();
 | 
						|
 | 
						|
    Services.prefs.setCharPref(kPrefCustomizationState, uiCustomizationState);
 | 
						|
    Services.prefs.setIntPref(kPrefDrawInTitlebar, drawInTitlebar);
 | 
						|
    Services.prefs.setIntPref(kPrefUIDensity, uiDensity);
 | 
						|
    Services.prefs.setBoolPref(kPrefAutoTouchMode, autoTouchMode);
 | 
						|
    Services.prefs.setBoolPref(
 | 
						|
      kPrefAutoHideDownloadsButton,
 | 
						|
      autoHideDownloadsButton
 | 
						|
    );
 | 
						|
    currentTheme.enable();
 | 
						|
    this.loadSavedState();
 | 
						|
    // If the user just customizes toolbar/titlebar visibility, gSavedState will be null
 | 
						|
    // and we don't need to do anything else here:
 | 
						|
    if (gSavedState) {
 | 
						|
      for (let areaId of Object.keys(gSavedState.placements)) {
 | 
						|
        let placements = gSavedState.placements[areaId];
 | 
						|
        gPlacements.set(areaId, placements);
 | 
						|
      }
 | 
						|
      this._rebuildRegisteredAreas();
 | 
						|
    }
 | 
						|
 | 
						|
    gUndoResetting = false;
 | 
						|
  },
 | 
						|
 | 
						|
  _clearPreviousUIState() {
 | 
						|
    Object.getOwnPropertyNames(gUIStateBeforeReset).forEach(prop => {
 | 
						|
      gUIStateBeforeReset[prop] = null;
 | 
						|
    });
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * @param {String|Node} aWidget - widget ID or a widget node (preferred for performance).
 | 
						|
   * @return {Boolean} whether the widget is removable
 | 
						|
   */
 | 
						|
  isWidgetRemovable(aWidget) {
 | 
						|
    let widgetId;
 | 
						|
    let widgetNode;
 | 
						|
    if (typeof aWidget == "string") {
 | 
						|
      widgetId = aWidget;
 | 
						|
    } else {
 | 
						|
      // Skipped items could just not have ids.
 | 
						|
      if (!aWidget.id && aWidget.getAttribute("skipintoolbarset") == "true") {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
      if (
 | 
						|
        !aWidget.id &&
 | 
						|
        !["toolbarspring", "toolbarspacer", "toolbarseparator"].includes(
 | 
						|
          aWidget.nodeName
 | 
						|
        )
 | 
						|
      ) {
 | 
						|
        throw new Error(
 | 
						|
          "No nodes without ids that aren't special widgets should ever come into contact with CUI"
 | 
						|
        );
 | 
						|
      }
 | 
						|
      // Use "spring" / "spacer" / "separator" for special widgets without ids
 | 
						|
      widgetId =
 | 
						|
        aWidget.id || aWidget.nodeName.substring(7 /* "toolbar".length */);
 | 
						|
      widgetNode = aWidget;
 | 
						|
    }
 | 
						|
    let provider = this.getWidgetProvider(widgetId);
 | 
						|
 | 
						|
    if (provider == CustomizableUI.PROVIDER_API) {
 | 
						|
      return gPalette.get(widgetId).removable;
 | 
						|
    }
 | 
						|
 | 
						|
    if (provider == CustomizableUI.PROVIDER_XUL) {
 | 
						|
      if (gBuildWindows.size == 0) {
 | 
						|
        // We don't have any build windows to look at, so just assume for now
 | 
						|
        // that its removable.
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
 | 
						|
      if (!widgetNode) {
 | 
						|
        // Pick any of the build windows to look at.
 | 
						|
        let [window] = [...gBuildWindows][0];
 | 
						|
        [, widgetNode] = this.getWidgetNode(widgetId, window);
 | 
						|
      }
 | 
						|
      // If we don't have a node, we assume it's removable. This can happen because
 | 
						|
      // getWidgetProvider returns PROVIDER_XUL by default, but this will also happen
 | 
						|
      // for API-provided widgets which have been destroyed.
 | 
						|
      if (!widgetNode) {
 | 
						|
        return true;
 | 
						|
      }
 | 
						|
      return widgetNode.getAttribute("removable") == "true";
 | 
						|
    }
 | 
						|
 | 
						|
    // Otherwise this is either a special widget, which is always removable, or
 | 
						|
    // an API widget which has already been removed from gPalette. Returning true
 | 
						|
    // here allows us to then remove its ID from any placements where it might
 | 
						|
    // still occur.
 | 
						|
    return true;
 | 
						|
  },
 | 
						|
 | 
						|
  canWidgetMoveToArea(aWidgetId, aArea) {
 | 
						|
    // Special widgets can't move to the menu panel.
 | 
						|
    if (
 | 
						|
      this.isSpecialWidget(aWidgetId) &&
 | 
						|
      gAreas.has(aArea) &&
 | 
						|
      gAreas.get(aArea).get("type") == CustomizableUI.TYPE_PANEL
 | 
						|
    ) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (
 | 
						|
      aArea == CustomizableUI.AREA_ADDONS &&
 | 
						|
      !CustomizableUI.isWebExtensionWidget(aWidgetId)
 | 
						|
    ) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (CustomizableUI.isWebExtensionWidget(aWidgetId)) {
 | 
						|
      // Extension widgets cannot move to the customization palette.
 | 
						|
      if (aArea == CustomizableUI.AREA_NO_AREA) {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
 | 
						|
      // Extension widgets cannot move to panels, with the exception of the
 | 
						|
      // AREA_ADDONS area.
 | 
						|
      if (
 | 
						|
        gAreas.get(aArea).get("type") == CustomizableUI.TYPE_PANEL &&
 | 
						|
        aArea != CustomizableUI.AREA_ADDONS
 | 
						|
      ) {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    let placement = this.getPlacementOfWidget(aWidgetId);
 | 
						|
    // Items in the palette can move, and items can move within their area:
 | 
						|
    if (!placement || placement.area == aArea) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
    // For everything else, just return whether the widget is removable.
 | 
						|
    return this.isWidgetRemovable(aWidgetId);
 | 
						|
  },
 | 
						|
 | 
						|
  ensureWidgetPlacedInWindow(aWidgetId, aWindow) {
 | 
						|
    let placement = this.getPlacementOfWidget(aWidgetId);
 | 
						|
    if (!placement) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    let areaNodes = gBuildAreas.get(placement.area);
 | 
						|
    if (!areaNodes) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    let container = [...areaNodes].filter(n => n.ownerGlobal == aWindow);
 | 
						|
    if (!container.length) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    let existingNode = container[0].getElementsByAttribute("id", aWidgetId)[0];
 | 
						|
    if (existingNode) {
 | 
						|
      return true;
 | 
						|
    }
 | 
						|
 | 
						|
    this.insertNodeInWindow(aWidgetId, container[0], true);
 | 
						|
    return true;
 | 
						|
  },
 | 
						|
 | 
						|
  _getCurrentWidgetsInContainer(container) {
 | 
						|
    // Get a list of all the widget IDs in this container, including any that
 | 
						|
    // are overflown.
 | 
						|
    let currentWidgets = new Set();
 | 
						|
    function addUnskippedChildren(parent) {
 | 
						|
      for (let node of parent.children) {
 | 
						|
        let realNode =
 | 
						|
          node.localName == "toolbarpaletteitem"
 | 
						|
            ? node.firstElementChild
 | 
						|
            : node;
 | 
						|
        if (realNode.getAttribute("skipintoolbarset") != "true") {
 | 
						|
          currentWidgets.add(realNode.id);
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
    addUnskippedChildren(this.getCustomizationTarget(container));
 | 
						|
    if (container.getAttribute("overflowing") == "true") {
 | 
						|
      let overflowTarget = container.getAttribute("default-overflowtarget");
 | 
						|
      addUnskippedChildren(
 | 
						|
        container.ownerDocument.getElementById(overflowTarget)
 | 
						|
      );
 | 
						|
      let webExtOverflowTarget = container.getAttribute(
 | 
						|
        "addon-webext-overflowtarget"
 | 
						|
      );
 | 
						|
      addUnskippedChildren(
 | 
						|
        container.ownerDocument.getElementById(webExtOverflowTarget)
 | 
						|
      );
 | 
						|
    }
 | 
						|
    // Then get the sorted list of placements, and filter based on the nodes
 | 
						|
    // that are present. This avoids including items that don't exist (e.g. ids
 | 
						|
    // of add-on items that the user has uninstalled).
 | 
						|
    let orderedPlacements = CustomizableUI.getWidgetIdsInArea(container.id);
 | 
						|
    return orderedPlacements.filter(w => currentWidgets.has(w));
 | 
						|
  },
 | 
						|
 | 
						|
  get inDefaultState() {
 | 
						|
    for (let [areaId, props] of gAreas) {
 | 
						|
      let defaultPlacements = props
 | 
						|
        .get("defaultPlacements")
 | 
						|
        .filter(item => this.widgetExists(item));
 | 
						|
      let currentPlacements = gPlacements.get(areaId);
 | 
						|
      // We're excluding all of the placement IDs for items that do not exist,
 | 
						|
      // and items that have removable="false",
 | 
						|
      // because we don't want to consider them when determining if we're
 | 
						|
      // in the default state. This way, if an add-on introduces a widget
 | 
						|
      // and is then uninstalled, the leftover placement doesn't cause us to
 | 
						|
      // automatically assume that the buttons are not in the default state.
 | 
						|
      let buildAreaNodes = gBuildAreas.get(areaId);
 | 
						|
      if (buildAreaNodes && buildAreaNodes.size) {
 | 
						|
        let container = [...buildAreaNodes][0];
 | 
						|
        let removableOrDefault = itemNodeOrItem => {
 | 
						|
          let item = (itemNodeOrItem && itemNodeOrItem.id) || itemNodeOrItem;
 | 
						|
          let isRemovable = this.isWidgetRemovable(itemNodeOrItem);
 | 
						|
          let isInDefault = defaultPlacements.includes(item);
 | 
						|
          return isRemovable || isInDefault;
 | 
						|
        };
 | 
						|
        // Toolbars need to deal with overflown widgets (if any) - so
 | 
						|
        // specialcase them:
 | 
						|
        if (props.get("type") == CustomizableUI.TYPE_TOOLBAR) {
 | 
						|
          currentPlacements =
 | 
						|
            this._getCurrentWidgetsInContainer(container).filter(
 | 
						|
              removableOrDefault
 | 
						|
            );
 | 
						|
        } else {
 | 
						|
          currentPlacements = currentPlacements.filter(item => {
 | 
						|
            let itemNode = container.getElementsByAttribute("id", item)[0];
 | 
						|
            return itemNode && removableOrDefault(itemNode || item);
 | 
						|
          });
 | 
						|
        }
 | 
						|
 | 
						|
        if (props.get("type") == CustomizableUI.TYPE_TOOLBAR) {
 | 
						|
          let collapsed = null;
 | 
						|
          let defaultCollapsed = props.get("defaultCollapsed");
 | 
						|
          let nondefaultState = false;
 | 
						|
          if (areaId == CustomizableUI.AREA_BOOKMARKS) {
 | 
						|
            collapsed = Services.prefs.getCharPref(
 | 
						|
              "browser.toolbars.bookmarks.visibility"
 | 
						|
            );
 | 
						|
            nondefaultState = Services.prefs.prefHasUserValue(
 | 
						|
              "browser.toolbars.bookmarks.visibility"
 | 
						|
            );
 | 
						|
          } else {
 | 
						|
            let attribute =
 | 
						|
              container.getAttribute("type") == "menubar"
 | 
						|
                ? "autohide"
 | 
						|
                : "collapsed";
 | 
						|
            collapsed = container.getAttribute(attribute) == "true";
 | 
						|
            nondefaultState = collapsed != defaultCollapsed;
 | 
						|
          }
 | 
						|
          if (defaultCollapsed !== null && nondefaultState) {
 | 
						|
            lazy.log.debug(
 | 
						|
              "Found " +
 | 
						|
                areaId +
 | 
						|
                " had non-default toolbar visibility" +
 | 
						|
                "(expected " +
 | 
						|
                defaultCollapsed +
 | 
						|
                ", was " +
 | 
						|
                collapsed +
 | 
						|
                ")"
 | 
						|
            );
 | 
						|
            return false;
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
      lazy.log.debug(
 | 
						|
        "Checking default state for " +
 | 
						|
          areaId +
 | 
						|
          ":\n" +
 | 
						|
          currentPlacements.join(",") +
 | 
						|
          "\nvs.\n" +
 | 
						|
          defaultPlacements.join(",")
 | 
						|
      );
 | 
						|
 | 
						|
      if (currentPlacements.length != defaultPlacements.length) {
 | 
						|
        return false;
 | 
						|
      }
 | 
						|
 | 
						|
      for (let i = 0; i < currentPlacements.length; ++i) {
 | 
						|
        if (
 | 
						|
          currentPlacements[i] != defaultPlacements[i] &&
 | 
						|
          !this.matchingSpecials(currentPlacements[i], defaultPlacements[i])
 | 
						|
        ) {
 | 
						|
          lazy.log.debug(
 | 
						|
            "Found " +
 | 
						|
              currentPlacements[i] +
 | 
						|
              " in " +
 | 
						|
              areaId +
 | 
						|
              " where " +
 | 
						|
              defaultPlacements[i] +
 | 
						|
              " was expected!"
 | 
						|
          );
 | 
						|
          return false;
 | 
						|
        }
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    if (Services.prefs.prefHasUserValue(kPrefUIDensity)) {
 | 
						|
      lazy.log.debug(kPrefUIDensity + " pref is non-default");
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (Services.prefs.prefHasUserValue(kPrefAutoTouchMode)) {
 | 
						|
      lazy.log.debug(kPrefAutoTouchMode + " pref is non-default");
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    if (Services.prefs.prefHasUserValue(kPrefDrawInTitlebar)) {
 | 
						|
      lazy.log.debug(kPrefDrawInTitlebar + " pref is non-default");
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    // This should just be `gDefaultTheme.isActive`, but bugs...
 | 
						|
    if (gDefaultTheme && gDefaultTheme.id != gSelectedTheme.id) {
 | 
						|
      lazy.log.debug(gSelectedTheme.id + " theme is non-default");
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
 | 
						|
    return true;
 | 
						|
  },
 | 
						|
 | 
						|
  getCollapsedToolbarIds(window) {
 | 
						|
    let collapsedToolbars = new Set();
 | 
						|
    for (let toolbarId of CustomizableUIInternal._builtinToolbars) {
 | 
						|
      let toolbar = window.document.getElementById(toolbarId);
 | 
						|
 | 
						|
      // Menubar toolbars are special in that they're hidden with the autohide
 | 
						|
      // attribute.
 | 
						|
      let hidingAttribute =
 | 
						|
        toolbar.getAttribute("type") == "menubar" ? "autohide" : "collapsed";
 | 
						|
 | 
						|
      if (toolbar.getAttribute(hidingAttribute) == "true") {
 | 
						|
        collapsedToolbars.add(toolbarId);
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    return collapsedToolbars;
 | 
						|
  },
 | 
						|
 | 
						|
  setToolbarVisibility(aToolbarId, aIsVisible) {
 | 
						|
    // We only persist the attribute the first time.
 | 
						|
    let isFirstChangedToolbar = true;
 | 
						|
    for (let window of CustomizableUI.windows) {
 | 
						|
      let toolbar = window.document.getElementById(aToolbarId);
 | 
						|
      if (toolbar) {
 | 
						|
        window.setToolbarVisibility(toolbar, aIsVisible, isFirstChangedToolbar);
 | 
						|
        isFirstChangedToolbar = false;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  },
 | 
						|
 | 
						|
  observe(aSubject, aTopic, aData) {
 | 
						|
    if (aTopic == "browser-set-toolbar-visibility") {
 | 
						|
      let [toolbar, visibility] = JSON.parse(aData);
 | 
						|
      CustomizableUI.setToolbarVisibility(toolbar, visibility == "true");
 | 
						|
    }
 | 
						|
  },
 | 
						|
};
 | 
						|
Object.freeze(CustomizableUIInternal);
 | 
						|
 | 
						|
export var CustomizableUI = {
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the navigation toolbar.
 | 
						|
   */
 | 
						|
  AREA_NAVBAR: "nav-bar",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the menubar's toolbar.
 | 
						|
   */
 | 
						|
  AREA_MENUBAR: "toolbar-menubar",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the tabstrip toolbar.
 | 
						|
   */
 | 
						|
  AREA_TABSTRIP: "TabsToolbar",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the bookmarks toolbar.
 | 
						|
   */
 | 
						|
  AREA_BOOKMARKS: "PersonalToolbar",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the non-dymanic (fixed) list in the overflow panel.
 | 
						|
   */
 | 
						|
  AREA_FIXED_OVERFLOW_PANEL: "widget-overflow-fixed-list",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the addons area.
 | 
						|
   */
 | 
						|
  AREA_ADDONS: "unified-extensions-area",
 | 
						|
  /**
 | 
						|
   * Constant reference to the ID of the customization palette, which is
 | 
						|
   * where widgets go when they're not assigned to an area. Note that this
 | 
						|
   * area is "virtual" in that it's never set as a value for a widgets
 | 
						|
   * currentArea or defaultArea. It's only used for the `canWidgetMoveToArea`
 | 
						|
   * function to check if widgets can be moved to the palette. Callers who
 | 
						|
   * wish to move items to the palette should use `removeWidgetFromArea`.
 | 
						|
   */
 | 
						|
  AREA_NO_AREA: "customization-palette",
 | 
						|
  /**
 | 
						|
   * Constant indicating the area is a panel.
 | 
						|
   */
 | 
						|
  TYPE_PANEL: "panel",
 | 
						|
  /**
 | 
						|
   * Constant indicating the area is a toolbar.
 | 
						|
   */
 | 
						|
  TYPE_TOOLBAR: "toolbar",
 | 
						|
 | 
						|
  /**
 | 
						|
   * Constant indicating a XUL-type provider.
 | 
						|
   */
 | 
						|
  PROVIDER_XUL: "xul",
 | 
						|
  /**
 | 
						|
   * Constant indicating an API-type provider.
 | 
						|
   */
 | 
						|
  PROVIDER_API: "api",
 | 
						|
  /**
 | 
						|
   * Constant indicating dynamic (special) widgets: spring, spacer, and separator.
 | 
						|
   */
 | 
						|
  PROVIDER_SPECIAL: "special",
 | 
						|
 | 
						|
  /**
 | 
						|
   * Constant indicating the widget is built-in
 | 
						|
   */
 | 
						|
  SOURCE_BUILTIN: "builtin",
 | 
						|
  /**
 | 
						|
   * Constant indicating the widget is externally provided
 | 
						|
   * (e.g. by add-ons or other items not part of the builtin widget set).
 | 
						|
   */
 | 
						|
  SOURCE_EXTERNAL: "external",
 | 
						|
 | 
						|
  /**
 | 
						|
   * Constant indicating the reason the event was fired was a window closing
 | 
						|
   */
 | 
						|
  REASON_WINDOW_CLOSED: "window-closed",
 | 
						|
  /**
 | 
						|
   * Constant indicating the reason the event was fired was an area being
 | 
						|
   * unregistered separately from window closing mechanics.
 | 
						|
   */
 | 
						|
  REASON_AREA_UNREGISTERED: "area-unregistered",
 | 
						|
 | 
						|
  /**
 | 
						|
   * An iteratable property of windows managed by CustomizableUI.
 | 
						|
   * Note that this can *only* be used as an iterator. ie:
 | 
						|
   *     for (let window of CustomizableUI.windows) { ... }
 | 
						|
   */
 | 
						|
  windows: {
 | 
						|
    *[Symbol.iterator]() {
 | 
						|
      for (let [window] of gBuildWindows) {
 | 
						|
        yield window;
 | 
						|
      }
 | 
						|
    },
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Add a listener object that will get fired for various events regarding
 | 
						|
   * customization.
 | 
						|
   *
 | 
						|
   * @param aListener the listener object to add
 | 
						|
   *
 | 
						|
   * Not all event handler methods need to be defined.
 | 
						|
   * CustomizableUI will catch exceptions. Events are dispatched
 | 
						|
   * synchronously on the UI thread, so if you can delay any/some of your
 | 
						|
   * processing, that is advisable. The following event handlers are supported:
 | 
						|
   *   - onWidgetAdded(aWidgetId, aArea, aPosition)
 | 
						|
   *     Fired when a widget is added to an area. aWidgetId is the widget that
 | 
						|
   *     was added, aArea the area it was added to, and aPosition the position
 | 
						|
   *     in which it was added.
 | 
						|
   *   - onWidgetMoved(aWidgetId, aArea, aOldPosition, aNewPosition)
 | 
						|
   *     Fired when a widget is moved within its area. aWidgetId is the widget
 | 
						|
   *     that was moved, aArea the area it was moved in, aOldPosition its old
 | 
						|
   *     position, and aNewPosition its new position.
 | 
						|
   *   - onWidgetRemoved(aWidgetId, aArea)
 | 
						|
   *     Fired when a widget is removed from its area. aWidgetId is the widget
 | 
						|
   *     that was removed, aArea the area it was removed from.
 | 
						|
   *
 | 
						|
   *   - onWidgetBeforeDOMChange(aNode, aNextNode, aContainer, aIsRemoval)
 | 
						|
   *     Fired *before* a widget's DOM node is acted upon by CustomizableUI
 | 
						|
   *     (to add, move or remove it). aNode is the DOM node changed, aNextNode
 | 
						|
   *     the DOM node (if any) before which a widget will be inserted,
 | 
						|
   *     aContainer the *actual* DOM container (could be an overflow panel in
 | 
						|
   *     case of an overflowable toolbar), and aWasRemoval is true iff the
 | 
						|
   *     action about to happen is the removal of the DOM node.
 | 
						|
   *   - onWidgetAfterDOMChange(aNode, aNextNode, aContainer, aWasRemoval)
 | 
						|
   *     Like onWidgetBeforeDOMChange, but fired after the change to the DOM
 | 
						|
   *     node of the widget.
 | 
						|
   *
 | 
						|
   *   - onWidgetReset(aNode, aContainer)
 | 
						|
   *     Fired after a reset to default placements moves a widget's node to a
 | 
						|
   *     different location. aNode is the widget's node, aContainer is the
 | 
						|
   *     area it was moved into (NB: it might already have been there and been
 | 
						|
   *     moved to a different position!)
 | 
						|
   *   - onWidgetUndoMove(aNode, aContainer)
 | 
						|
   *     Fired after undoing a reset to default placements moves a widget's
 | 
						|
   *     node to a different location. aNode is the widget's node, aContainer
 | 
						|
   *     is the area it was moved into (NB: it might already have been there
 | 
						|
   *     and been moved to a different position!)
 | 
						|
   *   - onAreaReset(aArea, aContainer)
 | 
						|
   *     Fired after a reset to default placements is complete on an area's
 | 
						|
   *     DOM node. Note that this is fired for each DOM node. aArea is the area
 | 
						|
   *     that was reset, aContainer the DOM node that was reset.
 | 
						|
   *
 | 
						|
   *   - onWidgetCreated(aWidgetId)
 | 
						|
   *     Fired when a widget with id aWidgetId has been created, but before it
 | 
						|
   *     is added to any placements or any DOM nodes have been constructed.
 | 
						|
   *     Only fired for API-based widgets.
 | 
						|
   *   - onWidgetAfterCreation(aWidgetId, aArea)
 | 
						|
   *     Fired after a widget with id aWidgetId has been created, and has been
 | 
						|
   *     added to either its default area or the area in which it was placed
 | 
						|
   *     previously. If the widget has no default area and/or it has never
 | 
						|
   *     been placed anywhere, aArea may be null. Only fired for API-based
 | 
						|
   *     widgets.
 | 
						|
   *   - onWidgetDestroyed(aWidgetId)
 | 
						|
   *     Fired when widgets are destroyed. aWidgetId is the widget that is
 | 
						|
   *     being destroyed. Only fired for API-based widgets.
 | 
						|
   *   - onWidgetInstanceRemoved(aWidgetId, aDocument)
 | 
						|
   *     Fired when a window is unloaded and a widget's instance is destroyed
 | 
						|
   *     because of this. Only fired for API-based widgets.
 | 
						|
   *
 | 
						|
   *   - onWidgetDrag(aWidgetId, aArea)
 | 
						|
   *     Fired both when and after customize mode drag handling system tries
 | 
						|
   *     to determine the width and height of widget aWidgetId when dragged to a
 | 
						|
   *     different area. aArea will be the area the item is dragged to, or
 | 
						|
   *     undefined after the measurements have been done and the node has been
 | 
						|
   *     moved back to its 'regular' area.
 | 
						|
   *
 | 
						|
   *   - onCustomizeStart(aWindow)
 | 
						|
   *     Fired when opening customize mode in aWindow.
 | 
						|
   *   - onCustomizeEnd(aWindow)
 | 
						|
   *     Fired when exiting customize mode in aWindow.
 | 
						|
   *
 | 
						|
   *   - onWidgetOverflow(aNode, aContainer)
 | 
						|
   *     Fired when a widget's DOM node is overflowing its container, a toolbar,
 | 
						|
   *     and will be displayed in the overflow panel.
 | 
						|
   *   - onWidgetUnderflow(aNode, aContainer)
 | 
						|
   *     Fired when a widget's DOM node is *not* overflowing its container, a
 | 
						|
   *     toolbar, anymore.
 | 
						|
   *   - onWindowOpened(aWindow)
 | 
						|
   *     Fired when a window has been opened that is managed by CustomizableUI,
 | 
						|
   *     once all of the prerequisite setup has been done.
 | 
						|
   *   - onWindowClosed(aWindow)
 | 
						|
   *     Fired when a window that has been managed by CustomizableUI has been
 | 
						|
   *     closed.
 | 
						|
   *   - onAreaNodeRegistered(aArea, aContainer)
 | 
						|
   *     Fired after an area node is first built when it is registered. This
 | 
						|
   *     is often when the window has opened, but in the case of add-ons,
 | 
						|
   *     could fire when the node has just been registered with CustomizableUI
 | 
						|
   *     after an add-on update or disable/enable sequence.
 | 
						|
   *   - onAreaNodeUnregistered(aArea, aContainer, aReason)
 | 
						|
   *     Fired when an area node is explicitly unregistered by an API caller,
 | 
						|
   *     or by a window closing. The aReason parameter indicates which of
 | 
						|
   *     these is the case.
 | 
						|
   */
 | 
						|
  addListener(aListener) {
 | 
						|
    CustomizableUIInternal.addListener(aListener);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Remove a listener added with addListener
 | 
						|
   * @param aListener the listener object to remove
 | 
						|
   */
 | 
						|
  removeListener(aListener) {
 | 
						|
    CustomizableUIInternal.removeListener(aListener);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Register a customizable area with CustomizableUI.
 | 
						|
   * @param aName   the name of the area to register. Can only contain
 | 
						|
   *                alphanumeric characters, dashes (-) and underscores (_).
 | 
						|
   * @param aProps  the properties of the area. The following properties are
 | 
						|
   *                recognized:
 | 
						|
   *                - type:   the type of area. Either TYPE_TOOLBAR (default) or
 | 
						|
   *                          TYPE_PANEL;
 | 
						|
   *                - anchor: for a menu panel or overflowable toolbar, the
 | 
						|
   *                          anchoring node for the panel.
 | 
						|
   *                - overflowable: set to true if your toolbar is overflowable.
 | 
						|
   *                                This requires an anchor, and only has an
 | 
						|
   *                                effect for toolbars.
 | 
						|
   *                - defaultPlacements: an array of widget IDs making up the
 | 
						|
   *                                     default contents of the area
 | 
						|
   *                - defaultCollapsed: (INTERNAL ONLY) applies if the type is TYPE_TOOLBAR, specifies
 | 
						|
   *                                    if toolbar is collapsed by default (default to true).
 | 
						|
   *                                    Specify null to ensure that reset/inDefaultArea don't care
 | 
						|
   *                                    about a toolbar's collapsed state
 | 
						|
   */
 | 
						|
  registerArea(aName, aProperties) {
 | 
						|
    CustomizableUIInternal.registerArea(aName, aProperties);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Register a concrete node for a registered area. This method needs to be called
 | 
						|
   * with any toolbar in the main browser window that has its "customizable" attribute
 | 
						|
   * set to true.
 | 
						|
   *
 | 
						|
   * Note that ideally, you should register your toolbar using registerArea
 | 
						|
   * before calling this. If you don't, the node will be saved for processing when
 | 
						|
   * you call registerArea. Note that CustomizableUI won't restore state in the area,
 | 
						|
   * allow the user to customize it in customize mode, or otherwise deal
 | 
						|
   * with it, until the area has been registered.
 | 
						|
   */
 | 
						|
  registerToolbarNode(aToolbar) {
 | 
						|
    CustomizableUIInternal.registerToolbarNode(aToolbar);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Register a panel node. A panel treated slightly differently from a toolbar in
 | 
						|
   * terms of what items can be moved into it. For example, a panel cannot have a
 | 
						|
   * spacer or a spring put into it.
 | 
						|
   *
 | 
						|
   * @param aPanelContents the panel contents DOM node being registered.
 | 
						|
   * @param aArea the area for which to register this node.
 | 
						|
   */
 | 
						|
  registerPanelNode(aNode, aArea) {
 | 
						|
    CustomizableUIInternal.registerPanelNode(aNode, aArea);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Unregister a customizable area. The inverse of registerArea.
 | 
						|
   *
 | 
						|
   * Unregistering an area will remove all the (removable) widgets in the
 | 
						|
   * area, which will return to the panel, and destroy all other traces
 | 
						|
   * of the area within CustomizableUI. Note that this means the *contents*
 | 
						|
   * of the area's DOM nodes will be moved to the panel or removed, but
 | 
						|
   * the area's DOM nodes *themselves* will stay.
 | 
						|
   *
 | 
						|
   * Furthermore, by default the placements of the area will be kept in the
 | 
						|
   * saved state (!) and restored if you re-register the area at a later
 | 
						|
   * point. This is useful for e.g. add-ons that get disabled and then
 | 
						|
   * re-enabled (e.g. when they update).
 | 
						|
   *
 | 
						|
   * You can override this last behaviour (and destroy the placements
 | 
						|
   * information in the saved state) by passing true for aDestroyPlacements.
 | 
						|
   *
 | 
						|
   * @param aName              the name of the area to unregister
 | 
						|
   * @param aDestroyPlacements whether to destroy the placements information
 | 
						|
   *                           for the area, too.
 | 
						|
   */
 | 
						|
  unregisterArea(aName, aDestroyPlacements) {
 | 
						|
    CustomizableUIInternal.unregisterArea(aName, aDestroyPlacements);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Add a widget to an area.
 | 
						|
   * If the area to which you try to add is not known to CustomizableUI,
 | 
						|
   * this will throw.
 | 
						|
   * If the area to which you try to add is the same as the area in which
 | 
						|
   * the widget is currently placed, this will do the same as
 | 
						|
   * moveWidgetWithinArea.
 | 
						|
   * If the widget cannot be removed from its original location, this will
 | 
						|
   * no-op.
 | 
						|
   *
 | 
						|
   * This will fire an onWidgetAdded notification,
 | 
						|
   * and an onWidgetBeforeDOMChange and onWidgetAfterDOMChange notification
 | 
						|
   * for each window CustomizableUI knows about.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget to add
 | 
						|
   * @param aArea     the ID of the area to add the widget to
 | 
						|
   * @param aPosition the position at which to add the widget. If you do not
 | 
						|
   *                  pass a position, the widget will be added to the end
 | 
						|
   *                  of the area.
 | 
						|
   */
 | 
						|
  addWidgetToArea(aWidgetId, aArea, aPosition) {
 | 
						|
    CustomizableUIInternal.addWidgetToArea(aWidgetId, aArea, aPosition);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Remove a widget from its area. If the widget cannot be removed from its
 | 
						|
   * area, or is not in any area, this will no-op. Otherwise, this will fire an
 | 
						|
   * onWidgetRemoved notification, and an onWidgetBeforeDOMChange and
 | 
						|
   * onWidgetAfterDOMChange notification for each window CustomizableUI knows
 | 
						|
   * about.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget to remove
 | 
						|
   */
 | 
						|
  removeWidgetFromArea(aWidgetId) {
 | 
						|
    CustomizableUIInternal.removeWidgetFromArea(aWidgetId);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Move a widget within an area.
 | 
						|
   * If the widget is not in any area, this will no-op.
 | 
						|
   * If the widget is already at the indicated position, this will no-op.
 | 
						|
   *
 | 
						|
   * Otherwise, this will move the widget and fire an onWidgetMoved notification,
 | 
						|
   * and an onWidgetBeforeDOMChange and onWidgetAfterDOMChange notification for
 | 
						|
   * each window CustomizableUI knows about.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget to move
 | 
						|
   * @param aPosition the position to move the widget to.
 | 
						|
   *                  Negative values or values greater than the number of
 | 
						|
   *                  widgets will be interpreted to mean moving the widget to
 | 
						|
   *                  respectively the first or last position.
 | 
						|
   */
 | 
						|
  moveWidgetWithinArea(aWidgetId, aPosition) {
 | 
						|
    CustomizableUIInternal.moveWidgetWithinArea(aWidgetId, aPosition);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Ensure a XUL-based widget created in a window after areas were
 | 
						|
   * initialized moves to its correct position.
 | 
						|
   * Always prefer this over moving items in the DOM yourself.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget that was just created
 | 
						|
   * @param aWindow the window in which you want to ensure it was added.
 | 
						|
   *
 | 
						|
   * NB: why is this API per-window, you wonder? Because if you need this,
 | 
						|
   * presumably you yourself need to create the widget in all the windows
 | 
						|
   * and need to loop through them anyway.
 | 
						|
   */
 | 
						|
  ensureWidgetPlacedInWindow(aWidgetId, aWindow) {
 | 
						|
    return CustomizableUIInternal.ensureWidgetPlacedInWindow(
 | 
						|
      aWidgetId,
 | 
						|
      aWindow
 | 
						|
    );
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Start a batch update of items.
 | 
						|
   * During a batch update, the customization state is not saved to the user's
 | 
						|
   * preferences file, in order to reduce (possibly sync) IO.
 | 
						|
   * Calls to begin/endBatchUpdate may be nested.
 | 
						|
   *
 | 
						|
   * Callers should ensure that NO MATTER WHAT they call endBatchUpdate once
 | 
						|
   * for each call to beginBatchUpdate, even if there are exceptions in the
 | 
						|
   * code in the batch update. Otherwise, for the duration of the
 | 
						|
   * Firefox session, customization state is never saved. Typically, you
 | 
						|
   * would do this using a try...finally block.
 | 
						|
   */
 | 
						|
  beginBatchUpdate() {
 | 
						|
    CustomizableUIInternal.beginBatchUpdate();
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * End a batch update. See the documentation for beginBatchUpdate above.
 | 
						|
   *
 | 
						|
   * State is not saved if we believe it is identical to the last known
 | 
						|
   * saved state. State is only ever saved when all batch updates have
 | 
						|
   * finished (ie there has been 1 endBatchUpdate call for each
 | 
						|
   * beginBatchUpdate call). If any of the endBatchUpdate calls pass
 | 
						|
   * aForceDirty=true, we will flush to the prefs file.
 | 
						|
   *
 | 
						|
   * @param aForceDirty force CustomizableUI to flush to the prefs file when
 | 
						|
   *                    all batch updates have finished.
 | 
						|
   */
 | 
						|
  endBatchUpdate(aForceDirty) {
 | 
						|
    CustomizableUIInternal.endBatchUpdate(aForceDirty);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Create a widget.
 | 
						|
   *
 | 
						|
   * To create a widget, you should pass an object with its desired
 | 
						|
   * properties. The following properties are supported:
 | 
						|
   *
 | 
						|
   * - id:            the ID of the widget (required).
 | 
						|
   * - type:          a string indicating the type of widget. Possible types
 | 
						|
   *                  are:
 | 
						|
   *                  'button' - for simple button widgets (the default)
 | 
						|
   *                  'view'   - for buttons that open a panel or subview,
 | 
						|
   *                             depending on where they are placed.
 | 
						|
   *                  'button-and-view' - A combination of 'button' and 'view',
 | 
						|
   *                             which looks different depending on whether it's
 | 
						|
   *                             located in the toolbar or in the panel: When
 | 
						|
   *                             located in the toolbar, the widget is shown as
 | 
						|
   *                             a combined item of a button and a dropmarker
 | 
						|
   *                             button. The button triggers the command and the
 | 
						|
   *                             dropmarker button opens the view. When located
 | 
						|
   *                             in the panel, shown as one item which opens the
 | 
						|
   *                             view, and the button command cannot be
 | 
						|
   *                             triggered separately.
 | 
						|
   *                  'custom' - for fine-grained control over the creation
 | 
						|
   *                             of the widget.
 | 
						|
   * - viewId:        Only useful for views and button-and-view widgets (and
 | 
						|
   *                  required there): the id of the <panelview> that should be
 | 
						|
   *                  shown when clicking the widget.  If used with a custom
 | 
						|
   *                  widget, the widget must also provide a toolbaritem where
 | 
						|
   *                  the first child is the view button.
 | 
						|
   * - onBuild(aDoc): Only useful for custom widgets (and required there); a
 | 
						|
   *                  function that will be invoked with the document in which
 | 
						|
   *                  to build a widget. Should return the DOM node that has
 | 
						|
   *                  been constructed.
 | 
						|
   * - onBeforeCreated(aDoc): Attached to all non-custom widgets; a function
 | 
						|
   *                  that will be invoked before the widget gets a DOM node
 | 
						|
   *                  constructed, passing the document in which that will happen.
 | 
						|
   *                  This is useful especially for 'view' type widgets that need
 | 
						|
   *                  to construct their views on the fly (e.g. from bootstrapped
 | 
						|
   *                  add-ons). If the function returns `false`, the widget will
 | 
						|
   *                  not be created.
 | 
						|
   * - onCreated(aNode): Attached to all widgets; a function that will be invoked
 | 
						|
   *                  whenever the widget has a DOM node constructed, passing the
 | 
						|
   *                  constructed node as an argument.
 | 
						|
   * - onDestroyed(aDoc): Attached to all non-custom widgets; a function that
 | 
						|
   *                  will be invoked after the widget has a DOM node destroyed,
 | 
						|
   *                  passing the document from which it was removed. This is
 | 
						|
   *                  useful especially for 'view' type widgets that need to
 | 
						|
   *                  cleanup after views that were constructed on the fly.
 | 
						|
   * - onBeforeCommand(aEvt, aNode): A function that will be invoked when the user
 | 
						|
   *                          activates the button but before the command
 | 
						|
   *                          is evaluated. Useful if code needs to run to
 | 
						|
   *                          change the button's icon in preparation to the
 | 
						|
   *                          pending command action. Called for any type that
 | 
						|
   *                          supports the handler.  The command type, either
 | 
						|
   *                          "view" or "command", may be returned to force the
 | 
						|
   *                          action that will occur.  View will open the panel
 | 
						|
   *                          and command will result in calling onCommand.
 | 
						|
   * - onCommand(aEvt): Useful for custom, button and button-and-view widgets; a
 | 
						|
   *                    function that will be invoked when the user activates
 | 
						|
   *                    the button. A custom widget with a view should
 | 
						|
   *                    return "view" or "command" to continue processing
 | 
						|
   *                    the command per the needs of the widget.
 | 
						|
   * - onClick(aEvt): Attached to all widgets; a function that will be invoked
 | 
						|
   *                  when the user clicks the widget.
 | 
						|
   * - onViewShowing(aEvt): Only useful for views and button-and-view widgets; a
 | 
						|
   *                  function that will be invoked when a user shows your view.
 | 
						|
   *                  If any event handler calls aEvt.preventDefault(), the view
 | 
						|
   *                  will not be shown.
 | 
						|
   *
 | 
						|
   *                  The event's `detail` property is an object with an
 | 
						|
   *                  `addBlocker` method. Handlers which need to
 | 
						|
   *                  perform asynchronous operations before the view is
 | 
						|
   *                  shown may pass this method a Promise, which will
 | 
						|
   *                  prevent the view from showing until it resolves.
 | 
						|
   *                  Additionally, if the promise resolves to the exact
 | 
						|
   *                  value `false`, the view will not be shown.
 | 
						|
   * - onViewHiding(aEvt): Only useful for views; a function that will be
 | 
						|
   *                  invoked when a user hides your view.
 | 
						|
   * - l10nId:        fluent string identifier to use for localizing attributes
 | 
						|
   *                  on the widget. If present, preferred over the
 | 
						|
   *                  label/tooltiptext.
 | 
						|
   * - tooltiptext:   string to use for the tooltip of the widget
 | 
						|
   * - label:         string to use for the label of the widget
 | 
						|
   * - localized:     If true, or undefined, attempt to retrieve the
 | 
						|
   *                  widget's string properties from the customizable
 | 
						|
   *                  widgets string bundle.
 | 
						|
   * - removable:     whether the widget is removable (optional, default: true)
 | 
						|
   *                  NB: if you specify false here, you must provide a
 | 
						|
   *                  defaultArea, too.
 | 
						|
   * - overflows:     whether widget can overflow when in an overflowable
 | 
						|
   *                  toolbar (optional, default: true)
 | 
						|
   * - defaultArea:   default area to add the widget to
 | 
						|
   *                  (optional, default: none; required if non-removable)
 | 
						|
   * - shortcutId:    id of an element that has a shortcut for this widget
 | 
						|
   *                  (optional, default: null). This is only used to display
 | 
						|
   *                  the shortcut as part of the tooltip for builtin widgets
 | 
						|
   *                  (which have strings inside
 | 
						|
   *                  customizableWidgets.properties). If you're in an add-on,
 | 
						|
   *                  you should not set this property.
 | 
						|
   *                  If l10nId is provided, the resulting shortcut is passed
 | 
						|
   *                  as the "$shortcut" variable to the fluent message.
 | 
						|
   * - showInPrivateBrowsing: whether to show the widget in private browsing
 | 
						|
   *                          mode (optional, default: true)
 | 
						|
   * - tabSpecific:      If true, closes the panel if the tab changes.
 | 
						|
   * - locationSpecific: If true, closes the panel if the location changes.
 | 
						|
   *                     This is similar to tabSpecific, but also if the location
 | 
						|
   *                     changes in the same tab, we may want to close the panel.
 | 
						|
   * - webExtension:  Set to true if this widget is being created on behalf of an
 | 
						|
   *                  extension.
 | 
						|
   *
 | 
						|
   * @param aProperties the specifications for the widget.
 | 
						|
   * @return a wrapper around the created widget (see getWidget)
 | 
						|
   */
 | 
						|
  createWidget(aProperties) {
 | 
						|
    return CustomizableUIInternal.wrapWidget(
 | 
						|
      CustomizableUIInternal.createWidget(aProperties)
 | 
						|
    );
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Destroy a widget
 | 
						|
   *
 | 
						|
   * If the widget is part of the default placements in an area, this will
 | 
						|
   * remove it from there. It will also remove any DOM instances. However,
 | 
						|
   * it will keep the widget in the placements for whatever area it was
 | 
						|
   * in at the time. You can remove it from there yourself by calling
 | 
						|
   * CustomizableUI.removeWidgetFromArea(aWidgetId).
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget to destroy
 | 
						|
   */
 | 
						|
  destroyWidget(aWidgetId) {
 | 
						|
    CustomizableUIInternal.destroyWidget(aWidgetId);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Get a wrapper object with information about the widget.
 | 
						|
   * The object provides the following properties
 | 
						|
   * (all read-only unless otherwise indicated):
 | 
						|
   *
 | 
						|
   * - id:            the widget's ID;
 | 
						|
   * - type:          the type of widget (button, view, custom). For
 | 
						|
   *                  XUL-provided widgets, this is always 'custom';
 | 
						|
   * - provider:      the provider type of the widget, id est one of
 | 
						|
   *                  PROVIDER_API or PROVIDER_XUL;
 | 
						|
   * - forWindow(w):  a method to obtain a single window wrapper for a widget,
 | 
						|
   *                  in the window w passed as the only argument;
 | 
						|
   * - instances:     an array of all instances (single window wrappers)
 | 
						|
   *                  of the widget. This array is NOT live;
 | 
						|
   * - areaType:      the type of the widget's current area
 | 
						|
   * - isGroup:       true; will be false for wrappers around single widget nodes;
 | 
						|
   * - source:        for API-provided widgets, whether they are built-in to
 | 
						|
   *                  Firefox or add-on-provided;
 | 
						|
   * - disabled:      for API-provided widgets, whether the widget is currently
 | 
						|
   *                  disabled. NB: this property is writable, and will toggle
 | 
						|
   *                  all the widgets' nodes' disabled states;
 | 
						|
   * - label:         for API-provied widgets, the label of the widget;
 | 
						|
   * - tooltiptext:   for API-provided widgets, the tooltip of the widget;
 | 
						|
   * - showInPrivateBrowsing: for API-provided widgets, whether the widget is
 | 
						|
   *                          visible in private browsing;
 | 
						|
   *
 | 
						|
   * Single window wrappers obtained through forWindow(someWindow) or from the
 | 
						|
   * instances array have the following properties
 | 
						|
   * (all read-only unless otherwise indicated):
 | 
						|
   *
 | 
						|
   * - id:            the widget's ID;
 | 
						|
   * - type:          the type of widget (button, view, custom). For
 | 
						|
   *                  XUL-provided widgets, this is always 'custom';
 | 
						|
   * - provider:      the provider type of the widget, id est one of
 | 
						|
   *                  PROVIDER_API or PROVIDER_XUL;
 | 
						|
   * - node:          reference to the corresponding DOM node;
 | 
						|
   * - anchor:        the anchor on which to anchor panels opened from this
 | 
						|
   *                  node. This will point to the overflow chevron on
 | 
						|
   *                  overflowable toolbars if and only if your widget node
 | 
						|
   *                  is overflowed, to the anchor for the panel menu
 | 
						|
   *                  if your widget is inside the panel menu, and to the
 | 
						|
   *                  node itself in all other cases;
 | 
						|
   * - overflowed:    boolean indicating whether the node is currently in the
 | 
						|
   *                  overflow panel of the toolbar;
 | 
						|
   * - isGroup:       false; will be true for the group widget;
 | 
						|
   * - label:         for API-provided widgets, convenience getter for the
 | 
						|
   *                  label attribute of the DOM node;
 | 
						|
   * - tooltiptext:   for API-provided widgets, convenience getter for the
 | 
						|
   *                  tooltiptext attribute of the DOM node;
 | 
						|
   * - disabled:      for API-provided widgets, convenience getter *and setter*
 | 
						|
   *                  for the disabled state of this single widget. Note that
 | 
						|
   *                  you may prefer to use the group wrapper's getter/setter
 | 
						|
   *                  instead.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget whose information you need
 | 
						|
   * @return a wrapper around the widget as described above, or null if the
 | 
						|
   *         widget is known not to exist (anymore). NB: non-null return
 | 
						|
   *         is no guarantee the widget exists because we cannot know in
 | 
						|
   *         advance if a XUL widget exists or not.
 | 
						|
   */
 | 
						|
  getWidget(aWidgetId) {
 | 
						|
    return CustomizableUIInternal.wrapWidget(aWidgetId);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Get an array of widget wrappers (see getWidget) for all the widgets
 | 
						|
   * which are currently not in any area (so which are in the palette).
 | 
						|
   *
 | 
						|
   * @param aWindowPalette the palette (and by extension, the window) in which
 | 
						|
   *                       CustomizableUI should look. This matters because of
 | 
						|
   *                       course XUL-provided widgets could be available in
 | 
						|
   *                       some windows but not others, and likewise
 | 
						|
   *                       API-provided widgets might not exist in a private
 | 
						|
   *                       window (because of the showInPrivateBrowsing
 | 
						|
   *                       property).
 | 
						|
   *
 | 
						|
   * @return an array of widget wrappers (see getWidget)
 | 
						|
   */
 | 
						|
  getUnusedWidgets(aWindowPalette) {
 | 
						|
    return CustomizableUIInternal.getUnusedWidgets(aWindowPalette).map(
 | 
						|
      CustomizableUIInternal.wrapWidget,
 | 
						|
      CustomizableUIInternal
 | 
						|
    );
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Get an array of all the widget IDs placed in an area.
 | 
						|
   * Modifying the array will not affect CustomizableUI.
 | 
						|
   *
 | 
						|
   * @param aArea the ID of the area whose placements you want to obtain.
 | 
						|
   * @return an array containing the widget IDs that are in the area.
 | 
						|
   *
 | 
						|
   * NB: will throw if called too early (before placements have been fetched)
 | 
						|
   *     or if the area is not currently known to CustomizableUI.
 | 
						|
   */
 | 
						|
  getWidgetIdsInArea(aArea) {
 | 
						|
    if (!gAreas.has(aArea)) {
 | 
						|
      throw new Error("Unknown customization area: " + aArea);
 | 
						|
    }
 | 
						|
    if (!gPlacements.has(aArea)) {
 | 
						|
      throw new Error("Area not yet restored");
 | 
						|
    }
 | 
						|
 | 
						|
    // We need to clone this, as we don't want to let consumers muck with placements
 | 
						|
    return [...gPlacements.get(aArea)];
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Get an array of widget wrappers for all the widgets in an area. This is
 | 
						|
   * the same as calling getWidgetIdsInArea and .map() ing the result through
 | 
						|
   * CustomizableUI.getWidget. Careful: this means that if there are IDs in there
 | 
						|
   * which don't have corresponding DOM nodes, there might be nulls in this array,
 | 
						|
   * or items for which wrapper.forWindow(win) will return null.
 | 
						|
   *
 | 
						|
   * @param aArea the ID of the area whose widgets you want to obtain.
 | 
						|
   * @return an array of widget wrappers and/or null values for the widget IDs
 | 
						|
   *         placed in an area.
 | 
						|
   *
 | 
						|
   * NB: will throw if called too early (before placements have been fetched)
 | 
						|
   *     or if the area is not currently known to CustomizableUI.
 | 
						|
   */
 | 
						|
  getWidgetsInArea(aArea) {
 | 
						|
    return this.getWidgetIdsInArea(aArea).map(
 | 
						|
      CustomizableUIInternal.wrapWidget,
 | 
						|
      CustomizableUIInternal
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Ensure the customizable widget that matches up with this view node
 | 
						|
   * will get the right subview showing/shown/hiding/hidden events when
 | 
						|
   * they fire.
 | 
						|
   * @param aViewNode the view node to add listeners to if they haven't
 | 
						|
   *                  been added already.
 | 
						|
   */
 | 
						|
  ensureSubviewListeners(aViewNode) {
 | 
						|
    return CustomizableUIInternal.ensureSubviewListeners(aViewNode);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Obtain an array of all the area IDs known to CustomizableUI.
 | 
						|
   * This array is created for you, so is modifiable without CustomizableUI
 | 
						|
   * being affected.
 | 
						|
   */
 | 
						|
  get areas() {
 | 
						|
    return [...gAreas.keys()];
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check what kind of area (toolbar or menu panel) an area is. This is
 | 
						|
   * useful if you have a widget that needs to behave differently depending
 | 
						|
   * on its location. Note that widget wrappers have a convenience getter
 | 
						|
   * property (areaType) for this purpose.
 | 
						|
   *
 | 
						|
   * @param aArea the ID of the area whose type you want to know
 | 
						|
   * @return TYPE_TOOLBAR or TYPE_PANEL depending on the area, null if
 | 
						|
   *         the area is unknown.
 | 
						|
   */
 | 
						|
  getAreaType(aArea) {
 | 
						|
    let area = gAreas.get(aArea);
 | 
						|
    return area ? area.get("type") : null;
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check if a toolbar is collapsed by default.
 | 
						|
   *
 | 
						|
   * @param aArea the ID of the area whose default-collapsed state you want to know.
 | 
						|
   * @return `true` or `false` depending on the area, null if the area is unknown,
 | 
						|
   *         or its collapsed state cannot normally be controlled by the user
 | 
						|
   */
 | 
						|
  isToolbarDefaultCollapsed(aArea) {
 | 
						|
    let area = gAreas.get(aArea);
 | 
						|
    return area ? area.get("defaultCollapsed") : null;
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Obtain the DOM node that is the customize target for an area in a
 | 
						|
   * specific window.
 | 
						|
   *
 | 
						|
   * Areas can have a customization target that does not correspond to the
 | 
						|
   * node itself. In particular, toolbars that have a customizationtarget
 | 
						|
   * attribute set will have their customization target set to that node.
 | 
						|
   * This means widgets will end up in the customization target, not in the
 | 
						|
   * DOM node with the ID that corresponds to the area ID. This is useful
 | 
						|
   * because it lets you have fixed content in a toolbar (e.g. the panel
 | 
						|
   * menu item in the navbar) and have all the customizable widgets use
 | 
						|
   * the customization target.
 | 
						|
   *
 | 
						|
   * Using this API yourself is discouraged; you should generally not need
 | 
						|
   * to be asking for the DOM container node used for a particular area.
 | 
						|
   * In particular, if you're wanting to check it in relation to a widget's
 | 
						|
   * node, your DOM node might not be a direct child of the customize target
 | 
						|
   * in a window if, for instance, the window is in customization mode, or if
 | 
						|
   * this is an overflowable toolbar and the widget has been overflowed.
 | 
						|
   *
 | 
						|
   * @param aArea   the ID of the area whose customize target you want to have
 | 
						|
   * @param aWindow the window where you want to fetch the DOM node.
 | 
						|
   * @return the customize target DOM node for aArea in aWindow
 | 
						|
   */
 | 
						|
  getCustomizeTargetForArea(aArea, aWindow) {
 | 
						|
    return CustomizableUIInternal.getCustomizeTargetForArea(aArea, aWindow);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Reset the customization state back to its default.
 | 
						|
   *
 | 
						|
   * This is the nuclear option. You should never call this except if the user
 | 
						|
   * explicitly requests it. Firefox does this when the user clicks the
 | 
						|
   * "Restore Defaults" button in customize mode.
 | 
						|
   */
 | 
						|
  reset() {
 | 
						|
    CustomizableUIInternal.reset();
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Undo the previous reset, can only be called immediately after a reset.
 | 
						|
   * @return a promise that will be resolved when the operation is complete.
 | 
						|
   */
 | 
						|
  undoReset() {
 | 
						|
    CustomizableUIInternal.undoReset();
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Remove a custom toolbar added in a previous version of Firefox or using
 | 
						|
   * an add-on. NB: only works on the customizable toolbars generated by
 | 
						|
   * the toolbox itself. Intended for use from CustomizeMode, not by
 | 
						|
   * other consumers.
 | 
						|
   * @param aToolbarId the ID of the toolbar to remove
 | 
						|
   */
 | 
						|
  removeExtraToolbar(aToolbarId) {
 | 
						|
    CustomizableUIInternal.removeExtraToolbar(aToolbarId);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Can the last Restore Defaults operation be undone.
 | 
						|
   *
 | 
						|
   * @return A boolean stating whether an undo of the
 | 
						|
   *         Restore Defaults can be performed.
 | 
						|
   */
 | 
						|
  get canUndoReset() {
 | 
						|
    return (
 | 
						|
      gUIStateBeforeReset.uiCustomizationState != null ||
 | 
						|
      gUIStateBeforeReset.drawInTitlebar != null ||
 | 
						|
      gUIStateBeforeReset.currentTheme != null ||
 | 
						|
      gUIStateBeforeReset.autoTouchMode != null ||
 | 
						|
      gUIStateBeforeReset.uiDensity != null
 | 
						|
    );
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Get the placement of a widget. This is by far the best way to obtain
 | 
						|
   * information about what the state of your widget is. The internals of
 | 
						|
   * this call are cheap (no DOM necessary) and you will know where the user
 | 
						|
   * has put your widget.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget whose placement you want to know
 | 
						|
   * @return
 | 
						|
   *   {
 | 
						|
   *     area: "somearea", // The ID of the area where the widget is placed
 | 
						|
   *     position: 42 // the index in the placements array corresponding to
 | 
						|
   *                  // your widget.
 | 
						|
   *   }
 | 
						|
   *
 | 
						|
   *   OR
 | 
						|
   *
 | 
						|
   *   null // if the widget is not placed anywhere (ie in the palette)
 | 
						|
   */
 | 
						|
  getPlacementOfWidget(aWidgetId, aOnlyRegistered = true, aDeadAreas = false) {
 | 
						|
    return CustomizableUIInternal.getPlacementOfWidget(
 | 
						|
      aWidgetId,
 | 
						|
      aOnlyRegistered,
 | 
						|
      aDeadAreas
 | 
						|
    );
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check if a widget can be removed from the area it's in.
 | 
						|
   *
 | 
						|
   * Note that if you're wanting to move the widget somewhere, you should
 | 
						|
   * generally be checking canWidgetMoveToArea, because that will return
 | 
						|
   * true if the widget is already in the area where you want to move it (!).
 | 
						|
   *
 | 
						|
   * NB: oh, also, this method might lie if the widget in question is a
 | 
						|
   *     XUL-provided widget and there are no windows open, because it
 | 
						|
   *     can obviously not check anything in this case. It will return
 | 
						|
   *     true. You will be able to move the widget elsewhere. However,
 | 
						|
   *     once the user reopens a window, the widget will move back to its
 | 
						|
   *     'proper' area automagically.
 | 
						|
   *
 | 
						|
   * @param aWidgetId a widget ID or DOM node to check
 | 
						|
   * @return true if the widget can be removed from its area,
 | 
						|
   *          false otherwise.
 | 
						|
   */
 | 
						|
  isWidgetRemovable(aWidgetId) {
 | 
						|
    return CustomizableUIInternal.isWidgetRemovable(aWidgetId);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check if a widget can be moved to a particular area. Like
 | 
						|
   * isWidgetRemovable but better, because it'll return true if the widget
 | 
						|
   * is already in the right area.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the widget ID or DOM node you want to move somewhere
 | 
						|
   * @param aArea     the area ID you want to move it to. This can also be
 | 
						|
   *                  AREA_NO_AREA to see if the widget can move to the
 | 
						|
   *                  customization palette, whether it's removable or not.
 | 
						|
   * @return true if this is possible, false if it is not. The same caveats as
 | 
						|
   *              for isWidgetRemovable apply, however, if no windows are open.
 | 
						|
   */
 | 
						|
  canWidgetMoveToArea(aWidgetId, aArea) {
 | 
						|
    return CustomizableUIInternal.canWidgetMoveToArea(aWidgetId, aArea);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Whether we're in a default state. Note that non-removable non-default
 | 
						|
   * widgets and non-existing widgets are not taken into account in determining
 | 
						|
   * whether we're in the default state.
 | 
						|
   *
 | 
						|
   * NB: this is a property with a getter. The getter is NOT cheap, because
 | 
						|
   * it does smart things with non-removable non-default items, non-existent
 | 
						|
   * items, and so forth. Please don't call unless necessary.
 | 
						|
   */
 | 
						|
  get inDefaultState() {
 | 
						|
    return CustomizableUIInternal.inDefaultState;
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Set a toolbar's visibility state in all windows.
 | 
						|
   * @param aToolbarId    the toolbar whose visibility should be adjusted
 | 
						|
   * @param aIsVisible    whether the toolbar should be visible
 | 
						|
   */
 | 
						|
  setToolbarVisibility(aToolbarId, aIsVisible) {
 | 
						|
    CustomizableUIInternal.setToolbarVisibility(aToolbarId, aIsVisible);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns a Set with the IDs of any registered toolbar areas that are
 | 
						|
   * currently collapsed in a particular window. Menubars that are set to
 | 
						|
   * autohide and are in the temporary "open" state are still considered
 | 
						|
   * collapsed by default.
 | 
						|
   *
 | 
						|
   * @param {Window} window The browser window to check for collapsed toolbars.
 | 
						|
   * @return {Set<string>}
 | 
						|
   */
 | 
						|
  getCollapsedToolbarIds(window) {
 | 
						|
    return CustomizableUIInternal.getCollapsedToolbarIds(window);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * DEPRECATED! Use fluent instead.
 | 
						|
   *
 | 
						|
   * Get a localized property off a (widget?) object.
 | 
						|
   *
 | 
						|
   * NB: this is unlikely to be useful unless you're in Firefox code, because
 | 
						|
   *     this code uses the builtin widget stringbundle, and can't be told
 | 
						|
   *     to use add-on-provided strings. It's mainly here as convenience for
 | 
						|
   *     custom builtin widgets that build their own DOM but use the same
 | 
						|
   *     stringbundle as the other builtin widgets.
 | 
						|
   *
 | 
						|
   * @param aWidget     the object whose property we should use to fetch a
 | 
						|
   *                    localizable string;
 | 
						|
   * @param aProp       the property on the object to use for the fetching;
 | 
						|
   * @param aFormatArgs (optional) any extra arguments to use for a formatted
 | 
						|
   *                    string;
 | 
						|
   * @param aDef        (optional) the default to return if we don't find the
 | 
						|
   *                    string in the stringbundle;
 | 
						|
   *
 | 
						|
   * @return the localized string, or aDef if the string isn't in the bundle.
 | 
						|
   *         If no default is provided,
 | 
						|
   *           if aProp exists on aWidget, we'll return that,
 | 
						|
   *           otherwise we'll return the empty string
 | 
						|
   *
 | 
						|
   */
 | 
						|
  getLocalizedProperty(aWidget, aProp, aFormatArgs, aDef) {
 | 
						|
    return CustomizableUIInternal.getLocalizedProperty(
 | 
						|
      aWidget,
 | 
						|
      aProp,
 | 
						|
      aFormatArgs,
 | 
						|
      aDef
 | 
						|
    );
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Utility function to detect, find and set a keyboard shortcut for a menuitem
 | 
						|
   * or (toolbar)button.
 | 
						|
   *
 | 
						|
   * @param aShortcutNode the XUL node where the shortcut will be derived from;
 | 
						|
   * @param aTargetNode   (optional) the XUL node on which the `shortcut`
 | 
						|
   *                      attribute will be set. If NULL, the shortcut will be
 | 
						|
   *                      set on aShortcutNode;
 | 
						|
   */
 | 
						|
  addShortcut(aShortcutNode, aTargetNode) {
 | 
						|
    return CustomizableUIInternal.addShortcut(aShortcutNode, aTargetNode);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Given a node, walk up to the first panel in its ancestor chain, and
 | 
						|
   * close it.
 | 
						|
   *
 | 
						|
   * @param aNode a node whose panel should be closed;
 | 
						|
   */
 | 
						|
  hidePanelForNode(aNode) {
 | 
						|
    CustomizableUIInternal.hidePanelForNode(aNode);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check if a widget is a "special" widget: a spring, spacer or separator.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the widget ID to check.
 | 
						|
   * @return true if the widget is 'special', false otherwise.
 | 
						|
   */
 | 
						|
  isSpecialWidget(aWidgetId) {
 | 
						|
    return CustomizableUIInternal.isSpecialWidget(aWidgetId);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Check if a widget is provided by an extension. This effectively checks
 | 
						|
   * whether `webExtension: true` passed when the widget was being created.
 | 
						|
   *
 | 
						|
   * If the widget being referred to hasn't yet been created, or has been
 | 
						|
   * destroyed, we fallback to checking the ID for the "-browser-action"
 | 
						|
   * suffix.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the widget ID to check.
 | 
						|
   * @return true if the widget was provided by an extension, false otherwise.
 | 
						|
   */
 | 
						|
  isWebExtensionWidget(aWidgetId) {
 | 
						|
    let widget = CustomizableUI.getWidget(aWidgetId);
 | 
						|
    return widget?.webExtension || aWidgetId.endsWith("-browser-action");
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Add listeners to a panel that will close it. For use from the menu panel
 | 
						|
   * and overflowable toolbar implementations, unlikely to be useful for
 | 
						|
   * consumers.
 | 
						|
   *
 | 
						|
   * @param aPanel the panel to which listeners should be attached.
 | 
						|
   */
 | 
						|
  addPanelCloseListeners(aPanel) {
 | 
						|
    CustomizableUIInternal.addPanelCloseListeners(aPanel);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Remove close listeners that have been added to a panel with
 | 
						|
   * addPanelCloseListeners. For use from the menu panel and overflowable
 | 
						|
   * toolbar implementations, unlikely to be useful for consumers.
 | 
						|
   *
 | 
						|
   * @param aPanel the panel from which listeners should be removed.
 | 
						|
   */
 | 
						|
  removePanelCloseListeners(aPanel) {
 | 
						|
    CustomizableUIInternal.removePanelCloseListeners(aPanel);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Notify listeners a widget is about to be dragged to an area. For use from
 | 
						|
   * Customize Mode only, do not use otherwise.
 | 
						|
   *
 | 
						|
   * @param aWidgetId the ID of the widget that is being dragged to an area.
 | 
						|
   * @param aArea     the ID of the area to which the widget is being dragged.
 | 
						|
   */
 | 
						|
  onWidgetDrag(aWidgetId, aArea) {
 | 
						|
    CustomizableUIInternal.notifyListeners("onWidgetDrag", aWidgetId, aArea);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Notify listeners that a window is entering customize mode. For use from
 | 
						|
   * Customize Mode only, do not use otherwise.
 | 
						|
   * @param aWindow the window entering customize mode
 | 
						|
   */
 | 
						|
  notifyStartCustomizing(aWindow) {
 | 
						|
    CustomizableUIInternal.notifyListeners("onCustomizeStart", aWindow);
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Notify listeners that a window is exiting customize mode. For use from
 | 
						|
   * Customize Mode only, do not use otherwise.
 | 
						|
   * @param aWindow the window exiting customize mode
 | 
						|
   */
 | 
						|
  notifyEndCustomizing(aWindow) {
 | 
						|
    CustomizableUIInternal.notifyListeners("onCustomizeEnd", aWindow);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Notify toolbox(es) of a particular event. If you don't pass aWindow,
 | 
						|
   * all toolboxes will be notified. For use from Customize Mode only,
 | 
						|
   * do not use otherwise.
 | 
						|
   * @param aEvent the name of the event to send.
 | 
						|
   * @param aDetails optional, the details of the event.
 | 
						|
   * @param aWindow optional, the window in which to send the event.
 | 
						|
   */
 | 
						|
  dispatchToolboxEvent(aEvent, aDetails = {}, aWindow = null) {
 | 
						|
    CustomizableUIInternal.dispatchToolboxEvent(aEvent, aDetails, aWindow);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Check whether an area is overflowable.
 | 
						|
   *
 | 
						|
   * @param aAreaId the ID of an area to check for overflowable-ness
 | 
						|
   * @return true if the area is overflowable, false otherwise.
 | 
						|
   */
 | 
						|
  isAreaOverflowable(aAreaId) {
 | 
						|
    let area = gAreas.get(aAreaId);
 | 
						|
    return area
 | 
						|
      ? area.get("type") == this.TYPE_TOOLBAR && area.get("overflowable")
 | 
						|
      : false;
 | 
						|
  },
 | 
						|
  /**
 | 
						|
   * Obtain a string indicating the place of an element. This is intended
 | 
						|
   * for use from customize mode; You should generally use getPlacementOfWidget
 | 
						|
   * instead, which is cheaper because it does not use the DOM.
 | 
						|
   *
 | 
						|
   * @param aElement the DOM node whose place we need to check
 | 
						|
   * @return "toolbar" if the node is in a toolbar, "panel" if it is in the
 | 
						|
   *         menu panel, "palette" if it is in the (visible!) customization
 | 
						|
   *         palette, undefined otherwise.
 | 
						|
   */
 | 
						|
  getPlaceForItem(aElement) {
 | 
						|
    let place;
 | 
						|
    let node = aElement;
 | 
						|
    while (node && !place) {
 | 
						|
      if (node.localName == "toolbar") {
 | 
						|
        place = "toolbar";
 | 
						|
      } else if (node.id == CustomizableUI.AREA_FIXED_OVERFLOW_PANEL) {
 | 
						|
        place = "panel";
 | 
						|
      } else if (node.id == "customization-palette") {
 | 
						|
        place = "palette";
 | 
						|
      }
 | 
						|
 | 
						|
      node = node.parentNode;
 | 
						|
    }
 | 
						|
    return place;
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Check if a toolbar is builtin or not.
 | 
						|
   * @param aToolbarId the ID of the toolbar you want to check
 | 
						|
   */
 | 
						|
  isBuiltinToolbar(aToolbarId) {
 | 
						|
    return CustomizableUIInternal._builtinToolbars.has(aToolbarId);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Create an instance of a spring, spacer or separator.
 | 
						|
   * @param aId       the type of special widget (spring, spacer or separator)
 | 
						|
   * @param aDocument the document in which to create it.
 | 
						|
   */
 | 
						|
  createSpecialWidget(aId, aDocument) {
 | 
						|
    return CustomizableUIInternal.createSpecialWidget(aId, aDocument);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * Fills a submenu with menu items.
 | 
						|
   * @param aMenuItems the menu items to display.
 | 
						|
   * @param aSubview   the subview to fill.
 | 
						|
   */
 | 
						|
  fillSubviewFromMenuItems(aMenuItems, aSubview) {
 | 
						|
    let attrs = [
 | 
						|
      "oncommand",
 | 
						|
      "onclick",
 | 
						|
      "label",
 | 
						|
      "key",
 | 
						|
      "disabled",
 | 
						|
      "command",
 | 
						|
      "observes",
 | 
						|
      "hidden",
 | 
						|
      "class",
 | 
						|
      "origin",
 | 
						|
      "image",
 | 
						|
      "checked",
 | 
						|
      "style",
 | 
						|
    ];
 | 
						|
 | 
						|
    // Use ownerGlobal.document to ensure we get the right doc even for
 | 
						|
    // elements in template tags.
 | 
						|
    let doc = aSubview.ownerGlobal.document;
 | 
						|
    let fragment = doc.createDocumentFragment();
 | 
						|
    for (let menuChild of aMenuItems) {
 | 
						|
      if (menuChild.hidden) {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
 | 
						|
      let subviewItem;
 | 
						|
      if (menuChild.localName == "menuseparator") {
 | 
						|
        // Don't insert duplicate or leading separators. This can happen if there are
 | 
						|
        // menus (which we don't copy) above the separator.
 | 
						|
        if (
 | 
						|
          !fragment.lastElementChild ||
 | 
						|
          fragment.lastElementChild.localName == "toolbarseparator"
 | 
						|
        ) {
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
        subviewItem = doc.createXULElement("toolbarseparator");
 | 
						|
      } else if (menuChild.localName == "menuitem") {
 | 
						|
        subviewItem = doc.createXULElement("toolbarbutton");
 | 
						|
        CustomizableUI.addShortcut(menuChild, subviewItem);
 | 
						|
 | 
						|
        let item = menuChild;
 | 
						|
        if (!item.hasAttribute("onclick")) {
 | 
						|
          subviewItem.addEventListener("click", event => {
 | 
						|
            let newEvent = new doc.defaultView.MouseEvent(event.type, event);
 | 
						|
 | 
						|
            // Telemetry should only pay attention to the original event.
 | 
						|
            lazy.BrowserUsageTelemetry.ignoreEvent(newEvent);
 | 
						|
            item.dispatchEvent(newEvent);
 | 
						|
          });
 | 
						|
        }
 | 
						|
 | 
						|
        if (!item.hasAttribute("oncommand")) {
 | 
						|
          subviewItem.addEventListener("command", event => {
 | 
						|
            let newEvent = doc.createEvent("XULCommandEvent");
 | 
						|
            newEvent.initCommandEvent(
 | 
						|
              event.type,
 | 
						|
              event.bubbles,
 | 
						|
              event.cancelable,
 | 
						|
              event.view,
 | 
						|
              event.detail,
 | 
						|
              event.ctrlKey,
 | 
						|
              event.altKey,
 | 
						|
              event.shiftKey,
 | 
						|
              event.metaKey,
 | 
						|
              0,
 | 
						|
              event.sourceEvent,
 | 
						|
              0
 | 
						|
            );
 | 
						|
 | 
						|
            // Telemetry should only pay attention to the original event.
 | 
						|
            lazy.BrowserUsageTelemetry.ignoreEvent(newEvent);
 | 
						|
            item.dispatchEvent(newEvent);
 | 
						|
          });
 | 
						|
        }
 | 
						|
      } else {
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
      for (let attr of attrs) {
 | 
						|
        let attrVal = menuChild.getAttribute(attr);
 | 
						|
        if (attrVal) {
 | 
						|
          subviewItem.setAttribute(attr, attrVal);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      // We do this after so the .subviewbutton class doesn't get overriden.
 | 
						|
      if (menuChild.localName == "menuitem") {
 | 
						|
        subviewItem.classList.add("subviewbutton");
 | 
						|
      }
 | 
						|
 | 
						|
      // We make it possible to supply an alternative Fluent key when cloning
 | 
						|
      // this menuitem into the AppMenu or panel contexts. This is because
 | 
						|
      // we often use Title Case in menuitems in native menus, but want to use
 | 
						|
      // Sentence case in the AppMenu / panels.
 | 
						|
      let l10nId = menuChild.getAttribute("appmenu-data-l10n-id");
 | 
						|
      if (l10nId) {
 | 
						|
        doc.l10n.setAttributes(subviewItem, l10nId);
 | 
						|
      }
 | 
						|
 | 
						|
      fragment.appendChild(subviewItem);
 | 
						|
    }
 | 
						|
    aSubview.appendChild(fragment);
 | 
						|
  },
 | 
						|
 | 
						|
  /**
 | 
						|
   * A helper function for clearing subviews.
 | 
						|
   * @param aSubview the subview to clear.
 | 
						|
   */
 | 
						|
  clearSubview(aSubview) {
 | 
						|
    let parent = aSubview.parentNode;
 | 
						|
    // We'll take the container out of the document before cleaning it out
 | 
						|
    // to avoid reflowing each time we remove something.
 | 
						|
    parent.removeChild(aSubview);
 | 
						|
 | 
						|
    while (aSubview.firstChild) {
 | 
						|
      aSubview.firstChild.remove();
 | 
						|
    }
 | 
						|
 | 
						|
    parent.appendChild(aSubview);
 | 
						|
  },
 | 
						|
 | 
						|
  getCustomizationTarget(aElement) {
 | 
						|
    return CustomizableUIInternal.getCustomizationTarget(aElement);
 | 
						|
  },
 | 
						|
 | 
						|
  getTestOnlyInternalProp(aProp) {
 | 
						|
    if (!Cu.isInAutomation) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
    switch (aProp) {
 | 
						|
      case "CustomizableUIInternal":
 | 
						|
        return CustomizableUIInternal;
 | 
						|
      case "gAreas":
 | 
						|
        return gAreas;
 | 
						|
      case "gFuturePlacements":
 | 
						|
        return gFuturePlacements;
 | 
						|
      case "gPalette":
 | 
						|
        return gPalette;
 | 
						|
      case "gPlacements":
 | 
						|
        return gPlacements;
 | 
						|
      case "gSavedState":
 | 
						|
        return gSavedState;
 | 
						|
      case "gSeenWidgets":
 | 
						|
        return gSeenWidgets;
 | 
						|
      case "kVersion":
 | 
						|
        return kVersion;
 | 
						|
    }
 | 
						|
    return null;
 | 
						|
  },
 | 
						|
  setTestOnlyInternalProp(aProp, aValue) {
 | 
						|
    if (!Cu.isInAutomation) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    switch (aProp) {
 | 
						|
      case "gSavedState":
 | 
						|
        gSavedState = aValue;
 | 
						|
        break;
 | 
						|
      case "kVersion":
 | 
						|
        kVersion = aValue;
 | 
						|
        break;
 | 
						|
      case "gDirty":
 | 
						|
        gDirty = aValue;
 | 
						|
        break;
 | 
						|
    }
 | 
						|
  },
 | 
						|
};
 | 
						|
 | 
						|
Object.freeze(CustomizableUI);
 | 
						|
Object.freeze(CustomizableUI.windows);
 | 
						|
 | 
						|
/**
 | 
						|
 * All external consumers of widgets are really interacting with these wrappers
 | 
						|
 * which provide a common interface.
 | 
						|
 */
 | 
						|
 | 
						|
/**
 | 
						|
 * WidgetGroupWrapper is the common interface for interacting with an entire
 | 
						|
 * widget group - AKA, all instances of a widget across a series of windows.
 | 
						|
 * This particular wrapper is only used for widgets created via the provider
 | 
						|
 * API.
 | 
						|
 */
 | 
						|
function WidgetGroupWrapper(aWidget) {
 | 
						|
  this.isGroup = true;
 | 
						|
 | 
						|
  const kBareProps = [
 | 
						|
    "id",
 | 
						|
    "source",
 | 
						|
    "type",
 | 
						|
    "disabled",
 | 
						|
    "label",
 | 
						|
    "tooltiptext",
 | 
						|
    "showInPrivateBrowsing",
 | 
						|
    "viewId",
 | 
						|
    "disallowSubView",
 | 
						|
    "webExtension",
 | 
						|
  ];
 | 
						|
  for (let prop of kBareProps) {
 | 
						|
    let propertyName = prop;
 | 
						|
    this.__defineGetter__(propertyName, () => aWidget[propertyName]);
 | 
						|
  }
 | 
						|
 | 
						|
  this.__defineGetter__("provider", () => CustomizableUI.PROVIDER_API);
 | 
						|
 | 
						|
  this.__defineSetter__("disabled", function (aValue) {
 | 
						|
    aValue = !!aValue;
 | 
						|
    aWidget.disabled = aValue;
 | 
						|
    for (let [, instance] of aWidget.instances) {
 | 
						|
      instance.disabled = aValue;
 | 
						|
    }
 | 
						|
  });
 | 
						|
 | 
						|
  this.forWindow = function WidgetGroupWrapper_forWindow(aWindow) {
 | 
						|
    let wrapperMap;
 | 
						|
    if (!gSingleWrapperCache.has(aWindow)) {
 | 
						|
      wrapperMap = new Map();
 | 
						|
      gSingleWrapperCache.set(aWindow, wrapperMap);
 | 
						|
    } else {
 | 
						|
      wrapperMap = gSingleWrapperCache.get(aWindow);
 | 
						|
    }
 | 
						|
    if (wrapperMap.has(aWidget.id)) {
 | 
						|
      return wrapperMap.get(aWidget.id);
 | 
						|
    }
 | 
						|
 | 
						|
    let instance = aWidget.instances.get(aWindow.document);
 | 
						|
    if (!instance) {
 | 
						|
      instance = CustomizableUIInternal.buildWidget(aWindow.document, aWidget);
 | 
						|
    }
 | 
						|
 | 
						|
    let wrapper = new WidgetSingleWrapper(aWidget, instance);
 | 
						|
    wrapperMap.set(aWidget.id, wrapper);
 | 
						|
    return wrapper;
 | 
						|
  };
 | 
						|
 | 
						|
  this.__defineGetter__("instances", function () {
 | 
						|
    // Can't use gBuildWindows here because some areas load lazily:
 | 
						|
    let placement = CustomizableUIInternal.getPlacementOfWidget(aWidget.id);
 | 
						|
    if (!placement) {
 | 
						|
      return [];
 | 
						|
    }
 | 
						|
    let area = placement.area;
 | 
						|
    let buildAreas = gBuildAreas.get(area);
 | 
						|
    if (!buildAreas) {
 | 
						|
      return [];
 | 
						|
    }
 | 
						|
    return Array.from(buildAreas, node => this.forWindow(node.ownerGlobal));
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("areaType", function () {
 | 
						|
    let areaProps = gAreas.get(aWidget.currentArea);
 | 
						|
    return areaProps && areaProps.get("type");
 | 
						|
  });
 | 
						|
 | 
						|
  Object.freeze(this);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * A WidgetSingleWrapper is a wrapper around a single instance of a widget in
 | 
						|
 * a particular window.
 | 
						|
 */
 | 
						|
function WidgetSingleWrapper(aWidget, aNode) {
 | 
						|
  this.isGroup = false;
 | 
						|
 | 
						|
  this.node = aNode;
 | 
						|
  this.provider = CustomizableUI.PROVIDER_API;
 | 
						|
 | 
						|
  const kGlobalProps = ["id", "type"];
 | 
						|
  for (let prop of kGlobalProps) {
 | 
						|
    this[prop] = aWidget[prop];
 | 
						|
  }
 | 
						|
 | 
						|
  const kNodeProps = ["label", "tooltiptext"];
 | 
						|
  for (let prop of kNodeProps) {
 | 
						|
    let propertyName = prop;
 | 
						|
    // Look at the node for these, instead of the widget data, to ensure the
 | 
						|
    // wrapper always reflects this live instance.
 | 
						|
    this.__defineGetter__(propertyName, () => aNode.getAttribute(propertyName));
 | 
						|
  }
 | 
						|
 | 
						|
  this.__defineGetter__("disabled", () => aNode.disabled);
 | 
						|
  this.__defineSetter__("disabled", function (aValue) {
 | 
						|
    aNode.disabled = !!aValue;
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("anchor", function () {
 | 
						|
    let anchorId;
 | 
						|
    // First check for an anchor for the area:
 | 
						|
    let placement = CustomizableUIInternal.getPlacementOfWidget(aWidget.id);
 | 
						|
    if (placement) {
 | 
						|
      anchorId = gAreas.get(placement.area).get("anchor");
 | 
						|
    }
 | 
						|
    if (!anchorId) {
 | 
						|
      anchorId = aNode.getAttribute("cui-anchorid");
 | 
						|
    }
 | 
						|
    if (!anchorId) {
 | 
						|
      anchorId = aNode.getAttribute("view-button-id");
 | 
						|
    }
 | 
						|
    if (anchorId) {
 | 
						|
      return aNode.ownerDocument.getElementById(anchorId);
 | 
						|
    }
 | 
						|
    if (aWidget.type == "button-and-view") {
 | 
						|
      return aNode.lastElementChild;
 | 
						|
    }
 | 
						|
    return aNode;
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("overflowed", function () {
 | 
						|
    return aNode.getAttribute("overflowedItem") == "true";
 | 
						|
  });
 | 
						|
 | 
						|
  Object.freeze(this);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * XULWidgetGroupWrapper is the common interface for interacting with an entire
 | 
						|
 * widget group - AKA, all instances of a widget across a series of windows.
 | 
						|
 * This particular wrapper is only used for widgets created via the old-school
 | 
						|
 * XUL method (overlays, or programmatically injecting toolbaritems, or other
 | 
						|
 * such things).
 | 
						|
 */
 | 
						|
// XXXunf Going to need to hook this up to some events to keep it all live.
 | 
						|
function XULWidgetGroupWrapper(aWidgetId) {
 | 
						|
  this.isGroup = true;
 | 
						|
  this.id = aWidgetId;
 | 
						|
  this.type = "custom";
 | 
						|
  // XUL Widgets can never be provided by extensions.
 | 
						|
  this.webExtension = false;
 | 
						|
  this.provider = CustomizableUI.PROVIDER_XUL;
 | 
						|
 | 
						|
  this.forWindow = function XULWidgetGroupWrapper_forWindow(aWindow) {
 | 
						|
    let wrapperMap;
 | 
						|
    if (!gSingleWrapperCache.has(aWindow)) {
 | 
						|
      wrapperMap = new Map();
 | 
						|
      gSingleWrapperCache.set(aWindow, wrapperMap);
 | 
						|
    } else {
 | 
						|
      wrapperMap = gSingleWrapperCache.get(aWindow);
 | 
						|
    }
 | 
						|
    if (wrapperMap.has(aWidgetId)) {
 | 
						|
      return wrapperMap.get(aWidgetId);
 | 
						|
    }
 | 
						|
 | 
						|
    let instance = aWindow.document.getElementById(aWidgetId);
 | 
						|
    if (!instance) {
 | 
						|
      // Toolbar palettes aren't part of the document, so elements in there
 | 
						|
      // won't be found via document.getElementById().
 | 
						|
      instance = aWindow.gNavToolbox.palette.getElementsByAttribute(
 | 
						|
        "id",
 | 
						|
        aWidgetId
 | 
						|
      )[0];
 | 
						|
    }
 | 
						|
 | 
						|
    let wrapper = new XULWidgetSingleWrapper(
 | 
						|
      aWidgetId,
 | 
						|
      instance,
 | 
						|
      aWindow.document
 | 
						|
    );
 | 
						|
    wrapperMap.set(aWidgetId, wrapper);
 | 
						|
    return wrapper;
 | 
						|
  };
 | 
						|
 | 
						|
  this.__defineGetter__("areaType", function () {
 | 
						|
    let placement = CustomizableUIInternal.getPlacementOfWidget(aWidgetId);
 | 
						|
    if (!placement) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
 | 
						|
    let areaProps = gAreas.get(placement.area);
 | 
						|
    return areaProps && areaProps.get("type");
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("instances", function () {
 | 
						|
    return Array.from(gBuildWindows, wins => this.forWindow(wins[0]));
 | 
						|
  });
 | 
						|
 | 
						|
  Object.freeze(this);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * A XULWidgetSingleWrapper is a wrapper around a single instance of a XUL
 | 
						|
 * widget in a particular window.
 | 
						|
 */
 | 
						|
function XULWidgetSingleWrapper(aWidgetId, aNode, aDocument) {
 | 
						|
  this.isGroup = false;
 | 
						|
 | 
						|
  this.id = aWidgetId;
 | 
						|
  this.type = "custom";
 | 
						|
  this.provider = CustomizableUI.PROVIDER_XUL;
 | 
						|
 | 
						|
  let weakDoc = Cu.getWeakReference(aDocument);
 | 
						|
  // If we keep a strong ref, the weak ref will never die, so null it out:
 | 
						|
  aDocument = null;
 | 
						|
 | 
						|
  this.__defineGetter__("node", function () {
 | 
						|
    // If we've set this to null (further down), we're sure there's nothing to
 | 
						|
    // be gotten here, so bail out early:
 | 
						|
    if (!weakDoc) {
 | 
						|
      return null;
 | 
						|
    }
 | 
						|
    if (aNode) {
 | 
						|
      // Return the last known node if it's still in the DOM...
 | 
						|
      if (aNode.isConnected) {
 | 
						|
        return aNode;
 | 
						|
      }
 | 
						|
      // ... or the toolbox
 | 
						|
      let toolbox = aNode.ownerGlobal.gNavToolbox;
 | 
						|
      if (toolbox && toolbox.palette && aNode.parentNode == toolbox.palette) {
 | 
						|
        return aNode;
 | 
						|
      }
 | 
						|
      // If it isn't, clear the cached value and fall through to the "slow" case:
 | 
						|
      aNode = null;
 | 
						|
    }
 | 
						|
 | 
						|
    let doc = weakDoc.get();
 | 
						|
    if (doc) {
 | 
						|
      // Store locally so we can cache the result:
 | 
						|
      aNode = CustomizableUIInternal.findWidgetInWindow(
 | 
						|
        aWidgetId,
 | 
						|
        doc.defaultView
 | 
						|
      );
 | 
						|
      return aNode;
 | 
						|
    }
 | 
						|
    // The weakref to the document is dead, we're done here forever more:
 | 
						|
    weakDoc = null;
 | 
						|
    return null;
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("anchor", function () {
 | 
						|
    let anchorId;
 | 
						|
    // First check for an anchor for the area:
 | 
						|
    let placement = CustomizableUIInternal.getPlacementOfWidget(aWidgetId);
 | 
						|
    if (placement) {
 | 
						|
      anchorId = gAreas.get(placement.area).get("anchor");
 | 
						|
    }
 | 
						|
 | 
						|
    let node = this.node;
 | 
						|
    if (!anchorId && node) {
 | 
						|
      anchorId = node.getAttribute("cui-anchorid");
 | 
						|
    }
 | 
						|
 | 
						|
    return anchorId && node
 | 
						|
      ? node.ownerDocument.getElementById(anchorId)
 | 
						|
      : node;
 | 
						|
  });
 | 
						|
 | 
						|
  this.__defineGetter__("overflowed", function () {
 | 
						|
    let node = this.node;
 | 
						|
    if (!node) {
 | 
						|
      return false;
 | 
						|
    }
 | 
						|
    return node.getAttribute("overflowedItem") == "true";
 | 
						|
  });
 | 
						|
 | 
						|
  Object.freeze(this);
 | 
						|
}
 | 
						|
 | 
						|
/**
 | 
						|
 * OverflowableToolbar is a class that gives a <xul:toolbar> the ability to send
 | 
						|
 * toolbar items that are "overflowable" to lists in separate panels if and
 | 
						|
 * when the toolbar shrinks enough so that those items overflow out of bounds.
 | 
						|
 * Secondly, this class manages moving things out from those panels and back
 | 
						|
 * into the toolbar once it underflows and has the space to accommodate the
 | 
						|
 * items that had originally overflowed out.
 | 
						|
 *
 | 
						|
 * There are two panels that toolbar items can be overflowed to:
 | 
						|
 *
 | 
						|
 * 1. The default items overflow panel
 | 
						|
 *   This is where built-in default toolbar items will go to.
 | 
						|
 * 2. The Unified Extensions panel
 | 
						|
 *   This is where browser_action toolbar buttons created by extensions will
 | 
						|
 *   go to if the Unified Extensions UI is enabled - otherwise, those items will
 | 
						|
 *   go to the default items overflow panel.
 | 
						|
 *
 | 
						|
 * Finally, OverflowableToolbar manages the showing of the default items
 | 
						|
 * overflow panel when the associated anchor is clicked or dragged over. The
 | 
						|
 * Unified Extensions panel is managed separately by the extension code.
 | 
						|
 *
 | 
						|
 * In theory, we could have multiple overflowable toolbars, but in practice,
 | 
						|
 * only the nav-bar (CustomizableUI.AREA_NAVBAR) makes use of this class.
 | 
						|
 */
 | 
						|
class OverflowableToolbar {
 | 
						|
  /**
 | 
						|
   * The OverflowableToolbar class is constructed during browser window
 | 
						|
   * creation, but to optimize for window painting, we defer most work until
 | 
						|
   * after the window has painted. This property is set to true once
 | 
						|
   * initialization has completed.
 | 
						|
   *
 | 
						|
   * @type {boolean}
 | 
						|
   */
 | 
						|
  #initialized = false;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the <xul:toolbar> that is overflowable.
 | 
						|
   *
 | 
						|
   * @type {Element}
 | 
						|
   */
 | 
						|
  #toolbar = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the part of the <xul:toolbar> that accepts CustomizableUI
 | 
						|
   * widgets.
 | 
						|
   *
 | 
						|
   * @type {Element}
 | 
						|
   */
 | 
						|
  #target = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A mapping from the ID of a toolbar item that has overflowed to the width
 | 
						|
   * that the toolbar item occupied in the toolbar at the time of overflow. Any
 | 
						|
   * item that is currently overflowed will have an entry in this map.
 | 
						|
   *
 | 
						|
   * @type {Map<string, number>}
 | 
						|
   */
 | 
						|
  #overflowedInfo = new Map();
 | 
						|
 | 
						|
  /**
 | 
						|
   * The set of overflowed DOM nodes that were hidden at the time of overflowing.
 | 
						|
   */
 | 
						|
  #hiddenOverflowedNodes = new WeakSet();
 | 
						|
 | 
						|
  /**
 | 
						|
   * True if the overflowable toolbar is actively handling overflows and
 | 
						|
   * underflows. This value is set internally by the private #enable() and
 | 
						|
   * #disable() methods.
 | 
						|
   *
 | 
						|
   * @type {boolean}
 | 
						|
   */
 | 
						|
  #enabled = true;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the element that overflowed toolbar items will be
 | 
						|
   * appended to as children upon overflow.
 | 
						|
   *
 | 
						|
   * @type {Element}
 | 
						|
   */
 | 
						|
  #defaultList = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the button that opens the overflow panel. This is also
 | 
						|
   * the element that the panel will anchor to.
 | 
						|
   *
 | 
						|
   * @type {Element}
 | 
						|
   */
 | 
						|
  #defaultListButton = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the <xul:panel> overflow panel that contains the #defaultList
 | 
						|
   * element.
 | 
						|
   *
 | 
						|
   * @type {Element}
 | 
						|
   */
 | 
						|
  #defaultListPanel = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A reference to the the element that overflowed extension browser action
 | 
						|
   * toolbar items will be appended to as children upon overflow if the
 | 
						|
   * Unified Extension UI is enabled. This is created lazily and might be null,
 | 
						|
   * so you should use the #webExtList memoizing getter instead to get this.
 | 
						|
   *
 | 
						|
   * @type {Element|null}
 | 
						|
   */
 | 
						|
  #webExtListRef = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * An empty object that is created in #checkOverflow to identify individual
 | 
						|
   * calls to #checkOverflow and avoid re-entrancy (since #checkOverflow is
 | 
						|
   * asynchronous, and in theory, could be called multiple times before any of
 | 
						|
   * those times have a chance to fully exit).
 | 
						|
   *
 | 
						|
   * @type {Object}
 | 
						|
   */
 | 
						|
  #checkOverflowHandle = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * A timeout ID returned by setTimeout that identifies a timeout function that
 | 
						|
   * runs to hide the #defaultListPanel if the user happened to open the panel by dragging
 | 
						|
   * over the #defaultListButton and then didn't hover any part of the #defaultListPanel.
 | 
						|
   *
 | 
						|
   * @type {number}
 | 
						|
   */
 | 
						|
  #hideTimeoutId = null;
 | 
						|
 | 
						|
  /**
 | 
						|
   * Public methods start here.
 | 
						|
   */
 | 
						|
 | 
						|
  /**
 | 
						|
   * OverflowableToolbar constructor. This is run very early on in the lifecycle
 | 
						|
   * of a browser window, so it tries to defer most work to the init() method
 | 
						|
   * instead after first paint.
 | 
						|
   *
 | 
						|
   * Upon construction, a "overflowable" attribute will be set on the
 | 
						|
   * toolbar, set to the value of "true".
 | 
						|
   *
 | 
						|
   * Part of the API for OverflowableToolbar is declarative, in that it expects
 | 
						|
   * certain attributes to be set on the <xul:toolbar> that is overflowable.
 | 
						|
   * Those attributes are:
 | 
						|
   *
 | 
						|
   * default-overflowbutton:
 | 
						|
   *   The ID of the button that is used to open and anchor the overflow panel.
 | 
						|
   * default-overflowtarget:
 | 
						|
   *   The ID of the element that overflowed items will be appended to as
 | 
						|
   *   children. Note that the overflowed toolbar items are moved into and out
 | 
						|
   *   of this overflow target, so it is definitely advisable to let
 | 
						|
   *   OverflowableToolbar own managing the children of default-overflowtarget,
 | 
						|
   *   and to not modify it outside of this class.
 | 
						|
   * default-overflowpanel:
 | 
						|
   *   The ID of the <xul:panel> that contains the default-overflowtarget.
 | 
						|
   * addon-webext-overflowbutton:
 | 
						|
   *   The ID of the button that is used to open and anchor the Unified
 | 
						|
   *   Extensions panel.
 | 
						|
   * addon-webext-overflowtarget:
 | 
						|
   *   The ID of the element that overflowed extension toolbar buttons will
 | 
						|
   *   be appended to as children if the Unified Extensions UI is enabled.
 | 
						|
   *   Note that the overflowed toolbar items are moved into and out of this
 | 
						|
   *   overflow target, so it is definitely advisable to let OverflowableToolbar
 | 
						|
   *   own managing the children of addon-webext-overflowtarget, and to not
 | 
						|
   *   modify it outside of this class.
 | 
						|
   *
 | 
						|
   * @param {Element} aToolbarNode The <xul:toolbar> that will be overflowable.
 | 
						|
   * @throws {Error} Throws if the customization target of the toolbar somehow
 | 
						|
   *   isn't a direct descendent of the toolbar.
 | 
						|
   */
 | 
						|
  constructor(aToolbarNode) {
 | 
						|
    this.#toolbar = aToolbarNode;
 | 
						|
    this.#target = CustomizableUI.getCustomizationTarget(this.#toolbar);
 | 
						|
    if (this.#target.parentNode != this.#toolbar) {
 | 
						|
      throw new Error(
 | 
						|
        "Customization target must be a direct child of an overflowable toolbar."
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    this.#toolbar.setAttribute("overflowable", "true");
 | 
						|
    let doc = this.#toolbar.ownerDocument;
 | 
						|
    this.#defaultList = doc.getElementById(
 | 
						|
      this.#toolbar.getAttribute("default-overflowtarget")
 | 
						|
    );
 | 
						|
    this.#defaultList._customizationTarget = this.#defaultList;
 | 
						|
 | 
						|
    let window = this.#toolbar.ownerGlobal;
 | 
						|
 | 
						|
    if (window.gBrowserInit.delayedStartupFinished) {
 | 
						|
      this.init();
 | 
						|
    } else {
 | 
						|
      Services.obs.addObserver(this, "browser-delayed-startup-finished");
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Does final initialization of the OverflowableToolbar after the window has
 | 
						|
   * first painted. This will also kick off the first check to see if overflow
 | 
						|
   * has already occurred at the time of initialization.
 | 
						|
   */
 | 
						|
  init() {
 | 
						|
    let doc = this.#toolbar.ownerDocument;
 | 
						|
    let window = doc.defaultView;
 | 
						|
    window.addEventListener("resize", this);
 | 
						|
    window.gNavToolbox.addEventListener("customizationstarting", this);
 | 
						|
    window.gNavToolbox.addEventListener("aftercustomization", this);
 | 
						|
 | 
						|
    let defaultListButton = this.#toolbar.getAttribute(
 | 
						|
      "default-overflowbutton"
 | 
						|
    );
 | 
						|
    this.#defaultListButton = doc.getElementById(defaultListButton);
 | 
						|
    this.#defaultListButton.addEventListener("mousedown", this);
 | 
						|
    this.#defaultListButton.addEventListener("keypress", this);
 | 
						|
    this.#defaultListButton.addEventListener("dragover", this);
 | 
						|
    this.#defaultListButton.addEventListener("dragend", this);
 | 
						|
 | 
						|
    let panelId = this.#toolbar.getAttribute("default-overflowpanel");
 | 
						|
    this.#defaultListPanel = doc.getElementById(panelId);
 | 
						|
    this.#defaultListPanel.addEventListener("popuphiding", this);
 | 
						|
    CustomizableUIInternal.addPanelCloseListeners(this.#defaultListPanel);
 | 
						|
 | 
						|
    CustomizableUI.addListener(this);
 | 
						|
 | 
						|
    this.#checkOverflow();
 | 
						|
 | 
						|
    this.#initialized = true;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Almost the exact reverse of init(). This is called when the browser window
 | 
						|
   * is unloading.
 | 
						|
   */
 | 
						|
  uninit() {
 | 
						|
    this.#toolbar.removeAttribute("overflowable");
 | 
						|
 | 
						|
    if (!this.#initialized) {
 | 
						|
      Services.obs.removeObserver(this, "browser-delayed-startup-finished");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    this.#disable();
 | 
						|
 | 
						|
    let window = this.#toolbar.ownerGlobal;
 | 
						|
    window.removeEventListener("resize", this);
 | 
						|
    window.gNavToolbox.removeEventListener("customizationstarting", this);
 | 
						|
    window.gNavToolbox.removeEventListener("aftercustomization", this);
 | 
						|
    this.#defaultListButton.removeEventListener("mousedown", this);
 | 
						|
    this.#defaultListButton.removeEventListener("keypress", this);
 | 
						|
    this.#defaultListButton.removeEventListener("dragover", this);
 | 
						|
    this.#defaultListButton.removeEventListener("dragend", this);
 | 
						|
    this.#defaultListPanel.removeEventListener("popuphiding", this);
 | 
						|
 | 
						|
    CustomizableUI.removeListener(this);
 | 
						|
    CustomizableUIInternal.removePanelCloseListeners(this.#defaultListPanel);
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Opens the overflow #defaultListPanel if it's not already open. If the panel is in
 | 
						|
   * the midst of hiding when this is called, the panel will be re-opened.
 | 
						|
   *
 | 
						|
   * @returns {Promise}
 | 
						|
   * @resolves {undefined} once the panel is open.
 | 
						|
   */
 | 
						|
  show(aEvent) {
 | 
						|
    if (this.#defaultListPanel.state == "open") {
 | 
						|
      return Promise.resolve();
 | 
						|
    }
 | 
						|
    return new Promise(resolve => {
 | 
						|
      let doc = this.#defaultListPanel.ownerDocument;
 | 
						|
      this.#defaultListPanel.hidden = false;
 | 
						|
      let multiview = this.#defaultListPanel.querySelector("panelmultiview");
 | 
						|
      let mainViewId = multiview.getAttribute("mainViewId");
 | 
						|
      let mainView = doc.getElementById(mainViewId);
 | 
						|
      let contextMenu = doc.getElementById(mainView.getAttribute("context"));
 | 
						|
      contextMenu.addEventListener("command", this, {
 | 
						|
        capture: true,
 | 
						|
        mozSystemGroup: true,
 | 
						|
      });
 | 
						|
      let anchor = this.#defaultListButton.icon;
 | 
						|
 | 
						|
      let popupshown = false;
 | 
						|
      this.#defaultListPanel.addEventListener(
 | 
						|
        "popupshown",
 | 
						|
        () => {
 | 
						|
          popupshown = true;
 | 
						|
          this.#defaultListPanel.addEventListener("dragover", this);
 | 
						|
          this.#defaultListPanel.addEventListener("dragend", this);
 | 
						|
          // Wait until the next tick to resolve so all popupshown
 | 
						|
          // handlers have a chance to run before our promise resolution
 | 
						|
          // handlers do.
 | 
						|
          Services.tm.dispatchToMainThread(resolve);
 | 
						|
        },
 | 
						|
        { once: true }
 | 
						|
      );
 | 
						|
 | 
						|
      let openPanel = () => {
 | 
						|
        // Ensure we update the gEditUIVisible flag when opening the popup, in
 | 
						|
        // case the edit controls are in it.
 | 
						|
        this.#defaultListPanel.addEventListener(
 | 
						|
          "popupshowing",
 | 
						|
          () => {
 | 
						|
            doc.defaultView.updateEditUIVisibility();
 | 
						|
          },
 | 
						|
          { once: true }
 | 
						|
        );
 | 
						|
 | 
						|
        this.#defaultListPanel.addEventListener(
 | 
						|
          "popuphidden",
 | 
						|
          () => {
 | 
						|
            if (!popupshown) {
 | 
						|
              // The panel was hidden again before it was shown. This can break
 | 
						|
              // consumers waiting for the panel to show. So we try again.
 | 
						|
              openPanel();
 | 
						|
            }
 | 
						|
          },
 | 
						|
          { once: true }
 | 
						|
        );
 | 
						|
 | 
						|
        lazy.PanelMultiView.openPopup(
 | 
						|
          this.#defaultListPanel,
 | 
						|
          anchor || this.#defaultListButton,
 | 
						|
          {
 | 
						|
            triggerEvent: aEvent,
 | 
						|
          }
 | 
						|
        );
 | 
						|
        this.#defaultListButton.open = true;
 | 
						|
      };
 | 
						|
 | 
						|
      openPanel();
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Exposes whether #checkOverflow is currently running.
 | 
						|
   *
 | 
						|
   * @returns {boolean} True if #checkOverflow is currently running.
 | 
						|
   */
 | 
						|
  isHandlingOverflow() {
 | 
						|
    return !!this.#checkOverflowHandle;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Finds the most appropriate place to insert toolbar item aNode if we've been
 | 
						|
   * asked to put it into the overflowable toolbar without being told exactly
 | 
						|
   * where.
 | 
						|
   *
 | 
						|
   * @param {Element} aNode The toolbar item being inserted.
 | 
						|
   * @returns {Array} [parent, nextNode]
 | 
						|
   *   parent: {Element} The parent element that should contain aNode.
 | 
						|
   *   nextNode: {Element|null} The node that should follow aNode after
 | 
						|
   *     insertion, if any. If this is null, aNode should be placed at the end
 | 
						|
   *     of parent.
 | 
						|
   */
 | 
						|
  findOverflowedInsertionPoints(aNode) {
 | 
						|
    let newNodeCanOverflow = aNode.getAttribute("overflows") != "false";
 | 
						|
    let areaId = this.#toolbar.id;
 | 
						|
    let placements = gPlacements.get(areaId);
 | 
						|
    let nodeIndex = placements.indexOf(aNode.id);
 | 
						|
    let nodeBeforeNewNodeIsOverflown = false;
 | 
						|
 | 
						|
    let loopIndex = -1;
 | 
						|
    // Loop through placements to find where to insert this item.
 | 
						|
    // As soon as we find an overflown widget, we will only
 | 
						|
    // insert in the overflow panel (this is why we check placements
 | 
						|
    // before the desired location for the new node). Once we pass
 | 
						|
    // the desired location of the widget, we look for placement ids
 | 
						|
    // that actually have DOM equivalents to insert before. If all
 | 
						|
    // else fails, we insert at the end of either the overflow list
 | 
						|
    // or the toolbar target.
 | 
						|
    while (++loopIndex < placements.length) {
 | 
						|
      let nextNodeId = placements[loopIndex];
 | 
						|
      if (loopIndex > nodeIndex) {
 | 
						|
        // Note that if aNode is in a template, its `ownerDocument` is *not*
 | 
						|
        // going to be the browser.xhtml document, so we cannot rely on it.
 | 
						|
        let nextNode = this.#toolbar.ownerDocument.getElementById(nextNodeId);
 | 
						|
        // If the node we're inserting can overflow, and the next node
 | 
						|
        // in the toolbar is overflown, we should insert this node
 | 
						|
        // in the overflow panel before it.
 | 
						|
        if (
 | 
						|
          newNodeCanOverflow &&
 | 
						|
          this.#overflowedInfo.has(nextNodeId) &&
 | 
						|
          nextNode &&
 | 
						|
          nextNode.parentNode == this.#defaultList
 | 
						|
        ) {
 | 
						|
          return [this.#defaultList, nextNode];
 | 
						|
        }
 | 
						|
        // Otherwise (if either we can't overflow, or the previous node
 | 
						|
        // wasn't overflown), and the next node is in the toolbar itself,
 | 
						|
        // insert the node in the toolbar.
 | 
						|
        if (
 | 
						|
          (!nodeBeforeNewNodeIsOverflown || !newNodeCanOverflow) &&
 | 
						|
          nextNode &&
 | 
						|
          (nextNode.parentNode == this.#target ||
 | 
						|
            // Also check if the next node is in a customization wrapper
 | 
						|
            // (toolbarpaletteitem). We don't need to do this for the
 | 
						|
            // overflow case because overflow is disabled in customize mode.
 | 
						|
            (nextNode.parentNode.localName == "toolbarpaletteitem" &&
 | 
						|
              nextNode.parentNode.parentNode == this.#target))
 | 
						|
        ) {
 | 
						|
          return [this.#target, nextNode];
 | 
						|
        }
 | 
						|
      } else if (
 | 
						|
        loopIndex < nodeIndex &&
 | 
						|
        this.#overflowedInfo.has(nextNodeId)
 | 
						|
      ) {
 | 
						|
        nodeBeforeNewNodeIsOverflown = true;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    let overflowList = CustomizableUI.isWebExtensionWidget(aNode.id)
 | 
						|
      ? this.#webExtList
 | 
						|
      : this.#defaultList;
 | 
						|
 | 
						|
    let containerForAppending =
 | 
						|
      this.#overflowedInfo.size && newNodeCanOverflow
 | 
						|
        ? overflowList
 | 
						|
        : this.#target;
 | 
						|
    return [containerForAppending, null];
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Allows callers to query for the current parent of a toolbar item that may
 | 
						|
   * or may not be overflowed. That parent will either be #defaultList,
 | 
						|
   * #webExtList (if it's an extension button) or #target.
 | 
						|
   *
 | 
						|
   * Note: It is assumed that the caller has verified that aNode is placed
 | 
						|
   * within the toolbar customizable area according to CustomizableUI.
 | 
						|
   *
 | 
						|
   * @param {Element} aNode the node that can be overflowed by this
 | 
						|
   *   OverflowableToolbar.
 | 
						|
   * @returns {Element} The current containing node for aNode.
 | 
						|
   */
 | 
						|
  getContainerFor(aNode) {
 | 
						|
    if (aNode.getAttribute("overflowedItem") == "true") {
 | 
						|
      return CustomizableUI.isWebExtensionWidget(aNode.id)
 | 
						|
        ? this.#webExtList
 | 
						|
        : this.#defaultList;
 | 
						|
    }
 | 
						|
    return this.#target;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Private methods start here.
 | 
						|
   */
 | 
						|
 | 
						|
  /**
 | 
						|
   * Handle overflow in the toolbar by moving items to the overflow menu.
 | 
						|
   */
 | 
						|
  async #onOverflow() {
 | 
						|
    if (!this.#enabled) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let win = this.#target.ownerGlobal;
 | 
						|
    let checkOverflowHandle = this.#checkOverflowHandle;
 | 
						|
    let webExtButtonID = this.#toolbar.getAttribute(
 | 
						|
      "addon-webext-overflowbutton"
 | 
						|
    );
 | 
						|
 | 
						|
    let { isOverflowing, targetContentWidth } = await this.#getOverflowInfo();
 | 
						|
 | 
						|
    // Stop if the window has closed or if we re-enter while waiting for
 | 
						|
    // layout.
 | 
						|
    if (win.closed || this.#checkOverflowHandle != checkOverflowHandle) {
 | 
						|
      lazy.log.debug("Window closed or another overflow handler started.");
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let webExtList = this.#webExtList;
 | 
						|
 | 
						|
    let child = this.#target.lastElementChild;
 | 
						|
    while (child && isOverflowing) {
 | 
						|
      let prevChild = child.previousElementSibling;
 | 
						|
 | 
						|
      if (child.getAttribute("overflows") != "false") {
 | 
						|
        this.#overflowedInfo.set(child.id, targetContentWidth);
 | 
						|
        let { width: childWidth } =
 | 
						|
          win.windowUtils.getBoundsWithoutFlushing(child);
 | 
						|
        if (!childWidth) {
 | 
						|
          this.#hiddenOverflowedNodes.add(child);
 | 
						|
        }
 | 
						|
 | 
						|
        child.setAttribute("overflowedItem", true);
 | 
						|
        CustomizableUIInternal.ensureButtonContextMenu(
 | 
						|
          child,
 | 
						|
          this.#toolbar,
 | 
						|
          true
 | 
						|
        );
 | 
						|
        CustomizableUIInternal.notifyListeners(
 | 
						|
          "onWidgetOverflow",
 | 
						|
          child,
 | 
						|
          this.#target
 | 
						|
        );
 | 
						|
 | 
						|
        if (webExtList && CustomizableUI.isWebExtensionWidget(child.id)) {
 | 
						|
          child.setAttribute("cui-anchorid", webExtButtonID);
 | 
						|
          webExtList.insertBefore(child, webExtList.firstElementChild);
 | 
						|
        } else {
 | 
						|
          child.setAttribute("cui-anchorid", this.#defaultListButton.id);
 | 
						|
          this.#defaultList.insertBefore(
 | 
						|
            child,
 | 
						|
            this.#defaultList.firstElementChild
 | 
						|
          );
 | 
						|
          if (!CustomizableUI.isSpecialWidget(child.id) && childWidth) {
 | 
						|
            this.#toolbar.setAttribute("overflowing", "true");
 | 
						|
          }
 | 
						|
        }
 | 
						|
      }
 | 
						|
      child = prevChild;
 | 
						|
      ({ isOverflowing, targetContentWidth } = await this.#getOverflowInfo());
 | 
						|
      // Stop if the window has closed or if we re-enter while waiting for
 | 
						|
      // layout.
 | 
						|
      if (win.closed || this.#checkOverflowHandle != checkOverflowHandle) {
 | 
						|
        lazy.log.debug("Window closed or another overflow handler started.");
 | 
						|
        return;
 | 
						|
      }
 | 
						|
    }
 | 
						|
 | 
						|
    win.UpdateUrlbarSearchSplitterState();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns a Promise that resolves to a an object that describes the state
 | 
						|
   * that this OverflowableToolbar is currently in.
 | 
						|
   *
 | 
						|
   * @returns {Promise}
 | 
						|
   * @resolves {Object}
 | 
						|
   *   An object with the following properties:
 | 
						|
   *
 | 
						|
   *   isOverflowing: {boolean} True if at least one toolbar item has overflowed
 | 
						|
   *     into an overflow panel.
 | 
						|
   *   targetContentWidth: {number} The total width of the items within the
 | 
						|
   *     customization target area of the toolbar.
 | 
						|
   *   totalAvailWidth: {number} The maximum width items in the toolbar may
 | 
						|
   *     occupy before causing an overflow.
 | 
						|
   */
 | 
						|
  async #getOverflowInfo() {
 | 
						|
    function getInlineSize(aElement) {
 | 
						|
      return aElement.getBoundingClientRect().width;
 | 
						|
    }
 | 
						|
 | 
						|
    function sumChildrenInlineSize(aParent, aExceptChild = null) {
 | 
						|
      let sum = 0;
 | 
						|
      for (let child of aParent.children) {
 | 
						|
        let style = win.getComputedStyle(child);
 | 
						|
        if (
 | 
						|
          style.display == "none" ||
 | 
						|
          win.XULPopupElement.isInstance(child) ||
 | 
						|
          (style.position != "static" && style.position != "relative")
 | 
						|
        ) {
 | 
						|
          continue;
 | 
						|
        }
 | 
						|
        sum += parseFloat(style.marginLeft) + parseFloat(style.marginRight);
 | 
						|
        if (child != aExceptChild) {
 | 
						|
          sum += getInlineSize(child);
 | 
						|
        }
 | 
						|
      }
 | 
						|
      return sum;
 | 
						|
    }
 | 
						|
 | 
						|
    let win = this.#target.ownerGlobal;
 | 
						|
    let totalAvailWidth;
 | 
						|
    let targetWidth;
 | 
						|
    let targetChildrenWidth;
 | 
						|
 | 
						|
    await win.promiseDocumentFlushed(() => {
 | 
						|
      let style = win.getComputedStyle(this.#toolbar);
 | 
						|
      let toolbarChildrenWidth = sumChildrenInlineSize(
 | 
						|
        this.#toolbar,
 | 
						|
        this.#target
 | 
						|
      );
 | 
						|
      totalAvailWidth =
 | 
						|
        getInlineSize(this.#toolbar) -
 | 
						|
        parseFloat(style.paddingLeft) -
 | 
						|
        parseFloat(style.paddingRight) -
 | 
						|
        toolbarChildrenWidth;
 | 
						|
      targetWidth = getInlineSize(this.#target);
 | 
						|
      targetChildrenWidth =
 | 
						|
        this.#target == this.#toolbar
 | 
						|
          ? toolbarChildrenWidth
 | 
						|
          : sumChildrenInlineSize(this.#target);
 | 
						|
    });
 | 
						|
 | 
						|
    lazy.log.debug(
 | 
						|
      `Getting overflow info: target width: ${targetWidth} (${targetChildrenWidth}); avail: ${totalAvailWidth}`
 | 
						|
    );
 | 
						|
 | 
						|
    // If the target has min-width: 0, their children might actually overflow
 | 
						|
    // it, so check for both cases explicitly.
 | 
						|
    let targetContentWidth = Math.max(targetWidth, targetChildrenWidth);
 | 
						|
    let isOverflowing = Math.floor(targetContentWidth) > totalAvailWidth;
 | 
						|
    return { isOverflowing, targetContentWidth, totalAvailWidth };
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Tries to move toolbar items back to the toolbar from the overflow panel.
 | 
						|
   *
 | 
						|
   * @param {boolean} shouldMoveAllItems
 | 
						|
   *        Whether we should move everything (e.g. because we're being
 | 
						|
   *        disabled)
 | 
						|
   * @param {number} [totalAvailWidth=undefined]
 | 
						|
   *        Optional; the width of the toolbar area in which we can put things.
 | 
						|
   *        Some consumers pass this to avoid reflows.
 | 
						|
   *
 | 
						|
   *        While there are items in the list, this width won't change, and so
 | 
						|
   *        we can avoid flushing layout by providing it and/or caching it.
 | 
						|
   *        Note that if `shouldMoveAllItems` is true, we never need the width
 | 
						|
   *        anyway, and this value is ignored.
 | 
						|
   * @returns {Promise}
 | 
						|
   * @resolves {undefined} Once moving of items has completed.
 | 
						|
   */
 | 
						|
  async #moveItemsBackToTheirOrigin(shouldMoveAllItems, totalAvailWidth) {
 | 
						|
    lazy.log.debug(
 | 
						|
      `Attempting to move ${shouldMoveAllItems ? "all" : "some"} items back`
 | 
						|
    );
 | 
						|
    let placements = gPlacements.get(this.#toolbar.id);
 | 
						|
    let win = this.#target.ownerGlobal;
 | 
						|
    let doc = this.#target.ownerDocument;
 | 
						|
    let checkOverflowHandle = this.#checkOverflowHandle;
 | 
						|
 | 
						|
    let overflowedItemStack = Array.from(this.#overflowedInfo.entries());
 | 
						|
 | 
						|
    for (let i = overflowedItemStack.length - 1; i >= 0; --i) {
 | 
						|
      let [childID, minSize] = overflowedItemStack[i];
 | 
						|
 | 
						|
      // The item may have been placed inside of a <xul:panel> that is lazily
 | 
						|
      // loaded and still in the view cache. PanelMultiView.getViewNode will
 | 
						|
      // do the work of checking the DOM for the child, and then falling back to
 | 
						|
      // the cache if that is the case.
 | 
						|
      let child = lazy.PanelMultiView.getViewNode(doc, childID);
 | 
						|
 | 
						|
      if (!child) {
 | 
						|
        this.#overflowedInfo.delete(childID);
 | 
						|
        continue;
 | 
						|
      }
 | 
						|
 | 
						|
      lazy.log.debug(
 | 
						|
        `Considering moving ${child.id} back, minSize: ${minSize}`
 | 
						|
      );
 | 
						|
 | 
						|
      if (!shouldMoveAllItems && minSize) {
 | 
						|
        if (!totalAvailWidth) {
 | 
						|
          ({ totalAvailWidth } = await this.#getOverflowInfo());
 | 
						|
 | 
						|
          // If the window has closed or if we re-enter because we were waiting
 | 
						|
          // for layout, stop.
 | 
						|
          if (win.closed || this.#checkOverflowHandle != checkOverflowHandle) {
 | 
						|
            lazy.log.debug("Window closed or #checkOverflow called again.");
 | 
						|
            return;
 | 
						|
          }
 | 
						|
        }
 | 
						|
        if (totalAvailWidth <= minSize) {
 | 
						|
          lazy.log.debug(
 | 
						|
            `Need ${minSize} but width is ${totalAvailWidth} so bailing`
 | 
						|
          );
 | 
						|
          break;
 | 
						|
        }
 | 
						|
      }
 | 
						|
 | 
						|
      lazy.log.debug(`Moving ${child.id} back`);
 | 
						|
      this.#overflowedInfo.delete(child.id);
 | 
						|
      let beforeNodeIndex = placements.indexOf(child.id) + 1;
 | 
						|
      // If this is a skipintoolbarset item, meaning it doesn't occur in the placements list,
 | 
						|
      // we're inserting it at the end. This will mean first-in, first-out (more or less)
 | 
						|
      // leading to as little change in order as possible.
 | 
						|
      if (beforeNodeIndex == 0) {
 | 
						|
        beforeNodeIndex = placements.length;
 | 
						|
      }
 | 
						|
      let inserted = false;
 | 
						|
      for (; beforeNodeIndex < placements.length; beforeNodeIndex++) {
 | 
						|
        let beforeNode = this.#target.getElementsByAttribute(
 | 
						|
          "id",
 | 
						|
          placements[beforeNodeIndex]
 | 
						|
        )[0];
 | 
						|
        // Unfortunately, XUL add-ons can mess with nodes after they are inserted,
 | 
						|
        // and this breaks the following code if the button isn't where we expect
 | 
						|
        // it to be (ie not a child of the target). In this case, ignore the node.
 | 
						|
        if (beforeNode && this.#target == beforeNode.parentElement) {
 | 
						|
          this.#target.insertBefore(child, beforeNode);
 | 
						|
          inserted = true;
 | 
						|
          break;
 | 
						|
        }
 | 
						|
      }
 | 
						|
      if (!inserted) {
 | 
						|
        this.#target.appendChild(child);
 | 
						|
      }
 | 
						|
      child.removeAttribute("cui-anchorid");
 | 
						|
      child.removeAttribute("overflowedItem");
 | 
						|
      CustomizableUIInternal.ensureButtonContextMenu(child, this.#target);
 | 
						|
      CustomizableUIInternal.notifyListeners(
 | 
						|
        "onWidgetUnderflow",
 | 
						|
        child,
 | 
						|
        this.#target
 | 
						|
      );
 | 
						|
    }
 | 
						|
 | 
						|
    win.UpdateUrlbarSearchSplitterState();
 | 
						|
 | 
						|
    let defaultListItems = Array.from(this.#defaultList.children);
 | 
						|
    if (
 | 
						|
      defaultListItems.every(
 | 
						|
        item =>
 | 
						|
          CustomizableUI.isSpecialWidget(item.id) ||
 | 
						|
          this.#hiddenOverflowedNodes.has(item)
 | 
						|
      )
 | 
						|
    ) {
 | 
						|
      this.#toolbar.removeAttribute("overflowing");
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Checks to see if there are overflowable items within the customization
 | 
						|
   * target of the toolbar that should be moved into the overflow panel, and
 | 
						|
   * if there are, moves them.
 | 
						|
   *
 | 
						|
   * Note that since this is an async function that can be called in bursts
 | 
						|
   * by resize events on the window, this function is often re-called even
 | 
						|
   * when a prior call hasn't yet resolved. In that situation, the older calls
 | 
						|
   * resolve early without doing any work and leave any DOM manipulation to the
 | 
						|
   * most recent call.
 | 
						|
   *
 | 
						|
   * This function is a no-op if the OverflowableToolbar is disabled or the
 | 
						|
   * DOM fullscreen UI is currently being used.
 | 
						|
   *
 | 
						|
   * @returns {Promise}
 | 
						|
   * @resolves {undefined} Once any movement of toolbar items has completed.
 | 
						|
   */
 | 
						|
  async #checkOverflow() {
 | 
						|
    if (!this.#enabled) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let win = this.#target.ownerGlobal;
 | 
						|
    if (win.document.documentElement.hasAttribute("inDOMFullscreen")) {
 | 
						|
      // Toolbars are hidden and cannot be made visible in DOM fullscreen mode
 | 
						|
      // so there's nothing to do here.
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let checkOverflowHandle = (this.#checkOverflowHandle = {});
 | 
						|
 | 
						|
    lazy.log.debug("Checking overflow");
 | 
						|
    let { isOverflowing, totalAvailWidth } = await this.#getOverflowInfo();
 | 
						|
    if (win.closed || this.#checkOverflowHandle != checkOverflowHandle) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    if (isOverflowing) {
 | 
						|
      await this.#onOverflow();
 | 
						|
    } else {
 | 
						|
      await this.#moveItemsBackToTheirOrigin(false, totalAvailWidth);
 | 
						|
    }
 | 
						|
 | 
						|
    if (checkOverflowHandle == this.#checkOverflowHandle) {
 | 
						|
      this.#checkOverflowHandle = null;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Makes the OverflowableToolbar inert and moves all overflowable items back
 | 
						|
   * into the customization target of the toolbar.
 | 
						|
   */
 | 
						|
  #disable() {
 | 
						|
    // Abort any ongoing overflow check. #enable() will #checkOverflow()
 | 
						|
    // anyways, so this is enough.
 | 
						|
    this.#checkOverflowHandle = {};
 | 
						|
    this.#moveItemsBackToTheirOrigin(true);
 | 
						|
    this.#enabled = false;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Puts the OverflowableToolbar into the enabled state and then checks to see
 | 
						|
   * if any of the items in the customization target should be overflowed into
 | 
						|
   * the overflow panel list.
 | 
						|
   */
 | 
						|
  #enable() {
 | 
						|
    this.#enabled = true;
 | 
						|
    this.#checkOverflow();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Shows the overflow panel and sets a timeout to automatically re-hide the
 | 
						|
   * panel if it is not being hovered.
 | 
						|
   */
 | 
						|
  #showWithTimeout() {
 | 
						|
    const OVERFLOW_PANEL_HIDE_DELAY_MS = 500;
 | 
						|
 | 
						|
    this.show().then(() => {
 | 
						|
      let window = this.#toolbar.ownerGlobal;
 | 
						|
      if (this.#hideTimeoutId) {
 | 
						|
        window.clearTimeout(this.#hideTimeoutId);
 | 
						|
      }
 | 
						|
      this.#hideTimeoutId = window.setTimeout(() => {
 | 
						|
        if (!this.#defaultListPanel.firstElementChild.matches(":hover")) {
 | 
						|
          lazy.PanelMultiView.hidePopup(this.#defaultListPanel);
 | 
						|
        }
 | 
						|
      }, OVERFLOW_PANEL_HIDE_DELAY_MS);
 | 
						|
    });
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Gets and caches a reference to the DOM node with the ID set as the value
 | 
						|
   * of addon-webext-overflowtarget. If a cache already exists, that's returned
 | 
						|
   * instead. If addon-webext-overflowtarget has no value, null is returned.
 | 
						|
   *
 | 
						|
   * @returns {Element|null} the list that overflowed extension toolbar
 | 
						|
   *   buttons should go to if the Unified Extensions UI is enabled, or null
 | 
						|
   *   if no such list exists.
 | 
						|
   */
 | 
						|
  get #webExtList() {
 | 
						|
    if (!this.#webExtListRef) {
 | 
						|
      let targetID = this.#toolbar.getAttribute("addon-webext-overflowtarget");
 | 
						|
      if (!targetID) {
 | 
						|
        throw new Error(
 | 
						|
          "addon-webext-overflowtarget was not defined on the " +
 | 
						|
            `overflowable toolbar with id: ${this.#toolbar.id}`
 | 
						|
        );
 | 
						|
      }
 | 
						|
      let win = this.#toolbar.ownerGlobal;
 | 
						|
      let { panel } = win.gUnifiedExtensions;
 | 
						|
      this.#webExtListRef = panel.querySelector(`#${targetID}`);
 | 
						|
    }
 | 
						|
    return this.#webExtListRef;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Returns true if aNode is not null and is one of either this.#webExtList or
 | 
						|
   * this.#defaultList.
 | 
						|
   *
 | 
						|
   * @param {DOMElement} aNode The node to test.
 | 
						|
   * @returns {boolean}
 | 
						|
   */
 | 
						|
  #isOverflowList(aNode) {
 | 
						|
    return aNode == this.#defaultList || aNode == this.#webExtList;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Private event handlers start here.
 | 
						|
   */
 | 
						|
 | 
						|
  /**
 | 
						|
   * Handles clicks on the #defaultListButton element.
 | 
						|
   *
 | 
						|
   * @param {MouseEvent} aEvent the click event.
 | 
						|
   */
 | 
						|
  #onClickDefaultListButton(aEvent) {
 | 
						|
    if (this.#defaultListButton.open) {
 | 
						|
      this.#defaultListButton.open = false;
 | 
						|
      lazy.PanelMultiView.hidePopup(this.#defaultListPanel);
 | 
						|
    } else if (
 | 
						|
      this.#defaultListPanel.state != "hiding" &&
 | 
						|
      !this.#defaultListButton.disabled
 | 
						|
    ) {
 | 
						|
      this.show(aEvent);
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Handles the popuphiding event firing on the #defaultListPanel.
 | 
						|
   *
 | 
						|
   * @param {WidgetMouseEvent} aEvent the popuphiding event that fired on the
 | 
						|
   *   #defaultListPanel.
 | 
						|
   */
 | 
						|
  #onPanelHiding(aEvent) {
 | 
						|
    if (aEvent.target != this.#defaultListPanel) {
 | 
						|
      // Ignore context menus, <select> popups, etc.
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    this.#defaultListButton.open = false;
 | 
						|
    this.#defaultListPanel.removeEventListener("dragover", this);
 | 
						|
    this.#defaultListPanel.removeEventListener("dragend", this);
 | 
						|
    let doc = aEvent.target.ownerDocument;
 | 
						|
    doc.defaultView.updateEditUIVisibility();
 | 
						|
    let contextMenuId = this.#defaultListPanel.getAttribute("context");
 | 
						|
    if (contextMenuId) {
 | 
						|
      let contextMenu = doc.getElementById(contextMenuId);
 | 
						|
      contextMenu.removeEventListener("command", this, {
 | 
						|
        capture: true,
 | 
						|
        mozSystemGroup: true,
 | 
						|
      });
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * Handles a resize event fired on the window hosting this
 | 
						|
   * OverflowableToolbar.
 | 
						|
   *
 | 
						|
   * @param {UIEvent} aEvent the resize event.
 | 
						|
   */
 | 
						|
  #onResize(aEvent) {
 | 
						|
    // Ignore bubbled-up resize events.
 | 
						|
    if (aEvent.target != aEvent.currentTarget) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    this.#checkOverflow();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * CustomizableUI listener methods start here.
 | 
						|
   */
 | 
						|
 | 
						|
  onWidgetBeforeDOMChange(aNode, aNextNode, aContainer) {
 | 
						|
    // This listener method is used to handle the case where a widget is
 | 
						|
    // moved or removed from an area via the CustomizableUI API while
 | 
						|
    // overflowed. It reorganizes the internal state of this OverflowableToolbar
 | 
						|
    // to handle that change.
 | 
						|
    if (!this.#enabled || !this.#isOverflowList(aContainer)) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
    // When we (re)move an item, update all the items that come after it in the list
 | 
						|
    // with the minsize *of the item before the to-be-removed node*. This way, we
 | 
						|
    // ensure that we try to move items back as soon as that's possible.
 | 
						|
    let updatedMinSize;
 | 
						|
    if (aNode.previousElementSibling) {
 | 
						|
      updatedMinSize = this.#overflowedInfo.get(
 | 
						|
        aNode.previousElementSibling.id
 | 
						|
      );
 | 
						|
    } else {
 | 
						|
      // Force (these) items to try to flow back into the bar:
 | 
						|
      updatedMinSize = 1;
 | 
						|
    }
 | 
						|
    let nextItem = aNode.nextElementSibling;
 | 
						|
    while (nextItem) {
 | 
						|
      this.#overflowedInfo.set(nextItem.id, updatedMinSize);
 | 
						|
      nextItem = nextItem.nextElementSibling;
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  onWidgetAfterDOMChange(aNode, aNextNode, aContainer) {
 | 
						|
    // This listener method is used to handle the case where a widget is
 | 
						|
    // moved or removed from an area via the CustomizableUI API while
 | 
						|
    // overflowed. It updates the DOM in the event that the movement or removal
 | 
						|
    // causes overflow or underflow of the toolbar.
 | 
						|
    if (
 | 
						|
      !this.#enabled ||
 | 
						|
      (aContainer != this.#target && !this.#isOverflowList(aContainer))
 | 
						|
    ) {
 | 
						|
      return;
 | 
						|
    }
 | 
						|
 | 
						|
    let nowOverflowed = this.#isOverflowList(aNode.parentNode);
 | 
						|
    let wasOverflowed = this.#overflowedInfo.has(aNode.id);
 | 
						|
 | 
						|
    // If this wasn't overflowed before...
 | 
						|
    if (!wasOverflowed) {
 | 
						|
      // ... but it is now, then we added to one of the overflow panels.
 | 
						|
      if (nowOverflowed) {
 | 
						|
        // We could be the first item in the overflow panel if we're being inserted
 | 
						|
        // before the previous first item in it. We can't assume the minimum
 | 
						|
        // size is the same (because the other item might be much wider), so if
 | 
						|
        // there is no previous item, just allow this item to be put back in the
 | 
						|
        // toolbar immediately by specifying a very low minimum size.
 | 
						|
        let sourceOfMinSize = aNode.previousElementSibling;
 | 
						|
        let minSize = sourceOfMinSize
 | 
						|
          ? this.#overflowedInfo.get(sourceOfMinSize.id)
 | 
						|
          : 1;
 | 
						|
        this.#overflowedInfo.set(aNode.id, minSize);
 | 
						|
        aNode.setAttribute("cui-anchorid", this.#defaultListButton.id);
 | 
						|
        aNode.setAttribute("overflowedItem", true);
 | 
						|
        CustomizableUIInternal.ensureButtonContextMenu(aNode, aContainer, true);
 | 
						|
        CustomizableUIInternal.notifyListeners(
 | 
						|
          "onWidgetOverflow",
 | 
						|
          aNode,
 | 
						|
          this.#target
 | 
						|
        );
 | 
						|
      }
 | 
						|
    } else if (!nowOverflowed) {
 | 
						|
      // If it used to be overflowed...
 | 
						|
      // ... and isn't anymore, let's remove our bookkeeping:
 | 
						|
      this.#overflowedInfo.delete(aNode.id);
 | 
						|
      aNode.removeAttribute("cui-anchorid");
 | 
						|
      aNode.removeAttribute("overflowedItem");
 | 
						|
      CustomizableUIInternal.ensureButtonContextMenu(aNode, aContainer);
 | 
						|
      CustomizableUIInternal.notifyListeners(
 | 
						|
        "onWidgetUnderflow",
 | 
						|
        aNode,
 | 
						|
        this.#target
 | 
						|
      );
 | 
						|
 | 
						|
      let collapsedWidgetIds = Array.from(this.#overflowedInfo.keys());
 | 
						|
      if (collapsedWidgetIds.every(w => CustomizableUI.isSpecialWidget(w))) {
 | 
						|
        this.#toolbar.removeAttribute("overflowing");
 | 
						|
      }
 | 
						|
    } else if (aNode.previousElementSibling) {
 | 
						|
      // but if it still is, it must have changed places. Bookkeep:
 | 
						|
      let prevId = aNode.previousElementSibling.id;
 | 
						|
      let minSize = this.#overflowedInfo.get(prevId);
 | 
						|
      this.#overflowedInfo.set(aNode.id, minSize);
 | 
						|
    }
 | 
						|
 | 
						|
    // We might overflow now if an item was added, or we may be able to move
 | 
						|
    // stuff back into the toolbar if an item was removed.
 | 
						|
    this.#checkOverflow();
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * @returns {Boolean} whether the given node is in the overflow list.
 | 
						|
   */
 | 
						|
  isInOverflowList(node) {
 | 
						|
    return node.parentNode == this.#defaultList;
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * nsIObserver implementation starts here.
 | 
						|
   */
 | 
						|
 | 
						|
  observe(aSubject, aTopic) {
 | 
						|
    // This nsIObserver method allows us to defer initialization until after
 | 
						|
    // this window has finished painting and starting up.
 | 
						|
    if (
 | 
						|
      aTopic == "browser-delayed-startup-finished" &&
 | 
						|
      aSubject == this.#toolbar.ownerGlobal
 | 
						|
    ) {
 | 
						|
      Services.obs.removeObserver(this, "browser-delayed-startup-finished");
 | 
						|
      this.init();
 | 
						|
    }
 | 
						|
  }
 | 
						|
 | 
						|
  /**
 | 
						|
   * nsIDOMEventListener implementation starts here.
 | 
						|
   */
 | 
						|
 | 
						|
  handleEvent(aEvent) {
 | 
						|
    switch (aEvent.type) {
 | 
						|
      case "aftercustomization": {
 | 
						|
        this.#enable();
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "mousedown": {
 | 
						|
        if (aEvent.button != 0) {
 | 
						|
          break;
 | 
						|
        }
 | 
						|
        if (aEvent.target == this.#defaultListButton) {
 | 
						|
          this.#onClickDefaultListButton(aEvent);
 | 
						|
        } else {
 | 
						|
          lazy.PanelMultiView.hidePopup(this.#defaultListPanel);
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "keypress": {
 | 
						|
        if (
 | 
						|
          aEvent.target == this.#defaultListButton &&
 | 
						|
          (aEvent.key == " " || aEvent.key == "Enter")
 | 
						|
        ) {
 | 
						|
          this.#onClickDefaultListButton(aEvent);
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "customizationstarting": {
 | 
						|
        this.#disable();
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "dragover": {
 | 
						|
        if (this.#enabled) {
 | 
						|
          this.#showWithTimeout();
 | 
						|
        }
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "dragend": {
 | 
						|
        lazy.PanelMultiView.hidePopup(this.#defaultListPanel);
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "popuphiding": {
 | 
						|
        this.#onPanelHiding(aEvent);
 | 
						|
        break;
 | 
						|
      }
 | 
						|
      case "resize": {
 | 
						|
        this.#onResize(aEvent);
 | 
						|
        break;
 | 
						|
      }
 | 
						|
    }
 | 
						|
  }
 | 
						|
}
 | 
						|
 | 
						|
CustomizableUIInternal.initialize();
 |