signal-desktop/ts/hooks/useHasWrapped.ts

59 lines
1.4 KiB
TypeScript
Raw Normal View History

2023-01-03 19:55:46 +00:00
// Copyright 2020 Signal Messenger, LLC
2021-09-17 22:24:21 +00:00
// SPDX-License-Identifier: AGPL-3.0-only
import type { Ref } from 'react';
import { useEffect, useState } from 'react';
2021-09-17 22:24:21 +00:00
import { first, last, noop } from 'lodash';
2024-11-12 21:56:09 +00:00
function getBottom(element: Readonly<Element>): number {
return element.getBoundingClientRect().bottom;
}
2021-09-17 22:24:21 +00:00
function getTop(element: Readonly<Element>): number {
return element.getBoundingClientRect().top;
}
function isWrapped(element: Readonly<null | HTMLElement>): boolean {
if (!element) {
return false;
}
const { children } = element;
const firstChild = first(children);
const lastChild = last(children);
return Boolean(
firstChild &&
lastChild &&
firstChild !== lastChild &&
2024-11-12 21:56:09 +00:00
getBottom(firstChild) <= getTop(lastChild)
2021-09-17 22:24:21 +00:00
);
}
/**
* A hook that returns a ref (to put on your element) and a boolean. The boolean will be
* `true` if the element's children have different `top`s, and `false` otherwise.
*/
export function useHasWrapped<T extends HTMLElement>(): [Ref<T>, boolean] {
const [element, setElement] = useState<null | T>(null);
const [hasWrapped, setHasWrapped] = useState(isWrapped(element));
useEffect(() => {
if (!element) {
return noop;
}
const observer = new ResizeObserver(() => {
2021-09-17 22:24:21 +00:00
setHasWrapped(isWrapped(element));
});
observer.observe(element);
return () => {
observer.disconnect();
};
}, [element]);
return [setElement, hasWrapped];
}