本篇为 从零开始配置 react + typescript 系列第三篇,将带大家完成模板项目的 webpack 配置。整个项目的配置我力求达到以下目标:
灵活: 我在配置 eslint 是选择使用 js 格式而不是 json,就是为了灵活性,使用 js 文件可以让你使用环境变量动态配置,充分发挥 js 语言的能力。当然了,用 js 作配置文件也是有缺点的,不能使用 json schema 校验。
新潮: 我觉得时刻保持对新事物的关注和尝试去使用它是一个优秀的素质。当然,追新很容易碰到坑,但是,没关系,我已经帮你们踩过了,踩不过去我也不会写出来 。从我 eslint parserOptions.ecmaVersion 设置为 2020, 还有经常来一发 yarn upgrade --latest 都可以体现出来。
严格: 就像我平时判断相等性我大多数情况都是使用严格等 ===,而不是非严格等 ==,我觉得越严格,分析起来就越清晰,越早能发现问题。例如我么后面会使用一些 webpack 插件来严格检查模块大小写,检查是否有循环依赖。
安逸: 项目中会尽量集成当前前端生态界实用的和能提高开发愉悦性的(换个词就是花里胡哨)工具。
生产 ready:配置的时候针对不同的打包环境针对性优化,并确保能够投入生产环境使用。
如果读者是初次看到这篇文章,建议先看下前两篇:
从零开始配置 react + typescript(一):dotfiles
从零开始配置 react + typescript(二):linters 和 formatter
dev server
想当初我刚开始学前端框架的那时候,也是被 webpack 折磨的欲仙欲死,我是先自学的 node 才开始写前端,写 nodejs 很方便,自带的模块化方案 commonjs,写前端项目就要配置打包工具。当时最火的打包工具已经是 webpack 了,其次就是 gulp。配置 webpack 总是记不住 webpack 配置有哪些字段,还要扯到一堆相关的工具像 ES6 编译器 babel,CSS 预处理器 sass/less,CSS 后处理器 postcss,以及各种 webpack 的 loader 和 plugin。然后嫌麻烦就有一段时间都是用官方的脚手架,react 就用 cra,也就是 create-react-app,vue 就用 vue-cli。其实也挺好用的,不过说实话,我个人觉得,cra 没 vue-cli 设计的好,无论是易用性和扩展性都完败,cra 不方便用户修改 webpack 配置,vue-cli 不但易于用户修改 webpack 配置,还能让用户保存模板以及自带插件系统。我感觉 react 官方也意识到了这点,所以官方声称近期将会重点优化相关工具链。现在的话,如果我新建一个前端项目,我会选择自己配,不会去采用官方的 cli,因为我觉得我自己已经相当熟悉前端各种构建工具了,等我上半年忙完毕业和找工作的事情我应该会将一些常用的配置抽成一个 npm 包,现在每次写一个项目都 copy 改太累了,一个项目的构建配置有优化点,其它项目都要手动同步一下,效率太低。
技术选型
TypeScript 作为静态类型语言,相对于 js 而言,在类型提示上带来的提升无疑是巨大的。借助 IDE 的类型提示和代码补全,我们需要知道 webpack 配置对象有哪些字段就不用去查官方文档了,而且还不会敲错,很安逸,所以开发语言就选择 TypeScript。
官方文档上有专门一节 Configuration Languages 介绍怎么使用 ts 格式的配置文件配置 webpack 命令行工具,我觉得 webpack-dev-server 命令行工具应该是一样的。
但是我不打算使用官方文档介绍的方式,我压根不打算使用命令行工具,用 node API 才是最灵活的配置方式。配置 webpack devServer 总结一下有以下方式:
webpack-dev-server,这是最不灵活的方式,当然使用场景简单的情况下还是很方便的
webpack-dev-server node API,在 node 脚本里面调用 web-dev-server 包提供的 node API 来启动 devServer
express + webpack devServer 相关中间件,实际上 webpack-dev-server 就是使用 express 以及一些 devServer 相关的中间件开发的。在这种方式下, 各种中间件直接暴露出来了,我们可以灵活配置各个中间件的选项。
koa + webpack devServer 相关中间件,我在 github 上还真的搜到了和 webpack devServer 相关的 webpack 中间件。其实 webpack devServer 就是一个 node server 嘛,用什么框架技术实现不重要,能实现我们需要的功能就行。
我最终采用 express + webpack devServer 相关中间件的方式,为什么不选择用 koa ?因为我觉得官方用的就是 express,用 express 肯定要比 koa 更成熟稳定,坑要少一些。
实现最基本的打包功能
从简到繁,我们先来实现最基本的打包功能使其能够打包 tsx 文件,在此基础上一步一步丰富,优化我们的配置。
配置入口文件
先安装 TypeScript:
yarn add typescript -D
复制代码每个 TypeScript 项目都需要有一个 tsconfig.json 配置文件,使用下面的命令在 src 目录下新建 tsconfig.json 文件:
cd src && npx tsc --init && cd …
复制代码我们暂时调整成这样:
// src/tsconfig.json
{
“compilerOptions”: {
/* Basic Options */
“jsx”: “react”,
“isolatedModules”: true,
/* Strict Type-Checking Options */
"strict": true,
"noImplicitAny": true,
"strictNullChecks": true,
"strictFunctionTypes": true,
"strictBindCallApply": true,
"strictPropertyInitialization": true,
"noImplicitThis": true,
"alwaysStrict": true,
/* Module Resolution Options */
"moduleResolution": "node",
"esModuleInterop": true,
"resolveJsonModule": true,
/* Experimental Options */
"experimentalDecorators": true,
"emitDecoratorMetadata": true,
/* Advanced Options */
"forceConsistentCasingInFileNames": true,
"skipLibCheck": true
}
}
复制代码我们将使用 babel 去编译 TypeScript,babel 在编译 TypeScript 代码是直接去掉 TypeScript 的类型,然后当成普通的 javascript 代码使用各种插件进行编译,因此 tsconfig.json 中很多选项例如 target 和 module 是没有用的。
启用 isolatedModules 选项会在 babel 编译代码时提供一些额外的检查,esModuleInterop 这个选项是用来为了让没有 default 属性的模块也可以使用默认导入,举个简单的例子,如果这个选项没开启,那你导入 fs 模块只能像下面这样导入:
import * as fs from ‘fs’;
复制代码开启了以后,可以直接使用默认导入:
import fs from ‘fs’;
复制代码本质上 ESM 默认导入是导入模块的 default 属性:
import fs from ‘fs’;
// 等同于
import * as module from ‘fs’;
let fs = module.default;
复制代码但是 node 内建模块 fs 是没有 default 属性的,开启 isolatedModules 选项就会在没有 default 属性的情况下自动转换:
import fs, { resolve } from ‘fs’;
// 转换成
import * as fs from ‘fs’;
let { resolve } = fs;
复制代码
我们添加一个入口文件 src/index.tsx,内容很简单:
import plus from ‘./plus’;
console.log(plus(404, 404, 404, 404, 404)); // => 2020
复制代码src/plus.ts 内容为:
export default function plus(…nums: number[]) {
return nums.reduce((pre, current) => pre + current, 0);
}
复制代码编译 TypeScript
我们知道 webpack 默认的模块化系统只支持 js 文件,对于其它类型的文件如 jsx, ts, tsx, vue 以及图片字体等文件类型,我们需要安装对应的 loader。对于 ts 文件,目前存在比较流行的方案有三种:
babel + @babel/preset-typescript
ts-loader
awesome-typescript-loader
awesome-typescript-loader 就算了,作者已经放弃维护了。首先 babel 我们一定要用的,因为 babel 生态有很多实用的插件。虽然 babel 是可以和 ts-loader 一起用,ts-loader 官方给了一个例子 react-babel-karma-gulp,但是我觉得既然 babel 已经能够编译 TypeScript 我们就没必要再加一个 ts-loader,所以我选择方案一。
添加 webpack 配置
我们将把所有 node 脚本放到项目根目的 scripts 文件夹,因为 src 文件夹是前端项目,而 scripts 文件夹是 node 项目,我们应该分别配置 tsconfig.json,通过下面的命令在其中生成初始的 tsconfig.json 文件:
cd ./scripts && npx tsc --init && cd …
复制代码我们调整成酱:
// scripts/tsconfig.json
{
“compilerOptions”: {
/* Basic Options */
“target”: “ES2020”,
“module”: “commonjs”,
/* Strict Type-Checking Options */
"strict": true,
"noImplicitAny": true,
"strictNullChecks": true,
"strictFunctionTypes": true,
"strictBindCallApply": true,
"strictPropertyInitialization": true,
"noImplicitThis": true,
"alwaysStrict": true,
/* Module Resolution Options */
"moduleResolution": "node",
"esModuleInterop": true,
"resolveJsonModule": true,
/* Experimental Options */
"experimentalDecorators": true,
"emitDecoratorMetadata": true,
/* Advanced Options */
"forceConsistentCasingInFileNames": true,
"skipLibCheck": true
}
}
复制代码提几个需要注意的地方:
“target”: “ES2020”,其实编译级别你调的很低是没问题的,你用高级语法 tsc 就转码呗,缺点就是转码后代码体积会变大,执行效率也会降低,原生语法一般都是被优化过的。我喜欢调高一点,一般来说只要不用那些在代码运行平台还不支持的语法就没问题。自从 TypeScript3.7 支持了可选链,我就开始尝试在 TypeScript 使用它,但是问题来了,我之前编译级别一直都是调成最高,也就是 ESNext,因为可选链在 ESNext 已经是标准了,所以 tsc 对于可选链不会转码的。然后 node 12 还不支持可选链,就会报语法错误,于是我就降到 ES2020 了。
Strict Type-Checking Options,这部分全开,既然上了 TypeScript 的船,就用最严格的类型检查,拒绝 AnyScript
删掉 Additional Checks 部份的配置,这部分 eslint 能做的更多,更好
接着我们新建 scripts/configs文件夹,里面用来存放包括 webpack 的配置文件。在其中新建三个 webpack 的配置文件 webpack.common.ts, webpack.dev.ts和 webapck.prod.ts。webpack.common.ts 保存一些公共的配置文件,webpack.dev.ts 是开发环境用的,会被 devServer 读取,webapck.prod.ts 是我们在构建生产环境的 bundle 时用的。
我们接着安装 webpack 和 webpack-merge 以及它们的类型声明文件:
yarn add webpack webpack-merge @types/webpack @types/webpack-merge -D
复制代码webpack-merge 是一个为 merge webpack 配置设计的 merge 工具,提供了一些高级的 merge 方式。不过我目前并没有用到那些高级的 merge 方式,就是当成普通的 merge 工具使用,后续可以探索一下这方面的优化。
为了编译 tsx,我们需要安装 babel-loader 和相关插件:
yarn add babel-loader @babel/core @babel/preset-typescript -D
复制代码新建 babel 配置文件 babel.config.js,现在我们只添加一个 TypeScript preset:
// babel.config.js
module.exports = function(api) {
api.cache(true);
const presets = [’@babel/preset-typescript’];
const plugins = [];
return {
presets,
plugins,
};
};
复制代码添加 babel-loader 到 webpack.common.ts:
// webpack.common.ts`
import { Configuration } from ‘webpack’;
import { projectRoot, resolvePath } from ‘…/env’;
const commonConfig: Configuration = {
context: projectRoot,
entry: resolvePath(projectRoot, ‘./src/index.tsx’),
output: {
publicPath: ‘/’,
path: resolvePath(projectRoot, ‘./dist’),
filename: ‘js/[name].js’,
},
resolve: {
extensions: [’.ts’, ‘.tsx’, ‘.js’, ‘.json’],
},
module: {
rules: [
{
// 导入 jsx 的人少喝点
test: /.(tsx?|js)$/,
loader: ‘babel-loader’,
// 开启缓存
options: { cacheDirectory: true },
exclude: /node_modules/,
},
],
},
};
复制代码我觉得这个 react + ts 项目不应该会出现 jsx 文件,如果导入了 jsx 文件 webpack 就会报错找不到对应的 loader,可以让我们及时处理掉这个有问题的文件。
使用 express 开发 devServer
我们先安装 express 以及和 webpack devServer 相关的一些中间件:
yarn add express webpack-dev-middleware webpack-hot-middleware @types/express @t
ypes/webpack-dev-middleware @types/webpack-hot-middleware -D
复制代码webpack-dev-middleware 这个 express 中间件的主要作用:
作为一个静态文件服务器,使用内存文件系统托管 webpack 编译出的 bundle
如果文件被修改了,会延迟服务器的请求直到编译完成
配合 webpack-hot-middleware 实现热更新功能
webpack-hot-middleware 这个 express 中间件会将自己注册为一个 webpack 插件,监听 webpack 的编译事件。 你哪个 chunck 需要实现热更新,就要在那个 chunk 中导入这个插件提供的 webpack-hot-middleware/client.js 客户端补丁。这个前端代码会获取 devServer 的 Server Sent Events 连接,当有编译事件发生,devServer 会发布通知给这个客户端。客户端接受到通知后,会通过比对 hash 值判断本地代码是不是最新的,如果不是就会向 devServer 拉取更新补丁借助一些其它的工具例如 react-hot-loader 实现热更新。
下面是我另外一个还在开发的 electron 项目修改了一行代码后, client 补丁发送的两次请求:
第一次请求返回的那个 h 值动动脚趾头就能猜出来就是 hash 值,发现和本地的 hash 值比对不上后,再次请求更新补丁。
我们新建文件 scripts/start.ts 用来启动我们的 devServer:
import chalk from ‘chalk’;
import getPort from ‘get-port’;
import logSymbols from ‘log-symbols’;
import open from ‘open’;
import { argv } from ‘yargs’;
import express, { Express } from ‘express’;
import webpack, { Compiler, Stats } from ‘webpack’;
import historyFallback from ‘connect-history-api-fallback’;
import cors from ‘cors’;
import webpackDevMiddleware from ‘webpack-dev-middleware’;
import webpackHotMiddleware from ‘webpack-hot-middleware’;
import proxy from ‘./proxy’;
import devConfig from ‘./configs/webpack.dev’;
import { hmrPath } from ‘./env’;
function openBrowser(compiler: Compiler, address: string) {
if (argv.open) {
let hadOpened = false;
// 第一次编译成功时打开浏览器
compiler.hooks.done.tap(‘open-browser-plugin’, async (stats: Stats) => {
if (!hadOpened && !stats.hasErrors()) {
await open(address);
hadOpened = true;
}
});
}
}
function setupMiddlewares(compiler: Compiler, server: Express) {
const publicPath = devConfig.output!.publicPath!;
// 设置代理
proxy(server);
// 使用 browserRouter 需要重定向所有 html 页面到首页
server.use(historyFallback());
// 开发 chrome 扩展的时候可能需要开启跨域,参考:https://juejin.im/post/5e2027096fb9a02fe971f6b8
server.use(cors());
const devMiddlewareOptions: webpackDevMiddleware.Options = {
// 保持和 webpack 中配置一致
publicPath,
// 只在发生错误或有新的编译时输出
stats: 'minimal',
// 需要输出文件到磁盘可以开启
// writeToDisk: true
};
server.use(webpackDevMiddleware(compiler, devMiddlewareOptions));
const hotMiddlewareOptions: webpackHotMiddleware.Options = {
// sse 路由
path: hmrPath,
// 编译出错会在网页中显示出错信息遮罩
overlay: true,
// webpack 卡住自动刷新页面
reload: true,
};
server.use(webpackHotMiddleware(compiler, hotMiddlewareOptions));
}
async function start() {
const HOST = ‘127.0.0.1’;
// 4个备选端口,都被占用会使用随机端口
const PORT = await getPort({ port: [3000, 4000, 8080, 8888] });
const address = http://${HOST}:${PORT}
;
// 加载 webpack 配置
const compiler = webpack(devConfig);
openBrowser(compiler, address);
const devServer = express();
setupMiddlewares(compiler, devServer);
const httpServer = devServer.listen(PORT, HOST, err => {
if (err) {
console.error(err);
return;
}
// logSymbols.success 在 windows 平台渲染为 √ ,支持的平台会显示 ✔
console.log(
`DevServer is running at ${chalk.magenta.underline(address)} ${logSymbols.success}`,
);
});
process.on('SIGINT', () => {
// 先关闭 devServer
httpServer.close();
// 在 ctrl + c 的时候随机输出 'See you again' 和 'Goodbye'
console.log(
chalk.greenBright.bold(`\n${Math.random() > 0.5 ? 'See you again' : 'Goodbye'}!`),
);
});
}
// 写过 python 的人应该不会陌生这种写法
// require.main === module 判断这个模块是不是被直接运行的
if (require.main === module) {
start();
}
复制代码很多细节我都写到注释里面了,安装其中用到的一些工具库:
yarn add get-port log-symbols -D
复制代码这两个都是 sindresorhus 大佬的作品,一个用于获取可用端口,另一个提供了一些常用的 log 字符:
webpack-dev-middleware 并不支持 webpack-dev-server 中的 historyFallback 和 proxy 功能,其实无所谓,我们可以通过 DIY 我们的 express server 来实现,需要安装对应的两个中间件:
yarn add connect-history-api-fallback http-proxy-middleware @types/connect-history-api-fallback @types/http-proxy-middleware -D
复制代码connect-history-api-fallback 可以直接作为 express 中间件继承到 express server,封装一下 http-proxy-middleware,可以在 proxyTable 中添加自己的代理配置:
// scripts/proxy.ts
import proxyMiddleware from ‘http-proxy-middleware’;
import chalk from ‘chalk’;
import { Express } from ‘express’;
import { Options } from ‘http-proxy-middleware/dist/types’;
interface ProxyTable {
[path: string]: Options;
}
const proxyTable: ProxyTable = {
// 示例配置
‘/path_to_be_proxy_1’: { target: ‘http://target1.domain.com’, changeOrigin: true },
‘/path_to_be_proxy_2’: { target: ‘http://target2.domain.com’, changeOrigin: true },
};
function renderLink(str: string) {
return chalk.magenta.underline(str);
}
function proxy(server: Express) {
Object.entries(proxyTable).forEach(([path, options]) => {
const from = path;
const to = options.target as string;
console.log(Proxy ${renderLink(from)} ${chalk.green('->')} ${renderLink(to)}
);
// eslint-disable-next-line no-param-reassign
if (!options.logLevel) options.logLevel = 'warn';
server.use(path, proxyMiddleware(options));
// 如果需要更灵活的定义方式,请在下面直接使用 server.use(path, proxyMiddleware(options)) 定义
});
}
export default proxy;
复制代码为了启动 devServer,我们还需要安装两个命令行工具:
yarn add ts-node cross-env -D
复制代码ts-node 可以让我们直接运行 TypeScript 代码,cross-env 是一个跨操作系统的设置环境变量的工具,添加启动命令到 npm script:
// package.json
{
“scripts”: {
“start”: “cross-env-shell NODE_ENV=development ts-node --files -P ./scripts/tsconfig.json ./scripts/start.ts”,
}
}
复制代码cross-env 官方文档提到如果要在 windows 平台处理 node 信号例如 SIGINT,也就是我们 ctrl + c 是触发的信号应该使用 cross-env-shell 命令而不是 cross-env 。
ts-node 为了提高执行速度,默认不会读取 tsconfig.json 中的 files, include 和 exclude 字段,而是基于模块依赖读取的。这会导致我们后面写的一些全局的 .d.ts 文件不会被读取,为此,我们需要指定 --files 参数,详情可以查看 help-my-types-are-missing。我们的 node 代码并不多,而且又不是经常性重启项目,直接让 ts-node 扫描整个 scripts 文件夹没多大影响。
启动我们的 dev server,通过 ctrl + c 退出:
npm start
复制代码
开发环境优化
plugins
每个 webpack plugin 都是一个包含 apply 方法的 class,在我们调用 compiler.run 或者 compiler.watch 的时候它就会被调用,并且把 compiler 作为参数传它。compiler 对象提供了各个时期的 hooks,我们可以通过这些 hooks 挂载回调函数来实现各种功能,例如压缩,优化统计信息,在在编译完弹个编译成功的通知等。
显示打包进度
webpack-dev-server 在打包时使用 --progress 参数会在控制台实时输出百分比表示当前的打包进度,但是从上面的图中可以看出只是输出了一些统计信息(stats)。想要实时显示打包进度我了解的有三种方式:
webpack 内置的 webpack.ProgressPlugin 插件
progress-bar-webpack-plugin
webpackbar
内置的 ProgressPlugig 非常的原始,你可以在回调函数获取当前进度,然后按照自己喜欢的格式去打印:
const handler = (percentage, message, …args) => {
// e.g. Output each progress message directly to the console:
console.info(percentage, message, …args);
};
new webpack.ProgressPlugin(handler);
复制代码progress-bar-webpack-plugin 这个插件不是显示百分比,而是显示一个用字符画出来的进度条:
这个插件其实还是挺简洁实用的,但是有个 bug ,如果在打印进度条的时候输出了其它语句,进度条就会错位,我们的 devServer 会在启动后会输出地址:
console.log(DevServer is running at ${chalk.magenta.underline(address)} ${logSymbols.success}
);
复制代码使用这个进度条插件就会出问题下面的问题,遂放弃。
webpackbar 是 nuxt 项目下的库,背靠 nuxt,质量绝对有保证。我之前有段时间用的是 progress-bar-webpack-plugin,因为我在 npm 官网搜索 webpack progress,综合看下来就它比较靠谱,webpackbar 都没搜出来。 看了下 webpackbar 的 package.json,果然 keywords 都是空的。webpackBar 还是我在研究 ant design 的 webpack 配置看到它用了这个插件,才发现了这个宝藏:
安装 webpackbar:
yarn add webpackbar @types/webpackbar -D
复制代码添加配置到 webpack.common.ts 的 plugins 数组,颜色我们使用 react 蓝:
import { Configuration } from ‘webpack’;
const commonConfig: Configuration = {
plugins: [
new WebpackBar({
name: ‘react-typescript-boilerplate’,
// react 蓝
color: ‘#61dafb’,
}),
],
};
复制代码添加版权声明
这个直接用 webpack 内置的 BannerPlugin 即可:
import { BannerPlugin, Configuration } from ‘webpack’;
const commonConfig: Configuration = {
plugins: [
new BannerPlugin({
raw: true,
banner: /** @preserve Powered by react-typescript-boilerplate (https://github.com/tjx666/react-typescript-boilerplate) */
,
}),
],
};
复制代码需要注意的是我们在版权声明的注释中加了 @preserve 标记,我们后面会使用 terser 在生产环境构建时压缩代码,压缩代码时会去掉所有注释,除了一些包含特殊标记的注释,例如我们添加的 @preserve。
优化控制台输出
我们使用 friendly-errors-webpack-plugin 插件让控制台的输出更加友好,下面使用了之后编译成功是的效果:
yarn add friendly-errors-webpack-plugin @types/friendly-errors-webpack-plugin -D
复制代码// webpack.common.ts
import FriendlyErrorsPlugin from ‘friendly-errors-webpack-plugin’;
const commonConfig: Configuration = {
plugins: [new FriendlyErrorsPlugin()],
};
复制代码构建通知
在我大四实习之前,我就没完整写过 vue 项目的,在上家公司实习的那段时间主要就是写 vue,当时我对 vue-cli 那个频繁的错误通知很反感,我和同事说我想去掉这个通知,没曾想同事都是比较喜欢那个通知,既然有人需要,那我们这个项目也配一下。
我们使用 webpack-build-notifier 来支持错误通知,这个插件时 TypeScript 写的,不需要安装 types:
yarn add webpack-build-notifier -D
复制代码// webpack.common.ts
import WebpackBuildNotifierPlugin from ‘webpack-build-notifier’;
const commonConfig: Configuration = {
plugins: [
// suppressSuccess: true 设置只在第一次编译成功时输出成功的通知, rebuild 成功的时候不通知
new WebpackBuildNotifierPlugin({ suppressSuccess: true }),
],
};
复制代码
因为我不喜欢弹通知,所以模板项目中的我注释掉了这个插件,有需要的自己打开就行了。
严格要求路径大小写
下面的测试表明 webpack 默认对路径的大小写不敏感:
我们使用 case-sensitive-paths-webpack-plugin 对路径进行严格的大小写检查:
yarn add case-sensitive-paths-webpack-plugin @types/case-sensitive-paths-webpack-plugin -D
复制代码// webpack.common.ts
import CaseSensitivePathsPlugin from ‘case-sensitive-paths-webpack-plugin’;
const commonConfig: Configuration = {
plugins: [new CaseSensitivePathsPlugin()],
};
复制代码
循环依赖检查
webpack 默认不会对循环依赖报错,通过 circular-dependency-plugin 这个 webpack 插件可以帮我们及时发现循环依赖的问题:
yarn add circular-dependency-plugin @types/circular-dependency-plugin -D
复制代码// webpack.common.ts
import CircularDependencyPlugin from ‘circular-dependency-plugin’;
import { projectRoot, resolvePath } from ‘…/env’;
const commonConfig: Configuration = {
plugins: [
new CircularDependencyPlugin({
exclude: /node_modules/,
failOnError: true,
allowAsyncCycles: false,
cwd: projectRoot,
}),
],
};
复制代码
这里顺便提一下 cwd 也就是工作路径的问题,官方文档直接用 process.cwd(),这是一种不好的做法,项目路径和工作路径是不同的两个概念。在 node 中表示项目路径永远不要用 preocess.cwd(),因为总会有些沙雕用户不去项目根目录启动。process.cwd() 也就是工作路径返回的是你运行 node 时所在的路径,假设说项目在 /code/projectRoot,有些用户直接在系统根目录打开 terminal,来一句 node ./code/projectRoot/index.js,这时 index.js 中 process.cwd() 返回的是就是系统根路径 /,不是有些人认为的还是 /code/projectRoot。
获取项目路径应该使用 path.resolve:
清理上次打包的 bundle
前面介绍了一些花里胡哨的插件,也介绍了一些让我们项目保持健康的插件,现在我们开始介绍一些打包用的插件。
clean-webpack-plugin 它会在第一次编译的时候删除 dist 目录中所有的文件,不过会保留 dist 文件夹,并且再每次 rebuild 的时候会删除所有不再被使用的文件。
这个项目也是 TypeScript 写的,总感觉 TypeScript 写的项目有种莫名的踏实感:
yarn add clean-webpack-plugin -D
复制代码// webpack.common.ts
import { CleanWebpackPlugin } from ‘clean-webpack-plugin’;
const commonConfig: Configuration = {
plugins: [new CleanWebpackPlugin()],
};
复制代码自动生成 index.html
众所周知,腾讯的前端面试很喜欢考计算机网络,我曾多次被问到过如何更新强缓存的问题。解决强缓存立即更新的问题我们一般就是采取在文件名中插入文件内容的 hash 值,然后首页不使用强缓存。这样只要你更新了某个被强缓存的资源文件,由于更新后内容的 hash 值会变化,生成的文件名也会变化,这样你请求首页的时候由于访问的是一个新的资源路径,就会向服务器请求最新的资源。关于浏览器 HTTP 缓存可以看我另一篇文章:通过-koa2-服务器实践探究浏览器 HTTP 缓存机制。
我们后续优化生产环境构建的时候会对将 CSS 拆分成单独的文件,如果 index.html 中插入的引入外部样式的 link 标签的 href 是我们手动设置的,那每次修改样式文件,都会生成一个新的 hash 值,我们都要手动去修改这个路径,太麻烦了,更不要说在开发环境下文件是保存在内存文件系统的,你都看不到文件名。
使用 html-webpack-plugin 可以自动生成 index.html,并且插入引用到了 bundle 和被拆分的 css 等资源路径。
参考 creat-react-app 的模板,我们新建 public 文件夹,并在其中加入 index.html,favico.ico,manifest.json 等文件。public 文件夹用于存放一些将被打包到 dist 文件夹一同发布的文件。
安装并配置 html-webpack-plugin:
yarn add html-webpack-plugin @types/html-webpack-plugin -D
复制代码import HtmlWebpackPlugin from ‘html-webpack-plugin’;
const commonConfig: Configuration = {
output: {
publicPath: ‘/’,
},
plugins: [
new HtmlWebpackPlugin({
// 指定 html 模板路径
template: resolvePath(projectRoot, ‘./public/index.html’),
// 类型不好定义,any 一时爽…
// 定义一些可以在模板中访问的模板参数
templateParameters: (…args: any[]) => {
const [compilation, assets, assetTags, options] = args;
const rawPublicPath = commonConfig.output!.publicPath!;
return {
compilation,
webpackConfig: compilation.options,
htmlWebpackPlugin: {
tags: assetTags,
files: assets,
options,
},
// 除掉 publicPath 的反斜杠,让用户在模板中拼接路径更自然
PUBLIC_PATH: rawPublicPath.endsWith(’/’)
? rawPublicPath.slice(0, -1)
: rawPublicPath,
};
},
}),
],
};
复制代码为了让用户可以像 create-react-app 一样在 index.html 里面通过 PUBLIC_PATH 访问发布路径,需要配置 templateParameters 选项添加 PUBLIC_PATH 变量到模板参数,html-webpack-plugin 默认支持部分 ejs 语法,我们可以通过下面的方式动态设置 favicon.ico , mainfest.json 等资源路径:
const commonConfig: Configuration = {
plugins: [
new CopyPlugin(
[
{
from: ‘*’,
to: resolvePath(projectRoot, ‘./dist’),
toType: ‘dir’,
// index.html 会通过 html-webpack-plugin 自动生成,所以需要被忽略掉
ignore: [‘index.html’],
},
],
{ context: resolvePath(projectRoot, ‘./public’) }
),
],
};
复制代码检查 TypeScript 类型
babel 为了提高编译速度只支持了 TypeScript 语法编译而不支持类型检查,为了在 webpack 打包的同时支持 ts 类型检查,我们会使用 webpack 插件 fork-ts-checker-webpack-plugin,这个 webpack 插件会在一个单独的进程并行的进行 TypeScript 的类型检查,这个项目也是 TypeScript 写的,我们不需要安装 types。
yarn add fork-ts-checker-webpack-plugin -D
复制代码添加到 webpack.dev.ts,限制使用的内存为 1G:
import ForkTsCheckerWebpackPlugin from ‘fork-ts-checker-webpack-plugin’;
const devConfig = merge(commonConfig, {
mode: ‘development’,
plugins: [
new ForkTsCheckerWebpackPlugin({
memoryLimit: 1024,
// babel 转换的是我们前端代码,所以是指向前端代码的 tsconfig.json
tsconfig: resolvePath(projectRoot, ‘./src/tsconfig.json’),
}),
],
});
复制代码同时修改 webpack.prod.ts,因为我们生产环境构建并不会长期的占用内存,所以可以调大点,我们就默认限制生产环境的构建使用的内存为 2G:
// webpack.prod.ts
const prodConfig = merge(commonConfig, {
mode: ‘production’,
plugins: [
new ForkTsCheckerWebpackPlugin({
memoryLimit: 1024 * 2,
tsconfig: resolvePath(projectRoot, ‘./src/tsconfig.json’),
}),
],
});
复制代码缓存神器
hard-source-webpack-plugin 是一个给 modules 提供中间缓存步骤的 webpack 插件,为了看到效果我们可能需要运行两次,第一次就是正常的编译速度,第二次可能会快上很多倍,拿我开发的一个 VSCode 插件 来测试一下:
我先把 node_modules/.cache/hard-source 缓存文件夹删掉,看看没有缓存的时候编译速度:
耗时 3.075 秒,重新编译:
哇 ,直接快了 3.6 倍多,直接跳过了 loaders…
牛逼归牛逼,但是在我的实际使用时发现它会偶尔会出 bug ,不过概率不是很高,本来有 bug 倒没什么,只要作者还在维护就行,但是这个插件的作者貌似维护不是很积极了,最后一次提交代码是 18 年 12 月份,也就是说一年多没维护了。
不过呢,出 bug 一般把缓存删掉就能解决了,虽然有点小毛病,我们这个项目还是配一下,禁不住太香了 :
yarn add hard-source-webpack-plugin @types/hard-source-webpack-plugin -D
复制代码import HardSourceWebpackPlugin from ‘hard-source-webpack-plugin’;
const commonConfig: Configuration = {
plugins: [new HardSourceWebpackPlugin({ info: { mode: ‘none’, level: ‘warn’ } })],
};
复制代码好了,插件部分介绍完了,接下来开始配置 loaders !
loaders
webpack 默认只支持导入 js,处理不了其它文件,需要配置对应的 loader,像 excel-loader 就可以解析 excel 为一个对象,file-loader 可以解析 png 图片为最终的发布路径。loader 是作用于一类文件的,plugin 是作用于 webpack 编译的全过程的。
前面我们只配置了 babel-loader, 使得 webpack 能够处理 TypeScript 文件,实际的开发中我们还需要支持导入样式文件,图片文件,字体文件等。
处理样式文件
我们最终要达到的目标是支持 css/less/sass 三种语法,以及通过 postcss 和 autoprefixer 插件实现自动补齐浏览器头等功能。
CSS
处理 css 文件我们需要安装 style-loader 和 css-loader:
yarn add css-loader style-loader -D
复制代码css-loader 作用是处理 CSS 文件中的 @import 和 url() 返回一个合并后的 CSS 字符串,而 style-loader 负责将返回的 CSS 字符串用 style 标签插到 DOM 中,并且还实现了 webpack 的热更新接口。
style-loader 官方示例配置是这样的:
module.exports = {
module: {
rules: [
{
// i 后缀忽略大小写
test: /.cssKaTeX parse error: Expected 'EOF', got '}' at position 56: …oader'], }̲, ], }, }…/,
use: [
‘style-loader’,
{
loader: ‘css-loader’,
options: {
// CSS modules 比较耗性能,默认就是禁用的
modules: false,
// 开启 sourcemap
sourceMap: true,
// 指定在 CSS loader 处理前使用的 laoder 数量
importLoaders: 0,
},
},
],
},
],
},
};
复制代码less
其实我本人从来不用 less 和 stylus,我一直用的是 sass,less-loader 依赖 less:
yarn add less less-loader -D
复制代码// webpack.common.ts
const commonConfig: Configuration = {
module: {
rules: [
{
test: /.lessKaTeX parse error: Expected 'EOF', got '}' at position 511: … ], }̲, ], }, }…/,
use: [
‘style-loader’,
{
loader: ‘css-loader’,
options: {
modules: false,
sourceMap: true,
importLoaders: 1,
},
},
{
loader: ‘sass-loader’,
options: {
// 中间每个 loader 都要开启 sourcemap,才能生成正确的 soucemap
sourceMap: true,
},
},
],
},
],
},
};
复制代码postcss
记得我在大一上网页设计课学到 CSS3 的时候,很多属性都要加浏览器头处理兼容性,当时就对 CSS 兴趣大减,太麻烦了。自从 node 的出现,前端工程化开始飞速发展,以前前端老被叫做切图仔,现在前端工程师也可以用 node 做伪全栈开发了。
postcss 是 CSS 后处理器工具,因为先有 CSS,postcss 后去处理它,所以叫后处理器。
less/sass 被称之为 CSS 预处理器,因为它们需要被 less 或 node-sass 预先编译代码到 CSS 嘛。
参考 create-react-app 对 postcss 的配置,安装以下插件:
yarn add postcss-loader postcss-flexbugs-fixes postcss-preset-env autoprefixer postcss-normalize -D
复制代码添加 postcss.config.js 用于配置 postcss:
module.exports = {
plugins: [
// 修复一些和 flex 布局相关的 bug
require(‘postcss-flexbugs-fixes’),
// 参考 browserslist 的浏览器兼容表自动对那些还不支持的现代 CSS 特性做转换
require(‘postcss-preset-env’)({
// 自动添加浏览器头
autoprefixer: {
// will add prefixes only for final and IE versions of specification
flexbox: ‘no-2009’,
},
stage: 3,
}),
// 根据 browserslist 自动导入需要的 normalize.css 内容
require(‘postcss-normalize’),
],
};
复制代码我们还需要添加 browserslist 配置到 package.json
// package.json
{
“browserslist”: [
“last 2 versions”,
“Firefox ESR”,
“> 1%”,
“ie >= 11”
],
}
复制代码回顾 CSS, less,sass 的配置可以看到有大量的重复,我们重构并修改 importLoaders 选项:
function getCssLoaders(importLoaders: number) {
return [
‘style-loader’,
{
loader: ‘css-loader’,
options: {
modules: false,
sourceMap: true,
importLoaders,
},
},
{
loader: ‘postcss-loader’,
options: { sourceMap: true },
},
];
}
const commonConfig: Configuration = {
module: {
rules: [
{
test: /.cssKaTeX parse error: Expected 'EOF', got '}' at position 41: …ders(1), }̲, { …/,
use: [
// postcss-loader + less-loader 两个 loader,所以 importLoaders 应该设置为 2
…getCssLoaders(2),
{
loader: ‘less-loader’,
options: {
sourceMap: true,
},
},
],
},
{
test: /.scssKaTeX parse error: Expected 'EOF', got '}' at position 169: … ], }̲, ], }, }…/, /.gifKaTeX parse error: Can't use function '\.' in math mode at position 5: /, /\̲.̲jpe?g/, /.pngKaTeX parse error: Expected 'EOF', got '}' at position 437: … ], }̲, { …/,
use: [
{
loader: ‘url-loader’,
options: {
name: ‘[name]-[contenthash].[ext]’,
outputPath: ‘fonts’,
},
},
],
},
],
},
};
复制代码注意到我这里文件名中都插入了文件内容 hash 值,这样就可以解决强缓存需要立即更新的问题。
sourcemap
devtool
构建速度
重新构建速度
生产环境
品质(quality)
(none)
+++
+++
yes
打包后的代码
eval
+++
+++
no
生成后的代码
cheap-eval-source-map
+
++
no
转换过的代码(仅限行)
cheap-module-eval-source-map
o
++
no
原始源代码(仅限行)
no
原始源代码
cheap-source-map
+
o
yes
转换过的代码(仅限行)
yes
原始源代码(仅限行)
inline-cheap-source-map
+
o
no
转换过的代码(仅限行)
no
原始源代码(仅限行)
–
yes
原始源代码
–
no
原始源代码
–
yes
原始源代码
–
yes
无源代码内容
+++ 非常快速, ++ 快速, + 比较快, o 中等, - 比较慢, – 慢
sourcemap 是现在前端界很多工具必不可缺的一个功能,webpack,TypeScript,babel,powser-assert 等转换代码的工具都要提供 sourcemap 功能,源代码被压缩,混淆,polyfill,没有 sourcemap,根本没办法调试定位问题,对 sourcemap 感兴趣可以看一下阮一峰的 JavaScript Source Map 详解。
考虑到编译速度,调式友好性,我选择 eval-source-map,如果用户在打包时觉得慢,而且能够忍受没有列号,可以考虑调成 cheap-eval-source-map。
我们修改 webpack.dev.ts 的 devtool 为 eval-source-map:
// webpack.dev.ts
import commonConfig from ‘./webpack.common’;
const devConfig = merge(commonConfig, {
devtool: ‘eval-source-map’,
});
复制代码这里顺便提一下 webpack 插件 error-overlay-webpack-plugin,它提供了和 create-react-app 一样的错误遮罩:
但是它有一个限制就是不能使用任何一种基于 eval 的 sourcemap,感兴趣的读者可以尝试以下。
热更新
我们前面给 devServer 添加了 webpack-hot-middleware 中间件,参考它的文档我们需要先添加 webapck 插件webpack.HotModuleReplacementPlugin:
// webpack.dev.ts
import { HotModuleReplacementPlugin, NamedModulesPlugin } from ‘webpack’;
const devConfig = merge(commonConfig, {
plugins: [new HotModuleReplacementPlugin(), new NamedModulesPlugin()],
});
复制代码NamedModulesPlugin 这个插件可以再 rebuild 的时候显示哪些模块被修改了。
还要添加 ‘webpack-hot-middleware/client’ 热更新补丁到我们的 bundle,加入 entry 数组即可:
// webpack.common.ts
import { isProduction, hmrPath } from ‘…/env’;
const commonConfig: Configuration = {
entry: [resolvePath(projectRoot, ‘./src/index.tsx’)],
};
if (isProduction) {
(commonConfig.entry as string[]).unshift(webpack-hot-middleware/client?path=${hmrPath}
);
}
复制代码通过在 entry 后面加 queryString 的方式可以让我们配置一些选项,它是怎么实现的呢?查看 ‘webpack-hot-middleware/client’ 源码可以看到,webpack 会将 queryString 作为全局变量注入这个文件:
其实到这我们也就支持了 CSS 的热更新(style-loader 实现了热更新接口),如果要支持 react 组件的热更新我们还需要配置 react-hot-loader ,配置它之前我们先来优化我们的 babel 配置。
babel 配置优化
前面我们在前面只配置了一个 @babel/preset-typescript 插件用于编译 TypeScript,其实还有很多可以优化的点。
@babel/preset-env
在 babel 中,preset 表示 plugin 的集合,@babel/preset-env 可以让 babel 根据我们配置的 browserslist 只添加需要转换的语法和 polyfill。
安装 @babel/preset-env:
yarn add @babel/preset-env -D
复制代码@babel/plugin-transform-runtime
我们知道默认情况下, babel 在编译每一个模块的时候在需要的时候会插入一些辅助函数例如 _extend,每一个需要的模块都会生成这个辅助函数会造成没必要的代码膨胀,@babel/plugin-transform-runtime 这个插件会将所有的辅助函数都从 @babel/runtime 导入,来减少代码体积。
yarn add @babel/plugin-transform-runtime -D
复制代码@babel/preset-react
虽然 @babel/preset-typescript 就能转换 tsx 成 js 代码,但是 @babel/preset-react 还集成了一些针对 react 项目的实用的插件。
@babel/preset-react 默认会开启下面这些插件:
@babel/plugin-syntax-jsx
@babel/plugin-transform-react-jsx
@babel/plugin-transform-react-display-name
如果设置了 development: true 还会开启:
@babel/plugin-transform-react-jsx-self
@babel/plugin-transform-react-jsx-source
安装依赖 @babel/preset-react:
yarn add @babel/preset-react -D
复制代码react-hot-loader
为了实现组件的局部刷新,我们需要安装 react-hot-loader 这个 babel 插件。
yarn add react-hot-loader
复制代码这个插件不需要安装成 devDependencies,它在生产环境下不会被执行并且确保它占用的体积最小。其实官方正在开发下一代的 react 热更新插件 React Fast Refresh,不过目前还不支持 webpack。
为了看到测试效果,我们安装 react 全家桶并且调整一下 src 文件夹下的默认内容:
yarn add react react-dom react-router-dom
yarn add @types/react @types/react-dom @types/react-router-dom -D
复制代码react 是框架核心接口,react-dom 负责挂载我们的 react 组件到真实的 DOM 上, react-dom-router 是实现了 react-router 接口的 web 平台的路由库。
让 react-hot-loader 接管我们的 react 根组件,其实这个 hot 函数就是一个 hoc 嘛:
// App.ts
import React from ‘react’;
import { hot } from ‘react-hot-loader/root’;
import ‘./App.scss’;
const App = () => {
return (
export default hot(App);
复制代码在 webpack entry 加入热更新补丁:
const commonConfig: Configuration = {
entry: [‘react-hot-loader/patch’, resolvePath(projectRoot, ‘./src/index.tsx’)],
};
复制代码官方文档提到如果需要支持 react hooks 的热更新,我们还需要安装 @hot-loader/react-dom,使用它来替换默认的 react-dom 来添加一些额外的热更新特性,为了替换 react-dom 我们需要配置 webpack alias:
// webpack.common.ts
module.exports = {
resolve: {
alias: {
‘react-dom’: ‘@hot-loader/react-dom’,
},
},
};
复制代码结合前面提到 babel 插件,最终修改 babel.config.js 成:
const envPreset = [
‘@babel/preset-env’,
{
// 只导入需要的 polyfill
useBuiltIns: ‘usage’,
// 指定 corjs 版本
corejs: 3,
// 禁用模块化方案转换
modules: false,
},
];
module.exports = function(api) {
api.cache(true);
return {
presets: [’@babel/preset-typescript’, envPreset],
plugins: [’@babel/plugin-transform-runtime’],
env: {
// 开发环境配置
development: {
presets: [[’@babel/preset-react’, { development: true }]],
plugins: [‘react-hot-loader/babel’],
},
// 生产环境配置
production: {
presets: [’@babel/preset-react’],
plugins: [’@babel/plugin-transform-react-constant-elements’, ‘@babel/plugin-transform-react-inline-elements’],
},
},
};
};
复制代码注意到我们生产环境下还安装了两个插件进行生产环境的优化:
yarn add @babel/plugin-transform-react-constant-elements @babel/plugin-transform-react-inline-elements -D
复制代码@babel/plugin-transform-react-constant-elements 的作用是像下面样将函数组件中的变量提升到函数外来避免每次重新调用函数组件重复声明和没必要的垃圾回收:
const Hr = () => {
return
// 转换成
const _ref =
const Hr = () => {
return _ref;
};
复制代码@babel/plugin-transform-react-inline-elements 的作用读者可以参考 react 的这个 issue:Optimizing Compiler: Inline ReactElements。
生产环境优化
CSS 拆分
如果 CSS 是包含在我们打包的 JS bundle 中那会导致最后体积很大,严重情况下访问首页会造成短暂的白屏。拆分 CSS 我们直接使用 mini-css-extract-plugin:
yarn add mini-css-extract-plugin -D
复制代码修改生产环境配置:
// webpack.prod.ts
import MiniCssExtractPlugin from ‘mini-css-extract-plugin’;
const prodConfig = merge(commonConfig, {
mode: ‘production’,
plugins: [
new MiniCssExtractPlugin({
filename: ‘css/[name].[contenthash].css’,
chunkFilename: ‘css/[id].[contenthash].css’,
ignoreOrder: false,
}),
],
});
复制代码mini-css-extract-plugin 还提供了 mini-css-extract-plugin.loader,它不能和 style-loader 共存,所以我们修改 webpack.common.ts 的配置使得开发环境下使用 style-loader 生产环境下使用 mini-css-extract-plugin.loader:
import { loader as MiniCssExtractLoader } from ‘mini-css-extract-plugin’;
function getCssLoaders(importLoaders: number) {
return [
isProduction ? MiniCssExtractLoader : ‘style-loader’,
{
loader: ‘css-loader’,
options: {
modules: false,
sourceMap: true,
importLoaders,
},
},
{
loader: ‘postcss-loader’,
options: { sourceMap: true },
},
];
}
复制代码代码压缩
JavaScript 压缩
网上很多教程在讲 webpack 压缩代码的时候都是使用 uglifyjs-webpack-plugin,其实这个仓库早就放弃维护了,而且它不支持 ES6 语法,webpack 项目的核心开发者 evilebottnawi 都转向维护 terser-webpack-plugin 了。我们使用 terser-webpack-plugin 在生产环境对代码进行压缩,并且我们可以利用 webpack4 新增的 tree-shaking 去除代码中的死代码,进一步减小 bundle 体积:
yarn add terser-webpack-plugin @types/terser-webpack-plugin -D
复制代码CSS 压缩
压缩 CSS 使用 optimize-css-assets-webpack-plugin:
yarn add optimize-css-assets-webpack-plugin @types/optimize-css-assets-webpack-plugin -D
复制代码最后我们修改 webpack.prod.ts:
import TerserPlugin from ‘terser-webpack-plugin’;
import OptimizeCSSAssetsPlugin from ‘optimize-css-assets-webpack-plugin’;
const prodConfig = merge(commonConfig, {
mode: ‘production’,
optimization: {
minimize: true,
minimizer: [new TerserPlugin({ extractComments: false }), new OptimizeCSSAssetsPlugin()],
},
});
复制代码构建分析
我们添加一些 webpack 插件用来进行构建分析