2024-09-30 21:19:29 +02:00
|
|
|
import * as React from "react";
|
|
|
|
import { useCallback, useState } from "react";
|
2024-09-04 03:43:59 +02:00
|
|
|
import { debounce } from "throttle-debounce";
|
2024-08-23 09:18:49 +02:00
|
|
|
|
2024-09-30 21:19:29 +02:00
|
|
|
// 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<T extends HTMLElement>(
|
|
|
|
debounceMs: number = null
|
|
|
|
): [(node: T) => void, React.RefObject<T>, DOMRectReadOnly] {
|
|
|
|
const [domRect, setDomRect] = useState<DOMRectReadOnly>(null);
|
|
|
|
const [htmlElem, setHtmlElem] = useState<T>(null);
|
|
|
|
const rszObjRef = React.useRef<ResizeObserver>(null);
|
|
|
|
const oldHtmlElem = React.useRef<T>(null);
|
|
|
|
const ref = React.useRef<T>(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);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
});
|
2024-09-04 03:43:59 +02:00
|
|
|
}
|
2024-09-30 21:19:29 +02:00
|
|
|
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();
|
|
|
|
};
|
2024-09-04 03:43:59 +02:00
|
|
|
}, []);
|
2024-09-30 21:19:29 +02:00
|
|
|
return [refCallback, ref, domRect];
|
|
|
|
}
|
2024-08-23 09:18:49 +02:00
|
|
|
|
2024-09-30 21:19:29 +02:00
|
|
|
// will not react to ref changes
|
|
|
|
// pass debounceMs of null to not debounce
|
|
|
|
export function useDimensionsWithExistingRef<T extends HTMLElement>(
|
|
|
|
ref: React.RefObject<T>,
|
|
|
|
debounceMs: number = null
|
|
|
|
): DOMRectReadOnly {
|
|
|
|
const [domRect, setDomRect] = useState<DOMRectReadOnly>(null);
|
|
|
|
const rszObjRef = React.useRef<ResizeObserver>(null);
|
|
|
|
const oldHtmlElem = React.useRef<T>(null);
|
|
|
|
const setDomRectDebounced = React.useCallback(debounceMs == null ? setDomRect : debounce(debounceMs, setDomRect), [
|
|
|
|
debounceMs,
|
|
|
|
setDomRect,
|
2024-08-23 09:18:49 +02:00
|
|
|
]);
|
2024-09-30 21:19:29 +02:00
|
|
|
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;
|
|
|
|
}
|