portal-www
Version:
Nova Portal Website. Based on Next starter by Ueno
57 lines (42 loc) • 1.48 kB
text/typescript
// useDebounce and useTimeoutFn from https://github.com/streamich/react-use
import { DependencyList, useCallback, useEffect, useRef } from 'react';
export type UseTimeoutFnReturn = [() => boolean | null, () => void, () => void];
type CallbackFunctionVariadic = (...args: unknown[]) => void;
export function useTimeoutFn(fn: CallbackFunctionVariadic, ms = 0): UseTimeoutFnReturn {
const ready = useRef<boolean | null>(false);
const timeout = useRef<ReturnType<typeof setTimeout>>();
const callback = useRef(fn);
const isReady = useCallback(() => ready.current, []);
const set = useCallback(() => {
ready.current = false;
timeout.current && clearTimeout(timeout.current);
timeout.current = setTimeout(() => {
ready.current = true;
callback.current();
}, ms);
}, [ms]);
const clear = useCallback(() => {
ready.current = null;
timeout.current && clearTimeout(timeout.current);
}, []);
// update ref when function changes
useEffect(() => {
callback.current = fn;
}, [fn]);
// set on mount, clear on unmount
useEffect(() => {
set();
return clear;
}, [ms]);
return [isReady, clear, set];
}
export type UseDebounceReturn = [() => boolean | null, () => void];
export function useDebounce(
fn: CallbackFunctionVariadic,
ms = 0,
deps: DependencyList = [],
): UseDebounceReturn {
const [isReady, cancel, reset] = useTimeoutFn(fn, ms);
useEffect(reset, deps);
return [isReady, cancel];
}