ant-design-admin 开发之旅 (1)-- 先用起来再说

本贴最后更新于 2189 天前,其中的信息可能已经天翻地覆

1、 安装软件

下载 node.js8.x 及以上(推荐 v8.1.2),然后安装并通过命令行检查 node --version。下载 webstorm 2017.3 版,并输入 http://idea.codebeta.cn 注册。

**1、 **环境搭建

克隆项目文件:

git clone https://github.com/zuiidea/antd-admin.git

进入目录安装依赖:

#开始前请确保没有安装 roadhog、webpack 到 NPM 全局目录

npm i 或者 yarn install

编译运行:

npm run build:dll #第一次 npm run dev 时需运行此命令,使开发时编译更快

npm run dev

打开 http://localhost:8000

目录结构

├── /dist/ # 项目输出目录

├── /src/ # 项目源码目录

│ ├── /public/ # 公共文件,编译时 copy 至 dist 目录

│ ├── /components/ # UI 组件及 UI 相关方法

│ │ ├── skin.less # 全局样式

│ │ └── vars.less # 全局样式变量

│ ├── /routes/ # 路由组件

│ │ └── app.js # 路由入口

│ ├── /models/ # 数据模型

│ ├── /services/ # 数据接口

│ ├── /themes/ # 项目样式

│ ├── /mock/ # 数据 mock

│ ├── /utils/ # 工具函数

│ │ ├── config.js # 项目常规配置

│ │ ├── menu.js # 菜单及面包屑配置

│ │ ├── config.js # 项目常规配置

│ │ ├── request.js # 异步请求函数

│ │ └── theme.js # 项目需要在 js 中使用到样式变量

│ ├── route.js # 路由配置

│ ├── index.js # 入口文件

│ └── index.html

├── package.json # 项目信息

├── .eslintrc # Eslint 配置

└── .roadhogrc.js # roadhog 配置

**2、 **跨域和普通接口调用

项目自身是没有对外接口调用的,调用的数据都是 mock 数据,使用 roadhog 模拟接口,mock 接口和数据都保存在 ant-design-pro-master/.roadhogrc.mock.js 中。

接口调用代码: ant-design-pro-master/src/services/api.js

比如登录接口 fakeAccountLogin,原来调用的是 mock 数据:

export async function fakeAccountLogin(params) {

return request('/api/login/account', {

method: 'POST',

body: params,

});

}

其中/api/login/account 就是.roadhogrc.mock.js 里提供的 mock 接口,如果要调用本地的接口,做如下改动:

export async function fakeAccountLogin(params) {

return request('http://localhost:8080/xxx/accountManager/login', {

method: 'POST',

body: params,

});

}

即可。

跨域问题:

由于调用本地优购接口存在跨域问题,需要请求端(ant-design-admin)和客户端(XXX)都做调整, ant-design-admin 需要修改 ant-design-pro-master/src/utils/request.js 文件的 request 函数如下:

