reblend-ui
Version:
Utilities for creating robust overlay components
59 lines (48 loc) • 1.65 kB
text/typescript
import ownerDocument from 'dom-helpers/ownerDocument';
import canUseDOM from 'dom-helpers/canUseDOM';
import { useState, useEffect } from 'reblendjs';
import useWindow from './useWindow';
import { VirtualElement } from './usePopper';
export type DOMContainer<T extends HTMLElement | VirtualElement = HTMLElement> =
| T
| Reblend.RefObject<T | null>
| null
| (() => T | Reblend.RefObject<T | null> | null);
export const resolveContainerRef = <T extends HTMLElement | VirtualElement>(
ref: DOMContainer<T> | undefined,
document?: Document,
): T | HTMLBodyElement | null => {
if (!canUseDOM) return null;
if (ref == null) return (document || ownerDocument()).body as HTMLBodyElement;
if (typeof ref === 'function') ref = ref();
if (ref && 'current' in ref) ref = ref.current;
if (ref && ('nodeType' in ref || 'getBoundingClientRect' in ref)) return ref;
return null;
};
export default function useWaitForDOMRef<
T extends HTMLElement | VirtualElement = HTMLElement,
>(
ref: DOMContainer<T> | undefined,
onResolved?: (element: T | HTMLBodyElement) => void,
) {
const window = useWindow();
const [resolvedRef, setRef] = useState(() =>
resolveContainerRef(ref, window?.document),
);
if (!resolvedRef) {
const earlyRef = resolveContainerRef(ref);
if (earlyRef) setRef(earlyRef);
}
useEffect(() => {
if (onResolved && resolvedRef) {
onResolved(resolvedRef);
}
}, [onResolved, resolvedRef]);
useEffect(() => {
const nextRef = resolveContainerRef(ref);
if (nextRef !== resolvedRef) {
setRef(nextRef);
}
}, [ref, resolvedRef]);
return resolvedRef;
}