当前位置: 首页 > news >正文

umi实现动态获取菜单权限

文章目录

    • 前景
    • 登录组件编写
    • 登录逻辑
    • 菜单的时机
    • 动态路由
    • 页面刷新
    • 手动修改地址

前景

不同用户拥有不同的菜单权限,现在我们实现登录动态获取权限菜单。

登录组件编写

//当我们需要使用dva的dispatch函数时,除了通过@connect函数包裹组件还可以使用这种方式来实现
import { getDvaApp } from 'umi';
const Login = ({ form ,dispatch}) => {const { getFieldDecorator, validateFields } = form; // 从 props 中解构出 form 方法/*** export default Form.create()(Login);*/const onSubmit = () => {validateFields((err, values) => {if (!err) {// 处理登录逻辑getDvaApp()._store.dispatch({type:"globalModel/login",payload:{username: values.username,password: values.password,}});} else {console.log('Validation Failed:', err);}});};return (<div className="login-page"><div className="login-container"  style ={{marginTop: '7%',marginRight: '13%'}}><div className="image-container"><img src={loginImage} alt="Login" /></div><div className="form-container"><h2>欢迎回来</h2><Formname="login"layout="vertical"><Form.Item label="用户名">{getFieldDecorator('username', {rules: [{ required: true, message: '请输入用户名!' }],})(<Input placeholder="请输入用户名" />)}</Form.Item><Form.Item label="密码">{getFieldDecorator('password', {rules: [{ required: true, message: '请输入密码!' }],})(<Input.Password placeholder="请输入密码" />)}</Form.Item><Form.Item><Button type="primary" onClick={onSubmit} className="login-button">登录</Button></Form.Item></Form></div></div></div>);
};
// 使用 Form.create 包裹 Login 组件 从而获取form中相关的函数 否则需要使用队Form组件设置ref 通过ref来实现详见loginbak.js
export default Form.create()(Login);

效果如下所示

在这里插入图片描述

登录逻辑

上面登录组件我们点击登录后,使用dispatch函数发送了一个访问请求,该请求处理逻辑处理来自dva中一个namespace叫做globalModel的model,代码如下所示:

getDvaApp()._store.dispatch({type:"globalModel/login",payload:{username: values.username,password: values.password,}});

在globalModel这个model中,我们在effect块中定义了一个login函数,他主要做一下操作:

  • 访问后端接口
  • 存储登录的数据
  • 跳转到/index页面

menuList: 是当前用户拥有的菜单权限

   //import {routerRedux} from 'dva/router'//import * as requestUtil from "../utils/request";//globalModel文件内state: {menuList: [],//当前用户拥有的菜单权限},*login({payload:{username,password}}, { select, call, put }) {const {data, code, msg} = yield call(globalModelService.login,{username,password});if (code === 200){requestUtil.save(data);yield put({type: 'updateState',payload:{menuList: data.menuList,}});//去首页信息yield put(routerRedux.push('/index'))} else {message.error("登陆失败!");}},
//requestUtil内容
export function save({accessToken,refreshToken,menuList}){sessionStorage.setItem("accessToken",accessToken);sessionStorage.setItem("refreshToken",refreshToken);sessionStorage.setItem("menuList",JSON.stringify(menuList));
}

菜单的时机

上面我们将数据存储在了sessionStorage中,下面我们来看如何使用这些数据,首先我们看看路由信息。

//src/routes 这个是定义的全局路由,各个模块的路由信息将在这里汇总module.exports = [{path: "/",exact: true,redirect:"/login",//跳转到登录页},{path: "/login",exact: true,component: "@/layouts/login/login.js",},{//不能加exact=truepath: "/",component: "@/layouts/index.js",routes: [//routes将会作为 index.js 中BasicRoute组件的props信息{path: '/index',component: '@/pages/atscript/basic.tsx'},...routeConsole(practice_routes),...routeConsole(lesson_routes)]}
];

这个routes将在.umirc.ts作为系统的路由配置

