Files
notesnook/packages/common/src/utils/debounce.ts
2023-06-17 13:36:01 +05:00

72 lines
2.2 KiB
TypeScript

/*
This file is part of the Notesnook project (https://notesnook.com/)
Copyright (C) 2023 Streetwriters (Private) Limited
This program is free software: you can redistribute it and/or modify
it under the terms of the GNU General Public License as published by
the Free Software Foundation, either version 3 of the License, or
(at your option) any later version.
This program is distributed in the hope that it will be useful,
but WITHOUT ANY WARRANTY; without even the implied warranty of
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
GNU General Public License for more details.
You should have received a copy of the GNU General Public License
along with this program. If not, see <http://www.gnu.org/licenses/>.
*/
/* eslint-disable @typescript-eslint/no-explicit-any */
interface DebouncedFunction<
Args extends any[],
F extends (...args: Args) => any
> {
(this: ThisParameterType<F>, ...args: Args & Parameters<F>): void;
}
interface DebouncedFunctionWithId<
Args extends any[],
F extends (...args: Args) => any
> {
(
this: ThisParameterType<F>,
id: string | number,
...args: Args & Parameters<F>
): void;
}
export function debounce<Args extends any[], F extends (...args: Args) => void>(
func: F,
waitFor: number
): DebouncedFunction<Args, F> {
let timeout: number | null;
return (...args: Args) => {
if (timeout) clearTimeout(timeout);
timeout = setTimeout(() => func(...args), waitFor) as unknown as number;
};
}
export function debounceWithId<
Args extends any[],
F extends (...args: Args) => void
>(func: F, waitFor: number): DebouncedFunctionWithId<Args, F> {
let timeout: number | null;
let debounceId: string | number | null = null;
return (id: string | number, ...args: Parameters<F>) => {
if (timeout && id === debounceId) clearTimeout(timeout);
debounceId = id;
timeout = setTimeout(() => {
func(...args);
}, waitFor) as unknown as number;
};
}
const DEBOUNCE_TIMEOUTS: Record<string, NodeJS.Timeout> = {};
export function inlineDebounce(id: string, func: () => void, waitFor: number) {
clearTimeout(DEBOUNCE_TIMEOUTS[id]);
DEBOUNCE_TIMEOUTS[id] = setTimeout(func, waitFor);
}