50

Seemingly, you can't (yet) programmatically copy an image to the clipboard from a JavaScript web app?

I have tried to copy a text in clipboard , and it's worked.

Now I would like to copy an image and after I press ctrl+v to paste into Word or Excel or Paint.

$(function() { 
    $("#btnSave").click(function() { 
        html2canvas($("#container1"), {
            onrendered: function(canvas) {
                theCanvas = canvas;

                document.body.appendChild(canvas);
                Canvas2Image.saveAsPNG(canvas); 
                $("#img-out").append(canvas);
            }
        });
    });
}); 
Uwe Keim
  • 39,551
  • 56
  • 175
  • 291
rachedbchir
  • 511
  • 1
  • 4
  • 4
  • The following might be useful as a relatively reliable/portable (though with added interaction cost) solution: spawn a new tab containing the contents of `theCanvas.toDataURL()`. Then the user can right-click that and "Copy Image" to clipboard. – Evgeni Sergeev Feb 02 '20 at 03:21
  • Earlier duplicate question: [Copy Image to Clipboard from Browser in Javascript?](https://stackoverflow.com/questions/2321606/copy-image-to-clipboard-from-browser-in-javascript) – Jon Schneider Jul 28 '22 at 16:24

11 Answers11

33

For those still looking, the ClipboardAPI now works with png images.

try {
    navigator.clipboard.write([
        new ClipboardItem({
            'image/png': pngImageBlob
        })
    ]);
} catch (error) {
    console.error(error);
}
withybetwixtye
  • 330
  • 3
  • 5
31

This worked across all browsers (as of 2016). I have uploaded on GitHub as well: https://github.com/owaisafaq/copier-js

//Cross-browser function to select content
function SelectText(element) {
  var doc = document;
  if (doc.body.createTextRange) {
    var range = document.body.createTextRange();
    range.moveToElementText(element);
    range.select();
  } else if (window.getSelection) {
    var selection = window.getSelection();
    var range = document.createRange();
    range.selectNodeContents(element);
    selection.removeAllRanges();
    selection.addRange(range);
  }
}
$(".copyable").click(function(e) {
  //Make the container Div contenteditable
  $(this).attr("contenteditable", true);
  //Select the image
  SelectText($(this).get(0));
  //Execute copy Command
  //Note: This will ONLY work directly inside a click listenner
  document.execCommand('copy');
  //Unselect the content
  window.getSelection().removeAllRanges();
  //Make the container Div uneditable again
  $(this).removeAttr("contenteditable");
  //Success!!
  alert("image copied!");
});
<script src="https://cdnjs.cloudflare.com/ajax/libs/jquery/3.3.1/jquery.min.js"></script>

<div class="copyable">
  <img src="https://via.placeholder.com/80" alt="Copy Image to Clipboard via Javascript." />
</div>

<div class="copyable">
  <img src="https://via.placeholder.com/80" alt="Copy Image to Clipboard via Javascript." />
</div>
isherwood
  • 58,414
  • 16
  • 114
  • 157
owaisafaq
  • 523
  • 4
  • 7
  • 9
    This doesn't add `DataTransferItem` of kind `file` to the `ClipboardEvent` but pure "Copy image" from web browser image context menu does. Basically this code passes images as HTML content instead of pure image. – czerny Sep 04 '17 at 13:17
  • 3
    Doesn't work in Firefox 56, Chrome 61 and Opera 48. Which browser did you use? – Gerold Meisinger Oct 12 '17 at 13:23
  • This is really great and very smart idea I tried and it worked for sure.. on chrome 64 and firefox 59 beta.. awesome thinking sir. it does copy the images and you can paste them wherever you want – Bashar Ali Labadi Feb 28 '18 at 03:19
  • 1
    However it worth noting that this has it's limitation as czerny said above.. some applications can understand the copied img html others don't. – Bashar Ali Labadi Mar 01 '18 at 05:34
  • 3
    Doesn't work in either Firefox Quantum or Chrome (2018) – kevinrodriguez-io Jul 06 '18 at 15:14
  • See @BasharAliLabadi's comment above. – owaisafaq Jul 09 '18 at 05:52
  • Works when I copy from Chrome into Keynote (and presumably other desktop apps), but not when I copy from one Chrome tab into another (i.e., attempting to paste into a browser-based tool like Figma). Does not work in Firefox. Nonetheless, the only working example I was able to find. – Ryan Burney Jan 24 '19 at 23:28
  • 2
    This has become outdated and no longer works, here is the exact copy in codepen - https://codepen.io/mikefreudiger/pen/xMXeoX?editors=1010 – Bilbo Feb 06 '19 at 14:36
8

So, i created the perfect solution with 1 liner-kinda solution to convert something with html2canvas to a canvas and then produce the image of it and then save it to clipboard as png. For example,

HTML:
<div id="copyToImage">Hello World!</div>

JavaScript:

$("#copyToImage").click(function() {
    html2canvas(document.querySelector("#copyToImage")).then(canvas => canvas.toBlob(blob => navigator.clipboard.write([new ClipboardItem({'image/png': blob})])));
});
Konstantinos
  • 943
  • 1
  • 9
  • 20
6

You are right. There is no support to copy image data into clipboard in chrome yet. https://bugs.chromium.org/p/chromium/issues/detail?id=150835. Looks like it has been open for about 4 years now.

There is a clipboard API spec that is coming up though https://w3c.github.io/clipboard-apis/

Luca
  • 9,259
  • 5
  • 46
  • 59
notsiddhartha
  • 356
  • 3
  • 6
4

This can be achieved by using the Clipboard API, which does require that your server is secured. The basic idea is to convert your image to a blob before writing it to the clipboard as an image.

const img = document.querySelector("#b64-img");
const copyBtn = document.querySelector("#copy");

copyBtn.onclick = e => {
    const canvas = document.createElement("canvas");
    canvas.width = img.width;
    canvas.height = img.height;
    canvas.getContext("2d").drawImage(img, 0, 0, img.width, img.height);
    canvas.toBlob((blob) => {
      navigator.clipboard.write([
          new ClipboardItem({ "image/png": blob })
      ]);
    }, "image/png");
};
<img id="b64-img" src="">
<br>
<button id="copy">Copy Above Image</button>

This will allow you to paste your image to Photoshop and other image-applications.

Vektor
  • 697
  • 5
  • 14
  • 1
    Doesn't work in Firefox, unfortunately (for most users): ClipboardItem is disabled in Firefox by default. Reference: https://developer.mozilla.org/en-US/docs/Web/API/ClipboardItem#browser_compatibility – Jon Schneider Jul 28 '22 at 16:15
2

Well, this is my first post in here with an answer i guess :)

