本文所用示例的倉庫地址: gayhubjavascript
上文使用 HtmlWebpackPlugin
生成了一個 index.html
文件,而且插件自動把打包後的資源添加到 index.html
文件中,使咱們能夠打開 index.html
在瀏覽器看到 js 的執行效果。本節咱們將用 Webpack Loaders 來處理工做中會用到的其餘三種類型文件:圖片、樣式、字體。css
從這一節開始,因爲項目資源類型變得複雜,每一節的測試項目會放到 demo/
下,再也不是文檔根目錄。html
官網對 Loaders 的介紹很簡單,只有簡單三點:vue
loaderName!
, 好比 import 'style-loader!css-loader?modules!./css/test.css'
rules
中配置,見下文yarn add file-loader
複製代碼
一般狀況下,咱們只須要把圖片複製到(打包)目標目錄,此時咱們使用 file-laoderjava
{
test: /\.(png|jpe?g|gif|webp)$/,
use: [
{
loader: 'file-loader',
options: {
// 文件命名
name: '[name].[ext]',
// 輸出路徑
outputPath: 'imgs/'
}
}
]
}
複製代碼
執行 webpack
命令後,咱們能夠看到圖片已經被打包到了 dist/imgs/
目錄下,而且命名未修改。webpack
yarn add url-loader
複製代碼
當咱們的圖片文件存在一些小標籤時,若是每一個小標籤都獨立請求,顯然是會形成帶寬浪費的,這不合規矩。這個時候咱們須要有相似雪碧圖同樣的方案,把多個小圖片整合爲一個請求。url-loader 就能夠作這樣的整合,但和雪碧圖不一樣的是,它是把大小小於限定值的圖片轉成 base64 編碼放到打包後的代碼中。ios
{
test: /\.(png|jpe?g|gif|webp)$/,
use: [
{
loader: 'url-loader',
options: {
// 文件命名
name: '[name].[ext]',
// 輸出路徑
outputPath: 'imgs/',
// 小於 10k 的圖片轉成 base64 編碼
limit: 10240
}
}
]
}
複製代碼
執行 webpack
命令後,咱們能夠看到只有 多啦A夢.jpg
圖片被打包到了 dist/imgs/
目錄下,其餘 5 個小圖被打包成了一串 base64 碼(在 dist/dist.js
中)。git
大於 10k 的圖github
/***/ "./src/imgs/多啦A夢.jpg":
/*!***************************!*\ !*** ./src/imgs/多啦A夢.jpg ***! \***************************/
/*! no static exports found */
/***/ (function(module, exports, __webpack_require__) {
eval("module.exports = __webpack_require__.p + \"imgs/多啦A夢.jpg\";\n\n//# sourceURL=webpack:///./src/imgs/%E5%A4%9A%E5%95%A6A%E6%A2%A6.jpg?");
/***/ }),
複製代碼
小於 10k 的圖web
/***/ "./src/imgs/分享.png":
/*!*************************!*\ !*** ./src/imgs/分享.png ***! \*************************/
/*! no static exports found */
/***/ (function(module, exports) {
eval("module.exports = \"\"\n\n//# sourceURL=webpack:///./src/imgs/%E5%88%86%E4%BA%AB.png?");
/***/ })
複製代碼
若是咱們不使用 loader 來處理樣式,執行 webpack
能夠看到這樣的提示 You may need an appropriate loader to handle this file type, currently no loaders are configured to process this file.
yarn add css-loader style-loader
複製代碼
css-loader
解析 CSS 文件style-loader
把樣式插入到 DOM 中loader 從右往左、從下往上執行
{
test: /\.css$/,
use: [
'style-loader',
'css-loader'
]
}
複製代碼
此時執行 webpack
就能正確打包了,但並不會再 dist/
下生成 .css
文件,而是把樣式以 style
標籤的形式插入到 index.html
的 head
標籤中(在瀏覽器控制檯查看,插入前樣式文件在 JS 中)。
因爲我是看黃軼講師的「高仿餓了麼」課程後,纔開始在項目中使用預處理器,因此會更喜歡 stylus ,本文的例子也使用 stylus 。
stylus-loader
默認項目中存在 stylus
依賴,因此安裝 stylus-loader
時不要忘記安裝 stylus
yarn add stylus stylus-loader css-loader style-loader
複製代碼
{
test: /\.styl(us)?$/,
use: [
'style-loader',
{
loader: 'css-loader',
options: {
importLoaders: 1 // 在 css-loader 前執行的 loader 數量
}
},
{
loader: 'stylus-loader',
options: {
preferPathResolver: 'webpack' // 優先使用 webpack 用於路徑解析,找不到再使用 stylus-loader 的路徑解析
}
}
]
}
複製代碼
幾年之前寫 CSS 有個蛋疼的地方,某些屬性咱們須要爲不一樣瀏覽器加上不一樣的前綴,好比 Firefox 的 -moz-transform
和 IE 的 -ms-transform
,但有了 PostCSS 後你只須要寫無前綴的屬性,PostCSS 會根據 Can i use 的數據爲你的 CSS 屬性補充上前綴。
yarn add postcss-loader -D
複製代碼
{
test: /\.styl(us)?$/,
use: [
'style-loader',
{
loader: 'css-loader',
options: {
importLoaders: 2 // 在 css-loader 前執行的 loader 數量
}
},
'postcss-loader',
{
loader: 'stylus-loader',
options: {
preferPathResolver: 'webpack' // 優先使用 webpack 用於路徑解析,找不到再使用 stylus-loader 的路徑解析
}
}
]
}
複製代碼
PostCSS 配置文件 postcss.config.js
,配置自動補全(須要 autoprefixer
插件)
yarn add autoprefixer -D
複製代碼
postcss.config.js
module.exports = {
plugins: [
require('autoprefixer')
]
}
複製代碼
此時執行 webpack
命令,能夠在瀏覽器中看到 transform rotate(180deg)
被翻譯成 -webkit-transform: rotate(180deg); transform: rotate(180deg);
。
兩個注意點:
postcss-loader
的順序,在 css-loader
前一步執行postcss-loader
的地方使用 options 來配置 PostCSS ,但更推薦在項目根目錄添加 postcss.config.js
配置文件,別人能更容易知道你使用了 PostCSS。上文有提到 style-loader
把樣式插入到 html 文件,這樣作減小了請求數。但正常項目樣式文件會佔據不小的體積,要知道在插入 index.html
前咱們的樣式文件是儲存在 JS 文件中的,因此 JS 文件會很是大,而且咱們可能須要更清晰的生成物結構,因此也應當清楚如何把樣式從 js 中分離出來 —— 使用 MiniCssExtractPlugin。
yarn add mini-css-extract-plugin -D
複製代碼
把 style-loader
替換爲 MiniCssExtractPlugin Loader
const MiniCssExtractPlugin = require('mini-css-extract-plugin');
module.exports = {
module: {
rules: [
{
test: /\.styl(us)?$/,
use: [
{
loader: MiniCssExtractPlugin.loader,
options: {
publicPath: '../',
hmr: process.env.NODE_ENV === 'development',
reloadAll: true
},
},
{
loader: 'css-loader',
options: {
importLoaders: 2 // 在 css-loader 前執行的 loader 數量
}
},
'postcss-loader',
{
loader: 'stylus-loader',
options: {
preferPathResolver: 'webpack' // 優先使用 webpack 用於路徑解析,找不到再使用 stylus-loader 的路徑解析
}
}
]
}
],
},
plugins: [
new MiniCssExtractPlugin({
filename: 'css/[name].css',
chunkFilename: 'css/[name].css',
}),
]
};
複製代碼
執行 webpack
命令後能夠看到,在 dist/css
目錄下生成了 main.css
文件,而且該文件在 index.html
中引入了。
字體文件和圖片相似,只須要拷貝並放到 dist/fonts/
目錄下,仍然使用 file-loader / url-loader
。
{
test: /\.(woff2?|eot|ttf|otf|svg)(\?.*)?$/i,
use: {
loader: 'url-loader',
options: {
limit: 4096,
name: '[name]_[hash:5].[ext]',
outputPath: 'fonts/'
}
}
}
複製代碼
另外提一句,如今更流行使用 svg 來繪製圖標,繪製路徑存在 iconfong.js
中(阿里圖標庫下載舉例),此時切記不要把 iconfong.js
放在 src/
目錄,由於打包 iconfont.js
毫無心義,只是增長打包時長罷了。應當放到不用打包的靜態目錄,好比 vue-cli 2 的 static/
和 vue-cli 3 的 public/
,並在 html 中引用他們,打包時使用 CopyWebpackPlugin
把 static / public
整個文件夾複製到 dist/
。