2023-01-03 16:55:26 +00:00
|
|
|
/*
|
|
|
|
Copyright 2022 New Vector Ltd
|
|
|
|
|
|
|
|
Licensed 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 CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
2023-02-02 14:32:44 +00:00
|
|
|
import React, { Key, useRef, useState } from "react";
|
2022-07-30 07:51:32 +00:00
|
|
|
import { AriaMenuOptions, useMenu, useMenuItem } from "@react-aria/menu";
|
|
|
|
import { TreeState, useTreeState } from "@react-stately/tree";
|
2021-12-07 01:34:10 +00:00
|
|
|
import { mergeProps } from "@react-aria/utils";
|
|
|
|
import { useFocus } from "@react-aria/interactions";
|
|
|
|
import classNames from "classnames";
|
2022-07-30 07:51:32 +00:00
|
|
|
import { Node } from "@react-types/shared";
|
|
|
|
|
|
|
|
import styles from "./Menu.module.css";
|
2021-12-07 01:34:10 +00:00
|
|
|
|
2022-07-30 07:51:32 +00:00
|
|
|
interface MenuProps<T> extends AriaMenuOptions<T> {
|
2023-02-02 14:32:44 +00:00
|
|
|
className?: String;
|
2022-08-01 22:46:16 +00:00
|
|
|
onClose?: () => void;
|
|
|
|
onAction: (value: Key) => void;
|
|
|
|
label?: string;
|
2022-07-30 07:51:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
export function Menu<T extends object>({
|
|
|
|
className,
|
|
|
|
onAction,
|
|
|
|
onClose,
|
2022-08-01 22:46:16 +00:00
|
|
|
label,
|
2022-07-30 07:51:32 +00:00
|
|
|
...rest
|
|
|
|
}: MenuProps<T>) {
|
2023-02-02 14:32:44 +00:00
|
|
|
const state = useTreeState<T>({ ...rest, selectionMode: "none" });
|
2021-12-07 01:34:10 +00:00
|
|
|
const menuRef = useRef();
|
2022-07-30 07:51:32 +00:00
|
|
|
const { menuProps } = useMenu<T>(rest, state, menuRef);
|
2021-12-07 01:34:10 +00:00
|
|
|
|
|
|
|
return (
|
|
|
|
<ul
|
|
|
|
{...mergeProps(menuProps, rest)}
|
|
|
|
ref={menuRef}
|
|
|
|
className={classNames(styles.menu, className)}
|
|
|
|
>
|
|
|
|
{[...state.collection].map((item) => (
|
|
|
|
<MenuItem
|
|
|
|
key={item.key}
|
|
|
|
item={item}
|
|
|
|
state={state}
|
|
|
|
onAction={onAction}
|
2022-07-30 07:51:32 +00:00
|
|
|
onClose={onClose}
|
2021-12-07 01:34:10 +00:00
|
|
|
/>
|
|
|
|
))}
|
|
|
|
</ul>
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
2022-07-30 07:51:32 +00:00
|
|
|
interface MenuItemProps<T> {
|
|
|
|
item: Node<T>;
|
|
|
|
state: TreeState<T>;
|
2022-08-01 22:46:16 +00:00
|
|
|
onAction: (value: Key) => void;
|
2022-07-30 07:51:32 +00:00
|
|
|
onClose: () => void;
|
|
|
|
}
|
|
|
|
|
|
|
|
function MenuItem<T>({ item, state, onAction, onClose }: MenuItemProps<T>) {
|
2021-12-07 01:34:10 +00:00
|
|
|
const ref = useRef();
|
|
|
|
const { menuItemProps } = useMenuItem(
|
|
|
|
{
|
|
|
|
key: item.key,
|
|
|
|
onAction,
|
|
|
|
onClose,
|
|
|
|
},
|
|
|
|
state,
|
|
|
|
ref
|
|
|
|
);
|
|
|
|
|
|
|
|
const [isFocused, setFocused] = useState(false);
|
|
|
|
const { focusProps } = useFocus({ onFocusChange: setFocused });
|
|
|
|
|
|
|
|
return (
|
|
|
|
<li
|
|
|
|
{...mergeProps(menuItemProps, focusProps)}
|
|
|
|
ref={ref}
|
|
|
|
className={classNames(styles.menuItem, {
|
|
|
|
[styles.focused]: isFocused,
|
|
|
|
})}
|
|
|
|
>
|
|
|
|
{item.rendered}
|
|
|
|
</li>
|
|
|
|
);
|
|
|
|
}
|