Actually i'm currently using cefsharp web browser component one of my project, cefsharp run on chrome based browser and i want to copy img element one of webpage

With cefsharp you can manipulate browser only javascript, so i think we can handle it with using canvas element.

/*
  'cause of lorempixel timeout, i used img onload function.
*/

function copyImage() {
  var imgCap = document.getElementById('imgCap');
  var imgCanvas = document.createElement('canvas');

  imgCanvas.id = 'imgCanvas';
  imgCanvas.height = 40;
  imgCanvas.width = 120;

  document.body.appendChild(imgCanvas);
  var originalContext = imgCanvas.getContext('2d');
  originalContext.drawImage(imgCap, 0, 0);

  //return imgCanvas.toDataURL();
}

//document.onload = copyImage();
<img id="imgCap" src="http://lorempixel.com/120/40" onload="copyImage();"/>

with return imgCanvas.toDataURL(); you can get base64 encoded value and use wherever you want.

this is my cefsharp code, it's working.

        string copyImageOtClipboardScript = "(function(){ try{var imgCap = document.getElementById('imgCap'); var imgCanvas = document.createElement('canvas'); imgCanvas.id = 'imgCanvas'; imgCanvas.height = 40; imgCanvas.width = 120; document.body.appendChild(imgCanvas); var originalContext = imgCanvas.getContext('2d'); originalContext.drawImage(imgCap, 0, 0); return imgCanvas.toDataURL();  }catch(e){ alert(e); } })();";

        var task = chromeBrowser.EvaluateScriptAsync(copyImageOtClipboardScript).ContinueWith(x =>
        {
            var resp = x.Result;

            if (resp.Success)
            {
                this.Invoke((MethodInvoker)delegate
                {
                    Bitmap bmp = null;
                    string captchaResult = "", captchaBase64;

                    var bytes = Convert.FromBase64String(resp.Result.ToString().Replace("data:image/png;base64,", ""));
                    using (var imageFile = new FileStream("temp_captcha.png", FileMode.Create))
                    {
                        imageFile.Write(bytes, 0, bytes.Length);
                        imageFile.Flush();
                    }

                });
            }
        });
