blob: 06d4cd33fcbca2f659ee1ef02ee7fcea29b71372 [file] [log] [blame]
// Copyright 2021 The Chromium Authors
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
/* eslint-disable @devtools/no-imperative-dom-api */
/*
* Copyright (C) 2007, 2008 Apple Inc. All rights reserved.
* Copyright (C) 2008 Matt Lilek <webkit@mattlilek.com>
* Copyright (C) 2009 Joseph Pecoraro
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions
* are met:
*
* 1. Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* 2. Redistributions in binary form must reproduce the above copyright
* notice, this list of conditions and the following disclaimer in the
* documentation and/or other materials provided with the distribution.
* 3. Neither the name of Apple Computer, Inc. ("Apple") nor the names of
* its contributors may be used to endorse or promote products derived
* from this software without specific prior written permission.
*
* THIS SOFTWARE IS PROVIDED BY APPLE AND ITS CONTRIBUTORS "AS IS" AND ANY
* EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
* WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
* DISCLAIMED. IN NO EVENT SHALL APPLE OR ITS CONTRIBUTORS BE LIABLE FOR ANY
* DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
* (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
* LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND
* ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
* (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF
* THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
*/
import '../../ui/components/adorners/adorners.js';
import '../../ui/components/buttons/buttons.js';
import * as Common from '../../core/common/common.js';
import * as Host from '../../core/host/host.js';
import * as i18n from '../../core/i18n/i18n.js';
import * as Platform from '../../core/platform/platform.js';
import * as Root from '../../core/root/root.js';
import * as SDK from '../../core/sdk/sdk.js';
import * as Protocol from '../../generated/protocol.js';
import * as AIAssistance from '../../models/ai_assistance/ai_assistance.js';
import * as Badges from '../../models/badges/badges.js';
import type * as Elements from '../../models/elements/elements.js';
import type * as IssuesManager from '../../models/issues_manager/issues_manager.js';
import * as TextUtils from '../../models/text_utils/text_utils.js';
import * as Workspace from '../../models/workspace/workspace.js';
import * as CodeMirror from '../../third_party/codemirror.next/codemirror.next.js';
import type * as Adorners from '../../ui/components/adorners/adorners.js';
import * as CodeHighlighter from '../../ui/components/code_highlighter/code_highlighter.js';
import * as Highlighting from '../../ui/components/highlighting/highlighting.js';
import * as TextEditor from '../../ui/components/text_editor/text_editor.js';
import {Icon} from '../../ui/kit/kit.js';
import * as Components from '../../ui/legacy/components/utils/utils.js';
import * as UI from '../../ui/legacy/legacy.js';
import * as Lit from '../../ui/lit/lit.js';
import type {DirectiveResult} from '../../ui/lit/lit.js';
import * as VisualLogging from '../../ui/visual_logging/visual_logging.js';
import * as PanelsCommon from '../common/common.js';
import * as Emulation from '../emulation/emulation.js';
import * as Media from '../media/media.js';
import * as ElementsComponents from './components/components.js';
import {canGetJSPath, cssPath, jsPath, xPath} from './DOMPath.js';
import {getElementIssueDetails} from './ElementIssueUtils.js';
import {ElementsPanel} from './ElementsPanel.js';
import {type ElementsTreeOutline, MappedCharToEntity} from './ElementsTreeOutline.js';
import {ImagePreviewPopover} from './ImagePreviewPopover.js';
import {getRegisteredDecorators, type MarkerDecorator, type MarkerDecoratorRegistration} from './MarkerDecorator.js';
const {html, nothing, render, Directives: {ref, repeat}} = Lit;
const {animateOn} = UI.UIUtils;
const UIStrings = {
/**
* @description Title for Ad adorner. This iframe is marked as advertisement frame.
*/
thisFrameWasIdentifiedAsAnAd: 'This frame was identified as an ad frame',
/**
* @description A context menu item in the Elements panel. Force is used as a verb, indicating intention to make the state change.
*/
forceState: 'Force state',
/**
* @description Hint element title in Elements Tree Element of the Elements panel
* @example {0} PH1
*/
useSInTheConsoleToReferToThis: 'Use {PH1} in the console to refer to this element.',
/**
* @description A context menu item in the Elements Tree Element of the Elements panel
*/
addAttribute: 'Add attribute',
/**
* @description Text to modify the attribute of an item
*/
editAttribute: 'Edit attribute',
/**
* @description Text to focus on something
*/
focus: 'Focus',
/**
* @description Text to scroll the displayed content into view
*/
scrollIntoView: 'Scroll into view',
/**
* @description A context menu item in the Elements Tree Element of the Elements panel
*/
editText: 'Edit text',
/**
* @description A context menu item in the Elements Tree Element of the Elements panel
*/
editAsHtml: 'Edit as HTML',
/**
* @description Text to cut an element, cut should be used as a verb
*/
cut: 'Cut',
/**
* @description Text for copying, copy should be used as a verb
*/
copy: 'Copy',
/**
* @description Text to paste an element, paste should be used as a verb
*/
paste: 'Paste',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copyOuterhtml: 'Copy outerHTML',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copySelector: 'Copy `selector`',
/**
* @description Text in Elements Tree Element of the Elements panel
*/
copyJsPath: 'Copy JS path',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copyStyles: 'Copy styles',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copyXpath: 'Copy XPath',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copyFullXpath: 'Copy full XPath',
/**
* @description Text in Elements Tree Element of the Elements panel, copy should be used as a verb
*/
copyElement: 'Copy element',
/**
* @description A context menu item in the Elements Tree Element of the Elements panel
*/
duplicateElement: 'Duplicate element',
/**
* @description Text to hide an element
*/
hideElement: 'Hide element',
/**
* @description A context menu item in the Elements Tree Element of the Elements panel
*/
deleteElement: 'Delete element',
/**
* @description Text to expand something recursively
*/
expandRecursively: 'Expand recursively',
/**
* @description Text to collapse children of a parent group
*/
collapseChildren: 'Collapse children',
/**
* @description Title of an action in the emulation tool to capture node screenshot
*/
captureNodeScreenshot: 'Capture node screenshot',
/**
* @description Title of a context menu item. When clicked DevTools goes to the Application panel and shows this specific iframe's details
*/
showFrameDetails: 'Show `iframe` details',
/**
* @description Text in Elements Tree Element of the Elements panel
*/
valueIsTooLargeToEdit: '<value is too large to edit>',
/**
* @description Element text content in Elements Tree Element of the Elements panel
*/
children: 'Children:',
/**
* @description ARIA label for Elements Tree adorners
*/
enableGridMode: 'Enable grid mode',
/**
* @description ARIA label for Elements Tree adorners
*/
disableGridMode: 'Disable grid mode',
/**
* @description ARIA label for Elements Tree adorners
*/
/**
* @description ARIA label for Elements Tree adorners
*/
enableGridLanesMode: 'Enable grid-lanes mode',
/**
* @description ARIA label for Elements Tree adorners
*/
disableGridLanesMode: 'Disable grid-lanes mode',
/**
* @description ARIA label for an elements tree adorner
*/
forceOpenPopover: 'Keep this popover open',
/**
* @description ARIA label for an elements tree adorner
*/
stopForceOpenPopover: 'Stop keeping this popover open',
/**
* @description Label of the adorner for flex elements in the Elements panel
*/
enableFlexMode: 'Enable flex mode',
/**
* @description Label of the adorner for flex elements in the Elements panel
*/
disableFlexMode: 'Disable flex mode',
/**
* @description Label of an adorner in the Elements panel. When clicked, it enables
* the overlay showing CSS scroll snapping for the current element.
*/
enableScrollSnap: 'Enable scroll-snap overlay',
/**
* @description Label of an adorner in the Elements panel. When clicked, it disables
* the overlay showing CSS scroll snapping for the current element.
*/
disableScrollSnap: 'Disable scroll-snap overlay',
/**
* @description Label of an adorner in the Elements panel. When clicked, it enables
* the overlay showing the container overlay for the current element.
*/
enableContainer: 'Enable container overlay',
/**
* @description Label of an adorner in the Elements panel. When clicked, it disables
* the overlay showing container for the current element.
*/
disableContainer: 'Disable container overlay',
/**
* @description Label of an adorner in the Elements panel. When clicked, it forces
* the element into applying its starting-style rules.
*/
enableStartingStyle: 'Enable @starting-style mode',
/**
* @description Label of an adorner in the Elements panel. When clicked, it no longer
* forces the element into applying its starting-style rules.
*/
disableStartingStyle: 'Disable @starting-style mode',
/**
* @description Label of an adorner in the Elements panel. When clicked, it redirects
* to the Media Panel.
*/
openMediaPanel: 'Jump to Media panel',
/**
* @description Text of a tooltip to redirect to another element in the Elements panel
*/
showPopoverTarget: 'Show element associated with the `popovertarget` attribute',
/**
* @description Text of a tooltip to redirect to another element in the Elements panel, associated with the `interesttarget` attribute
*/
showInterestTarget: 'Show element associated with the `interesttarget` attribute',
/**
* @description Text of a tooltip to redirect to another element in the Elements panel, associated with the `commandfor` attribute
*/
showCommandForTarget: 'Show element associated with the `commandfor` attribute',
/**
* @description Text of the tooltip for scroll adorner.
*/
elementHasScrollableOverflow: 'This element has a scrollable overflow',
/**
* @description Text of a context menu item to redirect to the AI assistance panel and to start a chat.
*/
startAChat: 'Start a chat',
/**
   * @description Label of an adorner next to the html node in the Elements panel.
   */
viewSourceCode: 'View source code',
/**
* @description Context menu item in Elements panel to assess visibility of an element via AI.
*/
assessVisibility: 'Assess visibility',
/**
* @description Context menu item in Elements panel to center an element via AI.
*/
centerElement: 'Center element',
/**
* @description Context menu item in Elements panel to wrap flex items via AI.
*/
wrapTheseItems: 'Wrap these items',
/**
* @description Context menu item in Elements panel to distribute flex items evenly via AI.
*/
distributeItemsEvenly: 'Distribute items evenly',
/**
* @description Context menu item in Elements panel to explain flexbox via AI.
*/
explainFlexbox: 'Explain flexbox',
/**
* @description Context menu item in Elements panel to align grid items via AI.
*/
alignItems: 'Align items',
/**
* @description Context menu item in Elements panel to add padding/gap to grid via AI.
*/
addPadding: 'Add padding',
/**
* @description Context menu item in Elements panel to explain grid layout via AI.
*/
explainGridLayout: 'Explain grid layout',
/**
* @description Context menu item in Elements panel to find grid definition for a subgrid item via AI.
*/
findGridDefinition: 'Find grid definition',
/**
* @description Context menu item in Elements panel to change parent grid properties for a subgrid item via AI.
*/
changeParentProperties: 'Change parent properties',
/**
* @description Context menu item in Elements panel to explain subgrids via AI.
*/
explainSubgrids: 'Explain subgrids',
/**
* @description Context menu item in Elements panel to remove scrollbars via AI.
*/
removeScrollbars: 'Remove scrollbars',
/**
* @description Context menu item in Elements panel to style scrollbars via AI.
*/
styleScrollbars: 'Style scrollbars',
/**
* @description Context menu item in Elements panel to explain scrollbars via AI.
*/
explainScrollbars: 'Explain scrollbars',
/**
* @description Context menu item in Elements panel to explain container queries via AI.
*/
explainContainerQueries: 'Explain container queries',
/**
* @description Context menu item in Elements panel to explain container types via AI.
*/
explainContainerTypes: 'Explain container types',
/**
* @description Context menu item in Elements panel to explain container context via AI.
*/
explainContainerContext: 'Explain container context',
/**
* @description Link text content in Elements Tree Outline of the Elements panel. When clicked, it "reveals" the true location of an element.
*/
reveal: 'reveal',
} as const;
const str_ = i18n.i18n.registerUIStrings('panels/elements/ElementsTreeElement.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
const enum TagType {
OPENING = 'OPENING_TAG',
CLOSING = 'CLOSING_TAG',
}
interface OpeningTagContext {
tagType: TagType.OPENING;
canAddAttributes: boolean;
}
interface ClosingTagContext {
tagType: TagType.CLOSING;
}
export type TagTypeContext = OpeningTagContext|ClosingTagContext;
export function isOpeningTag(context: TagTypeContext): context is OpeningTagContext {
return context.tagType === TagType.OPENING;
}
export interface ViewInput {
node: SDK.DOMModel.DOMNode|null;
isClosingTag: boolean;
expanded: boolean;
isExpandable: boolean;
isXMLMimeType: boolean;
updateRecord: Elements.ElementUpdateRecord.ElementUpdateRecord|null;
onHighlightSearchResults: () => void;
onExpand: () => void;
containerAdornerActive: boolean;
flexAdornerActive: boolean;
gridAdornerActive: boolean;
popoverAdornerActive: boolean;
showAdAdorner: boolean;
showContainerAdorner: boolean;
containerType?: string;
showFlexAdorner: boolean;
showGridAdorner: boolean;
showGridLanesAdorner: boolean;
showMediaAdorner: boolean;
showPopoverAdorner: boolean;
showTopLayerAdorner: boolean;
isSubgrid: boolean;
showViewSourceAdorner: boolean;
showScrollAdorner: boolean;
showScrollSnapAdorner: boolean;
topLayerIndex: number;
scrollSnapAdornerActive: boolean;
onGutterClick: (e: Event) => void;
onContainerAdornerClick: (e: Event) => void;
onFlexAdornerClick: (e: Event) => void;
onGridAdornerClick: (e: Event) => void;
onMediaAdornerClick: (e: Event) => void;
onPopoverAdornerClick: (e: Event) => void;
onScrollSnapAdornerClick: (e: Event) => void;
onTopLayerAdornerClick: (e: Event) => void;
onViewSourceAdornerClick: () => void;
onSlotAdornerClick: (e: Event) => void;
showSlotAdorner: boolean;
slotName?: string;
showStartingStyleAdorner: boolean;
startingStyleAdornerActive: boolean;
onStartingStyleAdornerClick: (e: Event) => void;
isHovered: boolean;
isSelected: boolean;
showAiButton: boolean;
aiButtonTitle?: string;
onAiButtonClick: (e: Event) => void;
decorations: Decoration[];
descendantDecorations: Decoration[];
decorationsTooltip: string;
indent: number;
}
export interface ViewOutput {
contentElement?: HTMLElement;
}
export function adornerRef(): DirectiveResult<typeof Lit.Directives.RefDirective> {
let adorner: Adorners.Adorner.Adorner|undefined;
return ref(el => {
if (adorner) {
ElementsPanel.instance().deregisterAdorner(adorner);
}
adorner = el as Adorners.Adorner.Adorner;
if (adorner) {
if (ElementsPanel.instance().isAdornerEnabled(adorner.name)) {
adorner.show();
} else {
adorner.hide();
}
ElementsPanel.instance().registerAdorner(adorner);
}
});
}
export interface Decoration {
title: string;
color: string;
}
const DOM_UPDATE_ANIMATION_CLASS_NAME = 'dom-update-highlight';
function handleAdornerKeydown(cb: (event: Event) => void): (event: KeyboardEvent) => void {
return (event: KeyboardEvent) => {
if (event.code === 'Enter' || event.code === 'Space') {
cb(event);
event.preventDefault();
event.stopPropagation();
}
};
}
function renderTitle(
node: SDK.DOMModel.DOMNode,
isClosingTag: boolean,
expanded: boolean,
isExpandable: boolean,
isXMLMimeType: boolean,
updateRecord: Elements.ElementUpdateRecord.ElementUpdateRecord|null,
onUpdateSearchHighlight: () => void,
onExpand: () => void,
): Lit.LitTemplate {
switch (node.nodeType()) {
case Node.ATTRIBUTE_NODE:
return renderAttribute({name: node.name as string, value: node.value as string}, updateRecord, true, node);
case Node.ELEMENT_NODE: {
if (node.pseudoType()) {
let pseudoElementName = node.nodeName();
const pseudoIdentifier = node.pseudoIdentifier();
if (pseudoIdentifier) {
pseudoElementName += `(${pseudoIdentifier})`;
}
return html`<span class="webkit-html-pseudo-element">${pseudoElementName}</span>\u200B`;
}
const tagName = node.nodeNameInCorrectCase();
if (isClosingTag) {
return renderTag(node, tagName, true, expanded, true, updateRecord);
}
const openingTag = renderTag(node, tagName, false, expanded, false, updateRecord);
if (isExpandable) {
if (!expanded) {
return html`${openingTag}<devtools-elements-tree-expand-button .data=${
{clickHandler: onExpand} as
ElementsComponents.ElementsTreeExpandButton
.ElementsTreeExpandButtonData}></devtools-elements-tree-expand-button><span style="font-size: 0;"
>…</span>\u200B${renderTag(node, tagName, true, expanded, false, updateRecord)}`;
}
return openingTag;
}
if (ElementsTreeElement.canShowInlineText(node)) {
const firstChild = node.firstChild;
if (!firstChild) {
throw new Error('ElementsTreeElement._nodeTitleInfo expects node.firstChild to be defined.');
}
const result = convertUnicodeCharsToHTMLEntities(firstChild.nodeValue());
const textContent = Platform.StringUtilities.collapseWhitespace(result.text);
const renderTextNode = ref(el => {
if (el) {
el.textContent = textContent;
Highlighting.highlightRangesWithStyleClass(el, result.entityRanges, 'webkit-html-entity-value');
}
});
return html`${openingTag}<span class="webkit-html-text-node" jslog=${
VisualLogging.value('text-node').track({change: true, dblclick: true})} ${
animateOn(
Boolean((updateRecord?.hasChangedChildren() || updateRecord?.isCharDataModified())),
DOM_UPDATE_ANIMATION_CLASS_NAME)} ${renderTextNode}></span>\u200B${
renderTag(node, tagName, true, expanded, false, updateRecord)}`;
}
if (isXMLMimeType || !ForbiddenClosingTagElements.has(tagName)) {
return html`${openingTag}${renderTag(node, tagName, true, expanded, false, updateRecord)}`;
}
return openingTag;
}
case Node.TEXT_NODE: {
if (node.parentNode && node.parentNode.nodeName().toLowerCase() === 'script') {
const text = node.nodeValue();
const highlightNode = ref(el => {
if (el) {
el.textContent = text.replace(/^[\n\r]+|\s+$/g, '');
void CodeHighlighter.CodeHighlighter.highlightNode(el, 'text/javascript').then(onUpdateSearchHighlight);
}
});
return html`<span class="webkit-html-text-node webkit-html-js-node" jslog=${
VisualLogging.value('script-text-node').track({change: true, dblclick: true})} ${highlightNode}></span>`;
}
if (node.parentNode && node.parentNode.nodeName().toLowerCase() === 'style') {
const text = node.nodeValue();
const highlightNode = ref(el => {
if (el) {
el.textContent = text.replace(/^[\n\r]+|\s+$/g, '');
void CodeHighlighter.CodeHighlighter.highlightNode(el, 'text/css').then(onUpdateSearchHighlight);
}
});
return html`<span class="webkit-html-text-node webkit-html-css-node" jslog=${
VisualLogging.value('css-text-node').track({change: true, dblclick: true})} ${highlightNode}></span>`;
}
const result = convertUnicodeCharsToHTMLEntities(node.nodeValue());
const textContent = Platform.StringUtilities.collapseWhitespace(result.text);
const renderTextNode = ref(el => {
if (el) {
el.textContent = textContent;
Highlighting.highlightRangesWithStyleClass(el, result.entityRanges, 'webkit-html-entity-value');
}
});
return html`"<span class="webkit-html-text-node" jslog=${VisualLogging.value('text-node').track({
change: true,
dblclick: true
})} ${animateOn(Boolean(updateRecord?.isCharDataModified()), DOM_UPDATE_ANIMATION_CLASS_NAME)} ${
renderTextNode}></span>"`;
}
case Node.COMMENT_NODE: {
return html`<span class="webkit-html-comment">&lt;!--${node.nodeValue()}--&gt;</span>`;
}
case Node.DOCUMENT_TYPE_NODE: {
let doctype = '<!DOCTYPE ' + node.nodeName();
if (node.publicId) {
doctype += ' PUBLIC "' + node.publicId + '"';
if (node.systemId) {
doctype += ' "' + node.systemId + '"';
}
} else if (node.systemId) {
doctype += ' SYSTEM "' + node.systemId + '"';
}
if (node.internalSubset) {
doctype += ' [' + node.internalSubset + ']';
}
doctype += '>';
return html`<span class="webkit-html-doctype">${doctype}</span>`;
}
case Node.CDATA_SECTION_NODE: {
return html`<span class="webkit-html-text-node">&lt;![CDATA[${node.nodeValue()}]]&gt;</span>`;
}
case Node.DOCUMENT_NODE: {
const text = (node as SDK.DOMModel.DOMDocument).documentURL;
return html`<span>#document (<span>${Components.Linkifier.Linkifier.renderLinkifiedUrl(text, {
text,
preventClick: true,
showColumnNumber: false,
inlineFrameIndex: 0,
})}</span>)</span>`;
}
case Node.DOCUMENT_FRAGMENT_NODE: {
return html`<span class="webkit-html-fragment">${
Platform.StringUtilities.collapseWhitespace(node.nodeNameInCorrectCase())}</span>`;
}
default: {
return html`${Platform.StringUtilities.collapseWhitespace(node.nodeNameInCorrectCase())}`;
}
}
}
function renderLinkifiedSrcset(tokens: Common.Srcset.Token[], node: SDK.DOMModel.DOMNode): Lit.TemplateResult {
return html`${repeat(tokens, token => {
switch (token.type) {
case Common.Srcset.TokenType.URL:
return renderLinkifiedValue(token.value, node);
case Common.Srcset.TokenType.LITERAL:
return token.value;
}
})}`;
}
const closingPunctuationRegex = /[\/;:\)\]\}]/g;
// FIXME: this should be made declarative next.
function setValueWithEntities(element: Element, value: string): void {
let highlightIndex = 0;
let highlightCount = 0;
let additionalHighlightOffset = 0;
const result = convertUnicodeCharsToHTMLEntities(value);
highlightCount = result.entityRanges.length;
const newValue = result.text.replace(closingPunctuationRegex, (match, replaceOffset) => {
while (highlightIndex < highlightCount && result.entityRanges[highlightIndex].offset < replaceOffset) {
result.entityRanges[highlightIndex].offset += additionalHighlightOffset;
++highlightIndex;
}
additionalHighlightOffset += 1;
return match + '\u200B';
});
while (highlightIndex < highlightCount) {
result.entityRanges[highlightIndex].offset += additionalHighlightOffset;
++highlightIndex;
}
element.setTextContentTruncatedIfNeeded(newValue);
Highlighting.highlightRangesWithStyleClass(element, result.entityRanges, 'webkit-html-entity-value');
}
function renderLinkifiedValue(value: string, node: SDK.DOMModel.DOMNode): Lit.TemplateResult {
const rewrittenHref = node ? node.resolveURL(value) : null;
if (rewrittenHref === null) {
return html`<span ${ref(el => {
if (el) {
setValueWithEntities(el, value);
}
})}}></span>`;
}
value = value.replace(closingPunctuationRegex, '$&\u200B');
if (value.startsWith('data:')) {
value = Platform.StringUtilities.trimMiddle(value, 60);
}
const isAnchor = node && node.nodeName().toLowerCase() === 'a';
if (isAnchor) {
return html`<devtools-link class="devtools-link image-url" href=${rewrittenHref} ${ref(el => {
if (el) {
ImagePreviewPopover.setImageUrl(el, rewrittenHref);
}
})}>${Platform.StringUtilities.trimMiddle(value, 150)}</devtools-link>`;
}
return Components.Linkifier.Linkifier.renderLinkifiedUrl(rewrittenHref, {
text: value,
preventClick: true,
showColumnNumber: false,
inlineFrameIndex: 0,
onRef: link => {
ImagePreviewPopover.setImageUrl(link, rewrittenHref);
}
});
}
function renderAttribute(
attr: {name: string, value?: string}, updateRecord: Elements.ElementUpdateRecord.ElementUpdateRecord|null,
isDiff: boolean, node: SDK.DOMModel.DOMNode): Lit.LitTemplate {
const name = attr.name;
const value = attr.value || '';
const forceValue = isDiff;
const hasText = (forceValue || value.length > 0);
const jslog = VisualLogging.value(name === 'style' ? 'style-attribute' : 'attribute').track({
change: true,
dblclick: true,
});
const relationRef =
(relation: Protocol.DOM.GetElementByRelationRequestRelation, tooltip: string): ReturnType<typeof ref> =>
ref((el): void => {
if (!el) {
return;
}
void (async(): Promise<void> => {
const relatedElementId = await node.domModel().getElementByRelation(node.id, relation);
const relatedElement = node.domModel().nodeForId(relatedElementId);
if (!relatedElement) {
return;
}
const link = PanelsCommon.DOMLinkifier.Linkifier.instance().linkify(relatedElement, {
preventKeyboardFocus: true,
tooltip,
textContent: el.textContent || undefined,
isDynamicLink: true,
});
el.removeChildren();
el.append(link);
})();
});
let relationRefDirective: ReturnType<typeof relationRef> = ref(() => {});
if (!value) {
if (name === 'popovertarget') {
relationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.PopoverTarget, i18nString(UIStrings.showPopoverTarget));
} else if (name === 'interesttarget') {
relationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.InterestTarget, i18nString(UIStrings.showInterestTarget));
} else if (name === 'commandfor') {
relationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.CommandFor, i18nString(UIStrings.showCommandForTarget));
}
}
let valueRelationRefDirective: ReturnType<typeof relationRef> = ref(() => {});
if (value) {
if (name === 'popovertarget') {
valueRelationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.PopoverTarget, i18nString(UIStrings.showPopoverTarget));
} else if (name === 'interesttarget') {
valueRelationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.InterestTarget, i18nString(UIStrings.showInterestTarget));
} else if (name === 'commandfor') {
valueRelationRefDirective = relationRef(
Protocol.DOM.GetElementByRelationRequestRelation.CommandFor, i18nString(UIStrings.showCommandForTarget));
}
}
const nodeName = node ? node.nodeName().toLowerCase() : '';
const enum ValueType {
UNKNOWN = 0,
SRC = 1,
SRCSET = 2,
}
let valueType = ValueType.UNKNOWN;
if (nodeName && (name === 'src' || name === 'href') && value) {
valueType = ValueType.SRC;
} else if ((nodeName === 'img' || nodeName === 'source') && name === 'srcset') {
valueType = ValueType.SRCSET;
} else if (nodeName === 'image' && (name === 'xlink:href' || name === 'href')) {
valueType = ValueType.SRCSET;
}
const withEntitiesRef = valueType === ValueType.UNKNOWN ? ref(el => {
if (el) {
setValueWithEntities(el, value);
}
}) :
nothing;
// clang-format off
return html`<span class="webkit-html-attribute" jslog=${jslog}><span class="webkit-html-attribute-name"
${animateOn(Boolean(updateRecord?.isAttributeModified(name) && !hasText), DOM_UPDATE_ANIMATION_CLASS_NAME)} ${
relationRefDirective}>${name}</span>${
hasText ? html`=\u200B"<span class="webkit-html-attribute-value" ${
animateOn(
Boolean(updateRecord?.isAttributeModified(name) && hasText),
DOM_UPDATE_ANIMATION_CLASS_NAME)} ${valueRelationRefDirective} ${withEntitiesRef}>
${valueType === ValueType.SRC ? renderLinkifiedValue(value, node) : nothing}
${valueType === ValueType.SRCSET ? renderLinkifiedSrcset(Common.Srcset.parseSrcset(value), node) : nothing}
</span>"` :
nothing}</span>`;
// clang-format on
}
function renderTag(
node: SDK.DOMModel.DOMNode, tagName: string, isClosingTag: boolean, expanded: boolean,
isDistinctTreeElement: boolean,
updateRecord: Elements.ElementUpdateRecord.ElementUpdateRecord|null): Lit.LitTemplate {
const classMap = {
'webkit-html-tag': true,
close: isClosingTag && isDistinctTreeElement,
};
let hasUpdates = false;
const attributes = !isClosingTag && node.hasAttributes() ? node.attributes() : [];
if (!isClosingTag && updateRecord) {
hasUpdates = updateRecord.hasRemovedAttributes() || updateRecord.hasRemovedChildren();
hasUpdates = hasUpdates || (!expanded && updateRecord.hasChangedChildren());
}
// We are taking full text content of the tag, including attributes and children, to set the aria label.
// FIXME: we should compute the aria label ourselves if it is event needed.
const setAriaLabel = ref(el => {
if (el?.textContent) {
UI.ARIAUtils.setLabel(el, el.textContent);
}
});
const tagNameClass = isClosingTag ? 'webkit-html-close-tag-name' : 'webkit-html-tag-name';
const tagString = (isClosingTag ? '/' : '') + tagName;
const jslog = !isClosingTag ? VisualLogging.value('tag-name').track({change: true, dblclick: true}) : '';
return html`<span
class=${Lit.Directives.classMap(classMap)} ${setAriaLabel}
>&lt;<span class=${tagNameClass} jslog=${jslog || nothing} ${
animateOn(hasUpdates, DOM_UPDATE_ANIMATION_CLASS_NAME)}>${tagString}</span>${
attributes.map(attr => html` ${renderAttribute(attr, updateRecord, false, node)}`)}&gt;</span>\u200B`;
}
export const DEFAULT_VIEW = (input: ViewInput, output: ViewOutput, target: HTMLElement): void => {
const hasAdorners = input.showAdAdorner || input.showContainerAdorner || input.showFlexAdorner ||
input.showGridAdorner || input.showGridLanesAdorner || input.showMediaAdorner || input.showPopoverAdorner ||
input.showTopLayerAdorner || input.showViewSourceAdorner || input.showScrollAdorner ||
input.showScrollSnapAdorner || input.showSlotAdorner || input.showStartingStyleAdorner;
const gutterContainerClasses = {
'has-decorations': input.decorations.length || input.descendantDecorations.length,
'gutter-container': true,
};
// clang-format off
render(html`
<div ${ref(el => { output.contentElement = el as HTMLElement; })}>
${input.node ? html`<span class="highlight">${renderTitle(
input.node,
input.isClosingTag,
input.expanded,
input.isExpandable,
input.isXMLMimeType,
input.updateRecord,
input.onHighlightSearchResults,
input.onExpand,
)}</span>` : nothing}
${input.isHovered || input.isSelected ? html`
<div class="selection fill" style=${`margin-left: ${-input.indent}px`}></div>
` : nothing}
<div class=${Lit.Directives.classMap(gutterContainerClasses)}
style="left: ${-input.indent}px"
@click=${input.onGutterClick}>
<devtools-icon name="dots-horizontal"></devtools-icon>
${input.decorations.length || input.descendantDecorations.length ? html`
<div class="elements-gutter-decoration-container"
title=${input.decorationsTooltip}>
${input.decorations.map(d => html`<div class="elements-gutter-decoration" style="--decoration-color: ${d.color}"></div>`)}
${input.descendantDecorations.map(d => html`<div class="elements-gutter-decoration elements-has-decorated-children" style="--decoration-color: ${d.color}"></div>`)}
</div>` : nothing}
</div>
${hasAdorners ? html`<div class="adorner-container ${!hasAdorners ? 'hidden' : ''}">
${input.showAdAdorner ? html`<devtools-adorner
aria-label=${i18nString(UIStrings.thisFrameWasIdentifiedAsAnAd)}
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.AD}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.AD)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.AD}</span>
</devtools-adorner>` : nothing}
${input.showViewSourceAdorner ? html`<devtools-adorner
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.VIEW_SOURCE}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.VIEW_SOURCE)}
aria-label=${i18nString(UIStrings.viewSourceCode)}
@click=${input.onViewSourceAdornerClick}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.VIEW_SOURCE}</span>
</devtools-adorner>` : nothing}
${input.showContainerAdorner ? html`<devtools-adorner
class=clickable
role=button
toggleable=true
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.CONTAINER}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.CONTAINER).track({click: true})}
active=${input.containerAdornerActive}
aria-label=${input.containerAdornerActive ? i18nString(UIStrings.enableContainer) : i18nString(UIStrings.disableContainer)}
@click=${input.onContainerAdornerClick}
@keydown=${handleAdornerKeydown(input.onContainerAdornerClick)}
${adornerRef()}>
<span class="adorner-with-icon">
<devtools-icon name="container"></devtools-icon>
<span>${input.containerType}</span>
</span>
</devtools-adorner>`: nothing}
${input.showFlexAdorner ? html`<devtools-adorner
class=clickable
role=button
toggleable=true
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.FLEX}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.FLEX).track({click: true})}
active=${input.flexAdornerActive}
aria-label=${input.flexAdornerActive ? i18nString(UIStrings.disableFlexMode) : i18nString(UIStrings.enableFlexMode)}
@click=${input.onFlexAdornerClick}
@keydown=${handleAdornerKeydown(input.onFlexAdornerClick)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.FLEX}</span>
</devtools-adorner>`: nothing}
${input.showGridAdorner ? html`<devtools-adorner
class=clickable
role=button
toggleable=true
tabindex=0
.name=${input.isSubgrid ? ElementsComponents.AdornerManager.RegisteredAdorners.SUBGRID : ElementsComponents.AdornerManager.RegisteredAdorners.GRID}
jslog=${VisualLogging.adorner(input.isSubgrid ? ElementsComponents.AdornerManager.RegisteredAdorners.SUBGRID : ElementsComponents.AdornerManager.RegisteredAdorners.GRID).track({click: true})}
active=${input.gridAdornerActive}
aria-label=${input.gridAdornerActive ? i18nString(UIStrings.disableGridMode) : i18nString(UIStrings.enableGridMode)}
@click=${input.onGridAdornerClick}
@keydown=${handleAdornerKeydown(input.onGridAdornerClick)}
${adornerRef()}>
<span>${input.isSubgrid ? ElementsComponents.AdornerManager.RegisteredAdorners.SUBGRID : ElementsComponents.AdornerManager.RegisteredAdorners.GRID}</span>
</devtools-adorner>`: nothing}
${input.showGridLanesAdorner ? html`<devtools-adorner
class=clickable
role=button
toggleable=true
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.GRID_LANES}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.GRID_LANES).track({click: true})}
active=${input.gridAdornerActive}
aria-label=${input.gridAdornerActive ? i18nString(UIStrings.disableGridLanesMode) : i18nString(UIStrings.enableGridLanesMode)}
@click=${input.onGridAdornerClick}
@keydown=${handleAdornerKeydown(input.onGridAdornerClick)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.GRID_LANES}</span>
</devtools-adorner>`: nothing}
${input.showMediaAdorner ? html`<devtools-adorner
class=clickable
role=button
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.MEDIA}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.MEDIA).track({click: true})}
aria-label=${i18nString(UIStrings.openMediaPanel)}
@click=${input.onMediaAdornerClick}
@keydown=${handleAdornerKeydown(input.onMediaAdornerClick)}
${adornerRef()}>
<span class="adorner-with-icon">
${ElementsComponents.AdornerManager.RegisteredAdorners.MEDIA}<devtools-icon name="select-element"></devtools-icon>
</span>
</devtools-adorner>`: nothing}
${input.showPopoverAdorner ? html`<devtools-adorner
class=clickable
role=button
toggleable=true
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.POPOVER}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.POPOVER).track({click: true})}
active=${input.popoverAdornerActive}
aria-label=${input.popoverAdornerActive ? i18nString(UIStrings.stopForceOpenPopover) : i18nString(UIStrings.forceOpenPopover)}
@click=${input.onPopoverAdornerClick}
@keydown=${handleAdornerKeydown(input.onPopoverAdornerClick)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.POPOVER}</span>
</devtools-adorner>`: nothing}
${input.showTopLayerAdorner ? html`<devtools-adorner
class=clickable
role=button
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.TOP_LAYER}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.TOP_LAYER).track({click: true})}
aria-label=${i18nString(UIStrings.reveal)}
@click=${input.onTopLayerAdornerClick}
@keydown=${handleAdornerKeydown(input.onTopLayerAdornerClick)}
${adornerRef()}>
<span class="adorner-with-icon">
${`top-layer (${input.topLayerIndex})`}<devtools-icon name="select-element"></devtools-icon>
</span>
</devtools-adorner>`: nothing}
${input.showStartingStyleAdorner ? html`<devtools-adorner
class="starting-style"
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.STARTING_STYLE}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.STARTING_STYLE).track({click: true})}
active=${input.startingStyleAdornerActive}
toggleable=true
aria-label=${input.startingStyleAdornerActive ? i18nString(UIStrings.disableStartingStyle) : i18nString(UIStrings.enableStartingStyle)}
@click=${input.onStartingStyleAdornerClick}
@keydown=${handleAdornerKeydown(input.onStartingStyleAdornerClick)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.STARTING_STYLE}</span>
</devtools-adorner>` : nothing}
${input.showScrollAdorner ? html`<devtools-adorner
class="scroll"
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL).track({click: true})}
aria-label=${i18nString(UIStrings.elementHasScrollableOverflow)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL}</span>
</devtools-adorner>` : nothing}
${input.showSlotAdorner ? html`<devtools-adorner
class=clickable
role=button
tabindex=0
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.SLOT}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.SLOT).track({click: true})}
@click=${input.onSlotAdornerClick}
@mousedown=${(e: Event) => e.stopPropagation()}
${adornerRef()}>
<span class="adorner-with-icon">
<devtools-icon name="select-element"></devtools-icon>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.SLOT}</span>
</span>
</devtools-adorner>`: nothing}
${input.showScrollSnapAdorner ? html`<devtools-adorner
class="scroll-snap"
.name=${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL_SNAP}
jslog=${VisualLogging.adorner(ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL_SNAP).track({click: true})}
active=${input.scrollSnapAdornerActive}
toggleable=true
aria-label=${input.scrollSnapAdornerActive ? i18nString(UIStrings.disableScrollSnap) : i18nString(UIStrings.enableScrollSnap)}
@click=${input.onScrollSnapAdornerClick}
@keydown=${handleAdornerKeydown(input.onScrollSnapAdornerClick)}
${adornerRef()}>
<span>${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL_SNAP}</span>
</devtools-adorner>` : nothing}
</div>`: nothing}
${input.isSelected ? html`
<span class="selected-hint" title=${i18nString(UIStrings.useSInTheConsoleToReferToThis, {PH1: '$0'})} aria-hidden="true"></span>
` : nothing}
${input.showAiButton ? html`
<span class="ai-button-container">
<devtools-floating-button
icon-name=${AIAssistance.AiUtils.getIconName()}
title=${input.aiButtonTitle || ''}
jslogcontext="ask-ai"
@click=${input.onAiButtonClick}
@mousedown=${(e: Event) => e.stopPropagation()}>
</devtools-floating-button>
</span>
` : nothing}
</div>
`, target);
// clang-format on
};
export class ElementsTreeElement extends UI.TreeOutline.TreeElement {
nodeInternal: SDK.DOMModel.DOMNode;
override treeOutline: ElementsTreeOutline|null;
private searchQuery: string|null;
#expandedChildrenLimit: number;
private readonly decorationsThrottler: Common.Throttler.Throttler;
private inClipboard: boolean;
#hovered: boolean;
private editing: EditorHandles|null;
private htmlEditElement?: HTMLElement;
expandAllButtonElement: UI.TreeOutline.TreeElement|null;
#elementIssues = new Map<string, IssuesManager.Issue.Issue>();
#nodeElementToIssue = new Map<Element, IssuesManager.Issue.Issue[]>();
#highlights: Range[] = [];
readonly tagTypeContext: TagTypeContext;
#adornersThrottler = new Common.Throttler.Throttler(100);
#containerAdornerActive = false;
#flexAdornerActive = false;
#gridAdornerActive = false;
#popoverAdornerActive = false;
#scrollSnapAdornerActive = false;
#startingStyleAdornerActive = false;
#layout: SDK.CSSModel.LayoutProperties|null = null;
#decorations: Decoration[] = [];
#descendantDecorations: Decoration[] = [];
#decorationsTooltip = '';
#updateRecord: Elements.ElementUpdateRecord.ElementUpdateRecord|null = null;
// Used to add the content to TreeElement's title element.
// Relied on by web tests.
#contentElement?: HTMLElement;
constructor(node: SDK.DOMModel.DOMNode, isClosingTag?: boolean) {
// The title will be updated in onattach.
super();
this.nodeInternal = node;
this.treeOutline = null;
this.listItemElement.setAttribute(
'jslog', `${VisualLogging.treeItem().parent('elementsTreeOutline').track({
keydown: 'ArrowUp|ArrowDown|ArrowLeft|ArrowRight|Backspace|Delete|Enter|Space|Home|End',
drag: true,
click: true,
})}`);
this.searchQuery = null;
this.#expandedChildrenLimit = InitialChildrenLimit;
this.decorationsThrottler = new Common.Throttler.Throttler(100);
this.inClipboard = false;
this.#hovered = false;
this.editing = null;
if (isClosingTag) {
this.tagTypeContext = {tagType: TagType.CLOSING};
} else {
this.tagTypeContext = {
tagType: TagType.OPENING,
canAddAttributes: this.nodeInternal.nodeType() === Node.ELEMENT_NODE,
};
void this.#updateAdorners();
}
this.expandAllButtonElement = null;
this.performUpdate();
if (this.nodeInternal.retained && !this.isClosingTag()) {
const icon = new Icon();
icon.name = 'small-status-dot';
icon.style.color = 'var(--icon-error)';
icon.classList.add('extra-small');
icon.style.setProperty('vertical-align', 'middle');
this.setLeadingIcons([icon]);
this.listItemNode.classList.add('detached-elements-detached-node');
this.listItemNode.style.setProperty('display', '-webkit-box');
this.listItemNode.setAttribute('title', 'Retained Node');
}
if (this.nodeInternal.detached && !this.isClosingTag()) {
this.listItemNode.setAttribute('title', 'Detached Tree Node');
}
}
static animateOnDOMUpdate(treeElement: ElementsTreeElement): void {
const tagName = treeElement.listItemElement.querySelector('.webkit-html-tag-name');
UI.UIUtils.runCSSAnimationOnce(tagName || treeElement.listItemElement, DOM_UPDATE_ANIMATION_CLASS_NAME);
}
static visibleShadowRoots(node: SDK.DOMModel.DOMNode): SDK.DOMModel.DOMNode[] {
let roots = node.shadowRoots();
if (roots.length && !Common.Settings.Settings.instance().moduleSetting('show-ua-shadow-dom').get()) {
roots = roots.filter(filter);
}
function filter(root: SDK.DOMModel.DOMNode): boolean {
return root.shadowRootType() !== SDK.DOMModel.DOMNode.ShadowRootTypes.UserAgent;
}
return roots;
}
static canShowInlineText(node: SDK.DOMModel.DOMNode): boolean {
if (node.contentDocument() || node.templateContent() || ElementsTreeElement.visibleShadowRoots(node).length ||
node.hasPseudoElements()) {
return false;
}
if (node.nodeType() !== Node.ELEMENT_NODE) {
return false;
}
if (!node.firstChild || node.firstChild !== node.lastChild || node.firstChild.nodeType() !== Node.TEXT_NODE) {
return false;
}
const textChild = node.firstChild;
const maxInlineTextChildLength = 80;
if (textChild.nodeValue().length < maxInlineTextChildLength) {
return true;
}
return false;
}
static populateForcedPseudoStateItems(contextMenu: UI.ContextMenu.ContextMenu, node: SDK.DOMModel.DOMNode): void {
const pseudoClasses = ['active', 'hover', 'focus', 'visited', 'focus-within', 'focus-visible'];
const forcedPseudoState = node.domModel().cssModel().pseudoState(node);
const stateMenu =
contextMenu.debugSection().appendSubMenuItem(i18nString(UIStrings.forceState), false, 'force-state');
for (const pseudoClass of pseudoClasses) {
const pseudoClassForced = forcedPseudoState ? forcedPseudoState.indexOf(pseudoClass) >= 0 : false;
stateMenu.defaultSection().appendCheckboxItem(
':' + pseudoClass, setPseudoStateCallback.bind(null, pseudoClass, !pseudoClassForced),
{checked: pseudoClassForced, jslogContext: pseudoClass});
}
function setPseudoStateCallback(pseudoState: string, enabled: boolean): void {
node.domModel().cssModel().forcePseudoState(node, pseudoState, enabled);
}
}
// ClearNode param is used to clean DOM after in-place editing..
performUpdate(clearNode = false): void {
if (this.editing) {
return;
}
const output: ViewOutput = {};
DEFAULT_VIEW(
{
node: !clearNode ? this.nodeInternal : null,
isClosingTag: this.isClosingTag(),
expanded: this.expanded,
isExpandable: this.isExpandable(),
isXMLMimeType: Boolean(this.treeOutline?.isXMLMimeType),
updateRecord: this.#updateRecord,
onHighlightSearchResults: () => this.#highlightSearchResults(),
onExpand: () => this.expand(),
containerAdornerActive: this.#containerAdornerActive,
showAdAdorner: this.nodeInternal.isAdFrameNode(),
showContainerAdorner: Boolean(this.#layout?.containerType) && !this.isClosingTag(),
containerType: this.#layout?.containerType,
showFlexAdorner: Boolean(this.#layout?.isFlex) && !this.isClosingTag(),
flexAdornerActive: this.#flexAdornerActive,
showGridAdorner: Boolean(this.#layout?.isGrid) && !this.isClosingTag(),
showGridLanesAdorner: Boolean(this.#layout?.isGridLanes) && !this.isClosingTag(),
showMediaAdorner: this.node().isMediaNode() && !this.isClosingTag(),
showPopoverAdorner: Boolean(Root.Runtime.hostConfig.devToolsAllowPopoverForcing?.enabled) &&
Boolean(this.node().attributes().find(attr => attr.name === 'popover')) && !this.isClosingTag(),
showTopLayerAdorner: this.node().topLayerIndex() !== -1 && !this.isClosingTag(),
gridAdornerActive: this.#gridAdornerActive,
popoverAdornerActive: this.#popoverAdornerActive,
isSubgrid: Boolean(this.#layout?.isSubgrid),
showViewSourceAdorner: this.nodeInternal.isRootNode() && isOpeningTag(this.tagTypeContext),
showScrollAdorner: ((this.node().nodeName() === 'HTML' && this.node().ownerDocument?.isScrollable()) ||
(this.node().nodeName() !== '#document' && this.node().isScrollable())) &&
!this.isClosingTag(),
decorations: this.#decorations,
descendantDecorations: this.expanded ? [] : this.#descendantDecorations,
decorationsTooltip: this.#decorationsTooltip,
indent: this.computeLeftIndent(),
showScrollSnapAdorner: Boolean(this.#layout?.hasScroll) && !this.isClosingTag(),
scrollSnapAdornerActive: this.#scrollSnapAdornerActive,
showSlotAdorner: Boolean(this.nodeInternal.assignedSlot) && !this.isClosingTag(),
showStartingStyleAdorner: this.nodeInternal.affectedByStartingStyles() && !this.isClosingTag(),
startingStyleAdornerActive: this.#startingStyleAdornerActive,
onStartingStyleAdornerClick: (event: Event) => this.#onStartingStyleAdornerClick(event),
onSlotAdornerClick: () => {
if (this.nodeInternal.assignedSlot) {
const deferredNode = this.nodeInternal.assignedSlot.deferredNode;
deferredNode.resolve(node => {
void Common.Revealer.reveal(node);
});
}
},
topLayerIndex: this.node().topLayerIndex(),
onViewSourceAdornerClick: this.revealHTMLInSources.bind(this),
onGutterClick: this.showContextMenu.bind(this),
onContainerAdornerClick: (event: Event) => this.#onContainerAdornerClick(event),
onFlexAdornerClick: (event: Event) => this.#onFlexAdornerClick(event),
onGridAdornerClick: (event: Event) => this.#onGridAdornerClick(event),
onMediaAdornerClick: (event: Event) => this.#onMediaAdornerClick(event),
onPopoverAdornerClick: (event: Event) => this.#onPopoverAdornerClick(event),
onScrollSnapAdornerClick: (event: Event) => this.#onScrollSnapAdornerClick(event),
onTopLayerAdornerClick: () => {
if (!this.treeOutline) {
return;
}
this.treeOutline.revealInTopLayer(this.node());
},
isHovered: this.#hovered,
isSelected: this.selected,
showAiButton: Boolean(this.#hovered || this.selected) && this.node().nodeType() === Node.ELEMENT_NODE &&
UI.ActionRegistry.ActionRegistry.instance().hasAction('freestyler.elements-floating-button'),
aiButtonTitle: UI.ActionRegistry.ActionRegistry.instance().hasAction('freestyler.elements-floating-button') ?
UI.ActionRegistry.ActionRegistry.instance().getAction('freestyler.elements-floating-button').title() :
undefined,
onAiButtonClick: (ev: Event) => {
ev.stopPropagation();
this.select(true, false);
const action = UI.ActionRegistry.ActionRegistry.instance().getAction('freestyler.elements-floating-button');
if (action) {
void action.execute();
}
},
},
output, this.listItemElement);
this.#contentElement = output.contentElement;
if (this.#updateRecord) {
this.#updateRecord = null;
}
}
#onContainerAdornerClick(event: Event): void {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
const model = node.domModel().overlayModel();
if (model.isHighlightedContainerQueryInPersistentOverlay(nodeId)) {
model.hideContainerQueryInPersistentOverlay(nodeId);
this.#containerAdornerActive = false;
} else {
model.highlightContainerQueryInPersistentOverlay(nodeId);
this.#containerAdornerActive = true;
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.MODERN_DOM_BADGE_CLICKED);
}
void this.updateAdorners();
}
#onFlexAdornerClick(event: Event): void {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
const model = node.domModel().overlayModel();
if (model.isHighlightedFlexContainerInPersistentOverlay(nodeId)) {
model.hideFlexContainerInPersistentOverlay(nodeId);
this.#flexAdornerActive = false;
} else {
model.highlightFlexContainerInPersistentOverlay(nodeId);
this.#flexAdornerActive = true;
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.MODERN_DOM_BADGE_CLICKED);
}
void this.updateAdorners();
}
#onGridAdornerClick(event: Event): void {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
const model = node.domModel().overlayModel();
if (model.isHighlightedGridInPersistentOverlay(nodeId)) {
model.hideGridInPersistentOverlay(nodeId);
this.#gridAdornerActive = false;
} else {
model.highlightGridInPersistentOverlay(nodeId);
this.#gridAdornerActive = true;
if (this.#layout?.isSubgrid) {
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.MODERN_DOM_BADGE_CLICKED);
}
}
void this.updateAdorners();
}
async #onMediaAdornerClick(event: Event): Promise<void> {
event.stopPropagation();
await UI.ViewManager.ViewManager.instance().showView('medias');
const view = UI.ViewManager.ViewManager.instance().view('medias');
if (view) {
const widget = await view.widget();
if (widget instanceof Media.MainView.MainView) {
await widget.waitForInitialPlayers();
widget.selectPlayerByDOMNodeId(this.node().backendNodeId());
}
}
}
highlightAttribute(attributeName: string): void {
// If the attribute is not found, we highlight the tag name instead.
let animationElement = this.listItemElement.querySelector('.webkit-html-tag-name') ?? this.listItemElement;
if (this.nodeInternal.getAttribute(attributeName) !== undefined) {
const tag = this.listItemElement.getElementsByClassName('webkit-html-tag')[0];
const attributes = tag.getElementsByClassName('webkit-html-attribute');
for (const attribute of attributes) {
const attributeElement = attribute.getElementsByClassName('webkit-html-attribute-name')[0];
if (attributeElement.textContent === attributeName) {
animationElement = attributeElement;
break;
}
}
}
UI.UIUtils.runCSSAnimationOnce(animationElement, DOM_UPDATE_ANIMATION_CLASS_NAME);
}
isClosingTag(): boolean {
return !isOpeningTag(this.tagTypeContext);
}
node(): SDK.DOMModel.DOMNode {
return this.nodeInternal;
}
isEditing(): boolean {
return Boolean(this.editing);
}
highlightSearchResults(searchQuery: string): void {
this.searchQuery = searchQuery;
if (!this.editing) {
this.#highlightSearchResults();
}
}
hideSearchHighlights(): void {
Highlighting.HighlightManager.HighlightManager.instance().removeHighlights(this.#highlights);
this.#highlights = [];
}
setInClipboard(inClipboard: boolean): void {
if (this.inClipboard === inClipboard) {
return;
}
this.inClipboard = inClipboard;
this.listItemElement.classList.toggle('in-clipboard', inClipboard);
}
get hovered(): boolean {
return this.#hovered;
}
set hovered(isHovered: boolean) {
if (this.#hovered === isHovered) {
return;
}
this.#hovered = isHovered;
if (this.listItemElement) {
if (isHovered) {
this.listItemElement.classList.add('hovered');
} else {
this.listItemElement.classList.remove('hovered');
}
this.performUpdate();
}
}
addIssue(newIssue: IssuesManager.Issue.Issue): void {
if (this.#elementIssues.has(newIssue.primaryKey())) {
return;
}
this.#elementIssues.set(newIssue.primaryKey(), newIssue);
this.#applyIssueStyleAndTooltip(newIssue);
}
#applyIssueStyleAndTooltip(issue: IssuesManager.Issue.Issue): void {
const elementIssueDetails = getElementIssueDetails(issue);
if (!elementIssueDetails) {
return;
}
if (elementIssueDetails.attribute) {
this.#highlightViolatingAttr(elementIssueDetails.attribute, issue);
} else {
this.#highlightTagAsViolating(issue);
}
}
get issuesByNodeElement(): Map<Element, IssuesManager.Issue.Issue[]> {
return this.#nodeElementToIssue;
}
#highlightViolatingAttr(name: string, issue: IssuesManager.Issue.Issue): void {
const tag = this.listItemElement.getElementsByClassName('webkit-html-tag')[0];
const attributes = tag.getElementsByClassName('webkit-html-attribute');
for (const attribute of attributes) {
if (attribute.getElementsByClassName('webkit-html-attribute-name')[0].textContent === name) {
const attributeElement = attribute.getElementsByClassName('webkit-html-attribute-name')[0];
attributeElement.classList.add('violating-element');
this.#updateNodeElementToIssue(attributeElement, issue);
}
}
}
#highlightTagAsViolating(issue: IssuesManager.Issue.Issue): void {
const tagElement = this.listItemElement.getElementsByClassName('webkit-html-tag-name')[0];
tagElement.classList.add('violating-element');
this.#updateNodeElementToIssue(tagElement, issue);
}
#updateNodeElementToIssue(nodeElement: Element, issue: IssuesManager.Issue.Issue): void {
let issues = this.#nodeElementToIssue.get(nodeElement);
if (!issues) {
issues = [];
this.#nodeElementToIssue.set(nodeElement, issues);
}
issues.push(issue);
this.treeOutline?.updateNodeElementToIssue(nodeElement, issues);
}
expandedChildrenLimit(): number {
return this.#expandedChildrenLimit;
}
setExpandedChildrenLimit(expandedChildrenLimit: number): void {
this.#expandedChildrenLimit = expandedChildrenLimit;
}
onTopLayerIndexChanged(): void {
this.performUpdate();
}
override onbind(): void {
this.performUpdate();
if (this.treeOutline && !this.isClosingTag()) {
this.treeOutline.treeElementByNode.set(this.nodeInternal, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.TOP_LAYER_INDEX_CHANGED, this.onTopLayerIndexChanged, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.SCROLLABLE_FLAG_UPDATED, this.#onScrollableFlagUpdated, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.CONTAINER_QUERY_OVERLAY_STATE_CHANGED,
this.#onPersistentContainerQueryOverlayStateChanged, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.FLEX_CONTAINER_OVERLAY_STATE_CHANGED,
this.#onPersistentFlexContainerOverlayStateChanged, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.GRID_OVERLAY_STATE_CHANGED, this.#onPersistentGridOverlayStateChanged, this);
this.nodeInternal.addEventListener(
SDK.DOMModel.DOMNodeEvents.SCROLL_SNAP_OVERLAY_STATE_CHANGED, this.#onPersistentScrollSnapOverlayStateChanged,
this);
}
}
override onunbind(): void {
if (this.editing) {
this.editing.cancel();
}
// Update the element to clean up adorner registrations with the
// ElementsPanel.
// We do not change the ElementsTreeElement state in case the
// element is bound again.
DEFAULT_VIEW(
{
node: null,
isClosingTag: false,
expanded: false,
isExpandable: false,
isXMLMimeType: false,
updateRecord: null,
onHighlightSearchResults: () => {},
onExpand: () => {},
containerAdornerActive: false,
showAdAdorner: false,
showContainerAdorner: false,
containerType: this.#layout?.containerType,
showFlexAdorner: false,
flexAdornerActive: false,
showGridAdorner: false,
showGridLanesAdorner: false,
showMediaAdorner: false,
showPopoverAdorner: false,
showTopLayerAdorner: false,
gridAdornerActive: false,
popoverAdornerActive: false,
isSubgrid: false,
showViewSourceAdorner: false,
showScrollAdorner: false,
showScrollSnapAdorner: false,
scrollSnapAdornerActive: false,
showSlotAdorner: false,
showStartingStyleAdorner: false,
startingStyleAdornerActive: false,
onStartingStyleAdornerClick: () => {},
onSlotAdornerClick: () => {},
topLayerIndex: -1,
onViewSourceAdornerClick: () => {},
onGutterClick: () => {},
onContainerAdornerClick: () => {},
onFlexAdornerClick: () => {},
onGridAdornerClick: () => {},
onMediaAdornerClick: () => {},
onPopoverAdornerClick: () => {},
onScrollSnapAdornerClick: () => {},
onTopLayerAdornerClick: () => {},
isHovered: false,
isSelected: false,
showAiButton: false,
onAiButtonClick: () => {},
decorations: [],
descendantDecorations: [],
decorationsTooltip: '',
indent: 0,
},
{}, this.listItemElement);
if (this.treeOutline && this.treeOutline.treeElementByNode.get(this.nodeInternal) === this) {
this.treeOutline.treeElementByNode.delete(this.nodeInternal);
}
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.TOP_LAYER_INDEX_CHANGED, this.onTopLayerIndexChanged, this);
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.SCROLLABLE_FLAG_UPDATED, this.#onScrollableFlagUpdated, this);
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.CONTAINER_QUERY_OVERLAY_STATE_CHANGED,
this.#onPersistentContainerQueryOverlayStateChanged, this);
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.FLEX_CONTAINER_OVERLAY_STATE_CHANGED,
this.#onPersistentFlexContainerOverlayStateChanged, this);
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.GRID_OVERLAY_STATE_CHANGED, this.#onPersistentGridOverlayStateChanged, this);
this.nodeInternal.removeEventListener(
SDK.DOMModel.DOMNodeEvents.SCROLL_SNAP_OVERLAY_STATE_CHANGED, this.#onPersistentScrollSnapOverlayStateChanged,
this);
}
#onScrollableFlagUpdated(): void {
void this.#updateAdorners();
}
#onPersistentContainerQueryOverlayStateChanged(event: Common.EventTarget.EventTargetEvent<{enabled: boolean}>): void {
this.#containerAdornerActive = event.data.enabled;
this.performUpdate();
}
#onPersistentFlexContainerOverlayStateChanged(event: Common.EventTarget.EventTargetEvent<{enabled: boolean}>): void {
this.#flexAdornerActive = event.data.enabled;
this.performUpdate();
}
#onPersistentGridOverlayStateChanged(event: Common.EventTarget.EventTargetEvent<{enabled: boolean}>): void {
this.#gridAdornerActive = event.data.enabled;
this.performUpdate();
}
#onPersistentScrollSnapOverlayStateChanged(event: Common.EventTarget.EventTargetEvent<{enabled: boolean}>): void {
this.#scrollSnapAdornerActive = event.data.enabled;
this.performUpdate();
}
#onScrollSnapAdornerClick(event: Event): void {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
const model = node.domModel().overlayModel();
if (this.#scrollSnapAdornerActive) {
model.hideScrollSnapInPersistentOverlay(nodeId);
} else {
model.highlightScrollSnapInPersistentOverlay(nodeId);
}
}
override onattach(): void {
if (this.#hovered) {
this.listItemElement.classList.add('hovered');
this.performUpdate();
}
this.updateTitle();
this.listItemElement.draggable = true;
}
override async onpopulate(): Promise<void> {
if (this.treeOutline) {
return await this.treeOutline.populateTreeElement(this);
}
}
override async expandRecursively(): Promise<void> {
await this.nodeInternal.getSubtree(100, true);
await super.expandRecursively(Number.MAX_VALUE);
}
override onexpand(): void {
if (this.isClosingTag()) {
return;
}
this.updateTitle();
}
override oncollapse(): void {
if (this.isClosingTag()) {
return;
}
this.updateTitle();
}
override select(omitFocus?: boolean, selectedByUser?: boolean): boolean {
if (this.editing) {
return false;
}
return super.select(omitFocus, selectedByUser);
}
override onselect(selectedByUser?: boolean): boolean {
if (!this.treeOutline) {
return false;
}
this.treeOutline.suppressRevealAndSelect = true;
this.treeOutline.selectDOMNode(this.nodeInternal, selectedByUser);
if (selectedByUser) {
this.nodeInternal.highlight();
Host.userMetrics.actionTaken(Host.UserMetrics.Action.ChangeInspectedNodeInElementsPanel);
}
this.performUpdate();
this.treeOutline.suppressRevealAndSelect = false;
return true;
}
override ondelete(): boolean {
if (!this.treeOutline) {
return false;
}
const startTagTreeElement = this.treeOutline.findTreeElement(this.nodeInternal);
startTagTreeElement ? (void startTagTreeElement.remove()) : (void this.remove());
return true;
}
override onenter(): boolean {
// On Enter or Return start editing the first attribute
// or create a new attribute on the selected element.
if (this.editing) {
return false;
}
this.startEditing();
// prevent a newline from being immediately inserted
return true;
}
override selectOnMouseDown(event: MouseEvent): void {
super.selectOnMouseDown(event);
if (this.editing) {
return;
}
// Prevent selecting the nearest word on double click.
if (event.detail >= 2) {
event.preventDefault();
}
}
override ondblclick(event: Event): boolean {
if (this.editing || this.isClosingTag()) {
return false;
}
if (this.startEditingTarget((event.target as Element))) {
return false;
}
if (this.isExpandable() && !this.expanded) {
this.expand();
}
return false;
}
hasEditableNode(): boolean {
return !this.nodeInternal.isShadowRoot() && !this.nodeInternal.ancestorUserAgentShadowRoot();
}
private insertInLastAttributePosition(tag: Element, node: Element): void {
if (tag.getElementsByClassName('webkit-html-attribute').length > 0) {
tag.insertBefore(node, tag.lastChild);
} else if (tag.textContent !== null) {
const matchResult = tag.textContent.match(/^<(.*?)>$/);
if (!matchResult) {
return;
}
const nodeName = matchResult[1];
tag.textContent = '';
UI.UIUtils.createTextChild(tag, '<' + nodeName);
tag.appendChild(node);
UI.UIUtils.createTextChild(tag, '>');
}
}
private startEditingTarget(eventTarget: Element): boolean {
if (!this.treeOutline || this.treeOutline.selectedDOMNode() !== this.nodeInternal) {
return false;
}
if (this.nodeInternal.nodeType() !== Node.ELEMENT_NODE && this.nodeInternal.nodeType() !== Node.TEXT_NODE) {
return false;
}
const textNode = eventTarget.enclosingNodeOrSelfWithClass('webkit-html-text-node');
if (textNode) {
return this.startEditingTextNode(textNode);
}
const attribute = eventTarget.enclosingNodeOrSelfWithClass('webkit-html-attribute');
if (attribute) {
return this.startEditingAttribute(attribute, eventTarget);
}
const tagName = eventTarget.enclosingNodeOrSelfWithClass('webkit-html-tag-name');
if (tagName) {
return this.startEditingTagName(tagName);
}
const newAttribute = eventTarget.enclosingNodeOrSelfWithClass('add-attribute');
if (newAttribute) {
return this.addNewAttribute();
}
return false;
}
private showContextMenu(event: Event): void {
this.treeOutline && void this.treeOutline.showContextMenu(this, event);
}
private revealHTMLInSources(): void {
const frameOwnerId = this.nodeInternal.frameOwnerFrameId();
if (frameOwnerId) {
const frame = SDK.FrameManager.FrameManager.instance().getFrame(frameOwnerId);
if (frame) {
const sourceCode = Workspace.Workspace.WorkspaceImpl.instance().uiSourceCodeForURL(frame.url);
void Common.Revealer.reveal(sourceCode);
}
}
}
async populateTagContextMenu(contextMenu: UI.ContextMenu.ContextMenu, event: Event): Promise<void> {
// Add attribute-related actions.
const treeElement =
this.isClosingTag() && this.treeOutline ? this.treeOutline.findTreeElement(this.nodeInternal) : this;
if (!treeElement) {
return;
}
contextMenu.editSection().appendItem(
i18nString(UIStrings.addAttribute), treeElement.addNewAttribute.bind(treeElement),
{jslogContext: 'add-attribute'});
const target = (event.target as Element);
const attribute = target.enclosingNodeOrSelfWithClass('webkit-html-attribute');
const newAttribute = target.enclosingNodeOrSelfWithClass('add-attribute');
if (attribute && !newAttribute) {
contextMenu.editSection().appendItem(
i18nString(UIStrings.editAttribute), this.startEditingAttribute.bind(this, attribute, target),
{jslogContext: 'edit-attribute'});
}
await this.populateNodeContextMenu(contextMenu);
ElementsTreeElement.populateForcedPseudoStateItems(contextMenu, treeElement.node());
this.populateScrollIntoView(contextMenu);
contextMenu.viewSection().appendItem(i18nString(UIStrings.focus), async () => {
await this.nodeInternal.focus();
}, {jslogContext: 'focus'});
}
populatePseudoElementContextMenu(contextMenu: UI.ContextMenu.ContextMenu): void {
if (this.childCount() !== 0) {
this.populateExpandRecursively(contextMenu);
}
this.populateScrollIntoView(contextMenu);
}
private populateExpandRecursively(contextMenu: UI.ContextMenu.ContextMenu): void {
contextMenu.viewSection().appendItem(
i18nString(UIStrings.expandRecursively), this.expandRecursively.bind(this),
{jslogContext: 'expand-recursively'});
}
private populateScrollIntoView(contextMenu: UI.ContextMenu.ContextMenu): void {
contextMenu.viewSection().appendItem(
i18nString(UIStrings.scrollIntoView), () => this.nodeInternal.scrollIntoView(),
{jslogContext: 'scroll-into-view'});
}
async populateTextContextMenu(contextMenu: UI.ContextMenu.ContextMenu, textNode: Element): Promise<void> {
if (!this.editing) {
contextMenu.editSection().appendItem(
i18nString(UIStrings.editText), this.startEditingTextNode.bind(this, textNode), {jslogContext: 'edit-text'});
}
return await this.populateNodeContextMenu(contextMenu);
}
async populateNodeContextMenu(contextMenu: UI.ContextMenu.ContextMenu): Promise<void> {
// Add free-form node-related actions.
const isEditable = this.hasEditableNode();
// clang-format off
if (isEditable && !this.editing) {
contextMenu.editSection().appendItem(i18nString(UIStrings.editAsHtml), this.editAsHTML.bind(this), { jslogContext: 'elements.edit-as-html' });
}
// clang-format on
const isShadowRoot = this.nodeInternal.isShadowRoot();
const createShortcut = UI.KeyboardShortcut.KeyboardShortcut.shortcutToString.bind(null);
const modifier = UI.KeyboardShortcut.Modifiers.CtrlOrMeta.value;
const treeOutline = this.treeOutline;
if (!treeOutline) {
return;
}
let menuItem;
const openAiAssistanceId = 'freestyler.element-panel-context';
if (UI.ActionRegistry.ActionRegistry.instance().hasAction(openAiAssistanceId)) {
function appendSubmenuPromptAction(
submenu: UI.ContextMenu.SubMenu, action: UI.ActionRegistration.Action, label: Common.UIString.LocalizedString,
prompt: string, jslogContext: string): void {
submenu.defaultSection().appendItem(label, () => {
void action.execute({prompt});
UI.UIUtils.PromotionManager.instance().recordFeatureInteraction(openAiAssistanceId);
}, {disabled: !action.enabled(), jslogContext});
}
UI.Context.Context.instance().setFlavor(SDK.DOMModel.DOMNode, this.nodeInternal);
const action = UI.ActionRegistry.ActionRegistry.instance().getAction(openAiAssistanceId);
const submenu = contextMenu.footerSection().appendSubMenuItem(action.title(), false, openAiAssistanceId);
submenu.defaultSection().appendAction(openAiAssistanceId, i18nString(UIStrings.startAChat));
const submenuConfigs = [
{
condition: (props: SDK.CSSModel.LayoutProperties|null): boolean => Boolean(props?.isFlex),
items: [
{
label: i18nString(UIStrings.wrapTheseItems),
prompt: 'How can I make flex items wrap?',
jslogContextSuffix: '.flex-wrap',
},
{
label: i18nString(UIStrings.distributeItemsEvenly),
prompt: 'How do I distribute flex items evenly?',
jslogContextSuffix: '.flex-distribute',
},
{
label: i18nString(UIStrings.explainFlexbox),
prompt: 'What is flexbox?',
jslogContextSuffix: '.flex-what',
},
],
},
{
condition: (props: SDK.CSSModel.LayoutProperties|null): boolean =>
Boolean(props?.isGrid && !props?.isSubgrid),
items: [
{
label: i18nString(UIStrings.alignItems),
prompt: 'How do I align items in a grid?',
jslogContextSuffix: '.grid-align',
},
{
label: i18nString(UIStrings.addPadding),
prompt: 'How to add spacing between grid items?',
jslogContextSuffix: '.grid-gap',
},
{
label: i18nString(UIStrings.explainGridLayout),
prompt: 'How does grid layout work?',
jslogContextSuffix: '.grid-how',
},
],
},
{
condition: (props: SDK.CSSModel.LayoutProperties|null): boolean => Boolean(props?.isSubgrid),
items: [
{
label: i18nString(UIStrings.findGridDefinition),
prompt: 'Where is this grid defined?',
jslogContextSuffix: '.subgrid-where',
},
{
label: i18nString(UIStrings.changeParentProperties),
prompt: 'How to overwrite parent grid properties?',
jslogContextSuffix: '.subgrid-override',
},
{
label: i18nString(UIStrings.explainSubgrids),
prompt: 'How do subgrids work?',
jslogContextSuffix: '.subgrid-how',
},
],
},
{
condition: (props: SDK.CSSModel.LayoutProperties|null): boolean => Boolean(props?.hasScroll),
items: [
{
label: i18nString(UIStrings.removeScrollbars),
prompt: 'How do I remove scrollbars for this element?',
jslogContextSuffix: '.scroll-remove',
},
{
label: i18nString(UIStrings.styleScrollbars),
prompt: 'How can I style a scrollbar?',
jslogContextSuffix: '.scroll-style',
},
{
label: i18nString(UIStrings.explainScrollbars),
prompt: 'Why does this element scroll?',
jslogContextSuffix: '.scroll-why',
},
],
},
{
condition: (props: SDK.CSSModel.LayoutProperties|null): boolean => Boolean(props?.containerType),
items: [
{
label: i18nString(UIStrings.explainContainerQueries),
prompt: 'What are container queries?',
jslogContextSuffix: '.container-what',
},
{
label: i18nString(UIStrings.explainContainerTypes),
prompt: 'How do I use container-type?',
jslogContextSuffix: '.container-how',
},
{
label: i18nString(UIStrings.explainContainerContext),
prompt: 'What\'s the container context for this element?',
jslogContextSuffix: '.container-context',
},
],
},
{
// Default items
condition: (): boolean => true,
items: [
{
label: i18nString(UIStrings.assessVisibility),
prompt: 'Why isn’t this element visible?',
jslogContextSuffix: '.visibility',
},
{
label: i18nString(UIStrings.centerElement),
prompt: 'How do I center this element?',
jslogContextSuffix: '.center',
},
],
},
];
const layoutProps =
await this.nodeInternal.domModel().cssModel().getLayoutPropertiesFromComputedStyle(this.nodeInternal.id);
const config = submenuConfigs.find(c => c.condition(layoutProps));
if (config) {
for (const item of config.items) {
appendSubmenuPromptAction(
submenu, action, item.label, item.prompt, openAiAssistanceId + item.jslogContextSuffix);
}
}
}
menuItem = contextMenu.clipboardSection().appendItem(
i18nString(UIStrings.cut), treeOutline.performCopyOrCut.bind(treeOutline, true, this.nodeInternal),
{disabled: !this.hasEditableNode(), jslogContext: 'cut'});
menuItem.setShortcut(createShortcut('X', modifier));
// Place it here so that all "Copy"-ing items stick together.
const copyMenu = contextMenu.clipboardSection().appendSubMenuItem(i18nString(UIStrings.copy), false, 'copy');
const section = copyMenu.section();
if (!isShadowRoot) {
menuItem = section.appendItem(
i18nString(UIStrings.copyOuterhtml), treeOutline.performCopyOrCut.bind(treeOutline, false, this.nodeInternal),
{jslogContext: 'copy-outer-html'});
menuItem.setShortcut(createShortcut('V', modifier));
}
if (this.nodeInternal.nodeType() === Node.ELEMENT_NODE) {
section.appendItem(
i18nString(UIStrings.copySelector), this.copyCSSPath.bind(this), {jslogContext: 'copy-selector'});
section.appendItem(
i18nString(UIStrings.copyJsPath), this.copyJSPath.bind(this),
{disabled: !canGetJSPath(this.nodeInternal), jslogContext: 'copy-js-path'});
section.appendItem(
i18nString(UIStrings.copyStyles), this.copyStyles.bind(this), {jslogContext: 'elements.copy-styles'});
}
if (!isShadowRoot) {
section.appendItem(i18nString(UIStrings.copyXpath), this.copyXPath.bind(this), {jslogContext: 'copy-xpath'});
section.appendItem(
i18nString(UIStrings.copyFullXpath), this.copyFullXPath.bind(this), {jslogContext: 'copy-full-xpath'});
}
menuItem = copyMenu.clipboardSection().appendItem(
i18nString(UIStrings.copyElement),
treeOutline.performCopyOrCut.bind(treeOutline, false, this.nodeInternal, true), {jslogContext: 'copy-element'});
menuItem.setShortcut(createShortcut('C', modifier));
if (!isShadowRoot) {
// Duplicate element, disabled on root element and ShadowDOM.
const isRootElement = !this.nodeInternal.parentNode || this.nodeInternal.parentNode.nodeName() === '#document';
menuItem = contextMenu.editSection().appendItem(
i18nString(UIStrings.duplicateElement), treeOutline.duplicateNode.bind(treeOutline, this.nodeInternal), {
disabled: (this.nodeInternal.isInShadowTree() || isRootElement),
jslogContext: 'elements.duplicate-element',
});
}
menuItem = contextMenu.clipboardSection().appendItem(
i18nString(UIStrings.paste), treeOutline.pasteNode.bind(treeOutline, this.nodeInternal),
{disabled: !treeOutline.canPaste(this.nodeInternal), jslogContext: 'paste'});
menuItem.setShortcut(createShortcut('V', modifier));
menuItem = contextMenu.debugSection().appendCheckboxItem(
i18nString(UIStrings.hideElement), treeOutline.toggleHideElement.bind(treeOutline, this.nodeInternal),
{checked: treeOutline.isToggledToHidden(this.nodeInternal), jslogContext: 'elements.hide-element'});
menuItem.setShortcut(
UI.ShortcutRegistry.ShortcutRegistry.instance().shortcutTitleForAction('elements.hide-element') || '');
if (isEditable) {
contextMenu.editSection().appendItem(
i18nString(UIStrings.deleteElement), this.remove.bind(this), {jslogContext: 'delete-element'});
}
this.populateExpandRecursively(contextMenu);
contextMenu.viewSection().appendItem(
i18nString(UIStrings.collapseChildren), this.collapseChildren.bind(this), {jslogContext: 'collapse-children'});
const deviceModeWrapperAction = new Emulation.DeviceModeWrapper.ActionDelegate();
contextMenu.viewSection().appendItem(
i18nString(UIStrings.captureNodeScreenshot),
deviceModeWrapperAction.handleAction.bind(
null, UI.Context.Context.instance(), 'emulation.capture-node-screenshot'),
{jslogContext: 'emulation.capture-node-screenshot'});
if (this.nodeInternal.frameOwnerFrameId()) {
contextMenu.viewSection().appendItem(i18nString(UIStrings.showFrameDetails), () => {
const frameOwnerFrameId = this.nodeInternal.frameOwnerFrameId();
if (frameOwnerFrameId) {
const frame = SDK.FrameManager.FrameManager.instance().getFrame(frameOwnerFrameId);
void Common.Revealer.reveal(frame);
}
}, {jslogContext: 'show-frame-details'});
}
}
private startEditing(): boolean|undefined {
if (!this.treeOutline || this.treeOutline.selectedDOMNode() !== this.nodeInternal) {
return;
}
const listItem = this.listItemElement;
if (isOpeningTag(this.tagTypeContext) && this.tagTypeContext.canAddAttributes) {
const attribute = listItem.getElementsByClassName('webkit-html-attribute')[0];
if (attribute) {
return this.startEditingAttribute(
attribute, attribute.getElementsByClassName('webkit-html-attribute-value')[0]);
}
return this.addNewAttribute();
}
if (this.nodeInternal.nodeType() === Node.TEXT_NODE) {
const textNode = listItem.getElementsByClassName('webkit-html-text-node')[0];
if (textNode) {
return this.startEditingTextNode(textNode);
}
}
return;
}
private addNewAttribute(): boolean {
// Cannot just convert the textual html into an element without
// a parent node. Use a temporary span container for the HTML.
const container = document.createElement('span');
// eslint-disable-next-line @devtools/no-lit-render-outside-of-view
Lit.render(renderAttribute({name: ' ', value: ''}, null, false, this.nodeInternal), container);
const attr = container.firstElementChild as HTMLElement;
attr.style.marginLeft = '2px'; // overrides the .editing margin rule
attr.style.marginRight = '2px'; // overrides the .editing margin rule
attr.setAttribute('jslog', `${VisualLogging.value('new-attribute').track({change: true, resize: true})}`);
const tag = this.listItemElement.getElementsByClassName('webkit-html-tag')[0];
this.insertInLastAttributePosition(tag, attr);
attr.scrollIntoViewIfNeeded(true);
return this.startEditingAttribute(attr, attr);
}
private triggerEditAttribute(attributeName: string): boolean|undefined {
const attributeElements = this.listItemElement.getElementsByClassName('webkit-html-attribute-name');
for (let i = 0, len = attributeElements.length; i < len; ++i) {
if (attributeElements[i].textContent === attributeName) {
for (let elem: (ChildNode|null) = attributeElements[i].nextSibling; elem; elem = elem.nextSibling) {
if (elem.nodeType !== Node.ELEMENT_NODE) {
continue;
}
if ((elem as Element).classList.contains('webkit-html-attribute-value')) {
return this.startEditingAttribute((elem.parentElement as HTMLElement), (elem as Element));
}
}
}
}
return;
}
private startEditingAttribute(attribute: Element, elementForSelection: Element): boolean {
console.assert(this.listItemElement.isAncestor(attribute));
if (UI.UIUtils.isBeingEdited(attribute)) {
return true;
}
const attributeNameElement = attribute.getElementsByClassName('webkit-html-attribute-name')[0];
if (!attributeNameElement) {
return false;
}
const attributeName = attributeNameElement.textContent;
const attributeValueElement = attribute.getElementsByClassName('webkit-html-attribute-value')[0];
// Make sure elementForSelection is not a child of attributeValueElement.
elementForSelection =
attributeValueElement?.isAncestor(elementForSelection) ? attributeValueElement : elementForSelection;
function removeZeroWidthSpaceRecursive(node: Node): void {
if (node.nodeType === Node.TEXT_NODE) {
node.nodeValue = node.nodeValue ? node.nodeValue.replace(/\u200B/g, '') : '';
return;
}
if (node.nodeType !== Node.ELEMENT_NODE) {
return;
}
for (let child: (ChildNode|null) = node.firstChild; child; child = child.nextSibling) {
removeZeroWidthSpaceRecursive(child);
}
}
const attributeValue = attributeName && attributeValueElement ?
this.nodeInternal.getAttribute(attributeName)?.replaceAll('"', '&quot;') :
undefined;
if (attributeValue !== undefined) {
attributeValueElement.setTextContentTruncatedIfNeeded(
attributeValue, i18nString(UIStrings.valueIsTooLargeToEdit));
}
// Remove zero-width spaces that were added by nodeTitleInfo.
removeZeroWidthSpaceRecursive(attribute);
const config = new UI.InplaceEditor.Config(
this.attributeEditingCommitted.bind(this), this.editingCancelled.bind(this), attributeName);
function postKeyDownFinishHandler(event: Event): string {
UI.UIUtils.handleElementValueModifications(event, attribute);
return '';
}
if (!Common.ParsedURL.ParsedURL.fromString(attributeValueElement?.textContent || '')) {
config.setPostKeydownFinishHandler(postKeyDownFinishHandler);
}
this.updateEditorHandles(attribute, config);
const componentSelection = this.listItemElement.getComponentSelection();
componentSelection?.selectAllChildren(elementForSelection);
return true;
}
private startEditingTextNode(textNodeElement: Element): boolean {
if (UI.UIUtils.isBeingEdited(textNodeElement)) {
return true;
}
let textNode: SDK.DOMModel.DOMNode = this.nodeInternal;
// We only show text nodes inline in elements if the element only
// has a single child, and that child is a text node.
if (textNode.nodeType() === Node.ELEMENT_NODE && textNode.firstChild) {
textNode = textNode.firstChild;
}
const container = textNodeElement.enclosingNodeOrSelfWithClass('webkit-html-text-node');
if (container) {
container.textContent = textNode.nodeValue();
} // Strip the CSS or JS highlighting if present.
const config = new UI.InplaceEditor.Config(
this.textNodeEditingCommitted.bind(this, textNode), this.editingCancelled.bind(this), null);
this.updateEditorHandles(textNodeElement, config);
const componentSelection = this.listItemElement.getComponentSelection();
componentSelection?.selectAllChildren(textNodeElement);
return true;
}
private startEditingTagName(tagNameElement?: Element): boolean {
if (!tagNameElement) {
tagNameElement = this.listItemElement.getElementsByClassName('webkit-html-tag-name')[0];
if (!tagNameElement) {
return false;
}
}
const tagName = tagNameElement.textContent;
if (tagName !== null && EditTagBlocklist.has(tagName.toLowerCase())) {
return false;
}
if (UI.UIUtils.isBeingEdited(tagNameElement)) {
return true;
}
const closingTagElement = this.distinctClosingTagElement();
function keyupListener(): void {
if (closingTagElement && tagNameElement) {
closingTagElement.textContent = '</' + tagNameElement.textContent + '>';
}
}
const keydownListener = (event: Event): void => {
if ((event as KeyboardEvent).key !== ' ') {
return;
}
this.editing?.commit();
event.consume(true);
};
function editingCommitted(
this: ElementsTreeElement,
element: Element,
newTagName: string,
oldText: string|null,
tagName: string|null,
moveDirection: string,
): void {
if (!tagNameElement) {
return;
}
tagNameElement.removeEventListener('keyup', keyupListener, false);
tagNameElement.removeEventListener('keydown', keydownListener, false);
this.tagNameEditingCommitted(element, newTagName, oldText, tagName, moveDirection);
}
function editingCancelled(this: ElementsTreeElement, element: Element, tagName: string|null): void {
if (!tagNameElement) {
return;
}
tagNameElement.removeEventListener('keyup', keyupListener, false);
tagNameElement.removeEventListener('keydown', keydownListener, false);
this.editingCancelled(element, tagName);
}
tagNameElement.addEventListener('keyup', keyupListener, false);
tagNameElement.addEventListener('keydown', keydownListener, false);
const config =
new UI.InplaceEditor.Config<string|null>(editingCommitted.bind(this), editingCancelled.bind(this), tagName);
this.updateEditorHandles(tagNameElement, config);
const componentSelection = this.listItemElement.getComponentSelection();
componentSelection?.selectAllChildren(tagNameElement);
return true;
}
private updateEditorHandles<T>(element: Element, config: UI.InplaceEditor.Config<T>): void {
const editorHandles = UI.InplaceEditor.InplaceEditor.startEditing(element, config);
if (!editorHandles) {
this.editing = null;
} else {
this.editing = {
commit: editorHandles.commit,
cancel: editorHandles.cancel,
editor: undefined,
resize: () => {},
};
}
}
private async startEditingAsHTML(
commitCallback: (arg0: string, arg1: string) => void, disposeCallback: () => void,
maybeInitialValue: string|null): Promise<void> {
if (maybeInitialValue === null) {
return;
}
if (this.editing) {
return;
}
const initialValue = convertUnicodeCharsToHTMLEntities(maybeInitialValue).text;
this.htmlEditElement = document.createElement('div');
this.htmlEditElement.className = 'source-code elements-tree-editor';
// Hide header items.
let child: (ChildNode|null) = this.listItemElement.firstChild;
while (child) {
if (child instanceof HTMLElement) {
child.style.display = 'none';
}
child = child.nextSibling;
}
// Hide children item.
if (this.childrenListElement) {
this.childrenListElement.style.display = 'none';
}
// Append editor.
this.listItemElement.append(this.htmlEditElement);
this.htmlEditElement.addEventListener('keydown', event => {
if (event.key === 'Escape') {
event.consume(true);
}
});
const editor = new TextEditor.TextEditor.TextEditor(CodeMirror.EditorState.create({
doc: initialValue,
extensions: [
CodeMirror.keymap.of([
{
key: 'Mod-Enter',
run: () => {
this.editing?.commit();
return true;
},
},
{
key: 'Escape',
run: () => {
this.editing?.cancel();
return true;
},
},
]),
TextEditor.Config.baseConfiguration(initialValue),
TextEditor.Config.closeBrackets.instance(),
TextEditor.Config.autocompletion.instance(),
CodeMirror.html.html({autoCloseTags: false, selfClosingTags: true}),
TextEditor.Config.domWordWrap.instance(),
CodeMirror.EditorView.theme({
'&.cm-editor': {maxHeight: '300px'},
'.cm-scroller': {overflowY: 'auto'},
}),
CodeMirror.EditorView.domEventHandlers({
focusout: event => {
// The relatedTarget is null when no element gains focus, e.g. switching windows.
const relatedTarget = (event.relatedTarget as Node | null);
if (relatedTarget && !relatedTarget.isSelfOrDescendant(editor)) {
this.editing?.commit();
}
},
}),
],
}));
this.editing = {commit: commit.bind(this), cancel: dispose.bind(this), editor, resize: resize.bind(this)};
resize.call(this);
this.htmlEditElement.appendChild(editor);
editor.editor.focus();
this.treeOutline?.setMultilineEditing(this.editing);
function resize(this: ElementsTreeElement): void {
if (this.treeOutline && this.htmlEditElement) {
this.htmlEditElement.style.width = this.treeOutline.visibleWidth() - this.computeLeftIndent() - 30 + 'px';
}
}
function commit(this: ElementsTreeElement): void {
if (this.editing?.editor) {
commitCallback(initialValue, this.editing.editor.state.doc.toString());
}
dispose.call(this);
}
function dispose(this: ElementsTreeElement): void {
if (!this.editing?.editor) {
return;
}
this.editing = null;
// Remove editor.
if (this.htmlEditElement) {
this.listItemElement.removeChild(this.htmlEditElement);
}
this.htmlEditElement = undefined;
// Unhide children item.
if (this.childrenListElement) {
this.childrenListElement.style.removeProperty('display');
}
// Unhide header items.
let child: (ChildNode|null) = this.listItemElement.firstChild;
while (child) {
if (child instanceof HTMLElement) {
child.style.removeProperty('display');
}
child = child.nextSibling;
}
if (this.treeOutline) {
this.treeOutline.setMultilineEditing(null);
this.treeOutline.focus();
}
disposeCallback();
}
}
private attributeEditingCommitted(
element: Element,
newText: string,
oldText: string|null,
attributeName: string|null,
moveDirection: string,
): void {
this.editing = null;
const treeOutline = this.treeOutline;
function moveToNextAttributeIfNeeded(this: ElementsTreeElement, error?: string|null): void {
if (error) {
this.editingCancelled(element, attributeName);
}
if (!moveDirection) {
return;
}
if (treeOutline) {
treeOutline.runPendingUpdates();
treeOutline.focus();
}
// Search for the attribute's position, and then decide where to move to.
const attributes = this.nodeInternal.attributes();
for (let i = 0; i < attributes.length; ++i) {
if (attributes[i].name !== attributeName) {
continue;
}
if (moveDirection === 'backward') {
if (i === 0) {
this.startEditingTagName();
} else {
this.triggerEditAttribute(attributes[i - 1].name);
}
} else if (i === attributes.length - 1) {
this.addNewAttribute();
} else {
this.triggerEditAttribute(attributes[i + 1].name);
}
return;
}
// Moving From the "New Attribute" position.
if (moveDirection === 'backward') {
if (newText === ' ') {
// Moving from "New Attribute" that was not edited
if (attributes.length > 0) {
this.triggerEditAttribute(attributes[attributes.length - 1].name);
}
// Moving from "New Attribute" that holds new value
} else if (attributes.length > 1) {
this.triggerEditAttribute(attributes[attributes.length - 2].name);
}
} else if (moveDirection === 'forward') {
if (!Platform.StringUtilities.isWhitespace(newText)) {
this.addNewAttribute();
} else {
this.startEditingTagName();
}
}
}
if (attributeName !== null && (attributeName.trim() || newText.trim()) && oldText !== newText) {
this.nodeInternal.setAttribute(attributeName, newText, moveToNextAttributeIfNeeded.bind(this));
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.DOM_ELEMENT_OR_ATTRIBUTE_EDITED);
return;
}
this.updateTitle();
moveToNextAttributeIfNeeded.call(this);
}
private tagNameEditingCommitted(
element: Element,
newText: string,
oldText: string|null,
tagName: string|null,
moveDirection: string,
): void {
this.editing = null;
const self = this;
function cancel(): void {
const closingTagElement = self.distinctClosingTagElement();
if (closingTagElement) {
closingTagElement.textContent = '</' + tagName + '>';
}
self.editingCancelled(element, tagName);
moveToNextAttributeIfNeeded.call(self);
}
function moveToNextAttributeIfNeeded(this: ElementsTreeElement): void {
if (moveDirection !== 'forward') {
this.addNewAttribute();
return;
}
const attributes = this.nodeInternal.attributes();
if (attributes.length > 0) {
this.triggerEditAttribute(attributes[0].name);
} else {
this.addNewAttribute();
}
}
newText = newText.trim();
if (newText === oldText) {
cancel();
return;
}
const treeOutline = this.treeOutline;
const wasExpanded = this.expanded;
this.nodeInternal.setNodeName(newText, (error, newNode) => {
if (error || !newNode) {
cancel();
return;
}
if (!treeOutline) {
return;
}
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.DOM_ELEMENT_OR_ATTRIBUTE_EDITED);
const newTreeItem = treeOutline.selectNodeAfterEdit(wasExpanded, error, newNode);
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// @ts-expect-error
moveToNextAttributeIfNeeded.call(newTreeItem);
});
}
private textNodeEditingCommitted(textNode: SDK.DOMModel.DOMNode, _element: Element, newText: string): void {
this.editing = null;
function callback(this: ElementsTreeElement): void {
this.updateTitle();
}
textNode.setNodeValue(newText, callback.bind(this));
}
private editingCancelled(_element: Element, _tagName: string|null): void {
this.editing = null;
// Need to restore attributes structure.
this.updateTitle();
}
private distinctClosingTagElement(): Element|null {
// FIXME: Improve the Tree Element / Outline Abstraction to prevent crawling the DOM
// For an expanded element, it will be the last element with class "close"
// in the child element list.
if (this.expanded) {
const closers = this.childrenListElement.querySelectorAll('.close');
return closers[closers.length - 1];
}
// Remaining cases are single line non-expanded elements with a closing
// tag, or HTML elements without a closing tag (such as <br>). Return
// null in the case where there isn't a closing tag.
const tags = this.listItemElement.getElementsByClassName('webkit-html-tag');
return tags.length === 1 ? null : tags[tags.length - 1];
}
updateTitle(updateRecord?: Elements.ElementUpdateRecord.ElementUpdateRecord|null): void {
// If we are editing, return early to prevent canceling the edit.
// After editing is committed updateTitle will be called.
if (this.editing) {
return;
}
this.performUpdate(/* clearNode= */ true);
this.#updateRecord = updateRecord ?? null;
if (this.nodeInternal.nodeType() === Node.DOCUMENT_FRAGMENT_NODE && this.nodeInternal.isInShadowTree() &&
this.nodeInternal.shadowRootType()) {
this.childrenListElement.classList.add('shadow-root');
let depth = 4;
for (let node: (SDK.DOMModel.DOMNode|null) = (this.nodeInternal as SDK.DOMModel.DOMNode | null); depth && node;
node = node.parentNode) {
if (node.nodeType() === Node.DOCUMENT_FRAGMENT_NODE) {
depth--;
}
}
if (!depth) {
this.childrenListElement.classList.add('shadow-root-deep');
} else {
this.childrenListElement.classList.add('shadow-root-depth-' + depth);
}
}
this.performUpdate();
if (this.#contentElement) {
// fixme: we probably do not need a title element in the new tree outline.
this.title = this.#contentElement;
}
this.updateDecorations();
// If there is an issue with this node, make sure to update it.
for (const issue of this.#elementIssues.values()) {
this.#applyIssueStyleAndTooltip(issue);
}
this.#highlightSearchResults();
}
private computeLeftIndent(): number {
let treeElement: (UI.TreeOutline.TreeElement|null) = this.parent;
let depth = 0;
while (treeElement !== null) {
depth++;
treeElement = treeElement.parent;
}
/** Keep it in sync with elementsTreeOutline.css **/
return 12 * (depth - 2) + (this.isExpandable() && this.isCollapsible() ? 1 : 12);
}
updateDecorations(): void {
if (this.isClosingTag()) {
return;
}
if (this.nodeInternal.nodeType() !== Node.ELEMENT_NODE) {
return;
}
void this.decorationsThrottler.schedule(this.#updateDecorations.bind(this));
}
#updateDecorations(): Promise<void> {
if (!this.treeOutline) {
return Promise.resolve();
}
const node = this.nodeInternal;
if (!this.treeOutline.decoratorExtensions) {
this.treeOutline.decoratorExtensions = getRegisteredDecorators();
}
const markerToExtension = new Map<string, MarkerDecoratorRegistration>();
for (const decoratorExtension of this.treeOutline.decoratorExtensions) {
markerToExtension.set(decoratorExtension.marker, decoratorExtension);
}
const promises: Array<Promise<void>> = [];
const decorations: Array<{
title: string,
color: string,
}> = [];
const descendantDecorations: Array<{
title: string,
color: string,
}> = [];
node.traverseMarkers(visitor);
function visitor(n: SDK.DOMModel.DOMNode, marker: string): void {
const extension = markerToExtension.get(marker);
if (!extension) {
return;
}
promises.push(Promise.resolve(extension.decorator()).then(collectDecoration.bind(null, n)));
}
function collectDecoration(n: SDK.DOMModel.DOMNode, decorator: MarkerDecorator): void {
const decoration = decorator.decorate(n);
if (!decoration) {
return;
}
(n === node ? decorations : descendantDecorations).push(decoration);
}
return Promise.all(promises).then(updateDecorationsUI.bind(this));
function updateDecorationsUI(this: ElementsTreeElement): void {
this.#decorations = decorations;
this.#descendantDecorations = descendantDecorations;
if (!decorations.length && !descendantDecorations.length) {
this.#decorationsTooltip = '';
this.performUpdate();
return;
}
const tooltip: string[] = [];
for (const decoration of decorations) {
tooltip.push(decoration.title);
}
if (!this.expanded && descendantDecorations.length) {
tooltip.push(i18nString(UIStrings.children));
for (const decoration of descendantDecorations) {
tooltip.push(decoration.title);
}
}
this.#decorationsTooltip = tooltip.join('\n');
this.performUpdate();
}
}
async remove(): Promise<void> {
if (this.treeOutline?.isToggledToHidden(this.nodeInternal)) {
// Unhide the node before removing. This avoids inconsistent state if the node is restored via undo.
await this.treeOutline.toggleHideElement(this.nodeInternal);
}
if (this.nodeInternal.pseudoType()) {
return;
}
const parentElement = this.parent;
if (!parentElement) {
return;
}
if (!this.nodeInternal.parentNode || this.nodeInternal.parentNode.nodeType() === Node.DOCUMENT_NODE) {
return;
}
void this.nodeInternal.removeNode();
}
toggleEditAsHTML(callback?: ((arg0: boolean) => void), startEditing?: boolean): void {
if (this.editing && this.htmlEditElement) {
this.editing.commit();
return;
}
if (startEditing === false) {
return;
}
function selectNode(error: string|null): void {
if (callback) {
callback(!error);
}
}
function commitChange(initialValue: string, value: string): void {
if (initialValue !== value) {
node.setOuterHTML(value, selectNode);
}
}
function disposeCallback(): void {
if (callback) {
callback(false);
}
}
const node = this.nodeInternal;
void node.getOuterHTML().then(this.startEditingAsHTML.bind(this, commitChange, disposeCallback));
}
private copyCSSPath(): void {
Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(cssPath(this.nodeInternal, true));
}
private copyJSPath(): void {
Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(jsPath(this.nodeInternal, true));
}
private copyXPath(): void {
Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(xPath(this.nodeInternal, true));
}
private copyFullXPath(): void {
Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(xPath(this.nodeInternal, false));
}
async copyStyles(): Promise<void> {
const node = this.nodeInternal;
const cssModel = node.domModel().cssModel();
const cascade = await cssModel.cachedMatchedCascadeForNode(node);
if (!cascade) {
return;
}
const indent = Common.Settings.Settings.instance().moduleSetting('text-editor-indent').get();
const lines: string[] = [];
for (const style of cascade.nodeStyles().reverse()) {
for (const property of style.leadingProperties()) {
if (!property.parsedOk || property.disabled || !property.activeInStyle() || property.implicit) {
continue;
}
if (cascade.isInherited(style) && !SDK.CSSMetadata.cssMetadata().isPropertyInherited(property.name)) {
continue;
}
if (style.parentRule?.isUserAgent()) {
continue;
}
if (cascade.propertyState(property) !== SDK.CSSMatchedStyles.PropertyState.ACTIVE) {
continue;
}
lines.push(`${indent}${property.name}: ${property.value};`);
}
}
Host.InspectorFrontendHost.InspectorFrontendHostInstance.copyText(lines.join('\n'));
}
#highlightSearchResults(): void {
this.hideSearchHighlights();
if (!this.searchQuery) {
return;
}
const text = this.listItemElement.textContent || '';
const regexObject = Platform.StringUtilities.createPlainTextSearchRegex(this.searchQuery, 'gi');
const matchRanges = [];
let match = regexObject.exec(text);
while (match) {
matchRanges.push(new TextUtils.TextRange.SourceRange(match.index, match[0].length));
match = regexObject.exec(text);
}
// Fall back for XPath, etc. matches.
if (!matchRanges.length) {
matchRanges.push(new TextUtils.TextRange.SourceRange(0, text.length));
}
this.#highlights = Highlighting.HighlightManager.HighlightManager.instance().highlightOrderedTextRanges(
this.listItemElement, matchRanges);
}
private editAsHTML(): void {
const promise = Common.Revealer.reveal(this.node());
void promise.then(() => {
const action = UI.ActionRegistry.ActionRegistry.instance().getAction('elements.edit-as-html');
return action.execute();
});
}
updateAdorners(): void {
// TODO: remove adornersThrottler in favour of throttled updated (requestUpdate/performUpdate).
void this.#adornersThrottler.schedule(this.#updateAdorners.bind(this));
}
async #updateAdorners(): Promise<void> {
if (this.isClosingTag()) {
return;
}
const node = this.node();
const nodeId = node.id;
if (node.nodeType() !== Node.COMMENT_NODE && node.nodeType() !== Node.DOCUMENT_FRAGMENT_NODE &&
node.nodeType() !== Node.TEXT_NODE && nodeId !== undefined) {
this.#layout = await node.domModel().cssModel().getLayoutPropertiesFromComputedStyle(nodeId);
} else {
this.#layout = null;
}
this.performUpdate();
}
async #onPopoverAdornerClick(event: Event): Promise<void> {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
await node.domModel().agent.invoke_forceShowPopover({nodeId, enable: !this.#popoverAdornerActive});
this.#popoverAdornerActive = !this.#popoverAdornerActive;
if (this.#popoverAdornerActive) {
Badges.UserBadges.instance().recordAction(Badges.BadgeAction.MODERN_DOM_BADGE_CLICKED);
}
this.performUpdate();
}
#onStartingStyleAdornerClick(event: Event): void {
event.stopPropagation();
const node = this.node();
const nodeId = node.id;
if (!nodeId) {
return;
}
const model = node.domModel().cssModel();
if (this.#startingStyleAdornerActive) {
model.forceStartingStyle(node, false);
} else {
model.forceStartingStyle(node, true);
}
this.#startingStyleAdornerActive = !this.#startingStyleAdornerActive;
this.performUpdate();
}
}
export const InitialChildrenLimit = 500;
/**
* A union of HTML4 and HTML5-Draft elements that explicitly
* or implicitly (for HTML5) forbid the closing tag.
**/
export const ForbiddenClosingTagElements = new Set<string>([
'area', 'base', 'basefont', 'br', 'canvas', 'col', 'command', 'embed', 'frame', 'hr',
'img', 'input', 'keygen', 'link', 'menuitem', 'meta', 'param', 'source', 'track', 'wbr',
]);
/** These tags we do not allow editing their tag name. **/
export const EditTagBlocklist = new Set<string>(['html', 'head', 'body']);
export function convertUnicodeCharsToHTMLEntities(text: string): {
text: string,
entityRanges: TextUtils.TextRange.SourceRange[],
} {
let result = '';
let lastIndexAfterEntity = 0;
const entityRanges = [];
const charToEntity = MappedCharToEntity;
for (let i = 0, size = text.length; i < size; ++i) {
const char = text.charAt(i);
if (charToEntity.has(char)) {
result += text.substring(lastIndexAfterEntity, i);
const entityValue = '&' + charToEntity.get(char) + ';';
entityRanges.push(new TextUtils.TextRange.SourceRange(result.length, entityValue.length));
result += entityValue;
lastIndexAfterEntity = i + 1;
}
}
if (result) {
result += text.substring(lastIndexAfterEntity);
}
return {text: result || text, entityRanges};
}
export interface EditorHandles {
commit: () => void;
cancel: () => void;
editor?: TextEditor.TextEditor.TextEditor;
resize: () => void;
}
/**
* As a privacy measure we are logging elements tree outline as a flat list where every tree item is a
* child of a tree outline.
**/
function loggingParentProvider(e: Element): Element|undefined {
const treeElement = UI.TreeOutline.TreeElement.getTreeElementBylistItemNode(e);
return treeElement?.treeOutline?.contentElement;
}
VisualLogging.registerParentProvider('elementsTreeOutline', loggingParentProvider);