前端采用蚂蚁金服的前端集成框架:ant-design-pro, 官网:https://pro.ant.design/
前端采用ant-design,其中是用react写的,所以针对后端开发人员而言,学习也不是那么快,该文章用于能够快速的让后端人员上手。
注意:下面介绍的是基于ant-design-pro的v2.x版本
一、主要文件
这里主要介绍,平常开发中可能用到的一些文件,这里分为两类:
配置文件
界面文件
1.配置文件
该文件用于非界面文件,用于项目中的配置的介绍
- 后端配置文件
- 菜单中文设置文件
- 菜单层级设置文件
1.后端配置文件
这里在当前的ant-design版本中文件为
/src/config.js
// 其中最重要的是这里
proxy: {
// 下面是在遇到这种路径的时候进行替换
'/like/tina/api/v1/': {
// 自己的路径,其中8084是后端的端口号
target: 'http://localhost:8084/like/tina/api/v1/',
changeOrigin: true,
pathRewrite: { '^/like/tina/api/v1': '' },
},
},
其中该配置文件主要用于 xxxxApi.js 文件针对后端的url的替换
注意:这里面最后请求的路径会自动修改掉,但是在浏览器下面查看的ip是不会变的,参考(这里),但是实际的链接已经被代理成,而且能够获取数据,请注意
在console中展示的可能还是Http://localhost:8000/xxxxx,但是实际上已经替换为我们自己的这个了,比如:http://localhost:8080/like/tina/api/v1//user/table
2.菜单中文设置文件
文件为(其中我们只考虑中文的转换,其他的暂时不考虑):
/src/locales/zh-CN/menu.js
文件介绍,该文件主要是给左侧菜单中的中文展示使用
export default {
'menu.home': '首页',
'menu.dashboard': 'Dashboard',
'menu.dashboard.analysis': '分析页',
'menu.dashboard.monitor': '监控页',
'menu.dashboard.workplace': '工作台',
'menu.form': '表单页',
'menu.form.basicform': '基础表单',
'menu.form.stepform': '分步表单',
'menu.form.stepform.info': '分步表单(填写转账信息)',
'menu.form.stepform.confirm': '分步表单(确认转账信息)',
'menu.form.stepform.result': '分步表单(完成)',
'menu.form.advancedform': '高级表单',
'menu.list': '列表页',
'menu.list.searchtable': '查询表格',
'menu.list.basiclist': '标准列表',
'menu.list.cardlist': '卡片列表',
'menu.list.searchlist': '搜索列表',
'menu.list.searchlist.articles': '搜索列表(文章)',
'menu.list.searchlist.projects': '搜索列表(项目)',
'menu.list.searchlist.applications': '搜索列表(应用)',
'menu.config': '配置',
'menu.config.configgrouplist': '配置组',
'menu.config.configitemlist': '配置项',
'menu.profile': '详情页',
'menu.profile.basic': '基础详情页',
'menu.profile.advanced': '高级详情页',
'menu.result': '结果页',
'menu.result.success': '成功页',
'menu.result.fail': '失败页',
'menu.exception': '异常页',
'menu.exception.not-permission': '403',
'menu.exception.not-find': '404',
'menu.exception.server-error': '500',
'menu.exception.trigger': '触发错误',
'menu.account': '个人页',
'menu.account.center': '个人中心',
'menu.account.settings': '个人设置',
'menu.account.trigger': '触发报错',
'menu.account.logout': '退出登录',
'menu.task': '调度',
'menu.task.tasklist': '任务调度',
};
比如其中的一条数据:menu.task.tasklist:‘任务调度’,该条数据用于在菜单层级设置中,比如如下,其中就是在界面展示的
export default [
...
{
path: '/task',
icon: 'project',
name: 'task',
routes: [
{
path: '/task/task-list',
// 这里就是跟中文对应的文件名
name: 'tasklist',
component: './like/TaskList',
},
],
},
...
];
3.菜单层级设置文件
文件为:
/config/router.config.js
该文件用于显示菜单,还是比较重要的,但是要展示自己的菜单的话,还是要自己进行单独设置,不过一键式工具中我们已经可以直接生成了,下面主要针对文件进行介绍下,全量文件如下
export default [
// user
{
path: '/user',
component: '../layouts/UserLayout',
routes: [
{ path: '/user', redirect: '/user/login' },
{ path: '/user/login', component: './User/Login' },
{ path: '/user/register', component: './User/Register' },
{ path: '/user/register-result', component: './User/RegisterResult' },
],
},
// app
{
path: '/',
component: '../layouts/BasicLayout',
Routes: ['src/pages/Authorized'],
authority: ['admin', 'user'],
routes: [
// dashboard
{ path: '/', redirect: '/dashboard/analysis' },
{
path: '/config',
icon: 'setting',
name: 'config',
routes: [
{
path: '/config/config-group-list',
name: 'configgrouplist',
component: './config/ConfigGroupList',
},
{
path: '/config/config-item-list',
name: 'configitemlist',
component: './config/ConfigItemList',
},
],
},
{
path: '/task',
icon: 'project',
name: 'task',
routes: [
{
path: '/task/task-list',
name: 'tasklist',
component: './like/TaskList',
},
],
},
{
component: '404',
},
],
},
];
其中如果有routes则就表示是父级菜单,可以进行测试下,其中内部的为子菜单,而子菜单还是可以继续向下拆分,其中component就是指向我们自己的component,就是界面xxxList.js文件
2.界面文件
1.界面展示文件(默认用List结尾)
自动生成工具会直接生成对应的表名+List.js,该文件就是直接在界面上展示的文件,下面主要介绍下这文件的结构
import React, { PureComponent } from 'react';
import { connect } from 'dva';
// 引入antdesign的组件
import {
Row,
Col,
Card,
Badge,
Form,
Input,
Button,
Table,
Icon,
Select,
Divider,
Drawer,
Pagination,
InputNumber,
Tabs,
Modal,
} from 'antd';
// 引用第三方文件
import 'codemirror/theme/solarized.css';
// 下面@connect 这个是将Model文件中的数据和当前界面List关联起来
/* eslint react/no-multi-comp:0 */
@connect(({ taskModel, loading }) => ({
// 这里是Model文件的namespace(注意:一定要保持一致)
taskModel,
loading: loading.models.taskModel,
}))
// @Form.create() 是一个注解,就简化了xxx = Form.create(xxx);export xxx
@Form.create()
class TaskList extends PureComponent {
// 这个是List的一些数据,也可以放到Model中
state = {
addModalVisible: false,
editModalVisible: false,
item: {},
};
// 界面展示的列
columns = [
{
name: 'id',
title: 'index',
dataIndex: 'id',
width: '8%',
},
];
// 界面初始化
componentDidMount() {
...
}
// 普通的函数
onClose = () => {
// 重要:当我们要调用Model中的函数的时候,应该通过dispatch来调用
const { dispatch } = this.props;
// 其中
dispatch({
type: 'configItem/getListCount',
payload: {
paneIndex: index,
searchParam: param,
},
});
};
// 界面加载,每次数据有更新,该函数都会执行
render() {
// 变量引用:props方式,props是外部数据引用过来的变量,比如Model文件,我们这里引入,Model中的数据就应该这样使用
const {
taskModel: { selectState, groupAllCodeList, drawerRecord, resultOfRun, drawerVisible },
} = this.props;
// 变量引用:state方式,这个其中的变量都是自己内部的
const { addModalVisible, editModalVisible, item } = this.state;
// 定义的一些定量
const components = {
body: {
row: EditableFormRow,
cell: EditableCell,
},
};
// 封装的表格
const tabPanes = panes.map(pane => (
<Tabs.TabPane tab={pane.title} key={pane.name}>
<Card bordered={false}>
<div className={styles.tableList}>
{/*这里是搜索框部分*/}
<div className={styles.tableListForm}>{this.renderSearchForm()}</div>
<div className={styles.tableListOperator} />
{/*这是表格*/}
<Table
rowKey={record => record.id}
dataSource={pane.content.tableList}
columns={this.columns}
loading={pane.content.tableLoading}
pagination={false}
expandedRowRender={this.expandedRowRender}
/>
{/*这里是页面分页部分*/}
<Pagination
showQuickJumper
onChange={this.onChange}
defaultCurrent={1}
total={pane.content.totalNumber}
current={pane.content.pager.pageNo}
defaultPageSize={pane.content.pager.pageSize}
/>
</div>
</Card>
</Tabs.TabPane>
));
// 这里返回界面元素
return (
// 该标签为antdesign的的标签,用于菜单和界面显示
<PageHeaderWrapper>
<Tabs
onChange={this.onTabChange}
activeKey={activePaneName}
defaultActiveKey="1"
type="editable-card"
onEdit={this.onEdit}
>
{tabPanes}
</Tabs>
</PageHeaderWrapper>
);
}
}
export default TaskList;java
2.界面数据文件(默认用Model结尾)
// 引入Api文件
import {
pageList,
add,
deleteData,
update,
pageCount,
getCodeList,
getAllCodeList,
disable,
enable,
run,
handRun,
} from '@/services/taskApi';
export default {
namespace: 'taskModel', // 这个是标示当前model的,用于跟List文件关联
// 下面是定义的数据模型
state: {
maxTabIndex: 1, // 最大的标签页索引,用于标签新增计数用
activePaneName: '1', // tabPane 的激活的key
tabIndexList: ['1'], // 当前存在的标签的列表
panes: [
{
name: '1',
title: '任务调度1',
content: {
tableList: [],
tableLoading: false,
searchParam: {},
totalNumber: 0,
pager: {
pageNo: 1,
pageSize: 20,
},
},
},
],
},
// 异步处理函数
effects: {
// 增加组配置,payload为List调度过来的参数,call为调用Api函数的方法,put为调用其他异步或者同步的方法
*add({ payload }, { call, put }) {
console.log('task.add 参数:');
// 打印复杂函数
console.log(JSON.stringify(payload));
// 调用Api文件中的某个函数add,这里
const response = yield call(add, payload);
yield put({
type: 'handleAddResult',
payload: response,
});
// 调用其他的函数,同步或者异步函数均可
yield put({
type: 'tableFresh',
payload: {
paneIndex: payload.paneIndex,
},
});
},
// 其他函数
...
},
// 同步函数
reducers: {
// 同步函数xxx
setTableLoading(state) {
const newPanes = state.panes;
const index = newPanes.findIndex(pane => pane.name === state.activePaneName);
newPanes[index].content.tableLoading = true;
// 下面这个retuen 在同步函数中必须存在
return {
...state, // 这里...是不修改除了下面这个panes之外的其他数据,注意:...state必须有,否则其他的数据都会被清空
panes: newPanes, // 这里是更新state中的panes字段
};
},
// 其他函数
...
},
};
3.跟后端交互文件(默认用Api结尾)
该文件是跟后端发送url的地方
import request from '@/utils/request';
// 这里我用一个变量统一表示,这个正好可以给config.js文件中的代理进行替换
const path = '/like/tina/api/v1';
export async function add(params) {
console.log('taskApi.add 发送的参数');
console.log(JSON.stringify(params));
// request是框架封装的函数,body为对应的参数,method不填写,则默认为GET
return request(`${path}/task/add`, {
method: 'PUT',
body: {
...params,
},
});
}
export async function deleteData(params) {
console.log('taskApi.deleteData 发送的参数');
console.log(JSON.stringify(params));
return request(`${path}/task/delete/${params}`, {
method: 'DELETE',
});
}
//
export async function getCodeList() {
console.log('taskApi.getCodeList 发送');
return request(`${path}/task/codeList`);
}
export async function update(params) {
console.log('taskApi.update 发送的参数');
console.log(JSON.stringify(params));
return request(`${path}/task/update`, {
method: 'POST',
body: {
...params,
},
});
}
二、页面加载流程
整体的流程是这样的:
List->Model→Api——>后端
其中Model中放置的是数据,发起方是List模块的dispatch,一旦数据获取到更新到Model中,List界面会自动刷新,整体的流程图是这样的
解释:
connect :对应@connect这个注解
dispatch:对应的是dispatch函数
action:新的版本中已经将action这个模块删除掉了,早期的写法中是还有一层Action,List那边dispatch的其实是Action这一层中的函数,而这一层再调用Model,现在新的版本中已经直接采用dispatch调用Model中的函数了,Action层已经嵌入到内核中
Model:对应的就是我们的Model文件
Effect:这个是Model中的异步函数,用于界面中调用获取数据比较耗时的操作,一般用于调用后端数据
Reducer:同步函数,用于界面中的数据,一般用于在界面处理后可以直接更新,并展示到界面中,是阻塞性函数
Subscriptioin:这个函数我们这边用的不多,主要是一种订阅,详细可参考这里:https://dvajs.com/guide/concepts.html#effectServer:就是我们的后端
三、文件数据用法
其中主要介绍下List怎么调用Model中的数据,其实在上面的List文件中已经介绍了,下面简单介绍下
// 变量引用:props方式,props是外部数据引用过来的变量,比如Model文件,我们这里引入,Model中的数据就应该这样使用
const {
taskModel: { selectState, groupAllCodeList, drawerRecord, resultOfRun, drawerVisible },
} = this.props;
要这样写才行,引用某个变量的数据
四、注意事项(重要)
主要介绍一些语法糖,很多都是小细节,小坑,网上资料也较少,建议关注下
1.变量的…含义
// 同步函数xxx
setTableLoading(state) {
const newPanes = state.panes;
const index = newPanes.findIndex(pane => pane.name === state.activePaneName);
newPanes[index].content.tableLoading = true;
// 下面这个retuen 在同步函数中必须存在
return {
...state, // 这里...是不修改除了下面这个panes之外的其他数据,注意:...state必须有,否则其他的数据都会被清空
panes: newPanes, // 这里是更新state中的panes字段
};
},
在代码中经常会见到这种“…”三个点的变量,这种表示是将变量的key展示出来
比如如果一个类型是这种:tem:{“key1”:123,“key2”:222},这个变量为tem,…tem 这个就表示:“key1”:123, “key2”:222,其实就是将外层剥离
2.变量,这种写法
在代码中有时候会见到这种写法,如下
dispatch({
type: 'configItem/getPageList',
payload: {
paneIndex: index,
pager, // 这里其实就是pager:pager, 这个的简化版,就是key和value的名字相同,就可以这样写,而且建议这样写,否则核查会有告警
searchParam: param,
},
});
3.变量的打印
在日志中打印平常都是使用console.log(xxxx) 即可,但是对于复杂类型,这个打印就显示有点复杂,这里建议使用JSON打印,如下这样
console.log(JSON.stringify(item));
注意:
其中不能这么写,如下:
console.log("变量为:"+JSON.stringify(item));
不能将JSON的解析和前面的字符链接,这样后来发现,就是这个JSON不生效
4.函数的写法
函数的写法跟旧的js写法不一样,没有function关键字,平常函数都这么写
// 没有参数
renderSearchForm = () => {
};
// 一个参数
enable = record => {
};
// 多个参数
contain = (object1, object2) => {
};
5.列中属性函数调用写法
在column中可能会添加Button或者Icon等控件,进行函数的跳转,在写的时候一定要注意使用如下的方式
{
key: 'delete',
title: '删除',
dataIndex: 'delete',
editable: false,
width: '5%',
render: (text, row) => (
<span>
<Button type="danger" icon="delete" onClick={() => this.showDeleteConfirm(row)} />
</span>
),
},
注意:其中函数调用一定要写成
onClick={() => this.fun()}
而不能写成如下: 下面这个会自动调用函数fun()
onClick={this.fun()}
6.js文件中的注释
其中"//"之后要添加一个空格,否则会报错
7.List页面调用
这个页面可以调用异步也可以调用同步函数,但是异步一般用于获取数据耗时比较长的,一般用于从后端获取数据并覆盖前端数据,同步函数用于直接将进行数据操作
8.Model页面函数调用
Model页面中有两种函数,异步函数和同步函数,但是异步函数可以调用同步函数也可以调用异步函数,而同步函数不可以调用其他的函数。
五、参考
dva概念:https://dvajs.com/guide/concepts.html#effectreact语法:https://www.jianshu.com/p/7e872afeae42antdesign官网:https://pro.ant.design/docs/getting-started-cnantdesign组件库:https://ant.design/docs/react/introduce-cnantdesign目录解释:https://pro.ant.design/docs/getting-started-cnesignpro详细介绍:https://www.ctolib.com/topics-134955.html
六、学习参考:
https://zhuanlan.zhihu.com/p/36134136https://pro.ant.design/index-cnhttps://www.ru23.com/note/ab0115f0.html