import { h, Fragment } from 'preact'; import { createPortal } from 'preact/compat'; import { useCallback, useEffect, useRef, useState } from 'preact/hooks'; export default function RelativeModal({ className, role = 'dialog', children, onDismiss, portalRootID, relativeTo }) { const [position, setPosition] = useState({ top: -999, left: 0, width: 0 }); const [show, setShow] = useState(false); const portalRoot = portalRootID && document.getElementById(portalRootID); const ref = useRef(null); const handleDismiss = useCallback( (event) => { onDismiss && onDismiss(event); }, [onDismiss] ); const handleKeydown = useCallback( (event) => { const focusable = ref.current.querySelectorAll('[tabindex]'); if (event.key === 'Tab' && focusable.length) { if (event.shiftKey && document.activeElement === focusable[0]) { focusable[focusable.length - 1].focus(); event.preventDefault(); } else if (document.activeElement === focusable[focusable.length - 1]) { focusable[0].focus(); event.preventDefault(); } return; } if (event.key === 'Escape') { setShow(false); return; } }, [ref.current] ); useEffect(() => { if (ref && ref.current && relativeTo && relativeTo.current) { const windowWidth = window.innerWidth; const windowHeight = window.innerHeight; const { width: menuWidth, height: menuHeight } = ref.current.getBoundingClientRect(); const { x, y, width, height } = relativeTo.current.getBoundingClientRect(); let top = y + height; let left = x; // too far right if (left + menuWidth > windowWidth) { left = windowWidth - menuWidth; } // too far left else if (left < 0) { left = 0; } // too close to bottom if (top + menuHeight > windowHeight) { top = y - menuHeight; } setPosition({ left, top, width }); const focusable = ref.current.querySelector('[tabindex]'); focusable && console.log('focusing'); focusable && focusable.focus(); } }, [relativeTo && relativeTo.current, ref && ref.current]); useEffect(() => { if (position.width) { setShow(true); } else { setShow(false); } }, [show, position.width, ref.current]); const menu = (
0 ? `width: ${position.width}px; top: ${position.top}px; left: ${position.left}px` : ''} > {children}
); return portalRoot ? createPortal(menu, portalRoot) : menu; }