Pasting image from clipboard seems to download the image anyway

I’m on Linux, gnome with x11 and when I add images to notes, I often just click on copy image in Chrome, and paste to anki.

However, I have been noticing Anki freezing for short or long time (seconds) after I paste each image which shouldn’t happen if it’s pasted from the clipboard locally.

Moreover, I occasionally got popups of 4xx errors after the freeze (but the image seems to paste successfully anyways. I didn’t bother to inspect the field html when those happened but I suspect Anki falls back to use a remote url, as my later example shows below.)

After a successful paste, I usually get this in my field:

<img alt="国際公務員 | 法律・公務員 | 未来の仕事を探せ! | 未来の仕事 | 学研 ..." src="be28e623c96b6e8f914752a00cc41728.jpg">

an img tag sometimes with an alt field and a local hashed and downloaded image file.

Today, rarely after tens of seconds of freezing, I got an SSL error, but the image successfully appeared anyway:

<img alt="発熱」は何度から?熱が出る原因・熱が出た際の対処法について - 健康 ..." src="https://www.suntory-kenko.com/column2/wp-content/uploads/2023/01/928-1.jpg">

and it seems to be a remote image instead, as I checked my media folder and it didn’t have the image.

I wonder why is an internet request even neccesary in this case? xclip -selection clipboard -t image/png -o > /tmp/test.png on my terminal would just paste the image instantly too.

Additionally, If I would just copy a screenshot to the clipboard using my system screenshot tool though, it pastes as:

<img src="paste-24e3c3884be88f587a08c765c7961bc7c93f573a.jpg">

and I indeed observe no internet request.

2 Likes

This usually depends on what you picked up in your copy buffer, which is mostly determined by how the website is serving up the image. But even when you are pasting the actual image file and not a link, Anki still needs to save the file to your collection media, compare it to your existing media to avoid duplication, encode/de-duplicate the filename, and add the HTML that you see in the field – Media - Anki Manual. So there are a few things going on there.

If the freezing is happening consistently, you should go through the Troubleshooting Checklist – Troubleshooting - Anki Manual . But not on that list is Check Media – Media - Anki Manual – which you should definitely consider doing. You’ve got over 11GB of media in your account, which is built-in drag on the media-attaching process (not to mention syncing and server space). Anything you can do to cut that down will probably help.

To fix the images that were inadvertently added as remote, but intended to be local – you can use the Localize Media add-on.

1 Like

The behavior that OP is describing indeed seems to be a bug. Usually, the copy buffer has both the image and the link and in such cases, Anki should prefer using the image.

I ran into the same issue today. I tried to copy the first image from https://www.nejm.org/doi/abs/10.1056/NEJMicm1801693. When I pasted it into Anki, I got the error “Unexpected response code: 403” and the pasted image was referenced from the web (not locally stored in collection.media).

I am sure that the image is on my clipboard because running magick clipboard:image image.jpg in a new folder saves the image to the disk. However, trying to download the image from the link using wget https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg gives an error:

Resolving www.nejm.org (www.nejm.org)... 104.18.41.121, 172.64.146.135
Connecting to www.nejm.org (www.nejm.org)|104.18.41.121|:443... connected.
HTTP request sent, awaiting response... 403 Forbidden
2025-02-11 18:39:09 ERROR 403: Forbidden.

