diff --git a/src/hooks/useHasFocus/useHasFocus.test.tsx b/src/hooks/useHasFocus/useHasFocus.test.tsx
index f5bfed7..d780188 100644
--- a/src/hooks/useHasFocus/useHasFocus.test.tsx
+++ b/src/hooks/useHasFocus/useHasFocus.test.tsx
@@ -1,6 +1,6 @@
/* eslint-disable react/jsx-no-literals */
/* eslint-disable react/no-multi-comp */
-import { act, render, waitFor } from '@testing-library/react';
+import { act, render } from '@testing-library/react';
import { useRef, type ReactElement } from 'react';
import { useHasFocus } from './useHasFocus.js';
diff --git a/src/index.ts b/src/index.ts
index 4670f9f..402a77a 100644
--- a/src/index.ts
+++ b/src/index.ts
@@ -18,3 +18,4 @@ export * from './hooks/useToggle/useToggle.js';
export * from './hooks/useUnmount/useUnmount.js';
export * from './hooks/useWindowEventListener/useWindowEventListener.js';
export * from './utils/arrayRef/arrayRef.js';
+export * from './utils/objectFit/objectFit.js';
diff --git a/src/utils/objectFit/objectFit.mdx b/src/utils/objectFit/objectFit.mdx
new file mode 100644
index 0000000..1a31acf
--- /dev/null
+++ b/src/utils/objectFit/objectFit.mdx
@@ -0,0 +1,155 @@
+import { Meta } from '@storybook/blocks';
+
+
+
+# objectFit
+
+This util mimics the CSS property `object-fit` for all HTML elements;
+
+It exports two reusable methods: `contain` and `cover`. Given the sizes of an parent element and its
+child element: Contain returns the size to be applied to the element to let it fits its parent and
+keeping its apect ratio. Cover returns the size to be applied to the element to let it fill its
+parent, keeping its aspect ratio, most likely overflowing the parent element.
+
+If the sizes or aspect ratio are initially known, it's better to use values instead of retrieving
+sizes from an image because its faster from a performance perspective.
+
+## Reference
+
+```ts
+function objectFit(fit: 'contain' | 'cover') {
+ return (
+ parentWidth: number,
+ parentHeight: number,
+ childWidth: number,
+ childHeight: number,
+ ): { x: number; y: number; width: number; height: number; scale: number; cssText: string } => {
+ if ([parentWidth, parentHeight, childWidth, childHeight].some((value) => value <= 0)) {
+ throw new Error(`All arguments should have a positive value`);
+ }
+
+ const mathMethod = fit === 'contain' ? Math.min : Math.max;
+ const scale = mathMethod(parentWidth / childWidth, parentHeight / childHeight);
+ const width = Math.ceil(childWidth * scale);
+ const height = Math.ceil(childHeight * scale);
+ const x = Math.trunc((parentWidth - width) * 0.5);
+ const y = Math.trunc((parentHeight - height) * 0.5);
+
+ return {
+ x,
+ y,
+ width,
+ height,
+ scale,
+ cssText: `left:${x}px;top:${y}px;width:${width}px;height:${height}px;`,
+ };
+ };
+}
+
+export const contain = objectFit('contain');
+export const cover = objectFit('cover');
+```
+
+### Parameters
+
+- parentWidth: number
+- parentHeight: number
+- childWidth: number
+- childHeight: number
+
+### Returns
+
+An object containing:
+
+- x: number
+- y: number
+- width: number
+- height: number
+- scale: number
+- cssText: string (easily add CSS values to child element)
+
+## Usage
+
+Contain:
+
+With the contain method you can use both position absolute and relative on the child element.
+Relative can be useful if you want to position elements inside absolute to the parent.
+
+```tsx
+import { contain } from './objectFit.js';
+
+export function Contain(): ReactElement {
+ const parentRef = useRef(null);
+ const childRef = useRef(null);
+
+ const onResize = useCallback(() => {
+ if (!parentRef.current || !childRef.current) {
+ return;
+ }
+
+ const objectFit = contain(parentRef.current.offsetWidth, parentRef.current.offsetHeight, 1, 1);
+
+ childRef.current.style.cssText += objectFit.cssText;
+ }, [parentRef, childRef]);
+
+ useResizeObserver(parentRef, onResize);
+
+ return (
+
+ );
+}
+```
+
+Cover:
+
+With contain you need to use position absolute to position the child.
+
+```tsx
+import { cover } from './objectFit.js';
+
+export function Cover(): ReactElement {
+ const parentRef = useRef(null);
+ const childRef = useRef(null);
+
+ const onResize = useCallback(() => {
+ if (!parentRef.current || !childRef.current) {
+ return;
+ }
+
+ const objectFit = cover(
+ parentRef.current.offsetWidth,
+ parentRef.current.offsetHeight,
+ 1920,
+ 1080,
+ );
+
+ childRef.current.style.cssText += objectFit.cssText;
+ }, [parentRef, childRef]);
+
+ useResizeObserver(parentRef, onResize);
+
+ return (
+
+ );
+}
+```
diff --git a/src/utils/objectFit/objectFit.stories.tsx b/src/utils/objectFit/objectFit.stories.tsx
new file mode 100644
index 0000000..86a8f00
--- /dev/null
+++ b/src/utils/objectFit/objectFit.stories.tsx
@@ -0,0 +1,101 @@
+/* eslint-disable react/no-multi-comp */
+import { type ReactElement, useCallback, useRef } from 'react';
+import { useResizeObserver } from '../../hooks/useResizeObserver/useResizeObserver.js';
+import { contain, cover } from './objectFit.js';
+
+export default {
+ title: 'utils/objectFit',
+};
+
+export function Contain(): ReactElement {
+ const parentRef = useRef(null);
+ const childRef = useRef(null);
+ const infoRef = useRef(null);
+
+ const onResize = useCallback(() => {
+ if (!parentRef.current || !childRef.current || !infoRef.current) {
+ return;
+ }
+
+ const objectFit = contain(parentRef.current.offsetWidth, parentRef.current.offsetHeight, 1, 1);
+ childRef.current.style.cssText += objectFit.cssText;
+
+ infoRef.current.innerHTML = JSON.stringify(objectFit);
+ }, [parentRef, childRef, infoRef]);
+
+ useResizeObserver(parentRef, onResize);
+
+ return (
+
+
+
+
+
+
+ );
+}
+
+export function Cover(): ReactElement {
+ const parentRef = useRef(null);
+ const childRef = useRef(null);
+ const infoRef = useRef(null);
+ const demoRef = useRef(null);
+
+ const onResize = useCallback(() => {
+ if (!parentRef.current || !childRef.current || !infoRef.current || !demoRef.current) {
+ return;
+ }
+
+ const objectFit = cover(
+ parentRef.current.offsetWidth,
+ parentRef.current.offsetHeight,
+ childRef.current.naturalWidth,
+ childRef.current.naturalHeight,
+ );
+ childRef.current.style.cssText += objectFit.cssText;
+
+ infoRef.current.innerHTML = JSON.stringify(objectFit);
+ // Using a second image to show the overflowing from the child element on the parent element because the css resize property doesn't allow overflowing.
+ const size = childRef.current.getBoundingClientRect();
+ demoRef.current.style.cssText += `left:${size.left}px;width:${size.width}px;height:${size.height}px;`;
+ }, [parentRef, childRef, infoRef]);
+
+ useResizeObserver(parentRef, onResize);
+
+ const dataUri =
+ '';
+ return (
+
+
+
+
+
+
+
+ );
+}
diff --git a/src/utils/objectFit/objectFit.test.ts b/src/utils/objectFit/objectFit.test.ts
new file mode 100644
index 0000000..ca84eaf
--- /dev/null
+++ b/src/utils/objectFit/objectFit.test.ts
@@ -0,0 +1,37 @@
+import { contain, cover } from './objectFit.js';
+
+describe('objectFit', () => {
+ describe('contain', () => {
+ it('returns expected values for positive input arguments', () => {
+ expect(contain(100, 100, 50, 50)).toEqual({
+ x: 0,
+ y: 0,
+ width: 100,
+ height: 100,
+ scale: 2,
+ cssText: 'left:0px;top:0px;width:100px;height:100px;',
+ });
+ });
+
+ it('throws an error for non-positive input arguments', () => {
+ expect(() => contain(1, 1, 1, 0)).toThrow('All arguments should have a positive value');
+ });
+ });
+
+ describe('cover', () => {
+ it('returns expected values for positive input arguments', () => {
+ expect(cover(100, 100, 50, 50)).toEqual({
+ x: 0,
+ y: 0,
+ width: 100,
+ height: 100,
+ scale: 2,
+ cssText: 'left:0px;top:0px;width:100px;height:100px;',
+ });
+ });
+
+ it('throws an error for non-positive input arguments', () => {
+ expect(() => cover(1, 1, -1, 1)).toThrow('All arguments should have a positive value');
+ });
+ });
+});
diff --git a/src/utils/objectFit/objectFit.ts b/src/utils/objectFit/objectFit.ts
new file mode 100644
index 0000000..e40178f
--- /dev/null
+++ b/src/utils/objectFit/objectFit.ts
@@ -0,0 +1,31 @@
+function objectFit(fit: 'contain' | 'cover') {
+ return (
+ parentWidth: number,
+ parentHeight: number,
+ childWidth: number,
+ childHeight: number,
+ ): { x: number; y: number; width: number; height: number; scale: number; cssText: string } => {
+ if ([parentWidth, parentHeight, childWidth, childHeight].some((value) => value <= 0)) {
+ throw new Error(`All arguments should have a positive value`);
+ }
+
+ const mathMethod = fit === 'contain' ? Math.min : Math.max;
+ const scale = mathMethod(parentWidth / childWidth, parentHeight / childHeight);
+ const width = Math.ceil(childWidth * scale);
+ const height = Math.ceil(childHeight * scale);
+ const x = Math.trunc((parentWidth - width) * 0.5);
+ const y = Math.trunc((parentHeight - height) * 0.5);
+
+ return {
+ x,
+ y,
+ width,
+ height,
+ scale,
+ cssText: `left:${x}px;top:${y}px;width:${width}px;height:${height}px;`,
+ };
+ };
+}
+
+export const contain = objectFit('contain');
+export const cover = objectFit('cover');