UNPKG

chrome-devtools-frontend

Version:
177 lines (154 loc) • 5.65 kB
// Copyright 2018 The Chromium Authors. All rights reserved. // Use of this source code is governed by a BSD-style license that can be // found in the LICENSE file. import * as Common from '../common/common.js'; import * as i18n from '../i18n/i18n.js'; import * as SDK from '../sdk/sdk.js'; import * as UI from '../ui/ui.js'; export const UIStrings = { /** *@description Text in DOMLinkifier of the Elements panel */ node: '<node>', }; const str_ = i18n.i18n.registerUIStrings('elements/DOMLinkifier.js', UIStrings); const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_); /** * @param {!SDK.DOMModel.DOMNode} node * @param {!HTMLElement} parentElement * @param {string=} tooltipContent */ export const decorateNodeLabel = function(node, parentElement, tooltipContent) { const originalNode = node; const isPseudo = node.nodeType() === Node.ELEMENT_NODE && node.pseudoType(); if (isPseudo && node.parentNode) { node = node.parentNode; } let title = node.nodeNameInCorrectCase(); const nameElement = parentElement.createChild('span', 'node-label-name'); nameElement.textContent = title; const idAttribute = node.getAttribute('id'); if (idAttribute) { const idElement = parentElement.createChild('span', 'node-label-id'); const part = '#' + idAttribute; title += part; UI.UIUtils.createTextChild(idElement, part); // Mark the name as extra, since the ID is more important. nameElement.classList.add('extra'); } const classAttribute = node.getAttribute('class'); if (classAttribute) { const classes = classAttribute.split(/\s+/); if (classes.length) { const foundClasses = new Set(); const classesElement = parentElement.createChild('span', 'extra node-label-class'); for (let i = 0; i < classes.length; ++i) { const className = classes[i]; if (className && !foundClasses.has(className)) { const part = '.' + className; title += part; UI.UIUtils.createTextChild(classesElement, part); foundClasses.add(className); } } } } if (isPseudo) { const pseudoElement = parentElement.createChild('span', 'extra node-label-pseudo'); const pseudoText = '::' + originalNode.pseudoType(); UI.UIUtils.createTextChild(pseudoElement, pseudoText); title += pseudoText; } UI.Tooltip.Tooltip.install(parentElement, tooltipContent || title); }; /** * @param {?SDK.DOMModel.DOMNode} node * @param {!Common.Linkifier.Options=} options * @return {!Node} */ export const linkifyNodeReference = function(node, options = { tooltip: undefined, preventKeyboardFocus: undefined, }) { if (!node) { return document.createTextNode(i18nString(UIStrings.node)); } const root = document.createElement('span'); root.classList.add('monospace'); const shadowRoot = UI.Utils.createShadowRootWithCoreStyles( root, {cssFile: 'elements/domLinkifier.css', enableLegacyPatching: false, delegatesFocus: undefined}); const link = /** @type {!HTMLDivElement} */ (shadowRoot.createChild('div', 'node-link')); decorateNodeLabel(node, link, options.tooltip); link.addEventListener('click', () => Common.Revealer.reveal(node, false) && false, false); link.addEventListener('mouseover', node.highlight.bind(node, undefined), false); link.addEventListener('mouseleave', () => SDK.OverlayModel.OverlayModel.hideDOMNodeHighlight(), false); if (!options.preventKeyboardFocus) { link.addEventListener('keydown', event => event.key === 'Enter' && Common.Revealer.reveal(node, false) && false); link.tabIndex = 0; UI.ARIAUtils.markAsLink(link); } return root; }; /** * @param {!SDK.DOMModel.DeferredDOMNode} deferredNode * @param {!Common.Linkifier.Options=} options * @return {!Node} */ export const linkifyDeferredNodeReference = function(deferredNode, options = { tooltip: undefined, preventKeyboardFocus: undefined, }) { const root = document.createElement('div'); const shadowRoot = UI.Utils.createShadowRootWithCoreStyles( root, {cssFile: 'elements/domLinkifier.css', enableLegacyPatching: false, delegatesFocus: undefined}); const link = /** @type {!HTMLDivElement} */ (shadowRoot.createChild('div', 'node-link')); link.createChild('slot'); link.addEventListener('click', deferredNode.resolve.bind(deferredNode, onDeferredNodeResolved), false); link.addEventListener('mousedown', e => e.consume(), false); if (!options.preventKeyboardFocus) { link.addEventListener('keydown', event => event.key === 'Enter' && deferredNode.resolve(onDeferredNodeResolved)); link.tabIndex = 0; UI.ARIAUtils.markAsLink(link); } /** * @param {?SDK.DOMModel.DOMNode} node */ function onDeferredNodeResolved(node) { Common.Revealer.reveal(node); } return root; }; /** * @type {!Linkifier} */ let linkifierInstance; /** * @implements {Common.Linkifier.Linkifier} */ export class Linkifier { /** * @param {{forceNew: ?boolean}} opts */ static instance(opts = {forceNew: null}) { const {forceNew} = opts; if (!linkifierInstance || forceNew) { linkifierInstance = new Linkifier(); } return linkifierInstance; } /** * @override * @param {!Object} object * @param {!Common.Linkifier.Options=} options * @return {!Node} */ linkify(object, options) { if (object instanceof SDK.DOMModel.DOMNode) { return linkifyNodeReference(object, options); } if (object instanceof SDK.DOMModel.DeferredDOMNode) { return linkifyDeferredNodeReference(object, options); } throw new Error('Can\'t linkify non-node'); } }