UNPKG

@react-aria/menu

Version:
87 lines (78 loc) 3.03 kB
/* * Copyright 2020 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 {AriaMenuProps} from '@react-types/menu'; import {DOMAttributes, KeyboardDelegate, KeyboardEvents, RefObject} from '@react-types/shared'; import {filterDOMProps, mergeProps} from '@react-aria/utils'; import {menuData} from './utils'; import {TreeState} from '@react-stately/tree'; import {useSelectableList} from '@react-aria/selection'; export interface MenuAria { /** Props for the menu element. */ menuProps: DOMAttributes } export interface AriaMenuOptions<T> extends Omit<AriaMenuProps<T>, 'children'>, KeyboardEvents { /** Whether the menu uses virtual scrolling. */ isVirtualized?: boolean, /** * An optional keyboard delegate implementation for type to select, * to override the default. */ keyboardDelegate?: KeyboardDelegate, /** * Whether the menu items should use virtual focus instead of being focused directly. */ shouldUseVirtualFocus?: boolean } /** * Provides the behavior and accessibility implementation for a menu component. * A menu displays a list of actions or options that a user can choose. * @param props - Props for the menu. * @param state - State for the menu, as returned by `useListState`. */ export function useMenu<T>(props: AriaMenuOptions<T>, state: TreeState<T>, ref: RefObject<HTMLElement | null>): MenuAria { let { shouldFocusWrap = true, onKeyDown, onKeyUp, ...otherProps } = props; if (!props['aria-label'] && !props['aria-labelledby'] && process.env.NODE_ENV !== 'production') { console.warn('An aria-label or aria-labelledby prop is required for accessibility.'); } let domProps = filterDOMProps(props, {labelable: true}); let {listProps} = useSelectableList({ ...otherProps, ref, selectionManager: state.selectionManager, collection: state.collection, disabledKeys: state.disabledKeys, shouldFocusWrap, linkBehavior: 'override' }); menuData.set(state, { onClose: props.onClose, onAction: props.onAction, shouldUseVirtualFocus: props.shouldUseVirtualFocus }); return { menuProps: mergeProps(domProps, {onKeyDown, onKeyUp}, { role: 'menu', ...listProps, onKeyDown: (e) => { // don't clear the menu selected keys if the user is presses escape since escape closes the menu if (e.key !== 'Escape' || props.shouldUseVirtualFocus) { listProps.onKeyDown?.(e); } } }) }; }