import type { JSX } from 'solid-js'; import { createComponent, createEffect, mergeProps } from 'solid-js'; import { omitProps } from 'solid-use/props'; import type { DisclosureStateRenderProps } from '../../states/create-disclosure-state'; import { DisclosureStateChild, useDisclosureState, } from '../../states/create-disclosure-state'; import createDynamic from '../../utils/create-dynamic'; import type { UnmountableProps } from '../../utils/create-unmountable'; import { createUnmountable } from '../../utils/create-unmountable'; import type { DynamicProps, HeadlessPropsWithRef, ValidConstructor, } from '../../utils/dynamic-prop'; import { createForwardRef } from '../../utils/dynamic-prop'; import { focusFirst, lockFocus } from '../../utils/focus-navigation'; import getFocusableElements from '../../utils/focus-query'; import { createDisabledState, createExpandedState, } from '../../utils/state-props'; import type { Prettify } from '../../utils/types'; import useEventListener from '../../utils/use-event-listener'; import { useContextMenuContext } from './ContextMenuContext'; import { CONTEXT_MENU_PANEL_TAG } from './tags'; export type ContextMenuPanelBaseProps = Prettify< DisclosureStateRenderProps & UnmountableProps >; export type ContextMenuPanelProps = HeadlessPropsWithRef; export function ContextMenuPanel( props: ContextMenuPanelProps, ): JSX.Element { const context = useContextMenuContext('ContextMenuPanel'); const state = useDisclosureState(); const [internalRef, setInternalRef] = createForwardRef(props); createEffect(() => { const current = internalRef(); if (current instanceof HTMLElement) { if (state.isOpen()) { focusFirst(getFocusableElements(current), false); useEventListener(current, 'keydown', e => { if (!props.disabled) { switch (e.key) { case 'Tab': { e.preventDefault(); lockFocus(current, e.shiftKey, false); break; } case 'Escape': { state.close(); break; } default: break; } } }); useEventListener(document, 'click', e => { if (!current.contains(e.target as Node)) { state.close(); } }); } } }); return createUnmountable( props, () => state.isOpen(), () => createDynamic( () => props.as || ('div' as T), mergeProps( CONTEXT_MENU_PANEL_TAG, { id: context.panelID, ref: setInternalRef, get children() { return createComponent(DisclosureStateChild, { get children() { return props.children; }, }); }, }, createDisabledState(() => state.disabled()), createExpandedState(() => state.isOpen()), omitProps(props, ['as', 'unmount', 'children', 'ref']), ) as DynamicProps, ), ); }