/* This Source Code Form is subject to the terms of the Mozilla Public * License, v. 2.0. If a copy of the MPL was not distributed with this
* file, You can obtain one at http://mozilla.org/MPL/2.0/. */
"use strict";
/** * This is the main module loaded in Firefox desktop that handles browser * windows and coordinates devtools around each window. * * This module is loaded lazily by devtools-clhandler.js, once the first * browser window is ready (i.e. fired browser-delayed-startup-finished event)
**/
/** * gDevToolsBrowser exposes functions to connect the gDevTools instance with a * Firefox instance.
*/ var gDevToolsBrowser = (exports.gDevToolsBrowser = { /** * A record of the windows whose menus we altered, so we can undo the changes * as the window is closed
*/
_trackedBrowserWindows: new Set(),
/** * WeakMap keeping track of the devtools-browser stylesheets loaded in the various * tracked windows.
*/
_browserStyleSheets: new WeakMap(),
/** * This function is for the benefit of Tools:DevToolbox in * browser/base/content/browser-sets.inc and should not be used outside * of there
*/ // used by browser-sets.inc, command
toggleToolboxCommand(gBrowser, startTime) { const toolbox = gDevTools.getToolboxForTab(gBrowser.selectedTab);
// If a toolbox exists, using toggle from the Main window : // - should close a docked toolbox // - should focus a windowed toolbox const isDocked = toolbox && toolbox.hostType != Toolbox.HostType.WINDOW; if (isDocked) {
gDevTools.closeToolboxForTab(gBrowser.selectedTab);
} else {
gDevTools.showToolboxForTab(gBrowser.selectedTab, { startTime });
}
},
/** * This function ensures the right commands are enabled in a window, * depending on their relevant prefs. It gets run when a window is registered, * or when any of the devtools prefs change.
*/
updateCommandAvailability(win) { const doc = win.document;
/** * This function makes sure that the "devtoolstheme" attribute is set on the * browser window to make it possible to change colors on elements in the * browser (like the splitter between the toolbox and web content).
*/
updateDevtoolsThemeAttribute(win) { // Set an attribute on root element of each window to make it possible // to change colors based on the selected devtools theme.
let devtoolsTheme = getTheme(); if (devtoolsTheme != "dark") {
devtoolsTheme = "light";
}
win.document.documentElement.setAttribute("devtoolstheme", devtoolsTheme);
},
observe(subject, topic, prefName) { switch (topic) { case"browser-delayed-startup-finished": this._registerBrowserWindow(subject); break; case"nsPref:changed": if (prefName.endsWith("enabled")) { for (const win of this._trackedBrowserWindows) { this.updateCommandAvailability(win);
}
} break; case"quit-application":
gDevToolsBrowser.destroy({ shuttingDown: true }); break; case"devtools:loader:destroy": // This event is fired when the devtools loader unloads, which happens // only when the add-on workflow ask devtools to be reloaded. if (subject.wrappedJSObject == require("@loader/unload")) {
gDevToolsBrowser.destroy({ shuttingDown: false });
} break;
}
},
_observersRegistered: false,
/** * This function is for the benefit of Tools:{toolId} commands, * triggered from the WebDeveloper menu and keyboard shortcuts. * * selectToolCommand's behavior: * - if the current page is about:devtools-toolbox * we select the targeted tool * - if the toolbox is closed, * we open the toolbox and select the tool * - if the toolbox is open, and the targeted tool is not selected, * we select it * - if the toolbox is open, and the targeted tool is selected, * and the host is NOT a window, we close the toolbox * - if the toolbox is open, and the targeted tool is selected, * and the host is a window, we raise the toolbox window * * Used when: - registering a new tool * - new xul window, to add menu items
*/
async selectToolCommand(win, toolId, startTime) { if (gDevToolsBrowser._isAboutDevtoolsToolbox(win)) { const toolbox = gDevToolsBrowser._getAboutDevtoolsToolbox(win);
await toolbox.selectTool(toolId, "key_shortcut"); return;
}
/** * Called by devtools/client/devtools-startup.js when a key shortcut is pressed * * @param {Window} window * The top level browser window from which the key shortcut is pressed. * @param {Object} key * Key object describing the key shortcut being pressed. It comes * from devtools-startup.js's KeyShortcuts array. The useful fields here * are: * - `toolId` used to identify a toolbox's panel like inspector or webconsole, * - `id` used to identify any other key shortcuts like about:debugging * @param {Number} startTime * Optional, indicates the time at which the key event fired. This is a * `Cu.now()` timing.
*/
async onKeyShortcut(window, key, startTime) { // Avoid to open devtools when the about:devtools-toolbox page is showing // on the window now. if (
gDevToolsBrowser._isAboutDevtoolsToolbox(window) &&
(key.id === "toggleToolbox" || key.id === "toggleToolboxF12")
) { return;
}
// If this is a toolbox's panel key shortcut, delegate to selectToolCommand if (key.toolId) {
await gDevToolsBrowser.selectToolCommand(window, key.toolId, startTime); return;
} // Otherwise implement all other key shortcuts individually here switch (key.id) { case"toggleToolbox":
gDevToolsBrowser.toggleToolboxCommand(window.gBrowser, startTime); break; case"toggleToolboxF12": if (Services.prefs.getBoolPref(DEVTOOLS_F12_ENABLED_PREF, true)) {
gDevToolsBrowser.toggleToolboxCommand(window.gBrowser, startTime);
} break; case"browserToolbox":
lazy.BrowserToolboxLauncher.init(); break; case"browserConsole": const {
BrowserConsoleManager,
} = require("resource://devtools/client/webconsole/browser-console-manager.js");
BrowserConsoleManager.openBrowserConsoleOrFocus(); break; case"responsiveDesignMode":
ResponsiveUIManager.toggle(window, window.gBrowser.selectedTab, {
trigger: "shortcut",
}); break; case"javascriptTracingToggle": const toolbox = gDevTools.getToolboxForTab(window.gBrowser.selectedTab); if (!toolbox) { break;
}
await toolbox.commands.tracerCommand.toggle(); break;
}
},
/** * Open a tab on "about:debugging", optionally pre-select a given tab.
*/ // Used by browser-sets.inc, command
openAboutDebugging(gBrowser, hash) { const url = "about:debugging" + (hash ? "#" + hash : "");
gBrowser.selectedTab = gBrowser.addTrustedTab(url);
},
/** * Add the devtools-browser stylesheet to browser window's document. Returns a promise. * * @param {Window} win * The window on which the stylesheet should be added. * @return {Promise} promise that resolves when the stylesheet is loaded (or rejects * if it fails to load).
*/
loadBrowserStyleSheet(win) { if (this._browserStyleSheets.has(win)) { return Promise.resolve();
}
/** * Add this DevTools's presence to a browser window's document * * @param {HTMLDocument} doc * The document to which devtools should be hooked to.
*/
_registerBrowserWindow(win) { if (gDevToolsBrowser._trackedBrowserWindows.has(win)) { return;
} if (!win.document.getElementById("menuWebDeveloperPopup")) { // Menus etc. set up here are browser specific. return;
}
gDevToolsBrowser._trackedBrowserWindows.add(win);
BrowserMenus.addMenus(win.document);
_onThemeChanged() { for (const win of this._trackedBrowserWindows) { this.updateDevtoolsThemeAttribute(win);
}
},
/** * Add the menuitem for a tool to all open browser windows. * * @param {object} toolDefinition * properties of the tool to add
*/
_addToolToWindows(toolDefinition) { // No menu item or global shortcut is required for options panel. if (!toolDefinition.inMenu) { return;
}
// Skip if the tool is disabled. try { if (
toolDefinition.visibilityswitch &&
!Services.prefs.getBoolPref(toolDefinition.visibilityswitch)
) { return;
}
} catch (e) { // Prevent breaking everything if the pref doesn't exists.
}
// We need to insert the new tool in the right place, which means knowing // the tool that comes before the tool that we're trying to add const allDefs = gDevTools.getToolDefinitionArray();
let prevDef; for (const def of allDefs) { if (!def.inMenu) { continue;
} if (def === toolDefinition) { break;
}
prevDef = def;
}
for (const win of gDevToolsBrowser._trackedBrowserWindows) {
BrowserMenus.insertToolMenuElements(
win.document,
toolDefinition,
prevDef
); // If we are on a page where devtools menu items are hidden such as // about:devtools-toolbox, we need to call _updateMenuItems to update the // visibility of the newly created menu item.
gDevToolsBrowser._updateMenuItems(win);
}
},
hasToolboxOpened(win) { const tab = win.gBrowser.selectedTab; for (const commands of gDevTools._toolboxesPerCommands.keys()) { if (commands.descriptorFront.localTab == tab) { returntrue;
}
} returnfalse;
},
/** * Update developer tools menu items and the "Toggle Tools" checkbox. This is * called when a toolbox is created or destroyed.
*/
_updateMenu() { for (const win of gDevToolsBrowser._trackedBrowserWindows) {
gDevToolsBrowser._updateMenuItems(win);
}
},
/** * Update developer tools menu items and the "Toggle Tools" checkbox of XULWindow. * * @param {XULWindow} win
*/
_updateMenuItems(win) { const menu = win.document.getElementById("menu_devToolbox");
// Hide the "Toggle Tools" menu item if we are on about:devtools-toolbox.
menu.hidden =
gDevToolsBrowser._isAboutDevtoolsToolbox(win) ||
Services.prefs.getBoolPref("devtools.policy.disabled", false);
// Add a checkmark for the "Toggle Tools" menu item if a toolbox is already opened. const hasToolbox = gDevToolsBrowser.hasToolboxOpened(win); if (hasToolbox) {
menu.setAttribute("checked", "true");
} else {
menu.removeAttribute("checked");
}
},
/** * Retrieve the Toolbox instance loaded in the current page if the page is * about:devtools-toolbox, null otherwise. * * @param {XULWindow} win * The chrome window containing about:devtools-toolbox. Will match * toolbox.topWindow. * @return {Toolbox} The toolbox instance loaded in about:devtools-toolbox *
*/
_getAboutDevtoolsToolbox(win) { if (!gDevToolsBrowser._isAboutDevtoolsToolbox(win)) { returnnull;
} return gDevTools.getToolboxes().find(toolbox => toolbox.topWindow === win);
},
/** * Remove the menuitem for a tool to all open browser windows. * * @param {string} toolId * id of the tool to remove
*/
_removeToolFromWindows(toolId) { for (const win of gDevToolsBrowser._trackedBrowserWindows) {
BrowserMenus.removeToolFromMenu(toolId, win.document);
}
},
/** * Called on browser unload to remove menu entries, toolboxes and event * listeners from the closed browser window. * * @param {XULWindow} win * The window containing the menu entry
*/
_forgetBrowserWindow(win) { if (!gDevToolsBrowser._trackedBrowserWindows.has(win)) { return;
}
gDevToolsBrowser._trackedBrowserWindows.delete(win);
win.removeEventListener("unload", this);
BrowserMenus.removeMenus(win.document);
// Destroy toolboxes for closed window for (const [commands, toolbox] of gDevTools._toolboxesPerCommands) { if (
commands.descriptorFront.localTab?.ownerDocument?.defaultView == win
) {
toolbox.destroy();
}
}
const styleSheet = this._browserStyleSheets.get(win); if (styleSheet) {
styleSheet.remove(); this._browserStyleSheets.delete(win);
}
/** * Either the DevTools Loader has been destroyed by the add-on contribution * workflow, or firefox is shutting down.
* @param {boolean} shuttingDown * True if firefox is currently shutting down. We may prevent doing * some cleanups to speed it up. Otherwise everything need to be * cleaned up in order to be able to load devtools again.
*/
destroy({ shuttingDown }) {
Services.prefs.removeObserver("devtools.", gDevToolsBrowser);
removeThemeObserver(this._onThemeChanged);
Services.obs.removeObserver(
gDevToolsBrowser, "browser-delayed-startup-finished"
);
Services.obs.removeObserver(gDevToolsBrowser, "quit-application");
Services.obs.removeObserver(gDevToolsBrowser, "devtools:loader:destroy");
for (const win of gDevToolsBrowser._trackedBrowserWindows) {
gDevToolsBrowser._forgetBrowserWindow(win);
}
// Remove scripts loaded in content process to support the Browser Content Toolbox.
DevToolsServer.removeContentServerScript();
gDevTools.destroy({ shuttingDown });
},
});
// Handle all already registered tools,
gDevTools
.getToolDefinitionArray()
.forEach(def => gDevToolsBrowser._addToolToWindows(def)); // and the new ones.
gDevTools.on("tool-registered", function (toolId) { const toolDefinition = gDevTools._tools.get(toolId); // If the tool has been registered globally, add to all the // available windows. if (toolDefinition) {
gDevToolsBrowser._addToolToWindows(toolDefinition);
}
});
gDevTools.on("tool-unregistered", function (toolId) {
gDevToolsBrowser._removeToolFromWindows(toolId);
});
Services.obs.addObserver(gDevToolsBrowser, "quit-application");
Services.obs.addObserver(gDevToolsBrowser, "browser-delayed-startup-finished"); // Watch for module loader unload. Fires when the tools are reloaded.
Services.obs.addObserver(gDevToolsBrowser, "devtools:loader:destroy");
// Fake end of browser window load event for all already opened windows // that is already fully loaded. for (const win of Services.wm.getEnumerator(gDevTools.chromeWindowType)) { if (win.gBrowserInit?.delayedStartupFinished) {
gDevToolsBrowser._registerBrowserWindow(win);
}
}
¤ Dauer der Verarbeitung: 0.16 Sekunden
(vorverarbeitet)
¤
Die Informationen auf dieser Webseite wurden
nach bestem Wissen sorgfältig zusammengestellt. Es wird jedoch weder Vollständigkeit, noch Richtigkeit,
noch Qualität der bereit gestellten Informationen zugesichert.
Bemerkung:
Die farbliche Syntaxdarstellung ist noch experimentell.