资源模块 | webpack 中文文档 (docschina.org)
为了演示我们项目中可以加载图片,我们需要在项目中使用图片,比较常见的使用图片的方式是两种:
// 创建img元素
const imgEl = document.createElement("img")
imgEl.src = zznhImage
document.body.append(imgEl)
// 创建div元素, 设置背景
const divBgEl = document.createElement("div")
divBgEl.classList.add("img-bg")
document.body.append(divBgEl)
.img-bg {
width: 500px;
height: 500px;
background: url(../img/nhlt.jpg);
}
这个时候,打包会报错
我们当前使用的webpack版本是webpack5:
资源模块类型(asset module type),通过添加 4 种新的模块类型,来替换所有这些 loader:
之前通过使用 file-loader 实现;
之前通过使用 url-loader 实现;
之前通过使用 raw-loader 实现;
之前通过使用 url-loader,并且配置资源体积限制实现;
比如加载图片,我们可以使用下面的方式:
{
test: /\.(png|jpe?g|svg|gif)$/,
type: "asset/resource"
}
但是,如何可以自定义文件的输出路径和文件名呢?
output: {
filename: "bundle.js",
path: path.resolve(__dirname, "./build"),
assetModuleFilename: "img/[name]_[hash:8][ext]"
}
{
test: /\.(png|jpe?g|svg|gif)$/i,
type: "asset/resource",
generator: {
// 占位符
// name: 指向原来的图片名称
// ext: 扩展名
// hash: webpack生成的hash,生成的hash值太长,我们可以使用 :数字 的语法进行部分截取
filename: "img/[name]_[hash:8][ext]"
}
}
我们这里介绍几个最常用的placeholder:
开发中我们往往是小的图片需要转换,但是大的图片直接使用图片即可
我们需要两个步骤来实现:
webpack.config.js
const path = require("path")
module.exports = {
entry: "./src/main.js",
output: {
filename: "bundle.js",
path: path.resolve(__dirname, "./build"),
// assetModuleFilename: "abc.png" // 一般都不会指定确切的名字,同时也不会使用这种方式
},
module: {
rules: [
{
test: /\.(png|jpe?g|svg|gif)$/,
// 1.打包两张图片, 并且这两张图片有自己的地址, 将地址设置到img/bgi中
// 缺点: 多图片加载的两次网络请求
// type: "asset/resource",
// 2.将图片进行base64的编码, 并且直接编码后的源码放到打包的js文件中
// 缺点: 造成js文件非常大, 下载js文件本身消耗时间非常长, 造成js代码的下载和解析/执行时间过长
// type: "asset/inline"
// 3.合理的规范:
// 3.1.对于小一点的图片, 可以进行base64编码
// 3.2.对于大一点的图片, 单独的图片打包, 形成url地址, 单独的请求这个url图片
type: "asset",
parser: {
dataUrlCondition: {
// 图片大于60kb时,不会进行base64编码处理
maxSize: 60 * 1024
}
},
generator: {
// 占位符
// name: 指向原来的图片名称
// ext: 扩展名
// hash: webpack生成的hash,生成的hash值太长,我们可以使用 :数字 的语法进行部分截取
filename: "img/[name]_[hash:8][ext]"
}
},
]
}
}
事实上,在开发中我们很少直接去接触babel,但是babel对于前端开发来说,目前是不可缺少的一部分:
那么,Babel到底是什么呢?
babel本身可以作为一个独立的工具(和postcss一样),不和webpack等构建工具配置来单独使用。
如果我们希望在命令行尝试使用babel,需要安装如下库:
npm install @babel/cli @babel/core -D
使用babel来处理我们的源代码:
npx babel src --out-dir dist
比如我们需要转换箭头函数,那么我们就可以使用箭头函数转换相关的插件:
npm install @babel/plugin-transform-arrow-functions -D
npx babel src --out-dir dist --plugins=@babel/plugin-transform-arrow-functions
查看转换后的结果:我们会发现 const 并没有转成 var
npm install @babel/plugin-transform-block-scoping -D
npx babel src --out-dir dist --plugins=@babel/plugin-transform-block-scoping ,@babel/plugin-transform-arrow-functions
npm install @babel/preset-env -D
npx babel src --out-dir dist --presets=@babel/preset-env
如果之前已经安装了@babel/core,那么这里不需要再次安装;
npm install babel-loader -D
module: {
rules: [
{
test: /\.js$/,
use: [
{
loader: "babel-loader",
// options: {
// plugins: [
// "@babel/plugin-transform-arrow-functions",
// "@babel/plugin-transform-block-scoping"
// ]
// }
}
]
}
但是,再次运行打包,发现里面的js文件完全没有替换,这是因为我们没有配置插件,造成的。当我们把注释打开后,再次运行,这时候就发现,起作用了。
如果我们一个个去安装使用插件,那么需要手动来管理大量的babel插件,我们可以直接给webpack提供一个preset,webpack会根据我们的预设来加载对应的插件列表,并且将其传递给babel。
比如常见的预设有三个:
安装preset-env:
npm install @babel/preset-env
刚才上述的那种发送写在webpack的配置文件中管理起来很不方便,我们可以定义一个babel.config.js,在这里面配置相关的插件信息,babel运行时会自动查找这个文件名,并根据里面的配置项,来进行设置。
babel.config.js
module.exports = {
// plugins: [
// "@babel/plugin-transform-arrow-functions",
// "@babel/plugin-transform-block-scoping"
// ]
presets: [
"@babel/preset-env"
]
}
在webpack中只需要添加babel-loader插件即可。
在开发中我们会编写Vue 相关的代码, webpack 可以对 Vue 代码进行解析:
{{title}}
我是内容, 哈哈哈哈哈哈哈哈
main.js引入
import { createApp } from 'vue'
import Hello from './vue_demo/Hello'
// Vue代码
createApp(Hello).mount("#app")
我们对代码打包会报错:我们需要合适的Loader来处理文件。
这个时候我们需要使用vue-loader:
npm install vue-loader -D
在webpack的模板规则中进行配置:
{
test: /\.vue$/,
loader: "vue-loader"
}
打包依然会报错,这是因为我们必须添加@vue/compiler-sfc来对template进行解析:
npm install @vue/compiler-sfc -D
注意:现在下载vue时,大部分都存在这个解析器,所以有时候可以不用下载
另外我们需要配置对应的Vue插件:
完整示例:
const path = require("path")
const { VueLoaderPlugin } = require("vue-loader/dist/index")
module.exports = {
entry: "./src/main.js",
output: {
filename: "bundle.js",
path: path.resolve(__dirname, "./build"),
},
module: {
rules: [
{
test: /\.vue$/,
loader: "vue-loader"
}
]
},
plugins: [
new VueLoaderPlugin() // 配置的vue插件,必须要
]
}
重新打包即可支持App.vue 的写法
resolve用于设置模块如何被解析:
webpack能解析三种文件路径:
绝对路径:由于已经获得文件的绝对路径,因此不需要再做进一步解析。
相对路径
模块路径
如果是一个文件:
如果是一个文件夹:
会在文件夹中根据 resolve.mainFiles配置选项中指定的文件顺序查找;
- resolve.mainFiles的默认值是 ['index'];
- 再根据 resolve.extensions来解析扩展名;
extensions是解析到文件时自动添加扩展名:
另一个非常好用的功能是配置别名alias:
const path = require("path")
const { VueLoaderPlugin } = require("vue-loader/dist/index")
module.exports = {
entry: "./src/main.js",
output: {
filename: "bundle.js",
path: path.resolve(__dirname, "./build"),
// assetModuleFilename: "abc.png"
},
// ==========关键=============
resolve: {
// 配置扩展名
extensions: [".js", ".json", ".vue", ".jsx", ".ts", ".tsx"],
// 配置别名
alias: {
utils: path.resolve(__dirname, "./src/utils")
}
},
// ==========关键=============
module: {
rules: [
{
test: /\.vue$/,
loader: "vue-loader"
}
]
},
plugins: [
new VueLoaderPlugin()
]
}
示例:
test.js
// 没有配置别名前使用方式,注意:这里可以省略.js后缀,因为extension中配置了后缀的,webpack会自动帮我们加上
// import { sum } from "../../../../math"
// 配置别名后使用方式
import { sum } from "utils/math.js"
console.log(sum(20, 30))