2016-09-18 17:39:11 +08:00
---
2019-12-17 22:25:28 +08:00
order: 3
2016-09-18 17:39:11 +08:00
title: 项目实战
---
2020-07-08 16:15:11 +08:00
在真实项目开发中,你可能会需要 Redux 或者 MobX 这样的数据流方案, Ant Design React 作为一个 UI 库,可以和任何 React 生态圈内的数据流方案以及应用框架搭配使用。我们基于业务场景的场景,推出了可插拔的企业级应用框架 umi, 推荐你在项目中使用。
2016-09-18 17:39:11 +08:00
2021-02-25 10:51:43 +08:00
[umi ](https://umijs.org/zh-CN ) 则是一个可插拔的企业级 react 应用框架。umi 以路由为基础的,支持[类 next.js 的约定式路由](https://umijs.org/zh-CN/docs/convention-routing),以及各种进阶的路由功能,并以此进行功能扩展,比如[支持路由级的按需加载](https://umijs.org/zh-CN/config#dynamicimport)。然后配以完善的[插件体系](https://umijs.org/zh-CN/plugins/api),覆盖从源码到构建产物的每个生命周期,支持各种功能扩展和业务需求,同时提供 [Umi UI ](https://umijs.org/zh-CN/docs/use-umi-ui ) 通过可视化辅助编程( VAP) 提高开发体验和研发效率。
2016-09-18 17:39:11 +08:00
2018-10-10 20:21:15 +08:00
> 你可能也会对 [Ant Design Pro](https://pro.ant.design/) 感兴趣,这是一个基于 umi、dva 和 ant design 的开箱即用的中台前端/设计解决方案。
2016-09-18 17:39:11 +08:00
2020-03-04 16:12:06 +08:00
本文会引导你使用 Umi、dva 和 antd 从 0 开始创建一个简单应用。
2016-09-18 17:39:11 +08:00
2020-03-04 16:12:06 +08:00
## 安装 Umi
2019-10-23 18:00:15 +08:00
2020-03-04 16:12:06 +08:00
推荐使用 yarn 创建 Umi 脚手架,执行以下命令。
2016-09-18 17:39:11 +08:00
```bash
2020-03-04 16:12:06 +08:00
$ mkdir myapp & & cd myapp
2021-02-25 15:45:05 +08:00
$ yarn create @umijs/umi -app
2020-03-04 16:12:06 +08:00
$ yarn
2016-09-18 17:39:11 +08:00
```
2021-02-25 15:45:05 +08:00
> 如果你使用 npm, 可执行 `npx @umijs/create-umi-app`,效果一致。
2016-09-18 17:39:11 +08:00
2021-02-25 15:45:05 +08:00
> 默认使用 `"antd": "^4.0.0"`,如果要使用固定版本的 antd, 你可以在项目里安装额外的 antd 依赖,`package.json` 里声明的 antd 依赖会被优先使用。
2016-09-18 17:39:11 +08:00
2018-10-10 20:21:15 +08:00
## 新建路由
2016-09-18 17:39:11 +08:00
我们要写个应用来先显示产品列表。首先第一步是创建路由,路由可以想象成是组成应用的不同页面。
2018-10-10 20:21:15 +08:00
然后通过命令创建 `/products` 路由,
```bash
2020-03-04 16:12:06 +08:00
$ npx umi g page products --typescript
Write: src/pages/products.tsx
Write: src/pages/products.css
```
在 `.umirc.ts` 中配置路由,如果有国际化需要,可以配置 `locale` 开启 antd 国际化:
2016-09-18 17:39:11 +08:00
2020-03-04 16:12:06 +08:00
```diff
import { defineConfig } from 'umi';
export default defineConfig({
+ locale: { antd: true },
routes: [
{ path: '/', component: '@/pages/index' },
+ { path: '/products', component: '@/pages/products' },
],
});
2016-09-18 17:39:11 +08:00
```
2020-03-04 16:12:06 +08:00
运行 `yarn start` 然后在浏览器里打开 [http://localhost:8000/products ](http://localhost:8000/products ),你应该能看到对应的页面。
2016-09-18 17:39:11 +08:00
## 编写 UI Component
2018-10-10 20:21:15 +08:00
随着应用的发展,你会需要在多个页面分享 UI 元素 (或在一个页面使用多次),在 umi 里你可以把这部分抽成 component 。
2016-09-18 17:39:11 +08:00
我们来编写一个 `ProductList` component, 这样就能在不同的地方显示产品列表了。
2020-03-04 16:12:06 +08:00
然后新建 `src/components/ProductList.tsx` 文件:
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
```tsx
2016-09-18 17:39:11 +08:00
import { Table, Popconfirm, Button } from 'antd';
2020-07-08 16:15:11 +08:00
const ProductList: React.FC< { products: { name: string }[]; onDelete: (id: string) => void }> = ({
onDelete,
products,
}) => {
2019-05-07 14:57:32 +08:00
const columns = [
{
title: 'Name',
dataIndex: 'name',
2016-09-18 17:39:11 +08:00
},
2019-05-07 14:57:32 +08:00
{
title: 'Actions',
render: (text, record) => {
return (
< Popconfirm title = "Delete?" onConfirm = {() = > onDelete(record.id)}>
< Button > Delete< / Button >
< / Popconfirm >
);
},
},
];
return < Table dataSource = {products} columns = {columns} / > ;
2016-09-18 17:39:11 +08:00
};
export default ProductList;
```
2020-07-08 16:15:11 +08:00
## 简单数据流方案
2016-09-18 17:39:11 +08:00
2020-11-08 15:04:02 +08:00
`@umijs/plugin-model` 是一种基于 hooks 范式的简单数据流方案,可以在一定情况下替代 dva 来进行中台的全局数据流。我们约定在 `src/models` 目录下的文件为项目定义的 model 文件。每个文件需要默认导出一个 function, 该 function 定义了一个 Hook, 不符合规范的文件我们会过滤掉。
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
文件名则对应最终 model 的 name, 你可以通过插件提供的 API 来消费 model 中的数据。
2016-09-18 17:39:11 +08:00
2022-02-10 18:24:53 +08:00
我们以一个简单的表格作为示例。首先新建一个 `src/services/product.ts` 存放产品相关的接口。
```tsx
/*
export function queryProductList() {
return fetch('/api/products').then(res => res.json());
}
*/
// 先用 setTimeout 模拟一个请求,正常的写法如上所示
export function queryProductList() {
return new Promise(resolve => {
setTimeout(() => {
resolve([
{
id: 1,
name: 'dva',
},
{
id: 2,
name: 'antd',
},
]);
}, 2000);
});
}
```
然后新建文件 `src/models/useProductList.ts` 。
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
```tsx
import { useRequest } from 'umi';
import { queryProductList } from '@/services/product';
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
export default function useProductList(params: { pageSize: number; current: number }) {
2021-12-27 10:03:17 +08:00
const msg = useRequest(() => queryProductList(params));
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
const deleteProducts = async (id: string) => {
try {
await removeProducts(id);
message.success('success');
msg.run();
} catch (error) {
message.error('fail');
}
};
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
return {
dataSource: msg.data,
reload: msg.run,
loading: msg.loading,
deleteProducts,
};
}
```
2016-09-18 17:39:11 +08:00
2020-03-04 16:12:06 +08:00
编辑 `src/pages/products.tsx` ,替换为以下内容:
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
```tsx
import { useModel } from 'umi';
2020-03-04 16:12:06 +08:00
import ProductList from '@/components/ProductList';
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
const Products = () => {
const { dataSource, reload, deleteProducts } = useModel('useProductList');
2016-09-18 17:39:11 +08:00
return (
< div >
2020-07-08 16:15:11 +08:00
< a onClick = {() = > reload()}>reload< / a >
< ProductList onDelete = {deleteProducts} products = {dataSource} / >
2016-09-18 17:39:11 +08:00
< / div >
);
};
2020-07-08 16:15:11 +08:00
export default Products;
2016-09-18 17:39:11 +08:00
```
2020-03-04 16:12:06 +08:00
执行启动命令:
2018-10-10 20:21:15 +08:00
2020-03-04 16:12:06 +08:00
```bash
$ yarn start
2016-09-18 17:39:11 +08:00
```
2020-03-04 16:12:06 +08:00
访问 [http://localhost:8000 ](http://localhost:8000/ ),应该能看到以下效果:
2016-09-18 17:39:11 +08:00
2020-03-04 16:12:06 +08:00
< img src = "https://gw.alipayobjects.com/zos/antfincdn/dPsy4tFHN3/umi.gif" / >
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
## ProLayout
一个标准的中后台页面, 一般都需要一个布局, 这个布局很多时候都是高度雷同的, ProLayout 封装了常用的菜单,面包屑,页头等功能,提供了一个不依赖的框架且开箱即用的高级布局组件。
并且支持 `side` , `mix` , `top` 三种模式,更是内置了菜单选中,菜单生成面包屑,自动设置页面标题的逻辑。可以帮助你快速的开始一个项目。
![site ](https://gw.alipayobjects.com/zos/antfincdn/gXkuc%26RmT7/64038246-E2BF-4840-8898-5AF531897A44.png )
使用方式也是极为简单,只需要进行几个简单的设置。
```tsx
import { Button } from 'antd';
import ProLayout, { PageContainer } from '@ant-design/pro-layout';
export default (
< ProLayout >
< PageContainer
extra={[
< Button key = "3" > Operating< / Button > ,
< Button key = "2" > Operating< / Button > ,
< Button key = "1" type = "primary" >
Main Operating
< / Button > ,
]}
footer={[< Button > reset< / Button > , < Button type = "primary" > submit< / Button > ]}
>
{children}
< / PageContainer >
< / ProLayout >
);
```
2022-02-16 19:50:38 +08:00
点击这里[快速开始](https://procomponents.ant.design/components/layout)。
2020-07-08 16:15:11 +08:00
## ProTable
一个中后台页面中很多数据都不需要跨页面共享, models 在一些时候也是不需要的。
```tsx
import ProTable from '@ant-design/pro-table';
import { Popconfirm, Button } from 'antd';
import { queryProductList } from '@/services/product';
const Products = () => {
const actionRef = useRef< ActionType > ();
const deleteProducts = async (id: string) => {
try {
await removeProducts(id);
message.success('success');
actionRef.current?.reload();
} catch (error) {
message.error('fail');
}
};
const columns = [
{
title: 'Name',
dataIndex: 'name',
},
{
title: 'Actions',
render: (text, record) => {
return (
< Popconfirm title = "Delete?" onConfirm = {() = > onDelete(record.id)}>
< Button > Delete< / Button >
< / Popconfirm >
);
},
},
];
return (
< ProTable < { name: string } >
headerTitle="查询表格"
actionRef={actionRef}
rowKey="name"
request={(params, sorter, filter) => queryProductList({ ...params, sorter, filter })}
columns={columns}
/>
);
};
```
2022-02-16 19:50:38 +08:00
ProTable 提供了预设逻辑来处理 loading, 分页 和搜索表单,可以大大减少代码量,点击这里[快速开始](https://procomponents.ant.design/components/table)。
2020-07-08 16:15:11 +08:00
2016-09-18 17:39:11 +08:00
## 构建应用
2020-03-04 16:12:06 +08:00
完成开发并且在开发环境验证之后,就需要部署给我们的用户了,执行以下命令:
```bash
$ yarn build
```
2016-09-18 17:39:11 +08:00
2020-07-08 16:15:11 +08:00
![build ](https://gw.alipayobjects.com/zos/antfincdn/Zd3f%242NdOK/b911d244-f1a5-4d61-adc5-3710cd86cd1b.png )
2016-09-18 17:39:11 +08:00
2019-10-23 18:00:15 +08:00
构建会打包所有的资源,包含 JavaScript, CSS, web fonts, images, html 等。你可以在 `dist/` 目录下找到这些文件。
2016-09-18 17:39:11 +08:00
## 下一步
我们已经完成了一个简单应用,你可能还有很多疑问,比如:
2019-05-07 14:57:32 +08:00
- 如何统一处理出错?
- 如何处理更多路由,比如动态路由,嵌套路由,权限路由等?
- 如何 mock 数据?
- 如何部署?
- 等等
2016-09-18 17:39:11 +08:00
你可以:
2019-05-07 14:57:32 +08:00
- 访问 [umi 官网 ](https://umijs.org/ )和 [dva 官网 ](https://dvajs.com/ )
- 理解 [umi 的路由 ](https://umijs.org/zh/guide/router.html )
- 理解 [如何部署 umi 应用 ](https://umijs.org/zh/guide/deploy.html )
2020-07-08 16:15:11 +08:00
- 开箱即用的脚手架 [Ant Design Pro ](https://pro.ant.design )
2022-02-16 19:50:38 +08:00
- 高级布局 [ProLayout ](https://procomponents.ant.design/components/layout )
- 高级表格 [ProTable ](https://procomponents.ant.design/components/table )