web-dev-qa-db-fra.com

Forcer le navigateur à télécharger des fichiers image par clic

J'ai besoin du navigateur pour télécharger les fichiers image, comme en cliquant sur une feuille Excel.

Y a-t-il un moyen de faire cela en utilisant uniquement la programmation côté client?

<html xmlns="http://www.w3.org/1999/xhtml">
    <head>
        <title></title>
        <script type="text/javascript" src="Scripts/jquery-1.10.2.js">
        $(document).ready(function () {
            $("*").click(function () {
                $("p").hide();
            });
        });
        </script>
    </head>

    <script type="text/javascript">
        document.onclick = function (e) {
            e = e || window.event;
            var element = e.target || e.srcElement;
            if (element.innerHTML == "Image") {
                //someFunction(element.href);
                var name = element.nameProp;
                var address = element.href;
                saveImageAs1(element.nameProp, element.href);
                return false; // Prevent default action and stop event propagation
            }
            else
                return true;
        };

        function saveImageAs1(name, adress) {
            if (confirm('you wanna save this image?')) {
                window.win = open(adress);
                //response.redirect("~/testpage.html");
                setTimeout('win.document.execCommand("SaveAs")', 100);
                setTimeout('win.close()', 500);
            }
        }
    </script>

    <body>
        <form id="form1" runat="server">
            <div>
                <p>
                    <a href="http://localhost:55298/SaveImage/demo/Sample2.xlsx" target="_blank">Excel</a><br />
                    <a href="http://localhost:55298/SaveImage/demo/abc.jpg" id="abc">Image</a>
                </p>
            </div>
        </form>
    </body>
</html>

Comment cela devrait-il fonctionner en cas de téléchargement d'une feuille Excel (que font les navigateurs)?

82
Amit

En utilisant HTML5, vous pouvez ajouter l'attribut "télécharger" à vos liens.

<a href="/path/to/image.png" download>

Les navigateurs compatibles inviteront ensuite à télécharger l'image avec le même nom de fichier (dans cet exemple, image.png).

Si vous spécifiez une valeur pour cet attribut, cela deviendra le nouveau nom de fichier:

<a href="/path/to/image.png" download="AwesomeImage.png">

UPDATE: À partir du printemps 2018, ce n'est plus possible pour l'origine croisée hrefs . Donc, si vous voulez créer <a href="https://i.imgur.com/IskAzqA.jpg" download> sur un domaine autre que imgur.com, cela ne fonctionnera pas comme prévu. Annonce de désapprobation et de suppression de Chrome

145

J'ai réussi à le faire fonctionner dans Chrome et dans Firefox également en ajoutant un lien vers le document.

var link = document.createElement('a');
link.href = 'images.jpg';
link.download = 'Download.jpg';
document.body.appendChild(link);
link.click();
document.body.removeChild(link);
84
DrowsySaturn

Leeroy et Richard Parnaby-King:

MISE À JOUR: À compter du printemps 2018, cela n'est plus possible pour les hrefs d'origine croisée. Donc, si vous voulez créer sur un domaine autre que imgur.com, cela ne fonctionnera pas comme prévu. Annonce de désapprobation et de suppression de Chrome

function forceDownload(url, fileName){
    var xhr = new XMLHttpRequest();
    xhr.open("GET", url, true);
    xhr.responseType = "blob";
    xhr.onload = function(){
        var urlCreator = window.URL || window.webkitURL;
        var imageUrl = urlCreator.createObjectURL(this.response);
        var tag = document.createElement('a');
        tag.href = imageUrl;
        tag.download = fileName;
        document.body.appendChild(tag);
        tag.click();
        document.body.removeChild(tag);
    }
    xhr.send();
}
23
iXs

Mise à jour du printemps 2018

<a href="/path/to/image.jpg" download="FileName.jpg">

Bien que cela soit toujours pris en charge, à partir de février 2018, chrome a désactivé cette fonctionnalité pour le téléchargement inter-origines, ce qui signifie que cela ne fonctionnera que si le fichier se trouve sur le même nom de domaine.

J'ai trouvé une solution de contournement pour télécharger des images entre domaines après la nouvelle mise à jour de Chrome, qui désactivait le téléchargement entre domaines. Vous pouvez le modifier en une fonction adaptée à vos besoins. Vous pourrez peut-être obtenir l'image de type mime (jpeg, png, gif, etc.) avec quelques recherches supplémentaires si vous en aviez besoin. Il peut également y avoir un moyen de faire quelque chose de similaire avec des vidéos. J'espère que ça aide quelqu'un!

Leeroy et Richard Parnaby-King:

MISE À JOUR: À compter du printemps 2018, cela n'est plus possible pour les hrefs d'origine croisée. Donc, si vous voulez créer sur un domaine autre que imgur.com, cela ne fonctionnera pas comme prévu. Annonce de désapprobation et de suppression de Chrome

