| 
									
										
										
										
											2023-01-03 11:55:46 -08:00
										 |  |  | // Copyright 2018 Signal Messenger, LLC
 | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | // SPDX-License-Identifier: AGPL-3.0-only
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  | import type { KeyboardEvent, ReactNode } from 'react'; | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  | import type { Options, VirtualElement } from '@popperjs/core'; | 
					
						
							|  |  |  | import React, { useEffect, useRef, useState } from 'react'; | 
					
						
							| 
									
										
										
										
											2023-04-05 13:48:00 -07:00
										 |  |  | import { createPortal } from 'react-dom'; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 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'; | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  | import { getClassNamesFor } from '../util/getClassNamesFor'; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | import { themeClassName } from '../util/theme'; | 
					
						
							| 
									
										
										
										
											2022-09-14 18:58:35 -07:00
										 |  |  | import { handleOutsideClick } from '../util/handleOutsideClick'; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  | export type ContextMenuOptionType<T> = Readonly<{ | 
					
						
							|  |  |  |   description?: string; | 
					
						
							|  |  |  |   icon?: string; | 
					
						
							|  |  |  |   label: string; | 
					
						
							|  |  |  |   group?: string; | 
					
						
							|  |  |  |   onClick: (value?: T) => unknown; | 
					
						
							|  |  |  |   value?: T; | 
					
						
							|  |  |  | }>; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  | type RenderButtonProps = Readonly<{ | 
					
						
							| 
									
										
										
										
											2024-03-10 14:14:22 -04:00
										 |  |  |   onClick: (ev: React.MouseEvent) => void; | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |   onKeyDown: (ev: KeyboardEvent) => void; | 
					
						
							|  |  |  |   isMenuShowing: boolean; | 
					
						
							|  |  |  |   ref: React.Ref<HTMLButtonElement> | null; | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   menuNode: ReactNode; | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  | }>; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  | export type PropsType<T> = Readonly<{ | 
					
						
							|  |  |  |   ariaLabel?: string; | 
					
						
							|  |  |  |   // contents of the button OR a function that will render the whole button
 | 
					
						
							|  |  |  |   children?: ReactNode | ((props: RenderButtonProps) => JSX.Element); | 
					
						
							|  |  |  |   i18n: LocalizerType; | 
					
						
							|  |  |  |   menuOptions: ReadonlyArray<ContextMenuOptionType<T>>; | 
					
						
							|  |  |  |   moduleClassName?: string; | 
					
						
							|  |  |  |   button?: () => JSX.Element; | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   onClick?: (ev: React.MouseEvent) => unknown; | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |   onMenuShowingChanged?: (value: boolean) => unknown; | 
					
						
							|  |  |  |   popperOptions?: Pick<Options, 'placement' | 'strategy'>; | 
					
						
							| 
									
										
										
										
											2023-04-05 13:48:00 -07:00
										 |  |  |   portalToRoot?: boolean; | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |   theme?: Theme; | 
					
						
							|  |  |  |   title?: string; | 
					
						
							|  |  |  |   value?: T; | 
					
						
							|  |  |  | }>; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-12 17:32:27 -04:00
										 |  |  | let closeCurrentOpenContextMenu: undefined | (() => unknown); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  | // 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 { | 
					
						
							| 
									
										
										
										
											2023-01-12 19:04:49 -05:00
										 |  |  |     getBoundingClientRect: () => new DOMRect(x, y), | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  |   }; | 
					
						
							|  |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  | export function ContextMenu<T>({ | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  |   ariaLabel, | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   children, | 
					
						
							|  |  |  |   i18n, | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   menuOptions, | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   moduleClassName, | 
					
						
							|  |  |  |   onClick, | 
					
						
							|  |  |  |   onMenuShowingChanged, | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   popperOptions, | 
					
						
							| 
									
										
										
										
											2023-04-05 13:48:00 -07:00
										 |  |  |   portalToRoot, | 
					
						
							| 
									
										
										
										
											2022-05-06 15:02:44 -04:00
										 |  |  |   theme, | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   title, | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   value, | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  | }: PropsType<T>): JSX.Element { | 
					
						
							|  |  |  |   const [isMenuShowing, setIsMenuShowing] = useState<boolean>(false); | 
					
						
							|  |  |  |   const [focusedIndex, setFocusedIndex] = useState<number | undefined>( | 
					
						
							|  |  |  |     undefined | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   const [popperElement, setPopperElement] = useState<HTMLDivElement | null>( | 
					
						
							|  |  |  |     null | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |   const virtualElement = useRef<VirtualElement>(generateVirtualElement(0, 0)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   const [referenceElement, setReferenceElement] = | 
					
						
							|  |  |  |     useState<HTMLButtonElement | null>(null); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  |   const { styles, attributes } = usePopper( | 
					
						
							|  |  |  |     virtualElement.current, | 
					
						
							|  |  |  |     popperElement, | 
					
						
							|  |  |  |     { | 
					
						
							|  |  |  |       placement: 'top-start', | 
					
						
							|  |  |  |       strategy: 'fixed', | 
					
						
							|  |  |  |       ...popperOptions, | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   ); | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-27 18:13:28 -05:00
										 |  |  |   // In Electron v23+, new elements added to the DOM may not trigger a recalculation of
 | 
					
						
							|  |  |  |   // draggable regions, so if a ContextMenu is shown on top of a draggable region, its
 | 
					
						
							|  |  |  |   // buttons may be unclickable. We add a class so that we can disable those draggable
 | 
					
						
							|  |  |  |   // regions while the context menu is shown. It has the added benefit of ensuring that
 | 
					
						
							|  |  |  |   // click events outside of the context menu onto an otherwise draggable region are
 | 
					
						
							|  |  |  |   // propagated and trigger the menu to close.
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     document.body.classList.toggle('context-menu-open', isMenuShowing); | 
					
						
							|  |  |  |   }, [isMenuShowing]); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     // Remove it on unmount in case the component is unmounted when the menu is open
 | 
					
						
							|  |  |  |     return () => document.body.classList.remove('context-menu-open'); | 
					
						
							|  |  |  |   }, []); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (onMenuShowingChanged) { | 
					
						
							|  |  |  |       onMenuShowingChanged(isMenuShowing); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   }, [isMenuShowing, onMenuShowingChanged]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (!isMenuShowing) { | 
					
						
							|  |  |  |       return noop; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-09-14 18:58:35 -07:00
										 |  |  |     return handleOutsideClick( | 
					
						
							|  |  |  |       () => { | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |         setIsMenuShowing(false); | 
					
						
							| 
									
										
										
										
											2022-08-12 17:32:27 -04:00
										 |  |  |         closeCurrentOpenContextMenu = undefined; | 
					
						
							| 
									
										
										
										
											2022-09-14 18:58:35 -07:00
										 |  |  |         return true; | 
					
						
							|  |  |  |       }, | 
					
						
							| 
									
										
										
										
											2022-09-27 13:24:21 -07:00
										 |  |  |       { | 
					
						
							|  |  |  |         containerElements: [referenceElement, popperElement], | 
					
						
							|  |  |  |         name: 'ContextMenu', | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-09-14 18:58:35 -07:00
										 |  |  |     ); | 
					
						
							|  |  |  |   }, [isMenuShowing, referenceElement, popperElement]); | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-04-05 13:48:00 -07:00
										 |  |  |   const [portalNode, setPortalNode] = React.useState<HTMLElement | null>(null); | 
					
						
							|  |  |  |   useEffect(() => { | 
					
						
							|  |  |  |     if (!portalToRoot || !isMenuShowing) { | 
					
						
							|  |  |  |       return noop; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const div = document.createElement('div'); | 
					
						
							|  |  |  |     document.body.appendChild(div); | 
					
						
							|  |  |  |     setPortalNode(div); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     return () => { | 
					
						
							|  |  |  |       document.body.removeChild(div); | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  |   }, [portalToRoot, isMenuShowing]); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |   const handleKeyDown = (ev: KeyboardEvent) => { | 
					
						
							| 
									
										
										
										
											2022-10-28 12:41:50 -04:00
										 |  |  |     if ((ev.key === 'Enter' || ev.key === 'Space') && !isMenuShowing) { | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu?.(); | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu = () => setIsMenuShowing(false); | 
					
						
							|  |  |  |       if (referenceElement) { | 
					
						
							|  |  |  |         const box = referenceElement.getBoundingClientRect(); | 
					
						
							|  |  |  |         virtualElement.current = generateVirtualElement(box.x, box.y); | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-10-28 12:41:50 -04:00
										 |  |  |       setIsMenuShowing(true); | 
					
						
							|  |  |  |       setFocusedIndex(0); | 
					
						
							|  |  |  |       ev.preventDefault(); | 
					
						
							|  |  |  |       ev.stopPropagation(); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (!isMenuShowing) { | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |       return; | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-28 12:41:50 -04:00
										 |  |  |     if (ev.key === 'ArrowDown' || ev.key === 'Tab') { | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |       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) { | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |         const focusedOption = menuOptions[focusedIndex]; | 
					
						
							|  |  |  |         focusedOption.onClick(focusedOption.value); | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |       } | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |       setIsMenuShowing(false); | 
					
						
							| 
									
										
										
										
											2022-08-12 17:32:27 -04:00
										 |  |  |       closeCurrentOpenContextMenu = undefined; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |       ev.stopPropagation(); | 
					
						
							|  |  |  |       ev.preventDefault(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-11-01 14:56:19 -04:00
										 |  |  | 
 | 
					
						
							|  |  |  |     if (ev.key === 'Escape') { | 
					
						
							|  |  |  |       setIsMenuShowing(false); | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu = undefined; | 
					
						
							|  |  |  |       ev.stopPropagation(); | 
					
						
							|  |  |  |       ev.preventDefault(); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  |   }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-08-19 14:36:47 -04:00
										 |  |  |   const handleClick = (ev: React.MouseEvent) => { | 
					
						
							| 
									
										
										
										
											2024-03-10 14:14:22 -04:00
										 |  |  |     if (isMenuShowing && ev.type !== 'contextmenu') { | 
					
						
							|  |  |  |       setIsMenuShowing(false); | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu = undefined; | 
					
						
							|  |  |  |     } else { | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu?.(); | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu = () => setIsMenuShowing(false); | 
					
						
							|  |  |  |       virtualElement.current = generateVirtualElement(ev.clientX, ev.clientY); | 
					
						
							|  |  |  |       setIsMenuShowing(true); | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2022-03-04 16:14:52 -05:00
										 |  |  |     ev.stopPropagation(); | 
					
						
							|  |  |  |     ev.preventDefault(); | 
					
						
							|  |  |  |   }; | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |   const getClassName = getClassNamesFor('ContextMenu', moduleClassName); | 
					
						
							| 
									
										
										
										
											2021-11-30 21:14:25 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  |   const optionElements = new Array<JSX.Element>(); | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |   const isAnyOptionSelected = typeof value !== 'undefined'; | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   for (const [index, option] of menuOptions.entries()) { | 
					
						
							|  |  |  |     const previous = menuOptions[index - 1]; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     const needsDivider = previous && previous.group !== option.group; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (needsDivider) { | 
					
						
							|  |  |  |       optionElements.push( | 
					
						
							|  |  |  |         <div | 
					
						
							|  |  |  |           className={getClassName('__divider')} | 
					
						
							|  |  |  |           key={`${option.label}-divider`} | 
					
						
							|  |  |  |         /> | 
					
						
							|  |  |  |       ); | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     // eslint-disable-next-line no-loop-func
 | 
					
						
							|  |  |  |     const onElementClick = (ev: React.MouseEvent): void => { | 
					
						
							|  |  |  |       ev.preventDefault(); | 
					
						
							|  |  |  |       ev.stopPropagation(); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       option.onClick(option.value); | 
					
						
							|  |  |  |       setIsMenuShowing(false); | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |       closeCurrentOpenContextMenu = undefined; | 
					
						
							|  |  |  |     }; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |     const isOptionSelected = isAnyOptionSelected && value === option.value; | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  |     optionElements.push( | 
					
						
							|  |  |  |       <button | 
					
						
							|  |  |  |         aria-label={option.label} | 
					
						
							|  |  |  |         className={classNames( | 
					
						
							|  |  |  |           getClassName('__option'), | 
					
						
							|  |  |  |           focusedIndex === index ? getClassName('__option--focused') : undefined | 
					
						
							|  |  |  |         )} | 
					
						
							|  |  |  |         key={option.label} | 
					
						
							|  |  |  |         type="button" | 
					
						
							|  |  |  |         onClick={onElementClick} | 
					
						
							|  |  |  |       > | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  |         <div | 
					
						
							|  |  |  |           className={classNames( | 
					
						
							|  |  |  |             getClassName('__option--container'), | 
					
						
							|  |  |  |             isAnyOptionSelected | 
					
						
							|  |  |  |               ? getClassName('__option--container--with-selection') | 
					
						
							|  |  |  |               : undefined, | 
					
						
							|  |  |  |             isOptionSelected | 
					
						
							|  |  |  |               ? getClassName('__option--container--selected') | 
					
						
							|  |  |  |               : undefined | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |         > | 
					
						
							| 
									
										
										
										
											2022-10-18 10:12:02 -07:00
										 |  |  |           {option.icon && ( | 
					
						
							|  |  |  |             <div | 
					
						
							|  |  |  |               className={classNames( | 
					
						
							|  |  |  |                 getClassName('__option--icon'), | 
					
						
							|  |  |  |                 option.icon | 
					
						
							|  |  |  |               )} | 
					
						
							|  |  |  |             /> | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |           <div> | 
					
						
							|  |  |  |             <div className={getClassName('__option--title')}> | 
					
						
							|  |  |  |               {option.label} | 
					
						
							|  |  |  |             </div> | 
					
						
							|  |  |  |             {option.description && ( | 
					
						
							|  |  |  |               <div className={getClassName('__option--description')}> | 
					
						
							|  |  |  |                 {option.description} | 
					
						
							|  |  |  |               </div> | 
					
						
							|  |  |  |             )} | 
					
						
							|  |  |  |           </div> | 
					
						
							|  |  |  |         </div> | 
					
						
							|  |  |  |       </button> | 
					
						
							|  |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   const menuNode = isMenuShowing ? ( | 
					
						
							|  |  |  |     <div className={theme ? themeClassName(theme) : undefined}> | 
					
						
							|  |  |  |       <div | 
					
						
							|  |  |  |         className={classNames( | 
					
						
							|  |  |  |           getClassName('__popper'), | 
					
						
							|  |  |  |           menuOptions.length === 1 | 
					
						
							|  |  |  |             ? getClassName('__popper--single-item') | 
					
						
							|  |  |  |             : undefined | 
					
						
							|  |  |  |         )} | 
					
						
							|  |  |  |         ref={setPopperElement} | 
					
						
							|  |  |  |         style={styles.popper} | 
					
						
							|  |  |  |         {...attributes.popper} | 
					
						
							|  |  |  |       > | 
					
						
							|  |  |  |         {title && <div className={getClassName('__title')}>{title}</div>} | 
					
						
							|  |  |  |         {optionElements} | 
					
						
							|  |  |  |       </div> | 
					
						
							|  |  |  |     </div> | 
					
						
							|  |  |  |   ) : undefined; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |   let buttonNode: JSX.Element; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |   if (typeof children === 'function') { | 
					
						
							| 
									
										
										
										
											2023-08-08 17:53:06 -07:00
										 |  |  |     buttonNode = ( | 
					
						
							|  |  |  |       <> | 
					
						
							|  |  |  |         {(children as (props: RenderButtonProps) => JSX.Element)({ | 
					
						
							| 
									
										
										
										
											2024-03-10 14:14:22 -04:00
										 |  |  |           onClick: onClick || handleClick, | 
					
						
							| 
									
										
										
										
											2023-08-08 17:53:06 -07:00
										 |  |  |           onKeyDown: handleKeyDown, | 
					
						
							|  |  |  |           isMenuShowing, | 
					
						
							|  |  |  |           ref: setReferenceElement, | 
					
						
							|  |  |  |           menuNode, | 
					
						
							|  |  |  |         })} | 
					
						
							|  |  |  |         {portalNode ? createPortal(menuNode, portalNode) : menuNode} | 
					
						
							|  |  |  |       </> | 
					
						
							|  |  |  |     ); | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |   } else { | 
					
						
							|  |  |  |     buttonNode = ( | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |       <div | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |         className={classNames( | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |           getClassName('__container'), | 
					
						
							| 
									
										
										
										
											2023-11-13 09:56:48 -05:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |           theme ? themeClassName(theme) : undefined | 
					
						
							| 
									
										
										
										
											2022-07-25 14:55:44 -04:00
										 |  |  |         )} | 
					
						
							|  |  |  |       > | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |         <button | 
					
						
							| 
									
										
										
										
											2023-03-29 17:03:25 -07:00
										 |  |  |           aria-label={ariaLabel || i18n('icu:ContextMenu--button')} | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |           className={classNames( | 
					
						
							|  |  |  |             getClassName('__button'), | 
					
						
							|  |  |  |             isMenuShowing ? getClassName('__button--active') : undefined | 
					
						
							|  |  |  |           )} | 
					
						
							|  |  |  |           onClick={onClick || handleClick} | 
					
						
							|  |  |  |           onContextMenu={handleClick} | 
					
						
							|  |  |  |           onKeyDown={handleKeyDown} | 
					
						
							|  |  |  |           ref={setReferenceElement} | 
					
						
							|  |  |  |           type="button" | 
					
						
							|  |  |  |         > | 
					
						
							|  |  |  |           {children} | 
					
						
							|  |  |  |         </button> | 
					
						
							| 
									
										
										
										
											2023-04-05 13:48:00 -07:00
										 |  |  |         {portalNode ? createPortal(menuNode, portalNode) : menuNode} | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |       </div> | 
					
						
							| 
									
										
										
										
											2022-10-12 10:14:35 -06:00
										 |  |  |     ); | 
					
						
							|  |  |  |   } | 
					
						
							| 
									
										
										
										
											2022-11-04 07:22:07 -06:00
										 |  |  |   return buttonNode; | 
					
						
							| 
									
										
										
										
											2021-12-13 19:15:24 -06:00
										 |  |  | } |