forked from processing/p5.js-web-editor
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathMenubarMenu.jsx
102 lines (82 loc) · 2.67 KB
/
MenubarMenu.jsx
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
// https://blog.logrocket.com/building-accessible-menubar-component-react
import classNames from 'classnames';
import PropTypes from 'prop-types';
import React, { useContext, useMemo } from 'react';
import TriangleIcon from '../../images/down-filled-triangle.svg';
import { MenuOpenContext, MenubarContext, ParentMenuContext } from './contexts';
export function useMenuProps(id) {
const activeMenu = useContext(MenuOpenContext);
const isOpen = id === activeMenu;
const { createMenuHandlers } = useContext(MenubarContext);
const handlers = useMemo(() => createMenuHandlers(id), [
createMenuHandlers,
id
]);
return { isOpen, handlers };
}
/* -------------------------------------------------------------------------------------------------
* MenubarTrigger
* -----------------------------------------------------------------------------------------------*/
function MenubarTrigger({ id, title, ...props }) {
const { isOpen, handlers } = useMenuProps(id);
return (
<button
{...handlers}
{...props}
role="menuitem"
aria-haspopup="menu"
aria-expanded={isOpen}
>
<span className="nav__item-header">{title}</span>
<TriangleIcon
className="nav__item-header-triangle"
focusable="false"
aria-hidden="true"
/>
</button>
);
}
MenubarTrigger.propTypes = {
id: PropTypes.string.isRequired,
title: PropTypes.node.isRequired
};
/* -------------------------------------------------------------------------------------------------
* MenubarList
* -----------------------------------------------------------------------------------------------*/
function MenubarList({ id, children }) {
return (
<ul className="nav__dropdown" role="menu">
<ParentMenuContext.Provider value={id}>
{children}
</ParentMenuContext.Provider>
</ul>
);
}
MenubarList.propTypes = {
id: PropTypes.string.isRequired,
children: PropTypes.node
};
MenubarList.defaultProps = {
children: null
};
/* -------------------------------------------------------------------------------------------------
* MenubarMenu
* -----------------------------------------------------------------------------------------------*/
function MenubarMenu({ id, title, children }) {
const { isOpen } = useMenuProps(id);
return (
<li className={classNames('nav__item', isOpen && 'nav__item--open')}>
<MenubarTrigger id={id} title={title} />
<MenubarList id={id}>{children}</MenubarList>
</li>
);
}
MenubarMenu.propTypes = {
id: PropTypes.string.isRequired,
title: PropTypes.node.isRequired,
children: PropTypes.node
};
MenubarMenu.defaultProps = {
children: null
};
export default MenubarMenu;