import * as React from 'react'; import { ItemGroup } from 'rc-menu'; import type { MenuItemType as RcMenuItemType, MenuDividerType as RcMenuDividerType, SubMenuType as RcSubMenuType, MenuItemGroupType as RcMenuItemGroupType, } from 'rc-menu/lib/interface'; import SubMenu from '../SubMenu'; import MenuDivider from '../MenuDivider'; import MenuItem from '../MenuItem'; interface MenuItemType extends RcMenuItemType { danger?: boolean; icon?: React.ReactNode; title?: string; } interface SubMenuType extends Omit { icon?: React.ReactNode; theme?: 'dark' | 'light'; children: ItemType[]; } interface MenuItemGroupType extends Omit { children?: ItemType[]; key?: React.Key; } interface MenuDividerType extends RcMenuDividerType { dashed?: boolean; key?: React.Key; } export type ItemType = MenuItemType | SubMenuType | MenuItemGroupType | MenuDividerType | null; function convertItemsToNodes(list: ItemType[]) { return (list || []) .map((opt, index) => { if (opt && typeof opt === 'object') { const { label, children, key, type, ...restProps } = opt as any; const mergedKey = key ?? `tmp-${index}`; // MenuItemGroup & SubMenuItem if (children || type === 'group') { if (type === 'group') { // Group return ( {convertItemsToNodes(children)} ); } // Sub Menu return ( {convertItemsToNodes(children)} ); } // MenuItem & Divider if (type === 'divider') { return ; } return ( {label} ); } return null; }) .filter(opt => opt); } // FIXME: Move logic here in v5 /** * We simply convert `items` to ReactNode for reuse origin component logic. But we need move all the * logic from component into this hooks when in v5 */ export default function useItems(items?: ItemType[]) { return React.useMemo(() => { if (!items) { return items; } return convertItemsToNodes(items); }, [items]); }