跳到主要内容

搭建 React 项目(6)

安装 TS

yarn add typescript -D

安装@types的类型声明包。当使用第三方库时,我们需要引用它的声明文件,才能获得对应的代码补全、接口提示等功能,声明文件是以index.d.ts的一个文件,一般放在项目的根目录中,或者可以通过package.json中的typings/types属性来指定项目中声明文件的位置。

获取一个库类型声明文件有两种方式:

  • 一种是该库包含了自己的声明文件,即包含了index.d.ts文件,这样的第三方库在使用的时候只需要安装库本身就可以了;
  • 另一种是通过 TypeScript 提供的 DefinitelyTyped 仓库获取本身没有包含声明文件的第三方库的声明文件

React 库本身没有包含声明文件,需要通过安装@types类型声明包才可以:

yarn add @types/react @types/react-dom -D

可以看到@types/react中只包含了 React 相关的声明文件而已。

image-20201003174520708

项目配置

使用yarn run tsc --init可以自动在当前终端打开的项目目录生成一个tsconfig.json文件,也可以直接手动创建这个文件,具体的配置项参见 —— TSConfig 参考

这个文件指定了项目根文件和编译项目所需的编译器配置,例如指定需要编译的文件夹include,忽略编译部分文件夹exclude,以及压缩编译的具体配置等compilerOptions

{
"compilerOptions": {
"target": "es5",
"module": "commonjs",
"jsx": "react"
},
"include": ["src/**/*"],
"exclude": ["node_modules", "build", "dll", "public"]
}

配置 webpack

TypeScript 具有特定的文件格式,例如ts或者tsx;webpack 需要安装额外的 loader 来解析他们,babel 7 之前的解决方案是通过安装ts-loader或者awesome-typescript-loader,在项目已经存在 babel 编译器的时候,这样的编译流程是冗余的,一份tsx代码经历的是tsx->ts-loader->js->babel-loader-js这样的流程,也就是一份 JS 文件需要被两个 loader 处理;于是从 babel 7 开始引入了@babel/preset-typescript,可以很方便的利用 babel 来编译所有 JS 相关的代码了。

yarn add @babel/preset-typescript -D

@babel/preset-typescript只包含一个 plugin —— @babel/plugin-transform-typescript,该 plugin 的配置项如下

配置项类型默认值含义
isTSXbooleanfalse当启用的时候,将旧式的 TS 类型断言语法var foo = <string>bar解析为 JSX 标签语法;如果开启,需要同时配置allExtensions: true
jsxPragmastringReact替换编译 JSX 表达式时使用的函数,默认是使用 React 的声明React
allowNamespacesbooleanfalse启用对 TS 的namespace的编译,将来可能将该配置项默认设置成true
onlyRemoveTypeImportsbooleanfalse当启用的时候,编译的时候只会移除type-only语法
allowDeclareFieldsbooleanfalse当启用的时候,在类中使用declare声明的类型会被移除;Babel 8 以后这个配置项会默认开启
module.exports = {
module: {
rules: [
{
test: /\.(js|jsx|tsx|ts)$/,
exclude: /node_modules/,
loader: 'babel-loader',
options: {
presets: [
[
'@babel/preset-env',
{
modules: false,
},
],
['@babel/preset-react'],
['@babel/preset-typescript'],
],
},
},
],
},
resolve: {
extensions: ['.ts', '.tsx'],
},
};

注意事项

@babel/preset-typescript不支持const enum语法,解决方法如下:

@babel/preset-typescript不支持 TypeScript 特定的export =import =语法,解决方法如下:

对于使用了 webpack 的resolve.alias配置模块路径别名的,需要在tsconfig.json中配置以下两个参数:

// webpack.config.js
module.exports = {
resolve: {
alias: {
'@': path.resolve(__dirname, 'src'),
},
},
};
// tsconfig.json
{
"compilerOptions": {
"baseUrl": ".",
"paths": {
"@/*": ["./src/*"]
}
}
}

修改项目

重命名项目文件为ts或者tsx后缀,这样 TS 的类型提示就会生效了