Resize Observer is the most preferred way to detect element size changes and makes changing content/styles or any custom action on a page based on resizing.
--MDN
Resize observer can be implemented in 3 easy steps:
const resizeHandler = (entries) => {
entries.forEach((entry) => {
const isSmall = entry.contentRect.width < 6;
entry.target.style.backgroundColor = isSmall ? "blue" : "orange";
});
};
const observer = new ResizeObserver(resizeHandler);
ResizeObserver
takes one callback function, triggers any time an element you are observing changes size it will trigger the function passed to ResizeObserver.
Callback function has one parameter of type array - entries
. This array just lists all the elements we are observing that have had their size change.
Each element may contain ``contentRect` property.
his property contains information such as the width, height, top, left, bottom, right, etc. of our element. Finally, we are using the target property of our entry to get the current element that is being observed.
<div id="resize-target"></div>
<div id="input" contenteditable></div>
input
and length of content reaches 50 or beyong then size of resize-target
should get increased to 400px
and background color should get changed to red.<!DOCTYPE html>
<html lang="en">
<head>
<meta charset="UTF-8">
<meta name="viewport" content="width=device-width, initial-scale=1.0">
<title>Resize observer</title>
</head>
<style>
#resize-target {
background-color: blue;
}
.container {
display: flex;
}
#input {
background-color: antiquewhite;
width: 20%;
font-size: xx-large;
}
</style>
<body>
<div class="container">
<div id="resize-target"></div>
<div id="input" contenteditable></div>
</div>
<script>
const input = document.getElementById('input');
const targetDiv = document.getElementById('resize-target');
const resizeObserver = new ResizeObserver(entries => {
// Iterate through each entry in the entries array
entries.forEach(entry => {
// Get the contentRect property from the entry object
const { contentRect, target } = entry;
console.log(target.innerText.length);
if (target.innerText.length >= 50) {
targetDiv.style.width = '400px';
targetDiv.style.backgroundColor = 'red';
} else {
targetDiv.style.width = '100px';
}
});
});
// Observe the root element of the document
resizeObserver.observe(input);
</script>
</body>
</html>
Like other observer APIs, ResizeObserver
has options property but unlike the other observer APIs, Resize Observer has very limited options you can configure. It has box
property allows you to change which box model is used to determine size changes.
box
property has content-box
as default value.box
has following property:content-box
: includes changes in the actual content of the element.device-pixel-content-box
: is similar to the content-box option but it takes into account the actual pixel size of the device it is rendering too.border-box
: option takes into account things like border and padding changes.const observer = new ResizeObserver(changeColor);
observer.observe(document.getElementById("input"), { box: "border-box" });
The blockSize property defines the height of the element while the inlineSize defines the width. If the writing mode of your document is vertical, though, the blockSize will define the width while the inlineSize will define the height.
The disconnect() method of the ResizeObserver interface unobserves all observed Element.
btn.addEventListener("click", () => {
resizeObserver.disconnect();
});
The unobserve() method of the ResizeObserver interface ends the observing of a specified Element.
observer.unobserve(entry.target)
useResizeObserver
.import { useEffect, useRef } from 'react';
function useResizeObserver<T extends HTMLElement>(
callback: (target: T, entry: ResizeObserverEntry) => void
) {
const ref = useRef<T>(null)
useEffect(() => {
const element = ref?.current;
if (!element) {
return;
}
const observer = new ResizeObserver((entries) => {
callback(element, entries[0]);
});
observer.observe(element);
return () => {
observer.disconnect();
};
}, [callback, ref]);
return ref
}
export default useResizeObserver;
"use client";
import { useCallback, useState } from "react";
import useResizeObserver from "../hooks/useResizeObserver";
const ResizeObserverPage = () => {
const [size, setSize] = useState<string>();
const [color, setColor] = useState<string>();
const onResize = useCallback((target: HTMLDivElement, entry: ResizeObserverEntry) => {
// Handle the resize event
const { contentRect } = entry;
if (target.innerText.length >= 50) {
setSize('400px');
setColor('red');
} else {
setSize('100px');
}
}, []);
const ref = useResizeObserver(onResize);
return (
<div className="container">
<div id="resize-target" style={{width: size, backgroundColor: color}} ></div>
<div id="input" ref={ref} contentEditable></div>
</div>
);
};
export default ResizeObserverPage;
We have CSS Container Queries(https://techcookies.com/post/CSS-container-queries) as well handling similar scenarios but container queries can work for child elements only.
Whereas, the Resize Observer is a very simple API to understand and can work for all the use cases and it can be incredibly powerful in specific situations.