-
Notifications
You must be signed in to change notification settings - Fork 0
/
menu-toggle.tsx
74 lines (64 loc) · 1.59 KB
/
menu-toggle.tsx
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
'use client'
import {
ButtonHTMLAttributes,
PropsWithChildren,
ReactElement,
useCallback,
useContext,
useEffect,
useState,
} from 'react'
import { NavContext } from '../context'
import { Button } from '../components'
import { extendClass } from '../utils'
interface Props
extends PropsWithChildren<ButtonHTMLAttributes<HTMLButtonElement>> {
'aria-controls': string
className?: string
label?: string
closeLabel?: string
renderIcon?: (navOpen: boolean) => ReactElement
}
const MenuToggle = ({
'aria-controls': ariaControls,
className = '',
label = 'Open Nav',
closeLabel = 'Close Nav',
renderIcon = undefined,
...props
}: Props) => {
const { navOpen, toggleNav } = useContext(NavContext)
const [icon, setIcon] = useState<ReactElement | string>(navOpen ? '×' : '꠵')
useEffect(() => {
if (renderIcon) {
setIcon(renderIcon(navOpen))
return
}
setIcon(navOpen ? '×' : '꠵')
}, [navOpen, renderIcon])
const handleClick = useCallback(() => {
if (
document &&
document?.activeElement instanceof Element &&
'blur' in document?.activeElement
) {
;(document.activeElement as HTMLElement)?.blur()
}
toggleNav()
}, [toggleNav])
return (
<Button
className={`menu-toggle ${className}`}
onClick={handleClick}
aria-expanded={navOpen}
aria-controls={ariaControls}
label_a11y={navOpen ? closeLabel : label}
{...props}
>
<span className={`menu-toggle__icon ${extendClass(className, 'icon')}`}>
{icon}
</span>
</Button>
)
}
export default MenuToggle