export default function request(url, options) {

const defaultOptions = {

credentials: 'include',

** mode: 'cors',**

};

const newOptions = { ...defaultOptions, ...options };

if (newOptions.method === 'POST' || newOptions.method === 'PUT') {

if (!(newOptions.body instanceof FormData)) {

newOptions.headers = {

Accept: 'application/json',

'Content-Type': 'application/json; charset=utf-8',

...newOptions.headers,

};

newOptions.body = JSON.stringify(newOptions.body);

} else {

// newOptions.body is FormData

newOptions.headers = {

Accept: 'application/json',

'Content-Type': 'multipart/form-data',

'Access-Control-Allow-Origin': '*',

'Access-Control-Allow-Headers': 'X-Requested-With',

'Access-Control-Allow-Methods': 'PUT,POST,GET,DELETE,OPTIONS',

...newOptions.headers,

};

}

}

服务端登录接口示例:

打开 src/com/xxx/controller/accountManager/ManagerController.java

新增代码如下:

@Resource(name="userService")

private UserService userService;

@Resource(name="roleService")

private RoleService roleService;

@Resource(name="menuService")

private MenuService menuService;

/**

  • 访问登录页 http://localhost:8080/xxx/accountManager/login

  • @return

*/

** @CrossOrigin("http://localhost:8000")**

@ApiOperation(value = "访问登录页")

@ResponseBody

@RequestMapping(value="/login",method={RequestMethod.POST},consumes="application/json;charset=UTF-8")

public ResponseEntity toLogin(@ApiParam @RequestBody(required=true) String body)throws Exception{

Map map = new HashMap();

Map params = new HashMap();

PageData pd = new PageData();

pd = this.getPageData();

String errInfo = "";

params = GsonTools.changeGsonToMap(body);

//shiro 管理的 session

Subject currentUser = SecurityUtils.getSubject();

Session session = currentUser.getSession();

if(null != params){

String sessionCode = (String)session.getAttribute(Const.SESSION_SECURITY_CODE); //获取 session 中的验证码

String USERNAME = params.get("userName");

String PASSWORD = params.get("password");

pd.put("USERNAME", USERNAME);

String passwd = new SimpleHash("SHA-1", USERNAME, PASSWORD).toString(); //密码加密

pd.put("PASSWORD", passwd);

pd = userService.getUserByNameAndPwd(pd);

if(pd != null){

pd.put("LAST_LOGIN",DateUtil.getTime().toString());

userService.updateLastLogin(pd);

User user = new User();

Role role = roleService.getRoleById(String.valueOf(pd.getLong("ROLE_ID")));

user.setUSER_ID(String.valueOf(pd.getLong("USER_ID")));

user.setUSERNAME(pd.getString("USERNAME"));

user.setPASSWORD(pd.getString("PASSWORD"));

user.setNAME(pd.getString("NAME"));

user.setRIGHTS(pd.getString("RIGHTS"));

user.setROLE_ID(String.valueOf(pd.getLong("ROLE_ID")));

user.setLAST_LOGIN(pd.getString("LAST_LOGIN"));

user.setIP(pd.getString("IP"));

           user.setSTATUS(pd.getString("STATUS"));

user.setPhone(pd.getString("PHONE"));

if(role != null){

user.setRole(role);

}

session.setAttribute(Const.SESSION_USER, user);

session.removeAttribute(Const.SESSION_SECURITY_CODE);

//shiro 加入身份验证

Subject subject = SecurityUtils.getSubject();

UsernamePasswordToken token = new UsernamePasswordToken(USERNAME, PASSWORD);

try {

subject.login(token);

} catch (AuthenticationException e) {

            logger.error(e.toString(), e);

errInfo = "身份验证失败!";

map.put("type", "account");

map.put("currentAuthority", "changeLoginStatus");

map.put("status", "false");

return new ResponseEntity(map,HttpStatus.OK);

}

}else{

errInfo = "usererror"; //用户名或密码有误

map.put("type", "account");

map.put("currentAuthority", "changeLoginStatus");

map.put("status", "false");

return new ResponseEntity(map,HttpStatus.OK);

}

}

if(Tools.isEmpty(errInfo)){

List allmenuList = new ArrayList();

User user = (User)session.getAttribute(Const.SESSION_USER);

if (user != null) {

User userr = (User)session.getAttribute(Const.SESSION_USERROL);

if(null == userr){

user = userService.getUserAndRoleById(user.getUSER_ID());

session.setAttribute(Const.SESSION_USERROL, user);

}else{

user = userr;

}

Role role = user.getRole();

String roleRights = role!=null ? role.getRIGHTS() : "";

//避免每次拦截用户操作时查询数据库,以下将用户所属角色权限、用户权限限都存入 session

session.setAttribute(Const.SESSION_ROLE_RIGHTS, roleRights); //将角色权限存入 session

session.setAttribute(Const.SESSION_USERNAME, user.getUSERNAME()); //放入用户名

if(null == session.getAttribute(Const.SESSION_allmenuList)){

allmenuList = menuService.listAllMenu();

if(Tools.notEmpty(roleRights)){

for(Menu menu : allmenuList){

menu.setHasMenu(RightsHelper.testRights(roleRights, menu.getMENU_ID()));

if(menu.isHasMenu()){

         List subMenuList = menu.getSubMenu();

for(Menu sub : subMenuList){

sub.setHasMenu(RightsHelper.testRights(roleRights, sub.getMENU_ID()));

                     }

}

}

}

session.setAttribute(Const.SESSION_allmenuList, allmenuList); //菜单权限放入 session 中

}else{

allmenuList = (List)session.getAttribute(Const.SESSION_allmenuList);

}

}

errInfo = "success";

map.put("type", "account");

map.put("currentAuthority", "admin");

map.put("status", "ok");

map.put("menuList", allmenuList);

//验证成功

return new ResponseEntity(map,HttpStatus.OK);

}

return null;

}

注意和原有接口注解和返回类型的不同

**3、 **菜单数据读取

Router 决定接口调用后使用哪个 model,文件在 ant-design-pro-master/src/common/router.js,登录时会指向 model BasicLayout ,具体如下

export const getRouterData = (app) => {

const routerConfig = {

'/': {

component: dynamicWrapper(app, ['user', 'login'], () => import('../layouts/BasicLayout')),

},

登录后 BasicLayout 会加载菜单组件,代码 ant-design-pro-master/src/layouts/BasicLayout.js

/**

  • 根据菜单取得重定向地址.

*/

const redirectData = [];

const getRedirect = (item) => {

if (item && item.children) {

if (item.children[0] && item.children[0].path) {

redirectData.push({

from: /${item.path},

to: /${item.children[0].path},

});

item.children.forEach((children) => {

getRedirect(children);

});

}

}

};

getMenuData().forEach(getRedirect);

getMenuData 是通过 import { getMenuData } from '../common/menu';调用的 menu.js 里的 demo 数据,如果要使用优购的菜单数据,就需要在登录成功时先把菜单数据存储到 localStorage,然后 BasicLayout 通过 localStorage 取出来,

在登录过程中增加存储菜单函数,具体如下:

在 model :ant-design-pro-master/src/models/login.js 中先引入 menu.js 中的存储函数:** **import { setMenuData } from '../common/menu';

effects: {

*login({ payload }, { call, put }) {

const response = yield call(fakeAccountLogin, payload);

yield put({

type: 'changeLoginStatus',

payload: response,

});

// Login successfully

if (response.status === 'ok') {

** setMenuData(response);**

reloadAuthorized();

yield put(routerRedux.push('/'));

}

},

setMenuData****如下:

export function setMenuData(menus) {

return localStorage.setItem('menus', menus);

}

获取菜单数据:

export const getMenuData = () => formatter(localStorage.getItem('menus'));

可以看到 setMenuData 会将登录接口返回的菜单数据存到 localStorage 中,BasicLayout.js 再通过 getMenuData 获取菜单数据:

/**

  • 根据菜单取得重定向地址.

*/

const redirectData = [];

const getRedirect = (item) => {

if (item && item.children) {

if (item.children[0] && item.children[0].path) {

redirectData.push({

from: /${item.path},

to: /${item.children[0].path},

});

item.children.forEach((children) => {

getRedirect(children);

});

}

}

};

getMenuData().forEach(getRedirect);

  • Ant-Design

    Ant Design 是服务于企业级产品的设计体系,基于确定和自然的设计价值观上的模块化解决方案,让设计者和开发者专注于更好的用户体验。

    17 引用 • 23 回帖 • 4 关注

相关帖子

欢迎来到这里!

我们正在构建一个小众社区,大家在这里相互信任,以平等 • 自由 • 奔放的价值观进行分享交流。最终,希望大家能够找到与自己志同道合的伙伴,共同成长。

注册 关于
请输入回帖内容 ...

推荐标签 标签

  • 996
    13 引用 • 200 回帖 • 11 关注
  • Q&A

    提问之前请先看《提问的智慧》,好的问题比好的答案更有价值。

    8447 引用 • 38477 回帖 • 154 关注
  • danl
    146 关注
  • 笔记

    好记性不如烂笔头。

    308 引用 • 793 回帖
  • 禅道

    禅道是一款国产的开源项目管理软件,她的核心管理思想基于敏捷方法 scrum,内置了产品管理和项目管理,同时又根据国内研发现状补充了测试管理、计划管理、发布管理、文档管理、事务管理等功能,在一个软件中就可以将软件研发中的需求、任务、bug、用例、计划、发布等要素有序的跟踪管理起来,完整地覆盖了项目管理的核心流程。

    5 引用 • 15 回帖 • 102 关注
  • V2Ray
    1 引用 • 15 回帖 • 1 关注
  • SpaceVim

    SpaceVim 是一个社区驱动的模块化 vim/neovim 配置集合,以模块的方式组织管理插件以
    及相关配置,为不同的语言开发量身定制了相关的开发模块,该模块提供代码自动补全,
    语法检查、格式化、调试、REPL 等特性。用户仅需载入相关语言的模块即可得到一个开箱
    即用的 Vim-IDE。

    3 引用 • 31 回帖 • 105 关注
  • 安装

    你若安好,便是晴天。

    132 引用 • 1184 回帖 • 1 关注
  • Git

    Git 是 Linux Torvalds 为了帮助管理 Linux 内核开发而开发的一个开放源码的版本控制软件。

    209 引用 • 358 回帖 • 1 关注
  • 爬虫

    网络爬虫(Spider、Crawler),是一种按照一定的规则,自动地抓取万维网信息的程序。

    106 引用 • 275 回帖 • 1 关注
  • Sandbox

    如果帖子标签含有 Sandbox ,则该帖子会被视为“测试帖”,主要用于测试社区功能,排查 bug 等,该标签下内容不定期进行清理。

    409 引用 • 1246 回帖 • 587 关注
  • Ubuntu

    Ubuntu(友帮拓、优般图、乌班图)是一个以桌面应用为主的 Linux 操作系统,其名称来自非洲南部祖鲁语或豪萨语的“ubuntu”一词,意思是“人性”、“我的存在是因为大家的存在”,是非洲传统的一种价值观,类似华人社会的“仁爱”思想。Ubuntu 的目标在于为一般用户提供一个最新的、同时又相当稳定的主要由自由软件构建而成的操作系统。

    126 引用 • 169 回帖
  • JavaScript

    JavaScript 一种动态类型、弱类型、基于原型的直译式脚本语言,内置支持类型。它的解释器被称为 JavaScript 引擎,为浏览器的一部分,广泛用于客户端的脚本语言,最早是在 HTML 网页上使用,用来给 HTML 网页增加动态功能。

    728 引用 • 1273 回帖 • 1 关注
  • 面试

    面试造航母,上班拧螺丝。多面试,少加班。

    325 引用 • 1395 回帖 • 1 关注
  • Notion

    Notion - The all-in-one workspace for your notes, tasks, wikis, and databases.

    7 引用 • 40 回帖
  • TensorFlow

    TensorFlow 是一个采用数据流图(data flow graphs),用于数值计算的开源软件库。节点(Nodes)在图中表示数学操作,图中的线(edges)则表示在节点间相互联系的多维数据数组,即张量(tensor)。

    20 引用 • 19 回帖 • 1 关注
  • 创造

    你创造的作品可能会帮助到很多人,如果是开源项目的话就更赞了!

    178 引用 • 997 回帖
  • Latke

    Latke 是一款以 JSON 为主的 Java Web 框架。

    71 引用 • 535 回帖 • 789 关注
  • 酷鸟浏览器

    安全 · 稳定 · 快速
    为跨境从业人员提供专业的跨境浏览器

    3 引用 • 59 回帖 • 26 关注
  • 七牛云

    七牛云是国内领先的企业级公有云服务商,致力于打造以数据为核心的场景化 PaaS 服务。围绕富媒体场景,七牛先后推出了对象存储,融合 CDN 加速,数据通用处理,内容反垃圾服务,以及直播云服务等。

    27 引用 • 225 回帖 • 163 关注
  • 新人

    让我们欢迎这对新人。哦,不好意思说错了,让我们欢迎这位新人!
    新手上路,请谨慎驾驶!

    52 引用 • 228 回帖
  • Markdown

    Markdown 是一种轻量级标记语言,用户可使用纯文本编辑器来排版文档,最终通过 Markdown 引擎将文档转换为所需格式(比如 HTML、PDF 等)。

    167 引用 • 1520 回帖
  • ZooKeeper

    ZooKeeper 是一个分布式的,开放源码的分布式应用程序协调服务,是 Google 的 Chubby 一个开源的实现,是 Hadoop 和 HBase 的重要组件。它是一个为分布式应用提供一致性服务的软件,提供的功能包括:配置维护、域名服务、分布式同步、组服务等。

    59 引用 • 29 回帖 • 14 关注
  • Webswing

    Webswing 是一个能将任何 Swing 应用通过纯 HTML5 运行在浏览器中的 Web 服务器,详细介绍请看 将 Java Swing 应用变成 Web 应用

    1 引用 • 15 回帖 • 637 关注
  • Sillot

    Insights(注意当前设置 master 为默认分支)

    汐洛彖夲肜矩阵(Sillot T☳Converbenk Matrix),致力于服务智慧新彖乄,具有彖乄驱动、极致优雅、开发者友好的特点。其中汐洛绞架(Sillot-Gibbet)基于自思源笔记(siyuan-note),前身是思源笔记汐洛版(更早是思源笔记汐洛分支),是智慧新录乄终端(多端融合,移动端优先)。

    主仓库地址:Hi-Windom/Sillot

    文档地址:sillot.db.sc.cn

    注意事项:

    1. ⚠️ 汐洛仍在早期开发阶段,尚不稳定
    2. ⚠️ 汐洛并非面向普通用户设计,使用前请了解风险
    3. ⚠️ 汐洛绞架基于思源笔记,开发者尽最大努力与思源笔记保持兼容,但无法实现 100% 兼容
    29 引用 • 25 回帖 • 86 关注
  • InfluxDB

    InfluxDB 是一个开源的没有外部依赖的时间序列数据库。适用于记录度量,事件及实时分析。

    2 引用 • 76 关注
  • VirtualBox

    VirtualBox 是一款开源虚拟机软件,最早由德国 Innotek 公司开发,由 Sun Microsystems 公司出品的软件,使用 Qt 编写,在 Sun 被 Oracle 收购后正式更名成 Oracle VM VirtualBox。

    10 引用 • 2 回帖