blob: 94bd9b1d4795f3b760981ce350c6b70fbadd32fa [file] [log] [blame] [edit]
// Copyright (C) 2023 The Android Open Source Project
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
import {Vector2D} from './geom';
export type CSSCursor =
| 'alias'
| 'all-scroll'
| 'auto'
| 'cell'
| 'context-menu'
| 'col-resize'
| 'copy'
| 'crosshair'
| 'default'
| 'e-resize'
| 'ew-resize'
| 'grab'
| 'grabbing'
| 'help'
| 'move'
| 'n-resize'
| 'ne-resize'
| 'nesw-resize'
| 'ns-resize'
| 'nw-resize'
| 'nwse-resize'
| 'no-drop'
| 'none'
| 'not-allowed'
| 'pointer'
| 'progress'
| 'row-resize'
| 's-resize'
| 'se-resize'
| 'sw-resize'
| 'text'
| 'vertical-text'
| 'w-resize'
| 'wait'
| 'zoom-in'
| 'zoom-out';
// Check whether a DOM element contains another, or whether they're the same
export function isOrContains(container: Element, target: Element): boolean {
return container === target || container.contains(target);
}
// Find a DOM element with a given "ref" attribute
export function findRef(root: Element, ref: string): Element | null {
const query = `[ref=${ref}]`;
if (root.matches(query)) {
return root;
} else {
return root.querySelector(query);
}
}
// Safely cast an Element to an HTMLElement.
// Throws if the element is not an HTMLElement.
export function toHTMLElement(el: Element): HTMLElement {
if (!(el instanceof HTMLElement)) {
throw new Error('Element is not an HTMLElement');
}
return el as HTMLElement;
}
// Return true if EventTarget is or is inside an editable element.
// Editable elements incluce: <input type="text">, <textarea>, or elements with
// the |contenteditable| attribute set.
export function elementIsEditable(target: EventTarget | null): boolean {
if (target === null) {
return false;
}
if (!(target instanceof Element)) {
return false;
}
const editable = target.closest('input, textarea, [contenteditable=true]');
if (editable === null) {
return false;
}
if (editable instanceof HTMLInputElement) {
if (['radio', 'checkbox', 'button'].includes(editable.type)) {
return false;
}
}
return true;
}
// Returns the mouse pointer's position relative to |e.currentTarget| for a
// given |MouseEvent|.
// Similar to |offsetX|, |offsetY| but for |currentTarget| rather than |target|.
// If the event has no currentTarget or it is not an element, offsetX & offsetY
// are returned instead.
export function currentTargetOffset(e: MouseEvent): Vector2D {
if (e.currentTarget === e.target) {
return new Vector2D({x: e.offsetX, y: e.offsetY});
}
if (e.currentTarget && e.currentTarget instanceof Element) {
const rect = e.currentTarget.getBoundingClientRect();
const offsetX = e.clientX - rect.left;
const offsetY = e.clientY - rect.top;
return new Vector2D({x: offsetX, y: offsetY});
}
return new Vector2D({x: e.offsetX, y: e.offsetY});
}
// Adds an event listener to a DOM element, returning a disposable to remove it.
export function bindEventListener<K extends keyof HTMLElementEventMap>(
element: EventTarget,
event: K,
handler: (event: HTMLElementEventMap[K]) => void,
): Disposable {
element.addEventListener(event, handler as EventListener);
return {
[Symbol.dispose]() {
element.removeEventListener(event, handler as EventListener);
},
};
}