SysEng
  • 21
  • 3
  • 1
    I tried the mentioned JavaScript code it's not working. – Omar Mar 20 '19 at 15:50
  • @Omar, what did you get ? it is working in code snippet. – SysEng Mar 26 '19 at 05:51
  • 1
    I'm trying the code within the normal standalone Chrome browser (not within a .NET app using CefSharp) .. it seems that your code is intended only to work along with CefSharp within .NET apps. I'm looking for a way to work within the normal daily usage browsers (Chrome, Firefox, …) – Omar Mar 29 '19 at 09:33
0

Check out this guide to copying and pasting with JavaScript: https://www.lucidchart.com/techblog/2014/12/02/definitive-guide-copying-pasting-javascript/

According to this, Chrome, Safari, and Firefox all support copying images along with plain text, while IE only allows copying text. The page linked above describes how this service uses an extension to add this functionality to a context menu, but it appears that several browsers support programmatic copying of images.

OverlappingElvis
  • 621
  • 6
  • 20
  • 1
    There is no support to copy image data into clipboard in chrome yet. https://bugs.chromium.org/p/chromium/issues/detail?id=150835. Looks like it has been open for about 4 years now. – notsiddhartha Sep 13 '16 at 06:31
  • Lucid Chart link wasn't working for me, but I found the article here: https://dzone.com/articles/definitive-guide-copying-and – Dylan Watson Oct 09 '19 at 01:47
0
const copyImageToClipboard = async () => {
      try {
      await navigator.clipboard.write([
        new ClipboardItem({
          'image/png': await fetch(img).then((r) => r.blob()),
        }),
      ]);
    } catch (error) {
      console.error(error);
    }
  };

You might run into CORS errors testing on localhost. But you can install a CORS extension to get rid of that issue :)

Ruben Verster
  • 301
  • 1
  • 8
-1

var image = new Image();

        image.src = url;

        copyImageToClipboard(image.src);

install it from https://www.npmjs.com/package/copy-image-clipboard

Rakesh Jat
  • 11
  • 2
  • 1
    A good answer will always include an explanation why this would solve the issue, so that the OP and any future readers can learn from it. – Tyler2P Jan 22 '22 at 09:19
  • `Uncaught ReferenceError: copyImageToClipboard is not defined` ... – Sarout Jan 26 '22 at 11:08
-1

Here is the best answer i found for you, very simple and its working 100% exactly what you need.

await navigator.clipboard.writeText(code);

Check out this link: enter link description here

Rizwan
  • 122
  • 1
  • 7
-6

You cannot copy to clip board with Javascript for security reasons, a work around can be found in a discussion here. Involves flash.Click button copy to clipboard using jQuery

Community
  • 1
  • 1
Casey ScriptFu Pharr
  • 1,672
  • 1
  • 16
  • 36