@react-aria/menu
Version:
Spectrum UI components in React
241 lines (217 loc) • 7.74 kB
text/typescript
/*
* Copyright 2023 Adobe. All rights reserved.
* This file is licensed to you under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License. You may obtain a copy
* of the License at http://www.apache.org/licenses/LICENSE-2.0
*
* Unless required by applicable law or agreed to in writing, software distributed under
* the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR REPRESENTATIONS
* OF ANY KIND, either express or implied. See the License for the specific language
* governing permissions and limitations under the License.
*/
import {AriaMenuItemProps} from './useMenuItem';
import {AriaMenuOptions} from './useMenu';
import type {AriaPopoverProps, OverlayProps} from '@react-aria/overlays';
import {FocusableElement, FocusStrategy, KeyboardEvent, PressEvent, Node as RSNode} from '@react-types/shared';
import {RefObject, useCallback, useRef} from 'react';
import type {SubmenuTriggerState} from '@react-stately/menu';
import {useEffectEvent, useId, useLayoutEffect} from '@react-aria/utils';
import {useLocale} from '@react-aria/i18n';
import {useSafelyMouseToSubmenu} from './useSafelyMouseToSubmenu';
export interface AriaSubmenuTriggerProps {
/** An object representing the submenu trigger menu item. Contains all the relevant information that makes up the menu item. */
node: RSNode<unknown>,
/** Whether the submenu trigger is disabled. */
isDisabled?: boolean,
/** The type of the contents that the submenu trigger opens. */
type?: 'dialog' | 'menu',
/** Ref of the menu that contains the submenu trigger. */
parentMenuRef: RefObject<HTMLElement>,
/** Ref of the submenu opened by the submenu trigger. */
submenuRef: RefObject<HTMLElement>,
/**
* The delay time in milliseconds for the submenu to appear after hovering over the trigger.
* @default 200
*/
delay?: number
}
interface SubmenuTriggerProps extends AriaMenuItemProps {
/** Whether the submenu trigger is in an expanded state. */
isOpen: boolean
}
interface SubmenuProps<T> extends AriaMenuOptions<T> {
/** The level of the submenu. */
submenuLevel: number
}
export interface SubmenuTriggerAria<T> {
/** Props for the submenu trigger menu item. */
submenuTriggerProps: SubmenuTriggerProps,
/** Props for the submenu controlled by the submenu trigger menu item. */
submenuProps: SubmenuProps<T>,
/** Props for the submenu's popover container. */
popoverProps: Pick<AriaPopoverProps, 'isNonModal' | 'shouldCloseOnInteractOutside'> & Pick<OverlayProps, 'disableFocusManagement'>
}
/**
* Provides the behavior and accessibility implementation for a submenu trigger and its associated submenu.
* @param props - Props for the submenu trigger and refs attach to its submenu and parent menu.
* @param state - State for the submenu trigger.
* @param ref - Ref to the submenu trigger element.
*/
export function useSubmenuTrigger<T>(props: AriaSubmenuTriggerProps, state: SubmenuTriggerState, ref: RefObject<FocusableElement>): SubmenuTriggerAria<T> {
let {parentMenuRef, submenuRef, type = 'menu', isDisabled, node, delay = 200} = props;
let submenuTriggerId = useId();
let overlayId = useId();
let {direction} = useLocale();
let openTimeout = useRef<ReturnType<typeof setTimeout> | undefined>();
let cancelOpenTimeout = useCallback(() => {
if (openTimeout.current) {
clearTimeout(openTimeout.current);
openTimeout.current = undefined;
}
}, [openTimeout]);
let onSubmenuOpen = useEffectEvent((focusStrategy?: FocusStrategy) => {
cancelOpenTimeout();
state.open(focusStrategy);
});
let onSubmenuClose = useEffectEvent(() => {
cancelOpenTimeout();
state.close();
});
useLayoutEffect(() => {
return () => {
cancelOpenTimeout();
};
}, [cancelOpenTimeout]);
let submenuKeyDown = (e: KeyboardEvent) => {
switch (e.key) {
case 'ArrowLeft':
if (direction === 'ltr' && e.currentTarget.contains(e.target as Element)) {
e.stopPropagation();
onSubmenuClose();
ref.current.focus();
}
break;
case 'ArrowRight':
if (direction === 'rtl' && e.currentTarget.contains(e.target as Element)) {
e.stopPropagation();
onSubmenuClose();
ref.current.focus();
}
break;
case 'Escape':
e.stopPropagation();
state.closeAll();
break;
}
};
let submenuProps = {
id: overlayId,
'aria-label': node.textValue,
submenuLevel: state.submenuLevel,
...(type === 'menu' && {
onClose: state.closeAll,
autoFocus: state.focusStrategy,
onKeyDown: submenuKeyDown
})
};
let submenuTriggerKeyDown = (e: KeyboardEvent) => {
switch (e.key) {
case 'ArrowRight':
if (!isDisabled) {
if (direction === 'ltr') {
if (!state.isOpen) {
onSubmenuOpen('first');
}
if (type === 'menu' && !!submenuRef?.current && document.activeElement === ref?.current) {
submenuRef.current.focus();
}
} else if (state.isOpen) {
onSubmenuClose();
} else {
e.continuePropagation();
}
}
break;
case 'ArrowLeft':
if (!isDisabled) {
if (direction === 'rtl') {
if (!state.isOpen) {
onSubmenuOpen('first');
}
if (type === 'menu' && !!submenuRef?.current && document.activeElement === ref?.current) {
submenuRef.current.focus();
}
} else if (state.isOpen) {
onSubmenuClose();
} else {
e.continuePropagation();
}
}
break;
case 'Escape':
state.closeAll();
break;
default:
e.continuePropagation();
break;
}
};
let onPressStart = (e: PressEvent) => {
if (!isDisabled && (e.pointerType === 'virtual' || e.pointerType === 'keyboard')) {
// If opened with a screen reader or keyboard, auto focus the first submenu item.
onSubmenuOpen('first');
}
};
let onPress = (e: PressEvent) => {
if (!isDisabled && (e.pointerType === 'touch' || e.pointerType === 'mouse')) {
// For touch or on a desktop device with a small screen open on press up to possible problems with
// press up happening on the newly opened tray items
onSubmenuOpen();
}
};
let onHoverChange = (isHovered) => {
if (!isDisabled) {
if (isHovered && !state.isOpen) {
if (!openTimeout.current) {
openTimeout.current = setTimeout(() => {
onSubmenuOpen();
}, delay);
}
} else if (!isHovered) {
cancelOpenTimeout();
}
}
};
let onBlur = (e) => {
if (state.isOpen && parentMenuRef.current.contains(e.relatedTarget)) {
onSubmenuClose();
}
};
let shouldCloseOnInteractOutside = (target) => {
if (target !== ref.current) {
return true;
}
return false;
};
useSafelyMouseToSubmenu({menuRef: parentMenuRef, submenuRef, isOpen: state.isOpen, isDisabled: isDisabled});
return {
submenuTriggerProps: {
id: submenuTriggerId,
'aria-controls': state.isOpen ? overlayId : undefined,
'aria-haspopup': !isDisabled ? type : undefined,
'aria-expanded': state.isOpen ? 'true' : 'false',
onPressStart,
onPress,
onHoverChange,
onKeyDown: submenuTriggerKeyDown,
onBlur,
isOpen: state.isOpen
},
submenuProps,
popoverProps: {
isNonModal: true,
disableFocusManagement: true,
shouldCloseOnInteractOutside
}
};
}