So, Anki should use the image from the clipboard rather than trying to download it from the link. Anki is supposed to do that (Prefer image pastes over HTML in the editor by abdnh · Pull Request #2856 · ankitects/anki · GitHub) but isn’t doing that for some reason. Oh okay, dae reverted this due to some regressions: Revert "Prefer image pastes over HTML in the editor (#2856)" · ankitects/anki@174a2e7 · GitHub

For the sake of completeness, here’s the error I got on trying to download OP’s image using wget.

Resolving www.suntory-kenko.com (www.suntory-kenko.com)... 2600:140f:5:2080::2da2, 2600:140f:5:2083::2da2, 23.57.251.33
Connecting to www.suntory-kenko.com (www.suntory-kenko.com)|2600:140f:5:2080::2da2|:443... connected.
ERROR: cannot verify www.suntory-kenko.com's certificate, issued by 'CN=Cybertrust Japan SureServer CA G4,O=Cybertrust Japan Co.\\, Ltd.,C=JP':
  Unable to locally verify the issuer's authority.
To connect to www.suntory-kenko.com insecurely, use `--no-check-certificate'.

Debug Info:

  • Anki 25.02 (stable)
  • Windows 11 24H2
  • Chromium: 133.0.6943.54
1 Like

By the way, why is it that Anki (and wget) are unable to fetch the image while Chrome can? Probably, Anki needs to switch to a different library/user-agent/something else? to be able fetch these images without relying on the image in the clipboard.

Regarding wget: javascript - Wget 403 Forbidden, But works in every browser - Stack Overflow

The site you shared is a cloudflared site as well.

2 Likes

Also: if I go to https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg and paste the image from there into Anki, I get a popup saying 403, but the image is still properly inserted:

Would be weird if it works for me but not for you.

I copied the picture from within LibreWolf (a firefox fork) into

Anki 24.11 (87ccd24e)  (ao)
Python 3.9.18 Qt 6.6.2 PyQt 6.6.1
Platform: Linux-6.1.0-0.deb11.21-amd64-x86_64-with-glibc2.31

The image is inserted for me too. The problem is that the card is referencing the image from the web instead of a local copy in the collection.media folder.

1 Like

sorry for not making myself clear. But @vaibhav was on point,

Usually, the copy buffer has both the image and the link and in such cases, Anki should prefer using the image.

As shown in my screenshot, Google Chrome has Copy image and Copy image address in its context menu. The first copies the data (along with the url I think, as Vaibhav said), and the second copies the url only. I consider Anki’s behavior buggy because it tries to download using the url anyway instead of using the image data.

I think my bottleneck was definitely the internet speed and not the deduplication, hashing, etc. As when I copy an image from my local OS screenshot tool and pasting to Anki. There is no freezing whatsoever.

You’ve got over 11GB of media in your account

:grimacing: Yeah I’m a bit self-conscious over that. What’s Anki’s policy on storage space? I’m not trying to abuse and my major deck just has really rich audio and image data.

1 Like

You’re right, same happens for me too.

Some testing

Summary
  1. Navigate to the image on your target site.
  2. Right click → Copy image.
  3. xclip -selection clipboard -o -t TARGETS shows a list of available targets in the clipboard:
The list of tagets in clipboard.
TIMESTAMP
TARGETS
MULTIPLE
SAVE_TARGETS
text/html
text/_moz_htmlinfo
text/_moz_htmlcontext
image/png
image/bmp
image/x-bmp
image/x-MS-bmp
image/x-icon
image/x-ico
image/x-win-bitmap
image/vnd.microsoft.icon
application/ico
image/ico
image/icon
text/ico
image/jpeg
image/tiff
image/webp
audio/x-riff
  1. xclip -selection clipboard -t text/html -o > /tmp/test.txt gives
<meta http-equiv="content-type" content="text/html; charset=utf-8"><img src="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" alt="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" width="848" height="638" class="shrinkToFit fullZoomIn">

whereas xclip -selection clipboard -t image/jpeg -o > /tmp/test.jpeg gives the picture.
6. Building from source an removing the comments in anki/qt/aqt/editor.py (l. 1591 – 1595), I can see the following output when pasting the image from above link:

use clipboard
html=True image=True urls=False txt=False
html <meta http-equiv="content-type" content="text/html; charset=utf-8"><img src="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" alt="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" width="1205" height="907" class="shrinkToFit">
urls []
text 
2025-02-11 15:53:10,881:DEBUG:urllib3.connectionpool: Starting new HTTPS connection (1): www.nejm.org:443
2025-02-11 15:53:11,020:DEBUG:urllib3.connectionpool: https://www.nejm.org:443 "GET /cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg HTTP/11" 403 None
[17467:17519:0211/155313.661525:ERROR:nss_util.cc(357)] After loading Root Certs, loaded==false: NSS error code: -8018

meaning it does seem to find both, html and the image.
8. A normal local screenshot gives this:

use clipboard
html=False image=True urls=False txt=False
html 
urls []
text 
2025-02-11 15:59:27,383:DEBUG:aqt.mediasrv: GET /paste-e7f953805215a3b51b11a0ff206a748c3c6b2e73.jpg

Attempted solution

So I changed the code slightly (qt/aqt/editor.py, line 1600+).

# try various content types in turn
        if mime.hasHtml() and not mime.hasImage():
            html_content = mime.html()[11:] if internal else mime.html()
            return html_content, internal

        # given _processUrls' extra allowed_suffixes kwarg, placate the typechecker
        def process_url(mime: QMimeData, extended: bool = False) -> str | None:
            return self._processUrls(mime, extended)

        # the clipboard stores the image and html containing a link to the remote
        # server. There's no need to fetch the image though if the clipboard
        # already contains it.
        if mime.hasImage() and mime.hasHtml:
            types = (self._processImage, self._processText)

With that change the image from the clipboard (if available) will be used. There is no network connection and anki stores the picture in the media folder, using something like this:

<img src="paste-dec03436e8a74f25918b1854a73ef8dde6f7eb8b.jpg">

Limitations

File names and alt tags are lost.

I have no idea how not remove the tags and file names though. Well, I mean I suppose one could do something like that:

html_content = mime.html()
start = html_content.find("<img")
html_content = mime.html()[start:]

which would give

<img src="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" alt="https://www.nejm.org/cms/10.1056/NEJMicm1801693/asset/849a05b6-eeb6-4f12-9b5e-11be7a704005/assets/images/large/nejmicm1801693_f1.jpg" width="1205" height="907" class="shrinkToFit">

And I guess we could use the same logic to get the alt fields and image name out of the html.

But then we’d need to modify _read_pasted_image() and _addMedia() too. I do not think I’m skilled enough to make these kind of changes. But maybe this helps someone.

1 Like

also I might have made it sound overly serious…
Most of the times the “freeze” is not perceptible, perhaps tens or hundreds of miliseconds when I paste a 400x400 image for example.

It only starts to get mildly inconvenient at night for me when the internet gets worse where I live. Then it can some times lag 1 or 2 seconds I think.

If I am interpreting your results correctly, Anki seems to be recompressing the clipboard image to JPG after your changes. This is not ideal if the original compressed image is also available. (JPG compression is lossy and recompression can even make the new file larger than the original one.)

Probably, that’s one of the reasons Anki currently prefers downloading the image from the server instead of relying on the clipboard image.

regarding the compression though there do be a setting in Prerences > Editing

That still won’t be ideal. If the website had a JPG,

  • enabling this option will cause the image to be pasted as a PNG with same visual quality but much larger size.
  • disabling this option will cause the image to be saved as a recompressed JPG with lower visual quality and similar size (size would actually depend upon the quality of original image and quality used by Anki for compression, which is 80)

Note: I have used the word quality for two different things here.

  • visual quality = how the image looks
  • quality = compression level
1 Like

But anki does this for every picture, doesn’t it?

Recently I found a problem similar to this when developing my add-on. If I copy an image generated by Google Gemini and paste it into Anki the download always fails. (Maybe they have an anti-bot scraping solution.)

Steps to reproduce this issue:

  1. AI-generate the image with Google Gemini (AI)
  2. Copy the generated image.
  3. When pasting the image into Anki, the redirect loops and the download always fails.

It seems to occur when using modules like requests. Such images are very rare so I don’t think any countermeasure is needed for now. (Images generated by ChatGPT can be copied so this problem does not occur.)

Workaround (add-on):

My purpose was to add an image directly to a card from my own AI sidebar (right click, not AnkiWebView), so I downloaded the image directly to the user’s media folder and added it to the card, like this:

  1. Set an action in QWebEngineProfile downloadRequested when a download request is received.
  2. Download directly to the user’s media file using QWebEngineDownloadRequest. (setDownloadDirectory, setDownloadFileName)
  3. Use QWebEngineView’s triggerPageAction to download the image.
  4. When the download is complete, look up the file extension (I bundled a filetype module), rename the file, and add it to the card.
1 Like

It looks like you’re in hands more capable than mine as far as the main question. But just to respond to this –

I wasn’t trying to embarrass you, so if hit a sore spot, I’m sorry! The policy is something along the lines of – use what you need, but don’t use more than you need – Are there limits on file sizes on AnkiWeb? - Anki FAQs . If you already make a habit of cleaning up your unused media, and being a good steward of this free mutual resource, then there’s probably nothing you need to do differently.

2 Likes

This topic was automatically closed 30 days after the last reply. New replies are no longer allowed.