META-INF.dirigible.dev-tools.layer_viewer.LayerDetailsView.js Maven / Gradle / Ivy
/*
* Copyright (C) 2013 Google Inc. All rights reserved.
*
* Redistribution and use in source and binary forms, with or without
* modification, are permitted provided that the following conditions are
* met:
*
* * Redistributions of source code must retain the above copyright
* notice, this list of conditions and the following disclaimer.
* * 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.
* * Neither the name of Google Inc. 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 THE COPYRIGHT HOLDERS AND 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 THE COPYRIGHT
* OWNER OR 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 * as Common from '../common/common.js';
import * as SDK from '../sdk/sdk.js';
import * as UI from '../ui/ui.js';
import {LayerView, LayerViewHost, ScrollRectSelection, Selection, SnapshotSelection, Type,} from './LayerViewHost.js'; // eslint-disable-line no-unused-vars
/**
* @implements {LayerView}
* @unrestricted
*/
export class LayerDetailsView extends UI.Widget.Widget {
/**
* @param {!LayerViewHost} layerViewHost
*/
constructor(layerViewHost) {
super(true);
this.registerRequiredCSS('layer_viewer/layerDetailsView.css');
this._layerViewHost = layerViewHost;
this._layerViewHost.registerView(this);
this._emptyWidget = new UI.EmptyWidget.EmptyWidget(Common.UIString.UIString('Select a layer to see its details'));
this._layerSnapshotMap = this._layerViewHost.getLayerSnapshotMap();
this._buildContent();
}
/**
* @param {?Selection} selection
* @override
*/
hoverObject(selection) {
}
/**
* @param {?Selection} selection
* @override
*/
selectObject(selection) {
this._selection = selection;
if (this.isShowing()) {
this.update();
}
}
/**
* @param {?SDK.LayerTreeBase.LayerTreeBase} layerTree
* @override
*/
setLayerTree(layerTree) {
}
/**
* @override
*/
wasShown() {
super.wasShown();
this.update();
}
/**
* @param {number} index
* @param {!Event} event
*/
_onScrollRectClicked(index, event) {
if (event.which !== 1) {
return;
}
this._layerViewHost.selectObject(new ScrollRectSelection(this._selection.layer(), index));
}
_invokeProfilerLink() {
const snapshotSelection = this._selection.type() === Type.Snapshot ?
this._selection :
this._layerSnapshotMap.get(this._selection.layer());
if (snapshotSelection) {
this.dispatchEventToListeners(Events.PaintProfilerRequested, snapshotSelection);
}
}
/**
* @param {!Protocol.LayerTree.ScrollRect} scrollRect
* @param {number} index
*/
_createScrollRectElement(scrollRect, index) {
if (index) {
this._scrollRectsCell.createTextChild(', ');
}
const element = this._scrollRectsCell.createChild('span', 'scroll-rect');
if (this._selection.scrollRectIndex === index) {
element.classList.add('active');
}
element.textContent = Common.UIString.UIString(
'%s %d × %d (at %d, %d)', slowScrollRectNames.get(scrollRect.type), scrollRect.rect.width,
scrollRect.rect.height, scrollRect.rect.x, scrollRect.rect.y);
element.addEventListener('click', this._onScrollRectClicked.bind(this, index), false);
}
/**
* @param {string} title
* @param {?SDK.LayerTreeBase.Layer} layer
* @return {string}
*/
_formatStickyAncestorLayer(title, layer) {
if (!layer) {
return '';
}
const node = layer.nodeForSelfOrAncestor();
const name = node ? node.simpleSelector() : Common.UIString.UIString('');
return Common.UIString.UIString('%s: %s (%s)', title, name, layer.id());
}
/**
* @param {string} title
* @param {?SDK.LayerTreeBase.Layer} layer
*/
_createStickyAncestorChild(title, layer) {
if (!layer) {
return;
}
this._stickyPositionConstraintCell.createTextChild(', ');
const child = this._stickyPositionConstraintCell.createChild('span');
child.textContent = this._formatStickyAncestorLayer(title, layer);
}
/**
* @param {?SDK.LayerTreeBase.StickyPositionConstraint} constraint
*/
_populateStickyPositionConstraintCell(constraint) {
this._stickyPositionConstraintCell.removeChildren();
if (!constraint) {
return;
}
const stickyBoxRect = constraint.stickyBoxRect();
const stickyBoxRectElement = this._stickyPositionConstraintCell.createChild('span');
stickyBoxRectElement.textContent = Common.UIString.UIString(
'Sticky Box %d × %d (at %d, %d)', stickyBoxRect.width, stickyBoxRect.height, stickyBoxRect.x, stickyBoxRect.y);
this._stickyPositionConstraintCell.createTextChild(', ');
const containingBlockRect = constraint.containingBlockRect();
const containingBlockRectElement = this._stickyPositionConstraintCell.createChild('span');
containingBlockRectElement.textContent = Common.UIString.UIString(
'Containing Block %d × %d (at %d, %d)', containingBlockRect.width, containingBlockRect.height,
containingBlockRect.x, containingBlockRect.y);
this._createStickyAncestorChild(
Common.UIString.UIString('Nearest Layer Shifting Sticky Box'), constraint.nearestLayerShiftingStickyBox());
this._createStickyAncestorChild(
Common.UIString.UIString('Nearest Layer Shifting Containing Block'),
constraint.nearestLayerShiftingContainingBlock());
}
update() {
const layer = this._selection && this._selection.layer();
if (!layer) {
this._tableElement.remove();
this._paintProfilerLink.remove();
this._emptyWidget.show(this.contentElement);
return;
}
this._emptyWidget.detach();
this.contentElement.appendChild(this._tableElement);
this.contentElement.appendChild(this._paintProfilerLink);
this._sizeCell.textContent =
Common.UIString.UIString('%d × %d (at %d,%d)', layer.width(), layer.height(), layer.offsetX(), layer.offsetY());
this._paintCountCell.parentElement.classList.toggle('hidden', !layer.paintCount());
this._paintCountCell.textContent = layer.paintCount();
this._memoryEstimateCell.textContent = Number.bytesToString(layer.gpuMemoryUsage());
layer.requestCompositingReasonIds().then(this._updateCompositingReasons.bind(this));
this._scrollRectsCell.removeChildren();
layer.scrollRects().forEach(this._createScrollRectElement.bind(this));
this._populateStickyPositionConstraintCell(layer.stickyPositionConstraint());
const snapshot = this._selection.type() === Type.Snapshot ?
/** @type {!SnapshotSelection} */ (this._selection).snapshot() :
null;
this._paintProfilerLink.classList.toggle('hidden', !(this._layerSnapshotMap.has(layer) || snapshot));
}
_buildContent() {
this._tableElement = this.contentElement.createChild('table');
this._tbodyElement = this._tableElement.createChild('tbody');
this._sizeCell = this._createRow(Common.UIString.UIString('Size'));
this._compositingReasonsCell = this._createRow(Common.UIString.UIString('Compositing Reasons'));
this._memoryEstimateCell = this._createRow(Common.UIString.UIString('Memory estimate'));
this._paintCountCell = this._createRow(Common.UIString.UIString('Paint count'));
this._scrollRectsCell = this._createRow(Common.UIString.UIString('Slow scroll regions'));
this._stickyPositionConstraintCell = this._createRow(Common.UIString.UIString('Sticky position constraint'));
this._paintProfilerLink = this.contentElement.createChild('span', 'hidden devtools-link link-margin');
UI.ARIAUtils.markAsLink(this._paintProfilerLink);
this._paintProfilerLink.textContent = ls`Paint Profiler`;
this._paintProfilerLink.tabIndex = 0;
this._paintProfilerLink.addEventListener('click', e => {
e.consume(true);
this._invokeProfilerLink();
});
this._paintProfilerLink.addEventListener('keydown', event => {
if (isEnterKey(event)) {
event.consume();
this._invokeProfilerLink();
}
});
}
/**
* @param {string} title
*/
_createRow(title) {
const tr = this._tbodyElement.createChild('tr');
const titleCell = tr.createChild('td');
titleCell.textContent = title;
return tr.createChild('td');
}
/**
* @param {!Array.} compositingReasonIds
*/
_updateCompositingReasons(compositingReasonIds) {
if (!compositingReasonIds || !compositingReasonIds.length) {
this._compositingReasonsCell.textContent = 'n/a';
return;
}
this._compositingReasonsCell.removeChildren();
const list = this._compositingReasonsCell.createChild('ul');
const compositingReasons = LayerDetailsView.getCompositingReasons(compositingReasonIds);
for (const compositingReason of compositingReasons) {
list.createChild('li').textContent = compositingReason;
}
}
/**
* @param {!Array.} compositingReasonIds
*/
static getCompositingReasons(compositingReasonIds) {
const compositingReasons = [];
for (const compositingReasonId of compositingReasonIds) {
const compositingReason = compositingReasonIdToReason.get(compositingReasonId);
if (compositingReason) {
compositingReasons.push(compositingReason);
} else {
console.error(`Compositing reason id '${compositingReasonId}' is not recognized.`);
}
}
return compositingReasons;
}
}
// The compositing reason IDs are defined in third_party/blink/renderer/platform/graphics/compositing_reasons.cc
const compositingReasonIdToReason = new Map([
['transform3D', ls`Has a 3d transform.`],
['video', ls`Is an accelerated video.`],
[
'canvas',
ls
`Is an accelerated canvas, or is a display list backed canvas that was promoted to a layer based on a performance heuristic.`
],
['plugin', ls`Is an accelerated plugin.`],
['iFrame', ls`Is an accelerated iFrame.`],
['backfaceVisibilityHidden', ls`Has backface-visibility: hidden.`],
['activeTransformAnimation', ls`Has an active accelerated transform animation or transition.`],
['activeOpacityAnimation', ls`Has an active accelerated opacity animation or transition.`],
['activeFilterAnimation', ls`Has an active accelerated filter animation or transition.`],
['activeBackdropFilterAnimation', ls`Has an active accelerated backdrop filter animation or transition.`],
['immersiveArOverlay', ls`Is DOM overlay for WebXR immersive-ar mode.`],
['scrollDependentPosition', ls`Is fixed or sticky position.`],
['overflowScrolling', ls`Is a scrollable overflow element.`],
['overflowScrollingParent', ls`Scroll parent is not an ancestor.`],
['outOfFlowClipping', ls`Has clipping ancestor.`],
['videoOverlay', ls`Is overlay controls for video.`],
['willChangeTransform', ls`Has a will-change: transform compositing hint.`],
['willChangeOpacity', ls`Has a will-change: opacity compositing hint.`],
['willChangeOther', ls`Has a will-change compositing hint other than transform and opacity.`],
['backdropFilter', ls`Has a backdrop filter.`],
['rootScroller', ls`Is the document.rootScroller.`],
['assumedOverlap', ls`Might overlap other composited content.`],
['overlap', ls`Overlaps other composited content.`],
['negativeZIndexChildren', ls`Parent with composited negative z-index content.`],
['squashingDisallowed', ls`Layer was separately composited because it could not be squashed.`],
[
'opacityWithCompositedDescendants',
ls`Has opacity that needs to be applied by compositor because of composited descendants.`
],
[
'maskWithCompositedDescendants',
ls`Has a mask that needs to be known by compositor because of composited descendants.`
],
[
'reflectionWithCompositedDescendants',
ls`Has a reflection that needs to be known by compositor because of composited descendants.`
],
[
'filterWithCompositedDescendants',
ls`Has a filter effect that needs to be known by compositor because of composited descendants.`
],
[
'blendingWithCompositedDescendants',
ls`Has a blending effect that needs to be known by compositor because of composited descendants.`
],
[
'clipsCompositingDescendants',
ls`Has a clip that needs to be known by compositor because of composited descendants.`
],
[
'perspectiveWith3DDescendants',
ls`Has a perspective transform that needs to be known by compositor because of 3d descendants.`
],
[
'preserve3DWith3DDescendants',
ls`Has a preserves-3d property that needs to be known by compositor because of 3d descendants.`
],
['isolateCompositedDescendants', ls`Should isolate descendants to apply a blend effect.`],
['positionFixedWithCompositedDescendants', ls`Is a position:fixed element with composited descendants.`],
['root', ls`Is the root layer.`],
['layerForHorizontalScrollbar', ls`Secondary layer, the horizontal scrollbar layer.`],
['layerForVerticalScrollbar', ls`Secondary layer, the vertical scrollbar layer.`],
['layerForOverflowControlsHost', ls`Secondary layer, the overflow controls host layer.`],
['layerForScrollCorner', ls`Secondary layer, the scroll corner layer.`],
['layerForScrollingContents', ls`Secondary layer, to house contents that can be scrolled.`],
['layerForScrollingContainer', ls`Secondary layer, used to position the scrolling contents while scrolling.`],
['layerForSquashingContents', ls`Secondary layer, home for a group of squashable content.`],
[
'layerForSquashingContainer',
ls`Secondary layer, no-op layer to place the squashing layer correctly in the composited layer tree.`
],
[
'layerForForeground',
ls`Secondary layer, to contain any normal flow and positive z-index contents on top of a negative z-index layer.`
],
['layerForMask', ls`Secondary layer, to contain the mask contents.`],
['layerForDecoration', ls`Layer painted on top of other layers as decoration.`],
['layerForOther', ls`Layer for link highlight, frame overlay, etc.`]
]);
/** @enum {symbol} */
export const Events = {
PaintProfilerRequested: Symbol('PaintProfilerRequested')
};
export const slowScrollRectNames = new Map([
[SDK.LayerTreeBase.Layer.ScrollRectType.NonFastScrollable, Common.UIString.UIString('Non fast scrollable')],
[SDK.LayerTreeBase.Layer.ScrollRectType.TouchEventHandler, Common.UIString.UIString('Touch event handler')],
[SDK.LayerTreeBase.Layer.ScrollRectType.WheelEventHandler, Common.UIString.UIString('Wheel event handler')],
[SDK.LayerTreeBase.Layer.ScrollRectType.RepaintsOnScroll, Common.UIString.UIString('Repaints on scroll')],
[
SDK.LayerTreeBase.Layer.ScrollRectType.MainThreadScrollingReason,
Common.UIString.UIString('Main thread scrolling reason')
]
]);
© 2015 - 2025 Weber Informatics LLC | Privacy Policy