实现图片本地化 ServiceWorker + IndexedDB
点击上方 前端Q,关注公众号
回复加群,加入前端Q技术交流群
1. 背景
在一些安全场景,或者一些本地化的场景(如本地化的 Markdown 记事本),如果有图片上传并需要查看的场景,在不上传到服务器的情况下,实现这个效果,通常是把图片 Base64 化,但编码后的字符串会非常长,体验较差。
这里不妨尝试使用 ServiceWorker + IndexedDB 来解决这个问题。
2. API 优势
IndexedDB 几乎什么都可以存储,二进制类的文件更不在话下,且存储空间在大多数浏览器中,是和系统空间持平的。
ServiceWorker 可以做页面的网络代理层,所以就不需要 Base64 化了,可以直接写一个特定规则的图片地址,实现从 IndexedDB 中 读/写 文件。
3. 实现
例如把这种特殊的图片采用 .dbimg
后缀。
上传图片时,就可以通过拦截上传请求的方式实现,流程如下:
![](https://filescdn.proginn.com/fdbecb6326f65121eecc5dc22f342048/e917ed1e83f2bef1e93312593c362eb7.webp)
self.addEventListener("fetch", (e) => {
const url = e.request.url;
if (e.request.method == "POST" && url.indexOf("update-img") > -1) {
e.respondWith(
e.request.formData().then((data) => {
const file = data.get("img");
const name = Date.now() + Math.random().toString().substr(2, 4);
db.put("img", { name, file });
return new Response(
JSON.stringify({
name: name + ".dbimg",
})
);
})
);
}
});
复制代码
读取时,通过判断 .dbimg
后缀实现,流程如下:
![](https://filescdn.proginn.com/f5941060a1309b5e3ea2484b39a775c9/c6ea80e7ac57fb4a38d86c2f7c0acd2f.webp)
self.addEventListener("fetch", (e) => {
const url = e.request.url;
if (e.request.method == "GET" && url.substr(url.length - 6) == ".dbimg") {
const name = url.substring(url.lastIndexOf("/") + 1, url.lastIndexOf("."));
e.respondWith(
db
.getAllMatching("img", { index: "name", query: IDBKeyRange.only(name) })
.then((data) => {
if (data.length) {
return new Response(data[0].file);
}
return new Response("", { status: 404 });
})
);
}
});
复制代码
4. 效果
体验地址:lecepin.github.io/file-proxy-…[1]
⭐ 仓库地址:github.com/lecepin/fil…[2]
![](https://filescdn.proginn.com/0d1e4ff2b8ba28d0e8ac43fe6f70779e/e335e065b86cbfb652ad5385039f862d.webp)
![](https://filescdn.proginn.com/552811c14a9eeb373a163b0a394be848/11b7bad44acba3c1d7da8a20c36e3ca9.webp)
![](https://filescdn.proginn.com/71a2202206434127fd7db7852daa3f4b/2cff511a88f0261891467c623c1b220d.webp)
关于本文
作者:lecepin
https://juejin.cn/post/7075650171471659015
![](https://filescdn.proginn.com/13c6acbd42fbc4dfc7285cc84e8255af/73ad0a099ca7430702dbc53739579cd7.webp)
往期推荐
![](https://filescdn.proginn.com/36fac610921518ea4e4bb4b7479828c5/d0d1482a9e53d5ba3dbd56206817e5e6.webp)
![](https://filescdn.proginn.com/2e91257aa86a617901a3634398b6c53b/2ad0120354530b140895f343da1580b7.webp)
![](https://filescdn.proginn.com/2e91257aa86a617901a3634398b6c53b/2ad0120354530b140895f343da1580b7.webp)
最后
欢迎加我微信,拉你进技术群,长期交流学习...
欢迎关注「前端Q」,认真学前端,做个专业的技术人...
![](https://filescdn.proginn.com/68e9df321c662be19a15b5962000fe91/9d0035892137394f12f091d657c1045f.webp)
![](https://filescdn.proginn.com/cff9049cfcfe60b78f1634e21dd2dd27/246a18618d49341cbc858cf50c2ae7ec.webp)
评论