You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
55 lines
1.5 KiB
55 lines
1.5 KiB
3 years ago
|
import { useState, useCallback, useLayoutEffect } from 'react';
|
||
|
import debounce from './utils/debounce';
|
||
|
|
||
|
function useDimensions(liveMeasure = true, delay = 250, initialDimensions = {}) {
|
||
|
const [dimensions, setDimensions] = useState(initialDimensions);
|
||
|
const [node, setNode] = useState(null);
|
||
|
|
||
|
const ref = useCallback((newNode) => {
|
||
|
setNode(newNode);
|
||
|
}, []);
|
||
|
|
||
|
useLayoutEffect(() => {
|
||
|
// need ref to continue
|
||
|
if (!node) {
|
||
|
return;
|
||
|
}
|
||
|
|
||
|
const measure = () => {
|
||
|
window.requestAnimationFrame(() => {
|
||
|
const newDimensions = node.getBoundingClientRect();
|
||
|
setDimensions(newDimensions);
|
||
|
});
|
||
|
};
|
||
|
// invoke measure right away
|
||
|
measure();
|
||
|
|
||
|
if (liveMeasure) {
|
||
|
const debounceMeasure = debounce(measure, delay);
|
||
|
|
||
|
if ('ResizeObserver' in window) {
|
||
|
const resizeObserver = new ResizeObserver(debounceMeasure);
|
||
|
resizeObserver.observe(node);
|
||
|
window.addEventListener('scroll', debounceMeasure);
|
||
|
|
||
|
return () => {
|
||
|
resizeObserver.disconnect();
|
||
|
window.removeEventListener('scroll', debounceMeasure);
|
||
|
};
|
||
|
}
|
||
|
window.addEventListener('resize', debounceMeasure);
|
||
|
window.addEventListener('scroll', debounceMeasure);
|
||
|
|
||
|
return () => {
|
||
|
window.removeEventListener('resize', debounceMeasure);
|
||
|
window.removeEventListener('scroll', debounceMeasure);
|
||
|
};
|
||
|
}
|
||
|
}, [node, liveMeasure, delay]);
|
||
|
|
||
|
return [ref, dimensions, node];
|
||
|
}
|
||
|
|
||
|
export default useDimensions;
|
||
|
|