mirror of
https://github.com/BookStackApp/BookStack.git
synced 2025-04-29 06:09:56 +00:00
259 lines
No EOL
8 KiB
TypeScript
259 lines
No EOL
8 KiB
TypeScript
import {cyrb53} from "./util";
|
|
|
|
/**
|
|
* Check if the given param is a HTMLElement
|
|
*/
|
|
export function isHTMLElement(el: any): el is HTMLElement {
|
|
return el instanceof HTMLElement;
|
|
}
|
|
|
|
/**
|
|
* Create a new element with the given attrs and children.
|
|
* Children can be a string for text nodes or other elements.
|
|
*/
|
|
export function elem(tagName: string, attrs: Record<string, string> = {}, children: Element[]|string[] = []): HTMLElement {
|
|
const el = document.createElement(tagName);
|
|
|
|
for (const [key, val] of Object.entries(attrs)) {
|
|
if (val === null) {
|
|
el.removeAttribute(key);
|
|
} else {
|
|
el.setAttribute(key, val);
|
|
}
|
|
}
|
|
|
|
for (const child of children) {
|
|
if (typeof child === 'string') {
|
|
el.append(document.createTextNode(child));
|
|
} else {
|
|
el.append(child);
|
|
}
|
|
}
|
|
|
|
return el;
|
|
}
|
|
|
|
/**
|
|
* Run the given callback against each element that matches the given selector.
|
|
*/
|
|
export function forEach(selector: string, callback: (el: Element) => any) {
|
|
const elements = document.querySelectorAll(selector);
|
|
for (const element of elements) {
|
|
callback(element);
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Helper to listen to multiple DOM events
|
|
*/
|
|
export function onEvents(listenerElement: Element|null, events: string[], callback: (e: Event) => any): void {
|
|
if (listenerElement) {
|
|
for (const eventName of events) {
|
|
listenerElement.addEventListener(eventName, callback);
|
|
}
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Helper to run an action when an element is selected.
|
|
* A "select" is made to be accessible, So can be a click, space-press or enter-press.
|
|
*/
|
|
export function onSelect(elements: HTMLElement|HTMLElement[], callback: (e: Event) => any): void {
|
|
if (!Array.isArray(elements)) {
|
|
elements = [elements];
|
|
}
|
|
|
|
for (const listenerElement of elements) {
|
|
listenerElement.addEventListener('click', callback);
|
|
listenerElement.addEventListener('keydown', event => {
|
|
if (event.key === 'Enter' || event.key === ' ') {
|
|
event.preventDefault();
|
|
callback(event);
|
|
}
|
|
});
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Listen to key press on the given element(s).
|
|
*/
|
|
function onKeyPress(key: string, elements: HTMLElement|HTMLElement[], callback: (e: KeyboardEvent) => any): void {
|
|
if (!Array.isArray(elements)) {
|
|
elements = [elements];
|
|
}
|
|
|
|
const listener = (event: KeyboardEvent) => {
|
|
if (event.key === key) {
|
|
callback(event);
|
|
}
|
|
};
|
|
|
|
elements.forEach(e => e.addEventListener('keydown', listener));
|
|
}
|
|
|
|
/**
|
|
* Listen to enter press on the given element(s).
|
|
*/
|
|
export function onEnterPress(elements: HTMLElement|HTMLElement[], callback: (e: KeyboardEvent) => any): void {
|
|
onKeyPress('Enter', elements, callback);
|
|
}
|
|
|
|
/**
|
|
* Listen to escape press on the given element(s).
|
|
*/
|
|
export function onEscapePress(elements: HTMLElement|HTMLElement[], callback: (e: KeyboardEvent) => any): void {
|
|
onKeyPress('Escape', elements, callback);
|
|
}
|
|
|
|
/**
|
|
* Set a listener on an element for an event emitted by a child
|
|
* matching the given childSelector param.
|
|
* Used in a similar fashion to jQuery's $('listener').on('eventName', 'childSelector', callback)
|
|
*/
|
|
export function onChildEvent(
|
|
listenerElement: HTMLElement,
|
|
childSelector: string,
|
|
eventName: string,
|
|
callback: (this: HTMLElement, e: Event, child: HTMLElement) => any
|
|
): void {
|
|
listenerElement.addEventListener(eventName, (event: Event) => {
|
|
const matchingChild = (event.target as HTMLElement|null)?.closest(childSelector) as HTMLElement;
|
|
if (matchingChild) {
|
|
callback.call(matchingChild, event, matchingChild);
|
|
}
|
|
});
|
|
}
|
|
|
|
/**
|
|
* Look for elements that match the given selector and contain the given text.
|
|
* Is case-insensitive and returns the first result or null if nothing is found.
|
|
*/
|
|
export function findText(selector: string, text: string): Element|null {
|
|
const elements = document.querySelectorAll(selector);
|
|
text = text.toLowerCase();
|
|
for (const element of elements) {
|
|
if ((element.textContent || '').toLowerCase().includes(text) && isHTMLElement(element)) {
|
|
return element;
|
|
}
|
|
}
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Show a loading indicator in the given element.
|
|
* This will effectively clear the element.
|
|
*/
|
|
export function showLoading(element: HTMLElement): void {
|
|
element.innerHTML = '<div class="loading-container"><div></div><div></div><div></div></div>';
|
|
}
|
|
|
|
/**
|
|
* Get a loading element indicator element.
|
|
*/
|
|
export function getLoading(): HTMLElement {
|
|
const wrap = document.createElement('div');
|
|
wrap.classList.add('loading-container');
|
|
wrap.innerHTML = '<div></div><div></div><div></div>';
|
|
return wrap;
|
|
}
|
|
|
|
/**
|
|
* Remove any loading indicators within the given element.
|
|
*/
|
|
export function removeLoading(element: HTMLElement): void {
|
|
const loadingEls = element.querySelectorAll('.loading-container');
|
|
for (const el of loadingEls) {
|
|
el.remove();
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Convert the given html data into a live DOM element.
|
|
* Initiates any components defined in the data.
|
|
*/
|
|
export function htmlToDom(html: string): HTMLElement {
|
|
const wrap = document.createElement('div');
|
|
wrap.innerHTML = html;
|
|
window.$components.init(wrap);
|
|
const firstChild = wrap.children[0];
|
|
if (!isHTMLElement(firstChild)) {
|
|
throw new Error('Could not find child HTMLElement when creating DOM element from HTML');
|
|
}
|
|
|
|
return firstChild;
|
|
}
|
|
|
|
/**
|
|
* For the given node and offset, return an adjusted offset that's relative to the given parent element.
|
|
*/
|
|
export function normalizeNodeTextOffsetToParent(node: Node, offset: number, parentElement: HTMLElement): number {
|
|
if (!parentElement.contains(node)) {
|
|
throw new Error('ParentElement must be a prent of element');
|
|
}
|
|
|
|
let normalizedOffset = offset;
|
|
let currentNode: Node|null = node.nodeType === Node.TEXT_NODE ?
|
|
node : node.childNodes[offset];
|
|
|
|
while (currentNode !== parentElement && currentNode) {
|
|
if (currentNode.previousSibling) {
|
|
currentNode = currentNode.previousSibling;
|
|
normalizedOffset += (currentNode.textContent?.length || 0);
|
|
} else {
|
|
currentNode = currentNode.parentNode;
|
|
}
|
|
}
|
|
|
|
return normalizedOffset;
|
|
}
|
|
|
|
/**
|
|
* Find the target child node and adjusted offset based on a parent node and text offset.
|
|
* Returns null if offset not found within the given parent node.
|
|
*/
|
|
export function findTargetNodeAndOffset(parentNode: HTMLElement, offset: number): ({node: Node, offset: number}|null) {
|
|
if (offset === 0) {
|
|
return { node: parentNode, offset: 0 };
|
|
}
|
|
|
|
let currentOffset = 0;
|
|
let currentNode = null;
|
|
|
|
for (let i = 0; i < parentNode.childNodes.length; i++) {
|
|
currentNode = parentNode.childNodes[i];
|
|
|
|
if (currentNode.nodeType === Node.TEXT_NODE) {
|
|
// For text nodes, count the length of their content
|
|
// Returns if within range
|
|
const textLength = currentNode.textContent.length;
|
|
if (currentOffset + textLength >= offset) {
|
|
return {
|
|
node: currentNode,
|
|
offset: offset - currentOffset
|
|
};
|
|
}
|
|
|
|
currentOffset += textLength;
|
|
} else if (currentNode.nodeType === Node.ELEMENT_NODE) {
|
|
// Otherwise, if an element, track the text length and search within
|
|
// if in range for the target offset
|
|
const elementTextLength = currentNode.textContent.length;
|
|
if (currentOffset + elementTextLength >= offset) {
|
|
return findTargetNodeAndOffset(currentNode, offset - currentOffset);
|
|
}
|
|
|
|
currentOffset += elementTextLength;
|
|
}
|
|
}
|
|
|
|
// Return null if not found within range
|
|
return null;
|
|
}
|
|
|
|
/**
|
|
* Create a hash for the given HTML element.
|
|
*/
|
|
export function hashElement(element: HTMLElement): string {
|
|
const normalisedElemHtml = element.outerHTML.replace(/\s{2,}/g, '');
|
|
return cyrb53(normalisedElemHtml);
|
|
} |