// Copyright 2018-2022 Signal Messenger, LLC // SPDX-License-Identifier: AGPL-3.0-only import type { KeyboardEvent, ReactNode } from 'react'; import type { Options, VirtualElement } from '@popperjs/core'; import FocusTrap from 'focus-trap-react'; import React, { useEffect, useRef, useState } from 'react'; import classNames from 'classnames'; import { usePopper } from 'react-popper'; import { noop } from 'lodash'; import type { Theme } from '../util/theme'; import type { LocalizerType } from '../types/Util'; import { getClassNamesFor } from '../util/getClassNamesFor'; import { themeClassName } from '../util/theme'; export type ContextMenuOptionType = { readonly description?: string; readonly icon?: string; readonly label: string; readonly onClick: (value?: T) => unknown; readonly value?: T; }; export type PropsType = { readonly ariaLabel?: string; readonly children?: ReactNode; readonly i18n: LocalizerType; readonly menuOptions: ReadonlyArray>; readonly moduleClassName?: string; readonly onClick?: () => unknown; readonly onMenuShowingChanged?: (value: boolean) => unknown; readonly popperOptions?: Pick; readonly theme?: Theme; readonly title?: string; readonly value?: T; }; let closeCurrentOpenContextMenu: undefined | (() => unknown); // https://popper.js.org/docs/v2/virtual-elements/ // Generating a virtual element here so that we can make the menu pop up // right under the mouse cursor. function generateVirtualElement(x: number, y: number): VirtualElement { return { getBoundingClientRect: () => ({ bottom: y, height: 0, left: x, right: x, toJSON: () => ({ x, y }), top: y, width: 0, x, y, }), }; } export function ContextMenu({ ariaLabel, children, i18n, menuOptions, moduleClassName, onClick, onMenuShowingChanged, popperOptions, theme, title, value, }: PropsType): JSX.Element { const [isMenuShowing, setIsMenuShowing] = useState(false); const [focusedIndex, setFocusedIndex] = useState( undefined ); const [popperElement, setPopperElement] = useState( null ); const virtualElement = useRef(generateVirtualElement(0, 0)); const [referenceElement, setReferenceElement] = useState(null); const { styles, attributes } = usePopper( virtualElement.current, popperElement, { placement: 'top-start', strategy: 'fixed', ...popperOptions, } ); useEffect(() => { if (onMenuShowingChanged) { onMenuShowingChanged(isMenuShowing); } }, [isMenuShowing, onMenuShowingChanged]); useEffect(() => { if (!isMenuShowing) { return noop; } const handleOutsideClick = (event: MouseEvent) => { if (!referenceElement?.contains(event.target as Node)) { setIsMenuShowing(false); closeCurrentOpenContextMenu = undefined; event.stopPropagation(); event.preventDefault(); } }; document.addEventListener('click', handleOutsideClick); return () => { document.removeEventListener('click', handleOutsideClick); }; }, [isMenuShowing, referenceElement]); const handleKeyDown = (ev: KeyboardEvent) => { if (!isMenuShowing) { if (ev.key === 'Enter') { setFocusedIndex(0); } return; } if (ev.key === 'ArrowDown') { const currFocusedIndex = focusedIndex || 0; const nextFocusedIndex = currFocusedIndex >= menuOptions.length - 1 ? 0 : currFocusedIndex + 1; setFocusedIndex(nextFocusedIndex); ev.stopPropagation(); ev.preventDefault(); } if (ev.key === 'ArrowUp') { const currFocusedIndex = focusedIndex || 0; const nextFocusedIndex = currFocusedIndex === 0 ? menuOptions.length - 1 : currFocusedIndex - 1; setFocusedIndex(nextFocusedIndex); ev.stopPropagation(); ev.preventDefault(); } if (ev.key === 'Enter') { if (focusedIndex !== undefined) { const focusedOption = menuOptions[focusedIndex]; focusedOption.onClick(focusedOption.value); } setIsMenuShowing(false); closeCurrentOpenContextMenu = undefined; ev.stopPropagation(); ev.preventDefault(); } }; const handleClick = (ev: React.MouseEvent) => { closeCurrentOpenContextMenu?.(); closeCurrentOpenContextMenu = () => setIsMenuShowing(false); virtualElement.current = generateVirtualElement(ev.clientX, ev.clientY); setIsMenuShowing(true); ev.stopPropagation(); ev.preventDefault(); }; const getClassName = getClassNamesFor('ContextMenu', moduleClassName); return (
{isMenuShowing && (
{title &&
{title}
} {menuOptions.map((option, index) => ( ))}
)}
); }