// Copyright (C) 2017-2024 Smart code 203358507 import React, { useLayoutEffect, useMemo, useRef, useState } from 'react'; import { createPortal } from 'react-dom'; import classNames from 'classnames'; import useKeyboardEvent from '../useKeyboardEvent'; import useOnClickOutside from '../useOnClickOutside'; import MenuContext from './MenuContext'; import { type Menu } from './types'; type Props = { children: JSX.Element, className: string, }; const MenuProvider = ({ children, className }: Props) => { const container = useRef(null); const [menus, setMenus] = useState([]); const [style, setStyle] = useState(null); const active = useMemo(() => menus.find(({ open }) => open), [menus]); const create = (menu: Omit) => { setMenus((menus) => ([ ...menus, { ...menu, open: false, }, ])); }; const remove = (id: string) => { setMenus((menus) => menus.filter((menu) => menu.id !== id)); }; const render = (content: JSX.Element) => { return createPortal(content, container.current); }; const toggle = (id: string) => { setMenus((menus) => menus.map((menu) => ({ ...menu, open: menu.id === id && !menu.open, }))); }; const closeAll = () => { setMenus((menus) => menus.map((menu) => ({ ...menu, open: false, }))); }; useLayoutEffect(() => { setStyle(null); if (container.current && active) { const menu = container.current.getBoundingClientRect(); const parent = active.parent.getBoundingClientRect(); const y = (parent.top + menu.height) < window.innerHeight ? 'bottom' : 'top'; const x = (parent.left + menu.width) < window.innerWidth ? 'left' : 'right'; setStyle({ top: y === 'top' ? parent.top - menu.height : parent.bottom, left: (active.align ?? x) === 'left' ? parent.left : parent.right - menu.width, }); } }, [active]); useKeyboardEvent('Escape', closeAll); useOnClickOutside([container.current, active?.parent], closeAll); return ( {children}
); }; export default MenuProvider;