# ReactPro
**Repository Path**: playerljc/ReactPro
## Basic Information
- **Project Name**: ReactPro
- **Description**: 自己的一个React的工程
- **Primary Language**: JavaScript
- **License**: Not specified
- **Default Branch**: master
- **Homepage**: None
- **GVP Project**: No
## Statistics
- **Stars**: 8
- **Forks**: 1
- **Created**: 2020-04-30
- **Last Updated**: 2024-09-12
## Categories & Tags
**Categories**: Uncategorized
**Tags**: None
## README
# ReactPro
一个基于 React、ant-design 和 less 开箱即用的模板工程
## 兼容性
- Chrom(支持)
- Firefox(支持)
- Edge(支持)
- Safari(没测试)
## 核心库
- [@ctsj/build](https://github.com/playerljc/CTSJ-BUILD)
- 对项目的打包进行管理
- [@ctsj/router](https://github.com/playerljc/CTSJ-ROUTER)
- 对项目中的路由进行管理
- [@ctsj/state](https://github.com/playerljc/CTSJ-STATE)
- 对项目中数据流进行管理
- [@baifendian/adhere-ui](https://github.com/playerljc/adhere)
- 基于 React 和 ant-design 的高可用组件
## 支持(特性)
- 支持 typescript
- 支持 vite 和 webpack 两种构建方式
- less 主题变量
- 动态换肤
- mock 数据
- jenkins 智能部署
- 动态修改 webpack 打包配置
- 缺省的 start 和 build 命令
- 内置@别名
- 缺省的 splitChunks
- 支持 cssModule
- assets 静态目录
- analysis 代码打包分析
- env 变量混入
- git 提交 hooks
- 一些列 lint 命令(基于@umijs/fabric)
- 支持 PingFang-SC-\*字体
- dev、stg、pe 和 te 环境的 development 和 production 动态配置
- 内置典型 CRUD 代码模板,使用脚本自动生成路由和页面代码
- 根据项目实际情况扩展内置模板代码
## 一些内置 lib 库
- 404 页面
- BasicLayout(左侧导航右侧路由切换的布局)
- NoAuthority(无权限组件)
- Router(路由组件)
## 一些常用的 package
- antd(5.x)
- axios
- classnames
- echarts
- echarts-for-react
- font-awesome
- js-md5
- lodash
- mockjs
- moment
- dayjs
- ol
- qs
- rc-queue-anim
- rc-tween-one
- swiper
- uuid
- @antv/g6
- faker
## 目录结构及重要文件说明
- assets - 静态目录
- EmitterExternal - 全局的发布订阅
- fonts - 字体
- pdf - pdfjs
- constent.js - 环境变量
- logo.ico - 浏览器的 ico
- theme-util.js 主题文件
- script - 实用脚本
- generatorResource - 自动生成典型代码
- deploy.js - 自动部署脚本
- generateSMM.js - 根据 swagger 自动生成 Service
- theme.js - 生成主题脚本
- publish.js - 发布(生成 tag 和 changelog)
- src - 源码目录
- themes 主题目录
- default - 缺省的主题目录
- var.js - 里面主要是全局的 less 变量
- (主题名)目录
- install.js - jenkins 上进行部署的脚本文件
- ctbuild.config.js - webpack 打包配置文件
- .eslint\* - eslint 配置文件
- .prettier\* - prettier 配置文件
- .stylelint\* - stylelint 配置文件
- .jsconfig
- .browserslistrc - 浏览器版本配置
- .postcss.config.js - PostCSS 配置
## src 源码目录
- components - 系统组件
- formitem - antd 表单的 FormItem
- AnnexFormItem - 附件的 FormItem
- ImageAnnexFormItem - 图片上传的 FormItem
- SearchTreeSelectFormItem - TreeSelect 的 FormItem
- TMapAddressSelectFormItem - 腾讯地图选择位置的 FormItem
- system - 系统级组件
- SystemAnnexList - 附件查看
- SystemBaseSearchTable - SearchTable 的基类
- SystemCard - 系统的 Card
- SystemDictDataWrap - 字典包装组件
- SystemFooter - 系统的 footer
- SystemHeader - 系统的 header
- SystemIframe - iframe 嵌入组件
- SystemFormTableGridLayout - 表单的 TableGridLayout
- SystemViewTableGridLayout - 查看的 TableGridLayout
- SystemImageAnnexList - 图片的附件查看列表
- SystemMediaListItem - 列表的 Item 项
- SystemPage - 页面的包装组件
- SystemSearchTree - 对 Tree 的查询封装组件
- SystemStatistic - 统计组件
- SystemTMapAddressDisplay - 基于腾讯地图选择地址的组件
- config - 路由和字典的配置
- dict - 字典项
- Dict
- dict.system.config.js - 系统的字典
- dict.config.js - 字典的注册
- router - 路由
- router.config.js - 路由注册的处理
- router.jsx - 路由的注册
- images - 图片
- lib - 库
- 404 - 404 组件
- BasicLayout - 导航组件
- NoAuthority - 无权限组件
- Mobile - 移动端支持
- PermissionConditionalrender - 权限的条件渲染组件
- ProgressBarPanel - 页面进度条
- Router - 路由的封装
- RouterConfigComponent.jsx - 获取路由信息的包装组件
- listen.js - 路由的监听
- path.js - 路由地址相关
- util.js - 相关工具方法
- locales - 国际化文件
- Model - 全局模型
- pages - 页面
- Service - 全局接口
- style - 样式
- normalize.less - 全局默认重写
- normalize-antd.less - 全局的 antd 样式重写
- normalize-self.less - 全局的 adhere 样式重写
- util - 工具
- index.js - 工具类
- request.js - ajax 请求
- saga.js - 数据流
- ServiceAdapter.js - 接口适配器
- app.jsx - 代表应用的文件
- index.esj - 主 html 模板文件
- index.js - 第一个加载的文件
- index.less - 系统的第一个 less 文件
## [ctbuild.config.js](https://www.npmjs.com/package/@ctsj/build)
webpack 的配置文件具体请参考上方链接,使用的是@ctsj/build 这个库。
## 安装
pmpm `pnpm install or pnpm i` (首选 pnpm 的方案)
npm `npm install or npm i`
yarn `yarn install or yarn`
## 本地运行
webpack `npm run startapp:dev`
vite `npm run startapp:vite:dev`
## 构建
webpack `npm run buildapp:dev`
vite `npm run buildapp:vite:dev`
startapp 和 buildapp 命令都有两组参数,ENV 变量和 Webpack 变量
ENV 变量
- environment - 部署环境(dev - 开发 | stg - 测试 | pe - 生产)
- mode - 模式(development - 开发 | production - 生产)
- mobile - 是否支持响应式(true - 支持移动端 | false - 不支持响应式)
- pathgen - 地址生成策略(static - 静态 | dynamic - 动态)
- publicPath - 项目名称
- router - 路由类型 browser | hash 默认是 browser
- antdCssPriority - antd的cssinjs的优先级 low | high 默认是low
使用 vite 的时候在 `.env` 和 `.evn.${environment}.${mode}` 中设置即可
Webpack 变量
- alias - 别名
- evnVars - 是否注入 ENV 变量到 process 中
- cssModules - 是否支持 CSSModules
- static - 静态资源目录名称
- analysis - 是否打开构建分析
## 预置命令
- alias - 启动和构建命令
一共有三组`startapp`和`buildapp`的命令,一个 startapp 和 buildapp 是一组,`:`后面的是环境名称,一共有三组环境 `dev`、`stg`和`pe`,分别代表开发、测试和生产环境,这三组环境所代表的配置在`assets/constent.js`文件中进行配置,可以随时对这个文件中的内容进行修改。
```js
let ins = null;
const { protocol, hostname, port } = window.location;
const mapping = {
// dev环境
'dev-development-static': {
apiPrefix: 'http://172.18.9.113:82/',
localPrefix: 'http://localhost:8000/',
systemWebPrefix: 'http://172.18.9.113:83/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'dev-production-static': {
apiPrefix: 'http://172.18.9.113:82/',
localPrefix: 'http://172.18.9.113:82/',
systemWebPrefix: 'http://172.18.9.113:83/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'dev-production-dynamic': {
apiPrefix: `${protocol}//${hostname}:${port || 80}/`,
localPrefix: `${protocol}//${hostname}:${port || 80}/`,
systemWebPrefix: `${protocol}//${hostname}:${port || 80}/`,
portalPath: `${protocol}//${hostname}:${port || 80}/`,
},
// stg环境
'stg-development-static': {
apiPrefix: 'http://172.18.9.113:84/',
localPrefix: 'http://localhost:8000/',
systemWebPrefix: 'http://172.18.9.113:85/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'stg-production-static': {
apiPrefix: 'http://172.18.9.113:84/',
localPrefix: 'http://172.18.9.113:84/',
systemWebPrefix: 'http://172.18.9.113:85/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'stg-production-dynamic': {
apiPrefix: `${protocol}//${hostname}:${port || 80}/`,
localPrefix: `${protocol}//${hostname}:${port || 80}/`,
systemWebPrefix: `${protocol}//${hostname}:${port || 80}/`,
portalPath: `${protocol}//${hostname}:${port || 80}/`,
},
// pe环境
'pe-development-static': {
apiPrefix: 'http://172.26.11.91/',
localPrefix: 'http://localhost:8000/',
systemWebPrefix: 'http://172.26.11.91:81/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'pe-production-static': {
apiPrefix: 'http://172.26.11.91/',
localPrefix: 'http://172.26.11.91/',
systemWebPrefix: 'http://172.26.11.91:81/',
portalPath: 'http://fj-portal-dev.percent.cn:8699/portal',
},
'pe-production-dynamic': {
apiPrefix: `${protocol}//${hostname}:${port || 80}/`,
localPrefix: `${protocol}//${hostname}:${port || 80}/`,
systemWebPrefix: `${protocol}//${hostname}:${port || 80}/`,
portalPath: `${protocol}//${hostname}:${port || 80}/`,
},
};
function Create({ mode, environment, pathgen }) {
const { apiPrefix, localPrefix, systemWebPrefix, portalPath } =
mapping[`${environment}-${mode}-${pathgen}`];
return {
apiPath: `${apiPrefix}api/env/`,
systemApiPath: `${systemWebPrefix}api/SystemManager/`,
systemWebPath: `${systemWebPrefix}system`,
localPath: localPrefix,
portalPath,
};
}
window.Constent = function (CustomEvnVars) {
if (!ins) {
ins = Create(CustomEvnVars);
}
return ins;
};
```
- deploy - 部署命令可以在没有持续集成环境的情况下进行部署
- generate:smm - 部署命令根据 swgger 生成 Service
## 响应式支持
启动的时候 ENV 变量加入 mobile=true 开启支持响应式,此开关会开启 px 自动转 REM,使用 amfe-flexible 自动计算 html 的 font-size,加入了 fast-click
使用[LessMediaQueryLoader][1]支持组件的响应式,组件一般都有一个 index.less 文件,如果想实现在 pad,phone 和 PC 上的不同控制可以那么文件会是这样
```css
// 在此文件中实现和响应式无关的样式
index.less
// 在此文件实现pc端样式
index.mq.pc.less
// 在此文件实现pad端样式
index.mq.pad.less
// 在此文件实现phone样式
index.mq.phone.less
```
## 自动生成代码
根据工程中已有的代码块(chunk)来自动生成业务代码,包括`pages`下的代码和`路由`,工程会根据`pages`下的目录结构来自动生成代码和路由,自动生成代码的过程如下:
1. 第一部 在`${root}/script/generatorResource/structure`下面编写结构文件,当前目录下已经有了一个预置好的结构文件,可以把这个目录转移到其他地方,或者基于这个目录进行修改。
2. 执行`$ npm run rp -- gen`命令
执行完上述三个步骤后`pages`下会生成相应代码,路由文件也会自动生成。
### rp 命令详解
运行`$ npm run rp -- gen --help`可以查看所有参数
- -a 路由文件中的上下文别名,如@ 默认值是@
- -b 存放路由视图文件夹所在的目录, + -p 参数值是结构文件实际拷贝的位置 默认值是`${root}/src`
- -p 存放路由视图文件夹的名称,默认是 pages
- -s 结构文件所在的位置 如果没设置则是`${root}/script/generatorResource/structure`
- -t 结构文件拷贝的位置 如果没设置则是`${root}/src/pages`
- -r 路由文件所在的位置 如果没设置则是`${root}/src/config/router/router.jsx`
### 结构文件详解
下面介绍一下组成结构文件的元素都有哪些:
- 结构目录
- 结构目录就是一个普通的目录(在路由中代表着一个父路由,通常这种路由对应的组件是一个返回子路由组件的包装组件,模版工程默认提供了一个包装组件 ``,你也可以自定义这个组件,但需要注意一定要在返回的 JSX 中包含接受到的子路由组件 `children` ),结构目录下面不能有功能文件(功能文件即路由树中,叶子路由节点对应的视图组件的文件),可以有其他结构目录、功能目录和配置文件,有一个特殊名字的结构目录,如果目录名称以`BasicLayout`结尾那么这个目录就是一个左侧是二级菜单右侧是切换路由的一个组件,刚才说到结构目录其实是路由中的一个包装节点,如果想实现自己的包装节点,则目录中可以有一个`index.[js | jsx | ts | tsx | vue]`文件,用来自定义包装节点的组件。
- 功能目录
- 含有功能文件的目录是功能目录,功能目录下只能有一个功能文件 Main.jsx 和配置文件,如果功能目录名称满足`!/\s*Or\s*/gm`正则表达式,那么这个功能目录实现了 2 个功能,如 SaveOrUpdate,它的路由配置和其他的有所不同,下面会介绍。
- 配置文件(config.json) 配置文件可以在结构目录和功能目录中,配置文件是 config.json,下面给出配置文件的定义 config.json
```json
{
"dir": {
"modelIgnore": false,
"serviceIgnore": false,
"mockIgnore": false,
"dictIgnore": false,
"componentIgnore": false,
"functionFileIgnore": false,
"routerIgnore": false
},
"file": {
"templateType": "View",
"functionFileIgnore": false,
"routerIgnore": false,
"contextRouteId": "Table"
},
"route": {
"name": "Menu",
"hide": false,
"subAuthority": [],
"icon": "images/a.png",
"sort": 0,
"ignore": false,
"skip": false
}
}
```
如果功能目录名称满足正则表达式 `!/\s*Or\s*/gm`(如:`SaveOrUpdate`),那么 router 配置如下
```json
{
"route": {
"Save": {
"name": "新增",
"hide": true
},
"Update": {
"name": "修改",
"hide": true
}
}
}
```
下面给出 ts 的定义
```typescript
interface DirConfig {
// 是否创建Model目录及其文件
modelIgnore?: boolean;
// 是否创建Service目录及其文件
serviceIgnore?: boolean;
// 是否创建Mock目录及其文件
mockIgnore?: boolean;
// 是否创建Dict目录及其文件
dictIgnore?: boolean;
// 是否创建Components目录及其文件
componentIgnore?: boolean;
// 如果有功能文件是否使用模板填充功能文件(目录是功能目录)
functionFileIgnore?: boolean;
// 是否是路由中的节点
routerIgnore?: boolean;
}
```
```typescript
interface FileConfig {
// 功能文件选用模板的名称,如果没有进行设置,则使用的目录名称
templateType?: string;
// 如果有功能文件是否使用模板填充功能文件,如果没设置则父目录的配置文件中的此属性生效
functionFileIgnore?: boolean;
// 是否是路由中的节点
routerIgnore?: boolean;
// 上下文模板类型
contextRouteId?: string;
}
```
```typescript
interface RouteItemConfig {
// 菜单的名称
name?: string;
// 是否在菜单中显示
hide?: string;
// 按钮权限
subAuthority?: string[];
// icon
icon?: string;
// 菜单排序属性
sort?: number;
// 是否忽略此节点
ignore?: boolean;
// 是否略过此节点
skip?: boolean;
}
```
```typescript
export type RouteConfig =
| RouteItemConfig
| {
Save?: RouteItemConfig;
Update?: RouteItemConfig;
};
```
### 代码块(chunk)介绍
当前的 template 中的模板类型有如下几种
- `AsyncSearchTreeList` - 左侧 Tree(异步)右侧列表的模板
- `AsyncSearchTreeTable` - 左侧 Tree(异步加载)右侧表格的模板
- `AsyncSearchTreeTreeDataTable` - 左侧 Tree(异步加载)右侧TreeData表格的模板
- `CardList` - 卡片列表的模板
- `CardSelectionList` - 可以选择的卡片列表模板
- `CBLayout` - CBL布局
- `CBRLayout` - CBR布局
- `CollapseList` - 可以折叠的卡片列表模板
- `CollapseSelectionList` - 可以选择和折叠的卡片列表模板
- `CRBLayout` - CRB布局
- `Back` - 可以返回的模板
- `ClassTable` - 使用类的方式实现一个纯净 sagger 的 Table
- `EditorCellRowDragSortTable` - 可以编辑Cell和拖拽行排序的表格模板
- `EditorCellRowDragSortTreeTreeDataTable` - 可以编辑Cell和拖拽行排序的TreeData表格模板
- `EditorCellTable` - 可以编辑表格Cell数据的Table模板
- `EditorCellTreeDataTable` - 可以编辑Cell的TreeData表格模板
- `EditorRowDragSortTable` - 可以拖拽行排序的表格模板
- `EditorRowDragSortTreeDataTable` - 可以拖拽排序的TreeData表格模板
- `EditorRowTable` - 可以编辑表格行的Table模板
- `EditorRowTreeDataTable` - 可以编辑表格行的TreeData模板
- `EditorTable` - 可以编辑整体表格的模板
- `EditorTableRowDragSortTable` - 可以编辑整体表格并且拖拽排序的Table模板
- `EditorTableRowDragSortTreeDataTable` - 可以编辑整体表格并且拖拽排序的TreeTable模板
- `EditorTreeDataTable` - 可以编辑整体表格的TreeData模板
- `IndicatorAsyncEditorCellRowDragSortTable` - 带有统计功能的AsyncEditorCellRowDragSortTable的模板
- `IndicatorAsyncEditorCellRowDragSortTreeDataTable` - 带有统计功能的AsyncEditorCellRowDragSortTreeDataTable模板
- `IndicatorAsyncEditorRowDragSortTable` - 带有统计功能的AsyncEditorRowDragSortTable模板
- `IndicatorAsyncEditorRowDragSortTreeDataTable` - 带有统计功能的AsyncEditorRowDragSortTreeDataTable
- `IndicatorAsyncEditorTableRowDragSortTable` - 带有统计功能的AsyncEditorTableRowDragSortTable
- `IndicatorAsyncEditorTableRowDragSortTreeDataTable` - 带有统计功能的IndicatorAsyncEditorTableRowDragSortTreeDataTable
- `IndicatorAsyncSearchTreeList` - 带有统计功能的AsyncSearchTreeList
- `IndicatorAsyncSearchTreeTable` - 带有统计功能的AsyncSearchTreeTable
- `IndicatorAsyncSearchTreeTreeDataTable` - 带有统计功能的AsyncSearchTreeTreeDataTable
- `IndicatorEditorCellRowDragSortTable` - 带有统计功能的EditorCellRowDragSortTable
- `IndicatorEditorCellRowDragSortTreeDataTable` - 带有统计功能的EditorCellRowDragSortTreeDataTable
- `IndicatorEditorRowDragSortTable` - 带有统计功能的EditorRowDragSortTable
- `IndicatorEditorRowDragSortTreeDataTable` - 带有统计功能的EditorRowDragSortTreeDataTable
- `IndicatorEditorTableRowDragSortTable` - 带有统计功能的EditorTableRowDragSortTable
- `IndicatorEditorTableRowDragSortTreeDataTable` - 带有统计功能的EditorTableRowDragSortTreeDataTable
- `IndicatorList` - 带有统计功能的List
- `IndicatorSearchTreeList` - 带有统计功能的SearchTreeList
- `IndicatorSearchTreeTable` - 带有统计功能的SearchTreeTable
- `IndicatorSearchTreeTreeDataTable` - 带有统计功能的SearchTreeTreeDataTable
- `IndicatorTable` - 带有统计功能的Table
- `IndicatorTreeDataTable` - 带有统计功能的TreeDataTable
- `LBCLayout` - LBC布局
- `LCBLayout` - LCB布局
- `LCLayout` - LC布局
- `LCRBLayout` - LCRB布局
- `List` - List列表模板
- `LRTCBLayout` - LRTCB布局
- `LTCBLayout` - LTCB布局
- `LTCLayout` - LTC布局
- `TBLCRLayout` - TBLCR布局
- `TCBRLayout` - TCBR布局
- `TCLayout` - TC布局
- `TCRLayout` - TCR布局
- `TLCLayout` - TLC布局
- `TLRCLayout` - TLRC布局
- `TRCLayout` - TRC布局
- `EditorCellTable` - 可编辑单元格的列表模板
- `EditorTable` - 可编辑表格的列表模板
- `EditorRowTable` - 可编辑行的列表模板
- `HookTable` - 使用 Hooks 的方式实现一个纯净 sagger 的 Table
- `IndicatorAsyncSearchTreeList` - 带有统计信息的AsyncSearchTreeList模板
- `IndicatorAsyncSearchTreeTable` - 带有统计信息的AsyncSearchTreeTable模板
- `IndicatorList` - 带有统计信息的列表
- `IndicatorSearchTreeList` - 带有统计信息的SearchTreeList模板
- `IndicatorSearchTreeTable` - 带有统计信息的SearchTreeTable模板
- `IndicatorTable` - 带有统计信息的Table模板
- `List` - 列表的模板
- `Table` - 表格的模板
- `RowDragSortTable` - Radius 卡片的查看模板
- `RowDragSortTable` - 可拖拽的列表模板
- `SaveOrUpdate` - 添加和编辑的模板
- `SearchTreeTable` - 左侧 Tree 右侧表格的模板
- `SearchTreeList` - 左侧 Tree 右侧列表的模板
- `SearchTreeTreeDataTable` - 左侧 Tree 右侧TreeData表格的模板
- `TreeDataTable` - TreeData的表格模板
- `SelecitonList` - 可以选择的列表
- `SmallCardsView` - Small 卡片的查看模板
- `StateClassTable` - 使用类的方式实现一个纯净 sagger 的 Table(数据源使用 state)
- `StateHooksTable` - 使用 Hooks 的方式实现一个纯净 sagger 的 Table(数据源使用 state)
- `TabsView` - 页卡的查看模板
- `View` - 查看的模板
- `ViewAdvanced` - 高级查看的模板
以上是工程内自带的代码块,如果以上的代码块不能满足需求,可以根据实际业务来增加代码块
1. 第一步 在`${root}/script/generatorResource/templates/function/`下建立一个目录,目录下面需要有如下几个子目录
- Build - 功能的实现
- render.js
- Components - 组件
- index.jsx
- Less - 样式
- index.less
- Page - 页面
- inndex.js
- structure - 在 e2e 中的结构文件
- config.json
- Main.jsx
2. 编写响应代码,可以参照之前已有模块的代码,这里就不在熬述了
### 代码块的测试
1. 运行 `npm run e2e:codeblock` 来启动一个服务从而查看工程中的代码块用例。
2. 生成代码块运行后的截图运行`npm run codeblock:genTemplateImage`生成代码块运行结果的截图。
## 典型代码和组件讲解
- **页面的代码结构**
一个典型页面(一般指 CRUD 功能)通常代表路由中的一个节点(组件),如列表、新增、修改和查看等,一个典型的页面一般应该有如下结构
```js
// 组件(一般只有一个组件)
Components;
// 弹窗
Modal;
// 组件入口
index.jsx;
// 组件样式入口
index.less;
// Mock数据
Mock;
// 模型(一般是定义状态管理的数据)
Model;
// 接口
Service;
// 测试
Test;
// 功能(页面)的入口文件
Main.jsx;
// 功能(页面)样式的入口文件
Main.less;
```
- **CRUD**
我们平时编写代码,编写频率最高的就是 CRUD 代码,列表的形式一般都不变,基本都是查询、列表体,分页,变化最多的就是 SaveOrUpdate 和 View,下面列出一下最简单的 SaveOrUpdate 和 View 的代码结构 1.View Main.jsx 入口文件
```jsx
// Main.jsx 入口文件
import qs from 'qs';
import React, { useContext } from 'react';
import { FlexLayout } from '@baifendian/adhere';
import SystemPage from '@/components/system/SystemPage';
import { getSearch } from '@/lib/Router/path';
import { findRouteConfig } from '@/lib/Router/util';
import { RouterConfigContext } from '@/lib/Router/RouterConfigComponent';
import ViewDataWrap from './Components/dataWrap';
import styles from './Main.less';
const { BackLayout, ScrollLayout } = FlexLayout;
function getQuery() {
return qs.parse(getSearch(), { ignoreQueryPrefix: true }) || {};
}
/**
* View - 查看页面入口
*/
function View(props) {
const { routerConfig } = useContext(RouterConfigContext);
return (
);
}
export default View;
```
Main.less 样式入口文件
```css
// Main.less 样式入口文件
.Wrap {
width: 100%;
height: 100%;
}
```
Components/dataWrap.jsx 获取 View 数据的包装组件
```jsx
// Components/dataWrap.jsx 获取View数据的包装组件
import PropTypes from 'prop-types';
import React, { useEffect, useState } from 'react';
import { fetchInfo } from '../Service';
import View from './index';
function ViewDataWrap(props) {
const { id } = props;
const [data, setData] = useState({});
useEffect(() => {
if (id) {
fetchInfo.call(id, true).then((res) => {
if (!res) return;
if (res.data.resCode === 0) {
setData(res.data.data);
}
res.hideIndicator();
});
}
}, [id]);
return ;
}
ViewDataWrap.defaultProps = {};
ViewDataWrap.propTypes = {
id: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
};
export default ViewDataWrap;
```
Components/index.jsx View 的展示组件
```jsx
import PropTypes from 'prop-types';
import React from 'react';
import { DateDisplay, Intl, Resource, TableGridLayout } from '@baifendian/adhere';
import SystemAnnexList from '@/components/system/SystemAnnexList';
import SystemCard from '@/components/system/SystemCard/SystemCard';
import SystemViewTableGridLayout from '@/components/system/SystemViewTableGridLayout';
const { Label, Value } = TableGridLayout;
/**
* View
* @param props
* @return {JSX.Element}
* @constructor
*/
function View(props) {
const {
data: { name, sex, birthDay, height, width, hometown, address, annex },
} = props;
return (
{Intl.v('姓名')}:,
value: {name},
},
{
key: 'sex',
label: ,
value: (
{Resource.Dict.value.ResourceNormalSexMap.value.get(sex)?.label}
),
},
{
key: 'birthDay',
label: ,
value: (
),
},
{
key: 'height',
label: ,
value: {height},
},
{
key: 'width',
label: ,
value: {width},
},
{
key: 'hometown',
label: ,
value: {hometown},
},
],
},
{
name: 'g2',
width: '100%',
columnCount: 1,
colgroup: [120, 'auto'],
data: [
{
key: 'address',
label: ,
value: {address},
},
{
key: 'annex',
label: ,
value: (
),
},
],
},
]}
/>
);
}
View.defaultProps = {
data: {
id: '',
hobby: null,
annex: [],
},
};
View.propTypes = {
data: PropTypes.shape({
id: PropTypes.oneOfType([PropTypes.string, PropTypes.number]),
hobby: PropTypes.arrayOf(PropTypes.string),
annex: PropTypes.arrayOf(
PropTypes.shape({
id: PropTypes.string,
name: PropTypes.string,
path: PropTypes.string,
}),
),
}),
};
export default View;
```
Service/index.js 接口文件
```js
// import ServiceAdapter from '@/util/ServiceAdapter';
import Mock from '@/pages/BasicLayout/Task/ImplementSubmit/View/Mock';
import request from '@/util/request';
export const fetchList = (() => {
return {
call: ({ page, limit }) => {
const path = JSON.parse(JSON.stringify(Mock.fetchList));
path.data.data.records = path.data.data.records.slice((page - 1) * limit, page * limit);
return request.get({
path,
mock: true,
loading: {
show: false,
},
});
},
defaultResult: () => ({
total: 0,
records: [],
}),
};
})();
export const fetchDel = (() => {
return {
call: (id, loading) => {
return request.get({
path: Mock.fetchDel,
mock: true,
loading: {
show: loading,
},
});
},
defaultResult: () => true,
};
})();
export const fetchInfo = (() => {
return {
call: (id, loading) => {
return request.get({
path: Mock.fetchInfo,
mock: true,
loading: {
show: loading,
},
});
},
defaultResult: () => ({
id: '',
name: '',
sex: '',
birthDay: '',
height: '',
width: '',
hometown: '',
address: '',
hobby: null,
annex: [],
}),
};
})();
export const fetchSave = (() => {
return {
call: (params, loading) => {
return request.post({
path: Mock.fetchSave,
mock: true,
loading: {
show: loading,
},
});
},
defaultResult: () => true,
};
})();
export const fetchUpdate = (() => {
return {
call: (params, loading) => {
return request.post({
path: Mock.fetchUpdate,
mock: true,
loading: {
show: loading,
},
});
},
defaultResult: () => true,
};
})();
export default {
codeKey: 'resCode',
codeSuccessKey: 0,
dataKey: 'data',
messageKey: 'resMsg',
name: 'AppBasicLayoutTaskImplementSubmitView',
};
```
Mock/index.js Mock 数据
```js
import faker from 'faker';
const data = [];
data.length = 300;
data.fill(0);
export default {
fetchList: {
data: {
resCode: 0,
data: {
total: data.length,
records: data.map((t, index) => ({
id: faker.random.uuid(),
name: faker.internet.userName(),
sex: `${(index + 1) % 2}`,
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
})),
},
resMsg: '',
},
},
fetchDel: {
resCode: 0,
data: true,
resMsg: '',
},
fetchSave: {
resCode: 0,
data: true,
resMsg: '',
},
fetchUpdate: {
resCode: 0,
data: true,
resMsg: '',
},
fetchInfo: {
resCode: 0,
data: {
id: faker.random.uuid(),
name: faker.internet.userName(),
sex: '0',
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
hobby: [1, 2, 3],
department: '0-0-0',
annex: [
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
{
path: faker.random.uuid(),
id: faker.random.uuid(),
name: faker.company.companyName(),
},
],
},
resMsg: '',
},
};
```
- **[Dict](http://49.232.163.126:8083/adhere/util/dict)**
程序中往往都会存储大量的常量和数据源信息,常量分为各种类型,如地址的信息、状态等信息等,数据源数据如用户信息、用户权限、Select 的下拉数据等用接口获取的数据,这种数据一般都是在整个系统中只需要调用一次即可,那么可以使用 Dict 来进行存储,Dict 的特点是统一了获取数据的编程接口,只有第一次使用才加载,其余都是读取缓存数据。Dict 也支持刷新缓存。
**定义 Dict**
Dict 定义在任意目录下,一般按层级可以进行分层定义,整个系统都需要的数据定义在项目根目录下的 Dict 里面
![][2]
和业务相关的可以定义在业务相关的目录里
![][3]
**项目中只要在 Dict 目录下创建的 js 文件都会认为是 Dict 的声明文件**
Dict 文件的写法
```js
import { Dict } from '@baifendian/adhere';
export default {
// 定义静态的Dict
initStatic() {
// 定义系统中用到的状态
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestRadio = () => [
{
value: 1,
label: '通过',
},
{
value: 2,
label: '不通过',
},
{
value: 3,
label: '退回',
},
];
// 也可以定义方法,通过kw进行筛选,如果在Dict中定义的是函数,则会有Memo的效果,类似于React的useMemo或Vue的计算效果
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestAutoCompleteSelect =
() => (kw) => {
const data = [
{
label: 'java',
value: 1,
},
{
label: 'javaScript',
value: 2,
},
{
label: 'html',
value: 3,
},
{
label: 'css',
value: 4,
},
{
label: 'spring',
value: 5,
},
{
label: 'react',
value: 6,
},
];
return Promise.resolve(data.filter((t) => t.label.includes(kw)));
};
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestTree = () => [
{
title: 'Node1',
value: '0-0',
leaf: false,
children: [
{
title: 'Child Node1',
value: '0-0-1',
leaf: true,
},
{
title: 'Child Node2',
value: '0-0-2',
leaf: true,
},
],
},
{
title: 'Node2',
value: '0-1',
leaf: true,
},
];
// Table
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestTable = () =>
Array.from({ length: 10 }).map((t, index) => ({
id: faker.random.uuid(),
isMore: !!Math.floor((Math.random() * 10) % 2),
name: faker.internet.userName(),
sex: `${(index + 1) % 2}`,
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
}));
// Cascader
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestCascader = () => [
{
value: 'zhejiang',
label: 'Zhejiang',
children: [
{
value: 'hangzhou',
label: 'Hangzhou',
children: [
{
value: 'xihu',
label: 'West Lake',
},
],
},
],
},
{
value: 'jiangsu',
label: 'Jiangsu',
children: [
{
value: 'nanjing',
label: 'Nanjing',
children: [
{
value: 'zhonghuamen',
label: 'Zhong Hua Men',
},
],
},
],
},
];
// List
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestList = () =>
Array.from({ length: 5 }).map((t, index) => ({
id: faker.random.uuid(),
isMore: !!Math.floor((Math.random() * 10) % 2),
name: faker.internet.userName(),
sex: `${(index + 1) % 2} `,
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
}));
},
// 动态的Dict,一般都是调用接口,如获取用户信息,获取用户权限,获取机构数据,获取Select等数据源数据,省市区级联等都可以在此定义,只要是调用接口的公共数据
initRemote() {
// 权限配置
Dict.handlers.SystemAuthorized = () => {
// 用户权限
return systemManagerRequest
.get(
injectionToken({
path: require('@/mock/system').default.SystemAuthorized,
mock: true,
loading: {
show: false,
},
showWarn: false,
}),
)
.then((data) => {
return data.data;
});
};
// 用户信息
Dict.handlers.SystemUserInfo = () => {
return systemManagerRequest
.get(
injectionToken({
path: require('@/mock/system').default.SystemUserInfo,
mock: true,
loading: {
show: false,
},
}),
)
.then((data) => {
return data.data;
});
};
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestTablePagination =
() => (paging) => {
const { current, pageSize } = paging;
const data = [];
data.length = 300;
data.fill(0);
const res = {
resCode: 0,
data: {
total: data.length,
pages: 30,
current: 1,
records: data
.slice((current - 1) * pageSize, (current - 1) * pageSize + pageSize)
.map((t, index) => ({
id: (current - 1) * pageSize + (index + 1),
isMore: !!Math.floor((Math.random() * 10) % 2),
name: faker.internet.userName(),
sex: `${(index + 1) % 2} `,
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
})),
},
resMsg: '',
};
return Promise.resolve(res.data);
};
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestDynamicCascader = () =>
Promise.resolve([
{
value: 'zhejiang',
label: 'Zhejiang',
children: [
{
value: 'hangzhou',
label: 'Hangzhou',
children: [
{
value: 'xihu',
label: 'West Lake',
},
],
},
],
},
{
value: 'jiangsu',
label: 'Jiangsu',
children: [
{
value: 'nanjing',
label: 'Nanjing',
children: [
{
value: 'zhonghuamen',
label: 'Zhong Hua Men',
},
],
},
],
},
]);
Dict.handlers.SystemAppBasicLayoutTaskImplementSubmitListTestListPagination =
() => (paging) => {
const { current, pageSize } = paging;
const data = [];
data.length = 300;
data.fill(0);
const res = {
resCode: 0,
data: {
total: data.length,
pages: 30,
current: 1,
records: data
.slice((current - 1) * pageSize, (current - 1) * pageSize + pageSize)
.map((t, index) => ({
id: (current - 1) * pageSize + (index + 1),
isMore: !!Math.floor((Math.random() * 10) % 2),
name: faker.internet.userName(),
sex: `${(index + 1) % 2}`,
birthDay: faker.time.recent(),
deptName: faker.company.companyName(),
height: faker.random.number(),
width: faker.random.number(),
hometown: faker.address.city(),
address: faker.address.streetAddress(),
})),
},
resMsg: '',
};
return Promise.resolve(res.data);
};
},
};
```
Dict 的用法
```js
import { Dict } from '@baifendian/adhere';
// 静态自定使用
Dict.value.SystemAppBasicLayoutTaskImplementSubmitListTestRadio.value;
// 静态方法
Dict.value.SystemAppBasicLayoutTaskImplementSubmitListTestAutoCompleteSelect.value('java');
// 动态字典的使用
// 获取用户信息
Dict.value.SystemUserInfo.value.then((userInfo) => {
console.log(userInfo);
});
// 级联
Dict.value.SystemAppBasicLayoutTaskImplementSubmitListTestListPagination.value(cascadeParams).then(
(res) => {
console.log(res);
},
);
```
Dict 的刷新
```js
Dict.value.SystemAppBasicLayoutTaskImplementSubmitListTestListPagination.refresh(params);
```
- **[FormItemGeneratorToDict](http://49.232.163.126:8083/adhere/ui/antdformitem)**
我们再写一个表单的时候,往往会有数据源表单项,如`Select`、`TreeSelect`、`AutoComplete`等、我们一般都是先调用接口然后用 state 进行存储,在遍历数据渲染 options。这只是简单的情况,如果有多个 Select 进行级联操作,Select 支持多选、搜索和全选,或者 TreeSelect 有限制规定,如只能选 leaf 节点,或者每个层级节点都能选这样的诉求时我们就需要编写大量代码来实现,而且这些代码是不能再其他地方进行复用的,如果想复用就需要把代码组装成组件,如果这样的组件太多又不能为每一个情况进行一一编写,FormItemGeneratorToDict 就是用了来解决上述这些问题的。`FormItemGeneratorToDict`的核心理念是使用 Dict 作为数据,然后使用 Dict 的名字来包装一个组件。`FormItemGeneratorToDict`使用约定的方式来定义 Dict。下面是一个 Select 的例子
定义 Dict
```js
import { Dict } from '@baifendian/adhere';
export default {
initStatic() {
Dict.handlers.SystemTestSelect = () => [
{
value: 1,
label: '通过',
},
{
value: 2,
label: '不通过',
},
{
value: 3,
label: '退回',
},
];
},
initRemote() {},
};
```
编写业务代码
```jsx
import React from 'react';
import { Form } from 'antd';
import { AntdFormItem } from '@baifendian/adhere';
export defult () => (
// 使用字典生成了一个Select的Item
)
```
级联
```js
import { Dict } from '@baifendian/adhere';
export default {
initStatic() {
// 省
Dict.handlers.SystemTestProvinceSelect = () => [
{
value: 1,
label: '辽宁省',
},
{
value: 2,
label: '安徽省',
},
{
value: 3,
label: '浙江省',
},
];
// 市
Dict.handlers.SystemTestCitySelect = () => (provinceValue) => {
if (provinceValue === 1)
return [
{ label: '沈阳市', vallue: 1 },
{ label: '铁岭市', vallue: 1 },
{ label: '沈阳市', vallue: 1 },
];
if (provinceValue === 2)
return [
{ label: '沈阳市', vallue: 1 },
{ label: '铁岭市', vallue: 1 },
{ label: '沈阳市', vallue: 1 },
];
if (provinceValue === 3)
return [
{ label: '沈阳市', vallue: 1 },
{ label: '铁岭市', vallue: 1 },
{ label: '沈阳市', vallue: 1 },
];
};
// 区
Dict.handlers.SystemTestAreaSelect = () => (cityValue) => {
if (provinceValue === 1)
return [
{ label: '铁西区', vallue: 1 },
{ label: '和平区', vallue: 1 },
,
{ label: '皇姑区', vallue: 1 },
];
if (provinceValue === 2)
return [
{ label: '铁西区', vallue: 1 },
{ label: '和平区', vallue: 1 },
{ label: '皇姑区', vallue: 1 },
];
if (provinceValue === 3)
return [
{ label: '沈阳市', vallue: 1 },
{ label: '铁岭市', vallue: 1 },
,
{ label: '沈阳市', vallue: 1 },
];
};
},
initRemote() {},
};
```
```jsx
import React , { useEffect } from 'react';
import { Form } from 'antd';
import { AntdFormItem } from '@baifendian/adhere';
export defult () => {
const [form] = Form.useForm();
const province = Form.useWatch('province', form);
const city = Form.useWatch('city', form);
useEffect(() => {
form.setFieldValue('city', '');
form.setFieldValue('area', '');
},[province]);
useEffect(() => {
form.setFieldValue('area', '');
},[city]);
return (
// 使用字典生成了一个Select的Item
// 使用字典生成了一个Select的Item
// 使用字典生成了一个Select的Item
);
}
```
用来做复杂的权限判断
```js
// 用户组织机构判断
export default {
initStatic() {
Dict.handlers.SystemUserOrganizationMap = () =>
new Map([
// 省
[
[2].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 2) resolve();
});
}),
],
// 市
[
[3].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 3) resolve();
});
}),
],
// 区
[
[4].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 4) resolve();
});
}),
],
// 营商
[
[37].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.stripeCatalog == 37) resolve();
});
}),
],
// 非营商
[
['!37'].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.stripeCatalog != 37) resolve();
});
}),
],
// 省部门
[
[2, '!37'].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 2 && user.stripeCatalog != 37) resolve();
});
}),
],
// 省营商
[
[2, 37].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 2 && user.stripeCatalog == 37) resolve();
});
}),
],
// 市营商
[
[3, 37].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 3 && user.stripeCatalog == 37) resolve();
});
}),
],
// 市部门
[
[3, '!37'].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 3 && user.stripeCatalog != 37) resolve();
});
}),
],
// 区营商
[
[4, 37].toString(),
() =>
new Promise((resolve) => {
Dict.value.SystemUserInfo.value.then((user) => {
if (user.standard == 4 && user.stripeCatalog == 37) resolve();
});
}),
],
]);
// 审核界面反馈来源的Select数据
Dict.handlers.SystemVerifyFeedbackSource = () =>
new Promise((resolve) => {
Promise.race([
// 省部门
Dict.value.SystemUserOrganizationMap.value.get([2, '!37'].toString())(),
// 省营商
Dict.value.SystemUserOrganizationMap.value.get([2, 37].toString())(),
]).then(() => {
Dict.value.SystemOrgGetRegionAreaList.value().then((res) => {
resolve(
res.map((t) => ({
label: t.name,
value: t.id,
})),
);
});
});
// 市营商
Dict.value.SystemUserOrganizationMap.value
.get([3, 37].toString())()
.then(() => {
Dict.value.SystemOrgGetOrgListByRegionCode.value().then((res) => {
resolve(
res.map((t) => ({
label: t.name,
value: t.id,
})),
);
});
});
});
},
initRemote() {},
};
```
当前 FormItemGeneratorToDict 内置了非常多的数据源包装组件,他们都已约定的后缀命名 Dict,比如以 Select 结尾的字典,如`Dict.handhers.SystemTestSelect`,那么引用的时候就是`SystemTestSelectFormItem`,此时 Dict 是静态的 Dict,也就是返回不是 Promise,如果以 DynamicSelect 结尾的 Dict,如`Dict.handlers.SystemTestDynamicSelect`,那么引用时候就是`SystemTestDynamicSelectFormItem`,此时 Dict 返回的是 Promise,多用于调用后端接口返回的数据。下面列出所有后缀。
- AutoComplete
- CascaderLeaf
- CascaderMulit
- CascaderLeafMulit
- RadioHorizontal
- RadioVertical
- RadioButton
- RadioSelect
- RadioCustom
- CheckBoxHorizontal
- CheckBoxVertical
- CheckBoxCheckAllVertical
- CheckBoxCheckAllHorizontal
- CheckBoxSelect
- CheckBoxCheckAllSelect
- CheckBoxCustom
- Select
- SelectMulit
- SelectCheckAllMulit
- AutoCompleteSelect
- AutoCompleteSelectMulit
- AutoCompleteSelectCheckAllMulit
- Tree
- TreeLeaf
- TreeMulit
- TreeLeafMulit
- Transfer
- TransferSelect
- Table
- TableSelect
- TableMulitSelect
- TablePagination
- TablePaginationSelect
- TablePaginationMulitSelect
- List
- ListSelect
- ListMulitSelect
- ListPagination
- ListPaginationSelect
- ListPaginationMulitSelect
- AutoComplete
- **[AntFormItemNormalize](http://49.232.163.126:8083/adhere/ui/antdformitem)**
AntFormItemNormalize 是对 antd 中的所有组件预定义默认值,下面是为 Select 组件设置的默认值
```js
import { Select as AntSelect } from 'antd';
export const Select = createFactory(AntSelect, {
showSearch: true,
allowClear: true,
placement: 'bottomLeft',
filterOption: (input, option) => option.children.toLowerCase().indexOf(input.toLowerCase()) >= 0,
});
```
结合[FlexLayout](http://49.232.163.126:8083/adhere/ui/flexlayout)的`ScrollLayout`来设置`getPopupContainer`属性,`FlexLayout.ScrollLayout`是一个可以滚动的容器,在此容器下使用`AntFormItemNormalize`包装的 antd 组件可以将`getPopupContainer`属性设置为`ScrollLayout`的`el`
可以统一动态设置`AntFormItemNormalize`包装组件属性的默认值
```js
import { AntdFormItem } from '@baifendian/adhere';
AntdFormItem.AntFormItemNormalize.Input.defaultProps.maxLength = 2000;
```
这样引用 AntdFormItem.AntFormItemNormalize.Input 组件的地方都会改变
- **[ConditionalRender](http://49.232.163.126:8083/adhere/ui/conditionalrender)**
我们一般在遇到条件渲染的时候都是使用三目运算符(条件 ? Match 功能 : NoMatch 功能)或者短路表达式(条件 && 功能),这样的弊端在于在 JSX 中使用表达式会让程序的可读性降低,不连贯最后导致不易扩展和维护,我们应该以标签的形式来处理条件渲染,下面是一个例子
Bad
```jsx
import React from 'react';
export default () => {
const [status, setStatus] = useState(0);
return status === 0 ? display
: null;
};
```
Good
```jsx
import React from 'react';
import { ConditionalRender } from '@baifendian/adhere';
export default () => {
const [status, setStatus] = useState(0);
return (
{() => display
}
);
};
```
`ConditionalRender`还支持`Show`和`Visibility`,Show 用于切换 css 的 display,Visibility 用于切换 css 的 visibility
```jsx
import React from 'react';
import { ConditionalRender } from '@baifendian/adhere';
export default () => {
const [status, setStatus] = useState(0);
return (
{() => display
}
);
};
```
```jsx
import React from 'react';
import { ConditionalRender } from '@baifendian/adhere';
export default () => {
const [status, setStatus] = useState(0);
return (
{() => display
}
);
};
```
- **Intl**
系统中的国际化文件在`src/locales`下面,Intl 的一大特点是不用定义`key`,`key`有组件自动生成,下面是`zh_CN.js`和`en_US`的定义
```js
export default ['姓名', '性别', '年龄'];
```
```js
export default ['name', 'sex', 'birthday'];
```
顺序要一一对应。
```js
import { Intl } from '@baifendian/adhere';
Intl.v('姓名');
```
如果一个词条代表多个意思也可以使用`key`和`value`来定义词条
```js
export default [
'姓名',
'性别',
'年龄',
{
key: 'name',
value: '姓名',
},
];
```
```js
import { Intl } from '@baifendian/adhere';
// 姓名词条
Intl.v('姓名');
// name词条
Intl.get('name');
```
支持表达式
```js
export default ['仓库数量合计:{count}个'];
```
```js
import { Intl } from '@baifendian/adhere';
Intl.v('仓库数量合计:{count}个', {
count: 6,
});
```
配合`AdhereConfigProvider`使用
```jsx
import { ConfigProvider as AdhereConfigProvider } from '@baifendian/adhere';
ReactDOM.createRoot(document.getElememtById('app')).render(
{() => App
}
,
);
```
- **[Ajax](http://49.232.163.126:8083/adhere/util/ajax)**
普通的请求需要在 Service 中进行定义,命名的约定`fetch${业务名称}${功能名称}`,如果 Service 是在业务目录中定义则可以省略${业务名称},组件已经默认加入和系统管理相关的 headers,token 是通过浏览器的 localStorage 中的 accessToken 来获取 token 的,组件也加入了 401 和 402 的默认处理,如需自定义可以修改`util/request/index.js`中的代码
定义和使用,一般在 Service 中进行接口定义
```js
import request from '@/util/request';
export const fetchList = (() => {
return {
call: ({ page, limit }) => {
const path = JSON.parse(JSON.stringify(Mock.fetchList));
path.data.data.records = path.data.data.records.slice((page - 1) * limit, page * limit);
return request.get({
path,
// 是否开启mock,开启mock则path就是返回的数据,如果没有开启,path是实际接口的相对地址
mock: true,
// 是否显示loading
loading: {
show: false,
},
});
},
// 用于状态管理的默认值,如果是直接调用则可以省略,最好是给出
defaultResult: () => ({
total: 0,
records: [],
}),
};
})();
```
```js
import { fetchList } from './Service';
fetchList.call({1, 10}).then(res => {
if(!res) return;
if(res.data.resCode === 0) {
// res.data.data
}
// 关闭遮罩层
res.hideIndicator();
});
```
理想的情况是一个业务系统前端之和一个业务系统后端通信,如果想要支持和多个业务系统通信,就需要在`util/request/index.js`中实例化多个 Ajax 对象并导出,同样在`assets/constent.js`中也要同时加入相应的地址配置
系统管理接口的调用,一般都在`config/dict/Dict/dict.system.config.js`中进行定义,然后应用 Dict 就可以,现在和系统管理通信的接口最重要的三个就是,获取用户信息,获取用户权限,获取组织机构。
- **[Preferences](http://49.232.163.126:8083/adhere/util/preferences)**
规范对 store 存取的写法
- **[Emiter](http://49.232.163.126:8083/adhere/util/emitter)**
全局的发布订阅 EventBUS,一般都是在组件之间连理通信
- **统一的提示**
为了统一系统中的提示信息,都是用同一套提示消息
[adhere-ui-prompt-errorprompt](http://49.232.163.126:8083/adhere/ui/errorprompt)
[adhere-ui-prompt-successprompt](http://49.232.163.126:8083/adhere/ui/successprompt)
[adhere-ui-prompt-warnprompt](http://49.232.163.126:8083/adhere/ui/warnprompt)
[adhere-ui-confirm-delconfirm](http://49.232.163.126:8083/adhere/ui/delconfirm)
[adhere-ui-confirm-importantconfirm](http://49.232.163.126:8083/adhere/ui/importantconfirm)
- **[DateDisplay](http://49.232.163.126:8083/adhere/ui/datedisplay)**
统一时间格式化
- **[HistoryBack](http://49.232.163.126:8083/adhere/ui/historyback)**
智能的路由跳转操作,如果有历史则跳回历史,如果没有历史则跳回给定的路由地址,一般都配合[`BackLayout`](http://49.232.163.126:8083/adhere/ui/flexlayout)使用
- **[MessageDialog](http://49.232.163.126:8083/adhere/ui/messagedialog)**
规范弹窗的编写方式,都是用指令式的写法,之前都是使用``的写法,这样好需要手动控制 visible 和清空重置等繁琐事项,规定弹窗都写到 Modal 目录中,以`open${}`业务名称的方式定义方法,下面是一个例子
```js
/**
* 所有相关的弹出框都在这里维
* 方法的命名规open${业务名称}
*/
export default {
/**
* openUpdateModal
* @description - 修改
* @param id
* @param success
* @return {HTMLDivElement}
*/
openUpdateModal(id, success) {
const ref = createRef();
const dialog = MessageDialog.Modal({
config: {
title: Intl.v('查看'),
width: '50%',
zIndex: Resource.Dict.value.ResourceNormalMaxZIndex.value,
maskClosable: false,
footer: [
,
],
},
children: (
{/**/}
),
});
return dialog;
},
};
```
```js
import Modal from './Modal'l
Modal.openUpdateModal();
```
- **[FlexLayout](http://49.232.163.126:8083/adhere/ui/flexlayout)**
提供布局相关组件来规范布局操作,长用到的组件有`HorizontalFlexLayout`、`VerticalFlexLayout`、`ToolBarLayout`、`BackLayout`、`ScrollLayout`,这里的`BackLayout`默认提供了一个返回按钮(使用`HistoryBack`实现),`ScrollLayout`结合`AntFormItemNormalize`使用,建议页面中都是用此组件进行布局
- **[TableGridLayout](http://49.232.163.126:8083/adhere/ui/tablegridlayout)**
提供 Grid 形式的布局,一般用来显示详情或者表单页面,建议页面中的详情和表单页面都使用此布局来显示。工程中也是基于此组件扩展出了`SystemViewTableGridLayout`和`SystemFormTableGridLayout`组件
- **[ReactQuillSandbox](http://49.232.163.126:8083/adhere/ui/reactquillsandbox)**
对 ReactQuill 进行了沙箱的封装,使得样式不会被全局样式覆盖。
- **[IframeIO](http://49.232.163.126:8083/adhere/util/iframeio)**
对 window 和 iframe 之间的通信封装出了一套通信协议。
- **[Globalindicator](http://49.232.163.126:8083/adhere/ui/globalindicator)和[Spin](http://49.232.163.126:8083/adhere/ui/spin)** 无侵入的提示信息
- **[Span](http://49.232.163.126:8083/adhere/ui/span)**、**[Split](http://49.232.163.126:8083/adhere/ui/split)**、**[SplitLayout](http://49.232.163.126:8083/adhere/ui/splitlayout)**
无侵入的分割、无侵入的空隙、分割窗体
- **[Suspense](http://49.232.163.126:8083/adhere/ui/suspense)**
数据加载单元(如第一次是骷髅骨架,其他是 loading),有数据加载的单元,第一次是骷髅骨架(或其他)mount,更新是 loading,一般用在 echart 图表展示做 loading,或者数据源组件做展示的时候,这里的数据源可以是 Promise 的也可不是。
- **[FormitemCreator](http://49.232.163.126:8083/adhere/ui/formitemcreator)**
使用配置方式来进行 Antd 的表单编程。
- **[SearchTable](http://49.232.163.126:8083/adhere/ui/searchtable)**
功能强大的重型表格组件,建议工程中的数据表格都是用这个组件,这个组件的功能太多,可以点击上方链接具体查看功能。
- **[Util](http://49.232.163.126:8083/adhere/util/util)**
工具类,类似于 loadsh
- **[Browsersniff](http://49.232.163.126:8083/adhere/util/browsersniff)**
客户端浏览器识别工具
- **[Resource](http://49.232.163.126:8083/adhere/util/resource)**
一些常用资源的集合,以 Dict 和 JSON 方式提供
- **[Validator](http://49.232.163.126:8083/adhere/util/validator)**
一些常用的典型验证方法,电话号码的验证极为强大。
## Router
路由文件在`config\router\router.jsx`文件中进行定义,下面是一个例子
```js
import { lazy } from 'react';
import BasicLayout from '@/lib/BasicLayout/tabs';
export default [
{
id: 'Root',
path: '/',
routes: [
{
path: '/',
redirect: ['/app'],
},
{
id: 'App',
path: '/app',
component: lazy(() => import(/* webpackChunkName: "app" */ '@/app.jsx')),
routes: [
{
path: '/',
redirect: ['/app/basiclayout'],
},
{
id: 'AppBasicLayout',
path: '/app/basiclayout',
component: BasicLayout,
routes: [
{
path: '/',
redirect: ['/app/basiclayout/dataflowdemo', '/app/basiclayout/task'],
},
{
id: 'AppBasicLayoutDataFlowDemo',
path: '/app/basiclayout/dataflowdemo',
name: 'DateFlowDemo',
sort: 2,
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/dataflowdemo/classlist',
'/app/basiclayout/dataflowdemo/hooklist',
'/app/basiclayout/dataflowdemo/stateclasslist',
'/app/basiclayout/dataflowdemo/statehookslist',
],
},
{
id: 'AppBasicLayoutDataFlowDemoClassListClassList',
path: '/app/basiclayout/dataflowdemo/classlist',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutDataFlowDemoClassList" */ '@/pages/BasicLayout/DataFlowDemo/ClassList/ClassList.jsx'
),
),
name: 'ClassList',
authority: ['AppBasicLayoutDataFlowDemoClassListClassList'],
subAuthority: [],
},
{
id: 'AppBasicLayoutDataFlowDemoHookListHookList',
path: '/app/basiclayout/dataflowdemo/hooklist',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutDataFlowDemoHookList" */ '@/pages/BasicLayout/DataFlowDemo/HookList/HookList.jsx'
),
),
name: 'HookList',
authority: ['AppBasicLayoutDataFlowDemoHookListHookList'],
subAuthority: [],
},
{
id: 'AppBasicLayoutDataFlowDemoStateClassListStateClassList',
path: '/app/basiclayout/dataflowdemo/stateclasslist',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutDataFlowDemoStateClassList" */ '@/pages/BasicLayout/DataFlowDemo/StateClassList/StateClassList.jsx'
),
),
name: 'StateClassList',
authority: ['AppBasicLayoutDataFlowDemoStateClassListStateClassList'],
subAuthority: [],
},
{
id: 'AppBasicLayoutDataFlowDemoStateHooksListStateHooksList',
path: '/app/basiclayout/dataflowdemo/statehookslist',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutDataFlowDemoStateHooksList" */ '@/pages/BasicLayout/DataFlowDemo/StateHooksList/StateHooksList.jsx'
),
),
name: 'StateHooksList',
authority: ['AppBasicLayoutDataFlowDemoStateHooksListStateHooksList'],
subAuthority: [],
},
],
},
{
id: 'AppBasicLayoutTask',
path: '/app/basiclayout/task',
name: '工作任务',
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/implementsubmit',
'/app/basiclayout/task/ledger',
'/app/basiclayout/task/ledgeraudit',
'/app/basiclayout/task/track',
],
},
{
id: 'AppBasicLayoutTaskImplementSubmit',
path: '/app/basiclayout/task/implementsubmit',
name: '落实报送',
sort: 2,
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/implementsubmit/feedback',
'/app/basiclayout/task/implementsubmit/list',
'/app/basiclayout/task/implementsubmit/view',
],
},
{
id: 'AppBasicLayoutTaskImplementSubmitFeedBackMain',
path: '/app/basiclayout/task/implementsubmit/feedback',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskImplementSubmitFeedBack" */ '@/pages/BasicLayout/Task/ImplementSubmit/FeedBack/Main.jsx'
),
),
name: '反馈',
authority: ['AppBasicLayoutTaskImplementSubmitFeedBackMain'],
subAuthority: [],
hide: true,
},
{
id: 'AppBasicLayoutTaskImplementSubmitListMain',
path: '/app/basiclayout/task/implementsubmit/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskImplementSubmitList" */ '@/pages/BasicLayout/Task/ImplementSubmit/List/Main.jsx'
),
),
name: '落实报送',
authority: ['AppBasicLayoutTaskImplementSubmitListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskImplementSubmitViewMain',
path: '/app/basiclayout/task/implementsubmit/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskImplementSubmitView" */ '@/pages/BasicLayout/Task/ImplementSubmit/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskImplementSubmitViewMain'],
subAuthority: [],
hide: true,
},
],
},
{
id: 'AppBasicLayoutTaskLedger',
path: '/app/basiclayout/task/ledger',
name: '台账管理',
sort: 1,
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/ledger/list',
'/app/basiclayout/task/ledger/save',
'/app/basiclayout/task/ledger/update',
'/app/basiclayout/task/ledger/view',
],
},
{
id: 'AppBasicLayoutTaskLedgerListMain',
path: '/app/basiclayout/task/ledger/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerList" */ '@/pages/BasicLayout/Task/Ledger/List/Main.jsx'
),
),
name: '台账管理',
authority: ['AppBasicLayoutTaskLedgerListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskLedgerSaveMain',
path: '/app/basiclayout/task/ledger/save',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerSaveOrUpdate" */ '@/pages/BasicLayout/Task/Ledger/SaveOrUpdate/Main.jsx'
),
),
name: '新增',
authority: ['AppBasicLayoutTaskLedgerSaveMain'],
subAuthority: [],
hide: true,
},
{
id: 'AppBasicLayoutTaskLedgerUpdateMain',
path: '/app/basiclayout/task/ledger/update',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerSaveOrUpdate" */ '@/pages/BasicLayout/Task/Ledger/SaveOrUpdate/Main.jsx'
),
),
name: '修改',
authority: ['AppBasicLayoutTaskLedgerUpdateMain'],
subAuthority: [],
hide: true,
},
{
id: 'AppBasicLayoutTaskLedgerViewMain',
path: '/app/basiclayout/task/ledger/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerView" */ '@/pages/BasicLayout/Task/Ledger/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskLedgerViewMain'],
subAuthority: [],
hide: true,
},
],
},
{
id: 'AppBasicLayoutTaskLedgerAudit',
path: '/app/basiclayout/task/ledgeraudit',
name: '台账审核',
sort: 4,
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/ledgeraudit/audit',
'/app/basiclayout/task/ledgeraudit/list',
'/app/basiclayout/task/ledgeraudit/view',
],
},
{
id: 'AppBasicLayoutTaskLedgerAuditAuditMain',
path: '/app/basiclayout/task/ledgeraudit/audit',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerAuditAudit" */ '@/pages/BasicLayout/Task/LedgerAudit/Audit/Main.jsx'
),
),
name: '审核',
authority: ['AppBasicLayoutTaskLedgerAuditAuditMain'],
subAuthority: [],
hide: true,
},
{
id: 'AppBasicLayoutTaskLedgerAuditListMain',
path: '/app/basiclayout/task/ledgeraudit/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerAuditList" */ '@/pages/BasicLayout/Task/LedgerAudit/List/Main.jsx'
),
),
name: '台账审核',
authority: ['AppBasicLayoutTaskLedgerAuditListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskLedgerAuditViewMain',
path: '/app/basiclayout/task/ledgeraudit/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskLedgerAuditView" */ '@/pages/BasicLayout/Task/LedgerAudit/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskLedgerAuditViewMain'],
subAuthority: [],
hide: true,
},
],
},
{
id: 'AppBasicLayoutTaskTrack',
path: '/app/basiclayout/task/track',
name: '落实跟踪',
sort: 3,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/track/feedbackprogress',
'/app/basiclayout/task/track/feedbackverification',
'/app/basiclayout/task/track/scheduleprogress',
],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackProgress',
path: '/app/basiclayout/task/track/feedbackprogress',
name: '反馈进度',
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/track/feedbackprogress/list',
'/app/basiclayout/task/track/feedbackprogress/view',
],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackProgressListMain',
path: '/app/basiclayout/task/track/feedbackprogress/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackFeedbackProgressList" */ '@/pages/BasicLayout/Task/Track/FeedbackProgress/List/Main.jsx'
),
),
name: '反馈进度',
authority: ['AppBasicLayoutTaskTrackFeedbackProgressListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackProgressViewMain',
path: '/app/basiclayout/task/track/feedbackprogress/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackFeedbackProgressView" */ '@/pages/BasicLayout/Task/Track/FeedbackProgress/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskTrackFeedbackProgressViewMain'],
subAuthority: [],
hide: true,
},
],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackVerification',
path: '/app/basiclayout/task/track/feedbackverification',
name: '反馈核验',
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/track/feedbackverification/list',
'/app/basiclayout/task/track/feedbackverification/verification',
'/app/basiclayout/task/track/feedbackverification/view',
],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackVerificationListMain',
path: '/app/basiclayout/task/track/feedbackverification/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackFeedbackVerificationList" */ '@/pages/BasicLayout/Task/Track/FeedbackVerification/List/Main.jsx'
),
),
name: '反馈核验',
authority: ['AppBasicLayoutTaskTrackFeedbackVerificationListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskTrackFeedbackVerificationVerificationMain',
path: '/app/basiclayout/task/track/feedbackverification/verification',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackFeedbackVerificationVerification" */ '@/pages/BasicLayout/Task/Track/FeedbackVerification/Verification/Main.jsx'
),
),
name: '核验',
authority: [
'AppBasicLayoutTaskTrackFeedbackVerificationVerificationMain',
],
subAuthority: [],
hide: true,
},
{
id: 'AppBasicLayoutTaskTrackFeedbackVerificationViewMain',
path: '/app/basiclayout/task/track/feedbackverification/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackFeedbackVerificationView" */ '@/pages/BasicLayout/Task/Track/FeedbackVerification/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskTrackFeedbackVerificationViewMain'],
subAuthority: [],
hide: true,
},
],
},
{
id: 'AppBasicLayoutTaskTrackScheduleProgress',
path: '/app/basiclayout/task/track/scheduleprogress',
name: '落实进度',
skip: true,
routes: [
{
path: '/',
redirect: [
'/app/basiclayout/task/track/scheduleprogress/list',
'/app/basiclayout/task/track/scheduleprogress/view',
],
},
{
id: 'AppBasicLayoutTaskTrackScheduleProgressListMain',
path: '/app/basiclayout/task/track/scheduleprogress/list',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackScheduleProgressList" */ '@/pages/BasicLayout/Task/Track/ScheduleProgress/List/Main.jsx'
),
),
name: '落实进度',
authority: ['AppBasicLayoutTaskTrackScheduleProgressListMain'],
subAuthority: [],
},
{
id: 'AppBasicLayoutTaskTrackScheduleProgressViewMain',
path: '/app/basiclayout/task/track/scheduleprogress/view',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutTaskTrackScheduleProgressView" */ '@/pages/BasicLayout/Task/Track/ScheduleProgress/View/Main.jsx'
),
),
name: '查看',
authority: ['AppBasicLayoutTaskTrackScheduleProgressViewMain'],
subAuthority: [],
hide: true,
},
],
},
],
},
],
},
],
},
],
},
],
},
];
```
router 节点属性说明
```ts
interface RouterConfig {
// 唯一代表一个节点的标致
id?: string;
// 重定向的path,匹配有权的第一个path
redirect?: string[];
// 路由地址
path: string;
// 组件 如果是BasicLayout或者是BasicLayoutTab那就是左右布局结构
component: React.ReactElement | () => React.ReactElement;
// 如果component是BasicLayout或者BasicLayoutTab,name是左侧菜单的名字
name?: string;
// 如果component是BasicLayout或者BasicLayoutTab生效,菜单的位置
sort?: number;
// 如果component是BasicLayout或者BasicLayoutTab生效,是否掠过此节点(是一个包装节点)
skip?: boolean;
// 如果component是BasicLayout或者BasicLayoutTab生效,页面的权限码列表
authority?: string[];
// 如果component是BasicLayout或者BasicLayoutTab生效,页面的按钮权限码列表
subAuthority?: {key: string, value: string}[];
// 如果component是BasicLayout或者BasicLayoutTab生效,是否在菜单中显示
hide?: boolean;
// 如果component是BasicLayout或者BasicLayoutTab生效,在菜单中显示的icon
icon?: React.ReactElement;
// 子路由配置
routers?: RouterConfig[];
}
```
路由的跳转
```jsx
// @ctsj/router 导出了react-router的所有对象
import { Link } from '@ctsj/router';
jump
;
```
```js
props.history.push();
props.history.replace();
```
获取路由地址
- 要使用routerConfig来获取路由的地址
```js
import { findRouteConfig } from '@/lib/Router/util';
// ../Table是基于当前页面路由的path,来寻找指定的路由 Table是路由的id
findRouteConfig('当前页面路由的path', '../Table')?.path;
```
- 获取当前页面路由的配置
```js
import { useContext } from 'react';
import { RouterConfigContext } from '@/lib/Router/RouterConfigComponent';
export default () => {
const { routerConfig } = useContext(RouterConfigContext);
return Page
}
```
切换路由类可以修改 ENV 的 router 变量来切换路由类型
## 权限
- 权限配置
权限码是一个字符串数组,里面包含了当前登录用户的所有可用权限,这里面包括`页面权限`和`按钮权限`,这两种权限在路由中进行配置,使用`authority`和`subAuthority`来定义,如
```javascript
{
id: 'AppBasicLayoutDataFlowDemoClassTableClassTable',
path: '/app/basiclayout/dataflowdemo/classtable',
component: lazy(() =>
import(
/* webpackChunkName: "appBasicLayoutDataFlowDemoClassTable" */ '@/pages/BasicLayout/DataFlowDemo/ClassTable/ClassTable.jsx'
),
),
name: 'ClassTable',
authority: ['AppBasicLayoutDataFlowDemoClassTableClassTable'],
subAuthority: [{ key: 'TaskResults', value: 'TaskResults' }],
},
```
`authority`代表页面权限,`subAuthority`代表这个页面的按钮权限,如果用户没有当前页面权限,那么在BasicLayout的左侧菜单中是不显示这个页面对应的菜单项,如果直接在浏览器地址栏上输入一个无权限的页面地址那么会显示403页面。
- 权限组件和获取权限
- 使用ConditionalRender.conditionalRender方法判断权限
```js
import { Permission } from '@baifendian/adhere';
import {findSubAuthorityById} from '@/lib/Router/util';
import Util from '@/util';
const { getPermission } = Permission;
ConditionalRender.conditionalRender({
conditional: Util.isAuthority(
[findSubAuthorityById('SystemSchedulingPlan', 'Save')],
getPermission(),
),
match: (
}
type="primary"
onClick={() => {
this.props.history.push(findRouteConfig('SystemSchedulingPlanSave').path);
}}
>
{Intl.v('新增')}
),
})
```
- 使用权限组件
可以使用`PermissionWrap`组件来在代码中使用权限,如
```js
import { Permission } from '@baifendian/adhere';
import {findSubAuthorityById} from '@/lib/Router/util';
const { Permission: PermissionWrap } = Permission;
permissions={['权限1','权限2','权限3']}
noMatch={{data.relFinishTarget}}
>
{() =>
{data.relFinishTarget}
}
```
- 获取按钮权限
使用`findSubAuthorityById`来获取按钮权限,第一个参数是父路由的id,第二个是按钮权限的key,给到后端实际的按钮权限值等于页面路由id + 按钮权限的value 例如 SystemSchedulingPlanListSave
```js
import { Permission } from '@baifendian/adhere';
import {findSubAuthorityById} from '@/lib/Router/util';
const { Permission: PermissionWrap } = Permission;
{data.relFinishTarget}}
>
{data.relFinishTarget}
```
## 工程内典型组件讲解
`components/system`下的组件
- **SystemCard**
一个增强的卡片布局,没有什么特别之处,就是为了统一写法,以后灵活应对修改
- **SystemAnnexList**
附件显示的组件,没有什么特别之处,就是为了统一写法,以后灵活应对修改
- **SystemIframe** 对 iframe 的一个包装,没有什么特别之处,就是为了统一写法,以后灵活应对修改
- **SystemFormTableGridLayout**
对 TableGridLayout 的一个扩展,复写的一些样式,用于表单布局
- **SystemViewTableGridLayout**
对 TableGridLayout 的一个扩展,复写的一些样式,用于详情布局
- **SystemImageAnnexList**
图片附件显示的组件,没有什么特别之处,就是为了统一写法,以后灵活应对修改
- **SystemMediaListItem**
List 的 Item 项布局,可以用来显示复杂的列表项元素显示。
- **SystemSearchTree**
封装了可以查询的 Tree,提供了各种数据源的初始化
- **SystemPage**
页面的顶层组件,如果是页面组件,必须用它来做顶层组件使用,为了以后扩展和修改预留了一个接口
- **SystemBaseSearchTable**
[SearchTable](http://49.232.163.126:8083/adhere/ui/searchtable)的一个包装,用于以扩展留出编程接口
`components/formitem`组件
- **AnnexFormItem**
附件的 FormIem
- **ImageAnnexFormItem**
图片附件的 FormItem
`lib`组件
- **BasicLayout**
一个左侧导航右侧联动的布局组件,左侧导航的数据根据路由配置获取,BasicLayout 还有个 tabs 实现版本,右侧的信息可以通过 tab 卡片的形式展示。
## less
工程中顶层 less 变量应该在`themes\vars`中进行定义
```js
/**
* 业务可以是A-B-C的方式
common-{业务}-font-size
common-{业务}-box-shadow
common-{业务}-background-color
*/
const commonVars = {
// 全局缺省的字体大小
'@common-normal-font-size': '15px',
// 系统名称大小
'@common-system-title-font-size': '21px',
// 界面主标题大小
'@common-primary-title-font-size': '16px',
// 辅助文字大小 | 链接 | 提示
'@common-assist-font-size': '12px',
// 主要文字颜色
'@common-primary-color': '#222',
// 辅助文字颜色
'@common-assist-color': '#888',
// 提示文字颜色
'@common-tip-color': '#999',
// 禁用文字颜色
'@common-disable-color': '#c3c3c3',
// 链接文字颜色
'@common-link-color': '#0099cb',
// 上传弹窗文字颜色
'@upload-primary-color': '#333',
'@upload-gray-color': '#999',
// 边框颜色
'@common-normal-border-color': '#ddd',
// 全局默认的boxShadow
'@common-normal-box-shadow': '0 0 5px 0 rgba(0,0,0,0.10)',
// 全局默认的背景色
'@common-normal-background-color': '#f8f8f8',
// 全局滚动条scrollbar-thumb背景色
'@common-normal-scrollbar-thumb-background-color': '#d8d8d8',
// 全局滚动条scrollbar背景色
'@common-normal-scrollbar-background-color': '#efefef',
// id为app的元素最小宽度
'@common-normal-min-width': '1366px',
// 主颜色
'@primary-color': '#17A293',
'@adhere-primary-color': '@primary-color',
// 导航颜色
'@common-navigator-background-color': '#13927D',
// 导航选中颜色
'@common-navigator-active-background-color': '#16867A',
// 面包屑背景色
'@common-bread-background-color': '#168F82',
// 错误颜色
'@common-error-color': '#D9001B',
// 鼠标经过背景色
'@common-mouse-hover-background-color': 'rgba(23, 162, 147, 80%)',
// 鼠标激活背景色
'@common-mouse-active-background-color': 'rgba(23, 162, 147, 10%)',
// block背景色
'@common-block-background-color': '#f0f2f5',
// 组件背景色
'@common-component-background-color': '#fff',
// vertical paddings
'@common-padding-lg': '24px',
'@common-padding-md': '16px',
'@common-padding-sm': '12px',
'@common-padding-xs': '8px',
'@common-padding-xss': '4px',
// vertical margins
'@common-margin-lg': '24px',
'@common-margin-md': '16px',
'@common-margin-sm': '12px',
'@common-margin-xs': '8px',
'@common-margin-xss': '4px',
// height rules
'@common-height-base': '32px',
'@common-height-lg': '40px',
'@common-height-sm': '24px',
// font Awesome
'@common-fontawesome-css-prefix': 'fa fa-',
// LINK
'@common-link-decoration': 'none',
'@common-link-hover-decoration': 'none',
'@common-link-focus-decoration': 'none',
'@common-link-focus-outline': '0',
// Border color
'@common-border-color-base': 'hsv(0, 0, 85%)',
'@common-border-color-split': 'hsv(0, 0, 94%)',
'@common-border-width-base': '1px',
'@common-border-style-base': 'solid',
// background
'@common-backgeound-hover-color': 'rgba(19,146,125,.1)',
// Outline
'@common-outline-blur-size': '0',
'@common-outline-width': '2px',
'@common-outline-fade': '20%',
'@common-background-color-light': 'hsv(0, 0, 98%);',
'@common-background-color-base': 'hsv(0, 0, 96%);',
// zIndex
'@common-max-zindex': '19999',
// menu
'@common-menu-background-color': '#071b28',
'@common-menu-split-color': '#13927D',
};
module.exports = commonVars;
```
这些变量可以直接使用,无需进行引用,这些变量也可以在脚本中使用`CustomEvnVars.skin`获取,另外[adhere 的 css](http://49.232.163.126:8083/adhere/ui/css)中也内置了很多 less 变量和实用函数
## lint
使用 IDE 的时候需要载入三个 lint 文件`.eslintrc.js`、`.prettierrc.js`和`.stylelintrc.js`
## git commit
使用`git cz`命令执行 git 的 commit 操作
```
$ git add .
$ git cz
```
![][5]
## Jenkins 部署
如果 package.json 有变化才执行 install,否则只执行 build
在 Jeknins 的 shell 中加入如下执行脚本
```
$ node install.js yarn install --ignore-engines
$ yarn buildapp:dev
```
## 访问 git 提交信息
在构建后会在`dist`中会自动生成`gitInfo.txt`文件,此文件会记录一些 git 信息,起到跟踪作用,生成的信息如下
```js
Git操作信息如下:
1、当前分支名称:ref: refs/heads/adhere-vite-ts-2.x
2、远程仓库地址:
3、操作时间:2023/3/10 16:38:47
4、最后一次提交记录id:cef7649e05875c3c88f90708d85449f9b9af2cac
5、最后一次提交记录描述:feat: 加入生成git文件操作
```
## 浏览器的支持
当前编译完的`dist`target是`ES5`代码,css是`stage4`,同时也包含了`corejs`的`polyfill`,但是代码中使用了`Proxy`这个api,如`dict`和`watchmemoized`这个是没办法`polyfill`,现代浏览器除了`IE`之外对`Proxy`的支持都是不错的,所以构建后的`dist`可以用在`chrom`、`firefox`、`Edge`等现代浏览中是没有问题的,但是有一种情况需要注意,就是我们用的第三方package中有的package并没有对es6的语法进行完全的转换,如`const`、`箭头函数 =>`等es6语法没有转换,这样使得有的浏览器(移动端居多),对有的es6语法支持的不够完善,如果用这样的浏览器访问就会出现语法错误,工程中提供了一个`transform:es5`命令,可以将`dist`进行完全的`ES5`转换,如:
```sh
$ npm run transform:es5
```
运行完之后会生成`build`目录,此目录中的内容和`dist`是一致的,这个目录中的所有js都是完全转换后成ES5之后的代码,可以放心使用。
[1]: https://github.com/playerljc/CTSJ-LessMediaQueryLoader/blob/HEAD/README_zh.md
[2]: 
[3]: 
[4]: 
[5]: 