UNPKG

chrome-devtools-frontend

Version:
280 lines (258 loc) • 9.54 kB
// Copyright (c) 2020 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 Root from '../root/root.js'; import * as SDK from '../sdk/sdk.js'; import * as UI from '../ui/ui.js'; import {ElementsPanel} from './ElementsPanel.js'; import {LayoutElement, LayoutPane} from './LayoutPane.js'; // eslint-disable-line no-unused-vars /** * @param {!SDK.DOMModel.DOMNode} node * @return {!LayoutElement} */ const nodeToLayoutElement = node => { const className = node.getAttribute('class'); const nodeId = node.id; return { id: nodeId, color: '#000', name: node.localName(), domId: node.getAttribute('id'), domClasses: className ? className.split(/\s+/).filter(s => Boolean(s)) : undefined, enabled: false, reveal: () => { ElementsPanel.instance().revealAndSelectNode(node, true, true); node.scrollIntoView(); }, highlight: () => { node.highlight(); }, hideHighlight: () => { SDK.OverlayModel.OverlayModel.hideDOMNodeHighlight(); }, toggle: value => { throw new Error('Not implemented'); }, setColor(value) { throw new Error('Not implemented'); }, }; }; /** * @param {!Array<!SDK.DOMModel.DOMNode>} nodes * @return {!Array<!LayoutElement>} */ const gridNodesToElements = nodes => { return nodes.map(node => { const layoutElement = nodeToLayoutElement(node); const nodeId = node.id; return { ...layoutElement, color: node.domModel().overlayModel().colorOfGridInPersistentOverlay(nodeId) || '#000', enabled: node.domModel().overlayModel().isHighlightedGridInPersistentOverlay(nodeId), toggle: value => { if (value) { node.domModel().overlayModel().highlightGridInPersistentOverlay(nodeId); } else { node.domModel().overlayModel().hideGridInPersistentOverlay(nodeId); } }, setColor(value) { this.color = value; node.domModel().overlayModel().setColorOfGridInPersistentOverlay(nodeId, value); }, }; }); }; /** @type {!LayoutSidebarPane} */ let layoutSidebarPaneInstance; /** * @param {!Array<!SDK.DOMModel.DOMNode>} nodes * @return {!Array<!LayoutElement>} */ const flexContainerNodesToElements = nodes => { return nodes.map(node => { const layoutElement = nodeToLayoutElement(node); const nodeId = node.id; return { ...layoutElement, color: node.domModel().overlayModel().colorOfFlexInPersistentOverlay(nodeId) || '#000', enabled: node.domModel().overlayModel().isHighlightedFlexContainerInPersistentOverlay(nodeId), toggle: value => { if (value) { node.domModel().overlayModel().highlightFlexContainerInPersistentOverlay(nodeId); } else { node.domModel().overlayModel().hideFlexContainerInPersistentOverlay(nodeId); } }, setColor(value) { this.color = value; node.domModel().overlayModel().setColorOfFlexInPersistentOverlay(nodeId, value); }, }; }); }; export class LayoutSidebarPane extends UI.ThrottledWidget.ThrottledWidget { constructor() { super(true /* isWebComponent */); this._layoutPane = new LayoutPane(); this.contentElement.appendChild(this._layoutPane); this._settings = ['showGridLineLabels', 'showGridTrackSizes', 'showGridAreas', 'extendGridLines']; this._uaShadowDOMSetting = Common.Settings.Settings.instance().moduleSetting('showUAShadowDOM'); this._boundOnSettingChanged = this.onSettingChanged.bind(this); /** * @type {!Array<!SDK.DOMModel.DOMModel>} */ this._domModels = []; } /** * @param {{forceNew: ?boolean}=} opts * @return {!LayoutSidebarPane} */ static instance(opts = {forceNew: null}) { const {forceNew} = opts; if (!layoutSidebarPaneInstance || forceNew) { layoutSidebarPaneInstance = new LayoutSidebarPane(); } return layoutSidebarPaneInstance; } /** * @param {!SDK.DOMModel.DOMModel} domModel */ modelAdded(domModel) { const overlayModel = domModel.overlayModel(); overlayModel.addEventListener(SDK.OverlayModel.Events.PersistentGridOverlayStateChanged, this.update, this); overlayModel.addEventListener( SDK.OverlayModel.Events.PersistentFlexContainerOverlayStateChanged, this.update, this); this._domModels.push(domModel); } /** * @param {!SDK.DOMModel.DOMModel} domModel */ modelRemoved(domModel) { const overlayModel = domModel.overlayModel(); overlayModel.removeEventListener(SDK.OverlayModel.Events.PersistentGridOverlayStateChanged, this.update, this); overlayModel.removeEventListener( SDK.OverlayModel.Events.PersistentFlexContainerOverlayStateChanged, this.update, this); this._domModels = this._domModels.filter(model => model !== domModel); } /** * @param {!Array<{ name: string, value: string }>} style */ async _fetchNodesByStyle(style) { const showUAShadowDOM = this._uaShadowDOMSetting.get(); const nodes = []; for (const domModel of this._domModels) { try { const nodeIds = await domModel.getNodesByStyle(style, true /* pierce */); for (const nodeId of nodeIds) { const node = domModel.nodeForId(nodeId); if (node !== null && (showUAShadowDOM || !node.ancestorUserAgentShadowRoot())) { nodes.push(node); } } } catch (error) { // TODO(crbug.com/1167706): Sometimes in E2E tests the layout panel is updated after a DOM node // has been removed. This causes an error that a node has not been found. // We can skip nodes that resulted in an error. console.warn(error); } } return nodes; } async _fetchGridNodes() { return await this._fetchNodesByStyle([{name: 'display', value: 'grid'}, {name: 'display', value: 'inline-grid'}]); } async _fetchFlexContainerNodes() { return await this._fetchNodesByStyle([{name: 'display', value: 'flex'}, {name: 'display', value: 'inline-flex'}]); } _mapSettings() { const settings = []; for (const settingName of this._settings) { const setting = Common.Settings.Settings.instance().moduleSetting(settingName); const settingValue = setting.get(); const settingType = setting.type(); if (!settingType) { throw new Error('A setting provided to LayoutSidebarPane does not have a setting type'); } if (settingType !== Common.Settings.SettingType.BOOLEAN && settingType !== Common.Settings.SettingType.ENUM) { throw new Error('A setting provided to LayoutSidebarPane does not have a supported setting type'); } const mappedSetting = { type: settingType, name: setting.name, title: setting.title(), }; if (typeof settingValue === 'boolean') { settings.push({ ...mappedSetting, value: settingValue, options: setting.options().map(opt => ({ ...opt, value: /** @type {boolean} */ (opt.value), })) }); } else if (typeof settingValue === 'string') { settings.push({ ...mappedSetting, value: settingValue, options: setting.options().map(opt => ({ ...opt, value: /** @type {string} */ (opt.value), })) }); } } return settings; } /** * @override * @protected * @return {!Promise<void>} */ async doUpdate() { this._layoutPane.data = { gridElements: gridNodesToElements(await this._fetchGridNodes()), flexContainerElements: Root.Runtime.experiments.isEnabled('cssFlexboxFeatures') ? flexContainerNodesToElements(await this._fetchFlexContainerNodes()) : undefined, settings: this._mapSettings(), }; } /** * @param {*} event */ onSettingChanged(event) { Common.Settings.Settings.instance().moduleSetting(event.data.setting).set(event.data.value); } /** * @override */ wasShown() { for (const setting of this._settings) { Common.Settings.Settings.instance().moduleSetting(setting).addChangeListener(this.update, this); } this._layoutPane.addEventListener('setting-changed', this._boundOnSettingChanged); for (const domModel of this._domModels) { this.modelRemoved(domModel); } this._domModels = []; SDK.SDKModel.TargetManager.instance().observeModels(SDK.DOMModel.DOMModel, this); UI.Context.Context.instance().addFlavorChangeListener(SDK.DOMModel.DOMNode, this.update, this); this._uaShadowDOMSetting.addChangeListener(this.update, this); this.update(); } /** * @override */ willHide() { for (const setting of this._settings) { Common.Settings.Settings.instance().moduleSetting(setting).removeChangeListener(this.update, this); } this._layoutPane.removeEventListener('setting-changed', this._boundOnSettingChanged); SDK.SDKModel.TargetManager.instance().unobserveModels(SDK.DOMModel.DOMModel, this); UI.Context.Context.instance().removeFlavorChangeListener(SDK.DOMModel.DOMNode, this.update, this); this._uaShadowDOMSetting.removeChangeListener(this.update, this); } }