11

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.

4 Answers4

15

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>
ryanafrish7
  • 3,222
  • 5
  • 20
  • 35
7

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.

MartinTeeVarga
  • 10,478
  • 12
  • 61
  • 98
5

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.

tanius
  • 14,003
  • 3
  • 51
  • 63
  • 3
    I'm the author of SingleFile. The purpose of the SVG in SingleFile is to give a size to the image without altering the CSS. So, I confirm you can avoid SVG if setting the width and the height of the image with CSS is not an issue for you. – check_ca May 16 '19 at 17:26
1

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>
HaulinOats
  • 3,628
  • 4
  • 21
  • 24
  • 1
    The application is the difficulty. I could do it in JQuery but I don't want any libraries or separate files. –  Jun 24 '16 at 13:47
  • 1
    Updated my answer @Devil'sAdvocate. Just another approach. – HaulinOats Jun 24 '16 at 13:55
  • 2
    This is what I was expecting, thanks for the update. But I prefer the CSS solution. –  Jun 24 '16 at 13:56
  • I've heard from different sources that using 'content' in CSS is bad since it separates the purposes of CSS, HTML and Javascript. CSS is supposed to be for presentation only, HTML for content, and Javascript for interactivity or DOM manipulation. Honestly, I don't think it really matters in your specific case, and i'm not an expert, just what the general consensus in the programming community is from what I've read: http://www.karlgroves.com/2013/08/26/css-generated-content-is-not-content/ – HaulinOats Jun 24 '16 at 14:04
  • I respect that argument but I was never a fan. Of course I also mix my food on my plate. –  Jun 24 '16 at 14:05