登录完成后页面被重定向到/index中,通过路由可以发现首先会加载父路径/下的@/layout/index.js的资源,相关代码如下所示,在该组件中我有一个SysMenu组件,所需的参数是当前用户获取的菜单权限信息

const BasicRoute = (props) => {const {globalModel,dispatch} = props;const historyHook = useHistory();let {menuList} = globalModel;useEffect(()=>{const curMenuList = (!!menuList && menuList.length > 0 )? menuList :(!!sessionStorage.getItem("menuList")? JSON.parse(sessionStorage.getItem("menuList")) : []);dispatch({type: 'globalModel/updateState',payload:{menuList: curMenuList,}});},[menuList]);const sysMenuProps ={menuList: menuList}return (<SysMenu {...sysMenuProps}>{props.children}</SysMenu>);
}// 指定订阅数据,这里关联了 model的namespace = globalModel
function mapStateToProps({globalModel }) {return {globalModel};
}// 建立数据关联关系 对于关联组件名称
export default connect(mapStateToProps)(BasicRoute);

useEffect 一是为了防止页面刷新数据导致菜单信息丢失,二是可以避免menuList数据有延迟导致渲染问题(第一次默认值为[])

下面我们来到SysMenu组件,最终生成菜单信息来自MenuTree 组件。

 <Layout><Sider trigger={null} collapsible collapsed={collapsed}><div className={layoutModule.logo} >学习系统 !!!</div><MenuTree {...menuTreeProps} /></Sider><Layout><Header style={{ background: '#fff', padding: 0 }}><IconclassName={layoutModule.trigger}type={collapsed ? 'menu-unfold' : 'menu-fold'}onClick={onCollapse}/></Header><Content className={layoutModule.content}><Breadcrumb  separator=">" style={{ margin: '16px 0' }}><Breadcrumb.Item>User</Breadcrumb.Item><Breadcrumb.Item>Bill</Breadcrumb.Item></Breadcrumb><div style={{ padding: 24, background: '#fff', minHeight: '90%' }}>{props.children}</div></Content><Footer style={{ textAlign: 'center' }}>Ant Design ©2018 Created by Ant UED</Footer></Layout></Layout>

动态路由

前面我们配置了路由地址,实际情况会根据用户的不同权限展示不同的菜单,下面我们将菜单抽取出一个专门的组件来动态处理。文件目录如下所示:
在这里插入图片描述
其中SysMenu文件中菜单配置变化如下:

