обслуживать несколько изображений base64 для загрузки - PullRequest
5 голосов
/ 05 апреля 2020

Приложение My django имеет модель с изображениями в кодировке base64.

Я хотел бы добавить опцию в свой ListView для загрузки всех отображаемых изображений в расположение по выбору пользователя.

Должен ли я создать AJAX представление или jQuery позаботится об этом?

Я погуглил и увидел несколько примеров предоставления одного файла для загрузки. Но как мне обслуживать все изображения одновременно?

1 Ответ

0 голосов
/ 12 апреля 2020

Вы можете вставить этот скрипт на страницу с изображениями:

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 покажет несколько сохранить диалоги, если пользователь не изменит настройки в разделе «Приложения» на «Сохранить файл». Но я сомневаюсь, что это можно исправить.

Почему-то мой Chrome блокирует несколько загрузок, когда я запускаю этот фрагмент, но он отлично работает здесь https://jsfiddle.net/w7nqvjgd/ и в других средах, которые я Попробовал.

Обновление: Или (спасибо Стивену за идею), если вы не удовлетворены опытом нескольких загрузок в современных браузерах, вы можете создать почтовый файл на клиенте. См 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="">
...