UNPKG

reblend-ui

Version:

Utilities for creating robust overlay components

59 lines (48 loc) 1.65 kB
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; }