目前我们指定菜单属性只有5个属性(后台已封装为父子结构),分别为nameurliconkeychildren,按照顺序依次表现为菜单名称,菜单路由,菜单图标,唯一key,菜单的子级节点。数据结构如下所示:

    public static List<Menu> getMenuList() {List<Menu> menuList = new ArrayList<>();menuList.addAll(Arrays.asList(new Menu("学习模块", "#", "build", "lesson", new ArrayList<>()).addChild(new Menu("Ref学习", "/lesson/reftest", "skin", "/lesson/reftest", new ArrayList<>())),new Menu("练习程序", "#", "book", "practice", new ArrayList<>()).addChild(new Menu("计算程序", "/practice/calculate", "snippets", "/practice/calculate", new ArrayList<>())).addChild(new Menu("中央空调", "/practice/air", "skin", "/practice/air", new ArrayList<>())).addChild(new Menu("流程管理", "/practice/activiti", "user", "/practice/activiti", new ArrayList<>())),//                new Menu("文件操作", "#", "build", "fileManage", new ArrayList<>())
//                        .addChild(new Menu("整体上传", "/fileManage/commonUploadFile", "user", "/fileManage/commonUploadFile", new ArrayList<>()))
//                        .addChild(new Menu("分片上传", "/fileManage/filePartUploadFile", "user", "/fileManage/filePartUploadFile", new ArrayList<>()))
//                        .addChild(new Menu("文件秒传", "/fileManage/flashUploadFile", "user", "/fileManage/flashUploadFile", new ArrayList<>()))
//                        .addChild(new Menu("断点续传", "/fileManage/breakPointUploadFile", "user", "/fileManage/breakPointUploadFile", new ArrayList<>())),new Menu("支付对接", "#", "build", "pay", new ArrayList<>()).addChild(new Menu("支付宝web支付", "/pay/AliWebPay", "user", "/pay/AliWebPay", new ArrayList<>())),new Menu("二维码", "/qrcode/qrcode", "build", "/qrcode/qrcode", new ArrayList<>())));return menuList;}

MenuTree组件整体结构如下,其中extractMenus方法则是将后台返回的权限菜单进行转化为对应的配置。

const MenuTree = ()=>{  return(<Menu theme="dark" mode="inline" defaultSelectedKeys={[menuList[0].key]}>{extractMenus(menuList)}</Menu>);
}
export default MenuTree;
  const extractMenus = (list) =>{return list.map((item, index) => (doExtractMenus(item)));}//显示菜单项const doExtractMenus = (item)=>{if(item.children.length == 0){return (<Menu.Item key={item.key}><Icon type={item.icon}/><span>{item.name}</span><Link to={item.url}/></Menu.Item>);}else{return (<SubMenu key={item.key} title={<span><Icon type={item.icon}/><span>{item.name}</span></span>}>{extractMenus(item.children)}</SubMenu>);}}

页面刷新

当我们刷新页面后丢失了菜单的选中信息,实际上需要还是在对应选中的菜单节点,下面我们来避免这个问题,我们需要记录以下信息:

  • 原来展开的菜单节点信息
  • 原来选中的菜单
//import { useHistory } from 'react-router-dom';
//import {useEffect, useState} from "react";
const MenuTree = (props)=>{const historyHook = useHistory();const menuList = props.menuList;const [state,setState] = useState ({visible:false,menuTree:[],defaultOpenKeys:[],defaultSelectedKeys:[historyHook.location.pathname]});useEffect(()=>{const defaultOpenKeys = [historyHook.location.pathname];const menuTree = extractMenus(menuList,null,defaultOpenKeys);console.log("defaultOpenKeys",defaultOpenKeys);setState({...state,visible: true,defaultOpenKeys: defaultOpenKeys,menuTree:menuTree,})},[menuList])//显示菜单列表const extractMenus = (list, parent,curOpenKeys) =>{return list.map((item, index) => doExtractMenus(item, parent,curOpenKeys));}//显示菜单项const doExtractMenus = (item, parent,curOpenKeys)=> {//需要展开父节点if (!!parent && item.url == state.defaultSelectedKeys[0]) {console.log("parent", parent);curOpenKeys.push( parent.key);}//没有子节点if(item.children.length == 0){return (<Menu.Item key={item.key}><Icon type={item.icon}/><span>{item.name}</span><Link to={item.url}/></Menu.Item>);}//当前时父节点else{return (<SubMenu key={item.key} title={<span><Icon type={item.icon}/><span>{item.name}</span></span>}>{extractMenus(item.children, item,curOpenKeys)}</SubMenu>);}}console.log("state",state);/*** defaultOpenKeys 的使用:defaultOpenKeys 只在组件首次渲染时生效。组件执行顺序* render => useEffect 所以利用state.visible来控制*/return(state.visible &&<Menu theme="dark" mode="inline" defaultSelectedKeys={state.defaultSelectedKeys} defaultOpenKeys={state.defaultOpenKeys}>{state.menuTree}</Menu>);}

上面我们通过监听菜单信息menuList(第一次进来为[])将原来extractMenus方法新增parent,curOpenKeys,前者是为了处理选中节点,后者是为了记录需要展开的父节点信息

当第一次进来时menuList为[],导致defaultOpenKeys一直为[] ,为了避免Menu 第一次挂在后,后续刷新defaultOpenKeys将不再生效,组件使用state.visible 来控制挂载时机

手动修改地址

登录完成后避免可以通过修改浏览器直接访问/login,也就是跳转到登录页面,我们需要配合globalModel中的监听函数subscriptions函数

//globalModel.js中subscriptions: {setup({ dispatch, history}) {return history.listen(location => {//如果有登录信息,直接访问/login则重定向到/index页面if (!!sessionStorage.getItem("refreshToken") && history.location.pathname === "/login"){history.push("/index");}}});},},

相关文章:

umi实现动态获取菜单权限

文章目录 前景登录组件编写登录逻辑菜单的时机动态路由页面刷新手动修改地址 前景 不同用户拥有不同的菜单权限&#xff0c;现在我们实现登录动态获取权限菜单。 登录组件编写 //当我们需要使用dva的dispatch函数时&#xff0c;除了通过connect函数包裹组件还可以使用这种方…...

Pytest-Bdd-Playwright 系列教程(14):Docstring 参数

Pytest-Bdd-Playwright 系列教程&#xff08;14&#xff09;&#xff1a;Docstring 参数 前言一、什么是docstring?二、基本语法三、主要特点四、实际例子五、注意事项六、使用建议总结 前言 在自动化测试的过程中&#xff0c;我们经常需要处理复杂的测试数据或需要输入多行文…...

交互开发---测量工具(适用VTK或OpenGL开发的应用程序)

简介&#xff1a; 经常使用RadiAnt DICOM Viewer来查看DICOM数据&#xff0c;该软件中的测量工具比较好用&#xff0c;就想着仿照其交互方式自己实现下。后采用VTK开发应用程序时&#xff0c;经常需要开发各种各样的测量工具&#xff0c;如果沿用VTK的widgets的思路&#xff0c…...

Qt 一个简单的QChart 绘图

Qt 一个简单的QChart 绘图 先上程序运行结果图&#xff1a; “sample9_1QChart.h” 文件代码如下&#xff1a; #pragma once#include <QtWidgets/QMainWindow> #include "ui_sample9_1QChart.h"#include <QtCharts> //必须这么设置 QT_CHARTS_USE_NAME…...

【Java笔记】LinkedList 底层结构

一、LinkedList 的全面说明 LinkedList底层实现了双向链表和双端队列特点可以添加任意元素(元素可以重复)&#xff0c;包括null线程不安全&#xff0c;没有实现同步 二、LinkedList 的底层操作机制 三、LinkedList的增删改查案例 public class LinkedListCRUD { public stati…...

el-table组件树形数据修改展开箭头

<style lang"scss" scoped> ::v-deep .el-table__expand-icon .el-icon-arrow-right:before {content: ">"; // 箭头样式font-size: 16px; }::v-deep .el-table__expand-icon{ // 没有展开的状态background-color: rgba(241, 242, 245, 1);color:…...

太速科技-FMC154-基于FMC 八路SFP+万兆光纤子卡

FMC154-基于FMC 八路SFP万兆光纤子卡 一、板卡概述 本卡是一个FPGA夹层卡&#xff08;FMC&#xff09;模块&#xff0c;可提供高达8个SFP / SFP 模块接口&#xff0c;直接插入千兆位级收发器&#xff08;MGT&#xff09;的赛灵思FPGA。支持业界标准的小型可插拔&#xff0…...

记:排查设备web时慢时快问题,速度提升100%

问题描述 问题1&#xff1a; 发现web登录界面刷新和登录功能都比较卡&#xff0c;开浏览器控制台看了下&#xff0c;让我很惊讶&#xff0c;居然能这么慢&#xff1a; 公司2个局域网内的表现不同&#xff0c;局域网A中的都比较卡&#xff0c;局域网B中的又不存在该现象。 问…...

音视频入门基础:MPEG2-TS专题(13)——FFmpeg源码中,解析Section Header的实现

一、引言 在《音视频入门基础&#xff1a;MPEG2-TS专题&#xff08;11&#xff09;—— TS中的Section》中讲述了Section Header的基本概念&#xff0c;本文讲述FFmpeg源码中是怎样解析Section Header的。 二、parse_section_header函数的定义 FFmpeg源码中通过parse_section…...

根据PDF模板单个PDF导出到浏览器和多个PDF打包ZIP导出到浏览器

一、单个PDF导出到浏览器 /*** * param templatePath 模板路径* param fileName 文件名称* param data 填充文本* param images 填充图片* param response* throws IOException*/public static void generateTempPDF(String templatePath, String fileName, Map<String, S…...

如何创建一个基本的Spring Boot应用程序

以下是一个简单的Spring Boot应用开发代码示例&#xff0c;它展示了如何创建一个基本的Spring Boot应用程序&#xff0c;并实现一个简单的RESTful API服务。 步骤1&#xff1a;创建项目 使用Spring Initializr或您喜欢的IDE&#xff08;如IntelliJ IDEA或Eclipse&#xff09;…...

1.2 计算机网络的分类和应用(重要知识点)

1.2.1 计算机网络的分类 计算机网络的定义&#xff1a; 由通信线路互相连接的、能自主工作的计算机构成&#xff0c;强调各计算机&#xff08;工作站&#xff09;拥有独立的计算资源和任务能力。与多终端分时系统不同&#xff0c;后者终端仅作为主机接口&#xff0c;不具备计…...

@JsonSerialize失效解决

当在实体类中加入这个注解时&#xff0c;本意是想如果是空值则返回0给页面&#xff0c;但是发现使用 JsonSerialize(using BigSerializer.class)无效&#xff0c;因为如果是null值会不走序列化的接口实现类&#xff0c;需要使用nullUsing 需要这样使用...

Docker部署WebRTC-Streamer

文章目录 WebRTC-Streamer概述Docker部署WebRTC-StreamerVue使用WebRTC-Streamer一些问题 WebRTC-Streamer概述 WebRTC-Streamer是一个基于WebRTC技术的流媒体传输工具&#xff0c;它可以通过Web浏览器实现实时音视频流的传输和播放。它提供了一种简单而强大的方式&#xff…...

2025年的大模型计划重点在于跨领域智能、工作流自动化、多模态能力强化

明年的计划和大模型发展方向可以围绕以下几个方面展开&#xff0c;结合实际应用场景和技术趋势&#xff0c;明确可执行的目标和期待的成果&#xff1a; 2025 年计划与展望&#xff1a;大模型能做些什么&#xff1f; 1. 更深层次的跨领域能力融合 目标&#xff1a;构建更强的跨…...

day12 接口测试 ——入门→精通→实战(1)

【没有所谓的运气&#x1f36c;&#xff0c;只有绝对的努力✊】 目录 1、接口测试分类 1.1 内部接口&#xff1a; 1.2 外部接口&#xff1a; 2、目前接口架构设计 2.1、基于SOAP架构&#xff0c; 2.2、基于RPC架构&#xff0c; 2.3、基于RestFul架构&#xff0c; 2.3.1…...

伏羲0.07(文生图)

为了使0.06代码能够有效运行并输出项目目录及所有文件&#xff0c;我们在代码中添加一些额外的功能。 项目目录结构 项目目录结构如下&#xff1a; text_to_image_project/ │ ├── config.yaml ├── data/ │ ├── train_data.csv │ └── test_data.txt ├── mod…...

scala的泛型特质的应用场景

//泛型特质的应用场景 //作比较找出最大值 //定义一个函数&#xff0c;用来求List元素中的最大值参考代码&#xff1a;object Test4 {def getMax[T](list:List[T])(implicit ev:T > Ordered[T]): T {list.reduce((a:T,b:T)> if(a>b) a else b)}def main(args: Array…...

Win10环境vscode+latex+中文快速配置

安装vscodelatex workshop 配置&#xff1a; {"liveServer.settings.donotVerifyTags": true,"liveServer.settings.donotShowInfoMsg": true,"explorer.confirmDelete": false,"files.autoSave": "afterDelay","exp…...

【vue2】el-select,虚拟滚动(vue-virtual-scroller)

需求背景​​​​​​ vue2+element-ui项目中,当el-select中数据量较大时(超出5000个dom节点),会导致页面加载和渲染卡顿、el-select下拉列表延迟展开。 在现在的el-select的基础上使用分页或者虚拟列表的形式去处理大量的下拉菜单,可以保证页面的正常渲染及el-select的…...

KubeSphere 容器平台高可用:环境搭建与可视化操作指南

Linux_k8s篇 欢迎来到Linux的世界&#xff0c;看笔记好好学多敲多打&#xff0c;每个人都是大神&#xff01; 题目&#xff1a;KubeSphere 容器平台高可用&#xff1a;环境搭建与可视化操作指南 版本号: 1.0,0 作者: 老王要学习 日期: 2025.06.05 适用环境: Ubuntu22 文档说…...

K8S认证|CKS题库+答案| 11. AppArmor

目录 11. AppArmor 免费获取并激活 CKA_v1.31_模拟系统 题目 开始操作&#xff1a; 1&#xff09;、切换集群 2&#xff09;、切换节点 3&#xff09;、切换到 apparmor 的目录 4&#xff09;、执行 apparmor 策略模块 5&#xff09;、修改 pod 文件 6&#xff09;、…...

前端倒计时误差!

提示:记录工作中遇到的需求及解决办法 文章目录 前言一、误差从何而来?二、五大解决方案1. 动态校准法(基础版)2. Web Worker 计时3. 服务器时间同步4. Performance API 高精度计时5. 页面可见性API优化三、生产环境最佳实践四、终极解决方案架构前言 前几天听说公司某个项…...

Nginx server_name 配置说明

Nginx 是一个高性能的反向代理和负载均衡服务器&#xff0c;其核心配置之一是 server 块中的 server_name 指令。server_name 决定了 Nginx 如何根据客户端请求的 Host 头匹配对应的虚拟主机&#xff08;Virtual Host&#xff09;。 1. 简介 Nginx 使用 server_name 指令来确定…...

鱼香ros docker配置镜像报错:https://registry-1.docker.io/v2/

使用鱼香ros一件安装docker时的https://registry-1.docker.io/v2/问题 一键安装指令 wget http://fishros.com/install -O fishros && . fishros出现问题&#xff1a;docker pull 失败 网络不同&#xff0c;需要使用镜像源 按照如下步骤操作 sudo vi /etc/docker/dae…...

工业自动化时代的精准装配革新:迁移科技3D视觉系统如何重塑机器人定位装配

AI3D视觉的工业赋能者 迁移科技成立于2017年&#xff0c;作为行业领先的3D工业相机及视觉系统供应商&#xff0c;累计完成数亿元融资。其核心技术覆盖硬件设计、算法优化及软件集成&#xff0c;通过稳定、易用、高回报的AI3D视觉系统&#xff0c;为汽车、新能源、金属制造等行…...

如何在网页里填写 PDF 表格?

有时候&#xff0c;你可能希望用户能在你的网站上填写 PDF 表单。然而&#xff0c;这件事并不简单&#xff0c;因为 PDF 并不是一种原生的网页格式。虽然浏览器可以显示 PDF 文件&#xff0c;但原生并不支持编辑或填写它们。更糟的是&#xff0c;如果你想收集表单数据&#xff…...

【Go语言基础【13】】函数、闭包、方法

文章目录 零、概述一、函数基础1、函数基础概念2、参数传递机制3、返回值特性3.1. 多返回值3.2. 命名返回值3.3. 错误处理 二、函数类型与高阶函数1. 函数类型定义2. 高阶函数&#xff08;函数作为参数、返回值&#xff09; 三、匿名函数与闭包1. 匿名函数&#xff08;Lambda函…...

Redis:现代应用开发的高效内存数据存储利器

一、Redis的起源与发展 Redis最初由意大利程序员Salvatore Sanfilippo在2009年开发&#xff0c;其初衷是为了满足他自己的一个项目需求&#xff0c;即需要一个高性能的键值存储系统来解决传统数据库在高并发场景下的性能瓶颈。随着项目的开源&#xff0c;Redis凭借其简单易用、…...

掌握 HTTP 请求:理解 cURL GET 语法

cURL 是一个强大的命令行工具&#xff0c;用于发送 HTTP 请求和与 Web 服务器交互。在 Web 开发和测试中&#xff0c;cURL 经常用于发送 GET 请求来获取服务器资源。本文将详细介绍 cURL GET 请求的语法和使用方法。 一、cURL 基本概念 cURL 是 "Client URL" 的缩写…...