# onecrm-pc 项目开发规范草案
# 背景
onecrm项目使用的是中台提供的脚手架搭建,技术选型使用的是 React + TypeScript
# TypeScript的优势
# 1、静态输入
静态类型化是一种功能,可以在开发人员编写脚本时检测错误。查找并修复错误是当今开发团队的迫切需求。有了这项功能,就会允许开发人员编写更健壮的代码并对其进行维护,以便使得代码质量更好、更清晰。
# 2、大型的开发项目
有时为了改进开发项目,需要对代码库进行小的增量更改。这些小小的变化可能会产生严重的、意想不到的后果,因此有必要撤销这些变化。使用TypeScript工具来进行重构更变的容易、快捷。
# 更好的协作
当发开大型项目时,会有许多开发人员,此时乱码和错误的机也会增加。类型安全是一种在编码期间检测错误的功能,而不是在编译项目时检测错误。这为开发团队创建了一个更高效的编码和调试过程。
# 更强的生产力
干净的 ECMAScript 6 代码,自动完成和动态输入等因素有助于提高开发人员的工作效率。这些功能也有助于编译器创建优化的代码。
# 函数组件
React Hooks 出现以后,函数组件有了更多的出错概率,由于函数组件只是一个普通的函数,它非常容易进行类型声明
- 1、使用 ComponentNameProps 形式命名
Props
类型,并导出()。 - 2、优先使用FC类型来声明函数组件
FC是FunctionComponent的简写, 这个类型定义了默认的 props(如 children)以及一些静态属性(如 defaultProps)
import React, { FC } from 'react';
/**
* 声明Props类型
*/
export interface OneCrmComponentProps {
className?: string;
style?: React.CSSProperties;
}
export const OneCrmComponent: FC<OneCrmComponentProps> = props => {
return <div>hello react</div>;
};
2
3
4
5
6
7
8
9
10
11
12
13
这里我们将 OneCrmComponentProps 接口类型定义导出,目的是为了后续的复用和向上层的抽象(可能多个函数组件的props是一致或者大部分类型一样,这样就可以抽象出更加基础的interface)。
你也可以直接使用普通函数来进行组件声明, 下文会看到这种形式更加灵活:
import React, { FC } from 'react';
export interface OneCrmComponentProps {
className?: string;
style?: React.CSSProperties;
// 手动声明children
children?: React.ReactNode;
}
export function OneCrmComponent(props: OneCrmComponentProps) {
return <div>hello react</div>;
}
2
3
4
5
6
7
8
9
10
11
12
- 3、不要直接使用export default导出组件
我们的组件定义中不仅包含组件本身的定义,还可能包含接口类型定义等其他需要导出的模块, 在使用时,尽量采用解构导入的语法:
import {
OneCrmComponent,
OneCrmComponentProps
} from "OneCrmComponent.tsx"
2
3
4
- 4、默认属性的声明 在使用js开发函数式组件的时,只需要给 defaultProps 设置一个对象即可。但是这种写法,在使用FC类型声明时,并不能完美支持。
import React, { FC } from 'react';
export interface OneCrmProps {
name: string;
}
export const OneCrm: FC<OneCrmProps> = ({ name }) => <div>Hello {name}!</div>;
OneCrm.defaultProps = { name: 'oneCrm' };
// ❌! missing name
<OneCrm />;
2
3
4
5
6
7
8
9
10
11
12
如何解决这个问题,可以使用ES6中对象的默认值属性语法。
import React, { FC } from 'react';
export interface OneCrmProps {
name?: string; // 声明为可选属性
}
// 利用对象默认属性值语法
export const OneCrm: FC<OneCrmProps> = ({ name = 'oneCrm' }) => <div>Hello {name}!</div>;
// 调用正常
<OneCrm />;
2
3
4
5
6
7
8
9
10
11
- 5、泛型函数组件
泛型在一些列表型或者容器型的组件中比较常用,直接使用FC无法满足需求:
import React from 'react';
export interface ListProps<T> {
visible: boolean;
list: T[];
renderItem: (item: T, index: number) => React.ReactNode;
}
export function List<T>(props: ListProps<T>) {
return <div />;
}
// Test
function Test() {
return (
<List
list={[1, 2, 3]}
renderItem={i => {
/*自动推断i为number类型*/
}}
/>
);
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
- 6、子组件的声明
使用
Parent.Child
形式的 JSX 可以让节点父子关系更加直观, 它类似于一种命名空间的机制, 可以避免命名冲突. 相比ParentChild
这种命名方式,Parent.Child
更为优雅些。
import React, { PropsWithChildren } from 'react';
export interface LayoutProps {}
export interface LayoutHeaderProps {} // 采用ParentChildProps形式命名
export interface LayoutFooterProps {}
export function Layout(props: PropsWithChildren<LayoutProps>) {
return <div className="layout">{props.children}</div>;
}
// 作为父组件的属性
Layout.Header = (props: PropsWithChildren<LayoutHeaderProps>) => {
return <div className="header">{props.children}</div>;
};
Layout.Footer = (props: PropsWithChildren<LayoutFooterProps>) => {
return <div className="footer">{props.children}</div>;
};
// Test
<Layout>
<Layout.Header>header</Layout.Header>
<Layout.Footer>footer</Layout.Footer>
</Layout>;
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
# 类组件
相比较函数组件来说,基于类的组件的类型检查,就更加好理解。
- 1、继承 Component 或 PureComponent
import React from 'react';
/**
* 首先导出Props声明, 同样是{ComponentName}Props形式命名
*/
export interface OneCrmComponentProps {
defaultCount: number; // 可选props, 不需要?修饰
}
/**
* 组件状态, 不需要暴露
*/
interface State {
count: number;
}
/**
* 类注释
* 继承React.Component, 并声明Props和State类型
*/
export class OneCrmComponent extends React.Component<OneCrmComponentProps, State> {
/**
* 默认参数
*/
public static defaultProps = {
defaultCount: 0,
};
/**
* 初始化State
*/
public state = {
count: this.props.defaultCount,
};
/**
* 声明周期方法
*/
public componentDidMount() {}
/**
* 建议靠近componentDidMount, 资源消费和资源释放靠近在一起, 方便review
*/
public componentWillUnmount() {}
public componentDidCatch() {}
public componentDidUpdate(prevProps: CounterProps, prevState: State) {}
/**
* 渲染函数
*/
public render() {
return (
<div>
{this.state.count}
<button onClick={this.increment}>Increment</button>
<button onClick={this.decrement}>Decrement</button>
</div>
);
}
/**
* ① 组件私有方法, 不暴露
* ② 使用类实例属性+箭头函数形式绑定this
*/
private increment = () => {
this.setState(({ count }) => ({ count: count + 1 }));
};
private decrement = () => {
this.setState(({ count }) => ({ count: count - 1 }));
};
}
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
- 2、使用
static defaultProps
定义默认的props
Typescript 3.0开始支持对使用 defaultProps 对 JSX props 进行推断, 在 defaultProps 中定义的 props 可以不需要'?'可选操作符修饰. 代码如上 👆
# 高阶组件
在React Hooks
出来之前,高阶组件是React的一个非常重要的逻辑复用方式,相比较而言,高阶组件比较重,并且并不容易理解,容易形成很多的嵌套,另外对ts类型的支持并不是很好。所以新的项目还是推荐使用 React Hooks
。
// TODO
# Render Props
React的props(包括 children)并没有限定类型,它可以是一个函数,于是就有了render props,这是和高阶组件一样常见的模式:
import React from 'react';
export interface ThemeConsumerProps {
children: (theme: Theme) => React.ReactNode;
}
export const ThemeConsumer = (props: ThemeConsumerProps) => {
const fakeTheme = { primary: 'red', secondary: 'blue' };
return props.children(fakeTheme);
};
// Test
<ThemeConsumer>
{({ primary }) => {
return <div style={{ color: primary }} />;
}}
</ThemeConsumer>;
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
# 无需使用 PropTypes
有了 Typescript 之后可以安全地约束 Props 和 State, 没有必要引入 React.PropTypes, 而且它的表达能力比较弱。
# 开启 strict 模式
为了真正把 Typescript 用起来, 应该始终开启 strict 模式, 避免使用 any 类型声明。