Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

How to reuse base64 image repeatedly in HTML file

I have an HTML file that will be portable to distribute to clients. It has an index of documents. I want a PDF or Word icon for each document but I don't want to repeat the base64 string 30 times if I can avoid it. How should I do this? Javascript is okay, but my goals are simplicity and portability.

like image 627
THE JOATMON Avatar asked Jun 24 '16 13:06

THE JOATMON


People also ask

Does browser cache Base64 images?

Browsers can cache downloaded files. If the Base64 string is in a text (or JSON) file then it can be cached.

Does Base64 encoding reduce image quality?

With the introduction of multiplexing that arrived with HTTP/2, web browsers have become incredibly efficient in delivering hundreds of files through a single connection. This works around most limits that the Base64 encoding solved and in fact means Base64 now does more bad than good.

Why Base64 is needed to transfer images?

Base64 is only useful for very small images. This is because when it comes to larger images, the encoded size of a picture in bytes will end up being much larger than JPEGs or PNG files.

How long is Base64 image?

It will however be larger than the source file. A ballpark figure is around 270 000 characters.


4 Answers

You can use a CSS variable.

In the <head> section of the page, name and define the variable and a class to use it:

:root {
    --img-123: url(data:null;base64,iVBORw………); /* your image data here */
}

.img-123 {
    background-blend-mode: normal;
    background-clip: content-box;
    background-position: 50% 50%;
    background-color: rgba(0,0,0,0);
    background-image:var(--img-123);
    background-size: 100% 100%;
    background-origin: content-box;
    background-repeat: no-repeat;
}

In the <body> section of the page, use the image repeatedly via element style section or CSS classes. The SVG data creates a transparent image through which to see the content.

<img 
  class="img-123" 
  src="data:image/svg+xml,&lt;svg xmlns=&quot;http://www.w3.org/2000/svg&quot; width=&quot;16&quot; height=&quot;16&quot;&gt;&lt;rect fill-opacity=&quot;0&quot;/&gt;&lt;/svg&gt;"
>

I observed this technique in the output of the open source Firefox / Chrome extension SingleFile. There might be ways to simplify the CSS and to avoid using SVG.

like image 28
tanius Avatar answered Oct 06 '22 19:10

tanius


Try using CSS class

img.word-icon {
  content: url();
  width: 32px;  
  height: 32px;
}
<html>
  <body>  
    <img class="word-icon">foo
    <img class="word-icon">bar
    <img class="word-icon">
  </body>
</html>
like image 199
ryanafrish7 Avatar answered Oct 06 '22 18:10

ryanafrish7


The other answer using CSS gives the right idea, but it is not compatible with all browsers. The content property of the CSS is only allowed in the pseudo-elements ::before and ::after. The img tag with a class with a content property is empty. The browsers have no obligation to display it. Chrome displays it nevertheless (incorrectly).

To make this work for all browsers, use for example the ::before on an empty span:

span.word-icon::before {
  content: url();
}
<html>
  <body>  
    <p><span class="word-icon"></span>Paragraph with icon.</p>
  </body>
</html>

You can also attach it directly to p::before.

like image 22
MartinTeeVarga Avatar answered Oct 06 '22 19:10

MartinTeeVarga


create a script to insert it into the images. I purposely put the script code inline since you're sending a single HTML file:

<img alt="pdf"/>
<img alt="pdf"/>
<img alt="pdf"/>
<img alt="pdf"/>
<img alt="pdf"/>
<img alt="pdf"/>
<script>
  var images = document.getElementsByTagName('img');
for (var i = 0; i < images.length; i++){
  images[i].src = "";
}
</script>
like image 36
HaulinOats Avatar answered Oct 06 '22 17:10

HaulinOats