提供多个 base64 图像以供下载

Posted

技术标签:

【中文标题】提供多个 base64 图像以供下载【英文标题】:serve multiple base64 images to download 【发布时间】:2020-07-17 00:14:42 【问题描述】:

我的 django 应用程序有一个带有 base64 编码图像的模型。

我想将选项添加到我的 ListView 以将所有显示的图像下载到用户选择的位置。

我应该创建一个 AJAX 视图,还是 jQuery 可以处理它?

我四处搜索,看到了一些提供单个文件下载的示例。但是如何同时提供所有图片呢?

【问题讨论】:

这能回答你的问题吗,***.com/questions/51027868/… 【参考方案1】:

您可以在带有图像的页面上插入此脚本:

function download(data, file_name) 
  const a = document.createElement("a")
  a.href = data
  const extension = data.match(/^data:(.*?)\/(.*?);/)[2]
  a.setAttribute("download", file_name + "." + extension)
  a.style.display = "none"
  a.addEventListener("click", e => e.stopPropagation()) // not relevant for modern browsers
  document.body.appendChild(a)
  setTimeout(() =>  // setTimeout - not relevant for modern browsers
    a.click()
    document.body.removeChild(a) 
  , 0)

function download_all() 
  [...document.getElementsByClassName("dl")]
    .forEach((img, i) => download(img.src, "img" + i))

document
  .getElementById("dl_all")
  .addEventListener("click", download_all)
<button id="dl_all">download all</button>
<img class="dl" src="">
<img class="dl" src="">

这并不理想:例如 Chrome 将警告用户多次下载,而 FF 将显示多个保存对话框,除非用户将“应用程序”部分中的设置更改为“保存存档”。但我怀疑它可以修复。

当我运行这个 sn-p 时,我的 Chrome 以某种方式阻止了多次下载,但它在 https://jsfiddle.net/w7nqvjgd/ 和我尝试过的其他环境中运行良好。

更新: 或者(感谢 Steven 的想法)如果您对现代浏览器中多次下载的用户体验不满意,您可以在客户端上创建一个 zip 文件。见https://github.com/Stuk/jszip

function download(data) 
  const a = document.createElement("a")
  a.href = "data:application/zip;base64," + data
  a.setAttribute("download", "imgs.zip")
  a.style.display = "none"
  a.addEventListener("click", e => e.stopPropagation()) // not relevant for modern browsers
  document.body.appendChild(a)
  setTimeout(() =>  // setTimeout - not relevant for modern browsers
    a.click()
    document.body.removeChild(a) 
  , 0)

function download_all() 
  var zip = new JSZip();
  [...document.getElementsByClassName("dl")]
    .forEach((img, i) => zip.file("img" + i, img.src.replace(/data:.*?;base64,/, ""), base64: true))
  zip.generateAsync(type: "base64").then(download)

document
  .getElementById("dl_all")
  .addEventListener("click", download_all)

// and an old school version
// for browsers incompatible with `download` attribute
function download_all_oldschool() 
  var zip = new JSZip();
  [...document.getElementsByClassName("dl")]
    .forEach((img, i) => zip.file("img" + i, img.src.replace(/data:.*?;base64,/, ""), base64: true))
  zip.generateAsync(type: "base64").then(data => window.location.href = "data:application/zip;base64," + data
)

document
  .getElementById("dl_all_oldschool")
  .addEventListener("click", download_all_oldschool)
<script src="https://cdn.jsdelivr.net/npm/jszip@3.3.0/dist/jszip.min.js"></script>
<button id="dl_all">download all</button>
<button id="dl_all_oldschool">download all (old school)</button>
<img class="dl" src="">
<img class="dl" src="">

【讨论】:

1) 在存档下载的情况下,用户必须从中提取文件。一个多余的步骤。这也是一个多余的请求。 2)我已经在代码后面的段落中解决了单独下载的问题 但是,是的,人们可以决定档案更适合他的需要。我只是展示了如何在客户端完成它:“或者 jQuery 可以处理它吗?” - 是的,它可以。有一些注意事项 再次更新了答案:)

以上是关于提供多个 base64 图像以供下载的主要内容,如果未能解决你的问题,请参考以下文章

下载图像,序列化为 base-64 字符串,转换为 ImageSource

浏览器/HTML 强制从 src="data:image/jpeg;base64..." 下载图像

如何在delphi中实现对文件进行base64编码

AES Base64 C++加密工具

如何将 Firebase 上传图片的图片下载 URL 转换为 base64

将 base64 上传到 firebase 存储并取回下载 url