forked from mirrors/gecko-dev
		
	 61ae9405d9
			
		
	
	
		61ae9405d9
		
	
	
	
	
		
			
			This is a very old and legacy attribute which always had a fuzzy definition. As of today it was a somewhat alias of "not debugging a local/remote tab". Differential Revision: https://phabricator.services.mozilla.com/D166904
		
			
				
	
	
		
			1998 lines
		
	
	
	
		
			62 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			1998 lines
		
	
	
	
		
			62 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/. */
 | |
| 
 | |
| "use strict";
 | |
| 
 | |
| const EventEmitter = require("resource://devtools/shared/event-emitter.js");
 | |
| const flags = require("resource://devtools/shared/flags.js");
 | |
| const { executeSoon } = require("resource://devtools/shared/DevToolsUtils.js");
 | |
| const { Toolbox } = require("resource://devtools/client/framework/toolbox.js");
 | |
| const createStore = require("resource://devtools/client/inspector/store.js");
 | |
| const InspectorStyleChangeTracker = require("resource://devtools/client/inspector/shared/style-change-tracker.js");
 | |
| 
 | |
| // Use privileged promise in panel documents to prevent having them to freeze
 | |
| // during toolbox destruction. See bug 1402779.
 | |
| const Promise = require("Promise");
 | |
| 
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "HTMLBreadcrumbs",
 | |
|   "resource://devtools/client/inspector/breadcrumbs.js",
 | |
|   true
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "KeyShortcuts",
 | |
|   "resource://devtools/client/shared/key-shortcuts.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "InspectorSearch",
 | |
|   "resource://devtools/client/inspector/inspector-search.js",
 | |
|   true
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "ToolSidebar",
 | |
|   "resource://devtools/client/inspector/toolsidebar.js",
 | |
|   true
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "MarkupView",
 | |
|   "resource://devtools/client/inspector/markup/markup.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "HighlightersOverlay",
 | |
|   "resource://devtools/client/inspector/shared/highlighters-overlay.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "ExtensionSidebar",
 | |
|   "resource://devtools/client/inspector/extensions/extension-sidebar.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "PICKER_TYPES",
 | |
|   "resource://devtools/shared/picker-constants.js"
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "captureAndSaveScreenshot",
 | |
|   "resource://devtools/client/shared/screenshot.js",
 | |
|   true
 | |
| );
 | |
| loader.lazyRequireGetter(
 | |
|   this,
 | |
|   "debounce",
 | |
|   "resource://devtools/shared/debounce.js",
 | |
|   true
 | |
| );
 | |
| 
 | |
| const {
 | |
|   LocalizationHelper,
 | |
|   localizeMarkup,
 | |
| } = require("resource://devtools/shared/l10n.js");
 | |
| const INSPECTOR_L10N = new LocalizationHelper(
 | |
|   "devtools/client/locales/inspector.properties"
 | |
| );
 | |
| const {
 | |
|   FluentL10n,
 | |
| } = require("resource://devtools/client/shared/fluent-l10n/fluent-l10n.js");
 | |
| 
 | |
| // Sidebar dimensions
 | |
| const INITIAL_SIDEBAR_SIZE = 350;
 | |
| 
 | |
| // How long we wait to debounce resize events
 | |
| const LAZY_RESIZE_INTERVAL_MS = 200;
 | |
| 
 | |
| // If the toolbox's width is smaller than the given amount of pixels, the sidebar
 | |
| // automatically switches from 'landscape/horizontal' to 'portrait/vertical' mode.
 | |
| const PORTRAIT_MODE_WIDTH_THRESHOLD = 700;
 | |
| // If the toolbox's width docked to the side is smaller than the given amount of pixels,
 | |
| // the sidebar automatically switches from 'landscape/horizontal' to 'portrait/vertical'
 | |
| // mode.
 | |
| const SIDE_PORTAIT_MODE_WIDTH_THRESHOLD = 1000;
 | |
| 
 | |
| const THREE_PANE_ENABLED_PREF = "devtools.inspector.three-pane-enabled";
 | |
| const THREE_PANE_ENABLED_SCALAR = "devtools.inspector.three_pane_enabled";
 | |
| const THREE_PANE_CHROME_ENABLED_PREF =
 | |
|   "devtools.inspector.chrome.three-pane-enabled";
 | |
| const TELEMETRY_EYEDROPPER_OPENED = "devtools.toolbar.eyedropper.opened";
 | |
| const TELEMETRY_SCALAR_NODE_SELECTION_COUNT =
 | |
|   "devtools.inspector.node_selection_count";
 | |
| 
 | |
| /**
 | |
|  * Represents an open instance of the Inspector for a tab.
 | |
|  * The inspector controls the breadcrumbs, the markup view, and the sidebar
 | |
|  * (computed view, rule view, font view and animation inspector).
 | |
|  *
 | |
|  * Events:
 | |
|  * - ready
 | |
|  *      Fired when the inspector panel is opened for the first time and ready to
 | |
|  *      use
 | |
|  * - new-root
 | |
|  *      Fired after a new root (navigation to a new page) event was fired by
 | |
|  *      the walker, and taken into account by the inspector (after the markup
 | |
|  *      view has been reloaded)
 | |
|  * - markuploaded
 | |
|  *      Fired when the markup-view frame has loaded
 | |
|  * - breadcrumbs-updated
 | |
|  *      Fired when the breadcrumb widget updates to a new node
 | |
|  * - boxmodel-view-updated
 | |
|  *      Fired when the box model updates to a new node
 | |
|  * - markupmutation
 | |
|  *      Fired after markup mutations have been processed by the markup-view
 | |
|  * - computed-view-refreshed
 | |
|  *      Fired when the computed rules view updates to a new node
 | |
|  * - computed-view-property-expanded
 | |
|  *      Fired when a property is expanded in the computed rules view
 | |
|  * - computed-view-property-collapsed
 | |
|  *      Fired when a property is collapsed in the computed rules view
 | |
|  * - computed-view-sourcelinks-updated
 | |
|  *      Fired when the stylesheet source links have been updated (when switching
 | |
|  *      to source-mapped files)
 | |
|  * - rule-view-refreshed
 | |
|  *      Fired when the rule view updates to a new node
 | |
|  * - rule-view-sourcelinks-updated
 | |
|  *      Fired when the stylesheet source links have been updated (when switching
 | |
|  *      to source-mapped files)
 | |
|  */
 | |
| function Inspector(toolbox, commands) {
 | |
|   EventEmitter.decorate(this);
 | |
| 
 | |
|   this._toolbox = toolbox;
 | |
|   this._commands = commands;
 | |
|   this.panelDoc = window.document;
 | |
|   this.panelWin = window;
 | |
|   this.panelWin.inspector = this;
 | |
|   this.telemetry = toolbox.telemetry;
 | |
|   this.store = createStore(this);
 | |
| 
 | |
|   // Map [panel id => panel instance]
 | |
|   // Stores all the instances of sidebar panels like rule view, computed view, ...
 | |
|   this._panels = new Map();
 | |
| 
 | |
|   this._clearSearchResultsLabel = this._clearSearchResultsLabel.bind(this);
 | |
|   this._handleRejectionIfNotDestroyed = this._handleRejectionIfNotDestroyed.bind(
 | |
|     this
 | |
|   );
 | |
|   this._onTargetAvailable = this._onTargetAvailable.bind(this);
 | |
|   this._onTargetDestroyed = this._onTargetDestroyed.bind(this);
 | |
|   this._onTargetSelected = this._onTargetSelected.bind(this);
 | |
|   this._onWillNavigate = this._onWillNavigate.bind(this);
 | |
|   this._updateSearchResultsLabel = this._updateSearchResultsLabel.bind(this);
 | |
| 
 | |
|   this.onDetached = this.onDetached.bind(this);
 | |
|   this.onHostChanged = this.onHostChanged.bind(this);
 | |
|   this.onNewSelection = this.onNewSelection.bind(this);
 | |
|   this.onResourceAvailable = this.onResourceAvailable.bind(this);
 | |
|   this.onRootNodeAvailable = this.onRootNodeAvailable.bind(this);
 | |
|   this._onLazyPanelResize = this._onLazyPanelResize.bind(this);
 | |
|   this.onPanelWindowResize = debounce(
 | |
|     this._onLazyPanelResize,
 | |
|     LAZY_RESIZE_INTERVAL_MS,
 | |
|     this
 | |
|   );
 | |
|   this.onPickerCanceled = this.onPickerCanceled.bind(this);
 | |
|   this.onPickerHovered = this.onPickerHovered.bind(this);
 | |
|   this.onPickerPicked = this.onPickerPicked.bind(this);
 | |
|   this.onSidebarHidden = this.onSidebarHidden.bind(this);
 | |
|   this.onSidebarResized = this.onSidebarResized.bind(this);
 | |
|   this.onSidebarSelect = this.onSidebarSelect.bind(this);
 | |
|   this.onSidebarShown = this.onSidebarShown.bind(this);
 | |
|   this.onSidebarToggle = this.onSidebarToggle.bind(this);
 | |
|   this.onReflowInSelection = this.onReflowInSelection.bind(this);
 | |
|   this.listenForSearchEvents = this.listenForSearchEvents.bind(this);
 | |
| }
 | |
| 
 | |
