mirror of
https://github.com/Feodor2/Mypal68.git
synced 2025-06-18 14:55:44 -04:00
803 lines
25 KiB
JavaScript
803 lines
25 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";
|
|
|
|
/**
|
|
* 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)
|
|
**/
|
|
|
|
const { Cc, Ci } = require("chrome");
|
|
const Services = require("Services");
|
|
const { gDevTools } = require("./devtools");
|
|
|
|
// Load target and toolbox lazily as they need gDevTools to be fully initialized
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"TargetFactory",
|
|
"devtools/client/framework/target",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"Toolbox",
|
|
"devtools/client/framework/toolbox",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"DebuggerServer",
|
|
"devtools/server/debugger-server",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"DebuggerClient",
|
|
"devtools/shared/client/debugger-client",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"BrowserMenus",
|
|
"devtools/client/framework/browser-menus"
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"appendStyleSheet",
|
|
"devtools/client/shared/stylesheet-utils",
|
|
true
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"ResponsiveUIManager",
|
|
"devtools/client/responsive/manager"
|
|
);
|
|
loader.lazyRequireGetter(
|
|
this,
|
|
"AppConstants",
|
|
"resource://gre/modules/AppConstants.jsm",
|
|
true
|
|
);
|
|
loader.lazyImporter(
|
|
this,
|
|
"BrowserToolboxProcess",
|
|
"resource://devtools/client/framework/ToolboxProcess.jsm"
|
|
);
|
|
|
|
const { LocalizationHelper } = require("devtools/shared/l10n");
|
|
const L10N = new LocalizationHelper(
|
|
"devtools/client/locales/toolbox.properties"
|
|
);
|
|
|
|
const BROWSER_STYLESHEET_URL = "chrome://devtools/skin/devtools-browser.css";
|
|
|
|
/**
|
|
* 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
|
|
async toggleToolboxCommand(gBrowser) {
|
|
const target = await TargetFactory.forTab(gBrowser.selectedTab);
|
|
const toolbox = gDevTools.getToolbox(target);
|
|
|
|
// 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;
|
|
isDocked ? gDevTools.closeToolbox(target) : gDevTools.showToolbox(target);
|
|
},
|
|
|
|
/**
|
|
* 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;
|
|
|
|
function toggleMenuItem(id, isEnabled) {
|
|
const cmd = doc.getElementById(id);
|
|
if (isEnabled) {
|
|
cmd.removeAttribute("disabled");
|
|
cmd.removeAttribute("hidden");
|
|
} else {
|
|
cmd.setAttribute("disabled", "true");
|
|
cmd.setAttribute("hidden", "true");
|
|
}
|
|
}
|
|
|
|
// Enable Browser Toolbox?
|
|
const chromeEnabled = Services.prefs.getBoolPref("devtools.chrome.enabled");
|
|
const devtoolsRemoteEnabled = Services.prefs.getBoolPref(
|
|
"devtools.debugger.remote-enabled"
|
|
);
|
|
const remoteEnabled = chromeEnabled && devtoolsRemoteEnabled;
|
|
toggleMenuItem("menu_browserToolbox", remoteEnabled);
|
|
toggleMenuItem(
|
|
"menu_browserContentToolbox",
|
|
remoteEnabled && win.gMultiProcessBrowser
|
|
);
|
|
|
|
// The profiler's popup is experimental. The plan is to eventually turn it on
|
|
// everywhere, but while it's under active development we don't want everyone
|
|
// having it enabled. For now the default pref is to turn it on with Nightly,
|
|
// with the option to flip the pref in other releases. This feature flag will
|
|
// go away once it is fully shipped.
|
|
const isPopupFeatureFlagEnabled = Services.prefs.getBoolPref(
|
|
"devtools.performance.popup.feature-flag", AppConstants.NIGHTLY_BUILD);
|
|
// If the feature flag is disabled, hide the menu item.
|
|
toggleMenuItem("menu_toggleProfilerButtonMenu", isPopupFeatureFlagEnabled);
|
|
|
|
if (isPopupFeatureFlagEnabled) {
|
|
// Did the user enable the profiler button in the menu? If it is then update the
|
|
// initial UI to show the menu item as checked.
|
|
if (Services.prefs.getBoolPref("devtools.performance.popup.enabled", false)) {
|
|
const cmd = doc.getElementById("menu_toggleProfilerButtonMenu");
|
|
cmd.setAttribute("checked", "true");
|
|
}
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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 = Services.prefs.getCharPref("devtools.theme");
|
|
if (devtoolsTheme != "dark") {
|
|
devtoolsTheme = "light";
|
|
}
|
|
|
|
// Style the splitter between the toolbox and page content. This used to
|
|
// set the attribute on the browser's root node but that regressed tpaint:
|
|
// bug 1331449.
|
|
win.document
|
|
.getElementById("browser-bottombox")
|
|
.setAttribute("devtoolstheme", devtoolsTheme);
|
|
win.document
|
|
.getElementById("appcontent")
|
|
.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);
|
|
}
|
|
}
|
|
if (prefName === "devtools.theme") {
|
|
for (const win of this._trackedBrowserWindows) {
|
|
this.updateDevtoolsThemeAttribute(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;
|
|
}
|
|
},
|
|
|
|
_prefObserverRegistered: false,
|
|
|
|
ensurePrefObserver() {
|
|
if (!this._prefObserverRegistered) {
|
|
this._prefObserverRegistered = true;
|
|
Services.prefs.addObserver("devtools.", this);
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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) {
|
|
if (gDevToolsBrowser._isAboutDevtoolsToolbox(win)) {
|
|
const toolbox = gDevToolsBrowser._getAboutDevtoolsToolbox(win);
|
|
toolbox.selectTool(toolId, "key_shortcut");
|
|
return;
|
|
}
|
|
|
|
const target = await TargetFactory.forTab(win.gBrowser.selectedTab);
|
|
const toolbox = gDevTools.getToolbox(target);
|
|
const toolDefinition = gDevTools.getToolDefinition(toolId);
|
|
|
|
if (
|
|
toolbox &&
|
|
(toolbox.currentToolId == toolId ||
|
|
(toolId == "webconsole" && toolbox.splitConsole))
|
|
) {
|
|
toolbox.fireCustomKey(toolId);
|
|
|
|
if (
|
|
toolDefinition.preventClosingOnKey ||
|
|
toolbox.hostType == Toolbox.HostType.WINDOW
|
|
) {
|
|
if (!toolDefinition.preventRaisingOnKey) {
|
|
toolbox.raise();
|
|
}
|
|
} else {
|
|
toolbox.destroy();
|
|
}
|
|
gDevTools.emit("select-tool-command", toolId);
|
|
} else {
|
|
gDevTools
|
|
.showToolbox(target, toolId).then(newToolbox => {
|
|
newToolbox.fireCustomKey(toolId);
|
|
gDevTools.emit("select-tool-command", toolId);
|
|
});
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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
|
|
*/
|
|
async onKeyShortcut(window, key) {
|
|
// 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);
|
|
return;
|
|
}
|
|
// Otherwise implement all other key shortcuts individually here
|
|
switch (key.id) {
|
|
case "toggleToolbox":
|
|
case "toggleToolboxF12":
|
|
await gDevToolsBrowser.toggleToolboxCommand(window.gBrowser);
|
|
break;
|
|
case "browserToolbox":
|
|
BrowserToolboxProcess.init();
|
|
break;
|
|
case "browserConsole":
|
|
const {
|
|
BrowserConsoleManager,
|
|
} = require("devtools/client/webconsole/browser-console-manager");
|
|
BrowserConsoleManager.openBrowserConsoleOrFocus();
|
|
break;
|
|
case "responsiveDesignMode":
|
|
ResponsiveUIManager.toggle(window, window.gBrowser.selectedTab, {
|
|
trigger: "shortcut",
|
|
});
|
|
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);
|
|
},
|
|
|
|
async _getContentProcessTarget(processId) {
|
|
// Create a DebuggerServer in order to connect locally to it
|
|
DebuggerServer.init();
|
|
DebuggerServer.registerAllActors();
|
|
DebuggerServer.allowChromeProcess = true;
|
|
|
|
const transport = DebuggerServer.connectPipe();
|
|
const client = new DebuggerClient(transport);
|
|
|
|
await client.connect();
|
|
const target = await client.mainRoot.getProcess(processId);
|
|
// Ensure closing the connection in order to cleanup
|
|
// the debugger client and also the server created in the
|
|
// content process
|
|
target.on("close", () => {
|
|
client.close();
|
|
});
|
|
return target;
|
|
},
|
|
|
|
/**
|
|
* Open the Browser Content Toolbox for the provided gBrowser instance.
|
|
* Returns a promise that resolves with a toolbox instance. If no content process is
|
|
* available, the promise will be rejected and a message will be displayed to the user.
|
|
*
|
|
* Used by menus.js
|
|
*/
|
|
openContentProcessToolbox(gBrowser) {
|
|
const { childCount } = Services.ppmm;
|
|
// Get the process message manager for the current tab
|
|
const mm = gBrowser.selectedBrowser.messageManager.processMessageManager;
|
|
let processId = null;
|
|
for (let i = 1; i < childCount; i++) {
|
|
const child = Services.ppmm.getChildAt(i);
|
|
if (child == mm) {
|
|
processId = i;
|
|
break;
|
|
}
|
|
}
|
|
if (processId) {
|
|
return this._getContentProcessTarget(processId)
|
|
.then(target => {
|
|
// Display a new toolbox in a new window
|
|
return gDevTools.showToolbox(target, null, Toolbox.HostType.WINDOW);
|
|
})
|
|
.catch(e => {
|
|
console.error(
|
|
"Exception while opening the browser content toolbox:",
|
|
e
|
|
);
|
|
});
|
|
}
|
|
|
|
const msg = L10N.getStr("toolbox.noContentProcessForTab.message");
|
|
Services.prompt.alert(null, "", msg);
|
|
return Promise.reject(msg);
|
|
},
|
|
|
|
/**
|
|
* Open a window-hosted toolbox to debug the worker associated to the provided
|
|
* worker actor.
|
|
*
|
|
* @param {WorkerTargetFront} workerTargetFront
|
|
* worker actor front to debug
|
|
* @param {String} toolId (optional)
|
|
* The id of the default tool to show
|
|
*/
|
|
async openWorkerToolbox(workerTarget, toolId) {
|
|
await gDevTools.showToolbox(workerTarget, toolId, Toolbox.HostType.WINDOW);
|
|
},
|
|
|
|
/**
|
|
* 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: function(win) {
|
|
if (this._browserStyleSheets.has(win)) {
|
|
return Promise.resolve();
|
|
}
|
|
|
|
const doc = win.document;
|
|
const { styleSheet, loadPromise } = appendStyleSheet(
|
|
doc,
|
|
BROWSER_STYLESHEET_URL
|
|
);
|
|
this._browserStyleSheets.set(win, styleSheet);
|
|
return loadPromise;
|
|
},
|
|
|
|
/**
|
|
* 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;
|
|
}
|
|
gDevToolsBrowser._trackedBrowserWindows.add(win);
|
|
|
|
BrowserMenus.addMenus(win.document);
|
|
|
|
this.updateCommandAvailability(win);
|
|
this.updateDevtoolsThemeAttribute(win);
|
|
this.ensurePrefObserver();
|
|
win.addEventListener("unload", this);
|
|
|
|
const tabContainer = win.gBrowser.tabContainer;
|
|
tabContainer.addEventListener("TabSelect", this);
|
|
},
|
|
|
|
/**
|
|
* Hook the JS debugger tool to the "Debug Script" button of the slow script
|
|
* dialog.
|
|
*/
|
|
setSlowScriptDebugHandler() {
|
|
const debugService = Cc["@mozilla.org/dom/slow-script-debug;1"].getService(
|
|
Ci.nsISlowScriptDebug
|
|
);
|
|
|
|
async function slowScriptDebugHandler(tab, callback) {
|
|
const target = await TargetFactory.forTab(tab);
|
|
|
|
gDevTools.showToolbox(target, "jsdebugger").then(toolbox => {
|
|
const threadFront = toolbox.threadFront;
|
|
|
|
// Break in place, which means resuming the debuggee thread and pausing
|
|
// right before the next step happens.
|
|
switch (threadFront.state) {
|
|
case "paused":
|
|
// When the debugger is already paused.
|
|
threadFront.resumeThenPause();
|
|
callback();
|
|
break;
|
|
case "attached":
|
|
// When the debugger is already open.
|
|
threadFront.interrupt().then(() => {
|
|
threadFront.resumeThenPause();
|
|
callback();
|
|
});
|
|
break;
|
|
case "resuming":
|
|
// The debugger is newly opened.
|
|
threadFront.once("resumed", () => {
|
|
threadFront.interrupt().then(() => {
|
|
threadFront.resumeThenPause();
|
|
callback();
|
|
});
|
|
});
|
|
break;
|
|
default:
|
|
throw Error(
|
|
"invalid thread client state in slow script debug handler: " +
|
|
threadFront.state
|
|
);
|
|
}
|
|
});
|
|
}
|
|
|
|
debugService.activationHandler = function(window) {
|
|
const chromeWindow = window.docShell.rootTreeItem.domWindow;
|
|
|
|
let setupFinished = false;
|
|
slowScriptDebugHandler(chromeWindow.gBrowser.selectedTab, () => {
|
|
setupFinished = true;
|
|
});
|
|
|
|
// Don't return from the interrupt handler until the debugger is brought
|
|
// up; no reason to continue executing the slow script.
|
|
const utils = window.windowUtils;
|
|
utils.enterModalState();
|
|
Services.tm.spinEventLoopUntil(() => {
|
|
return setupFinished;
|
|
});
|
|
utils.leaveModalState();
|
|
};
|
|
|
|
debugService.remoteActivationHandler = function(browser, callback) {
|
|
const chromeWindow = browser.ownerDocument.defaultView;
|
|
const tab = chromeWindow.gBrowser.getTabForBrowser(browser);
|
|
chromeWindow.gBrowser.selected = tab;
|
|
|
|
slowScriptDebugHandler(tab, function() {
|
|
callback.finishDebuggerStartup();
|
|
}).catch(console.error);
|
|
};
|
|
},
|
|
|
|
/**
|
|
* Unset the slow script debug handler.
|
|
*/
|
|
unsetSlowScriptDebugHandler() {
|
|
const debugService = Cc["@mozilla.org/dom/slow-script-debug;1"].getService(
|
|
Ci.nsISlowScriptDebug
|
|
);
|
|
debugService.activationHandler = undefined;
|
|
},
|
|
|
|
/**
|
|
* 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);
|
|
}
|
|
|
|
if (toolDefinition.id === "jsdebugger") {
|
|
gDevToolsBrowser.setSlowScriptDebugHandler();
|
|
}
|
|
},
|
|
|
|
hasToolboxOpened(win) {
|
|
const tab = win.gBrowser.selectedTab;
|
|
for (const [target] of gDevTools._toolboxes) {
|
|
if (target.tab == tab) {
|
|
return true;
|
|
}
|
|
}
|
|
return false;
|
|
},
|
|
|
|
/**
|
|
* 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.
|
|
const isAboutDevtoolsToolbox = gDevToolsBrowser._isAboutDevtoolsToolbox(
|
|
win
|
|
);
|
|
if (isAboutDevtoolsToolbox) {
|
|
menu.setAttribute("hidden", "true");
|
|
} else {
|
|
menu.removeAttribute("hidden");
|
|
}
|
|
|
|
// 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");
|
|
}
|
|
},
|
|
|
|
/**
|
|
* Check whether the window is showing about:devtools-toolbox page or not.
|
|
*
|
|
* @param {XULWindow} win
|
|
* @return {boolean} true: about:devtools-toolbox is showing
|
|
* false: otherwise
|
|
*/
|
|
_isAboutDevtoolsToolbox(win) {
|
|
const currentURI = win.gBrowser.currentURI;
|
|
return (
|
|
currentURI.scheme === "about" &&
|
|
currentURI.filePath === "devtools-toolbox"
|
|
);
|
|
},
|
|
|
|
/**
|
|
* 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)) {
|
|
return null;
|
|
}
|
|
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);
|
|
}
|
|
|
|
if (toolId === "jsdebugger") {
|
|
gDevToolsBrowser.unsetSlowScriptDebugHandler();
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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 [target, toolbox] of gDevTools._toolboxes) {
|
|
if (target.tab && target.tab.ownerDocument.defaultView == win) {
|
|
toolbox.destroy();
|
|
}
|
|
}
|
|
|
|
const styleSheet = this._browserStyleSheets.get(win);
|
|
if (styleSheet) {
|
|
styleSheet.remove();
|
|
this._browserStyleSheets.delete(win);
|
|
}
|
|
|
|
const tabContainer = win.gBrowser.tabContainer;
|
|
tabContainer.removeEventListener("TabSelect", this);
|
|
},
|
|
|
|
handleEvent(event) {
|
|
switch (event.type) {
|
|
case "TabSelect":
|
|
gDevToolsBrowser._updateMenu();
|
|
break;
|
|
case "unload":
|
|
// top-level browser window unload
|
|
gDevToolsBrowser._forgetBrowserWindow(event.target.defaultView);
|
|
break;
|
|
}
|
|
},
|
|
|
|
/**
|
|
* 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);
|
|
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.
|
|
DebuggerServer.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);
|
|
});
|
|
|
|
gDevTools.on("toolbox-ready", gDevToolsBrowser._updateMenu);
|
|
gDevTools.on("toolbox-destroyed", gDevToolsBrowser._updateMenu);
|
|
|
|
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);
|
|
}
|
|
}
|