import * as React from "react"; import { useCallback, useState } from "react"; import { debounce } from "throttle-debounce"; // returns a callback ref, a ref object (that is set from the callback), and the width // pass debounceMs of null to not debounce export function useDimensionsWithCallbackRef( debounceMs: number = null ): [(node: T) => void, React.RefObject, DOMRectReadOnly] { const [domRect, setDomRect] = useState(null); const [htmlElem, setHtmlElem] = useState(null); const rszObjRef = React.useRef(null); const oldHtmlElem = React.useRef(null); const ref = React.useRef(null); const refCallback = useCallback((node: T) => { setHtmlElem(node); ref.current = node; }, []); const setDomRectDebounced = React.useCallback(debounceMs == null ? setDomRect : debounce(debounceMs, setDomRect), [ debounceMs, setDomRect, ]); React.useEffect(() => { if (!rszObjRef.current) { rszObjRef.current = new ResizeObserver((entries) => { for (const entry of entries) { if (domRect == null) { setDomRect(entry.contentRect); } else { setDomRectDebounced(entry.contentRect); } } }); } if (htmlElem) { rszObjRef.current.observe(htmlElem); oldHtmlElem.current = htmlElem; } return () => { if (oldHtmlElem.current) { rszObjRef.current?.unobserve(oldHtmlElem.current); oldHtmlElem.current = null; } }; }, [htmlElem]); React.useEffect(() => { return () => { rszObjRef.current?.disconnect(); }; }, []); return [refCallback, ref, domRect]; } // will not react to ref changes // pass debounceMs of null to not debounce export function useDimensionsWithExistingRef( ref: React.RefObject, debounceMs: number = null ): DOMRectReadOnly { const [domRect, setDomRect] = useState(null); const rszObjRef = React.useRef(null); const oldHtmlElem = React.useRef(null); const setDomRectDebounced = React.useCallback(debounceMs == null ? setDomRect : debounce(debounceMs, setDomRect), [ debounceMs, setDomRect, ]); React.useEffect(() => { if (!rszObjRef.current) { rszObjRef.current = new ResizeObserver((entries) => { for (const entry of entries) { if (domRect == null) { setDomRect(entry.contentRect); } else { setDomRectDebounced(entry.contentRect); } } }); } if (ref.current) { rszObjRef.current.observe(ref.current); oldHtmlElem.current = ref.current; } return () => { if (oldHtmlElem.current) { rszObjRef.current?.unobserve(oldHtmlElem.current); oldHtmlElem.current = null; } }; }, [ref.current]); React.useEffect(() => { return () => { rszObjRef.current?.disconnect(); }; }, []); if (ref.current != null) { return ref.current.getBoundingClientRect(); } return null; }