React + Ant Design 实现带有侧边栏和路由切换的布局_ant design

React + Ant Design 实现带有侧边栏和路由切换的布局_react_02

步骤
  1. 安装依赖
    首先,确保安装了以下依赖:
npm install react react-dom antd @ant-design/icons react-router-dom
  • 1.
  1. 创建基本布局组件
    我们将创建一个名为 Main.js 的文件,它将包含我们的主要布局组件。
  2. 编写代码
    在 Main.js 中,我们将实现以下功能:
  • 使用 Ant Design 的 Layout 组件来组织页面结构。
  • 使用 Menu 组件作为侧边栏,其中包含可点击的菜单项。
  • 使用 useNavigate 从 react-router-dom 来处理页面间的导航。
  • 使用 Outlet 组件来渲染当前激活的路由组件。
import React, { useState } from 'react';
import {
  MenuFoldOutlined,
  MenuUnfoldOutlined,
  UploadOutlined,
  UserOutlined,
  VideoCameraOutlined,
  LogoutOutlined,
} from '@ant-design/icons';
import { Button, Layout, Menu, theme } from 'antd';
import { useNavigate } from 'react-router-dom'; // 导入 useNavigate
import '../css/main.css'; // 导入自定义的 CSS 文件
import { Outlet } from 'react-router-dom';

const { Header, Sider, Content } = Layout;

// 定义退出登录的函数
const handleLogout = (navigate) => {
  console.log('Logging out...');

  // 清除登录状态,例如从 localStorage 中删除 token
  // localStorage.removeItem('token');

  // 跳转到登录页面
  navigate('/login');
};

const Main = () => {
  const [collapsed, setCollapsed] = useState(false);
  const {
    token: { colorBgContainer, borderRadiusLG },
  } = theme.useToken();
  const navigate = useNavigate(); // 创建 navigate 函数

  // 定义菜单项点击事件处理器
  const handleMenuClick = (e) => {
    console.log('Menu clicked:', e);
    const path = e.item.props.path; // 从菜单项属性中获取路径
    if (path) {
      navigate(path); // 跳转到对应的路径
    }
  };

  return (
    <Layout
      style={{
        display: 'flex',
        height: '100vh', // 设置高度为视口高度
      }}
    >
      <Sider
        trigger={null}
        collapsible
        collapsed={collapsed}
        style={{ flex: 1 }} // 设置 flex 属性
      >
        <div className="demo-logo-vertical" />
        <Menu
          theme="dark"
          mode="inline"
          onClick={handleMenuClick} // 添加 onClick 事件处理器
          defaultSelectedKeys={['1']}
          items={[
            {
              key: '1',
              icon: <UserOutlined />,
              label: 'Home',
              path: '/home',
            },
            {
              key: '2',
              icon: <VideoCameraOutlined />,
              label: 'User',
              path: '/user',
            },
          ]}
        />
      </Sider>
      <Layout style={{ flex: 1 }}> {/* 设置 flex 属性 */}
        <Header
          style={{
            padding: 0,
            background: colorBgContainer,
            display: 'flex', // 设置 header 为 flex 布局
            justifyContent: 'space-between', // 使内容两端对齐
            alignItems: 'center', // 垂直居中
          }}
        >
          <Button
            type="text"
            icon={collapsed ? <MenuUnfoldOutlined /> : <MenuFoldOutlined />}
            onClick={() => setCollapsed(!collapsed)}
            style={{
              fontSize: '16px',
              width: 64,
              height: 64,
            }}
          />
          <Button
            type="text"
            icon={<LogoutOutlined />}
            onClick={() => handleLogout(navigate)} // 绑定退出登录的点击事件
            style={{
              fontSize: '16px',
              marginRight: '20px', // 将按钮放置在右侧
            }}
          />
        </Header>
        <Content
          style={{
            margin: '24px 16px',
            padding: 24,
            background: colorBgContainer,
            borderRadius: borderRadiusLG,
          }}
        >
          <Outlet /> {/* 渲染当前激活的路由组件 */}
        </Content>
      </Layout>
    </Layout>
  );
};

export default Main;
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
  • 22.
  • 23.
  • 24.
  • 25.
  • 26.
  • 27.
  • 28.
  • 29.
  • 30.
  • 31.
  • 32.
  • 33.
  • 34.
  • 35.
  • 36.
  • 37.
  • 38.
  • 39.
  • 40.
  • 41.
  • 42.
  • 43.
  • 44.
  • 45.
  • 46.
  • 47.
  • 48.
  • 49.
  • 50.
  • 51.
  • 52.
  • 53.
  • 54.
  • 55.
  • 56.
  • 57.
  • 58.
  • 59.
  • 60.
  • 61.
  • 62.
  • 63.
  • 64.
  • 65.
  • 66.
  • 67.
  • 68.
  • 69.
  • 70.
  • 71.
  • 72.
  • 73.
  • 74.
  • 75.
  • 76.
  • 77.
  • 78.
  • 79.
  • 80.
  • 81.
  • 82.
  • 83.
  • 84.
  • 85.
  • 86.
  • 87.
  • 88.
  • 89.
  • 90.
  • 91.
  • 92.
  • 93.
  • 94.
  • 95.
  • 96.
  • 97.
  • 98.
  • 99.
  • 100.
  • 101.
  • 102.
  • 103.
  • 104.
  • 105.
  • 106.
  • 107.
  • 108.
  • 109.
  • 110.
  • 111.
  • 112.
  • 113.
  • 114.
  • 115.
  • 116.
  • 117.
  • 118.
  • 119.
  • 120.
  • 121.
  • 122.
  • 123.
  • 124.

配置路由

确保在项目的某个位置配置了 BrowserRouter 和相应的路由规则。例如,在 App.js 文件中:

import React from 'react';
import { BrowserRouter as Router, Route, Routes } from 'react-router-dom';
import Main from './components/Main';
import Home from './pages/Home';
import User from './pages/User';

function App() {
  return (
    <Router>
      <Routes>
        <Route path="/" element={<Main />}>
          <Route index element={<Home />} />
          <Route path="/home" element={<Home />} />
          <Route path="/user" element={<User />} />
        </Route>
      </Routes>
    </Router>
  );
}

export default App;
  • 1.
  • 2.
  • 3.
  • 4.
  • 5.
  • 6.
  • 7.
  • 8.
  • 9.
  • 10.
  • 11.
  • 12.
  • 13.
  • 14.
  • 15.
  • 16.
  • 17.
  • 18.
  • 19.
  • 20.
  • 21.
结论

通过以上步骤,你已经成功创建了一个带有侧边栏和路由切换功能的基本布局。这为构建复杂的单页应用程序提供了一个良好的起点。你可以根据需要扩展更多的页面和功能。


注意事项

  • 确保每个菜单项都有一个唯一的 key
  • 在实际应用中,你可能还需要处理登录状态和权限验证等功能。
  • 可以进一步美化样式,使其更加符合你的设计需求。

希望这篇文章能帮助你快速上手使用 React 和 Ant Design 构建具有侧边栏和路由切换功能的应用程序!