mirror of
https://github.com/labring/FastGPT.git
synced 2025-07-23 13:03:50 +00:00
192 lines
4.7 KiB
TypeScript
192 lines
4.7 KiB
TypeScript
import React, { useMemo, useRef, useState } from 'react';
|
|
import {
|
|
Menu,
|
|
MenuList,
|
|
MenuItem,
|
|
Box,
|
|
useOutsideClick,
|
|
MenuButton,
|
|
MenuItemProps
|
|
} from '@chakra-ui/react';
|
|
import MyIcon from '../Icon';
|
|
import MyDivider from '../MyDivider';
|
|
import type { IconNameType } from '../Icon/type';
|
|
|
|
export type MenuItemType = 'primary' | 'danger';
|
|
|
|
export type Props = {
|
|
width?: number | string;
|
|
offset?: [number, number];
|
|
Button: React.ReactNode;
|
|
trigger?: 'hover' | 'click';
|
|
iconSize?: string;
|
|
menuList: {
|
|
label?: string;
|
|
children: {
|
|
isActive?: boolean;
|
|
type?: MenuItemType;
|
|
icon?: IconNameType | string;
|
|
label: string | React.ReactNode;
|
|
description?: string;
|
|
onClick: () => any;
|
|
}[];
|
|
}[];
|
|
};
|
|
|
|
const MyMenu = ({
|
|
width = 'auto',
|
|
trigger = 'hover',
|
|
offset,
|
|
iconSize = '1rem',
|
|
Button,
|
|
menuList
|
|
}: Props) => {
|
|
const typeMapStyle: Record<MenuItemType, MenuItemProps> = {
|
|
primary: {
|
|
_hover: {
|
|
backgroundColor: 'primary.50',
|
|
color: 'primary.600'
|
|
},
|
|
_focus: {
|
|
backgroundColor: 'primary.50',
|
|
color: 'primary.600'
|
|
},
|
|
_active: {
|
|
backgroundColor: 'primary.50',
|
|
color: 'primary.600'
|
|
}
|
|
},
|
|
danger: {
|
|
color: 'red.600',
|
|
_hover: {
|
|
background: 'red.1'
|
|
},
|
|
_focus: {
|
|
background: 'red.1'
|
|
},
|
|
_active: {
|
|
background: 'red.1'
|
|
}
|
|
}
|
|
};
|
|
const menuItemStyles: MenuItemProps = {
|
|
borderRadius: 'sm',
|
|
py: 2,
|
|
px: 3,
|
|
display: 'flex',
|
|
alignItems: 'center',
|
|
fontSize: 'sm'
|
|
};
|
|
const ref = useRef<HTMLDivElement>(null);
|
|
const closeTimer = useRef<any>();
|
|
const [isOpen, setIsOpen] = useState(false);
|
|
|
|
useOutsideClick({
|
|
ref: ref,
|
|
handler: () => {
|
|
setIsOpen(false);
|
|
}
|
|
});
|
|
|
|
const computeOffset = useMemo<[number, number]>(() => {
|
|
if (offset) return offset;
|
|
if (typeof width === 'number') return [-width / 2, 5];
|
|
return [0, 5];
|
|
}, [offset]);
|
|
|
|
return (
|
|
<Menu
|
|
offset={computeOffset}
|
|
isOpen={isOpen}
|
|
autoSelect={false}
|
|
direction={'ltr'}
|
|
isLazy
|
|
lazyBehavior={'keepMounted'}
|
|
placement="bottom-start"
|
|
>
|
|
<Box
|
|
ref={ref}
|
|
onMouseEnter={() => {
|
|
if (trigger === 'hover') {
|
|
setIsOpen(true);
|
|
}
|
|
clearTimeout(closeTimer.current);
|
|
}}
|
|
onMouseLeave={() => {
|
|
if (trigger === 'hover') {
|
|
closeTimer.current = setTimeout(() => {
|
|
setIsOpen(false);
|
|
}, 100);
|
|
}
|
|
}}
|
|
>
|
|
<Box
|
|
position={'relative'}
|
|
onClickCapture={(e) => {
|
|
e.stopPropagation();
|
|
if (trigger === 'click') {
|
|
setIsOpen(!isOpen);
|
|
}
|
|
}}
|
|
>
|
|
<MenuButton
|
|
w={'100%'}
|
|
h={'100%'}
|
|
position={'absolute'}
|
|
top={0}
|
|
right={0}
|
|
bottom={0}
|
|
left={0}
|
|
/>
|
|
<Box position={'relative'}>{Button}</Box>
|
|
</Box>
|
|
<MenuList
|
|
minW={isOpen ? `${width}px !important` : '80px'}
|
|
maxW={'300px'}
|
|
p={'6px'}
|
|
border={'1px solid #fff'}
|
|
boxShadow={'3'}
|
|
>
|
|
{menuList.map((item, i) => {
|
|
return (
|
|
<Box key={i}>
|
|
{item.label && <Box fontSize={'sm'}>{item.label}</Box>}
|
|
{i !== 0 && <MyDivider h={'1.5px'} my={1} />}
|
|
{item.children.map((child, index) => (
|
|
<MenuItem
|
|
key={index}
|
|
{...menuItemStyles}
|
|
onClickCapture={(e) => {
|
|
e.stopPropagation();
|
|
setIsOpen(false);
|
|
child.onClick && child.onClick();
|
|
}}
|
|
color={child.isActive ? 'primary.700' : 'myGray.600'}
|
|
whiteSpace={'pre-wrap'}
|
|
_notLast={{ mb: 0.5 }}
|
|
{...typeMapStyle[child.type || 'primary']}
|
|
>
|
|
{!!child.icon && <MyIcon name={child.icon as any} w={iconSize} mr={3} />}
|
|
<Box>
|
|
<Box color={child.description ? 'myGray.900' : 'inherit'} fontSize={'sm'}>
|
|
{child.label}
|
|
</Box>
|
|
{child.description && (
|
|
<Box color={'myGray.500'} fontSize={'mini'}>
|
|
{child.description}
|
|
</Box>
|
|
)}
|
|
</Box>
|
|
</MenuItem>
|
|
))}
|
|
</Box>
|
|
);
|
|
})}
|
|
</MenuList>
|
|
</Box>
|
|
</Menu>
|
|
);
|
|
};
|
|
|
|
export default React.memo(MyMenu);
|