前言
在平时自己由零搭建项目时,虽然基础配置都比较熟悉,比如配置 file-loader, url-loader, css-loader 等,配置不难,但究竟是怎么起作用的呢,今天就来说说如何编写一个 Webpack Loader。
Loader 作用
按我自己的简单理解,loader 通常指打包的方案,即按什么方式来处理打包,打包的时候它可以拿到模块源代码,经过特定 loader 的转换后返回新的结果。
比如 sass-loader 可以把 SCSS 代码转换成 CSS 代码
编写 Loader
保持功能单一
我们项目中可能会配置很多,但要记住,要保持一个 Loader 的功能单一,避免做多种功能,只需完成一种功能转换即可。
所以如 less 文件转换成 css 文件,也不是一步到位,而是 less-loader, css-loader, style-loader 几个 loader 的链式调用才能完成转换。
模块
因为 Webpack 本身是运行在 Node.js 之上的,一个 loader 其实就是一个 node 模块,这个模块导出的是一个函数,即:
module.exports = function (source) { // source 为 compiler 传递给 Loader 的一个文件的原内容 // 处理... return source // 需要返回处理后的内容 }
这个导出的函数的工作就是获得处理前的原内容,对原内容执行处理后,返回处理后的内容。
替换字符串的 loader
比如我们打包时,想要替换源文件的字符串,这时可以考虑使用 Loader,因为 loader 就是获得源文件内容然后对其进行处理,再返回。
比如 src 目录下有三个文件:
src/msg1.js
export const msg1 = '学习框架'
src/msg2.js
export const msg2 = '深入理解JS'
src/index.js
import { msg1 } from './msg1' import { msg2 } from './msg2' function print() { console.log(`输出:${msg1}, ${msg2}`) } print()
做的事情则是把 msg1 和 msg2 两个文件导入,然后输出两个字符串。
我们要做的事也很简单,把"框架"转为"React 框架", "JS"转为"JavaScript"。
新建 src/loaders/replaceLoader.js文件,
module.exports = function (source) { const handleContent = source.replace('框架', 'React框架').replace('JS', 'JavaScript') return handleContent }
就这样,loader 写完了!!!
上面我们讲到,source 是源文件内容,如果打印的话,则是:
使用 Loader
接下来,我们要来使用它,在根目录下新建文件 webpack.config.js
const path = require('path') module.exports = { mode: 'production', entry: './src/index.js', module: { rules: [ { test: /\.js$/, use: './src/loaders/replaceLoader.js', }, ], }, output: { path: path.resolve(__dirname, 'dist'), filename: '[name].js', }, }
执行npx webpack, 查看打包结果dist/main.js
(()=>{"use strict";console.log("输出:学习React框架, 深入理解JavaScript")})();
替换成功!
需要注意的是,use里面填写的 loader 是去node_modules目录里面找的,由于我们是自定义的 loader,所以不能直接写use: 'replaceLoader',但直接写路径的方式未免难看点,我们可以通过 webpack 来配置:
module.exports = { resolveLoader: { modules: ['node_modules', './src/loaders'], // node_modules找不到,就去./src/loaders找 }, module: { rules: [ { test: /\.js$/, use: 'replaceLoader', }, ], }, }
获取 loader 的 options
写完之后,让我们来想想,其实就是写一个功能函数嘛。
当然,这只是最简单的例子,如果 loader 可以传入参数呢,比如:
module: { rules: [ { test: /\.js$/, use: { loader: 'replaceLoader', options: { params: 'replaceString', }, }, }, ], },
这个时候可以使用this.query来获取,通过this.query.params就能拿到,这里需要注意的是,this 上下文是有用的,所以这个 loader 导出函数不能是箭头函数。
但 webpack 更推荐loader-utils模块来获取,它提供了许多有用的工具,最常用的一种工具是获取传递给 loader 的选项。
首先要安装
npm i -D loader-utils
修改src/loaders/replaceLoader.js
const { getOptions } = require('loader-utils') module.exports = function (source) { console.log(getOptions(this)) // { params: 'replaceString' } console.log(this.query.params) // replaceString const handleContent = source.replace('框架', 'React框架').replace('JS', 'JavaScript') return handleContent }
这里需要注意的是,getOptions(this)参数传入的是 this,也就是说
打印结果:
{ params: 'replaceString' }
{ params: 'replaceString' }
{ params: 'replaceString' }
this.callback()
上面都是返回原来内容转换后的内容,但有些场景下还需要返回其他东西比如 sourceMap
module.exports = function (source) { // 告诉 Webpack 返回的结果 this.callback(null, source, sourceMaps) }
另外也不需要 return 了,所以也可使用此 API 替代 return
const { getOptions } = require('loader-utils') module.exports = function (source) { const handleContent = source.replace('框架', 'React框架').replace('JS', 'JavaScript') this.callback(null, handleContent) }
自定义 loader 应用场景
在所有 function 外面加一层 try catch 代码块捕获错误,避免手动繁琐添加。
实现中英文替换:可以将文字用占位符如{{ title }}包裹,检测到占位符则根据环境变量替换为中英文。
Webpack,Loader
《魔兽世界》大逃杀!60人新游玩模式《强袭风暴》3月21日上线
暴雪近日发布了《魔兽世界》10.2.6 更新内容,新游玩模式《强袭风暴》即将于3月21 日在亚服上线,届时玩家将前往阿拉希高地展开一场 60 人大逃杀对战。
艾泽拉斯的冒险者已经征服了艾泽拉斯的大地及遥远的彼岸。他们在对抗世界上最致命的敌人时展现出过人的手腕,并且成功阻止终结宇宙等级的威胁。当他们在为即将于《魔兽世界》资料片《地心之战》中来袭的萨拉塔斯势力做战斗准备时,他们还需要在熟悉的阿拉希高地面对一个全新的敌人──那就是彼此。在《巨龙崛起》10.2.6 更新的《强袭风暴》中,玩家将会进入一个全新的海盗主题大逃杀式限时活动,其中包含极高的风险和史诗级的奖励。
《强袭风暴》不是普通的战场,作为一个独立于主游戏之外的活动,玩家可以用大逃杀的风格来体验《魔兽世界》,不分职业、不分装备(除了你在赛局中捡到的),光是技巧和战略的强弱之分就能决定出谁才是能坚持到最后的赢家。本次活动将会开放单人和双人模式,玩家在加入海盗主题的预赛大厅区域前,可以从强袭风暴角色画面新增好友。游玩游戏将可以累计名望轨迹,《巨龙崛起》和《魔兽世界:巫妖王之怒 经典版》的玩家都可以获得奖励。
更新日志
- 小骆驼-《草原狼2(蓝光CD)》[原抓WAV+CUE]
- 群星《欢迎来到我身边 电影原声专辑》[320K/MP3][105.02MB]
- 群星《欢迎来到我身边 电影原声专辑》[FLAC/分轨][480.9MB]
- 雷婷《梦里蓝天HQⅡ》 2023头版限量编号低速原抓[WAV+CUE][463M]
- 群星《2024好听新歌42》AI调整音效【WAV分轨】
- 王思雨-《思念陪着鸿雁飞》WAV
- 王思雨《喜马拉雅HQ》头版限量编号[WAV+CUE]
- 李健《无时无刻》[WAV+CUE][590M]
- 陈奕迅《酝酿》[WAV分轨][502M]
- 卓依婷《化蝶》2CD[WAV+CUE][1.1G]
- 群星《吉他王(黑胶CD)》[WAV+CUE]
- 齐秦《穿乐(穿越)》[WAV+CUE]
- 发烧珍品《数位CD音响测试-动向效果(九)》【WAV+CUE】
- 邝美云《邝美云精装歌集》[DSF][1.6G]
- 吕方《爱一回伤一回》[WAV+CUE][454M]