// @ts-ignore #hook-types import React, { useState, useRef, useEffect, useMemo } from "react"; import ContextMenu from "./ContextMenu"; type DivProps = React.DetailedHTMLProps< React.HTMLAttributes, HTMLDivElement >; interface Props extends DivProps { handle: React.ReactNode; top?: boolean; bottom?: boolean; } interface State { selected: boolean; } function getNodeOffset(node?: Element) { if (!node) { return { top: 0, left: 0 }; } const rect = node.getBoundingClientRect(); const scrollLeft = window.pageXOffset || document.documentElement!.scrollLeft; const scrollTop = window.pageYOffset || document.documentElement!.scrollTop; return { top: rect.top + scrollTop, left: rect.left + scrollLeft }; } function ContextMenuTarget(props: Props) { const handleNode = useRef(null); const [selected, setSelected] = useState(false); useEffect( () => { function handleGlobalClick(e: MouseEvent) { if ( selected && // Typescript does not believe that these click events are always fired on DOM nodes. e.target instanceof Element && selected && // Not sure how, but it's possible for this to get called when handleNode is null/undefined. // https://sentry.io/share/issue/2066cd79f21e4f279791319f4d2ea35d/ handleNode.current && !handleNode.current.contains(e.target!) ) { setSelected(false); } } document.addEventListener("click", handleGlobalClick); return () => { document.removeEventListener("click", handleGlobalClick); }; }, [selected, handleNode.current] ); const offset = useMemo(() => getNodeOffset(handleNode.current), [selected]); const { handle, children, top, bottom, ...passThroughProps } = props; return (
setSelected(!selected)} > {handle}
{children}
); } export default ContextMenuTarget;