import * as React from 'react'; import omit from '@rc-component/util/lib/omit'; import classNames from 'classnames'; import { SubMenu as RcSubMenu, useFullPath } from 'rc-menu'; import { useZIndex } from '../_util/hooks/useZIndex'; import { cloneElement } from '../_util/reactNode'; import type { SubMenuType } from './interface'; import type { MenuContextProps } from './MenuContext'; import MenuContext from './MenuContext'; export interface SubMenuProps extends Omit { title?: React.ReactNode; children?: React.ReactNode; } const SubMenu: React.FC = (props) => { const { popupClassName, icon, title, theme: customTheme } = props; const context = React.useContext(MenuContext); const { prefixCls, inlineCollapsed, theme: contextTheme } = context; const parentPath = useFullPath(); let titleNode: React.ReactNode; if (!icon) { titleNode = inlineCollapsed && !parentPath.length && title && typeof title === 'string' ? (
{title.charAt(0)}
) : ( {title} ); } else { // inline-collapsed.md demo 依赖 span 来隐藏文字,有 icon 属性,则内部包裹一个 span // ref: https://github.com/ant-design/ant-design/pull/23456 const titleIsSpan = React.isValidElement(title) && title.type === 'span'; titleNode = ( <> {cloneElement(icon, { className: classNames( React.isValidElement(icon) ? (icon as React.ReactElement<{ className?: string }>).props?.className : '', `${prefixCls}-item-icon`, ), })} {titleIsSpan ? title : {title}} ); } const contextValue = React.useMemo( () => ({ ...context, firstLevel: false }), [context], ); // ============================ zIndex ============================ const [zIndex] = useZIndex('Menu'); return ( ); }; export default SubMenu;