2022-10-21 00:14:17 -05:00
|
|
|
import type { RefObject } from 'react';
|
|
|
|
import { useCallback, useEffect, useState } from 'react';
|
2022-05-12 22:54:10 -05:00
|
|
|
|
2022-10-10 10:53:34 -05:00
|
|
|
export type VerticalAlignment = 'top' | 'center' | 'bottom';
|
2022-05-12 22:54:10 -05:00
|
|
|
|
|
|
|
export type HorizontalAlignment = 'start' | 'center' | 'end';
|
|
|
|
|
|
|
|
type Offset = {
|
|
|
|
vertical: number;
|
|
|
|
horizontal: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
type Props = {
|
|
|
|
verticalAlign: VerticalAlignment;
|
|
|
|
horizontalAlign: HorizontalAlignment;
|
|
|
|
offset: Offset;
|
|
|
|
anchorRef: RefObject<Element>;
|
|
|
|
overlayRef: RefObject<Element>;
|
|
|
|
};
|
|
|
|
|
|
|
|
type Position = {
|
|
|
|
top: number;
|
|
|
|
left: number;
|
|
|
|
};
|
|
|
|
|
|
|
|
// Leave space for box-shadow effect.
|
|
|
|
const windowSafePadding = 12;
|
|
|
|
|
|
|
|
const selectVerticalAlignment = ({
|
|
|
|
verticalAlign,
|
|
|
|
verticalTop,
|
2022-10-10 10:53:34 -05:00
|
|
|
verticalCenter,
|
2022-05-12 22:54:10 -05:00
|
|
|
verticalBottom,
|
|
|
|
overlayHeight,
|
|
|
|
}: {
|
|
|
|
verticalAlign: VerticalAlignment;
|
|
|
|
verticalTop: number;
|
2022-10-10 10:53:34 -05:00
|
|
|
verticalCenter: number;
|
2022-05-12 22:54:10 -05:00
|
|
|
verticalBottom: number;
|
|
|
|
overlayHeight: number;
|
|
|
|
}) => {
|
|
|
|
const minY = windowSafePadding;
|
|
|
|
const maxY = window.innerHeight - windowSafePadding;
|
|
|
|
|
|
|
|
const isTopAllowed = verticalTop >= minY;
|
2022-10-10 10:53:34 -05:00
|
|
|
const isCenterAllowed =
|
|
|
|
verticalCenter - overlayHeight / 2 >= minY && verticalCenter + overlayHeight / 2 <= maxY;
|
2022-05-12 22:54:10 -05:00
|
|
|
const isBottomAllowed = verticalBottom + overlayHeight <= maxY;
|
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
switch (verticalAlign) {
|
|
|
|
case 'top': {
|
|
|
|
if (isTopAllowed) {
|
|
|
|
return 'top';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isBottomAllowed) {
|
|
|
|
return 'bottom';
|
|
|
|
}
|
|
|
|
|
2022-10-10 10:53:34 -05:00
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
|
|
|
return verticalAlign;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'center': {
|
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isTopAllowed) {
|
|
|
|
return 'top';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isBottomAllowed) {
|
|
|
|
return 'bottom';
|
|
|
|
}
|
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
return verticalAlign;
|
2022-05-12 22:54:10 -05:00
|
|
|
}
|
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
case 'bottom': {
|
|
|
|
if (isBottomAllowed) {
|
|
|
|
return 'bottom';
|
|
|
|
}
|
2022-05-12 22:54:10 -05:00
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
if (isTopAllowed) {
|
|
|
|
return 'top';
|
|
|
|
}
|
|
|
|
|
2022-10-10 10:53:34 -05:00
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
return verticalAlign;
|
|
|
|
}
|
2022-05-12 22:54:10 -05:00
|
|
|
|
2022-05-12 23:13:44 -05:00
|
|
|
default: {
|
|
|
|
return verticalAlign;
|
|
|
|
}
|
|
|
|
}
|
2022-05-12 22:54:10 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
const selectHorizontalAlignment = ({
|
|
|
|
horizontalAlign,
|
|
|
|
horizontalStart,
|
|
|
|
horizontalCenter,
|
|
|
|
horizontalEnd,
|
|
|
|
overlayWidth,
|
|
|
|
}: {
|
|
|
|
horizontalAlign: HorizontalAlignment;
|
|
|
|
horizontalStart: number;
|
|
|
|
horizontalCenter: number;
|
|
|
|
horizontalEnd: number;
|
|
|
|
overlayWidth: number;
|
|
|
|
}) => {
|
|
|
|
const minX = windowSafePadding;
|
|
|
|
const maxX = window.innerWidth - windowSafePadding;
|
|
|
|
|
|
|
|
const isStartAllowed = horizontalStart + overlayWidth <= maxX;
|
|
|
|
const isCenterAllowed =
|
|
|
|
horizontalCenter - overlayWidth / 2 >= minX && horizontalCenter + overlayWidth / 2 <= maxX;
|
|
|
|
const isEndAllowed = horizontalEnd >= minX;
|
|
|
|
|
|
|
|
switch (horizontalAlign) {
|
|
|
|
case 'start': {
|
|
|
|
if (isStartAllowed) {
|
|
|
|
return 'start';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isEndAllowed) {
|
|
|
|
return 'end';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
|
|
|
return horizontalAlign;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'center': {
|
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isStartAllowed) {
|
|
|
|
return 'start';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isEndAllowed) {
|
|
|
|
return 'end';
|
|
|
|
}
|
|
|
|
|
|
|
|
return horizontalAlign;
|
|
|
|
}
|
|
|
|
|
|
|
|
case 'end': {
|
|
|
|
if (isEndAllowed) {
|
|
|
|
return 'end';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isStartAllowed) {
|
|
|
|
return 'start';
|
|
|
|
}
|
|
|
|
|
|
|
|
if (isCenterAllowed) {
|
|
|
|
return 'center';
|
|
|
|
}
|
|
|
|
|
|
|
|
return horizontalAlign;
|
|
|
|
}
|
|
|
|
|
|
|
|
default: {
|
|
|
|
return horizontalAlign;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
export default function usePosition({
|
|
|
|
verticalAlign,
|
|
|
|
horizontalAlign,
|
|
|
|
offset,
|
|
|
|
anchorRef,
|
|
|
|
overlayRef,
|
|
|
|
}: Props) {
|
|
|
|
const [position, setPosition] = useState<Position>();
|
|
|
|
const [currentVerticalAlign, setCurrentVerticalAlign] = useState(verticalAlign);
|
|
|
|
const [currentHorizontalAlign, setCurrentHorizontalAlign] = useState(horizontalAlign);
|
|
|
|
|
|
|
|
const updatePosition = useCallback(() => {
|
|
|
|
if (!anchorRef.current || !overlayRef.current) {
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
const anchorRect = anchorRef.current.getBoundingClientRect();
|
|
|
|
const overlayRect = overlayRef.current.getBoundingClientRect();
|
|
|
|
|
2022-10-23 10:38:11 -05:00
|
|
|
const verticalTop = anchorRect.y - overlayRect.height - offset.vertical;
|
2022-10-10 10:53:34 -05:00
|
|
|
const verticalCenter =
|
2022-10-23 10:38:11 -05:00
|
|
|
anchorRect.y - anchorRect.height / 2 - overlayRect.height / 2 + offset.vertical;
|
|
|
|
const verticalBottom = anchorRect.y + anchorRect.height + offset.vertical;
|
2022-05-12 22:54:10 -05:00
|
|
|
|
2022-10-10 10:53:34 -05:00
|
|
|
const verticalPositionMap = {
|
|
|
|
top: verticalTop,
|
|
|
|
center: verticalCenter,
|
|
|
|
bottom: verticalBottom,
|
|
|
|
};
|
2022-05-12 22:54:10 -05:00
|
|
|
|
2022-10-23 10:38:11 -05:00
|
|
|
const horizontalStart = anchorRect.x + offset.horizontal;
|
2022-05-12 22:54:10 -05:00
|
|
|
const horizontalCenter =
|
2022-10-23 10:38:11 -05:00
|
|
|
anchorRect.x + anchorRect.width / 2 - overlayRect.width / 2 + offset.horizontal;
|
|
|
|
const horizontalEnd = anchorRect.x + anchorRect.width - overlayRect.width + offset.horizontal;
|
2022-05-12 22:54:10 -05:00
|
|
|
|
|
|
|
const horizontalPositionMap = {
|
|
|
|
start: horizontalStart,
|
|
|
|
center: horizontalCenter,
|
|
|
|
end: horizontalEnd,
|
|
|
|
};
|
|
|
|
|
|
|
|
const selectedVerticalAlign = selectVerticalAlignment({
|
|
|
|
verticalAlign,
|
|
|
|
verticalTop,
|
2022-10-10 10:53:34 -05:00
|
|
|
verticalCenter,
|
2022-05-12 22:54:10 -05:00
|
|
|
verticalBottom,
|
|
|
|
overlayHeight: overlayRect.height,
|
|
|
|
});
|
|
|
|
|
|
|
|
const selectedHorizontalAlign = selectHorizontalAlignment({
|
|
|
|
horizontalAlign,
|
|
|
|
horizontalStart,
|
|
|
|
horizontalCenter,
|
|
|
|
horizontalEnd,
|
|
|
|
overlayWidth: overlayRect.width,
|
|
|
|
});
|
|
|
|
|
|
|
|
setCurrentVerticalAlign(selectedVerticalAlign);
|
|
|
|
setCurrentHorizontalAlign(selectedHorizontalAlign);
|
|
|
|
setPosition({
|
|
|
|
top: verticalPositionMap[selectedVerticalAlign],
|
|
|
|
left: horizontalPositionMap[selectedHorizontalAlign],
|
|
|
|
});
|
|
|
|
}, [anchorRef, horizontalAlign, offset.vertical, offset.horizontal, overlayRef, verticalAlign]);
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
updatePosition();
|
|
|
|
window.addEventListener('resize', updatePosition);
|
|
|
|
window.addEventListener('scroll', updatePosition);
|
|
|
|
|
|
|
|
return () => {
|
|
|
|
window.removeEventListener('resize', updatePosition);
|
|
|
|
window.removeEventListener('scroll', updatePosition);
|
|
|
|
};
|
|
|
|
}, [updatePosition]);
|
|
|
|
|
|
|
|
return {
|
|
|
|
position,
|
|
|
|
mutate: updatePosition,
|
|
|
|
positionState: {
|
|
|
|
verticalAlign: currentVerticalAlign,
|
|
|
|
horizontalAlign: currentHorizontalAlign,
|
|
|
|
},
|
|
|
|
};
|
|
|
|
}
|