From 009ba78286ee7572708609da7edfba3ac0aafc9b Mon Sep 17 00:00:00 2001 From: David Evans Date: Sat, 30 Nov 2024 13:49:12 +0000 Subject: [PATCH] Fix events inside popups by using a portal --- frontend/src/components/common/Popup.tsx | 66 +++++++++++++++--------- 1 file changed, 43 insertions(+), 23 deletions(-) diff --git a/frontend/src/components/common/Popup.tsx b/frontend/src/components/common/Popup.tsx index ccfafb4..fcda512 100644 --- a/frontend/src/components/common/Popup.tsx +++ b/frontend/src/components/common/Popup.tsx @@ -1,12 +1,11 @@ import { type FC, type PropsWithChildren, - useRef, - useEffect, - type KeyboardEvent, useId, + useLayoutEffect, useState, } from 'react'; +import { createPortal } from 'react-dom'; import { useEvent } from '../../hooks/useEvent'; import './Popup.less'; @@ -26,7 +25,15 @@ export const Popup: FC> = ({ onClose, children, }) => { + const titleID = useId(); + const [dialog] = useState(() => document.createElement('dialog')); const [lagDisplay, setLagDisplay] = useState(false); + const showContent = isOpen || lagDisplay; + + const handleCancel = useEvent((e: Event) => { + e.preventDefault(); + onClose(); + }); const handleKeyDown = useEvent((e: KeyboardEvent) => { e.stopPropagation(); @@ -47,34 +54,47 @@ export const Popup: FC> = ({ } }); - const id = useId(); - const dialog = useRef(null); - useEffect(() => { + useLayoutEffect(() => { + dialog.className = 'popup-content'; + dialog.setAttribute('aria-labelledby', titleID); + dialog.addEventListener('cancel', handleCancel); + dialog.addEventListener('keydown', handleKeyDown); + return () => { + dialog.removeEventListener('cancel', handleCancel); + dialog.removeEventListener('keydown', handleKeyDown); + }; + }, [dialog, titleID, handleCancel, handleKeyDown]); + + useLayoutEffect(() => { + if (!showContent) { + return; + } + document.body.append(dialog); + return () => dialog.remove(); + }, [dialog, showContent]); + + useLayoutEffect(() => { if (isOpen) { setLagDisplay(true); - dialog.current?.showModal(); - return () => dialog.current?.close(); + dialog.showModal(); + return () => dialog.close(); } else { const tm = setTimeout(() => setLagDisplay(false), 500); return () => clearTimeout(tm); } - }, [isOpen]); + }, [dialog, isOpen]); + + if (!showContent) { + return null; + } - return ( - { - e.preventDefault(); - onClose(); - }} - onKeyDown={handleKeyDown} - aria-labelledby={id} - > -

+ return createPortal( + <> +

{title}

- {isOpen || lagDisplay ? children : null} -
+ {children} + , + dialog, ); };