SiderMenu.tsx 3.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124
  1. import { Layout } from 'antd';
  2. import classNames from 'classnames';
  3. import React, { Component, Suspense } from 'react';
  4. import Link from 'umi/link';
  5. import defaultSettings from '../../../config/defaultSettings';
  6. import PageLoading from '../PageLoading';
  7. import { BaseMenuProps } from './BaseMenu';
  8. import styles from './index.less';
  9. import { getDefaultCollapsedSubMenus } from './SiderMenuUtils';
  10. const BaseMenu = React.lazy(() => import('./BaseMenu'));
  11. const { Sider } = Layout;
  12. const { title } = defaultSettings;
  13. let firstMount: boolean = true;
  14. export interface SiderMenuProps extends BaseMenuProps {
  15. logo?: string;
  16. fixSiderbar?: boolean;
  17. }
  18. interface SiderMenuState {
  19. pathname?: string;
  20. openKeys?: string[];
  21. flatMenuKeysLen?: number;
  22. }
  23. export default class SiderMenu extends Component<SiderMenuProps, SiderMenuState> {
  24. static defaultProps: Partial<SiderMenuProps> = {
  25. flatMenuKeys: [],
  26. onCollapse: () => void 0,
  27. isMobile: false,
  28. openKeys: [],
  29. collapsed: false,
  30. handleOpenChange: () => void 0,
  31. menuData: [],
  32. onOpenChange: () => void 0,
  33. };
  34. static getDerivedStateFromProps(props: SiderMenuProps, state: SiderMenuState) {
  35. const { pathname, flatMenuKeysLen } = state;
  36. if (props.location!.pathname !== pathname || props.flatMenuKeys!.length !== flatMenuKeysLen) {
  37. return {
  38. pathname: props.location!.pathname,
  39. flatMenuKeysLen: props.flatMenuKeys!.length,
  40. openKeys: getDefaultCollapsedSubMenus(props),
  41. };
  42. }
  43. return null;
  44. }
  45. constructor(props: SiderMenuProps) {
  46. super(props);
  47. this.state = {
  48. openKeys: getDefaultCollapsedSubMenus(props),
  49. };
  50. }
  51. componentDidMount() {
  52. firstMount = false;
  53. }
  54. isMainMenu: (key: string) => boolean = key => {
  55. const { menuData } = this.props;
  56. return menuData!.some(item => {
  57. if (key) {
  58. return item.key === key || item.path === key;
  59. }
  60. return false;
  61. });
  62. };
  63. handleOpenChange: (openKeys: string[]) => void = openKeys => {
  64. const moreThanOne = openKeys.filter(openKey => this.isMainMenu(openKey)).length > 1;
  65. if (moreThanOne) {
  66. this.setState({ openKeys: [openKeys.pop()].filter(item => item) as string[] });
  67. } else {
  68. this.setState({ openKeys: [...openKeys] });
  69. }
  70. };
  71. render() {
  72. const { logo, collapsed, onCollapse, fixSiderbar, theme, isMobile } = this.props;
  73. const { openKeys } = this.state;
  74. const defaultProps = collapsed ? {} : { openKeys };
  75. const siderClassName = classNames(styles.sider, {
  76. [styles.fixSiderBar]: fixSiderbar,
  77. [styles.light]: theme === 'light',
  78. });
  79. return (
  80. <Sider
  81. collapsible
  82. trigger={null}
  83. collapsed={collapsed}
  84. breakpoint="lg"
  85. onCollapse={collapse => {
  86. if (firstMount || !isMobile) {
  87. onCollapse!(collapse);
  88. }
  89. }}
  90. width={256}
  91. theme={theme}
  92. className={siderClassName}
  93. >
  94. <div className={styles.logo} id="logo">
  95. <Link to="/">
  96. <img src={logo} alt="logo" />
  97. <h1>{title}</h1>
  98. </Link>
  99. </div>
  100. <Suspense fallback={<PageLoading />}>
  101. <BaseMenu
  102. {...this.props}
  103. mode="inline"
  104. handleOpenChange={this.handleOpenChange}
  105. onOpenChange={this.handleOpenChange}
  106. style={{ padding: '16px 0', width: '100%' }}
  107. {...defaultProps}
  108. />
  109. </Suspense>
  110. </Sider>
  111. );
  112. }
  113. }