mirror of
https://github.com/Stremio/stremio-web.git
synced 2026-04-14 04:50:26 +00:00
84 lines
2.5 KiB
TypeScript
84 lines
2.5 KiB
TypeScript
// 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<HTMLDivElement>(null);
|
|
|
|
const [menus, setMenus] = useState<Menu[]>([]);
|
|
const [style, setStyle] = useState<React.CSSProperties>(null);
|
|
|
|
const active = useMemo(() => menus.find(({ open }) => open), [menus]);
|
|
|
|
const create = (menu: Omit<Menu, 'open'>) => {
|
|
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 (
|
|
<MenuContext.Provider value={{ active, create, remove, render, toggle }}>
|
|
{children}
|
|
<div ref={container} className={classNames(className, active?.className)} style={style} />
|
|
</MenuContext.Provider>
|
|
);
|
|
};
|
|
|
|
export default MenuProvider;
|