webui/bridge/utils.ts

57 lines
2.1 KiB
TypeScript
Raw Normal View History

/*
2024-07-11 10:10:29 -04:00
https://webui.me
https://github.com/webui-dev/webui
2025-01-30 15:52:24 -05:00
Copyright (c) 2020-2025 Hassan Draga.
Licensed under MIT License.
All rights reserved.
2024-07-11 10:10:29 -04:00
Canada.
File: WebUI Bridge Utils
Copyright (c) 2024 Oculi Julien.
*/
/**
* Allows you to automatically bind an event listener to newly added
* elements that match a specific selector within a given root element.
* Track dom update to rebind event listeners automatically.
*
* @param {HTMLElement} root - The root element to observe for changes.
* @param {string} targetSelector - Query selector matching elements that you want to bind the event listener to.
* @param {K} type - Type of event listener to bind (same as for addEventListener).
* @param listener - Event listener to bind (same as for addEventListener).
* @param {boolean | AddEventListenerOptions} [options] - Event listener options (same as for addEventListener).
* @returns the used observer to allow disconnect.
*/
2023-10-08 03:01:02 +02:00
export function addRefreshableEventListener<K extends keyof HTMLElementEventMap>(
root: HTMLElement,
targetSelector: string,
type: K,
listener: (this: HTMLElement, ev: HTMLElementEventMap[K]) => unknown,
2023-10-08 03:01:02 +02:00
options?: boolean | AddEventListenerOptions,
) {
function rebindListener(mutations: MutationRecord[]) {
for (const mutation of mutations) {
for (const node of mutation.addedNodes) {
2023-10-08 03:01:02 +02:00
if (!(node instanceof HTMLElement)) return; // Target only html elements
if (node.matches(targetSelector)) {
// Bind event on added nodes
2023-10-08 03:01:02 +02:00
node.addEventListener<K>(type, listener, options);
}
for (const child of node.querySelectorAll(targetSelector)) {
2023-10-08 03:01:02 +02:00
if (!(child instanceof HTMLElement)) continue; //Target only html elements
child.addEventListener<K>(type, listener, options);
}
}
}
}
2023-10-08 03:01:02 +02:00
const observer = new MutationObserver(rebindListener); //Set mutation observer callback
observer.observe(root, { subtree: true, childList: true }); // Observe root element and all his children
return observer; // Allow user to stop observer for performance issues
}
/**
* Async function constructor
*/
2023-10-08 03:01:02 +02:00
export const AsyncFunction = async function () {}.constructor;