var image = new Image();
image.crossOrigin = "anonymous";
image.src = "https://is3-ssl.mzstatic.com/image/thumb/Music62/v4/4b/f6/a2/4bf6a267-5a59-be4f-6947-d803849c6a7d/source/200x200bb.jpg";
// get file name - you might need to modify this if your image url doesn't contain a file extension otherwise you can set the file name manually
var fileName = image.src.split(/(\\|\/)/g).pop();
image.onload = function () {
    var canvas = document.createElement('canvas');
    canvas.width = this.naturalWidth; // or 'width' if you want a special/scaled size
    canvas.height = this.naturalHeight; // or 'height' if you want a special/scaled size
    canvas.getContext('2d').drawImage(this, 0, 0);
    var blob;
    // ... get as Data URI
    if (image.src.indexOf(".jpg") > -1) {
    blob = canvas.toDataURL("image/jpeg");
    } else if (image.src.indexOf(".png") > -1) {
    blob = canvas.toDataURL("image/png");
    } else if (image.src.indexOf(".gif") > -1) {
    blob = canvas.toDataURL("image/gif");
    } else {
    blob = canvas.toDataURL("image/png");
    }
    $("body").html("<b>Click image to download.</b><br><a download='" + fileName + "' href='" + blob + "'><img src='" + blob + "'/></a>");
};
<script src="https://ajax.googleapis.com/ajax/libs/jquery/2.1.1/jquery.min.js"></script>
13
Riley Bell
var pom = document.createElement('a');
pom.setAttribute('href', 'data:application/octet-stream,' + encodeURIComponent(text));
pom.setAttribute('download', filename);
pom.style.display = 'none';
document.body.appendChild(pom);
pom.click();
document.body.removeChild(pom);     
11
Sadi

Ceci est une solution générale à votre problème. Mais il y a une partie très importante pour laquelle l'extension de fichier doit correspondre à votre encodage. Et bien sûr, ce paramètre de contenu de la fonction downlowadImage doit être la chaîne encodée en base64 de votre image.

const clearUrl = url => url.replace(/^data:image\/\w+;base64,/, '');

const downloadImage = (name, content, type) => {
  var link = document.createElement('a');
  link.style = 'position: fixed; left -10000px;';
  link.href = `data:application/octet-stream;base64,${encodeURIComponent(content)}`;
  link.download = /\.\w+/.test(name) ? name : `${name}.${type}`;

  document.body.appendChild(link);
  link.click();
  document.body.removeChild(link);
}

['png', 'jpg', 'gif'].forEach(type => {
  var download = document.querySelector(`#${type}`);
  download.addEventListener('click', function() {
    var img = document.querySelector('#img');

    downloadImage('myImage', clearUrl(img.src), type);
  });
});
a gif image: <image id="img" src="" />


<button id="png">Download PNG</button>
<button id="jpg">Download JPG</button>
<button id="gif">Download GIF</button>
6
hakobpogh

Une approche plus moderne utilisant Promise et async/wait:

toDataURL(url) {
    return fetch(url).then((response) => {
            return response.blob();
        }).then(blob => {
            return URL.createObjectURL(blob);
        });
}

ensuite

async download() {
        const a = document.createElement("a");
        a.href = await toDataURL("https://cdn1.iconfinder.com/data/icons/ninja-things-1/1772/ninja-simple-512.png");
        a.download = "myImage.png";
        document.body.appendChild(a);
        a.click();
        document.body.removeChild(a);
}

Trouvez la documentation ici: https://developer.mozilla.org/en-US/docs/Web/API/Fetch_API/Using_Fetch

2
Curse

Essaye ça:

<a class="button" href="http://www.glamquotes.com/wp-content/uploads/2011/11/smile.jpg" download="smile.jpg">Download image</a>
1
Muhammad Awais

Vous pouvez directement télécharger ce fichier en utilisant balise d'ancrage sans beaucoup de code.
Copiez l'extrait puis collez-le dans votre éditeur de texte et essayez-le ...!

<html>
<head>
</head>
<body>
   <div>
     <img src="https://upload.wikimedia.org/wikipedia/commons/1/1f/SMirC-thumbsup.svg" width="200" height="200">
      <a href="#" download="https://upload.wikimedia.org/wikipedia/commons/1/1f/SMirC-thumbsup.svg"> Download Image </a>
   </div>
</body>
</html>
0
vinodh

Qu'en est-il de l'utilisation de la fonction .click () et de la balise?

(Version compressée)

<a id="downloadtag" href="examplefolder/testfile.txt" hidden download></a>

<button onclick="document.getElementById('downloadtag').click()">Download</button>

Maintenant, vous pouvez le déclencher avec js. Il ne s'ouvre pas non plus, à l'instar d'autres exemples, de fichiers image et texte!

(version js-function)

function download(){
document.getElementById('downloadtag').click();
}
<!-- HTML -->
<button onclick="download()">Download</button>
<a id="downloadtag" href="coolimages/awsome.png" hidden download></a>
0
Poisonous Whisper