blob: e2e593a59e79e210a9ea14e95e015689d143d399 [file] [log] [blame]
// Copyright 2021 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.
/*
* Copyright (C) 2011 Google Inc. All rights reserved.
* Copyright (C) 2006, 2007, 2008 Apple Inc. All rights reserved.
* Copyright (C) 2007 Matt Lilek ([email protected]).
* 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.
*/
/* eslint-disable rulesdir/no_underscored_properties */
import * as Common from '../../core/common/common.js';
import * as DOMExtension from '../../core/dom_extension/dom_extension.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 TextUtils from '../../models/text_utils/text_utils.js';
import * as ARIAUtils from './ARIAUtils.js';
import {Dialog} from './Dialog.js';
import {Size} from './Geometry.js';
import {GlassPane, PointerEventsBehavior, SizeBehavior} from './GlassPane.js';
import {Icon} from './Icon.js';
import {KeyboardShortcut} from './KeyboardShortcut.js';
import * as ThemeSupport from './theme_support/theme_support.js'; // eslint-disable-line rulesdir/es_modules_import
import {Toolbar, ToolbarButton} from './Toolbar.js'; // eslint-disable-line no-unused-vars
import {Tooltip} from './Tooltip.js';
import {TreeOutline} from './Treeoutline.js'; // eslint-disable-line no-unused-vars
import {createShadowRootWithCoreStyles} from './utils/create-shadow-root-with-core-styles.js';
import {focusChanged} from './utils/focus-changed.js';
import {injectCoreStyles} from './utils/inject-core-styles.js';
import {measuredScrollbarWidth} from './utils/measured-scrollbar-width.js';
import {registerCustomElement} from './utils/register-custom-element.js';
const UIStrings = {
/**
*@description Micros format in UIUtils
*@example {2} PH1
*/
fmms: '{PH1} μs',
/**
*@description Sub millis format in UIUtils
*@example {2.14} PH1
*/
fms: '{PH1} ms',
/**
*@description Seconds format in UIUtils
*@example {2.14} PH1
*/
fs: '{PH1} s',
/**
*@description Minutes format in UIUtils
*@example {2.2} PH1
*/
fmin: '{PH1} min',
/**
*@description Hours format in UIUtils
*@example {2.2} PH1
*/
fhrs: '{PH1} hrs',
/**
*@description Days format in UIUtils
*@example {2.2} PH1
*/
fdays: '{PH1} days',
/**
*@description label to open link externally
*/
openInNewTab: 'Open in new tab',
/**
*@description label to copy link address
*/
copyLinkAddress: 'Copy link address',
/**
*@description label to copy file name
*/
copyFileName: 'Copy file name',
/**
*@description label for the profiler control button
*/
anotherProfilerIsAlreadyActive: 'Another profiler is already active',
/**
*@description Text in UIUtils
*/
promiseResolvedAsync: 'Promise resolved (async)',
/**
*@description Text in UIUtils
*/
promiseRejectedAsync: 'Promise rejected (async)',
/**
*@description Text in UIUtils
*@example {Promise} PH1
*/
sAsync: '{PH1} (async)',
/**
*@description Text for the title of asynchronous function calls group in Call Stack
*/
asyncCall: 'Async Call',
/**
*@description Text for the name of anonymous functions
*/
anonymous: '(anonymous)',
/**
*@description Text to close something
*/
close: 'Close',
/**
*@description Text on a button for message dialog
*/
ok: 'OK',
/**
*@description Text to cancel something
*/
cancel: 'Cancel',
};
const str_ = i18n.i18n.registerUIStrings('ui/legacy/UIUtils.ts', UIStrings);
const i18nString = i18n.i18n.getLocalizedString.bind(undefined, str_);
export const highlightedSearchResultClassName = 'highlighted-search-result';
export const highlightedCurrentSearchResultClassName = 'current-search-result';
export function installDragHandle(
element: Element, elementDragStart: ((arg0: MouseEvent) => boolean)|null, elementDrag: (arg0: MouseEvent) => void,
elementDragEnd: ((arg0: MouseEvent) => void)|null, cursor: string|null, hoverCursor?: string|null,
startDelay?: number): void {
function onMouseDown(event: Event): void {
const dragHandler = new DragHandler();
const dragStart = (): void =>
dragHandler.elementDragStart(element, elementDragStart, elementDrag, elementDragEnd, cursor, event);
if (startDelay) {
startTimer = window.setTimeout(dragStart, startDelay);
} else {
dragStart();
}
}
function onMouseUp(): void {
if (startTimer) {
window.clearTimeout(startTimer);
}
startTimer = null;
}
let startTimer: number|null;
element.addEventListener('mousedown', onMouseDown, false);
if (startDelay) {
element.addEventListener('mouseup', onMouseUp, false);
}
if (hoverCursor !== null) {
(element as HTMLElement).style.cursor = hoverCursor || cursor || '';
}
}
export function elementDragStart(
targetElement: Element, elementDragStart: ((arg0: MouseEvent) => boolean)|null,
elementDrag: (arg0: MouseEvent) => void, elementDragEnd: ((arg0: MouseEvent) => void)|null, cursor: string|null,
event: Event): void {
const dragHandler = new DragHandler();
dragHandler.elementDragStart(targetElement, elementDragStart, elementDrag, elementDragEnd, cursor, event);
}
class DragHandler {
_glassPaneInUse?: boolean;
_elementDraggingEventListener?: ((arg0: MouseEvent) => void|boolean);
_elementEndDraggingEventListener?: ((arg0: MouseEvent) => void)|null;
_dragEventsTargetDocument?: Document;
_dragEventsTargetDocumentTop?: Document;
_restoreCursorAfterDrag?: (() => void);
constructor() {
this._elementDragMove = this._elementDragMove.bind(this);
this._elementDragEnd = this._elementDragEnd.bind(this);
this._mouseOutWhileDragging = this._mouseOutWhileDragging.bind(this);
}
_createGlassPane(): void {
this._glassPaneInUse = true;
if (!DragHandler._glassPaneUsageCount++) {
DragHandler._glassPane = new GlassPane();
DragHandler._glassPane.setPointerEventsBehavior(PointerEventsBehavior.BlockedByGlassPane);
if (DragHandler._documentForMouseOut) {
DragHandler._glassPane.show(DragHandler._documentForMouseOut);
}
}
}
_disposeGlassPane(): void {
if (!this._glassPaneInUse) {
return;
}
this._glassPaneInUse = false;
if (--DragHandler._glassPaneUsageCount) {
return;
}
if (DragHandler._glassPane) {
DragHandler._glassPane.hide();
DragHandler._glassPane = null;
}
DragHandler._documentForMouseOut = null;
DragHandler._rootForMouseOut = null;
}
elementDragStart(
targetElement: Element, elementDragStart: ((arg0: MouseEvent) => boolean)|null,
elementDrag: (arg0: MouseEvent) => void|boolean, elementDragEnd: ((arg0: MouseEvent) => void)|null,
cursor: string|null, ev: Event): void {
const event = (ev as MouseEvent);
// Only drag upon left button. Right will likely cause a context menu. So will ctrl-click on mac.
if (event.button || (Host.Platform.isMac() && event.ctrlKey)) {
return;
}
if (this._elementDraggingEventListener) {
return;
}
if (elementDragStart && !elementDragStart((event as MouseEvent))) {
return;
}
const targetDocument = (event.target instanceof Node && event.target.ownerDocument) as Document;
this._elementDraggingEventListener = elementDrag;
this._elementEndDraggingEventListener = elementDragEnd;
console.assert(
(DragHandler._documentForMouseOut || targetDocument) === targetDocument, 'Dragging on multiple documents.');
DragHandler._documentForMouseOut = targetDocument;
DragHandler._rootForMouseOut = event.target instanceof Node && event.target.getRootNode() || null;
this._dragEventsTargetDocument = targetDocument;
try {
if (targetDocument.defaultView) {
this._dragEventsTargetDocumentTop = targetDocument.defaultView.top.document;
}
} catch (e) {
this._dragEventsTargetDocumentTop = this._dragEventsTargetDocument;
}
targetDocument.addEventListener('mousemove', e => this._elementDragMove((e as MouseEvent)), true);
targetDocument.addEventListener('mouseup', this._elementDragEnd, true);
DragHandler._rootForMouseOut &&
DragHandler._rootForMouseOut.addEventListener('mouseout', this._mouseOutWhileDragging, {capture: true});
if (this._dragEventsTargetDocumentTop && targetDocument !== this._dragEventsTargetDocumentTop) {
this._dragEventsTargetDocumentTop.addEventListener('mouseup', this._elementDragEnd, true);
}
const targetHtmlElement = (targetElement as HTMLElement);
if (typeof cursor === 'string') {
this._restoreCursorAfterDrag = restoreCursor.bind(this, targetHtmlElement.style.cursor);
targetHtmlElement.style.cursor = cursor;
targetDocument.body.style.cursor = cursor;
}
function restoreCursor(this: DragHandler, oldCursor: string): void {
targetDocument.body.style.removeProperty('cursor');
targetHtmlElement.style.cursor = oldCursor;
this._restoreCursorAfterDrag = undefined;
}
event.preventDefault();
}
_mouseOutWhileDragging(): void {
this._unregisterMouseOutWhileDragging();
this._createGlassPane();
}
_unregisterMouseOutWhileDragging(): void {
if (!DragHandler._rootForMouseOut) {
return;
}
DragHandler._rootForMouseOut.removeEventListener('mouseout', this._mouseOutWhileDragging, {capture: true});
}
_unregisterDragEvents(): void {
if (!this._dragEventsTargetDocument) {
return;
}
this._dragEventsTargetDocument.removeEventListener('mousemove', this._elementDragMove, true);
this._dragEventsTargetDocument.removeEventListener('mouseup', this._elementDragEnd, true);
if (this._dragEventsTargetDocumentTop && this._dragEventsTargetDocument !== this._dragEventsTargetDocumentTop) {
this._dragEventsTargetDocumentTop.removeEventListener('mouseup', this._elementDragEnd, true);
}
delete this._dragEventsTargetDocument;
delete this._dragEventsTargetDocumentTop;
}
_elementDragMove(event: MouseEvent): void {
if (event.buttons !== 1) {
this._elementDragEnd(event);
return;
}
if (this._elementDraggingEventListener && this._elementDraggingEventListener(event)) {
this._cancelDragEvents(event);
}
}
_cancelDragEvents(_event: Event): void {
this._unregisterDragEvents();
this._unregisterMouseOutWhileDragging();
if (this._restoreCursorAfterDrag) {
this._restoreCursorAfterDrag();
}
this._disposeGlassPane();
delete this._elementDraggingEventListener;
delete this._elementEndDraggingEventListener;
}
_elementDragEnd(event: Event): void {
const elementDragEnd = this._elementEndDraggingEventListener;
this._cancelDragEvents((event as MouseEvent));
event.preventDefault();
if (elementDragEnd) {
elementDragEnd((event as MouseEvent));
}
}
static _glassPaneUsageCount = 0;
static _glassPane: GlassPane|null = null;
static _documentForMouseOut: Document|null = null;
static _rootForMouseOut: Node|null = null;
}
export function isBeingEdited(node?: Node|null): boolean {
if (!node || node.nodeType !== Node.ELEMENT_NODE) {
return false;
}
const element = (node as Element);
if (element.classList.contains('text-prompt') || element.nodeName === 'INPUT' || element.nodeName === 'TEXTAREA') {
return true;
}
if (!elementsBeingEdited.size) {
return false;
}
let currentElement: (Element|null)|Element = element;
while (currentElement) {
if (elementsBeingEdited.has(element)) {
return true;
}
currentElement = currentElement.parentElementOrShadowHost();
}
return false;
}
export function isEditing(): boolean {
if (elementsBeingEdited.size) {
return true;
}
const focused = document.deepActiveElement();
if (!focused) {
return false;
}
return focused.classList.contains('text-prompt') || focused.nodeName === 'INPUT' || focused.nodeName === 'TEXTAREA';
}
export function markBeingEdited(element: Element, value: boolean): boolean {
if (value) {
if (elementsBeingEdited.has(element)) {
return false;
}
element.classList.add('being-edited');
elementsBeingEdited.add(element);
} else {
if (!elementsBeingEdited.has(element)) {
return false;
}
element.classList.remove('being-edited');
elementsBeingEdited.delete(element);
}
return true;
}
const elementsBeingEdited = new Set<Element>();
// Avoids Infinity, NaN, and scientific notation (e.g. 1e20), see crbug.com/81165.
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
const _numberRegex = /^(-?(?:\d+(?:\.\d+)?|\.\d+))$/;
export const StyleValueDelimiters = ' \xA0\t\n"\':;,/()';
export function getValueModificationDirection(event: Event): string|null {
let direction: 'Up'|'Down'|null = null;
if (event.type === 'wheel') {
// When shift is pressed while spinning mousewheel, delta comes as wheelDeltaX.
const wheelEvent = (event as WheelEvent);
if (wheelEvent.deltaY < 0 || wheelEvent.deltaX < 0) {
direction = 'Up';
} else if (wheelEvent.deltaY > 0 || wheelEvent.deltaX > 0) {
direction = 'Down';
}
} else {
const keyEvent = (event as KeyboardEvent);
if (keyEvent.key === 'ArrowUp' || keyEvent.key === 'PageUp') {
direction = 'Up';
} else if (keyEvent.key === 'ArrowDown' || keyEvent.key === 'PageDown') {
direction = 'Down';
}
}
return direction;
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
function _modifiedHexValue(hexString: string, event: Event): string|null {
const direction = getValueModificationDirection(event);
if (!direction) {
return null;
}
const mouseEvent = (event as MouseEvent);
const number = parseInt(hexString, 16);
if (isNaN(number) || !isFinite(number)) {
return null;
}
const hexStrLen = hexString.length;
const channelLen = hexStrLen / 3;
// Colors are either rgb or rrggbb.
if (channelLen !== 1 && channelLen !== 2) {
return null;
}
// Precision modifier keys work with both mousewheel and up/down keys.
// When ctrl is pressed, increase R by 1.
// When shift is pressed, increase G by 1.
// When alt is pressed, increase B by 1.
// If no shortcut keys are pressed then increase hex value by 1.
// Keys can be pressed together to increase RGB channels. e.g trying different shades.
let delta = 0;
if (KeyboardShortcut.eventHasCtrlOrMeta(mouseEvent)) {
delta += Math.pow(16, channelLen * 2);
}
if (mouseEvent.shiftKey) {
delta += Math.pow(16, channelLen);
}
if (mouseEvent.altKey) {
delta += 1;
}
if (delta === 0) {
delta = 1;
}
if (direction === 'Down') {
delta *= -1;
}
// Increase hex value by 1 and clamp from 0 ... maxValue.
const maxValue = Math.pow(16, hexStrLen) - 1;
const result = Platform.NumberUtilities.clamp(number + delta, 0, maxValue);
// Ensure the result length is the same as the original hex value.
let resultString = result.toString(16).toUpperCase();
for (let i = 0, lengthDelta = hexStrLen - resultString.length; i < lengthDelta; ++i) {
resultString = '0' + resultString;
}
return resultString;
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
function _modifiedFloatNumber(number: number, event: Event, modifierMultiplier?: number): number|null {
const direction = getValueModificationDirection(event);
if (!direction) {
return null;
}
const mouseEvent = (event as MouseEvent);
// Precision modifier keys work with both mousewheel and up/down keys.
// When ctrl is pressed, increase by 100.
// When shift is pressed, increase by 10.
// When alt is pressed, increase by 0.1.
// Otherwise increase by 1.
let delta = 1;
if (KeyboardShortcut.eventHasCtrlOrMeta(mouseEvent)) {
delta = 100;
} else if (mouseEvent.shiftKey) {
delta = 10;
} else if (mouseEvent.altKey) {
delta = 0.1;
}
if (direction === 'Down') {
delta *= -1;
}
if (modifierMultiplier) {
delta *= modifierMultiplier;
}
// Make the new number and constrain it to a precision of 6, this matches numbers the engine returns.
// Use the Number constructor to forget the fixed precision, so 1.100000 will print as 1.1.
const result = Number((number + delta).toFixed(6));
if (!String(result).match(_numberRegex)) {
return null;
}
return result;
}
export function createReplacementString(
wordString: string, event: Event,
customNumberHandler?: ((arg0: string, arg1: number, arg2: string) => string)): string|null {
let prefix;
let suffix;
let number;
let replacementString: string|null = null;
let matches = /(.*#)([\da-fA-F]+)(.*)/.exec(wordString);
if (matches && matches.length) {
prefix = matches[1];
suffix = matches[3];
number = _modifiedHexValue(matches[2], event);
if (number !== null) {
replacementString = prefix + number + suffix;
}
} else {
matches = /(.*?)(-?(?:\d+(?:\.\d+)?|\.\d+))(.*)/.exec(wordString);
if (matches && matches.length) {
prefix = matches[1];
suffix = matches[3];
number = _modifiedFloatNumber(parseFloat(matches[2]), event);
if (number !== null) {
replacementString =
customNumberHandler ? customNumberHandler(prefix, number, suffix) : prefix + number + suffix;
}
}
}
return replacementString;
}
export function handleElementValueModifications(
event: Event, element: Element, finishHandler?: ((arg0: string, arg1: string) => void),
suggestionHandler?: ((arg0: string) => boolean),
customNumberHandler?: ((arg0: string, arg1: number, arg2: string) => string)): boolean {
const arrowKeyOrWheelEvent =
((event as KeyboardEvent).key === 'ArrowUp' || (event as KeyboardEvent).key === 'ArrowDown' ||
event.type === 'wheel');
const pageKeyPressed = ((event as KeyboardEvent).key === 'PageUp' || (event as KeyboardEvent).key === 'PageDown');
if (!arrowKeyOrWheelEvent && !pageKeyPressed) {
return false;
}
const selection = element.getComponentSelection();
if (!selection || !selection.rangeCount) {
return false;
}
const selectionRange = selection.getRangeAt(0);
if (!selectionRange.commonAncestorContainer.isSelfOrDescendant(element)) {
return false;
}
const originalValue = element.textContent;
const wordRange = DOMExtension.DOMExtension.rangeOfWord(
selectionRange.startContainer, selectionRange.startOffset, StyleValueDelimiters, element);
const wordString = wordRange.toString();
if (suggestionHandler && suggestionHandler(wordString)) {
return false;
}
const replacementString = createReplacementString(wordString, event, customNumberHandler);
if (replacementString) {
const replacementTextNode = document.createTextNode(replacementString);
wordRange.deleteContents();
wordRange.insertNode(replacementTextNode);
const finalSelectionRange = document.createRange();
finalSelectionRange.setStart(replacementTextNode, 0);
finalSelectionRange.setEnd(replacementTextNode, replacementString.length);
selection.removeAllRanges();
selection.addRange(finalSelectionRange);
event.handled = true;
event.preventDefault();
if (finishHandler) {
finishHandler(originalValue || '', replacementString);
}
return true;
}
return false;
}
Number.preciseMillisToString = function(ms: number, precision?: number): string {
precision = precision || 0;
return i18nString(UIStrings.fms, {PH1: ms.toFixed(precision)});
};
Number.millisToString = function(ms: number, higherResolution?: boolean): string {
if (!isFinite(ms)) {
return '-';
}
if (ms === 0) {
return '0';
}
if (higherResolution && ms < 0.1) {
return i18nString(UIStrings.fmms, {PH1: (ms * 1000).toFixed(0)});
}
if (higherResolution && ms < 1000) {
return i18nString(UIStrings.fms, {PH1: (ms).toFixed(2)});
}
if (ms < 1000) {
return i18nString(UIStrings.fms, {PH1: (ms).toFixed(0)});
}
const seconds = ms / 1000;
if (seconds < 60) {
return i18nString(UIStrings.fs, {PH1: (seconds).toFixed(2)});
}
const minutes = seconds / 60;
if (minutes < 60) {
return i18nString(UIStrings.fmin, {PH1: (minutes).toFixed(1)});
}
const hours = minutes / 60;
if (hours < 24) {
return i18nString(UIStrings.fhrs, {PH1: (hours).toFixed(1)});
}
const days = hours / 24;
return i18nString(UIStrings.fdays, {PH1: (days).toFixed(1)});
};
Number.secondsToString = function(seconds: number, higherResolution?: boolean): string {
if (!isFinite(seconds)) {
return '-';
}
return Number.millisToString(seconds * 1000, higherResolution);
};
Number.withThousandsSeparator = function(num: number): string {
let str = String(num);
const re = /(\d+)(\d{3})/;
while (str.match(re)) {
str = str.replace(re, '$1\xA0$2');
} // \xa0 is a non-breaking space
return str;
};
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function formatLocalized(format: string, substitutions: ArrayLike<any>|null): Element {
const formatters = {
s: (substitution: unknown): unknown => substitution,
};
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
function append(a: Element, b: any): Element {
a.appendChild(typeof b === 'string' ? document.createTextNode(b) : b as Element);
return a;
}
return Platform.StringUtilities.format(format, substitutions, formatters, document.createElement('span'), append)
.formattedResult;
}
export function openLinkExternallyLabel(): string {
return i18nString(UIStrings.openInNewTab);
}
export function copyLinkAddressLabel(): string {
return i18nString(UIStrings.copyLinkAddress);
}
export function copyFileNameLabel(): string {
return i18nString(UIStrings.copyFileName);
}
export function anotherProfilerActiveLabel(): string {
return i18nString(UIStrings.anotherProfilerIsAlreadyActive);
}
export function asyncStackTraceLabel(description: string|undefined): string {
if (description) {
if (description === 'Promise.resolve') {
return i18nString(UIStrings.promiseResolvedAsync);
}
if (description === 'Promise.reject') {
return i18nString(UIStrings.promiseRejectedAsync);
}
return i18nString(UIStrings.sAsync, {PH1: description});
}
return i18nString(UIStrings.asyncCall);
}
export function installComponentRootStyles(element: Element): void {
injectCoreStyles(element);
element.classList.add('platform-' + Host.Platform.platform());
// Detect overlay scrollbar enable by checking for nonzero scrollbar width.
if (!Host.Platform.isMac() && measuredScrollbarWidth(element.ownerDocument) === 0) {
element.classList.add('overlay-scrollbar-enabled');
}
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
function _windowFocused(document: Document, event: Event): void {
if (event.target instanceof Window && event.target.document.nodeType === Node.DOCUMENT_NODE) {
document.body.classList.remove('inactive');
}
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
function _windowBlurred(document: Document, event: Event): void {
if (event.target instanceof Window && event.target.document.nodeType === Node.DOCUMENT_NODE) {
document.body.classList.add('inactive');
}
}
export class ElementFocusRestorer {
_element: HTMLElement|null;
_previous: HTMLElement|null;
constructor(element: Element) {
this._element = (element as HTMLElement | null);
this._previous = (element.ownerDocument.deepActiveElement() as HTMLElement | null);
(element as HTMLElement).focus();
}
restore(): void {
if (!this._element) {
return;
}
if (this._element.hasFocus() && this._previous) {
this._previous.focus();
}
this._previous = null;
this._element = null;
}
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function highlightSearchResult(element: Element, offset: number, length: number, domChanges?: any[]): Element|
null {
const result = highlightSearchResults(element, [new TextUtils.TextRange.SourceRange(offset, length)], domChanges);
return result.length ? result[0] : null;
}
export function highlightSearchResults(
element: Element, resultRanges: TextUtils.TextRange.SourceRange[], changes?: HighlightChange[]): Element[] {
return highlightRangesWithStyleClass(element, resultRanges, highlightedSearchResultClassName, changes);
}
export function runCSSAnimationOnce(element: Element, className: string): void {
function animationEndCallback(): void {
element.classList.remove(className);
element.removeEventListener('webkitAnimationEnd', animationEndCallback, false);
}
if (element.classList.contains(className)) {
element.classList.remove(className);
}
element.addEventListener('webkitAnimationEnd', animationEndCallback, false);
element.classList.add(className);
}
export function highlightRangesWithStyleClass(
element: Element, resultRanges: TextUtils.TextRange.SourceRange[], styleClass: string,
changes?: HighlightChange[]): Element[] {
changes = changes || [];
const highlightNodes: Element[] = [];
const textNodes = element.childTextNodes();
const lineText = textNodes
.map(function(node) {
return node.textContent;
})
.join('');
const ownerDocument = element.ownerDocument;
if (textNodes.length === 0) {
return highlightNodes;
}
const nodeRanges: TextUtils.TextRange.SourceRange[] = [];
let rangeEndOffset = 0;
for (const textNode of textNodes) {
const range =
new TextUtils.TextRange.SourceRange(rangeEndOffset, textNode.textContent ? textNode.textContent.length : 0);
rangeEndOffset = range.offset + range.length;
nodeRanges.push(range);
}
let startIndex = 0;
for (let i = 0; i < resultRanges.length; ++i) {
const startOffset = resultRanges[i].offset;
const endOffset = startOffset + resultRanges[i].length;
while (startIndex < textNodes.length &&
nodeRanges[startIndex].offset + nodeRanges[startIndex].length <= startOffset) {
startIndex++;
}
let endIndex = startIndex;
while (endIndex < textNodes.length && nodeRanges[endIndex].offset + nodeRanges[endIndex].length < endOffset) {
endIndex++;
}
if (endIndex === textNodes.length) {
break;
}
const highlightNode = ownerDocument.createElement('span');
highlightNode.className = styleClass;
highlightNode.textContent = lineText.substring(startOffset, endOffset);
const lastTextNode = textNodes[endIndex];
const lastText = lastTextNode.textContent || '';
lastTextNode.textContent = lastText.substring(endOffset - nodeRanges[endIndex].offset);
changes.push({
node: (lastTextNode as Element),
type: 'changed',
oldText: lastText,
newText: lastTextNode.textContent,
nextSibling: undefined,
parent: undefined,
});
if (startIndex === endIndex && lastTextNode.parentElement) {
lastTextNode.parentElement.insertBefore(highlightNode, lastTextNode);
changes.push({
node: highlightNode,
type: 'added',
nextSibling: lastTextNode,
parent: lastTextNode.parentElement,
oldText: undefined,
newText: undefined,
});
highlightNodes.push(highlightNode);
const prefixNode =
ownerDocument.createTextNode(lastText.substring(0, startOffset - nodeRanges[startIndex].offset));
lastTextNode.parentElement.insertBefore(prefixNode, highlightNode);
changes.push({
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
node: (prefixNode as any),
type: 'added',
nextSibling: highlightNode,
parent: lastTextNode.parentElement,
oldText: undefined,
newText: undefined,
});
} else {
const firstTextNode = textNodes[startIndex];
const firstText = firstTextNode.textContent || '';
const anchorElement = firstTextNode.nextSibling;
if (firstTextNode.parentElement) {
firstTextNode.parentElement.insertBefore(highlightNode, anchorElement);
changes.push({
node: highlightNode,
type: 'added',
nextSibling: anchorElement || undefined,
parent: firstTextNode.parentElement,
oldText: undefined,
newText: undefined,
});
highlightNodes.push(highlightNode);
}
firstTextNode.textContent = firstText.substring(0, startOffset - nodeRanges[startIndex].offset);
changes.push({
node: (firstTextNode as Element),
type: 'changed',
oldText: firstText,
newText: firstTextNode.textContent,
nextSibling: undefined,
parent: undefined,
});
for (let j = startIndex + 1; j < endIndex; j++) {
const textNode = textNodes[j];
const text = textNode.textContent;
textNode.textContent = '';
changes.push({
node: (textNode as Element),
type: 'changed',
oldText: text || undefined,
newText: textNode.textContent,
nextSibling: undefined,
parent: undefined,
});
}
}
startIndex = endIndex;
nodeRanges[startIndex].offset = endOffset;
nodeRanges[startIndex].length = lastTextNode.textContent.length;
}
return highlightNodes;
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function applyDomChanges(domChanges: any[]): void {
for (let i = 0, size = domChanges.length; i < size; ++i) {
const entry = domChanges[i];
switch (entry.type) {
case 'added':
entry.parent.insertBefore(entry.node, entry.nextSibling);
break;
case 'changed':
entry.node.textContent = entry.newText;
break;
}
}
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function revertDomChanges(domChanges: any[]): void {
for (let i = domChanges.length - 1; i >= 0; --i) {
const entry = domChanges[i];
switch (entry.type) {
case 'added':
entry.node.remove();
break;
case 'changed':
entry.node.textContent = entry.oldText;
break;
}
}
}
export function measurePreferredSize(element: Element, containerElement?: Element|null): Size {
const oldParent = element.parentElement;
const oldNextSibling = element.nextSibling;
containerElement = containerElement || element.ownerDocument.body;
containerElement.appendChild(element);
element.positionAt(0, 0);
const result = element.getBoundingClientRect();
element.positionAt(undefined, undefined);
if (oldParent) {
oldParent.insertBefore(element, oldNextSibling);
} else {
element.remove();
}
return new Size(result.width, result.height);
}
class InvokeOnceHandlers {
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
_handlers: Map<any, any>|null;
_autoInvoke: boolean;
constructor(autoInvoke: boolean) {
this._handlers = null;
this._autoInvoke = autoInvoke;
}
add(object: Object, method: () => void): void {
if (!this._handlers) {
this._handlers = new Map();
if (this._autoInvoke) {
this.scheduleInvoke();
}
}
let methods = this._handlers.get(object);
if (!methods) {
methods = new Set();
this._handlers.set(object, methods);
}
methods.add(method);
}
scheduleInvoke(): void {
if (this._handlers) {
requestAnimationFrame(this._invoke.bind(this));
}
}
_invoke(): void {
const handlers = this._handlers || new Map(); // Make closure happy. This should not be null.
this._handlers = null;
for (const [object, methods] of handlers) {
for (const method of methods) {
method.call(object);
}
}
}
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
let _coalescingLevel = 0;
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
let _postUpdateHandlers: InvokeOnceHandlers|null = null;
export function startBatchUpdate(): void {
if (!_coalescingLevel++) {
_postUpdateHandlers = new InvokeOnceHandlers(false);
}
}
export function endBatchUpdate(): void {
if (--_coalescingLevel) {
return;
}
if (_postUpdateHandlers) {
_postUpdateHandlers.scheduleInvoke();
_postUpdateHandlers = null;
}
}
export function invokeOnceAfterBatchUpdate(object: Object, method: () => void): void {
if (!_postUpdateHandlers) {
_postUpdateHandlers = new InvokeOnceHandlers(true);
}
_postUpdateHandlers.add(object, method);
}
export function animateFunction(
window: Window, func: Function, params: {
from: number,
to: number,
}[],
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
duration: number, animationComplete?: (() => any)): () => void {
const start = window.performance.now();
let raf = window.requestAnimationFrame(animationStep);
function animationStep(timestamp: number): void {
const progress = Platform.NumberUtilities.clamp((timestamp - start) / duration, 0, 1);
func(...params.map(p => p.from + (p.to - p.from) * progress));
if (progress < 1) {
raf = window.requestAnimationFrame(animationStep);
} else if (animationComplete) {
animationComplete();
}
}
return (): void => window.cancelAnimationFrame(raf);
}
export class LongClickController extends Common.ObjectWrapper.ObjectWrapper {
_element: Element;
_callback: (arg0: Event) => void;
_editKey: (arg0: Event) => boolean;
_longClickData!: {
mouseUp: (arg0: Event) => void,
mouseDown: (arg0: Event) => void,
reset: () => void,
}|undefined;
_longClickInterval!: number|undefined;
constructor(
element: Element, callback: (arg0: Event) => void,
isEditKeyFunc: (arg0: Event) => boolean = (event): boolean => isEnterOrSpaceKey(event)) {
super();
this._element = element;
this._callback = callback;
this._editKey = isEditKeyFunc;
this._enable();
}
reset(): void {
if (this._longClickInterval) {
clearInterval(this._longClickInterval);
delete this._longClickInterval;
}
}
_enable(): void {
if (this._longClickData) {
return;
}
const boundKeyDown = keyDown.bind(this);
const boundKeyUp = keyUp.bind(this);
const boundMouseDown = mouseDown.bind(this);
const boundMouseUp = mouseUp.bind(this);
const boundReset = this.reset.bind(this);
this._element.addEventListener('keydown', boundKeyDown, false);
this._element.addEventListener('keyup', boundKeyUp, false);
this._element.addEventListener('mousedown', boundMouseDown, false);
this._element.addEventListener('mouseout', boundReset, false);
this._element.addEventListener('mouseup', boundMouseUp, false);
this._element.addEventListener('click', boundReset, true);
this._longClickData = {mouseUp: boundMouseUp, mouseDown: boundMouseDown, reset: boundReset};
function keyDown(this: LongClickController, e: Event): void {
if (this._editKey(e)) {
const callback = this._callback;
this._longClickInterval = window.setTimeout(callback.bind(null, e), LongClickController.TIME_MS);
}
}
function keyUp(this: LongClickController, e: Event): void {
if (this._editKey(e)) {
this.reset();
}
}
function mouseDown(this: LongClickController, e: Event): void {
if ((e as MouseEvent).which !== 1) {
return;
}
const callback = this._callback;
this._longClickInterval = window.setTimeout(callback.bind(null, e), LongClickController.TIME_MS);
}
function mouseUp(this: LongClickController, e: Event): void {
if ((e as MouseEvent).which !== 1) {
return;
}
this.reset();
}
}
dispose(): void {
if (!this._longClickData) {
return;
}
this._element.removeEventListener('mousedown', this._longClickData.mouseDown, false);
this._element.removeEventListener('mouseout', this._longClickData.reset, false);
this._element.removeEventListener('mouseup', this._longClickData.mouseUp, false);
this._element.addEventListener('click', this._longClickData.reset, true);
delete this._longClickData;
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/naming-convention
static TIME_MS = 200;
}
export function initializeUIUtils(document: Document, themeSetting: Common.Settings.Setting<string>): void {
document.body.classList.toggle('inactive', !document.hasFocus());
if (document.defaultView) {
document.defaultView.addEventListener('focus', _windowFocused.bind(undefined, document), false);
document.defaultView.addEventListener('blur', _windowBlurred.bind(undefined, document), false);
}
document.addEventListener('focus', focusChanged.bind(undefined), true);
if (!ThemeSupport.ThemeSupport.hasInstance()) {
ThemeSupport.ThemeSupport.instance({forceNew: true, setting: themeSetting});
}
ThemeSupport.ThemeSupport.instance().applyTheme(document);
const body = (document.body as Element);
GlassPane.setContainer(body);
}
export function beautifyFunctionName(name: string): string {
return name || i18nString(UIStrings.anonymous);
}
export const createTextChild = (element: Element|DocumentFragment, text: string): Text => {
const textNode = element.ownerDocument.createTextNode(text);
element.appendChild(textNode);
return textNode;
};
export const createTextChildren = (element: Element|DocumentFragment, ...childrenText: string[]): void => {
for (const child of childrenText) {
createTextChild(element, child);
}
};
export function createTextButton(
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
text: string, eventHandler?: ((arg0: Event) => any), className?: string, primary?: boolean,
alternativeEvent?: string): HTMLButtonElement {
const element = (document.createElement('button') as HTMLButtonElement);
if (className) {
element.className = className;
}
element.textContent = text;
element.classList.add('text-button');
if (primary) {
element.classList.add('primary-button');
}
if (eventHandler) {
element.addEventListener(alternativeEvent || 'click', eventHandler);
}
element.type = 'button';
return element;
}
export function createInput(className?: string, type?: string): HTMLInputElement {
const element = document.createElement('input');
if (className) {
element.className = className;
}
element.spellcheck = false;
element.classList.add('harmony-input');
if (type) {
element.type = type;
}
return /** @type {!HTMLInputElement} */ element as HTMLInputElement;
}
export function createSelect(name: string, options: string[]|Map<string, string[]>[]|Set<string>): HTMLSelectElement {
const select = (document.createElementWithClass('select', 'chrome-select') as HTMLSelectElement);
ARIAUtils.setAccessibleName(select, name);
for (const option of options) {
if (option instanceof Map) {
for (const [key, value] of option) {
const optGroup = (select.createChild('optgroup') as HTMLOptGroupElement);
optGroup.label = key;
for (const child of value) {
if (typeof child === 'string') {
optGroup.appendChild(new Option(child, child));
}
}
}
} else if (typeof option === 'string') {
select.add(new Option(option, option));
}
}
return select;
}
export function createLabel(title: string, className?: string, associatedControl?: Element): Element {
const element = document.createElement('label');
if (className) {
element.className = className;
}
element.textContent = title;
if (associatedControl) {
ARIAUtils.bindLabelToControl(element, associatedControl);
}
return element;
}
export function createRadioLabel(name: string, title: string, checked?: boolean): DevToolsRadioButton {
const element = (document.createElement('span', {is: 'dt-radio'}) as DevToolsRadioButton);
element.radioElement.name = name;
element.radioElement.checked = Boolean(checked);
createTextChild(element.labelElement, title);
return element;
}
export function createIconLabel(title: string, iconClass: string): HTMLElement {
const element = (document.createElement('span', {is: 'dt-icon-label'}) as DevToolsIconLabel);
element.createChild('span').textContent = title;
element.type = iconClass;
return element;
}
export function createSlider(min: number, max: number, tabIndex: number): Element {
const element = (document.createElement('span', {is: 'dt-slider'}) as DevToolsSlider);
element.sliderElement.min = String(min);
element.sliderElement.max = String(max);
element.sliderElement.step = String(1);
element.sliderElement.tabIndex = tabIndex;
return element;
}
export function setTitle(element: HTMLElement, title: string, actionId: string|undefined = undefined): void {
ARIAUtils.setAccessibleName(element, title);
Tooltip.install(element, title, actionId, {
anchorTooltipAtElement: true,
});
}
export class CheckboxLabel extends HTMLSpanElement {
_shadowRoot!: DocumentFragment;
checkboxElement!: HTMLInputElement;
textElement!: Element;
constructor() {
super();
CheckboxLabel._lastId = CheckboxLabel._lastId + 1;
const id = 'ui-checkbox-label' + CheckboxLabel._lastId;
this._shadowRoot = createShadowRootWithCoreStyles(
this, {cssFile: 'ui/legacy/checkboxTextLabel.css', enableLegacyPatching: true, delegatesFocus: undefined});
this.checkboxElement = (this._shadowRoot.createChild('input') as HTMLInputElement);
this.checkboxElement.type = 'checkbox';
this.checkboxElement.setAttribute('id', id);
this.textElement = this._shadowRoot.createChild('label', 'dt-checkbox-text');
this.textElement.setAttribute('for', id);
this._shadowRoot.createChild('slot');
}
static create(title?: string, checked?: boolean, subtitle?: string): CheckboxLabel {
if (!CheckboxLabel._constructor) {
CheckboxLabel._constructor = registerCustomElement('span', 'dt-checkbox', CheckboxLabel);
}
const element = (CheckboxLabel._constructor() as CheckboxLabel);
element.checkboxElement.checked = Boolean(checked);
if (title !== undefined) {
element.textElement.textContent = title;
ARIAUtils.setAccessibleName(element.checkboxElement, title);
if (subtitle !== undefined) {
element.textElement.createChild('div', 'dt-checkbox-subtitle').textContent = subtitle;
}
}
return element;
}
set backgroundColor(color: string) {
this.checkboxElement.classList.add('dt-checkbox-themed');
this.checkboxElement.style.backgroundColor = color;
}
set checkColor(color: string) {
this.checkboxElement.classList.add('dt-checkbox-themed');
const stylesheet = document.createElement('style');
stylesheet.textContent = 'input.dt-checkbox-themed:checked:after { background-color: ' + color + '}';
this._shadowRoot.appendChild(stylesheet);
}
set borderColor(color: string) {
this.checkboxElement.classList.add('dt-checkbox-themed');
this.checkboxElement.style.borderColor = color;
}
static _lastId = 0;
static _constructor: (() => Element)|null = null;
}
export class DevToolsIconLabel extends HTMLSpanElement {
_iconElement: Icon;
constructor() {
super();
const root = createShadowRootWithCoreStyles(this, {
enableLegacyPatching: true,
cssFile: undefined,
delegatesFocus: undefined,
});
this._iconElement = Icon.create();
this._iconElement.style.setProperty('margin-right', '4px');
root.appendChild(this._iconElement);
root.createChild('slot');
}
set type(type: string) {
this._iconElement.setIconType(type);
}
}
let labelId = 0;
export class DevToolsRadioButton extends HTMLSpanElement {
radioElement: HTMLInputElement;
labelElement: HTMLLabelElement;
constructor() {
super();
this.radioElement = (this.createChild('input', 'dt-radio-button') as HTMLInputElement);
this.labelElement = (this.createChild('label') as HTMLLabelElement);
const id = 'dt-radio-button-id' + (++labelId);
this.radioElement.id = id;
this.radioElement.type = 'radio';
this.labelElement.htmlFor = id;
const root = createShadowRootWithCoreStyles(
this, {cssFile: 'ui/legacy/radioButton.css', enableLegacyPatching: false, delegatesFocus: undefined});
root.createChild('slot');
this.addEventListener('click', this.radioClickHandler.bind(this), false);
}
radioClickHandler(): void {
if (this.radioElement.checked || this.radioElement.disabled) {
return;
}
this.radioElement.checked = true;
this.radioElement.dispatchEvent(new Event('change'));
}
}
registerCustomElement('span', 'dt-radio', DevToolsRadioButton);
registerCustomElement('span', 'dt-icon-label', DevToolsIconLabel);
export class DevToolsSlider extends HTMLSpanElement {
sliderElement: HTMLInputElement;
constructor() {
super();
const root = createShadowRootWithCoreStyles(
this, {cssFile: 'ui/legacy/slider.css', enableLegacyPatching: true, delegatesFocus: undefined});
this.sliderElement = document.createElement('input');
this.sliderElement.classList.add('dt-range-input');
this.sliderElement.type = 'range';
root.appendChild(this.sliderElement);
}
set value(amount: number) {
this.sliderElement.value = String(amount);
}
get value(): number {
return Number(this.sliderElement.value);
}
}
registerCustomElement('span', 'dt-slider', DevToolsSlider);
export class DevToolsSmallBubble extends HTMLSpanElement {
_textElement: Element;
constructor() {
super();
const root = createShadowRootWithCoreStyles(
this, {cssFile: 'ui/legacy/smallBubble.css', enableLegacyPatching: false, delegatesFocus: undefined});
this._textElement = root.createChild('div');
this._textElement.className = 'info';
this._textElement.createChild('slot');
}
set type(type: string) {
this._textElement.className = type;
}
}
registerCustomElement('span', 'dt-small-bubble', DevToolsSmallBubble);
export class DevToolsCloseButton extends HTMLDivElement {
_buttonElement: HTMLElement;
_hoverIcon: Icon;
_activeIcon: Icon;
constructor() {
super();
const root = createShadowRootWithCoreStyles(
this, {cssFile: 'ui/legacy/closeButton.css', enableLegacyPatching: false, delegatesFocus: undefined});
this._buttonElement = (root.createChild('div', 'close-button') as HTMLElement);
ARIAUtils.setAccessibleName(this._buttonElement, i18nString(UIStrings.close));
ARIAUtils.markAsButton(this._buttonElement);
const regularIcon = Icon.create('smallicon-cross', 'default-icon');
this._hoverIcon = Icon.create('mediumicon-red-cross-hover', 'hover-icon');
this._activeIcon = Icon.create('mediumicon-red-cross-active', 'active-icon');
this._buttonElement.appendChild(regularIcon);
this._buttonElement.appendChild(this._hoverIcon);
this._buttonElement.appendChild(this._activeIcon);
}
set gray(gray: boolean) {
if (gray) {
this._hoverIcon.setIconType('mediumicon-gray-cross-hover');
this._activeIcon.setIconType('mediumicon-gray-cross-active');
} else {
this._hoverIcon.setIconType('mediumicon-red-cross-hover');
this._activeIcon.setIconType('mediumicon-red-cross-active');
}
}
setAccessibleName(name: string): void {
ARIAUtils.setAccessibleName(this._buttonElement, name);
}
setTabbable(tabbable: boolean): void {
if (tabbable) {
this._buttonElement.tabIndex = 0;
} else {
this._buttonElement.tabIndex = -1;
}
}
}
registerCustomElement('div', 'dt-close-button', DevToolsCloseButton);
export function bindInput(
input: HTMLInputElement, apply: (arg0: string) => void, validate: (arg0: string) => {
valid: boolean,
errorMessage: (string | undefined),
},
numeric: boolean, modifierMultiplier?: number): (arg0: string) => void {
input.addEventListener('change', onChange, false);
input.addEventListener('input', onInput, false);
input.addEventListener('keydown', onKeyDown, false);
input.addEventListener('focus', input.select.bind(input), false);
function onInput(): void {
input.classList.toggle('error-input', !validate(input.value));
}
function onChange(): void {
const {valid} = validate(input.value);
input.classList.toggle('error-input', !valid);
if (valid) {
apply(input.value);
}
}
function onKeyDown(event: KeyboardEvent): void {
if (event.key === 'Enter') {
const {valid} = validate(input.value);
if (valid) {
apply(input.value);
}
event.preventDefault();
return;
}
if (!numeric) {
return;
}
const value = _modifiedFloatNumber(parseFloat(input.value), event, modifierMultiplier);
const stringValue = value ? String(value) : '';
const {valid} = validate(stringValue);
if (!valid || !value) {
return;
}
input.value = stringValue;
apply(input.value);
event.preventDefault();
}
function setValue(value: string): void {
if (value === input.value) {
return;
}
const {valid} = validate(value);
input.classList.toggle('error-input', !valid);
input.value = value;
}
return setValue;
}
export function trimText(
context: CanvasRenderingContext2D, text: string, maxWidth: number,
trimFunction: (arg0: string, arg1: number) => string): string {
const maxLength = 200;
if (maxWidth <= 10) {
return '';
}
if (text.length > maxLength) {
text = trimFunction(text, maxLength);
}
const textWidth = measureTextWidth(context, text);
if (textWidth <= maxWidth) {
return text;
}
let l = 0;
let r: number = text.length;
let lv = 0;
let rv: number = textWidth;
while (l < r && lv !== rv && lv !== maxWidth) {
const m = Math.ceil(l + (r - l) * (maxWidth - lv) / (rv - lv));
const mv = measureTextWidth(context, trimFunction(text, m));
if (mv <= maxWidth) {
l = m;
lv = mv;
} else {
r = m - 1;
rv = mv;
}
}
text = trimFunction(text, l);
return text !== '…' ? text : '';
}
export function trimTextMiddle(context: CanvasRenderingContext2D, text: string, maxWidth: number): string {
return trimText(context, text, maxWidth, (text, width) => Platform.StringUtilities.trimMiddle(text, width));
}
export function trimTextEnd(context: CanvasRenderingContext2D, text: string, maxWidth: number): string {
return trimText(context, text, maxWidth, (text, width) => Platform.StringUtilities.trimEndWithMaxLength(text, width));
}
export function measureTextWidth(context: CanvasRenderingContext2D, text: string): number {
const maxCacheableLength = 200;
if (text.length > maxCacheableLength) {
return context.measureText(text).width;
}
if (!measureTextWidthCache) {
measureTextWidthCache = new Map();
}
const font = context.font;
let textWidths = measureTextWidthCache.get(font);
if (!textWidths) {
textWidths = new Map();
measureTextWidthCache.set(font, textWidths);
}
let width = textWidths.get(text);
if (!width) {
width = context.measureText(text).width;
textWidths.set(text, width);
}
return width;
}
let measureTextWidthCache: Map<string, Map<string, number>>|null = null;
/**
* Adds a 'utm_source=devtools' as query parameter to the url.
*/
export function addReferrerToURL(url: string): string {
if (/(\?|&)utm_source=devtools/.test(url)) {
return url;
}
if (url.indexOf('?') === -1) {
// If the URL does not contain a query, add the referrer query after path
// and before (potential) anchor.
return url.replace(/^([^#]*)(#.*)?$/g, '$1?utm_source=devtools$2');
}
// If the URL already contains a query, add the referrer query after the last query
// and before (potential) anchor.
return url.replace(/^([^#]*)(#.*)?$/g, '$1&utm_source=devtools$2');
}
/**
* We want to add a referrer query param to every request to
* 'web.dev' or 'developers.google.com'.
*/
export function addReferrerToURLIfNecessary(url: string): string {
if (/(\/\/developers.google.com\/|\/\/web.dev\/)/.test(url)) {
return addReferrerToURL(url);
}
return url;
}
export function loadImage(url: string): Promise<HTMLImageElement|null> {
return new Promise(fulfill => {
const image = new Image();
image.addEventListener('load', () => fulfill(image));
image.addEventListener('error', () => fulfill(null));
image.src = url;
});
}
export function loadImageFromData(data: string|null): Promise<HTMLImageElement|null> {
return data ? loadImage('data:image/jpg;base64,' + data) : Promise.resolve(null);
}
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// eslint-disable-next-line @typescript-eslint/no-explicit-any
export function createFileSelectorElement(callback: (arg0: File) => any): HTMLInputElement {
const fileSelectorElement = (document.createElement('input') as HTMLInputElement);
fileSelectorElement.type = 'file';
fileSelectorElement.style.display = 'none';
fileSelectorElement.tabIndex = -1;
fileSelectorElement.onchange = (): void => {
if (fileSelectorElement.files) {
callback(fileSelectorElement.files[0]);
}
};
return fileSelectorElement;
}
export const MaxLengthForDisplayedURLs = 150;
export class MessageDialog {
static async show(message: string, where?: Element | Document): Promise<void> {
const dialog = new Dialog();
dialog.setSizeBehavior(SizeBehavior.MeasureContent);
dialog.setDimmed(true);
const shadowRoot = createShadowRootWithCoreStyles(
dialog.contentElement,
{cssFile: 'ui/legacy/confirmDialog.css', enableLegacyPatching: false, delegatesFocus: undefined});
const content = shadowRoot.createChild('div', 'widget');
await new Promise(resolve => {
const okButton = createTextButton(i18nString(UIStrings.ok), resolve, '', true);
content.createChild('div', 'message').createChild('span').textContent = message;
content.createChild('div', 'button').appendChild(okButton);
dialog.setOutsideClickCallback(event => {
event.consume();
resolve(undefined);
});
dialog.show(where);
okButton.focus();
});
dialog.hide();
}
}
export class ConfirmDialog {
static async show(message: string, where?: Element | Document): Promise<boolean> {
const dialog = new Dialog();
dialog.setSizeBehavior(SizeBehavior.MeasureContent);
dialog.setDimmed(true);
ARIAUtils.setAccessibleName(dialog.contentElement, message);
const shadowRoot = createShadowRootWithCoreStyles(
dialog.contentElement,
{cssFile: 'ui/legacy/confirmDialog.css', enableLegacyPatching: false, delegatesFocus: undefined});
const content = shadowRoot.createChild('div', 'widget');
content.createChild('div', 'message').createChild('span').textContent = message;
const buttonsBar = content.createChild('div', 'button');
const result = await new Promise<boolean>(resolve => {
const okButton = createTextButton(
/* text= */ i18nString(UIStrings.ok), /* clickHandler= */ () => resolve(true), /* className= */ '',
/* primary= */ true);
buttonsBar.appendChild(okButton);
buttonsBar.appendChild(createTextButton(i18nString(UIStrings.cancel), () => resolve(false)));
dialog.setOutsideClickCallback(event => {
event.consume();
resolve(false);
});
dialog.show(where);
okButton.focus();
});
dialog.hide();
return result;
}
}
export function createInlineButton(toolbarButton: ToolbarButton): Element {
const element = document.createElement('span');
const shadowRoot = createShadowRootWithCoreStyles(
element, {cssFile: 'ui/legacy/inlineButton.css', enableLegacyPatching: false, delegatesFocus: undefined});
element.classList.add('inline-button');
const toolbar = new Toolbar('');
toolbar.appendToolbarItem(toolbarButton);
shadowRoot.appendChild(toolbar.element);
return element;
}
export abstract class Renderer {
abstract render(object: Object, options?: Options): Promise<{
node: Node,
tree: TreeOutline|null,
}|null>;
static async render(object: Object, options?: Options): Promise<{
node: Node,
tree: TreeOutline|null,
}|null> {
if (!object) {
throw new Error('Can\'t render ' + object);
}
const extension = getApplicableRegisteredRenderers(object)[0];
if (!extension) {
return null;
}
const renderer = await extension.loadRenderer();
return renderer.render(object, options);
}
}
export function formatTimestamp(timestamp: number, full: boolean): string {
const date = new Date(timestamp);
const yymmdd = date.getFullYear() + '-' + leadZero(date.getMonth() + 1, 2) + '-' + leadZero(date.getDate(), 2);
const hhmmssfff = leadZero(date.getHours(), 2) + ':' + leadZero(date.getMinutes(), 2) + ':' +
leadZero(date.getSeconds(), 2) + '.' + leadZero(date.getMilliseconds(), 3);
return full ? (yymmdd + ' ' + hhmmssfff) : hhmmssfff;
function leadZero(value: number, length: number): string {
const valueString = String(value);
return valueString.padStart(length, '0');
}
}
export interface Options {
title?: string|Element;
editable?: boolean;
}
export interface HighlightChange {
node: Element;
type: string;
oldText?: string;
newText?: string;
nextSibling?: Node;
parent?: Node;
}
export const isScrolledToBottom = (element: Element): boolean => {
// This code works only for 0-width border.
// The scrollTop, clientHeight and scrollHeight are computed in double values internally.
// However, they are exposed to javascript differently, each being either rounded (via
// round, ceil or floor functions) or left intouch.
// This adds up a total error up to 2.
return Math.abs(element.scrollTop + element.clientHeight - element.scrollHeight) <= 2;
};
export function createSVGChild(element: Element, childType: string, className?: string): Element {
const child = element.ownerDocument.createElementNS('http://www.w3.org/2000/svg', childType);
if (className) {
child.setAttribute('class', className);
}
element.appendChild(child);
return child;
}
export const enclosingNodeOrSelfWithNodeNameInArray = (initialNode: Node, nameArray: string[]): Node|null => {
let node: (Node|null)|Node = initialNode;
for (; node && node !== initialNode.ownerDocument; node = node.parentNodeOrShadowHost()) {
for (let i = 0; i < nameArray.length; ++i) {
if (node.nodeName.toLowerCase() === nameArray[i].toLowerCase()) {
return node;
}
}
}
return null;
};
export const enclosingNodeOrSelfWithNodeName = function(node: Node, nodeName: string): Node|null {
return enclosingNodeOrSelfWithNodeNameInArray(node, [nodeName]);
};
export const deepElementFromPoint = (document: Document|ShadowRoot|null|undefined, x: number, y: number): Node|null => {
let container: (ShadowRoot|null)|(Document | ShadowRoot | null | undefined) = document;
let node: Element|null = null;
while (container) {
const innerNode = container.elementFromPoint(x, y);
if (!innerNode || node === innerNode) {
break;
}
node = innerNode;
container = node.shadowRoot;
}
return node;
};
export const deepElementFromEvent = (ev: Event): Node|null => {
const event = (ev as MouseEvent);
// Some synthetic events have zero coordinates which lead to a wrong element. Better return nothing in this case.
if (!event.which && !event.pageX && !event.pageY && !event.clientX && !event.clientY && !event.movementX &&
!event.movementY) {
return null;
}
const root = event.target && (event.target as Element).getComponentRoot();
return root ? deepElementFromPoint((root as Document | ShadowRoot), event.pageX, event.pageY) : null;
};
const registeredRenderers: RendererRegistration[] = [];
export function registerRenderer(registration: RendererRegistration): void {
registeredRenderers.push(registration);
}
export function getApplicableRegisteredRenderers(object: Object): RendererRegistration[] {
return registeredRenderers.filter(isRendererApplicableToContextTypes);
function isRendererApplicableToContextTypes(rendererRegistration: RendererRegistration): boolean {
if (!rendererRegistration.contextTypes) {
return true;
}
for (const contextType of rendererRegistration.contextTypes()) {
// TODO(crbug.com/1172300) Ignored during the jsdoc to ts migration
// @ts-expect-error
if (object instanceof contextType) {
return true;
}
}
return false;
}
}
export interface RendererRegistration {
loadRenderer: () => Promise<Renderer>;
contextTypes: () => Array<unknown>;
}