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

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

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 回帖 • 2 关注

相关帖子

欢迎来到这里!

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

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

推荐标签 标签

  • RabbitMQ

    RabbitMQ 是一个开源的 AMQP 实现,服务器端用 Erlang 语言编写,支持多种语言客户端,如:Python、Ruby、.NET、Java、C、PHP、ActionScript 等。用于在分布式系统中存储转发消息,在易用性、扩展性、高可用性等方面表现不俗。

    49 引用 • 60 回帖 • 399 关注
  • 开源中国

    开源中国是目前中国最大的开源技术社区。传播开源的理念,推广开源项目,为 IT 开发者提供了一个发现、使用、并交流开源技术的平台。目前开源中国社区已收录超过两万款开源软件。

    7 引用 • 86 回帖
  • Ant-Design

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

    17 引用 • 23 回帖 • 2 关注
  • ZooKeeper

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

    59 引用 • 29 回帖 • 18 关注
  • Jenkins

    Jenkins 是一套开源的持续集成工具。它提供了非常丰富的插件,让构建、部署、自动化集成项目变得简单易用。

    51 引用 • 37 回帖
  • LaTeX

    LaTeX(音译“拉泰赫”)是一种基于 ΤΕΧ 的排版系统,由美国计算机学家莱斯利·兰伯特(Leslie Lamport)在 20 世纪 80 年代初期开发,利用这种格式,即使使用者没有排版和程序设计的知识也可以充分发挥由 TeX 所提供的强大功能,能在几天,甚至几小时内生成很多具有书籍质量的印刷品。对于生成复杂表格和数学公式,这一点表现得尤为突出。因此它非常适用于生成高印刷质量的科技和数学类文档。

    9 引用 • 32 回帖 • 166 关注
  • JetBrains

    JetBrains 是一家捷克的软件开发公司,该公司位于捷克的布拉格,并在俄国的圣彼得堡及美国麻州波士顿都设有办公室,该公司最为人所熟知的产品是 Java 编程语言开发撰写时所用的集成开发环境:IntelliJ IDEA

    18 引用 • 54 回帖 • 2 关注
  • Dubbo

    Dubbo 是一个分布式服务框架,致力于提供高性能和透明化的 RPC 远程服务调用方案,是 [阿里巴巴] SOA 服务化治理方案的核心框架,每天为 2,000+ 个服务提供 3,000,000,000+ 次访问量支持,并被广泛应用于阿里巴巴集团的各成员站点。

    60 引用 • 82 回帖 • 609 关注
  • SMTP

    SMTP(Simple Mail Transfer Protocol)即简单邮件传输协议,它是一组用于由源地址到目的地址传送邮件的规则,由它来控制信件的中转方式。SMTP 协议属于 TCP/IP 协议簇,它帮助每台计算机在发送或中转信件时找到下一个目的地。

    4 引用 • 18 回帖 • 588 关注
  • GAE

    Google App Engine(GAE)是 Google 管理的数据中心中用于 WEB 应用程序的开发和托管的平台。2008 年 4 月 发布第一个测试版本。目前支持 Python、Java 和 Go 开发部署。全球已有数十万的开发者在其上开发了众多的应用。

    14 引用 • 42 回帖 • 687 关注
  • Hprose

    Hprose 是一款先进的轻量级、跨语言、跨平台、无侵入式、高性能动态远程对象调用引擎库。它不仅简单易用,而且功能强大。你无需专门学习,只需看上几眼,就能用它轻松构建分布式应用系统。

    9 引用 • 17 回帖 • 597 关注
  • 安装

    你若安好,便是晴天。

    128 引用 • 1184 回帖
  • Angular

    AngularAngularJS 的新版本。

    26 引用 • 66 回帖 • 512 关注
  • PWL

    组织简介

    用爱发电 (Programming With Love) 是一个以开源精神为核心的民间开源爱好者技术组织,“用爱发电”象征开源与贡献精神,加入组织,代表你将遵守组织的“个人开源爱好者”的各项条款。申请加入:用爱发电组织邀请帖
    用爱发电组织官网:https://programmingwithlove.stackoverflow.wiki/

    用爱发电组织的核心驱动力:

    • 遵守开源守则,体现开源&贡献精神:以分享为目的,拒绝非法牟利。
    • 自我保护:使用适当的 License 保护自己的原创作品。
    • 尊重他人:不以各种理由、各种漏洞进行未经允许的抄袭、散播、洩露;以礼相待,尊重所有对社区做出贡献的开发者;通过他人的分享习得知识,要留下足迹,表示感谢。
    • 热爱编程、热爱学习:加入组织,热爱编程是首当其要的。我们欢迎热爱讨论、分享、提问的朋友,也同样欢迎默默成就的朋友。
    • 倾听:正确并恳切对待、处理问题与建议,及时修复开源项目的 Bug ,及时与反馈者沟通。不抬杠、不无视、不辱骂。
    • 平视:不诋毁、轻视、嘲讽其他开发者,主动提出建议、施以帮助,以和谐为本。只要他人肯努力,你也可能会被昔日小看的人所超越,所以请保持谦虚。
    • 乐观且活跃:你的努力决定了你的高度。不要放弃,多年后回头俯瞰,才会发现自己已经成就往日所仰望的水平。积极地将项目开源,帮助他人学习、改进,自己也会获得相应的提升、成就与成就感。
    1 引用 • 487 回帖 • 6 关注
  • RIP

    愿逝者安息!

    8 引用 • 92 回帖 • 291 关注
  • IBM

    IBM(国际商业机器公司)或万国商业机器公司,简称 IBM(International Business Machines Corporation),总公司在纽约州阿蒙克市。1911 年托马斯·沃森创立于美国,是全球最大的信息技术和业务解决方案公司,拥有全球雇员 30 多万人,业务遍及 160 多个国家和地区。

    16 引用 • 53 回帖 • 123 关注
  • MyBatis

    MyBatis 本是 Apache 软件基金会 的一个开源项目 iBatis,2010 年这个项目由 Apache 软件基金会迁移到了 google code,并且改名为 MyBatis ,2013 年 11 月再次迁移到了 GitHub。

    170 引用 • 414 回帖 • 429 关注
  • 数据库

    据说 99% 的性能瓶颈都在数据库。

    330 引用 • 614 回帖
  • 小说

    小说是以刻画人物形象为中心,通过完整的故事情节和环境描写来反映社会生活的文学体裁。

    28 引用 • 108 回帖
  • JWT

    JWT(JSON Web Token)是一种用于双方之间传递信息的简洁的、安全的表述性声明规范。JWT 作为一个开放的标准(RFC 7519),定义了一种简洁的,自包含的方法用于通信双方之间以 JSON 的形式安全的传递信息。

    20 引用 • 15 回帖 • 19 关注
  • OpenResty

    OpenResty 是一个基于 NGINX 与 Lua 的高性能 Web 平台,其内部集成了大量精良的 Lua 库、第三方模块以及大多数的依赖项。用于方便地搭建能够处理超高并发、扩展性极高的动态 Web 应用、Web 服务和动态网关。

    17 引用 • 39 关注
  • 倾城之链
    23 引用 • 66 回帖 • 101 关注
  • 黑曜石

    黑曜石是一款强大的知识库工具,支持本地 Markdown 文件编辑,支持双向链接和关系图。

    A second brain, for you, forever.

    10 引用 • 85 回帖
  • 国际化

    i18n(其来源是英文单词 internationalization 的首末字符 i 和 n,18 为中间的字符数)是“国际化”的简称。对程序来说,国际化是指在不修改代码的情况下,能根据不同语言及地区显示相应的界面。

    7 引用 • 26 回帖 • 2 关注
  • 创业

    你比 99% 的人都优秀么?

    82 引用 • 1398 回帖 • 1 关注
  • Sublime

    Sublime Text 是一款可以用来写代码、写文章的文本编辑器。支持代码高亮、自动完成,还支持通过插件进行扩展。

    10 引用 • 5 回帖
  • 书籍

    宋真宗赵恒曾经说过:“书中自有黄金屋,书中自有颜如玉。”

    76 引用 • 390 回帖