// 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 * 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 A context menu item in the Elements Tree Element of the Elements panel */ editData: 'Edit data', /** * @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: '', /** * @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 { 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`${pseudoElementName}\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}\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}\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``; } 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``; } 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`""`; } case Node.COMMENT_NODE: { return html`<!--${node.nodeValue()}-->`; } case Node.DOCUMENT_TYPE_NODE: { let doctype = ''; return html`${doctype}`; } case Node.CDATA_SECTION_NODE: { return html`<![CDATA[${node.nodeValue()}]]>`; } case Node.DOCUMENT_NODE: { const text = (node as SDK.DOMModel.DOMDocument).documentURL; return html`#document (${Components.Linkifier.Linkifier.renderLinkifiedUrl(text, { text, preventClick: true, showColumnNumber: false, inlineFrameIndex: 0, })})`; } case Node.DOCUMENT_FRAGMENT_NODE: { return html`${ Platform.StringUtilities.collapseWhitespace(node.nodeNameInCorrectCase())}`; } case Node.PROCESSING_INSTRUCTION_NODE: { const nodeValue = node.nodeValue(); const maybeSpace = nodeValue ? ' ' : ''; return html`<?${ node.nodeName()}${maybeSpace}${nodeValue}?>`; } 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` { if (el) { setValueWithEntities(el, value); } })}}>`; } 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` { if (el) { ImagePreviewPopover.setImageUrl(el, rewrittenHref); } })}>${Platform.StringUtilities.trimMiddle(value, 150)}`; } 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 => ref((el): void => { if (!el) { return; } void (async(): Promise => { 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 = 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 = 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`${name}${ hasText ? html`=\u200B" ${valueType === ValueType.SRC ? renderLinkifiedValue(value, node) : nothing} ${valueType === ValueType.SRCSET ? renderLinkifiedSrcset(Common.Srcset.parseSrcset(value), node) : nothing} "` : nothing}`; // 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`<${tagString}${ attributes.map(attr => html` ${renderAttribute(attr, updateRecord, false, node)}`)}>\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`
{ output.contentElement = el as HTMLElement; })}> ${input.node ? html`${renderTitle( input.node, input.isClosingTag, input.expanded, input.isExpandable, input.isXMLMimeType, input.updateRecord, input.onHighlightSearchResults, input.onExpand, )}` : nothing} ${input.isHovered || input.isSelected ? html`
` : nothing}
${input.decorations.length || input.descendantDecorations.length ? html`
${input.decorations.map(d => html`
`)} ${input.descendantDecorations.map(d => html`
`)}
` : nothing}
${hasAdorners ? html`
${input.showAdAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.AD} ` : nothing} ${input.showViewSourceAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.VIEW_SOURCE} ` : nothing} ${input.showContainerAdorner ? html` ${input.containerType} `: nothing} ${input.showFlexAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.FLEX} `: nothing} ${input.showGridAdorner ? html` ${input.isSubgrid ? ElementsComponents.AdornerManager.RegisteredAdorners.SUBGRID : ElementsComponents.AdornerManager.RegisteredAdorners.GRID} `: nothing} ${input.showGridLanesAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.GRID_LANES} `: nothing} ${input.showMediaAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.MEDIA} `: nothing} ${input.showPopoverAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.POPOVER} `: nothing} ${input.showTopLayerAdorner ? html` ${`top-layer (${input.topLayerIndex})`} `: nothing} ${input.showStartingStyleAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.STARTING_STYLE} ` : nothing} ${input.showScrollAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL} ` : nothing} ${input.showSlotAdorner ? html` e.stopPropagation()} ${adornerRef()}> ${ElementsComponents.AdornerManager.RegisteredAdorners.SLOT} `: nothing} ${input.showScrollSnapAdorner ? html` ${ElementsComponents.AdornerManager.RegisteredAdorners.SCROLL_SNAP} ` : nothing}
`: nothing} ${input.isSelected ? html` ` : nothing} ${input.showAiButton ? html` e.stopPropagation()}> ` : nothing}
`, 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(); #nodeElementToIssue = new Map(); #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', resize: true, 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.isAdRelatedNode(), 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 { 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 { 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.AD_RELATED_STATE_UPDATED, this.#onAdRelatedStateUpdated, 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.AD_RELATED_STATE_UPDATED, this.#onAdRelatedStateUpdated, 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(); } #onAdRelatedStateUpdated(): 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 { if (this.treeOutline) { return await this.treeOutline.populateTreeElement(this); } } override async expandRecursively(): Promise { 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 && this.nodeInternal.nodeType() !== Node.PROCESSING_INSTRUCTION_NODE) { return false; } const textNode = eventTarget.enclosingNodeOrSelfWithClass('webkit-html-text-node') ?? eventTarget.enclosingNodeOrSelfWithClass('webkit-html-processing-instruction-value'); 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 { // 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 { 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 { // 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'}); } } async populateProcessingElementContextMenu(contextMenu: UI.ContextMenu.ContextMenu): Promise { const treeOutline = this.treeOutline; if (!treeOutline) { return; } contextMenu.editSection().appendItem( i18nString(UIStrings.editData), this.startEditingProcessingInstructionValue.bind(this), {jslogContext: 'elements.edit-data'}); contextMenu.editSection().appendItem( i18nString(UIStrings.duplicateElement), treeOutline.duplicateNode.bind(treeOutline, this.nodeInternal), { disabled: (this.nodeInternal.isInShadowTree()), jslogContext: 'elements.duplicate-element', }); contextMenu.editSection().appendItem( i18nString(UIStrings.deleteElement), this.remove.bind(this), {jslogContext: 'delete-element'}); } 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); } } if (this.nodeInternal.nodeType() === Node.PROCESSING_INSTRUCTION_NODE) { return this.startEditingProcessingInstructionValue(); } return; } private startEditingProcessingInstructionValue(): boolean|undefined { const processingInstructionValue = this.listItemElement.getElementsByClassName('webkit-html-processing-instruction-value')[0]; if (processingInstructionValue) { return this.startEditingTextNode(processingInstructionValue); } 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('"', '"') : 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 = ''; } } 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(editingCommitted.bind(this), editingCancelled.bind(this), tagName); this.updateEditorHandles(tagNameElement, config); const componentSelection = this.listItemElement.getComponentSelection(); componentSelection?.selectAllChildren(tagNameElement); return true; } private updateEditorHandles(element: Element, config: UI.InplaceEditor.Config): 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 { 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 = ''; } self.editingCancelled(element, tagName); moveToNextAttributeIfNeeded.call(self); } function moveToNextAttributeIfNeeded(this: ElementsTreeElement): void { if (this.nodeInternal.nodeType() === Node.PROCESSING_INSTRUCTION_NODE) { this.startEditingProcessingInstructionValue(); return; } 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
). 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 { if (!this.treeOutline) { return Promise.resolve(); } const node = this.nodeInternal; if (!this.treeOutline.decoratorExtensions) { this.treeOutline.decoratorExtensions = getRegisteredDecorators(); } const markerToExtension = new Map(); for (const decoratorExtension of this.treeOutline.decoratorExtensions) { markerToExtension.set(decoratorExtension.marker, decoratorExtension); } const promises: Array> = []; 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 { 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 { 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 { 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 { 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([ '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(['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);