webpack内联代码loader
单页开发很多时候index.html只有些html,外链一些css和js,通常导致一些打包后的代码非常大,造成头轻脚重。导致index.html很小,打包的js和css很臃肿,我们可以将一些常用的例如reset.css,jquery,vue等内联到index.html,合理分配文件大小,毕竟index.html一个http请求就拿那么点东西,太浪费了。
我们尝试写一个webpack plugin,通过配置实现内联代码!
要求:
- 传一个数组,第一个表示要内联的js路径列表,第二个是css路径列表
- css插入
<head>
标签中,js插入<body>
标签中 - 按数组顺序插入
1 | new WebpackInlineSourcePlugin([{ |
WebPackInlineSourcePlugin插件代码 webpack内联code插件
1 | const assert = require('assert'); //断言库 |
我们以Vue-cli为例,vue.config.js配置: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
35const webpack = require('webpack')
const path = require('path');
const WebpackInlineSourcePlugin = require('./plugin/plugin2');
const WebpackfilelistPlugin = require('./plugin/filelist.js');
function resolve(dir) {
return path.join(__dirname, dir)
}
module.exports = {
// 基本路径
baseUrl: './',
devServer: {
port: 8888,
open: true
},
configureWebpack: {
plugins: [
new webpack.ProvidePlugin({
jQuery: 'jquery',
$: 'jquery'
}),
new WebpackInlineSourcePlugin([{
path:['./src/lib/jquery-3.2.1.min.js','./src/lib/vue.min.js']
},{
path:['./src/lib/reset.min.css','./src/lib/A.css']
}]),
new WebpackfilelistPlugin()
]
},
productionSourceMap: false,
lintOnSave: false,
chainWebpack: config => {
config.entry.app = ["babel-polyfill", resolve('src/main.js')]
//config.resolve.alias.set('@', resolve('src'))
}
}
走起
plugin和loader的区别是什么
在 Webpack 运行的生命周期中会广播出许多事件,Plugin 可以监听这些事件,在合适的时机通过 Webpack 提供的 API 【事件钩子】改变输出结果。
对于loader,它就是一个转换器【文件解析器】,将A文件进行编译形成B文件,这里操作的是文件,比如将A.scss或A.less转变为B.css,单纯的文件转换过程
plugin是一个扩展器,它丰富了wepack本身,针对是loader结束后,webpack打包的整个过程,它并不直接操作文件,而是基于事件机制工作,会监听webpack打包过程中的某些节点,执行广泛的任务。
webpack整个构建流程有许多钩子,开发者可以在指定的阶段加入自己的行为到webpack构建流程中。插件由以下构成:
- 一个 JavaScript 命名函数。
- 在插件函数的 prototype 上定义一个 apply 方法。
- 指定一个绑定到 webpack 自身的事件钩子。
- 处理 webpack 内部实例的特定数据。
- 功能完成后调用 webpack 提供的回调。
整个webpack流程由compiler和compilation构成,compiler只会创建一次,compilation如果开起了watch文件变化,那么会多次生成compilation.
Webpack Plugin原理
Webpack 通过 Plugin 机制让其更加灵活,以适应各种应用场景。
在 Webpack 运行的生命周期中会广播出许多事件,Plugin 可以监听这些事件,在合适的时机通过 Webpack 提供的 API 改变输出结果。
一个最基础的 Plugin 的代码是这样的:1
2
3
4
5
6
7
8
9
10
11
12
13
14class BasicPlugin{
// 在构造函数中获取用户给该插件传入的配置
constructor(options){
}
// Webpack 会调用 BasicPlugin 实例的 apply 方法给插件实例传入 compiler 对象
apply(compiler){
compiler.plugin('compilation',function(compilation) {
})
}
}
// 导出 Plugin
module.exports = BasicPlugin;
在使用这个 Plugin 时,相关配置代码如下:1
2
3
4
5
6const BasicPlugin = require('./BasicPlugin.js');
module.export = {
plugins:[
new BasicPlugin(options),
]
}
Webpack 启动后,在读取配置的过程中会先执行 new BasicPlugin(options)
初始化一个 BasicPlugin 获得其实例。
在初始化 compiler 对象后,再调用 basicPlugin.apply(compiler)
给插件实例传入 compiler 对象。
插件实例在获取到 compiler 对象后,就可以通过 compiler.plugin(事件名称, 回调函数)
监听到 Webpack 广播出来的事件。
并且可以通过 compiler 对象去操作 Webpack。
通过以上最简单的 Plugin 相信你大概明白了 Plugin 的工作原理,但实际开发中还有很多细节需要注意,下面来详细介绍。
Compiler 和 Compilation
在开发 Plugin 时最常用的两个对象就是 Compiler 和 Compilation,它们是 Plugin 和 Webpack 之间的桥梁。
Compiler 和 Compilation 的含义如下:
- Compiler 对象包含了 Webpack 环境所有的的配置信息,包含 options,loaders,plugins 这些信息,这个对象在 Webpack 启动时候被实例化,它是全局唯一的,可以简单地把它理解为 Webpack 实例;
- Compilation 对象包含了当前的模块资源、编译生成资源、变化的文件等。当 Webpack 以开发模式运行时,每当检测到一个文件变化,一次新的 Compilation 将被创建。Compilation 对象也提供了很多事件回调供插件做扩展。通过 Compilation 也能读取到 Compiler 对象。
Compiler 和 Compilation 的区别在于:Compiler 代表了整个 Webpack 从启动到关闭的生命周期,而 Compilation 只是代表了一次新的编译。
事件流
Webpack 就像一条生产线,要经过一系列处理流程后才能将源文件转换成输出结果。
这条生产线上的每个处理流程的职责都是单一的,多个流程之间有存在依赖关系,只有完成当前处理后才能交给下一个流程去处理。
插件就像是一个插入到生产线中的一个功能,在特定的时机对生产线上的资源做处理。
Webpack 通过 Tapable 来组织这条复杂的生产线。
Webpack 在运行过程中会广播事件,插件只需要监听它所关心的事件,就能加入到这条生产线中,去改变生产线的运作。
Webpack 的事件流机制保证了插件的有序性,使得整个系统扩展性很好。
Webpack 的事件流机制应用了观察者模式,和 Node.js 中的 EventEmitter 非常相似。
Compiler 和 Compilation 都继承自 Tapable,可以直接在 Compiler 和 Compilation 对象上广播和监听事件,方法如下:
1 | /** |
同理,compilation.apply 和 compilation.plugin 使用方法和上面一致。
在开发插件时,你可能会不知道该如何下手,因为你不知道该监听哪个事件才能完成任务。
在开发插件时,还需要注意以下两点:
- 只要能拿到 Compiler 或 Compilation 对象,就能广播出新的事件,所以在新开发的插件中也能广播出事件,给其它插件监听使用。
- 传给每个插件的 Compiler 和 Compilation 对象都是同一个引用。也就是说在一个插件中修改了 Compiler 或 Compilation 对象上的属性,会影响到后面的插件。
- 有些事件是异步的,这些异步的事件会附带两个参数,第二个参数为回调函数,在插件处理完任务时需要调用回调函数通知 Webpack,才会进入下一处理流程。例如:
1
2
3
4
5
6
7compiler.plugin('emit',function(compilation, callback) {
// 支持处理逻辑
// 处理完毕后执行 callback 以通知 Webpack
// 如果不执行 callback,运行流程将会一直卡在这不往下执行
callback();
});
常用 API
插件可以用来修改输出文件、增加输出文件、甚至可以提升 Webpack 性能、等等,总之插件通过调用 Webpack 提供的 API 能完成很多事情。
由于 Webpack 提供的 API 非常多,有很多 API 很少用的上,又加上篇幅有限,下面来介绍一些常用的 API。
读取输出资源、代码块、模块及其依赖
有些插件可能需要读取 Webpack 的处理结果,例如输出资源、代码块、模块及其依赖,以便做下一步处理。
在 emit
事件发生时,代表源文件的转换和组装已经完成,在这里可以读取到最终将输出的资源、代码块、模块及其依赖,并且可以修改输出资源的内容。
插件代码如下:
1 | class Plugin { |
监听文件变化
在4-5使用自动刷新 中介绍过 Webpack 会从配置的入口模块出发,依次找出所有的依赖模块,当入口模块或者其依赖的模块发生变化时,
就会触发一次新的 Compilation。
在开发插件时经常需要知道是哪个文件发生变化导致了新的 Compilation,为此可以使用如下代码:
1 | // 当依赖的文件发生变化时会触发 watch-run 事件 |
默认情况下 Webpack 只会监视入口和其依赖的模块是否发生变化,在有些情况下项目可能需要引入新的文件,例如引入一个 HTML 文件。
由于 JavaScript 文件不会去导入 HTML 文件,Webpack 就不会监听 HTML 文件的变化,编辑 HTML 文件时就不会重新触发新的 Compilation。
为了监听 HTML 文件的变化,我们需要把 HTML 文件加入到依赖列表中,为此可以使用如下代码:
1 | compiler.plugin('after-compile', (compilation, callback) => { |
修改输出资源
有些场景下插件需要修改、增加、删除输出的资源,要做到这点需要监听 emit
事件,因为发生 emit
事件时所有模块的转换和代码块对应的文件已经生成好,
需要输出的资源即将输出,因此 emit
事件是修改 Webpack 输出资源的最后时机。
所有需要输出的资源会存放在 compilation.assets
中,compilation.assets
是一个键值对,键为需要输出的文件名称,值为文件对应的内容。
设置 compilation.assets
的代码如下:
1 | compiler.plugin('emit', (compilation, callback) => { |
读取 compilation.assets
的代码如下:1
2
3
4
5
6
7
8
9compiler.plugin('emit', (compilation, callback) => {
// 读取名称为 fileName 的输出资源
const asset = compilation.assets[fileName];
// 获取输出资源的内容
asset.source();
// 获取输出资源的文件大小
asset.size();
callback();
});
判断 Webpack 使用了哪些插件
在开发一个插件时可能需要根据当前配置是否使用了其它某个插件而做下一步决定,因此需要读取 Webpack 当前的插件配置情况。
以判断当前是否使用了 ExtractTextPlugin 为例,可以使用如下代码:
1 | // 判断当前配置使用使用了 ExtractTextPlugin, |
实战
下面我们举一个实际的例子,带你一步步去实现一个插件。
该插件的名称取名叫 EndWebpackPlugin,作用是在 Webpack 即将退出时再附加一些额外的操作,例如在 Webpack 成功编译和输出了文件后执行发布操作把输出的文件上传到服务器。
同时该插件还能区分 Webpack 构建是否执行成功。使用该插件时方法如下:
1 | module.exports = { |
要实现该插件,需要借助两个事件:
- done:在成功构建并且输出了文件后,Webpack 即将退出时发生;
- failed:在构建出现异常导致构建失败,Webpack 即将退出时发生;
实现该插件非常简单,完整代码如下:
1 | class EndWebpackPlugin { |
从开发这个插件可以看出,找到合适的事件点去完成功能在开发插件时显得尤为重要。
输出打包文件列表
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
52const gzipSize = require('gzip-size');
const filesize = require('filesize');
const herb = require('herb'); //表格输出
module.exports = class TablePlugin {
constructor(
options = {
errorSize: 1024 * 1024 * 500
}
) {
this.options = options;
}
apply(compiler) {
const errorSize = this.options.errorSize;
const beforeMsg = this.options.beforeMsg;
const afterMsg = this.options.afterMsg;
compiler.hooks.done.tapPromise('table-plugin', async (stats) => {
if (stats.hasErrors()) return;
const assets = stats.compilation.assets;
const gzipedSizes = await Promise.all(
Object.keys(assets).map(async (name) => {
console.log('--------------------------')
console.log(name)
const size = await gzipSize(assets[name].source());
return size;
})
);
const headers = ['asset', 'size', 'gziped'];
const rows = Object.keys(assets).map((name, i) => {
let row = [name, filesize(assets[name].size()), filesize(gzipedSizes[i])];
if (assets[name].size() >= errorSize) {
row = row.map(herb.red);
}
return row;
});
if (beforeMsg) {
console.log(beforeMsg, '\n');
}
herb.table({
headers,
rows,
borders: false
});
if (afterMsg) {
console.log('\n', afterMsg);
}
return Promise.resolve();
});
}
};