| Inspector.prototype = {
 | |
|   /**
 | |
|    * InspectorPanel.open() is effectively an asynchronous constructor.
 | |
|    * Set any attributes or listeners that rely on the document being loaded or fronts
 | |
|    * from the InspectorFront and Target here.
 | |
|    */
 | |
|   async init() {
 | |
|     // Localize all the nodes containing a data-localization attribute.
 | |
|     localizeMarkup(this.panelDoc);
 | |
| 
 | |
|     this._fluentL10n = new FluentL10n();
 | |
|     await this._fluentL10n.init(["devtools/client/compatibility.ftl"]);
 | |
| 
 | |
|     // Display the main inspector panel with: search input, markup view and breadcrumbs.
 | |
|     this.panelDoc.getElementById("inspector-main-content").style.visibility =
 | |
|       "visible";
 | |
| 
 | |
|     // Setup the splitter before watching targets & resources.
 | |
|     // The markup view will be initialized after we get the first root-node
 | |
|     // resource, and the splitter should be initialized before that.
 | |
|     // The markup view is rendered in an iframe and the splitter will move the
 | |
|     // parent of the iframe in the DOM tree which would reset the state of the
 | |
|     // iframe if it had already been initialized.
 | |
|     this.setupSplitter();
 | |
| 
 | |
|     await this.commands.targetCommand.watchTargets({
 | |
|       types: [this.commands.targetCommand.TYPES.FRAME],
 | |
|       onAvailable: this._onTargetAvailable,
 | |
|       onSelected: this._onTargetSelected,
 | |
|       onDestroyed: this._onTargetDestroyed,
 | |
|     });
 | |
| 
 | |
|     await this.toolbox.resourceCommand.watchResources(
 | |
|       [
 | |
|         this.toolbox.resourceCommand.TYPES.ROOT_NODE,
 | |
|         // To observe CSS change before opening changes view.
 | |
|         this.toolbox.resourceCommand.TYPES.CSS_CHANGE,
 | |
|         this.toolbox.resourceCommand.TYPES.DOCUMENT_EVENT,
 | |
|       ],
 | |
|       { onAvailable: this.onResourceAvailable }
 | |
|     );
 | |
| 
 | |
|     // Store the URL of the target page prior to navigation in order to ensure
 | |
|     // telemetry counts in the Grid Inspector are not double counted on reload.
 | |
|     this.previousURL = this.currentTarget.url;
 | |
| 
 | |
|     // Note: setupSidebar() really has to be called after the first target has
 | |
|     // been processed, so that the cssProperties getter works.
 | |
|     // But the rest could be moved before the watch* calls.
 | |
|     this.styleChangeTracker = new InspectorStyleChangeTracker(this);
 | |
|     this.setupSidebar();
 | |
|     this.breadcrumbs = new HTMLBreadcrumbs(this);
 | |
|     this.setupExtensionSidebars();
 | |
|     this.setupSearchBox();
 | |
| 
 | |
|     this.onNewSelection();
 | |
| 
 | |
|     this.toolbox.on("host-changed", this.onHostChanged);
 | |
|     this.toolbox.nodePicker.on("picker-node-hovered", this.onPickerHovered);
 | |
|     this.toolbox.nodePicker.on("picker-node-canceled", this.onPickerCanceled);
 | |
|     this.toolbox.nodePicker.on("picker-node-picked", this.onPickerPicked);
 | |
|     this.selection.on("new-node-front", this.onNewSelection);
 | |
|     this.selection.on("detached-front", this.onDetached);
 | |
| 
 | |
|     // Log the 3 pane inspector setting on inspector open. The question we want to answer
 | |
|     // is:
 | |
|     // "What proportion of users use the 3 pane vs 2 pane inspector on inspector open?"
 | |
|     this.telemetry.keyedScalarAdd(
 | |
|       THREE_PANE_ENABLED_SCALAR,
 | |
|       this.is3PaneModeEnabled,
 | |
|       1
 | |
|     );
 | |
| 
 | |
|     return this;
 | |
|   },
 | |
| 
 | |
|   async _onTargetAvailable({ targetFront }) {
 | |
|     // Ignore all targets but the top level one
 | |
|     if (!targetFront.isTopLevel) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     await this.initInspectorFront(targetFront);
 | |
| 
 | |
|     // the target might have been destroyed when reloading quickly,
 | |
|     // while waiting for inspector front initialization
 | |
|     if (targetFront.isDestroyed()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     await Promise.all([
 | |
|       this._getCssProperties(targetFront),
 | |
|       this._getAccessibilityFront(targetFront),
 | |
|     ]);
 | |
|   },
 | |
| 
 | |
|   async _onTargetSelected({ targetFront }) {
 | |
|     // We don't use this.highlighters since it creates a HighlightersOverlay if it wasn't
 | |
|     // the case yet.
 | |
|     if (this._highlighters) {
 | |
|       this._highlighters.hideAllHighlighters();
 | |
|     }
 | |
|     if (targetFront.isDestroyed()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     await this.initInspectorFront(targetFront);
 | |
| 
 | |
|     // the target might have been destroyed when reloading quickly,
 | |
|     // while waiting for inspector front initialization
 | |
|     if (targetFront.isDestroyed()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     const { walker } = await targetFront.getFront("inspector");
 | |
|     const rootNodeFront = await walker.getRootNode();
 | |
|     // When a given target is focused, don't try to reset the selection
 | |
|     this.selectionCssSelectors = [];
 | |
|     this._defaultNode = null;
 | |
| 
 | |
|     // onRootNodeAvailable will take care of populating the markup view
 | |
|     await this.onRootNodeAvailable(rootNodeFront);
 | |
|   },
 | |
| 
 | |
|   _onTargetDestroyed({ targetFront }) {
 | |
|     // Ignore all targets but the top level one
 | |
|     if (!targetFront.isTopLevel) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this._defaultNode = null;
 | |
|     this.selection.setNodeFront(null);
 | |
|   },
 | |
| 
 | |
|   onResourceAvailable(resources) {
 | |
|     // Store all onRootNodeAvailable calls which are asynchronous.
 | |
|     const rootNodeAvailablePromises = [];
 | |
| 
 | |
|     for (const resource of resources) {
 | |
|       const isTopLevelTarget = !!resource.targetFront?.isTopLevel;
 | |
|       const isTopLevelDocument = !!resource.isTopLevelDocument;
 | |
|       if (
 | |
|         resource.resourceType ===
 | |
|           this.toolbox.resourceCommand.TYPES.ROOT_NODE &&
 | |
|         // It might happen that the ROOT_NODE resource (which is a Front) is already
 | |
|         // destroyed, and in such case we want to ignore it.
 | |
|         !resource.isDestroyed() &&
 | |
|         isTopLevelTarget &&
 | |
|         isTopLevelDocument
 | |
|       ) {
 | |
|         rootNodeAvailablePromises.push(this.onRootNodeAvailable(resource));
 | |
|       }
 | |
| 
 | |
|       // Only consider top level document, and ignore remote iframes top document
 | |
|       if (
 | |
|         resource.resourceType ===
 | |
|           this.toolbox.resourceCommand.TYPES.DOCUMENT_EVENT &&
 | |
|         resource.name === "will-navigate" &&
 | |
|         isTopLevelTarget
 | |
|       ) {
 | |
|         this._onWillNavigate();
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return Promise.all(rootNodeAvailablePromises);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Reset the inspector on new root mutation.
 | |
|    */
 | |
|   async onRootNodeAvailable(rootNodeFront) {
 | |
|     // Record new-root timing for telemetry
 | |
|     this._newRootStart = this.panelWin.performance.now();
 | |
| 
 | |
|     this._defaultNode = null;
 | |
|     this.selection.setNodeFront(null);
 | |
|     this._destroyMarkup();
 | |
| 
 | |
|     try {
 | |
|       const defaultNode = await this._getDefaultNodeForSelection(rootNodeFront);
 | |
|       if (!defaultNode) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       this.selection.setNodeFront(defaultNode, {
 | |
|         reason: "inspector-default-selection",
 | |
|       });
 | |
| 
 | |
|       await this._initMarkupView();
 | |
| 
 | |
|       // Setup the toolbar again, since its content may depend on the current document.
 | |
|       this.setupToolbar();
 | |
|     } catch (e) {
 | |
|       this._handleRejectionIfNotDestroyed(e);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   async _initMarkupView() {
 | |
|     if (!this._markupFrame) {
 | |
|       this._markupFrame = this.panelDoc.createElement("iframe");
 | |
|       this._markupFrame.setAttribute(
 | |
|         "aria-label",
 | |
|         INSPECTOR_L10N.getStr("inspector.panelLabel.markupView")
 | |
|       );
 | |
|       this._markupFrame.setAttribute("flex", "1");
 | |
|       // This is needed to enable tooltips inside the iframe document.
 | |
|       this._markupFrame.setAttribute("tooltip", "aHTMLTooltip");
 | |
| 
 | |
|       this._markupBox = this.panelDoc.getElementById("markup-box");
 | |
|       this._markupBox.style.visibility = "hidden";
 | |
|       this._markupBox.appendChild(this._markupFrame);
 | |
| 
 | |
|       const onMarkupFrameLoaded = new Promise(r =>
 | |
|         this._markupFrame.addEventListener("load", r, {
 | |
|           capture: true,
 | |
|           once: true,
 | |
|         })
 | |
|       );
 | |
| 
 | |
|       this._markupFrame.setAttribute("src", "markup/markup.xhtml");
 | |
| 
 | |
|       await onMarkupFrameLoaded;
 | |
|     }
 | |
| 
 | |
|     this._markupFrame.contentWindow.focus();
 | |
|     this._markupBox.style.visibility = "visible";
 | |
|     this.markup = new MarkupView(this, this._markupFrame, this._toolbox.win);
 | |
|     // TODO: We might be able to merge markuploaded, new-root and reloaded.
 | |
|     this.emitForTests("markuploaded");
 | |
| 
 | |
|     const onExpand = this.markup.expandNode(this.selection.nodeFront);
 | |
| 
 | |
|     // Restore the highlighter states prior to emitting "new-root".
 | |
|     if (this._highlighters) {
 | |
|       await Promise.all([
 | |
|         this.highlighters.restoreFlexboxState(),
 | |
|         this.highlighters.restoreGridState(),
 | |
|       ]);
 | |
|     }
 | |
|     this.emit("new-root");
 | |
| 
 | |
|     // Wait for full expand of the selected node in order to ensure
 | |
|     // the markup view is fully emitted before firing 'reloaded'.
 | |
|     // 'reloaded' is used to know when the panel is fully updated
 | |
|     // after a page reload.
 | |
|     await onExpand;
 | |
| 
 | |
|     this.emit("reloaded");
 | |
| 
 | |
|     // Record the time between new-root event and inspector fully loaded.
 | |
|     if (this._newRootStart) {
 | |
|       // Only log the timing when inspector is not destroyed and is in foreground.
 | |
|       if (this.toolbox && this.toolbox.currentToolId == "inspector") {
 | |
|         const delay = this.panelWin.performance.now() - this._newRootStart;
 | |
|         const telemetryKey = "DEVTOOLS_INSPECTOR_NEW_ROOT_TO_RELOAD_DELAY_MS";
 | |
|         const histogram = this.telemetry.getHistogramById(telemetryKey);
 | |
|         histogram.add(delay);
 | |
|       }
 | |
|       delete this._newRootStart;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   async initInspectorFront(targetFront) {
 | |
|     this.inspectorFront = await targetFront.getFront("inspector");
 | |
|     this.walker = this.inspectorFront.walker;
 | |
|   },
 | |
| 
 | |
|   get toolbox() {
 | |
|     return this._toolbox;
 | |
|   },
 | |
| 
 | |
|   get commands() {
 | |
|     return this._commands;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Get the list of InspectorFront instances that correspond to all of the inspectable
 | |
|    * targets in remote frames nested within the document inspected here, as well as the
 | |
|    * current InspectorFront instance.
 | |
|    *
 | |
|    * @return {Array} The list of InspectorFront instances.
 | |
|    */
 | |
|   async getAllInspectorFronts() {
 | |
|     return this.commands.targetCommand.getAllFronts(
 | |
|       [this.commands.targetCommand.TYPES.FRAME],
 | |
|       "inspector"
 | |
|     );
 | |
|   },
 | |
| 
 | |
|   get highlighters() {
 | |
|     if (!this._highlighters) {
 | |
|       this._highlighters = new HighlightersOverlay(this);
 | |
|     }
 | |
| 
 | |
|     return this._highlighters;
 | |
|   },
 | |
| 
 | |
|   get _3PanePrefName() {
 | |
|     // All other contexts: webextension and browser toolbox
 | |
|     // are considered as "chrome"
 | |
|     return this.commands.descriptorFront.isTabDescriptor
 | |
|       ? THREE_PANE_ENABLED_PREF
 | |
|       : THREE_PANE_CHROME_ENABLED_PREF;
 | |
|   },
 | |
| 
 | |
|   get is3PaneModeEnabled() {
 | |
|     if (!this._is3PaneModeEnabled) {
 | |
|       this._is3PaneModeEnabled = Services.prefs.getBoolPref(
 | |
|         this._3PanePrefName
 | |
|       );
 | |
|     }
 | |
|     return this._is3PaneModeEnabled;
 | |
|   },
 | |
| 
 | |
|   set is3PaneModeEnabled(value) {
 | |
|     this._is3PaneModeEnabled = value;
 | |
|     Services.prefs.setBoolPref(this._3PanePrefName, this._is3PaneModeEnabled);
 | |
|   },
 | |
| 
 | |
|   get search() {
 | |
|     if (!this._search) {
 | |
|       this._search = new InspectorSearch(
 | |
|         this,
 | |
|         this.searchBox,
 | |
|         this.searchClearButton
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     return this._search;
 | |
|   },
 | |
| 
 | |
|   get selection() {
 | |
|     return this.toolbox.selection;
 | |
|   },
 | |
| 
 | |
|   get cssProperties() {
 | |
|     return this._cssProperties.cssProperties;
 | |
|   },
 | |
| 
 | |
|   get fluentL10n() {
 | |
|     return this._fluentL10n;
 | |
|   },
 | |
| 
 | |
|   // Duration in milliseconds after which to hide the highlighter for the picked node.
 | |
|   // While testing, disable auto hiding to prevent intermittent test failures.
 | |
|   // Some tests are very slow. If the highlighter is hidden after a delay, the test may
 | |
|   // find itself midway through without a highlighter to test.
 | |
|   // This value is exposed on Inspector so individual tests can restore it when needed.
 | |
|   HIGHLIGHTER_AUTOHIDE_TIMER: flags.testing ? 0 : 1000,
 | |
| 
 | |
|   /**
 | |
|    * Handle promise rejections for various asynchronous actions, and only log errors if
 | |
|    * the inspector panel still exists.
 | |
|    * This is useful to silence useless errors that happen when the inspector is closed
 | |
|    * while still initializing (and making protocol requests).
 | |
|    */
 | |
|   _handleRejectionIfNotDestroyed(e) {
 | |
|     if (!this._destroyed) {
 | |
|       console.error(e);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   _onWillNavigate() {
 | |
|     this._defaultNode = null;
 | |
|     this.selection.setNodeFront(null);
 | |
|     if (this._highlighters) {
 | |
|       this._highlighters.hideAllHighlighters();
 | |
|     }
 | |
|     this._destroyMarkup();
 | |
|     this._pendingSelectionUnique = null;
 | |
|   },
 | |
| 
 | |
|   async _getCssProperties(targetFront) {
 | |
|     this._cssProperties = await targetFront.getFront("cssProperties");
 | |
|   },
 | |
| 
 | |
|   async _getAccessibilityFront(targetFront) {
 | |
|     this.accessibilityFront = await targetFront.getFront("accessibility");
 | |
|     return this.accessibilityFront;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Return a promise that will resolve to the default node for selection.
 | |
|    *
 | |
|    * @param {NodeFront} rootNodeFront
 | |
|    *        The current root node front for the top walker.
 | |
|    */
 | |
|   async _getDefaultNodeForSelection(rootNodeFront) {
 | |
|     if (this._defaultNode) {
 | |
|       return this._defaultNode;
 | |
|     }
 | |
| 
 | |
|     // Save the _pendingSelectionUnique on the current inspector instance.
 | |
|     const pendingSelectionUnique = Symbol("pending-selection");
 | |
|     this._pendingSelectionUnique = pendingSelectionUnique;
 | |
| 
 | |
|     if (this._pendingSelectionUnique !== pendingSelectionUnique) {
 | |
|       // If this method was called again while waiting, bail out.
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     const walker = rootNodeFront.walkerFront;
 | |
|     const cssSelectors = this.selectionCssSelectors;
 | |
|     // Try to find a default node using three strategies:
 | |
|     const defaultNodeSelectors = [
 | |
|       // - first try to match css selectors for the selection
 | |
|       () =>
 | |
|         cssSelectors.length
 | |
|           ? this.commands.inspectorCommand.findNodeFrontFromSelectors(
 | |
|               cssSelectors
 | |
|             )
 | |
|           : null,
 | |
|       // - otherwise try to get the "body" element
 | |
|       () => walker.querySelector(rootNodeFront, "body"),
 | |
|       // - finally get the documentElement element if nothing else worked.
 | |
|       () => walker.documentElement(),
 | |
|     ];
 | |
| 
 | |
|     // Try all default node selectors until a valid node is found.
 | |
|     for (const selector of defaultNodeSelectors) {
 | |
|       const node = await selector();
 | |
|       if (this._pendingSelectionUnique !== pendingSelectionUnique) {
 | |
|         // If this method was called again while waiting, bail out.
 | |
|         return null;
 | |
|       }
 | |
| 
 | |
|       if (node) {
 | |
|         this._defaultNode = node;
 | |
|         return node;
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     return null;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Top level target front getter.
 | |
|    */
 | |
|   get currentTarget() {
 | |
|     return this.commands.targetCommand.targetFront;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Hooks the searchbar to show result and auto completion suggestions.
 | |
|    */
 | |
|   setupSearchBox() {
 | |
|     this.searchBox = this.panelDoc.getElementById("inspector-searchbox");
 | |
|     this.searchClearButton = this.panelDoc.getElementById(
 | |
|       "inspector-searchinput-clear"
 | |
|     );
 | |
|     this.searchResultsContainer = this.panelDoc.getElementById(
 | |
|       "inspector-searchlabel-container"
 | |
|     );
 | |
|     this.searchResultsLabel = this.panelDoc.getElementById(
 | |
|       "inspector-searchlabel"
 | |
|     );
 | |
| 
 | |
|     this.searchBox.addEventListener("focus", this.listenForSearchEvents, {
 | |
|       once: true,
 | |
|     });
 | |
| 
 | |
|     this.createSearchBoxShortcuts();
 | |
|   },
 | |
| 
 | |
|   listenForSearchEvents() {
 | |
|     this.search.on("search-cleared", this._clearSearchResultsLabel);
 | |
|     this.search.on("search-result", this._updateSearchResultsLabel);
 | |
|   },
 | |
| 
 | |
|   createSearchBoxShortcuts() {
 | |
|     this.searchboxShortcuts = new KeyShortcuts({
 | |
|       window: this.panelDoc.defaultView,
 | |
|       // The inspector search shortcuts need to be available from everywhere in the
 | |
|       // inspector, and the inspector uses iframes (markupview, sidepanel webextensions).
 | |
|       // Use the chromeEventHandler as the target to catch events from all frames.
 | |
|       target: this.toolbox.getChromeEventHandler(),
 | |
|     });
 | |
|     const key = INSPECTOR_L10N.getStr("inspector.searchHTML.key");
 | |
|     this.searchboxShortcuts.on(key, event => {
 | |
|       // Prevent overriding same shortcut from the computed/rule views
 | |
|       if (
 | |
|         event.originalTarget.closest("#sidebar-panel-ruleview") ||
 | |
|         event.originalTarget.closest("#sidebar-panel-computedview")
 | |
|       ) {
 | |
|         return;
 | |
|       }
 | |
| 
 | |
|       const win = event.originalTarget.ownerGlobal;
 | |
|       // Check if the event is coming from an inspector window to avoid catching
 | |
|       // events from other panels. Note, we are testing both win and win.parent
 | |
|       // because the inspector uses iframes.
 | |
|       if (win === this.panelWin || win.parent === this.panelWin) {
 | |
|         event.preventDefault();
 | |
|         this.searchBox.focus();
 | |
|       }
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   get searchSuggestions() {
 | |
|     return this.search.autocompleter;
 | |
|   },
 | |
| 
 | |
|   _clearSearchResultsLabel(result) {
 | |
|     return this._updateSearchResultsLabel(result, true);
 | |
|   },
 | |
| 
 | |
|   _updateSearchResultsLabel(result, clear = false) {
 | |
|     let str = "";
 | |
|     if (!clear) {
 | |
|       if (result) {
 | |
|         str = INSPECTOR_L10N.getFormatStr(
 | |
|           "inspector.searchResultsCount2",
 | |
|           result.resultsIndex + 1,
 | |
|           result.resultsLength
 | |
|         );
 | |
|       } else {
 | |
|         str = INSPECTOR_L10N.getStr("inspector.searchResultsNone");
 | |
|       }
 | |
| 
 | |
|       this.searchResultsContainer.hidden = false;
 | |
|     } else {
 | |
|       this.searchResultsContainer.hidden = true;
 | |
|     }
 | |
| 
 | |
|     this.searchResultsLabel.textContent = str;
 | |
|   },
 | |
| 
 | |
|   get React() {
 | |
|     return this._toolbox.React;
 | |
|   },
 | |
| 
 | |
|   get ReactDOM() {
 | |
|     return this._toolbox.ReactDOM;
 | |
|   },
 | |
| 
 | |
|   get ReactRedux() {
 | |
|     return this._toolbox.ReactRedux;
 | |
|   },
 | |
| 
 | |
|   get browserRequire() {
 | |
|     return this._toolbox.browserRequire;
 | |
|   },
 | |
| 
 | |
|   get InspectorTabPanel() {
 | |
|     if (!this._InspectorTabPanel) {
 | |
|       this._InspectorTabPanel = this.React.createFactory(
 | |
|         this.browserRequire(
 | |
|           "devtools/client/inspector/components/InspectorTabPanel"
 | |
|         )
 | |
|       );
 | |
|     }
 | |
|     return this._InspectorTabPanel;
 | |
|   },
 | |
| 
 | |
|   get InspectorSplitBox() {
 | |
|     if (!this._InspectorSplitBox) {
 | |
|       this._InspectorSplitBox = this.React.createFactory(
 | |
|         this.browserRequire(
 | |
|           "devtools/client/shared/components/splitter/SplitBox"
 | |
|         )
 | |
|       );
 | |
|     }
 | |
|     return this._InspectorSplitBox;
 | |
|   },
 | |
| 
 | |
|   get TabBar() {
 | |
|     if (!this._TabBar) {
 | |
|       this._TabBar = this.React.createFactory(
 | |
|         this.browserRequire("devtools/client/shared/components/tabs/TabBar")
 | |
|       );
 | |
|     }
 | |
|     return this._TabBar;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Check if the inspector should use the landscape mode.
 | |
|    *
 | |
|    * @return {Boolean} true if the inspector should be in landscape mode.
 | |
|    */
 | |
|   useLandscapeMode() {
 | |
|     if (!this.panelDoc) {
 | |
|       return true;
 | |
|     }
 | |
| 
 | |
|     const splitterBox = this.panelDoc.getElementById("inspector-splitter-box");
 | |
|     const width = splitterBox.clientWidth;
 | |
| 
 | |
|     return this.is3PaneModeEnabled &&
 | |
|       (this.toolbox.hostType == Toolbox.HostType.LEFT ||
 | |
|         this.toolbox.hostType == Toolbox.HostType.RIGHT)
 | |
|       ? width > SIDE_PORTAIT_MODE_WIDTH_THRESHOLD
 | |
|       : width > PORTRAIT_MODE_WIDTH_THRESHOLD;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Build Splitter located between the main and side area of
 | |
|    * the Inspector panel.
 | |
|    */
 | |
|   setupSplitter() {
 | |
|     const { width, height, splitSidebarWidth } = this.getSidebarSize();
 | |
| 
 | |
|     this.sidebarSplitBoxRef = this.React.createRef();
 | |
| 
 | |
|     const splitter = this.InspectorSplitBox({
 | |
|       className: "inspector-sidebar-splitter",
 | |
|       initialWidth: width,
 | |
|       initialHeight: height,
 | |
|       minSize: "10%",
 | |
|       maxSize: "80%",
 | |
|       splitterSize: 1,
 | |
|       endPanelControl: true,
 | |
|       startPanel: this.InspectorTabPanel({
 | |
|         id: "inspector-main-content",
 | |
|       }),
 | |
|       endPanel: this.InspectorSplitBox({
 | |
|         initialWidth: splitSidebarWidth,
 | |
|         minSize: "225px",
 | |
|         maxSize: "80%",
 | |
|         splitterSize: this.is3PaneModeEnabled ? 1 : 0,
 | |
|         endPanelControl: this.is3PaneModeEnabled,
 | |
|         startPanel: this.InspectorTabPanel({
 | |
|           id: "inspector-rules-container",
 | |
|         }),
 | |
|         endPanel: this.InspectorTabPanel({
 | |
|           id: "inspector-sidebar-container",
 | |
|         }),
 | |
|         ref: this.sidebarSplitBoxRef,
 | |
|       }),
 | |
|       vert: this.useLandscapeMode(),
 | |
|       onControlledPanelResized: this.onSidebarResized,
 | |
|     });
 | |
| 
 | |
|     this.splitBox = this.ReactDOM.render(
 | |
|       splitter,
 | |
|       this.panelDoc.getElementById("inspector-splitter-box")
 | |
|     );
 | |
| 
 | |
|     this.panelWin.addEventListener("resize", this.onPanelWindowResize, true);
 | |
|   },
 | |
| 
 | |
|   async _onLazyPanelResize() {
 | |
|     // We can be called on a closed window or destroyed toolbox because of the deferred task.
 | |
|     if (
 | |
|       window.closed ||
 | |
|       this._destroyed ||
 | |
|       this._toolbox.currentToolId !== "inspector"
 | |
|     ) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this.splitBox.setState({ vert: this.useLandscapeMode() });
 | |
|     this.emit("inspector-resize");
 | |
|   },
 | |
| 
 | |
|   getSidebarSize() {
 | |
|     let width;
 | |
|     let height;
 | |
|     let splitSidebarWidth;
 | |
| 
 | |
|     // Initialize splitter size from preferences.
 | |
|     try {
 | |
|       width = Services.prefs.getIntPref("devtools.toolsidebar-width.inspector");
 | |
|       height = Services.prefs.getIntPref(
 | |
|         "devtools.toolsidebar-height.inspector"
 | |
|       );
 | |
|       splitSidebarWidth = Services.prefs.getIntPref(
 | |
|         "devtools.toolsidebar-width.inspector.splitsidebar"
 | |
|       );
 | |
|     } catch (e) {
 | |
|       // Set width and height of the splitter. Only one
 | |
|       // value is really useful at a time depending on the current
 | |
|       // orientation (vertical/horizontal).
 | |
|       // Having both is supported by the splitter component.
 | |
|       width = this.is3PaneModeEnabled
 | |
|         ? INITIAL_SIDEBAR_SIZE * 2
 | |
|         : INITIAL_SIDEBAR_SIZE;
 | |
|       height = INITIAL_SIDEBAR_SIZE;
 | |
|       splitSidebarWidth = INITIAL_SIDEBAR_SIZE;
 | |
|     }
 | |
| 
 | |
|     return { width, height, splitSidebarWidth };
 | |
|   },
 | |
| 
 | |
|   onSidebarHidden() {
 | |
|     // Store the current splitter size to preferences.
 | |
|     const state = this.splitBox.state;
 | |
|     Services.prefs.setIntPref(
 | |
|       "devtools.toolsidebar-width.inspector",
 | |
|       state.width
 | |
|     );
 | |
|     Services.prefs.setIntPref(
 | |
|       "devtools.toolsidebar-height.inspector",
 | |
|       state.height
 | |
|     );
 | |
|     Services.prefs.setIntPref(
 | |
|       "devtools.toolsidebar-width.inspector.splitsidebar",
 | |
|       this.sidebarSplitBoxRef.current.state.width
 | |
|     );
 | |
|   },
 | |
| 
 | |
|   onSidebarResized(width, height) {
 | |
|     this.toolbox.emit("inspector-sidebar-resized", { width, height });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns inspector tab that is active.
 | |
|    */
 | |
|   getActiveSidebar() {
 | |
|     return Services.prefs.getCharPref("devtools.inspector.activeSidebar");
 | |
|   },
 | |
| 
 | |
|   setActiveSidebar(toolId) {
 | |
|     Services.prefs.setCharPref("devtools.inspector.activeSidebar", toolId);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns tab that is explicitly selected by user.
 | |
|    */
 | |
|   getSelectedSidebar() {
 | |
|     return Services.prefs.getCharPref("devtools.inspector.selectedSidebar");
 | |
|   },
 | |
| 
 | |
|   setSelectedSidebar(toolId) {
 | |
|     Services.prefs.setCharPref("devtools.inspector.selectedSidebar", toolId);
 | |
|   },
 | |
| 
 | |
|   onSidebarSelect(toolId) {
 | |
|     // Save the currently selected sidebar panel
 | |
|     this.setSelectedSidebar(toolId);
 | |
|     this.setActiveSidebar(toolId);
 | |
| 
 | |
|     // Then forces the panel creation by calling getPanel
 | |
|     // (This allows lazy loading the panels only once we select them)
 | |
|     this.getPanel(toolId);
 | |
| 
 | |
|     this.toolbox.emit("inspector-sidebar-select", toolId);
 | |
|   },
 | |
| 
 | |
|   onSidebarShown() {
 | |
|     const { width, height, splitSidebarWidth } = this.getSidebarSize();
 | |
|     this.splitBox.setState({ width, height });
 | |
|     this.sidebarSplitBoxRef.current.setState({ width: splitSidebarWidth });
 | |
|   },
 | |
| 
 | |
|   async onSidebarToggle() {
 | |
|     this.is3PaneModeEnabled = !this.is3PaneModeEnabled;
 | |
|     await this.setupToolbar();
 | |
|     this.addRuleView({ skipQueue: true });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Sets the inspector sidebar split box state. Shows the splitter inside the sidebar
 | |
|    * split box, specifies the end panel control and resizes the split box width depending
 | |
|    * on the width of the toolbox.
 | |
|    */
 | |
|   setSidebarSplitBoxState() {
 | |
|     const toolboxWidth = this.panelDoc.getElementById("inspector-splitter-box")
 | |
|       .clientWidth;
 | |
| 
 | |
|     // Get the inspector sidebar's (right panel in horizontal mode or bottom panel in
 | |
|     // vertical mode) width.
 | |
|     const sidebarWidth = this.splitBox.state.width;
 | |
|     // This variable represents the width of the right panel in horizontal mode or
 | |
|     // bottom-right panel in vertical mode width in 3 pane mode.
 | |
|     let sidebarSplitboxWidth;
 | |
| 
 | |
|     if (this.useLandscapeMode()) {
 | |
|       // Whether or not doubling the inspector sidebar's (right panel in horizontal mode
 | |
|       // or bottom panel in vertical mode) width will be bigger than half of the
 | |
|       // toolbox's width.
 | |
|       const canDoubleSidebarWidth = sidebarWidth * 2 < toolboxWidth / 2;
 | |
| 
 | |
|       // Resize the main split box's end panel that contains the middle and right panel.
 | |
|       // Attempts to resize the main split box's end panel to be double the size of the
 | |
|       // existing sidebar's width when switching to 3 pane mode. However, if the middle
 | |
|       // and right panel's width together is greater than half of the toolbox's width,
 | |
|       // split all 3 panels to be equally sized by resizing the end panel to be 2/3 of
 | |
|       // the current toolbox's width.
 | |
|       this.splitBox.setState({
 | |
|         width: canDoubleSidebarWidth
 | |
|           ? sidebarWidth * 2
 | |
|           : (toolboxWidth * 2) / 3,
 | |
|       });
 | |
| 
 | |
|       // In landscape/horizontal mode, set the right panel back to its original
 | |
|       // inspector sidebar width if we can double the sidebar width. Otherwise, set
 | |
|       // the width of the right panel to be 1/3 of the toolbox's width since all 3
 | |
|       // panels will be equally sized.
 | |
|       sidebarSplitboxWidth = canDoubleSidebarWidth
 | |
|         ? sidebarWidth
 | |
|         : toolboxWidth / 3;
 | |
|     } else {
 | |
|       // In portrait/vertical mode, set the bottom-right panel to be 1/2 of the
 | |
|       // toolbox's width.
 | |
|       sidebarSplitboxWidth = toolboxWidth / 2;
 | |
|     }
 | |
| 
 | |
|     // Show the splitter inside the sidebar split box. Sets the width of the inspector
 | |
|     // sidebar and specify that the end (right in horizontal or bottom-right in
 | |
|     // vertical) panel of the sidebar split box should be controlled when resizing.
 | |
|     this.sidebarSplitBoxRef.current.setState({
 | |
|       endPanelControl: true,
 | |
|       splitterSize: 1,
 | |
|       width: sidebarSplitboxWidth,
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Adds the rule view to the middle (in landscape/horizontal mode) or bottom-left panel
 | |
|    * (in portrait/vertical mode) or inspector sidebar depending on whether or not it is 3
 | |
|    * pane mode. Rule view is selected when switching to 2 pane mode. Selected sidebar pref
 | |
|    * is used otherwise.
 | |
|    */
 | |
|   addRuleView({ skipQueue = false } = {}) {
 | |
|     const selectedSidebar = this.getSelectedSidebar();
 | |
|     const ruleViewSidebar = this.sidebarSplitBoxRef.current.startPanelContainer;
 | |
| 
 | |
|     if (this.is3PaneModeEnabled) {
 | |
|       // Convert to 3 pane mode by removing the rule view from the inspector sidebar
 | |
|       // and adding the rule view to the middle (in landscape/horizontal mode) or
 | |
|       // bottom-left (in portrait/vertical mode) panel.
 | |
|       ruleViewSidebar.style.display = "block";
 | |
| 
 | |
|       this.setSidebarSplitBoxState();
 | |
| 
 | |
|       // Force the rule view panel creation by calling getPanel
 | |
|       this.getPanel("ruleview");
 | |
| 
 | |
|       this.sidebar.removeTab("ruleview");
 | |
|       this.sidebar.select(selectedSidebar);
 | |
| 
 | |
|       this.ruleViewSideBar.addExistingTab(
 | |
|         "ruleview",
 | |
|         INSPECTOR_L10N.getStr("inspector.sidebar.ruleViewTitle"),
 | |
|         true
 | |
|       );
 | |
| 
 | |
|       this.ruleViewSideBar.show();
 | |
|     } else {
 | |
|       // When switching to 2 pane view, always set rule view as the active sidebar.
 | |
|       this.setActiveSidebar("ruleview");
 | |
|       // Removes the rule view from the 3 pane mode and adds the rule view to the main
 | |
|       // inspector sidebar.
 | |
|       ruleViewSidebar.style.display = "none";
 | |
| 
 | |
|       // Set the width of the split box (right panel in horziontal mode and bottom panel
 | |
|       // in vertical mode) to be the width of the inspector sidebar.
 | |
|       const splitterBox = this.panelDoc.getElementById(
 | |
|         "inspector-splitter-box"
 | |
|       );
 | |
|       this.splitBox.setState({
 | |
|         width: this.useLandscapeMode()
 | |
|           ? this.sidebarSplitBoxRef.current.state.width
 | |
|           : splitterBox.clientWidth,
 | |
|       });
 | |
| 
 | |
|       // Hide the splitter to prevent any drag events in the sidebar split box and
 | |
|       // specify that the end (right panel in horziontal mode or bottom panel in vertical
 | |
|       // mode) panel should be uncontrolled when resizing.
 | |
|       this.sidebarSplitBoxRef.current.setState({
 | |
|         endPanelControl: false,
 | |
|         splitterSize: 0,
 | |
|       });
 | |
| 
 | |
|       this.ruleViewSideBar.hide();
 | |
|       this.ruleViewSideBar.removeTab("ruleview");
 | |
| 
 | |
|       if (skipQueue) {
 | |
|         this.sidebar.addExistingTab(
 | |
|           "ruleview",
 | |
|           INSPECTOR_L10N.getStr("inspector.sidebar.ruleViewTitle"),
 | |
|           true,
 | |
|           0
 | |
|         );
 | |
|       } else {
 | |
|         this.sidebar.queueExistingTab(
 | |
|           "ruleview",
 | |
|           INSPECTOR_L10N.getStr("inspector.sidebar.ruleViewTitle"),
 | |
|           true,
 | |
|           0
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     // Adding or removing a tab from sidebar sets selectedSidebar by the active tab,
 | |
|     // which we should revert.
 | |
|     this.setSelectedSidebar(selectedSidebar);
 | |
| 
 | |
|     this.emit("ruleview-added");
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns a boolean indicating whether a sidebar panel instance exists.
 | |
|    */
 | |
|   hasPanel(id) {
 | |
|     return this._panels.has(id);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Lazily get and create panel instances displayed in the sidebar
 | |
|    */
 | |
|   getPanel(id) {
 | |
|     if (this._panels.has(id)) {
 | |
|       return this._panels.get(id);
 | |
|     }
 | |
| 
 | |
|     let panel;
 | |
|     switch (id) {
 | |
|       case "animationinspector":
 | |
|         const AnimationInspector = this.browserRequire(
 | |
|           "devtools/client/inspector/animation/animation"
 | |
|         );
 | |
|         panel = new AnimationInspector(this, this.panelWin);
 | |
|         break;
 | |
|       case "boxmodel":
 | |
|         // box-model isn't a panel on its own, it used to, now it is being used by
 | |
|         // the layout view which retrieves an instance via getPanel.
 | |
|         const BoxModel = require("resource://devtools/client/inspector/boxmodel/box-model.js");
 | |
|         panel = new BoxModel(this, this.panelWin);
 | |
|         break;
 | |
|       case "changesview":
 | |
|         const ChangesView = this.browserRequire(
 | |
|           "devtools/client/inspector/changes/ChangesView"
 | |
|         );
 | |
|         panel = new ChangesView(this, this.panelWin);
 | |
|         break;
 | |
|       case "compatibilityview":
 | |
|         const CompatibilityView = this.browserRequire(
 | |
|           "devtools/client/inspector/compatibility/CompatibilityView"
 | |
|         );
 | |
|         panel = new CompatibilityView(this, this.panelWin);
 | |
|         break;
 | |
|       case "computedview":
 | |
|         const { ComputedViewTool } = this.browserRequire(
 | |
|           "devtools/client/inspector/computed/computed"
 | |
|         );
 | |
|         panel = new ComputedViewTool(this, this.panelWin);
 | |
|         break;
 | |
|       case "fontinspector":
 | |
|         const FontInspector = this.browserRequire(
 | |
|           "devtools/client/inspector/fonts/fonts"
 | |
|         );
 | |
|         panel = new FontInspector(this, this.panelWin);
 | |
|         break;
 | |
|       case "layoutview":
 | |
|         const LayoutView = this.browserRequire(
 | |
|           "devtools/client/inspector/layout/layout"
 | |
|         );
 | |
|         panel = new LayoutView(this, this.panelWin);
 | |
|         break;
 | |
|       case "ruleview":
 | |
|         const {
 | |
|           RuleViewTool,
 | |
|         } = require("resource://devtools/client/inspector/rules/rules.js");
 | |
|         panel = new RuleViewTool(this, this.panelWin);
 | |
|         break;
 | |
|       default:
 | |
|         // This is a custom panel or a non lazy-loaded one.
 | |
|         return null;
 | |
|     }
 | |
| 
 | |
|     if (panel) {
 | |
|       this._panels.set(id, panel);
 | |
|     }
 | |
| 
 | |
|     return panel;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Build the sidebar.
 | |
|    */
 | |
|   setupSidebar() {
 | |
|     const sidebar = this.panelDoc.getElementById("inspector-sidebar");
 | |
|     const options = {
 | |
|       showAllTabsMenu: true,
 | |
|       allTabsMenuButtonTooltip: INSPECTOR_L10N.getStr(
 | |
|         "allTabsMenuButton.tooltip"
 | |
|       ),
 | |
|       sidebarToggleButton: {
 | |
|         collapsed: !this.is3PaneModeEnabled,
 | |
|         collapsePaneTitle: INSPECTOR_L10N.getStr("inspector.hideThreePaneMode"),
 | |
|         expandPaneTitle: INSPECTOR_L10N.getStr("inspector.showThreePaneMode"),
 | |
|         onClick: this.onSidebarToggle,
 | |
|       },
 | |
|     };
 | |
| 
 | |
|     this.sidebar = new ToolSidebar(sidebar, this, "inspector", options);
 | |
|     this.sidebar.on("select", this.onSidebarSelect);
 | |
| 
 | |
|     const ruleSideBar = this.panelDoc.getElementById("inspector-rules-sidebar");
 | |
|     this.ruleViewSideBar = new ToolSidebar(ruleSideBar, this, "inspector", {
 | |
|       hideTabstripe: true,
 | |
|     });
 | |
| 
 | |
|     // Append all side panels
 | |
|     this.addRuleView();
 | |
| 
 | |
|     // Inspector sidebar panels in order of appearance.
 | |
|     const sidebarPanels = [];
 | |
|     sidebarPanels.push({
 | |
|       id: "layoutview",
 | |
|       title: INSPECTOR_L10N.getStr("inspector.sidebar.layoutViewTitle2"),
 | |
|     });
 | |
| 
 | |
|     sidebarPanels.push({
 | |
|       id: "computedview",
 | |
|       title: INSPECTOR_L10N.getStr("inspector.sidebar.computedViewTitle"),
 | |
|     });
 | |
| 
 | |
|     sidebarPanels.push({
 | |
|       id: "changesview",
 | |
|       title: INSPECTOR_L10N.getStr("inspector.sidebar.changesViewTitle"),
 | |
|     });
 | |
| 
 | |
|     if (
 | |
|       Services.prefs.getBoolPref("devtools.inspector.compatibility.enabled")
 | |
|     ) {
 | |
|       sidebarPanels.push({
 | |
|         id: "compatibilityview",
 | |
|         title: INSPECTOR_L10N.getStr(
 | |
|           "inspector.sidebar.compatibilityViewTitle"
 | |
|         ),
 | |
|       });
 | |
|     }
 | |
| 
 | |
|     sidebarPanels.push({
 | |
|       id: "fontinspector",
 | |
|       title: INSPECTOR_L10N.getStr("inspector.sidebar.fontInspectorTitle"),
 | |
|     });
 | |
| 
 | |
|     sidebarPanels.push({
 | |
|       id: "animationinspector",
 | |
|       title: INSPECTOR_L10N.getStr("inspector.sidebar.animationInspectorTitle"),
 | |
|     });
 | |
| 
 | |
|     const defaultTab = this.getActiveSidebar();
 | |
| 
 | |
|     for (const { id, title } of sidebarPanels) {
 | |
|       // The Computed panel is not a React-based panel. We pick its element container from
 | |
|       // the DOM and wrap it in a React component (InspectorTabPanel) so it behaves like
 | |
|       // other panels when using the Inspector's tool sidebar.
 | |
|       if (id === "computedview") {
 | |
|         this.sidebar.queueExistingTab(id, title, defaultTab === id);
 | |
|       } else {
 | |
|         // When `panel` is a function, it is called when the tab should render. It is
 | |
|         // expected to return a React component to populate the tab's content area.
 | |
|         // Calling this method on-demand allows us to lazy-load the requested panel.
 | |
|         this.sidebar.queueTab(
 | |
|           id,
 | |
|           title,
 | |
|           {
 | |
|             props: {
 | |
|               id,
 | |
|               title,
 | |
|             },
 | |
|             panel: () => {
 | |
|               return this.getPanel(id).provider;
 | |
|             },
 | |
|           },
 | |
|           defaultTab === id
 | |
|         );
 | |
|       }
 | |
|     }
 | |
| 
 | |
|     this.sidebar.addAllQueuedTabs();
 | |
| 
 | |
|     // Persist splitter state in preferences.
 | |
|     this.sidebar.on("show", this.onSidebarShown);
 | |
|     this.sidebar.on("hide", this.onSidebarHidden);
 | |
|     this.sidebar.on("destroy", this.onSidebarHidden);
 | |
| 
 | |
|     this.sidebar.show();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Setup any extension sidebar already registered to the toolbox when the inspector.
 | |
|    * has been created for the first time.
 | |
|    */
 | |
|   setupExtensionSidebars() {
 | |
|     for (const [sidebarId, { title }] of this.toolbox
 | |
|       .inspectorExtensionSidebars) {
 | |
|       this.addExtensionSidebar(sidebarId, { title });
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Create a side-panel tab controlled by an extension
 | |
|    * using the devtools.panels.elements.createSidebarPane and sidebar object API
 | |
|    *
 | |
|    * @param {String} id
 | |
|    *        An unique id for the sidebar tab.
 | |
|    * @param {Object} options
 | |
|    * @param {String} options.title
 | |
|    *        The tab title
 | |
|    */
 | |
|   addExtensionSidebar(id, { title }) {
 | |
|     if (this._panels.has(id)) {
 | |
|       throw new Error(
 | |
|         `Cannot create an extension sidebar for the existent id: ${id}`
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     const extensionSidebar = new ExtensionSidebar(this, { id, title });
 | |
| 
 | |
|     // TODO(rpl): pass some extension metadata (e.g. extension name and icon) to customize
 | |
|     // the render of the extension title (e.g. use the icon in the sidebar and show the
 | |
|     // extension name in a tooltip).
 | |
|     this.addSidebarTab(id, title, extensionSidebar.provider, false);
 | |
| 
 | |
|     this._panels.set(id, extensionSidebar);
 | |
| 
 | |
|     // Emit the created ExtensionSidebar instance to the listeners registered
 | |
|     // on the toolbox by the "devtools.panels.elements" WebExtensions API.
 | |
|     this.toolbox.emit(`extension-sidebar-created-${id}`, extensionSidebar);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Remove and destroy a side-panel tab controlled by an extension (e.g. when the
 | |
|    * extension has been disable/uninstalled while the toolbox and inspector were
 | |
|    * still open).
 | |
|    *
 | |
|    * @param {String} id
 | |
|    *        The id of the sidebar tab to destroy.
 | |
|    */
 | |
|   removeExtensionSidebar(id) {
 | |
|     if (!this._panels.has(id)) {
 | |
|       throw new Error(`Unable to find a sidebar panel with id "${id}"`);
 | |
|     }
 | |
| 
 | |
|     const panel = this._panels.get(id);
 | |
| 
 | |
|     if (!(panel instanceof ExtensionSidebar)) {
 | |
|       throw new Error(
 | |
|         `The sidebar panel with id "${id}" is not an ExtensionSidebar`
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     this._panels.delete(id);
 | |
|     this.sidebar.removeTab(id);
 | |
|     panel.destroy();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Register a side-panel tab. This API can be used outside of
 | |
|    * DevTools (e.g. from an extension) as well as by DevTools
 | |
|    * code base.
 | |
|    *
 | |
|    * @param {string} tab uniq id
 | |
|    * @param {string} title tab title
 | |
|    * @param {React.Component} panel component. See `InspectorPanelTab` as an example.
 | |
|    * @param {boolean} selected true if the panel should be selected
 | |
|    */
 | |
|   addSidebarTab(id, title, panel, selected) {
 | |
|     this.sidebar.addTab(id, title, panel, selected);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Method to check whether the document is a HTML document and
 | |
|    * pickColorFromPage method is available or not.
 | |
|    *
 | |
|    * @return {Boolean} true if the eyedropper highlighter is supported by the current
 | |
|    *         document.
 | |
|    */
 | |
|   async supportsEyeDropper() {
 | |
|     try {
 | |
|       return await this.inspectorFront.supportsHighlighters();
 | |
|     } catch (e) {
 | |
|       console.error(e);
 | |
|       return false;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   async setupToolbar() {
 | |
|     this.teardownToolbar();
 | |
| 
 | |
|     // Setup the add-node button.
 | |
|     this.addNode = this.addNode.bind(this);
 | |
|     this.addNodeButton = this.panelDoc.getElementById(
 | |
|       "inspector-element-add-button"
 | |
|     );
 | |
|     this.addNodeButton.addEventListener("click", this.addNode);
 | |
| 
 | |
|     // Setup the eye-dropper icon if we're in an HTML document and we have actor support.
 | |
|     const canShowEyeDropper = await this.supportsEyeDropper();
 | |
| 
 | |
|     // Bail out if the inspector was destroyed in the meantime and panelDoc is no longer
 | |
|     // available.
 | |
|     if (!this.panelDoc) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (canShowEyeDropper) {
 | |
|       this.onEyeDropperDone = this.onEyeDropperDone.bind(this);
 | |
|       this.onEyeDropperButtonClicked = this.onEyeDropperButtonClicked.bind(
 | |
|         this
 | |
|       );
 | |
|       this.eyeDropperButton = this.panelDoc.getElementById(
 | |
|         "inspector-eyedropper-toggle"
 | |
|       );
 | |
|       this.eyeDropperButton.disabled = false;
 | |
|       this.eyeDropperButton.title = INSPECTOR_L10N.getStr(
 | |
|         "inspector.eyedropper.label"
 | |
|       );
 | |
|       this.eyeDropperButton.addEventListener(
 | |
|         "click",
 | |
|         this.onEyeDropperButtonClicked
 | |
|       );
 | |
|     } else {
 | |
|       const eyeDropperButton = this.panelDoc.getElementById(
 | |
|         "inspector-eyedropper-toggle"
 | |
|       );
 | |
|       eyeDropperButton.disabled = true;
 | |
|       eyeDropperButton.title = INSPECTOR_L10N.getStr(
 | |
|         "eyedropper.disabled.title"
 | |
|       );
 | |
|     }
 | |
| 
 | |
|     this.emit("inspector-toolbar-updated");
 | |
|   },
 | |
| 
 | |
|   teardownToolbar() {
 | |
|     if (this.addNodeButton) {
 | |
|       this.addNodeButton.removeEventListener("click", this.addNode);
 | |
|       this.addNodeButton = null;
 | |
|     }
 | |
| 
 | |
|     if (this.eyeDropperButton) {
 | |
|       this.eyeDropperButton.removeEventListener(
 | |
|         "click",
 | |
|         this.onEyeDropperButtonClicked
 | |
|       );
 | |
|       this.eyeDropperButton = null;
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   _selectionCssSelectors: null,
 | |
| 
 | |
|   /**
 | |
|    * Set the array of CSS selectors for the currently selected node.
 | |
|    * We use an array of selectors in case the element is in iframes.
 | |
|    * Will store the current target url along with it to allow pre-selection at
 | |
|    * reload
 | |
|    */
 | |
|   set selectionCssSelectors(cssSelectors = []) {
 | |
|     if (this._destroyed) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this._selectionCssSelectors = {
 | |
|       selectors: cssSelectors,
 | |
|       url: this.currentTarget.url,
 | |
|     };
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Get the CSS selectors for the current selection if any, that is, if a node
 | |
|    * is actually selected and that node has been selected while on the same url
 | |
|    */
 | |
|   get selectionCssSelectors() {
 | |
|     if (
 | |
|       this._selectionCssSelectors &&
 | |
|       this._selectionCssSelectors.url === this.currentTarget.url
 | |
|     ) {
 | |
|       return this._selectionCssSelectors.selectors;
 | |
|     }
 | |
|     return [];
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * On any new selection made by the user, store the array of css selectors
 | |
|    * of the selected node so it can be restored after reload of the same page
 | |
|    */
 | |
|   updateSelectionCssSelectors() {
 | |
|     if (!this.selection.isElementNode()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this.commands.inspectorCommand
 | |
|       .getNodeFrontSelectorsFromTopDocument(this.selection.nodeFront)
 | |
|       .then(selectors => {
 | |
|         this.selectionCssSelectors = selectors;
 | |
|         // emit an event so tests relying on the property being set can properly wait
 | |
|         // for it.
 | |
|         this.emitForTests("selection-css-selectors-updated", selectors);
 | |
|       }, this._handleRejectionIfNotDestroyed);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Can a new HTML element be inserted into the currently selected element?
 | |
|    * @return {Boolean}
 | |
|    */
 | |
|   canAddHTMLChild() {
 | |
|     const selection = this.selection;
 | |
| 
 | |
|     // Don't allow to insert an element into these elements. This should only
 | |
|     // contain elements where walker.insertAdjacentHTML has no effect.
 | |
|     const invalidTagNames = ["html", "iframe"];
 | |
| 
 | |
|     return (
 | |
|       selection.isHTMLNode() &&
 | |
|       selection.isElementNode() &&
 | |
|       !selection.isPseudoElementNode() &&
 | |
|       !selection.isAnonymousNode() &&
 | |
|       !invalidTagNames.includes(selection.nodeFront.nodeName.toLowerCase())
 | |
|     );
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Update the state of the add button in the toolbar depending on the current selection.
 | |
|    */
 | |
|   updateAddElementButton() {
 | |
|     const btn = this.panelDoc.getElementById("inspector-element-add-button");
 | |
|     if (this.canAddHTMLChild()) {
 | |
|       btn.removeAttribute("disabled");
 | |
|     } else {
 | |
|       btn.setAttribute("disabled", "true");
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Handler for the "host-changed" event from the toolbox. Resets the inspector
 | |
|    * sidebar sizes when the toolbox host type changes.
 | |
|    */
 | |
|   async onHostChanged() {
 | |
|     // Eagerly call our resize handling code to process the fact that we
 | |
|     // switched hosts. If we don't do this, we'll wait for resize events + 200ms
 | |
|     // to have passed, which causes the old layout to noticeably show up in the
 | |
|     // new host, followed by the updated one.
 | |
|     await this._onLazyPanelResize();
 | |
|     // Note that we may have been destroyed by now, especially in tests, so we
 | |
|     // need to check if that's happened before touching anything else.
 | |
|     if (!this.currentTarget || !this.is3PaneModeEnabled) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // When changing hosts, the toolbox chromeEventHandler might change, for instance when
 | |
|     // switching from docked to window hosts. Recreate the searchbox shortcuts.
 | |
|     this.searchboxShortcuts.destroy();
 | |
|     this.createSearchBoxShortcuts();
 | |
| 
 | |
|     this.setSidebarSplitBoxState();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * When a new node is selected.
 | |
|    */
 | |
|   onNewSelection(value, reason) {
 | |
|     if (reason === "selection-destroy") {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     this.updateAddElementButton();
 | |
|     this.updateSelectionCssSelectors();
 | |
|     this.trackReflowsInSelection();
 | |
| 
 | |
|     const selfUpdate = this.updating("inspector-panel");
 | |
|     executeSoon(() => {
 | |
|       try {
 | |
|         selfUpdate(this.selection.nodeFront);
 | |
|         this.telemetry.scalarAdd(TELEMETRY_SCALAR_NODE_SELECTION_COUNT, 1);
 | |
|       } catch (ex) {
 | |
|         console.error(ex);
 | |
|       }
 | |
|     });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Starts listening for reflows in the targetFront of the currently selected nodeFront.
 | |
|    */
 | |
|   async trackReflowsInSelection() {
 | |
|     this.untrackReflowsInSelection();
 | |
|     if (!this.selection.nodeFront) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     if (this._destroyed) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     try {
 | |
|       await this.commands.resourceCommand.watchResources(
 | |
|         [this.commands.resourceCommand.TYPES.REFLOW],
 | |
|         {
 | |
|           onAvailable: this.onReflowInSelection,
 | |
|         }
 | |
|       );
 | |
|     } catch (e) {
 | |
|       // it can happen that watchResources fails as the client closes while we're processing
 | |
|       // some asynchronous call.
 | |
|       // In order to still get valid exceptions, we re-throw the exception if the inspector
 | |
|       // isn't destroyed.
 | |
|       if (!this._destroyed) {
 | |
|         throw e;
 | |
|       }
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Stops listening for reflows.
 | |
|    */
 | |
|   untrackReflowsInSelection() {
 | |
|     this.commands.resourceCommand.unwatchResources(
 | |
|       [this.commands.resourceCommand.TYPES.REFLOW],
 | |
|       {
 | |
|         onAvailable: this.onReflowInSelection,
 | |
|       }
 | |
|     );
 | |
|   },
 | |
| 
 | |
|   onReflowInSelection() {
 | |
|     // This event will be fired whenever a reflow is detected in the target front of the
 | |
|     // selected node front (so when a reflow is detected inside any of the windows that
 | |
|     // belong to the BrowsingContext when the currently selected node lives).
 | |
|     this.emit("reflow-in-selected-target");
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Delay the "inspector-updated" notification while a tool
 | |
|    * is updating itself.  Returns a function that must be
 | |
|    * invoked when the tool is done updating with the node
 | |
|    * that the tool is viewing.
 | |
|    */
 | |
|   updating(name) {
 | |
|     if (
 | |
|       this._updateProgress &&
 | |
|       this._updateProgress.node != this.selection.nodeFront
 | |
|     ) {
 | |
|       this.cancelUpdate();
 | |
|     }
 | |
| 
 | |
|     if (!this._updateProgress) {
 | |
|       // Start an update in progress.
 | |
|       const self = this;
 | |
|       this._updateProgress = {
 | |
|         node: this.selection.nodeFront,
 | |
|         outstanding: new Set(),
 | |
|         checkDone() {
 | |
|           if (this !== self._updateProgress) {
 | |
|             return;
 | |
|           }
 | |
|           // Cancel update if there is no `selection` anymore.
 | |
|           // It can happen if the inspector panel is already destroyed.
 | |
|           if (!self.selection || this.node !== self.selection.nodeFront) {
 | |
|             self.cancelUpdate();
 | |
|             return;
 | |
|           }
 | |
|           if (this.outstanding.size !== 0) {
 | |
|             return;
 | |
|           }
 | |
| 
 | |
|           self._updateProgress = null;
 | |
|           self.emit("inspector-updated", name);
 | |
|         },
 | |
|       };
 | |
|     }
 | |
| 
 | |
|     const progress = this._updateProgress;
 | |
|     const done = function() {
 | |
|       progress.outstanding.delete(done);
 | |
|       progress.checkDone();
 | |
|     };
 | |
|     progress.outstanding.add(done);
 | |
|     return done;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Cancel notification of inspector updates.
 | |
|    */
 | |
|   cancelUpdate() {
 | |
|     this._updateProgress = null;
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * When a node is deleted, select its parent node or the defaultNode if no
 | |
|    * parent is found (may happen when deleting an iframe inside which the
 | |
|    * node was selected).
 | |
|    */
 | |
|   onDetached(parentNode) {
 | |
|     this.breadcrumbs.cutAfter(this.breadcrumbs.indexOf(parentNode));
 | |
|     const nodeFront = parentNode ? parentNode : this._defaultNode;
 | |
|     this.selection.setNodeFront(nodeFront, { reason: "detached" });
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Destroy the inspector.
 | |
|    */
 | |
|   destroy() {
 | |
|     if (this._destroyed) {
 | |
|       return;
 | |
|     }
 | |
|     this._destroyed = true;
 | |
| 
 | |
|     this.cancelUpdate();
 | |
| 
 | |
|     this.panelWin.removeEventListener("resize", this.onPanelWindowResize, true);
 | |
|     this.selection.off("new-node-front", this.onNewSelection);
 | |
|     this.selection.off("detached-front", this.onDetached);
 | |
|     this.toolbox.nodePicker.off("picker-node-canceled", this.onPickerCanceled);
 | |
|     this.toolbox.nodePicker.off("picker-node-hovered", this.onPickerHovered);
 | |
|     this.toolbox.nodePicker.off("picker-node-picked", this.onPickerPicked);
 | |
| 
 | |
|     // Destroy the sidebar first as it may unregister stuff
 | |
|     // and still use random attributes on inspector and layout panel
 | |
|     this.sidebar.destroy();
 | |
|     // Unregister sidebar listener *after* destroying it
 | |
|     // in order to process its destroy event and save sidebar sizes
 | |
|     this.sidebar.off("select", this.onSidebarSelect);
 | |
|     this.sidebar.off("show", this.onSidebarShown);
 | |
|     this.sidebar.off("hide", this.onSidebarHidden);
 | |
|     this.sidebar.off("destroy", this.onSidebarHidden);
 | |
| 
 | |
|     for (const [, panel] of this._panels) {
 | |
|       panel.destroy();
 | |
|     }
 | |
|     this._panels.clear();
 | |
| 
 | |
|     if (this._highlighters) {
 | |
|       this._highlighters.destroy();
 | |
|     }
 | |
| 
 | |
|     if (this._search) {
 | |
|       this._search.destroy();
 | |
|       this._search = null;
 | |
|     }
 | |
| 
 | |
|     this.ruleViewSideBar.destroy();
 | |
|     this.ruleViewSideBar = null;
 | |
| 
 | |
|     this._destroyMarkup();
 | |
| 
 | |
|     this.teardownToolbar();
 | |
| 
 | |
|     this.breadcrumbs.destroy();
 | |
|     this.styleChangeTracker.destroy();
 | |
|     this.searchboxShortcuts.destroy();
 | |
|     this.searchboxShortcuts = null;
 | |
| 
 | |
|     this.commands.targetCommand.unwatchTargets({
 | |
|       types: [this.commands.targetCommand.TYPES.FRAME],
 | |
|       onAvailable: this._onTargetAvailable,
 | |
|       onSelected: this._onTargetSelected,
 | |
|       onDestroyed: this._onTargetDestroyed,
 | |
|     });
 | |
|     const { resourceCommand } = this.toolbox;
 | |
|     resourceCommand.unwatchResources(
 | |
|       [
 | |
|         resourceCommand.TYPES.ROOT_NODE,
 | |
|         resourceCommand.TYPES.CSS_CHANGE,
 | |
|         resourceCommand.TYPES.DOCUMENT_EVENT,
 | |
|       ],
 | |
|       { onAvailable: this.onResourceAvailable }
 | |
|     );
 | |
|     this.untrackReflowsInSelection();
 | |
| 
 | |
|     this._InspectorTabPanel = null;
 | |
|     this._TabBar = null;
 | |
|     this._InspectorSplitBox = null;
 | |
|     this.sidebarSplitBoxRef = null;
 | |
|     // Note that we do not unmount inspector-splitter-box
 | |
|     // as it regresses inspector closing performance while not releasing
 | |
|     // any object (bug 1729925)
 | |
|     this.splitBox = null;
 | |
| 
 | |
|     this._is3PaneModeEnabled = null;
 | |
|     this._markupBox = null;
 | |
|     this._markupFrame = null;
 | |
|     this._toolbox = null;
 | |
|     this._commands = null;
 | |
|     this.breadcrumbs = null;
 | |
|     this.inspectorFront = null;
 | |
|     this._cssProperties = null;
 | |
|     this.accessibilityFront = null;
 | |
|     this._highlighters = null;
 | |
|     this.walker = null;
 | |
|     this._defaultNode = null;
 | |
|     this.panelDoc = null;
 | |
|     this.panelWin.inspector = null;
 | |
|     this.panelWin = null;
 | |
|     this.resultsLength = null;
 | |
|     this.searchBox.removeEventListener("focus", this.listenForSearchEvents);
 | |
|     this.searchBox = null;
 | |
|     this.show3PaneTooltip = null;
 | |
|     this.sidebar = null;
 | |
|     this.store = null;
 | |
|     this.telemetry = null;
 | |
|   },
 | |
| 
 | |
|   _destroyMarkup() {
 | |
|     if (this.markup) {
 | |
|       this.markup.destroy();
 | |
|       this.markup = null;
 | |
|     }
 | |
| 
 | |
|     if (this._markupBox) {
 | |
|       this._markupBox.style.visibility = "hidden";
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   onEyeDropperButtonClicked() {
 | |
|     this.eyeDropperButton.classList.contains("checked")
 | |
|       ? this.hideEyeDropper()
 | |
|       : this.showEyeDropper();
 | |
|   },
 | |
| 
 | |
|   startEyeDropperListeners() {
 | |
|     this.toolbox.tellRDMAboutPickerState(true, PICKER_TYPES.EYEDROPPER);
 | |
|     this.inspectorFront.once("color-pick-canceled", this.onEyeDropperDone);
 | |
|     this.inspectorFront.once("color-picked", this.onEyeDropperDone);
 | |
|     this.once("new-root", this.onEyeDropperDone);
 | |
|   },
 | |
| 
 | |
|   stopEyeDropperListeners() {
 | |
|     this.toolbox.tellRDMAboutPickerState(false, PICKER_TYPES.EYEDROPPER);
 | |
|     this.inspectorFront.off("color-pick-canceled", this.onEyeDropperDone);
 | |
|     this.inspectorFront.off("color-picked", this.onEyeDropperDone);
 | |
|     this.off("new-root", this.onEyeDropperDone);
 | |
|   },
 | |
| 
 | |
|   onEyeDropperDone() {
 | |
|     this.eyeDropperButton.classList.remove("checked");
 | |
|     this.stopEyeDropperListeners();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Show the eyedropper on the page.
 | |
|    * @return {Promise} resolves when the eyedropper is visible.
 | |
|    */
 | |
|   showEyeDropper() {
 | |
|     // The eyedropper button doesn't exist, most probably because the actor doesn't
 | |
|     // support the pickColorFromPage, or because the page isn't HTML.
 | |
|     if (!this.eyeDropperButton) {
 | |
|       return null;
 | |
|     }
 | |
|     // turn off node picker when color picker is starting
 | |
|     this.toolbox.nodePicker.stop({ canceled: true }).catch(console.error);
 | |
|     this.telemetry.scalarSet(TELEMETRY_EYEDROPPER_OPENED, 1);
 | |
|     this.eyeDropperButton.classList.add("checked");
 | |
|     this.startEyeDropperListeners();
 | |
|     return this.inspectorFront
 | |
|       .pickColorFromPage({ copyOnSelect: true })
 | |
|       .catch(console.error);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Hide the eyedropper.
 | |
|    * @return {Promise} resolves when the eyedropper is hidden.
 | |
|    */
 | |
|   hideEyeDropper() {
 | |
|     // The eyedropper button doesn't exist, most probably  because the page isn't HTML.
 | |
|     if (!this.eyeDropperButton) {
 | |
|       return null;
 | |
|     }
 | |
| 
 | |
|     this.eyeDropperButton.classList.remove("checked");
 | |
|     this.stopEyeDropperListeners();
 | |
|     return this.inspectorFront.cancelPickColorFromPage().catch(console.error);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Create a new node as the last child of the current selection, expand the
 | |
|    * parent and select the new node.
 | |
|    */
 | |
|   async addNode() {
 | |
|     if (!this.canAddHTMLChild()) {
 | |
|       return;
 | |
|     }
 | |
| 
 | |
|     // turn off node picker when add node is triggered
 | |
|     this.toolbox.nodePicker.stop({ canceled: true });
 | |
| 
 | |
|     // turn off color picker when add node is triggered
 | |
|     this.hideEyeDropper();
 | |
| 
 | |
|     const nodeFront = this.selection.nodeFront;
 | |
|     const html = "<div></div>";
 | |
| 
 | |
|     // Insert the html and expect a childList markup mutation.
 | |
|     const onMutations = this.once("markupmutation");
 | |
|     await nodeFront.walkerFront.insertAdjacentHTML(
 | |
|       this.selection.nodeFront,
 | |
|       "beforeEnd",
 | |
|       html
 | |
|     );
 | |
|     await onMutations;
 | |
| 
 | |
|     // Expand the parent node.
 | |
|     this.markup.expandNode(nodeFront);
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Toggle a pseudo class.
 | |
|    */
 | |
|   togglePseudoClass(pseudo) {
 | |
|     if (this.selection.isElementNode()) {
 | |
|       const node = this.selection.nodeFront;
 | |
|       if (node.hasPseudoClassLock(pseudo)) {
 | |
|         return node.walkerFront.removePseudoClassLock(node, pseudo, {
 | |
|           parents: true,
 | |
|         });
 | |
|       }
 | |
| 
 | |
|       const hierarchical = pseudo == ":hover" || pseudo == ":active";
 | |
|       return node.walkerFront.addPseudoClassLock(node, pseudo, {
 | |
|         parents: hierarchical,
 | |
|       });
 | |
|     }
 | |
|     return Promise.resolve();
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Initiate screenshot command on selected node.
 | |
|    */
 | |
|   async screenshotNode() {
 | |
|     // Bug 1332936 - it's possible to call `screenshotNode` while the BoxModel highlighter
 | |
|     // is still visible, therefore showing it in the picture.
 | |
|     // Note that other highlighters will still be visible. See Bug 1663881
 | |
|     await this.highlighters.hideHighlighterType(
 | |
|       this.highlighters.TYPES.BOXMODEL
 | |
|     );
 | |
| 
 | |
|     const clipboardEnabled = Services.prefs.getBoolPref(
 | |
|       "devtools.screenshot.clipboard.enabled"
 | |
|     );
 | |
|     const args = {
 | |
|       file: !clipboardEnabled,
 | |
|       nodeActorID: this.selection.nodeFront.actorID,
 | |
|       clipboard: clipboardEnabled,
 | |
|     };
 | |
| 
 | |
|     const messages = await captureAndSaveScreenshot(
 | |
|       this.selection.nodeFront.targetFront,
 | |
|       this.panelWin,
 | |
|       args
 | |
|     );
 | |
|     const notificationBox = this.toolbox.getNotificationBox();
 | |
|     const priorityMap = {
 | |
|       error: notificationBox.PRIORITY_CRITICAL_HIGH,
 | |
|       warn: notificationBox.PRIORITY_WARNING_HIGH,
 | |
|     };
 | |
|     for (const { text, level } of messages) {
 | |
|       // captureAndSaveScreenshot returns "saved" messages, that indicate where the
 | |
|       // screenshot was saved. We don't want to display them as the download UI can be
 | |
|       // used to open the file.
 | |
|       if (level !== "warn" && level !== "error") {
 | |
|         continue;
 | |
|       }
 | |
|       notificationBox.appendNotification(text, null, null, priorityMap[level]);
 | |
|     }
 | |
|   },
 | |
| 
 | |
|   /**
 | |
|    * Returns an object containing the shared handler functions used in React components.
 | |
|    */
 | |
|   getCommonComponentProps() {
 | |
|     return {
 | |
|       setSelectedNode: this.selection.setNodeFront,
 | |
|     };
 | |
|   },
 | |
| 
 | |
|   onPickerCanceled() {
 | |
|     this.highlighters.hideHighlighterType(this.highlighters.TYPES.BOXMODEL);
 | |
|   },
 | |
| 
 | |
|   onPickerHovered(nodeFront) {
 | |
|     this.highlighters.showHighlighterTypeForNode(
 | |
|       this.highlighters.TYPES.BOXMODEL,
 | |
|       nodeFront
 | |
|     );
 | |
|   },
 | |
| 
 | |
|   onPickerPicked(nodeFront) {
 | |
|     if (this.toolbox.isDebugTargetFenix()) {
 | |
|       // When debugging a phone, as we don't have the "hover overlay", we want to provide
 | |
|       // feedback to the user so they know where they tapped
 | |
|       this.highlighters.showHighlighterTypeForNode(
 | |
|         this.highlighters.TYPES.BOXMODEL,
 | |
|         nodeFront,
 | |
|         { duration: this.HIGHLIGHTER_AUTOHIDE_TIMER }
 | |
|       );
 | |
|       return;
 | |
|     }
 | |
|     this.highlighters.hideHighlighterType(this.highlighters.TYPES.BOXMODEL);
 | |
|   },
 | |
| 
 | |
|   async inspectNodeActor(nodeGrip, reason) {
 | |
|     const nodeFront = await this.inspectorFront.getNodeFrontFromNodeGrip(
 | |
|       nodeGrip
 | |
|     );
 | |
|     if (!nodeFront) {
 | |
|       console.error(
 | |
|         "The object cannot be linked to the inspector, the " +
 | |
|           "corresponding nodeFront could not be found."
 | |
|       );
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     const isAttached = await this.walker.isInDOMTree(nodeFront);
 | |
|     if (!isAttached) {
 | |
|       console.error("Selected DOMNode is not attached to the document tree.");
 | |
|       return false;
 | |
|     }
 | |
| 
 | |
|     await this.selection.setNodeFront(nodeFront, { reason });
 | |
|     return true;
 | |
|   },
 | |
| };
 | |
| 
 | |
| exports.Inspector = Inspector;
 |