原文連結:http://dopro.io/webpack-sprites.html
前言
在HTTP/2.0還沒有普及的現在,css sprite(雪碧圖)仍是前端工程化中必備的一環,而隨著webpack應用的普及,構建工具切換到webpack後雪碧圖該如何處理呢?
目標
先說下我們希望達到的效果。
1、需要合成的雪碧圖icon統一放在src / assets / sprite資料夾下
data:image/s3,"s3://crabby-images/6c094/6c09480fd6dd294e414657ac5188e9d7b8f40208" alt="webpack雪碧圖生成"
data:image/s3,"s3://crabby-images/c873f/c873f5b1fc539741d593c1df4a401754080ab216" alt="webpack雪碧圖生成"
data:image/s3,"s3://crabby-images/7334c/7334c9d3beea094328731fe23b79540ba191c3ea" alt="webpack雪碧圖生成"
data:image/s3,"s3://crabby-images/62309/623098c3f89faf9ce2c93a151878f9159a0dae31" alt="webpack雪碧圖生成"
data:image/s3,"s3://crabby-images/07cc2/07cc266b1e0f37d054572e962faa139a1e27d0db" alt="webpack雪碧圖生成"
常見方案
目前主要的解決方案有兩種
一種是以webpack-spritesmith、postcss-sprites為代表的外掛,webpack-spritesmith主要的執行方式是會預先把sprite目錄中的圖示拼合好,頁面主要依賴生成後的雪碧圖。而postcss-sprites則是通過對已經生成的 CSS 檔案進行分析,將其中的 background 或 background-image 作為依賴收集,合併成雪碧圖後再將相關引數替換。
第二種是loader處理方式,不過網上基本上沒有很好的方案,比如說這種
data:image/s3,"s3://crabby-images/71606/716066269eb18e22e5dccedb290c7d9052ab2817" alt="webpack雪碧圖生成"
目前市面上的方案都不能滿足我們的個性化要求,相對於外掛針對某一資料夾/檔案處理方式,我們更希望採用loader鏈式處理的方式來實現。
解決方案
採用loader的話只能通過新增標記的方式告知需要合成,比如在background url加上“?_sprite”,然後利用正則獲取圖片路徑,給到相關的庫去合成。
data:image/s3,"s3://crabby-images/f67f3/f67f38cda557d9946f7d690072ed3e08ea2fc24b" alt="webpack雪碧圖生成"
其中sprites是需要被合成的圖片路徑,裡面包含了圖示的絕對路徑
var Spritesmith = require('spritesmith');
Spritesmith.run({src: sprites}, function handleResult (err, result) {
/* ... */
});
複製程式碼
另外還提供了多種排列的規則,以及可以設定圖片間的間距
data:image/s3,"s3://crabby-images/8d7a4/8d7a43ac38a8d4967145b8095df39e2bf77fbd96" alt="webpack雪碧圖生成"
Spritesmith.run({src: sprites,algorithm:'top-down',padding: 20}, function handleResult (err, result) {
if(err) {
throw err;
}
});
複製程式碼
列印result可以看到返回的資料主要包括圖片的絕對路徑,座標、寬高以及雪碧圖的Buffer
data:image/s3,"s3://crabby-images/f3f24/f3f24dde404e74a31014b345e1277d4e9997d165" alt="webpack雪碧圖生成"
let imageW = image.width;
let imageH = image.height;
if(mobile){
imageW = imageW/2;
imageH = imageH/2;
}
let imgWidth = 'width:' + imageW + 'px;';
let imgHeight = 'height:' + imageH + 'px;';
if(i < afterContent.length) {
if(afterContent[i] == ';') {
end = i + 1;
afterContent = afterContent.substring(0, end) + backgroundSize+ imgWidth+ '\n' + imgHeight + afterContent.substring(end);
} else {
end = i;
afterContent = afterContent.substring(0, end) + ';\n' + backgroundSize +imgWidth+ '\n' + imgHeight+ afterContent.substring(end);
}
}
let imagePosX = image.x;
let imagePosY = image.y;
if(mobile){
imagePosX = imagePosX/2;
imagePosY = imagePosY/2;
}
let imageX = image.x == 0 ? ' 0' : ' -' + imagePosX + 'px';
let imageY = image.y == 0 ? ' 0' : ' -' + imagePosY + 'px';
let imagePosition = '';
if(image.x || image.y){
imagePosition = imageX + imageY;
}
let cssVal = 'url("' + spriteRelativePath + '")' + imagePosition;
複製程式碼
一般專案中的h5採用的都是雙倍圖,這裡可以增加個判斷,如果h5的話則width、height、background-size都減半處理
module:{
rules:[
{
test:/\.(scss|css)$/,
use: ExtractTextPlugin.extract({
fallback: "style-loader",
publicPath: '../../',
use: [{
loader:'css-loader'
},{
loader:'isprite-loader',
options:{
outputPath:'./src/assets/img/',
mobile:true
}
},{
loader:'sass-loader'
}],
})
},
]
}
複製程式碼
最後
到這裡已經基本能達到預期的效果,不過仍有些問題沒有處理好,比如每次都會生成雪碧圖,這對於編譯速度會有一定的影響,針對這種問題,可以採用hash值進行對比,如果檔案沒有改動的話則不處理。
每一個業務都有不同的需求場景,這種方式可能不一定適用於所有專案,希望對大家有所幫助。
附上demo (https://github.com/Klchan-me/srpite)
歡迎關注“騰訊DeepOcean”微信公眾號,訂閱更多優質前端文章喲~
data:image/s3,"s3://crabby-images/86362/8636280240896d00e2f02bb3a5e44255afb88a94" alt="webpack雪碧圖生成"