All Downloads are FREE. Search and download functionalities are using the official Maven repository.

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