summaryrefslogtreecommitdiffstats
path: root/devtools/client/webconsole/webconsole-ui.js
diff options
context:
space:
mode:
Diffstat (limited to '')
-rw-r--r--devtools/client/webconsole/webconsole-ui.js716
1 files changed, 716 insertions, 0 deletions
diff --git a/devtools/client/webconsole/webconsole-ui.js b/devtools/client/webconsole/webconsole-ui.js
new file mode 100644
index 0000000000..914cf9b22d
--- /dev/null
+++ b/devtools/client/webconsole/webconsole-ui.js
@@ -0,0 +1,716 @@
+/* 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 KeyShortcuts = require("resource://devtools/client/shared/key-shortcuts.js");
+const {
+ l10n,
+} = require("resource://devtools/client/webconsole/utils/messages.js");
+
+const { BrowserLoader } = ChromeUtils.import(
+ "resource://devtools/shared/loader/browser-loader.js"
+);
+const {
+ getAdHocFrontOrPrimitiveGrip,
+} = require("resource://devtools/client/fronts/object.js");
+
+const { PREFS } = require("resource://devtools/client/webconsole/constants.js");
+
+const FirefoxDataProvider = require("resource://devtools/client/netmonitor/src/connector/firefox-data-provider.js");
+
+const lazy = {};
+ChromeUtils.defineESModuleGetters(lazy, {
+ AppConstants: "resource://gre/modules/AppConstants.sys.mjs",
+});
+
+loader.lazyRequireGetter(
+ this,
+ "START_IGNORE_ACTION",
+ "resource://devtools/client/shared/redux/middleware/ignore.js",
+ true
+);
+const ZoomKeys = require("resource://devtools/client/shared/zoom-keys.js");
+
+const PREF_SIDEBAR_ENABLED = "devtools.webconsole.sidebarToggle";
+const PREF_BROWSERTOOLBOX_SCOPE = "devtools.browsertoolbox.scope";
+
+/**
+ * A WebConsoleUI instance is an interactive console initialized *per target*
+ * that displays console log data as well as provides an interactive terminal to
+ * manipulate the target's document content.
+ *
+ * The WebConsoleUI is responsible for the actual Web Console UI
+ * implementation.
+ */
+class WebConsoleUI {
+ /*
+ * @param {WebConsole} hud: The WebConsole owner object.
+ */
+ constructor(hud) {
+ this.hud = hud;
+ this.hudId = this.hud.hudId;
+ this.isBrowserConsole = this.hud.isBrowserConsole;
+
+ this.isBrowserToolboxConsole =
+ this.hud.commands.descriptorFront.isBrowserProcessDescriptor &&
+ !this.isBrowserConsole;
+
+ this.window = this.hud.iframeWindow;
+
+ this._onPanelSelected = this._onPanelSelected.bind(this);
+ this._onChangeSplitConsoleState =
+ this._onChangeSplitConsoleState.bind(this);
+ this._onTargetAvailable = this._onTargetAvailable.bind(this);
+ this._onTargetDestroyed = this._onTargetDestroyed.bind(this);
+ this._onResourceAvailable = this._onResourceAvailable.bind(this);
+ this._onNetworkResourceUpdated = this._onNetworkResourceUpdated.bind(this);
+ this._onScopePrefChanged = this._onScopePrefChanged.bind(this);
+
+ if (this.isBrowserConsole) {
+ Services.prefs.addObserver(
+ PREF_BROWSERTOOLBOX_SCOPE,
+ this._onScopePrefChanged
+ );
+ }
+
+ EventEmitter.decorate(this);
+ }
+
+ /**
+ * Initialize the WebConsoleUI instance.
+ * @return object
+ * A promise object that resolves once the frame is ready to use.
+ */
+ init() {
+ if (this._initializer) {
+ return this._initializer;
+ }
+
+ this._initializer = (async () => {
+ this._initUI();
+
+ if (this.isBrowserConsole) {
+ // Bug 1605763:
+ // TargetCommand.startListening will start fetching additional targets
+ // and may overload the Browser Console with loads of targets and resources.
+ // We can call it from here, as `_attachTargets` is called after the UI is initialized.
+ // Bug 1642599:
+ // TargetCommand.startListening has to be called before:
+ // - `_attachTargets`, in order to set TargetCommand.watcherFront which is used by ResourceWatcher.watchResources.
+ // - `ConsoleCommands`, in order to set TargetCommand.targetFront which is wrapped by hud.currentTarget
+ await this.hud.commands.targetCommand.startListening();
+ if (this._destroyed) {
+ return;
+ }
+ }
+
+ await this.wrapper.init();
+ if (this._destroyed) {
+ return;
+ }
+
+ // Bug 1605763: It's important to call _attachTargets once the UI is initialized, as
+ // it may overload the Browser Console with many updates.
+ // It is also important to do it only after the wrapper is initialized,
+ // otherwise its `store` will be null while we already call a few dispatch methods
+ // from onResourceAvailable
+ await this._attachTargets();
+ if (this._destroyed) {
+ return;
+ }
+
+ // `_attachTargets` will process resources and throttle some actions
+ // Wait for these actions to be dispatched before reporting that the
+ // console is initialized. Otherwise `showToolbox` will resolve before
+ // all already existing console messages are displayed.
+ await this.wrapper.waitAsyncDispatches();
+ })();
+
+ return this._initializer;
+ }
+
+ destroy() {
+ if (this._destroyed) {
+ return;
+ }
+
+ this._destroyed = true;
+
+ this.React = this.ReactDOM = this.FrameView = null;
+
+ if (this.wrapper) {
+ this.wrapper.getStore()?.dispatch(START_IGNORE_ACTION);
+ this.wrapper.destroy();
+ }
+
+ if (this.jsterm) {
+ this.jsterm.destroy();
+ this.jsterm = null;
+ }
+
+ const { toolbox } = this.hud;
+ if (toolbox) {
+ toolbox.off("webconsole-selected", this._onPanelSelected);
+ toolbox.off("split-console", this._onChangeSplitConsoleState);
+ toolbox.off("select", this._onChangeSplitConsoleState);
+ }
+
+ if (this.isBrowserConsole) {
+ Services.prefs.removeObserver(
+ PREF_BROWSERTOOLBOX_SCOPE,
+ this._onScopePrefChanged
+ );
+ }
+
+ // Stop listening for targets
+ this.hud.commands.targetCommand.unwatchTargets({
+ types: this.hud.commands.targetCommand.ALL_TYPES,
+ onAvailable: this._onTargetAvailable,
+ onDestroyed: this._onTargetDestroyed,
+ });
+
+ const resourceCommand = this.hud.resourceCommand;
+ resourceCommand.unwatchResources(
+ [
+ resourceCommand.TYPES.CONSOLE_MESSAGE,
+ resourceCommand.TYPES.ERROR_MESSAGE,
+ resourceCommand.TYPES.PLATFORM_MESSAGE,
+ resourceCommand.TYPES.DOCUMENT_EVENT,
+ resourceCommand.TYPES.LAST_PRIVATE_CONTEXT_EXIT,
+ ],
+ { onAvailable: this._onResourceAvailable }
+ );
+ resourceCommand.unwatchResources([resourceCommand.TYPES.CSS_MESSAGE], {
+ onAvailable: this._onResourceAvailable,
+ });
+
+ this.stopWatchingNetworkResources();
+
+ if (this.networkDataProvider) {
+ this.networkDataProvider.destroy();
+ this.networkDataProvider = null;
+ }
+
+ // Nullify `hud` last as it nullify also target which is used on destroy
+ this.window = this.hud = this.wrapper = null;
+ }
+
+ /**
+ * Clear the Web Console output.
+ *
+ * This method emits the "messages-cleared" notification.
+ *
+ * @param boolean clearStorage
+ * True if you want to clear the console messages storage associated to
+ * this Web Console.
+ * @param object event
+ * If the event exists, calls preventDefault on it.
+ */
+ async clearOutput(clearStorage, event) {
+ if (event) {
+ event.preventDefault();
+ }
+ if (this.wrapper) {
+ this.wrapper.dispatchMessagesClear();
+ }
+
+ if (clearStorage) {
+ await this.clearMessagesCache();
+ }
+ this.emitForTests("messages-cleared");
+ }
+
+ async clearMessagesCache() {
+ if (this._destroyed) {
+ return;
+ }
+
+ // This can be called during console destruction and getAllFronts would reject in such case.
+ try {
+ const consoleFronts = await this.hud.commands.targetCommand.getAllFronts(
+ this.hud.commands.targetCommand.ALL_TYPES,
+ "console"
+ );
+ const promises = [];
+ for (const consoleFront of consoleFronts) {
+ promises.push(consoleFront.clearMessagesCacheAsync());
+ }
+ await Promise.all(promises);
+ this.emitForTests("messages-cache-cleared");
+ } catch (e) {
+ console.warn("Exception in clearMessagesCache", e);
+ }
+ }
+
+ /**
+ * Remove all of the private messages from the Web Console output.
+ *
+ * This method emits the "private-messages-cleared" notification.
+ */
+ clearPrivateMessages() {
+ if (this._destroyed) {
+ return;
+ }
+
+ this.wrapper.dispatchPrivateMessagesClear();
+ this.emitForTests("private-messages-cleared");
+ }
+
+ inspectObjectActor(objectActor) {
+ const { targetFront } = this.hud.commands.targetCommand;
+ this.wrapper.dispatchMessageAdd(
+ {
+ helperResult: {
+ type: "inspectObject",
+ object:
+ objectActor && objectActor.getGrip
+ ? objectActor
+ : getAdHocFrontOrPrimitiveGrip(objectActor, targetFront),
+ },
+ },
+ true
+ );
+ return this.wrapper;
+ }
+
+ disableAllNetworkMessages() {
+ if (this._destroyed) {
+ return;
+ }
+ this.wrapper.dispatchNetworkMessagesDisable();
+ }
+
+ getPanelWindow() {
+ return this.window;
+ }
+
+ logWarningAboutReplacedAPI() {
+ return this.hud.currentTarget.logWarningInPage(
+ l10n.getStr("ConsoleAPIDisabled"),
+ "ConsoleAPIDisabled"
+ );
+ }
+
+ /**
+ * Connect to the server using the remote debugging protocol.
+ *
+ * @private
+ * @return object
+ * A promise object that is resolved/reject based on the proxies connections.
+ */
+ async _attachTargets() {
+ const { commands, resourceCommand } = this.hud;
+ this.networkDataProvider = new FirefoxDataProvider({
+ commands,
+ actions: {
+ updateRequest: (id, data) =>
+ this.wrapper.batchedRequestUpdates({ id, data }),
+ },
+ owner: this,
+ });
+
+ // Listen for all target types, including:
+ // - frames, in order to get the parent process target
+ // which is considered as a frame rather than a process.
+ // - workers, for similar reason. When we open a toolbox
+ // for just a worker, the top level target is a worker target.
+ // - processes, as we want to spawn additional proxies for them.
+ await commands.targetCommand.watchTargets({
+ types: this.hud.commands.targetCommand.ALL_TYPES,
+ onAvailable: this._onTargetAvailable,
+ onDestroyed: this._onTargetDestroyed,
+ });
+
+ await resourceCommand.watchResources(
+ [
+ resourceCommand.TYPES.CONSOLE_MESSAGE,
+ resourceCommand.TYPES.ERROR_MESSAGE,
+ resourceCommand.TYPES.PLATFORM_MESSAGE,
+ resourceCommand.TYPES.DOCUMENT_EVENT,
+ resourceCommand.TYPES.LAST_PRIVATE_CONTEXT_EXIT,
+ ],
+ { onAvailable: this._onResourceAvailable }
+ );
+
+ if (this.isBrowserConsole || this.isBrowserToolboxConsole) {
+ const shouldEnableNetworkMonitoring = Services.prefs.getBoolPref(
+ PREFS.UI.ENABLE_NETWORK_MONITORING
+ );
+ if (shouldEnableNetworkMonitoring) {
+ await this.startWatchingNetworkResources();
+ } else {
+ await this.stopWatchingNetworkResources();
+ }
+ } else {
+ // We should always watch for network resources in the webconsole
+ await this.startWatchingNetworkResources();
+ }
+ }
+
+ async startWatchingNetworkResources() {
+ const { commands, resourceCommand } = this.hud;
+ await resourceCommand.watchResources(
+ [
+ resourceCommand.TYPES.NETWORK_EVENT,
+ resourceCommand.TYPES.NETWORK_EVENT_STACKTRACE,
+ ],
+ {
+ onAvailable: this._onResourceAvailable,
+ onUpdated: this._onNetworkResourceUpdated,
+ }
+ );
+
+ // When opening a worker toolbox from about:debugging,
+ // we do not instantiate any Watcher actor yet and would throw here.
+ // But even once we do, we wouldn't support network inspection anyway.
+ if (commands.targetCommand.hasTargetWatcherSupport()) {
+ const networkFront = await commands.watcherFront.getNetworkParentActor();
+ // There is no way to view response bodies from the Browser Console, so do
+ // not waste the memory.
+ const saveBodies =
+ !this.isBrowserConsole &&
+ Services.prefs.getBoolPref(
+ "devtools.netmonitor.saveRequestAndResponseBodies"
+ );
+ await networkFront.setSaveRequestAndResponseBodies(saveBodies);
+ }
+ }
+
+ async stopWatchingNetworkResources() {
+ if (this._destroyed) {
+ return;
+ }
+
+ await this.hud.resourceCommand.unwatchResources(
+ [
+ this.hud.resourceCommand.TYPES.NETWORK_EVENT,
+ this.hud.resourceCommand.TYPES.NETWORK_EVENT_STACKTRACE,
+ ],
+ {
+ onAvailable: this._onResourceAvailable,
+ onUpdated: this._onNetworkResourceUpdated,
+ }
+ );
+ }
+
+ handleDocumentEvent(resource) {
+ // Only consider top level document, and ignore remote iframes top document
+ if (!resource.targetFront.isTopLevel) {
+ return;
+ }
+
+ if (resource.name == "will-navigate") {
+ this.handleWillNavigate({
+ timeStamp: resource.time,
+ url: resource.newURI,
+ });
+ } else if (resource.name == "dom-complete") {
+ this.handleNavigated({
+ hasNativeConsoleAPI: resource.hasNativeConsoleAPI,
+ });
+ }
+ // For now, ignore all other DOCUMENT_EVENT's.
+ }
+
+ /**
+ * Handler for when the page is done loading.
+ *
+ * @param Boolean hasNativeConsoleAPI
+ * True if the `console` object is the native one and hasn't been overloaded by a custom
+ * object by the page itself.
+ */
+ async handleNavigated({ hasNativeConsoleAPI }) {
+ // Updates instant evaluation on page navigation
+ this.wrapper.dispatchUpdateInstantEvaluationResultForCurrentExpression();
+
+ // Wait for completion of any async dispatch before notifying that the console
+ // is fully updated after a page reload
+ await this.wrapper.waitAsyncDispatches();
+
+ if (!hasNativeConsoleAPI) {
+ this.logWarningAboutReplacedAPI();
+ }
+
+ this.emit("reloaded");
+ }
+
+ handleWillNavigate({ timeStamp, url }) {
+ this.wrapper.dispatchTabWillNavigate({ timeStamp, url });
+ }
+
+ async watchCssMessages() {
+ const { resourceCommand } = this.hud;
+ await resourceCommand.watchResources([resourceCommand.TYPES.CSS_MESSAGE], {
+ onAvailable: this._onResourceAvailable,
+ });
+ }
+
+ _onResourceAvailable(resources) {
+ if (this._destroyed) {
+ return;
+ }
+
+ const messages = [];
+ for (const resource of resources) {
+ const { TYPES } = this.hud.resourceCommand;
+ if (resource.resourceType === TYPES.DOCUMENT_EVENT) {
+ this.handleDocumentEvent(resource);
+ continue;
+ }
+ if (resource.resourceType == TYPES.LAST_PRIVATE_CONTEXT_EXIT) {
+ // Private messages only need to be removed from the output in Browser Console/Browser Toolbox
+ // (but in theory this resource should only be send from parent process watchers)
+ if (this.isBrowserConsole || this.isBrowserToolboxConsole) {
+ this.clearPrivateMessages();
+ }
+ continue;
+ }
+ // Ignore messages forwarded from content processes if we're in fission browser toolbox.
+ if (
+ !this.wrapper ||
+ ((resource.resourceType === TYPES.ERROR_MESSAGE ||
+ resource.resourceType === TYPES.CSS_MESSAGE) &&
+ resource.pageError?.isForwardedFromContentProcess &&
+ (this.isBrowserToolboxConsole || this.isBrowserConsole))
+ ) {
+ continue;
+ }
+
+ // Don't show messages emitted from a private window before the Browser Console was
+ // opened to avoid leaking data from past usage of the browser (e.g. content message
+ // from now closed private tabs)
+ if (
+ (this.isBrowserToolboxConsole || this.isBrowserConsole) &&
+ resource.isAlreadyExistingResource &&
+ (resource.pageError?.private || resource.message?.private)
+ ) {
+ continue;
+ }
+
+ if (resource.resourceType === TYPES.NETWORK_EVENT_STACKTRACE) {
+ this.networkDataProvider?.onStackTraceAvailable(resource);
+ continue;
+ }
+
+ if (resource.resourceType === TYPES.NETWORK_EVENT) {
+ this.networkDataProvider?.onNetworkResourceAvailable(resource);
+ }
+ messages.push(resource);
+ }
+ this.wrapper.dispatchMessagesAdd(messages);
+ }
+
+ _onNetworkResourceUpdated(updates) {
+ if (this._destroyed) {
+ return;
+ }
+
+ const messageUpdates = [];
+ for (const { resource } of updates) {
+ if (
+ resource.resourceType == this.hud.resourceCommand.TYPES.NETWORK_EVENT
+ ) {
+ this.networkDataProvider?.onNetworkResourceUpdated(resource);
+ messageUpdates.push(resource);
+ }
+ }
+ this.wrapper.dispatchMessagesUpdate(messageUpdates);
+ }
+
+ /**
+ * Called any time a new target is available.
+ * i.e. it was already existing or has just been created.
+ *
+ * @private
+ * @param Front targetFront
+ * The Front of the target that is available.
+ * This Front inherits from TargetMixin and is typically
+ * composed of a WindowGlobalTargetFront or ContentProcessTargetFront.
+ */
+ async _onTargetAvailable({ targetFront }) {
+ // onTargetAvailable is a mandatory argument for watchTargets,
+ // we still define it solely for being able to use onTargetDestroyed.
+ }
+
+ _onTargetDestroyed({ targetFront, isModeSwitching }) {
+ // Don't try to do anything if the WebConsole is being destroyed
+ if (this._destroyed) {
+ return;
+ }
+
+ // We only want to remove messages from a target destroyed when we're switching mode
+ // in the Browser Console/Browser Toolbox Console.
+ // For regular cases, we want to keep the message history (the output will still be
+ // cleared when the top level target navigates, if "Persist Logs" isn't true, via handleWillNavigate)
+ if (isModeSwitching) {
+ this.wrapper.dispatchTargetMessagesRemove(targetFront);
+ }
+ }
+
+ _initUI() {
+ this.document = this.window.document;
+ this.rootElement = this.document.documentElement;
+
+ this.outputNode = this.document.getElementById("app-wrapper");
+
+ const { toolbox } = this.hud;
+
+ // Initialize module loader and load all the WebConsoleWrapper. The entire code-base
+ // doesn't need any extra privileges and runs entirely in content scope.
+ const WebConsoleWrapper = BrowserLoader({
+ baseURI: "resource://devtools/client/webconsole/",
+ window: this.window,
+ }).require("resource://devtools/client/webconsole/webconsole-wrapper.js");
+
+ this.wrapper = new WebConsoleWrapper(
+ this.outputNode,
+ this,
+ toolbox,
+ this.document
+ );
+
+ this._initShortcuts();
+ this._initOutputSyntaxHighlighting();
+
+ if (toolbox) {
+ toolbox.on("webconsole-selected", this._onPanelSelected);
+ toolbox.on("split-console", this._onChangeSplitConsoleState);
+ toolbox.on("select", this._onChangeSplitConsoleState);
+ }
+ }
+
+ _initOutputSyntaxHighlighting() {
+ // Given a DOM node, we syntax highlight identically to how the input field
+ // looks. See https://codemirror.net/demo/runmode.html;
+ const syntaxHighlightNode = node => {
+ const editor = this.jsterm && this.jsterm.editor;
+ if (node && editor) {
+ node.classList.add("cm-s-mozilla");
+ editor.CodeMirror.runMode(
+ node.textContent,
+ "application/javascript",
+ node
+ );
+ }
+ };
+
+ // Use a Custom Element to handle syntax highlighting to avoid
+ // dealing with refs or innerHTML from React.
+ const win = this.window;
+ win.customElements.define(
+ "syntax-highlighted",
+ class extends win.HTMLElement {
+ connectedCallback() {
+ if (!this.connected) {
+ this.connected = true;
+ syntaxHighlightNode(this);
+
+ // Highlight Again when the innerText changes
+ // We remove the listener before running codemirror mode and add
+ // it again to capture text changes
+ this.observer = new win.MutationObserver((mutations, observer) => {
+ observer.disconnect();
+ syntaxHighlightNode(this);
+ observer.observe(this, { childList: true });
+ });
+
+ this.observer.observe(this, { childList: true });
+ }
+ }
+ }
+ );
+ }
+
+ _initShortcuts() {
+ const shortcuts = new KeyShortcuts({
+ window: this.window,
+ });
+
+ let clearShortcut;
+ if (lazy.AppConstants.platform === "macosx") {
+ const alternativaClearShortcut = l10n.getStr(
+ "webconsole.clear.alternativeKeyOSX"
+ );
+ shortcuts.on(alternativaClearShortcut, event =>
+ this.clearOutput(true, event)
+ );
+ clearShortcut = l10n.getStr("webconsole.clear.keyOSX");
+ } else {
+ clearShortcut = l10n.getStr("webconsole.clear.key");
+ }
+
+ shortcuts.on(clearShortcut, event => this.clearOutput(true, event));
+
+ if (this.isBrowserConsole) {
+ // Make sure keyboard shortcuts work immediately after opening
+ // the Browser Console (Bug 1461366).
+ this.window.focus();
+ shortcuts.on(
+ l10n.getStr("webconsole.close.key"),
+ this.window.close.bind(this.window)
+ );
+
+ ZoomKeys.register(this.window, shortcuts);
+
+ /* This is the same as DevelopmentHelpers.quickRestart, but it runs in all
+ * builds (even official). This allows a user to do a restart + session restore
+ * with Ctrl+Shift+J (open Browser Console) and then Ctrl+Alt+R (restart).
+ */
+ shortcuts.on("CmdOrCtrl+Alt+R", () => {
+ this.hud.commands.targetCommand.reloadTopLevelTarget();
+ });
+ } else if (Services.prefs.getBoolPref(PREF_SIDEBAR_ENABLED)) {
+ shortcuts.on("Esc", event => {
+ this.wrapper.dispatchSidebarClose();
+ if (this.jsterm) {
+ this.jsterm.focus();
+ }
+ });
+ }
+ }
+
+ /**
+ * Sets the focus to JavaScript input field when the web console tab is
+ * selected or when there is a split console present.
+ * @private
+ */
+ _onPanelSelected() {
+ // We can only focus when we have the jsterm reference. This is fine because if the
+ // jsterm is not mounted yet, it will be focused in JSTerm's componentDidMount.
+ if (this.jsterm) {
+ this.jsterm.focus();
+ }
+ }
+
+ _onChangeSplitConsoleState() {
+ this.wrapper.dispatchSplitConsoleCloseButtonToggle();
+ }
+
+ _onScopePrefChanged() {
+ if (this.isBrowserConsole) {
+ this.hud.updateWindowTitle();
+ }
+ }
+
+ getInputCursor() {
+ return this.jsterm && this.jsterm.getSelectionStart();
+ }
+
+ getJsTermTooltipAnchor() {
+ return this.outputNode.querySelector(".CodeMirror-cursor");
+ }
+
+ attachRef(id, node) {
+ this[id] = node;
+ }
+
+ getSelectedNodeActorID() {
+ const inspectorSelection = this.hud.getInspectorSelection();
+ return inspectorSelection?.nodeFront?.actorID;
+ }
+}
+
+exports.WebConsoleUI = WebConsoleUI;