偷偷摘套内射激情视频,久久精品99国产国产精,中文字幕无线乱码人妻,中文在线中文a,性爽19p

Vite插件:自動(dòng)打包壓縮圖片和轉(zhuǎn)webp

開發(fā)
最近寫頁面的時(shí)候,用webp優(yōu)化圖片大小,就想著有沒有相關(guān)插件可以開發(fā)和打包的時(shí)候自動(dòng)幫我轉(zhuǎn)化和壓縮。因?yàn)橛胿ite打包工具,就去社區(qū)找相關(guān)插件,可沒找到一個(gè)比較符合我要求的,就打算自己工作摸魚寫一個(gè)吧。

倉庫

github: github.com/illusionGD/…[1]

需求

  • 能壓縮圖片,壓縮質(zhì)量能配置
  • 能自動(dòng)轉(zhuǎn)webp格式,并且打包后能把圖片引用路徑的后綴改成.webp
  • 支持開發(fā)環(huán)境和生產(chǎn)環(huán)境
  • 不影響原項(xiàng)目圖片資源,開發(fā)要無感,使用簡(jiǎn)單

技術(shù)棧

  • sharp:圖片壓縮、格式轉(zhuǎn)換
  • @vitejs/plugin-vue:vite插件開發(fā)

實(shí)現(xiàn)思路

生產(chǎn)環(huán)境

生產(chǎn)環(huán)境要考慮兩個(gè)功能:
1、壓縮圖片:這個(gè)比較簡(jiǎn)單,在generateBundle鉤子函數(shù)里面處理圖片的chunk中的buffer就可以了

image.png

export default function ImageTools() {
    return {
        // hook
      async  generateBundle(_options, bundle) {
          for (const key in bundle){
              // 過濾圖片key
              const { ext } = parse(key)
              if (!/(png|jpg|jpeg|webp)$/.test(ext)) {
                  continue
              }
              
              // 處理圖片buffer
              if (chunk.source && chunk.source instanceof Buffer) {
                   // 壓縮圖片,這里就省略邏輯了,可以去看sharp文檔
                  const pressBuffer = await pressBufferToImage(chunk.source) 
                  // 替換處理后的buffer
                  chunk.source = pressBuffer
                }
          }
      }
    }
}

2、轉(zhuǎn)webp格式: 還是在generateBundle中,直接copy一份圖片的chunk,替換chunk的source和fileName,再添加到bundle中輸出

export default function ImageTools() {
    return {
        // hook
      async  generateBundle(_options, bundle) {
          for (const key in bundle){
              // 過濾圖片key
              ...
              // 處理圖片buffer
              ...
              
              /*webp相關(guān)邏輯*/
              // 克隆原本的chunk
              const webpChunk = structuredClone(chunk)
              // 生成webp的buffer, 邏輯省略
              const webpBuffer = await toWebpBuffer(chunk.source)
              
              // 更改新chunk的source和fileName
              webpChunk.source = webpBuffer
              
              const ext = extname(path)
              const webpName = key.replace(ext, '.wep')
              webpChunk.fileName = webpName
              
              // 添加到bundle中
              bundle[webpName] = webpChunk
          }
      }
    }
}

3、**替換路徑后綴為.webp**:這里就有點(diǎn)麻煩,需要考慮圖片的引入方式和打包的產(chǎn)物,解析產(chǎn)物去替換了

引入方式:

  • css:background、background-image
  • 組件、html文件中的標(biāo)簽:img、source<div style="background-image: url('')"></div>、<div style="background: url('')"></div>
  • import:import 'xxx/xxx/xx.png'

產(chǎn)物, 以vue為例:
css中引入的,打包后還是在css中 image.png image.png 組件中的標(biāo)簽引入,打包后是在js中 image.png

html文件中的標(biāo)簽:就在html中 

image.png

知道產(chǎn)物后就比較好替換了,我這里采用一種比較巧妙的方法,不需要轉(zhuǎn)ast就能精準(zhǔn)替換路徑后綴:
先在generateBundle中收集打包后圖片的名稱和對(duì)應(yīng)的webp名稱:

image.png

再替換上述產(chǎn)物文件中的圖片后綴:

function handleReplaceWebp(str: string) {
  let temp = str
  for (const key in map) { // 這里的map就是上述圖片中的對(duì)象
    temp = temp.replace(new RegExp(key, 'g'), map[key])
  }
  return temp
}
export default function ImageTools() {
    return {
        // hook
      async  generateBundle(_options, bundle) {
          for (const key in bundle){
              // 過濾圖片key
              ...
              // 處理圖片buffer
              ...
              // 替換js和css中的圖片后綴
             if (/(js|css)$/.test(key) && enableWebp) {
              if (/(js)$/.test(key)) {
                chunk.code = handleReplaceWebp(chunk.code)
              } else if (/(css)$/.test(key)) {
                chunk.source = handleReplaceWebp(chunk.source)
              }
            }
          }
      },
      
      // 替換html中的圖片后綴
      async writeBundle(opt, bundle) {
          for (const key in bundle) {
            const chunk = bundle[key] as any
            if (/(html)$/.test(key)) {
              const htmlCode = handleReplaceWebp(chunk.source)
              writeFileSync(join(opt.dir!, chunk.fileName), htmlCode)
            }
          }
        }
    }
}

好了,這就是生產(chǎn)環(huán)境大概實(shí)現(xiàn)思路了,接下來看開發(fā)環(huán)境中如何轉(zhuǎn)webp

開發(fā)環(huán)境

有人可能認(rèn)為,開發(fā)環(huán)境并不需要壓縮和轉(zhuǎn)webp功能,其實(shí)不然,開發(fā)環(huán)境主要是為了看圖片處理后的效果,是否符合預(yù)期效果,不然每次都要打包才能看,就有點(diǎn)麻煩了.

