mirror of
https://github.com/ant-design/ant-design.git
synced 2025-06-16 23:21:00 +08:00

* feat: ConfigProvider support classNames and styles for menu * fix * fix type * demo opt * fix * fix * chore: fix ts * chore: fix ts * chore: must fill * chore: other filled * chore: adjust logic * chore: fix cov --------- Co-authored-by: 二货机器人 <smith3816@gmail.com>
266 lines
8.3 KiB
TypeScript
266 lines
8.3 KiB
TypeScript
import * as React from 'react';
|
|
import { forwardRef } from 'react';
|
|
import EllipsisOutlined from '@ant-design/icons/EllipsisOutlined';
|
|
import type { MenuProps as RcMenuProps, MenuRef as RcMenuRef } from '@rc-component/menu';
|
|
import RcMenu from '@rc-component/menu';
|
|
import useEvent from '@rc-component/util/lib/hooks/useEvent';
|
|
import omit from '@rc-component/util/lib/omit';
|
|
import cls from 'classnames';
|
|
|
|
import useMergeSemantic from '../_util/hooks/useMergeSemantic';
|
|
import initCollapseMotion from '../_util/motion';
|
|
import { cloneElement } from '../_util/reactNode';
|
|
import type { GetProp } from '../_util/type';
|
|
import { devUseWarning } from '../_util/warning';
|
|
import { ConfigContext } from '../config-provider';
|
|
import { useComponentConfig } from '../config-provider/context';
|
|
import useCSSVarCls from '../config-provider/hooks/useCSSVarCls';
|
|
import type { SiderContextProps } from '../layout/Sider';
|
|
import type { ItemType } from './interface';
|
|
import type { MenuContextProps, MenuTheme } from './MenuContext';
|
|
import MenuContext from './MenuContext';
|
|
import Divider from './MenuDivider';
|
|
import MenuItem from './MenuItem';
|
|
import OverrideContext from './OverrideContext';
|
|
import useStyle from './style';
|
|
import SubMenu from './SubMenu';
|
|
|
|
function isEmptyIcon(icon?: React.ReactNode) {
|
|
return icon === null || icon === false;
|
|
}
|
|
|
|
const MENU_COMPONENTS: GetProp<RcMenuProps, '_internalComponents'> = {
|
|
item: MenuItem,
|
|
submenu: SubMenu,
|
|
divider: Divider,
|
|
};
|
|
|
|
export type SemanticName = 'root' | 'itemTitle' | 'list' | 'item' | 'itemIcon' | 'itemContent';
|
|
|
|
export type SubMenuName = 'item' | 'itemTitle' | 'list' | 'itemContent' | 'itemIcon';
|
|
|
|
export interface MenuProps
|
|
extends Omit<RcMenuProps, 'items' | '_internalComponents' | 'classNames' | 'styles'> {
|
|
theme?: MenuTheme;
|
|
inlineIndent?: number;
|
|
|
|
// >>>>> Private
|
|
/**
|
|
* @private Internal Usage. Not promise crash if used in production. Connect with chenshuai2144
|
|
* for removing.
|
|
*/
|
|
_internalDisableMenuItemTitleTooltip?: boolean;
|
|
|
|
items?: ItemType[];
|
|
classNames?: Partial<
|
|
Record<SemanticName, string> & {
|
|
popup?: string | { root?: string };
|
|
subMenu?: Partial<Record<SubMenuName, string>>;
|
|
}
|
|
>;
|
|
styles?: Partial<
|
|
Record<SemanticName, React.CSSProperties> & {
|
|
subMenu?: Partial<Record<SubMenuName, React.CSSProperties>>;
|
|
popup?: { root?: React.CSSProperties };
|
|
}
|
|
>;
|
|
}
|
|
|
|
type InternalMenuProps = MenuProps &
|
|
SiderContextProps & {
|
|
collapsedWidth?: string | number;
|
|
};
|
|
|
|
const InternalMenu = forwardRef<RcMenuRef, InternalMenuProps>((props, ref) => {
|
|
const override = React.useContext(OverrideContext);
|
|
const overrideObj = override || {};
|
|
|
|
const {
|
|
prefixCls: customizePrefixCls,
|
|
className,
|
|
style,
|
|
theme = 'light',
|
|
expandIcon,
|
|
_internalDisableMenuItemTitleTooltip,
|
|
inlineCollapsed,
|
|
siderCollapsed,
|
|
rootClassName,
|
|
mode,
|
|
selectable,
|
|
onClick,
|
|
overflowedIndicatorPopupClassName,
|
|
classNames,
|
|
styles,
|
|
...restProps
|
|
} = props;
|
|
|
|
const { menu } = React.useContext(ConfigContext);
|
|
|
|
const {
|
|
getPrefixCls,
|
|
getPopupContainer,
|
|
direction,
|
|
className: contextClassName,
|
|
style: contextStyle,
|
|
classNames: contextClassNames,
|
|
styles: contextStyles,
|
|
} = useComponentConfig('menu');
|
|
|
|
const [mergedClassNames, mergedStyles] = useMergeSemantic(
|
|
[contextClassNames, classNames],
|
|
[contextStyles, styles],
|
|
{
|
|
popup: {
|
|
_default: 'root',
|
|
},
|
|
subMenu: {
|
|
_default: 'root',
|
|
},
|
|
},
|
|
) as [MenuContextProps['classNames'], MenuContextProps['styles']];
|
|
|
|
const rootPrefixCls = getPrefixCls();
|
|
|
|
const passedProps = omit(restProps, ['collapsedWidth']);
|
|
|
|
// ======================== Warning ==========================
|
|
if (process.env.NODE_ENV !== 'production') {
|
|
const warning = devUseWarning('Menu');
|
|
|
|
warning(
|
|
!('inlineCollapsed' in props && mode !== 'inline'),
|
|
'usage',
|
|
'`inlineCollapsed` should only be used when `mode` is inline.',
|
|
);
|
|
warning.deprecated('items' in props && !props.children, 'children', 'items');
|
|
}
|
|
|
|
overrideObj.validator?.({ mode });
|
|
|
|
// ========================== Click ==========================
|
|
// Tell dropdown that item clicked
|
|
const onItemClick = useEvent<Required<MenuProps>['onClick']>((...args) => {
|
|
onClick?.(...args);
|
|
overrideObj.onClick?.();
|
|
});
|
|
|
|
// ========================== Mode ===========================
|
|
const mergedMode = overrideObj.mode || mode;
|
|
|
|
// ======================= Selectable ========================
|
|
const mergedSelectable = selectable ?? overrideObj.selectable;
|
|
|
|
// ======================== Collapsed ========================
|
|
// Inline Collapsed
|
|
const mergedInlineCollapsed = inlineCollapsed ?? siderCollapsed;
|
|
|
|
const defaultMotions: MenuProps['defaultMotions'] = {
|
|
horizontal: { motionName: `${rootPrefixCls}-slide-up` },
|
|
inline: initCollapseMotion(rootPrefixCls),
|
|
other: { motionName: `${rootPrefixCls}-zoom-big` },
|
|
};
|
|
|
|
const prefixCls = getPrefixCls('menu', customizePrefixCls || overrideObj.prefixCls);
|
|
const rootCls = useCSSVarCls(prefixCls);
|
|
const [hashId, cssVarCls] = useStyle(prefixCls, rootCls, !override);
|
|
const menuClassName = cls(`${prefixCls}-${theme}`, contextClassName, className);
|
|
|
|
// ====================== ExpandIcon ========================
|
|
const mergedExpandIcon = React.useMemo<MenuProps['expandIcon']>(() => {
|
|
if (typeof expandIcon === 'function' || isEmptyIcon(expandIcon)) {
|
|
return expandIcon || null;
|
|
}
|
|
if (typeof overrideObj.expandIcon === 'function' || isEmptyIcon(overrideObj.expandIcon)) {
|
|
return overrideObj.expandIcon || null;
|
|
}
|
|
if (typeof menu?.expandIcon === 'function' || isEmptyIcon(menu?.expandIcon)) {
|
|
return menu?.expandIcon || null;
|
|
}
|
|
const mergedIcon = expandIcon ?? overrideObj?.expandIcon ?? menu?.expandIcon;
|
|
return cloneElement(mergedIcon, {
|
|
className: cls(
|
|
`${prefixCls}-submenu-expand-icon`,
|
|
React.isValidElement<any>(mergedIcon)
|
|
? (
|
|
mergedIcon as React.ReactElement<{
|
|
className?: string;
|
|
}>
|
|
).props?.className
|
|
: undefined,
|
|
),
|
|
});
|
|
}, [expandIcon, overrideObj?.expandIcon, menu?.expandIcon, prefixCls]);
|
|
|
|
// ======================== Context ==========================
|
|
const contextValue = React.useMemo<MenuContextProps>(
|
|
() => ({
|
|
prefixCls,
|
|
inlineCollapsed: mergedInlineCollapsed || false,
|
|
direction,
|
|
firstLevel: true,
|
|
theme,
|
|
mode: mergedMode,
|
|
disableMenuItemTitleTooltip: _internalDisableMenuItemTitleTooltip,
|
|
classNames: mergedClassNames,
|
|
styles: mergedStyles,
|
|
}),
|
|
[
|
|
prefixCls,
|
|
mergedInlineCollapsed,
|
|
direction,
|
|
_internalDisableMenuItemTitleTooltip,
|
|
theme,
|
|
mergedClassNames,
|
|
mergedStyles,
|
|
],
|
|
);
|
|
|
|
// ========================= Render ==========================
|
|
return (
|
|
<OverrideContext.Provider value={null}>
|
|
<MenuContext.Provider value={contextValue}>
|
|
<RcMenu
|
|
getPopupContainer={getPopupContainer}
|
|
overflowedIndicator={<EllipsisOutlined />}
|
|
overflowedIndicatorPopupClassName={cls(
|
|
prefixCls,
|
|
`${prefixCls}-${theme}`,
|
|
overflowedIndicatorPopupClassName,
|
|
)}
|
|
classNames={{
|
|
list: mergedClassNames.list,
|
|
listTitle: mergedClassNames.itemTitle,
|
|
}}
|
|
styles={{
|
|
list: mergedStyles.list,
|
|
listTitle: mergedStyles.itemTitle,
|
|
}}
|
|
mode={mergedMode}
|
|
selectable={mergedSelectable}
|
|
onClick={onItemClick}
|
|
{...passedProps}
|
|
inlineCollapsed={mergedInlineCollapsed}
|
|
style={{ ...mergedStyles.root, ...contextStyle, ...style }}
|
|
className={menuClassName}
|
|
prefixCls={prefixCls}
|
|
direction={direction}
|
|
defaultMotions={defaultMotions}
|
|
expandIcon={mergedExpandIcon}
|
|
ref={ref}
|
|
rootClassName={cls(
|
|
rootClassName,
|
|
hashId,
|
|
overrideObj.rootClassName,
|
|
cssVarCls,
|
|
rootCls,
|
|
mergedClassNames.root,
|
|
)}
|
|
_internalComponents={MENU_COMPONENTS}
|
|
/>
|
|
</MenuContext.Provider>
|
|
</OverrideContext.Provider>
|
|
);
|
|
});
|
|
|
|
export default InternalMenu;
|