mirror of
https://github.com/ant-design/ant-design.git
synced 2024-12-15 08:59:15 +08:00
20d5502193
* chore(deps-dev): bump eslint-config-airbnb from 18.2.1 to 19.0.0 Bumps [eslint-config-airbnb](https://github.com/airbnb/javascript) from 18.2.1 to 19.0.0. - [Release notes](https://github.com/airbnb/javascript/releases) - [Commits](https://github.com/airbnb/javascript/compare/eslint-config-airbnb-v18.2.1...eslint-config-airbnb-v19.0.0) --- updated-dependencies: - dependency-name: eslint-config-airbnb dependency-type: direct:development update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * chore: code style * memoize-one * add comment * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * fix lint * improve useMemo deps Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com> Co-authored-by: afc163 <afc163@gmail.com>
141 lines
4.1 KiB
TypeScript
141 lines
4.1 KiB
TypeScript
import * as React from 'react';
|
|
import RcMenu, { ItemGroup, MenuProps as RcMenuProps } from 'rc-menu';
|
|
import classNames from 'classnames';
|
|
import omit from 'rc-util/lib/omit';
|
|
import EllipsisOutlined from '@ant-design/icons/EllipsisOutlined';
|
|
import memoize from 'memoize-one';
|
|
import SubMenu, { SubMenuProps } from './SubMenu';
|
|
import Item, { MenuItemProps } from './MenuItem';
|
|
import { ConfigConsumer, ConfigConsumerProps } from '../config-provider';
|
|
import devWarning from '../_util/devWarning';
|
|
import { SiderContext, SiderContextProps } from '../layout/Sider';
|
|
import collapseMotion from '../_util/motion';
|
|
import { cloneElement } from '../_util/reactNode';
|
|
import MenuContext, { MenuTheme } from './MenuContext';
|
|
import MenuDivider from './MenuDivider';
|
|
|
|
export { MenuDividerProps } from './MenuDivider';
|
|
|
|
export { MenuItemGroupProps } from 'rc-menu';
|
|
|
|
export type MenuMode = 'vertical' | 'vertical-left' | 'vertical-right' | 'horizontal' | 'inline';
|
|
|
|
export interface MenuProps extends RcMenuProps {
|
|
theme?: MenuTheme;
|
|
inlineIndent?: number;
|
|
}
|
|
|
|
type InternalMenuProps = MenuProps &
|
|
SiderContextProps & {
|
|
collapsedWidth?: string | number;
|
|
};
|
|
|
|
class InternalMenu extends React.Component<InternalMenuProps> {
|
|
static defaultProps: Partial<MenuProps> = {
|
|
theme: 'light', // or dark
|
|
};
|
|
|
|
constructor(props: InternalMenuProps) {
|
|
super(props);
|
|
|
|
devWarning(
|
|
!('inlineCollapsed' in props && props.mode !== 'inline'),
|
|
'Menu',
|
|
'`inlineCollapsed` should only be used when `mode` is inline.',
|
|
);
|
|
|
|
devWarning(
|
|
!(props.siderCollapsed !== undefined && 'inlineCollapsed' in props),
|
|
'Menu',
|
|
'`inlineCollapsed` not control Menu under Sider. Should set `collapsed` on Sider instead.',
|
|
);
|
|
}
|
|
|
|
getInlineCollapsed() {
|
|
const { inlineCollapsed, siderCollapsed } = this.props;
|
|
if (siderCollapsed !== undefined) {
|
|
return siderCollapsed;
|
|
}
|
|
return inlineCollapsed;
|
|
}
|
|
|
|
renderMenu = ({ getPopupContainer, getPrefixCls, direction }: ConfigConsumerProps) => {
|
|
const rootPrefixCls = getPrefixCls();
|
|
|
|
const {
|
|
prefixCls: customizePrefixCls,
|
|
className,
|
|
theme,
|
|
expandIcon,
|
|
...restProps
|
|
} = this.props;
|
|
|
|
const passedProps = omit(restProps, ['siderCollapsed', 'collapsedWidth']);
|
|
const inlineCollapsed = this.getInlineCollapsed();
|
|
|
|
const defaultMotions = {
|
|
horizontal: { motionName: `${rootPrefixCls}-slide-up` },
|
|
inline: collapseMotion,
|
|
other: { motionName: `${rootPrefixCls}-zoom-big` },
|
|
};
|
|
|
|
const prefixCls = getPrefixCls('menu', customizePrefixCls);
|
|
const menuClassName = classNames(`${prefixCls}-${theme}`, className);
|
|
|
|
// TODO: refactor menu with function component
|
|
const contextValue = memoize((cls, collapsed, the, dir) => ({
|
|
prefixCls: cls,
|
|
inlineCollapsed: collapsed || false,
|
|
antdMenuTheme: the,
|
|
direction: dir,
|
|
firstLevel: true,
|
|
}))(prefixCls, inlineCollapsed, theme, direction);
|
|
|
|
return (
|
|
<MenuContext.Provider value={contextValue}>
|
|
<RcMenu
|
|
getPopupContainer={getPopupContainer}
|
|
overflowedIndicator={<EllipsisOutlined />}
|
|
overflowedIndicatorPopupClassName={`${prefixCls}-${theme}`}
|
|
{...passedProps}
|
|
inlineCollapsed={inlineCollapsed}
|
|
className={menuClassName}
|
|
prefixCls={prefixCls}
|
|
direction={direction}
|
|
defaultMotions={defaultMotions}
|
|
expandIcon={cloneElement(expandIcon, {
|
|
className: `${prefixCls}-submenu-expand-icon`,
|
|
})}
|
|
/>
|
|
</MenuContext.Provider>
|
|
);
|
|
};
|
|
|
|
render() {
|
|
return <ConfigConsumer>{this.renderMenu}</ConfigConsumer>;
|
|
}
|
|
}
|
|
|
|
// We should keep this as ref-able
|
|
class Menu extends React.Component<MenuProps, {}> {
|
|
static Divider = MenuDivider;
|
|
|
|
static Item = Item;
|
|
|
|
static SubMenu = SubMenu;
|
|
|
|
static ItemGroup = ItemGroup;
|
|
|
|
render() {
|
|
return (
|
|
<SiderContext.Consumer>
|
|
{(context: SiderContextProps) => <InternalMenu {...this.props} {...context} />}
|
|
</SiderContext.Consumer>
|
|
);
|
|
}
|
|
}
|
|
|
|
export { MenuTheme, SubMenuProps, MenuItemProps };
|
|
|
|
export default Menu;
|