開發(fā)環(huán)境主要考慮以下兩點(diǎn):

  1. 和生產(chǎn)環(huán)境一樣,需要做壓縮和轉(zhuǎn)webp處理
  2. 需要加入緩存,避免每次熱更都進(jìn)行壓縮和轉(zhuǎn)webp

壓縮和轉(zhuǎn)webp處理
這里就比較簡(jiǎn)單了,不需要處理bunlde,在請(qǐng)求本地服務(wù)器資源hook中(configureServer) 處理并返回圖片資源就行:

export default function ImageTools() {
    return {
        // hook
     configureServer(server) {
         server.middlewares.use(async (req, res, next) => {
            if (!filterImage(req.url || '')) return next()

            try {
              const filePath = decodeURIComponent(
                path.resolve(process.cwd(), req.url?.split('?')[0].slice(1) || '')
              )
                
              // 過濾圖片請(qǐng)求
              ...
              
              const buffer = readFileSync(filePath)
              // 處理圖片壓縮和轉(zhuǎn)webp,返回新的buffer,邏輯省略
              const newBuffer = await pressBufferToImage(buffer)
             
              if (!newBuffer) {
                next()
              }

              res.setHeader('Content-Type', `image/webp`)
              res.end(newBuffer)
            } catch (e) {
              next()
            }
          })
    }
}

緩存圖片
這里的思路:

  • 第一次請(qǐng)求圖片時(shí),緩存對(duì)應(yīng)圖片的文件,并帶上hash值
  • 每次請(qǐng)求時(shí)都對(duì)比緩存文件的hash,有就返回,沒有就繼續(xù)走圖片處理邏輯

詳細(xì)代碼就不貼了,這里只寫大概邏輯

export function getCacheKey({ name, ext, content}: any, factor: AnyObject) {
  const hash = crypto
    .createHash('md5')
    .update(content)
    .update(JSON.stringify(factor))
    .digest('hex')
  return `${name}_${hash.slice(0, 8)}${ext}`
}

export default function ImageTools() {
    return {
        // hook
     configureServer(server) {
         server.middlewares.use(async (req, res, next) => {
            if (!filterImage(req.url || '')) return next()

            try {
              const filePath = decodeURIComponent(
                path.resolve(process.cwd(), req.url?.split('?')[0].slice(1) || '')
              )
                
              // 過濾圖片請(qǐng)求
              ...
              const { ext, name } = parse(filePath)
              const file = readFileSync(filePath)
              // 獲取圖片緩存的key,就是圖片hash的名稱
              const cacheKey = getCacheKey(
                {
                  name,
                  ext,
                  content: file
                },
                { quality, enableWebp, sharpConfig, enableDevWebp, ext } // 這里傳生成hash的因子,方便后續(xù)改配置重新緩存圖片
              )
              const cachePath = join('node_modules/.cache/vite-plugin-image', cacheKey)

              // 讀緩存
              if (existsSync(cachePath)) {
                return readFileSync(cachePath)
              }
             
              // 處理圖片壓縮和轉(zhuǎn)webp,返回新的buffer
              const buffer = readFileSync(filePath)
              // 處理圖片壓縮和轉(zhuǎn)webp,返回新的buffer,邏輯省略
              const newBuffer = await pressBufferToImage(buffer)
              // 寫入緩存
              writeFile(cachePath, newBuffer, () => {})
              ...
          })
    }
}

效果

這里就爬幾張?jiān)竦膱D片展示了(原神,啟動(dòng)!!)

開發(fā)環(huán)境: image.png

生產(chǎn)環(huán)境:

image.png image.png image.png image.png

總結(jié)

  • 以上就是大致思路了,代碼僅供參考
  • GitHub: vite-plugin-image-tools[2]
  • 后續(xù)打算繼續(xù)維護(hù)這個(gè)倉庫并更新更多圖片相關(guān)功能的,有問題歡迎提issue呀~

原文: https://juejin.cn/post/7489043337288794139

參考資料

[1] https://github.com/illusionGD/vite-plugin-image-tools

[2] https://github.com/illusionGD/vite-plugin-image-tools

責(zé)任編輯:龐桂玉 來源: web前端進(jìn)階
相關(guān)推薦

2023-08-29 08:19:19

PicGo插件開發(fā)

2010-03-12 15:10:16

Ubuntu壓縮圖片

2016-11-02 18:43:02

javascripthtml5vue.js

2020-08-05 16:09:52

javascript壓縮圖片前端

2021-06-22 06:52:46

Vite 插件機(jī)制Rollup

2020-05-07 19:21:53

WebP圖片編輯工具開源

2010-10-12 13:57:43

GoogleWebP

2022-05-19 09:48:07

webp瀏覽器

2021-09-05 15:55:05

WebP圖片格式瀏覽器

2021-11-27 07:09:39

P 圖工具工具應(yīng)用軟件

2021-05-24 05:36:31

壓縮圖片應(yīng)用壓縮工具

2009-06-05 16:43:54

Page Speed網(wǎng)頁加速Google

2022-06-30 10:38:53

Lepton無損壓縮圖片圖片質(zhì)量

2022-10-20 16:04:26

模型質(zhì)量

2024-07-17 09:23:58

Vite插件機(jī)制

2022-01-24 12:38:58

Vite插件開發(fā)

2023-11-22 08:21:25

Android開發(fā)場(chǎng)景

2025-05-09 08:40:42

插件頁面Vite

2022-10-17 14:59:17

圖像算法

2009-09-05 10:26:20

Linux壓縮打包命令Linux命令
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)