How can I use a Base64 encoded image string?

I would like to embed an image that I have as a Base64 encoded string directly into my Typst document. Normally, I can embed an image using the following syntax, but it only allows specific format only not string.

image("path/to/image.png", width: 100%)

Can you clarify: you have a Base64 encoded image (of a format that is currently supported by Typst), and you want to insert it in a Typst document as a regular image, correct?

In case this is correct, by using based package (love the name) you can easily achieve this:

#import "@preview/based:0.1.0": base64
#set page(width: auto, height: auto, margin: 0pt)

// `base64 -w 0 smile.png` (10x10):
#let base64-encoded-image = "iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAACXBIWXMAAC4jAAAuIwF4pT92AAAASUlEQVQYGa3OWwoAIAgEQNf739k0EjVCfxKCHtMqiEh0jcWjOKBAkQjPe7MFdun/IbRdDNb05nvolzWzEx0DdozK96W1PzjNHxcgphkBs9CoHwAAAABJRU5ErkJggg=="
#let raw-image = base64.decode(base64-encoded-image)

#image.decode(raw-image, format: "png")

image

image.decode

Yes, indeed. I have a Base64 encoded image that I want to include in a Typst document as a standard image.
This code snippet successfully solves the issue
Thanks!

1 Like

You shouldn’t use based if you have a document with a lot of pictures. I am testing with a document with 35 pictures in base64 format. Using based it takes 60 seconds to generate the pdf and just using

image.decode("<svg xmlns=\"http://www.w3.org/2000/svg\" xmlns:xlink=\"http://www.w3.org/1999/xlink\"><image xlink:href=\"data:image/jpg;base64," + foto.Foto + "\" /></svg>")

it takes less than 1 second. I provided the base64 string through the variable foto.Foto because it’s part of a for each loop.

1 Like

Did you use version 0.2.0 of based when it took 60 seconds? I know that 0.1.0 was incredibly slow, but I switched to a wasm plugin for 0.2.0, which improved performance by a lot.

1 Like

You shouldn’t embed (big/many) images into a Typst document by encoding it in a different format. That is clearly not how it is meant to be used, and obviously any data manipulation such as decoding Base64 to bytes will take some CPU cycles to finish. The SVG hack sounds interesting, though it might have some downsides, but I’m not an SVG nerd.

Hold on, I thought that WASM code should be always slower than the native Typst code. What’s the catch?

Wow, I can no longer edit my own message… Anyway, I wanted to make an update.

First is, of course, update the based package version, but the second point is that now image.decode is deprecated, and you should pass bytes directly, i.e.:

#image(base64.decode(base64-encoded-image))

I’ve also noticed that you don’t have to specify the format anymore, I assume it will auto-detect any format correctly, just like it does when you provide a path to the image.

Complete solution
#import "@preview/based:0.2.0": base64
#set page(width: auto, height: auto, margin: 0pt)

// `base64 -w 0 smile.png` (10x10):
#let base64-encoded-image = "iVBORw0KGgoAAAANSUhEUgAAAAoAAAAKCAYAAACNMs+9AAAACXBIWXMAAC4jAAAuIwF4pT92AAAASUlEQVQYGa3OWwoAIAgEQNf739k0EjVCfxKCHtMqiEh0jcWjOKBAkQjPe7MFdun/IbRdDNb05nvolzWzEx0DdozK96W1PzjNHxcgphkBs9CoHwAAAABJRU5ErkJggg=="
#let raw-image = base64.decode(base64-encoded-image)

#image(raw-image)