本文由 简悦 SimpRead 转码, 原文地址 mp.weixin.qq.com
前言
最近写页面的时候,用 webp 优化图片大小,就想着有没有相关插件可以开发和打包的时候自动帮我转化和压缩。因为用 vite 打包工具,就去社区找相关插件,可没找到一个比较符合我要求的,就打算自己工作摸鱼写一个吧👀。(也算是第一次写 vite 插件吧😂)
仓库
github: github.com/illusionGD/…[1]
需求
能压缩图片,压缩质量能配置
能自动转 webp 格式,并且打包后能把图片引用路径的后缀改成
.webp支持开发环境和生产环境
不影响原项目图片资源,开发要无感,使用简单
技术栈
sharp:图片压缩、格式转换
@vitejs/plugin-vue:vite 插件开发
实现思路
生产环境
生产环境要考虑两个功能:
1、压缩图片:这个比较简单,在 generateBundle 钩子函数里面处理图片的 chunk 中的 buffer 就可以了
image.png
exportdefaultfunctionImageTools() { 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、转 webp 格式: 还是在 generateBundle 中,直接 copy 一份图片的 chunk,替换 chunk 的 source 和 fileName,再添加到 bundle 中输出
exportdefaultfunctionImageTools() { return { // hook async generateBundle(_options, bundle) { for (const key in bundle){ // 过滤图片key ... // 处理图片buffer ... /*webp相关逻辑*/ // 克隆原本的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**:这里就有点麻烦,需要考虑图片的引入方式和打包的产物,解析产物去替换了
引入方式:
css:
background、background-image组件、html 文件中的标签:
img、source、<div></div>、<div></div>import:
import 'xxx/xxx/xx.png'
产物, 以 vue 为例:
css 中引入的,打包后还是在 css 中
组件中的标签引入,打包后是在 js 中
html 文件中的标签:就在 html 中
知道产物后就比较好替换了,我这里采用一种比较巧妙的方法,不需要转 ast 就能精准替换路径后缀:
先在 generateBundle 中收集打包后图片的名称和对应的 webp 名称:
image.png
再替换上述产物文件中的图片后缀:
functionhandleReplaceWebp(str: string) {let temp = strfor (const key in map) { // 这里的map就是上述图片中的对象 temp = temp.replace(newRegExp(key, 'g'), map[key]) }return temp}exportdefaultfunctionImageTools() { 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) } elseif (/(css)$/.test(key)) { chunk.source = handleReplaceWebp(chunk.source) } } } }, // 替换html中的图片后缀 async writeBundle(opt, bundle) { for (const key in bundle) { const chunk = bundle[key] asany if (/(html)$/.test(key)) { const htmlCode = handleReplaceWebp(chunk.source) writeFileSync(join(opt.dir!, chunk.fileName), htmlCode) } } } }}好了,这就是生产环境大概实现思路了,接下来看开发环境中如何转 webp
开发环境
有人可能认为,开发环境并不需要压缩和转 webp 功能,其实不然,开发环境主要是为了看图片处理后的效果,是否符合预期效果,不然每次都要打包才能看,就有点麻烦了.
开发环境主要考虑以下两点:
和生产环境一样,需要做压缩和转 webp 处理
需要加入缓存,避免每次热更都进行压缩和转 webp
压缩和转 webp 处理
这里就比较简单了,不需要处理 bunlde,在请求本地服务器资源 hook 中 (configureServer) 处理并返回图片资源就行:
exportdefaultfunctionImageTools() { 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) || '') ) // 过滤图片请求 ... const buffer = readFileSync(filePath) // 处理图片压缩和转webp,返回新的buffer,逻辑省略 const newBuffer = await pressBufferToImage(buffer) if (!newBuffer) { next() } res.setHeader('Content-Type', `image/webp`) res.end(newBuffer) } catch (e) { next() } }) }}缓存图片
这里的思路:
第一次请求图片时,缓存对应图片的文件,并带上 hash 值
每次请求时都对比缓存文件的 hash,有就返回,没有就继续走图片处理逻辑
详细代码就不贴了,这里只写大概逻辑
exportfunctiongetCacheKey({ 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}`}exportdefaultfunctionImageTools() { 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) || '') ) // 过滤图片请求 ... const { ext, name } = parse(filePath) const file = readFileSync(filePath) // 获取图片缓存的key,就是图片hash的名称 const cacheKey = getCacheKey( { name, ext, content: file }, { quality, enableWebp, sharpConfig, enableDevWebp, ext } // 这里传生成hash的因子,方便后续改配置重新缓存图片 ) const cachePath = join('node_modules/.cache/vite-plugin-image', cacheKey) // 读缓存 if (existsSync(cachePath)) { return readFileSync(cachePath) } // 处理图片压缩和转webp,返回新的buffer const buffer = readFileSync(filePath) // 处理图片压缩和转webp,返回新的buffer,逻辑省略 const newBuffer = await pressBufferToImage(buffer) // 写入缓存 writeFile(cachePath, newBuffer, () => {}) ... }) }}效果
这里就爬几张原神的图片展示了 (原神, 启动!!)
开发环境:
生产环境:
总结
以上就是大致思路了,代码仅供参考
GitHub: vite-plugin-image-tools[2]
后续打算继续维护这个仓库并更新更多图片相关功能的,有问题欢迎提 issue 呀~
作者:阿帕琪尔
https://juejin.cn/post/7489043337288794139
参考资料
[1]
https://github.com/illusionGD/vite-plugin-image-tools
[2]