边界: webpack编译期间,注册函数,干预webpack的打包过程

loader 的功能定位是转换代码,而一些其他的操作难以用loader来完成
如:

  • 当webpack生成文件的时,顺便多生成一个说明描述文件
  • 当webpack编译启动的时候,控制台输出一句话表示webpak启动了
  • 当xxx时,xxxx

这种类似的功能需要把功能嵌入到webpack的编译流程中,而这种事情的实现依托于plugin
2020-01-15-12-45-16.png

plugin的本质是一个带有apply方法的对象

  1. var plugin={
  2. apply:function(compiler){
  3. }
  4. }

通常,习惯上,我们会将该函数写成构造函数的模式

  1. class MyPlugin{
  2. apply(complier)
  3. }
  4. var plugin=new MyPlugin();

要将插件应用到webpack中,需要把对象配置到webpack的plugins的数组,如下

  1. module.exports={
  2. plugins:[new MyPlugin()]
  3. }

compiler对象是初始化阶段构建的,整个webpack打包期间只有一个compiler对象,后续完成打包的工作是compiler对象内部创建的compilation
apply方法会在创建好complier对象后调用,并向方法传入一个compiler对象

2020-01-15-12-49-26.png
compiler对象提供了大量的钩子函数(hooks,可以理解为事件),plugin的开发者可以注册这些钩子函数,参与webpack编译和生成

  1. class MyPlugin{
  2. apply(compiler){
  3. compiler.hooks.事件名称.事件类型(name,function(){
  4. //事件处理函数
  5. })
  6. }
  7. }

事件名称

即要监听的事件名,即钩子名,所有的钩子
https://www.webpackjs.com/api/compiler-hooks

事件类型

这一部分使用的是Tapable Api,这个小型库是一个专门用于钩子函数监听的库

它提供了一些事件类型

  • tap: 注册一个同步的钩子函数,函数运行完毕则表示事件处理结束
  • tapAsync: 注册一个基于回调的异步钩子函数,函数通过调用一个回调表示事件处理结束
  • tapPromise: 注册一个基于Promise的异步钩子函数,函数通过返回的Promise进入已决状态表示事件处理结束

处理函数

处理函有一个事件参数 comilation

demo 生成资源后,额外添加一个文件列表

  1. module.exports = class FileListPlugin {
  2. constructor(filename = "filelist.txt"){
  3. this.filename = filename;
  4. }
  5. apply(compiler) {
  6. compiler.hooks.emit.tap("FileListPlugin", complation => {
  7. var fileList = [];
  8. for (const key in complation.assets) {
  9. var content = `【${key}】
  10. 大小:${complation.assets[key].size()/1000}KB`;
  11. fileList.push(content);
  12. }
  13. var str = fileList.join("\n\n");
  14. complation.assets[this.filename] = {
  15. source() {
  16. return str
  17. },
  18. size() {
  19. return str.length;
  20. }
  21. }
  22. })
  23. }
  24. }