網站首頁 編程語言 正文
React-RouterV6 + AntdV4實現Menu菜單路由跳轉,采用子路由嵌套的方式
兩種實現方式:
方式一:編程式跳轉
使用useNavigate()
方式二:NavLink鏈接式
<Link to="/home">主頁</Link>
配置路由和主頁
App.js
import {
Routes,
Route,
Navigate,
useLocation
} from 'react-router-dom'
import Home from './pages/Home';
import Main from './pages/Main';
import User from './pages/User';
import Auth from './pages/Auth';
function App() {
// 獲取瀏覽器url
const location = useLocation()
const { path } = location
console.log(path);
return (
<Routes>
{/* 重定向到主頁 */}
<Route path='*' element={<Navigate to="/home" />} />
{/* 主頁及其子路由 */}
<Route exact path='/home' element={<Home />} >
{/* url為/home時主動觸發二級路由 */}
<Route exact index element={<Main />} />
<Route exact path='/home/user/list' element={<User />} />
<Route exact path='/home/user/auth' element={<Auth />} />
</Route>
</Routes>
);
}
export default App;
Home/index.js
react-router-dom的Outlet組件,類似于Vue中的router-view,在Outlet處會渲染任何匹配到的子路由組件。
import React from 'react';
import { Breadcrumb, Layout } from 'antd';
import '../../assets/css/layout.css';
import { LaptopOutlined, NotificationOutlined, UserOutlined } from '@ant-design/icons';
import { Outlet } from 'react-router-dom'
import SiderLeft from '../../components/SiderLeft.js';
import TopHeader from '../../components/TopHeader';
const { Header, Content, Sider } = Layout;
export default class Home extends React.Component {
constructor(props) {
super(props);
this.state = {};
}
// 生命周期函數
componentDidMount() {
console.log('componentDidMount');
}
componentWillUnmount() {
console.log('componentWillUnmount');
}
render() {
return (
<Layout>
{/* 頭部 */}
<Header>
<TopHeader />
</Header>
<Layout className="layout-main">
{/* 左側導航欄 */}
<Sider width={200} className="layout-nav-box"
style={{
position: 'fixed',
left: 0,
top: 64,
bottom: 0
}}
>
{/* 渲染左側菜單組件 */}
<SiderLeft />
</Sider>
{/* 右邊區域 */}
<Layout
style={{
position: 'relative',
right: 0,
top: 64,
marginLeft: 200,
padding: '0 24px 24px',
}}
>
<Breadcrumb
style={{
margin: '16px 0',
}}
>
<Breadcrumb.Item>Home</Breadcrumb.Item>
<Breadcrumb.Item>List</Breadcrumb.Item>
<Breadcrumb.Item>App</Breadcrumb.Item>
</Breadcrumb>
<Content
className="layout-content"
style={{
padding: 24,
margin: 0,
minHeight: 280,
}}
>
{/* 渲染子路由 匹配到子路由時,用子路由的組件替換此處內容*/}
{/* 類似Vue中的router-view */}
<Outlet />
</Content>
</Layout>
</Layout>
</Layout>
)
}
}
一、編程式跳轉
編程式跳轉的方式使用react-router-dom中的useNavigate方法,傳入url路徑即可進行頁面跳轉。
注意:useNavigate不能在類組件中使用,如果你非要在類組件中使用,可以使用高階組件,對類組件進行一個包裹,讓原始類組件擁有useNavigate功能。(函數組件中使用useNavigate請自行實現)
Menu/SiderLeft.js
import { Menu } from 'antd';
import React from 'react';
import { LaptopOutlined, NotificationOutlined, UserOutlined } from '@ant-design/icons';
// 高階組件,包裹useNavigate()功能
import WidthUseNavigate from './withComponents/WithUseNavigate';
class SiderLeft extends React.Component {
constructor(props) {
super(props);
this.state = {
items: [{
key: "/home",
icon: React.createElement(UserOutlined),
label: "概覽"
}, {
key: "/home/user",
icon: React.createElement(UserOutlined),
label: "用戶管理",
children: [{
key: "/home/user/list",
label: "成員管理"
}, {
key: "/home/user/auth",
label: "權限設置"
}, {
key: "sub23",
label: "菜單三"
}, {
key: "sub24",
label: "菜單四"
}, {
key: "sub25",
label: "菜單五"
}]
}]
};
}
click = (e) => {
console.log(e);
console.log(e.key);
//注意this指向問題,采用箭頭函數this就指向當前組件
this.props.to(e.key);
}
openChange() {
console.log('OpenChange');
}
render() {
return (
<Menu
theme="light"
mode="inline"
defaultSelectedKeys={['/home']}
defaultOpenKeys={['/home/user']}
style={{
height: '100%',
borderRight: 0,
}}
items={this.state.items}
onOpenChange={() => this.openChange()}
onClick={this.click}
/>
)
}
}
// 使用高階組件包裹當前類組件
const NavigateCompont = WidthUseNavigate(SiderLeft);
// 導出包裹后的類組件
export default NavigateCompont;
高階組件,包裹useNavigate()功能
widthUseNavigate.js
import { useNavigate } from 'react-router-dom'
// 高階組件包裝useNavigate()功能
// 原因:類組件中無法使用useNavigate(),會報錯
// React Hook "useNavigate" cannot be called in a class component.
function widthUseNavigate(WrapCompontent) {
// 設置別名
WrapCompontent.displayName = `widthUseNavigate${getDisplayName(WrapCompontent)}`
return function NavigateCompont() {
const navigate = useNavigate()
// 給傳入的組件新增一個to方法,傳給原始組件的props,在原始組件中通過this.props.to(參數)使用
return <WrapCompontent to={navigate}></WrapCompontent>
}
}
// 別名
function getDisplayName(WrapCompontent) {
return WrapCompontent.displayname || WrapCompontent.name || 'Component'
}
export default widthUseNavigate
二、鏈接式跳轉
在Menu組件的items中將label的值設置為<Link to="/home">主頁</Link>,菜單顯示“主頁”,同時具備鏈接跳轉功能。
Menu/SiderLeft.js
import { Menu } from 'antd';
import React from 'react';
import { LaptopOutlined, NotificationOutlined, UserOutlined } from '@ant-design/icons';
import { NavLink as Link } from 'react-router-dom';
export default class SiderLeft extends React.Component {
constructor(props) {
super(props);
this.state = {
items: [{
key: "/home",
icon: React.createElement(UserOutlined),
label: <Link to="/home">概覽</Link>
}, {
key: "/home/user",
icon: React.createElement(UserOutlined),
label: "用戶管理",
children: [{
key: "/home/user/list",
label: <Link to="/home/user/list">成員管理</Link>
}, {
key: "/home/user/auth",
label: <Link to="/home/user/auth">權限設置</Link>
}, {
pass...
}]
}]
};
}
openChange() {
console.log('OpenChange');
}
render() {
return (
<Menu
theme="light"
mode="inline"
defaultSelectedKeys={['/home']}
defaultOpenKeys={['/home/user']}
style={{
height: '100%',
borderRight: 0,
}}
items={this.state.items}
onOpenChange={() => this.openChange()}
/>
)
}
}
三、實現效果
兩種方式都能實現點擊左側菜單,右側內容區顯示不同組件。
Main/index.js
import React from 'react';
export default class Main extends React.Component {
constructor(props) {
super(props);
this.state = {};
}
render() {
return (
// React.Fragment一般跟在return后面,用來包裹元素,之前一般會用div進行包裹
// Fragment相比于div的好處是在dom中不會增加額外節點,也可以直接簡寫為<></>
<React.Fragment>
<div className="layout-content-display">
主內容區
</div>
</React.Fragment>
)
}
}
User/index.js
import React from 'react';
export default class User extends React.Component {
constructor(props) {
super(props);
this.state = {};
}
render() {
return (
<React.Fragment>
<div className="layout-content-display">
用戶列表
</div>
</React.Fragment>
)
}
}
Auth/index.js
import React from 'react';
export default class Auth extends React.Component {
constructor(props) {
super(props);
this.state = {};
}
render() {
return (
<React.Fragment>
<div className="layout-content-display">
權限設置
</div>
</React.Fragment>
)
}
}
輸入任何未匹配到路由的url,重定向到主頁,默認渲染Main組件
點擊“成員管理”,右側內容展示區渲染User組件
點擊“權限管理”,右側內容展示區渲染Auth組件
原文鏈接:https://blog.csdn.net/liyonghong3333/article/details/124969089
相關推薦
- 2021-12-10 Qt?QFile文件操作的具體使用_C 語言
- 2023-02-06 Golang泛型實現類型轉換的方法實例_Golang
- 2023-02-17 Go語言如何實現TCP通信詳解_Golang
- 2023-03-20 C#中如何生成安裝包_C#教程
- 2022-06-15 axios?gin的GET和POST請求實現示例_Golang
- 2023-02-25 C++?move()函數及priority_queue隊列使用記錄_C 語言
- 2022-09-17 C++實現棧的操作(push和pop)_C 語言
- 2022-04-12 C#實現六大設計原則之接口隔離原則_C#教程
- 最近更新
-
- window11 系統安裝 yarn
- 超詳細win安裝深度學習環境2025年最新版(
- Linux 中運行的top命令 怎么退出?
- MySQL 中decimal 的用法? 存儲小
- get 、set 、toString 方法的使
- @Resource和 @Autowired注解
- Java基礎操作-- 運算符,流程控制 Flo
- 1. Int 和Integer 的區別,Jav
- spring @retryable不生效的一種
- Spring Security之認證信息的處理
- Spring Security之認證過濾器
- Spring Security概述快速入門
- Spring Security之配置體系
- 【SpringBoot】SpringCache
- Spring Security之基于方法配置權
- redisson分布式鎖中waittime的設
- maven:解決release錯誤:Artif
- restTemplate使用總結
- Spring Security之安全異常處理
- MybatisPlus優雅實現加密?
- Spring ioc容器與Bean的生命周期。
- 【探索SpringCloud】服務發現-Nac
- Spring Security之基于HttpR
- Redis 底層數據結構-簡單動態字符串(SD
- arthas操作spring被代理目標對象命令
- Spring中的單例模式應用詳解
- 聊聊消息隊列,發送消息的4種方式
- bootspring第三方資源配置管理
- GIT同步修改后的遠程分支