2

Can you combine these 2 images into one external file on this JS fiddle and link to them as images?

Something like

<img src="base64.html#img1" />
<img src="base64.html#img2" />

I did see this answer, but it provides no examples
Can multiple base64 documents be stored in a single file?


What about using a dot in the filename and somehow making the file think that it's a different filename, like base64.img1.html and somehow using a file-header to split them so its one file but different parts.


Okay, new idea .. what if there was a way to specify the charStart and charEnd? Imagine there were 2 base64 encoded files and then you call one like

base64.0-3214.html // for one file (like font.woff)
base64.3215-5673.html // for another file (like demo.jpg)
Community
  • 1
  • 1
Kirk Strobeck
  • 17,984
  • 20
  • 75
  • 114
  • I bet you probalbly can put multiple images in a base64 file but you won't be able to navigate to them using hashtags. It seems to me you're looking for a service of some sort that returns a different stream depending on a hashtag or another variable. – Marvin Brouwer Feb 07 '14 at 11:11
  • Sure .. I like the answers so far, but I still believe there's gotta be something out of the box on this one. Maybe a header that makes the browser think its different files? And you can add a filename as a separator or something .. – Kirk Strobeck Feb 07 '14 at 17:34
  • Bottom line is this is a great optimization when you only need 1 cacheable http request for a site. – Kirk Strobeck Feb 07 '14 at 17:35
  • 1
    There used to be a spec prepared by the people from Mozilla for Resource packages for HTML. AFAIK never implemented, probably discontinued. These would be ZIP files with manifest describing caching. Browser would load the package linked by `link` element and pull resources from there. Old browser would not know about it and would still send request for each resource separately. http://stackoverflow.com/q/4165098/2157640 I guess that the motivation to ask this question was saving multiple requests overhead, too. – Palec Feb 08 '14 at 23:55
  • 1
    Bundling more images in one file can be done with SVG images. http://stackoverflow.com/q/14630035/2157640 – Palec Feb 09 '14 at 00:06
  • Also you could be looking for [sprites](http://en.wikipedia.org/wiki/Sprite_%28computer_graphics%29). With CSS it is possible to use them in HTML. – Palec Feb 09 '14 at 00:08
  • Will the images be always one next to the other? If you only want to have parts of one image able to serve as links, use [image map](http://www.w3.org/TR/html5/embedded-content-0.html#image-map) in HTML. – Palec Feb 09 '14 at 00:13
  • Not just images, any filetypes – Kirk Strobeck Feb 09 '14 at 03:14
  • Thanks all! There were many great answers here, was not an easy choice. Still with there was a non-js approach, but I'll keep dreaming :P – Kirk Strobeck Feb 13 '14 at 15:32

7 Answers7

3

Data-uris, which is the key here, are strings more often than not encoded as base-64.

As they are strings you can concatenate them as any other string. However, you cannot set a concatenated (data-uri) string directly as image source.

When concatenating you need to use a separator character so you can split them at arrival before setting them as image sources.

You could have the following strings as base-64 encoded data-uris:

var img1 = 'data:...',
    img2 = 'data:...';

A concatenated version could look like this with for example $ as a separator char:

var conc = img1 + '$' + img2;

This string can now be sent over the net containing the two data-uris combined.

To split them again at arrival from server:

var conc = getDataString();   /// some function to get the string from server
var images = conc.split('$'); /// split the string on the $ char
var img1 = images[0];         /// first index contains first string etc.
var img2 = images[1];

/// set the image sources
document.getElementById('image1').src = img1;
document.getElementById('image2').src = img2;

This of course will require error checking and validation and so forth, but for the principle this is one way of doing this.

Note that you cannot automatically split them simply by referencing them as a link. The server could of course parse the link and provide the single data-uri needed - just as what would be typical to do. But if you want, as I understood your question, to get a single string from server representing two (or more) images then you need to manually split them at client side.

Note 2: base-64 encoded images are 33% larger than an original binary image + the data-uri header needed for the browser to be able to use it as source. The question is probably if there is any gain doing it this way or if the total cost of an extra request to server retrieving binary data would be less in total due to the base64 overhead. Just my 2 cents..

Hope this helps (and that I didn't misunderstood your question entirely).

3

It is possible to put all resources (JavaScript, CSS, fonts, images) together in one big (cacheable) JavaScript file. The file will look messy (because of some lengthy CSS and Base64 strings), but don't let that put you off; you can use code generation to compile the JavaScript file from its separate components (see below).

Demo: http://jsfiddle.net/PgdXd/

CSS

Put your entire stylesheet in a single string literal (do not forget to escape quotes and line breaks) and embed that inside a JavaScript statement that dynamically creates a style element.

$('head').append("<style> ......... </style>");

Note: I am using jQuery here, but plain JavaScript will do just as well.

The statement above may have no effect in older web browser versions. For alternatives, see: How to dynamically create CSS class in JavaScript and apply?

Fonts

Embed your Base64-encoded font files inside your stylesheet, as explained here: http://sosweetcreative.com/2613/font-face-and-base64-data-uri

@font-face {
    font-family: 'latoregular';
    src: url(data:application/x-font-woff;charset=utf-8;base64,........) format('woff');
    font-weight: normal;
    font-style: normal;
}

As shown earlier, this should be embedded within the JavaScript statement that holds your stylesheet:

$('head').append("<style> @font-face { font-family:'latoregular'; src: url(data:application/x-font-woff;charset=utf-8;base64,........) format('woff'); font-weight:normal; font-style:normal; } .... </style>");

Images

There are two flavors. Background images can be specified in the stylesheet, just like we did with fonts:

.logo1 { background-image: url(data:image/png;base64,.......); }

The second flavor are regular images. In the HTML file, use img tags with an empty src attribute.

<img alt="" src="" class="icon1" />

Use javascript to populate the src attribute. The entire Base64 string is a string literal embedded inside the JavaScript statement.

$('.icon1').attr('src', 'data:image/png;base64,.......');
$('.icon2').attr('src', 'data:image/png;base64,.......');

The CSS class determines which data belongs in which img tag(s). It is alright for a certain class to be unused in one or more HTML pages that will be including this JavaScript file. The Base64 string just gets passed around a bit; the same negligible overhead is involved in other JavaScript-based approaches.

Code generation

Here's a simple example using bash scripts. But feel free to use Perl or T4 or whatever language or tool you like instead.

gendatauri.sh: outputs a data URI; parameters are filename and (optionally) MIME type.

#!/bin/bash
echo "data:${2:-$(file -bi $1)};base64,$(base64 -w0 $1)"

genimgsrc.sh: generates the jQuery statement to populate the src attribute of <img> elements; parameter is filename.

#!/bin/bash
filename=$1
basename=${filename##*/}
classname=${basename%.*}
echo "\$('.$classname').attr('src','$(./gendatauri.sh $filename)');"

gencss.sh: your stylesheet; use $(./gendatauri.sh FILENAME [MIMETYPE]) to inject file data. Example:

#!/bin/bash
cat << EOF
@font-face {
    font-family: 'latoregular';
    src: url($(./gendatauri.sh latoregular.woff "application/x-font-woff;charset=utf-8")) format('woff');
    font-weight: normal;
    font-style: normal;
}
.logo1 {
    background-image: url($(./gendatauri.sh icon1.png));
}
EOF

genjs.sh: combines all the different components into a single JavaScript file. Example:

#!/bin/bash

# CSS, including fonts and background images
(
    echo '$("head").append("<style>\n'
    ./gencss.sh
    echo '\n</style>");'
) | tr '\n' ' '
echo

# images
./genimgsrc.sh icon1.png
./genimgsrc.sh icon2.png

# JavaScript
cat YourOwnFunctions.js

Run it and redirect its output to generate the final JavaScript file. Make sure the .png, .woff and other resource files are present when you do.

./genjs.sh > GeneratedJavaScriptFile.js

Included the JavaScript file in each HTML file:

<script src="GeneratedJavaScriptFile.js"></script>

Just for clarity: on the web server, you only need GeneratedJavaScriptFile.js and the .html files; the original resource files do not need to be present there.

Community
  • 1
  • 1
Ruud Helderman
  • 10,563
  • 1
  • 26
  • 45
  • Still wishing there was a non-js solution, but thank you much! great answer. – Kirk Strobeck Feb 13 '14 at 15:31
  • @KirkStrobeck: The biggest reduction is in embedding the Base64-encoded resources (fonts and background images) in CSS. If you can live with one more HTTP request per page, and you don't mind using sprites, then you can do without the JavaScript hacks. – Ruud Helderman Feb 13 '14 at 16:16
1

It is not possible in pure HTML within the data URI specification. (http://en.wikipedia.org/wiki/Data_URI_scheme)

It is possible to use JavaScript to grab only a substring of the base64 data/document and set that as the img's src attribute, as the linked answer proposes.

I'm just not sure that it's worth the trouble to do it this way, it seems rather complicated for maybe a very small gain in loading time.


if you want to use it, one way to do it could be this (simple draft, no error handling, no fancy functions): in your HTML document:

<script type="text/javascript" src="base64file.js" />
....
<img id="image0" src="" />
....
<img id="image1" src="" />
....
<script type="text/javascript">
   var singleImages = allImages.split('$');
   document.getElementById('image0').src = 'data:image/png;base64,' + singleImages[0];
   document.getElementById('image1').src = 'data:image/png;base64,' + singleImages[1];
   ....
</script>

in your base64file.js:

allImages = 'abc... <- the base64 code for the first image
             $ <- split indicator that is not a part of the base64 character set
            cba.... <- the base64 code for the second image
            ';
cypherabe
  • 2,562
  • 1
  • 20
  • 35
  • didn't see Ken's answer while writing this, which is basically the same idea. should I delete my answer? – cypherabe Feb 07 '14 at 11:54
1

Put each base64-encoded picture in a file of its own, then use server-side code to insert the base-64 data inside the HTML page. The server will see three separate files, but the client (i.e. the web browser) is served one big file. Now you only have one big HTTP request for the HTML page together with the two pictures. If necessary, use mod_rewrite to make the web browser believe that it is a static cacheable HTML page rather than a server-side PHP/Perl/whatever page.

Simple PHP example (tested on Ubuntu 12.04):

<img src="data:image/png;base64,<?php readfile("image1.base64.txt"); ?>" />
<img src="data:image/png;base64,<?php readfile("image2.base64.txt"); ?>" />

Note: make sure the base64 files do not have a newline character at the end.

Ruud Helderman
  • 10,563
  • 1
  • 26
  • 45
  • @KirkStrobeck: Physically, the images _are_ in separate files; with _fopen wrappers_ enabled, they can even be files on a different web server. You just do not reveal this separation to the client. Is that a problem? Or is there something else I missed in your question? – Ruud Helderman Feb 07 '14 at 20:58
  • Think of 2 html files, both very short, and one external 500kb file that gets cached and is immediately available on both pages. – Kirk Strobeck Feb 09 '14 at 03:16
  • Good point. Sorry, this particular approach is less suitable for big pictures that appear on more than one page. – Ruud Helderman Feb 09 '14 at 11:07
1

Use an image array. This requires no javascript, only HTML and CSS.

First, use a graphics editor (e.g. GIMP) to stitch your pictures together into one single big picture.

image array

Then use CSS attributes (width, height, background-position) to crop the image on the fly. <img> does not support this, but background images do.

.iconarray
{
    display: block;
    background-image: url(iconarray.png);
    background-repeat: no-repeat;
}

.icon1
{
    width: 64px;
    height: 64px;
    background-position: 0px 0px;
}

.icon2
{
    width: 64px;
    height: 64px;
    background-position: 0px -64px;
}

Use the CSS classes in a <span>:

<span class="iconarray icon1"></span>
<span class="iconarray icon2"></span>

Demo: http://jsfiddle.net/rFTpL/

Kudos go to:

Community
  • 1
  • 1
Ruud Helderman
  • 10,563
  • 1
  • 26
  • 45
  • Right, that's pretty standard sprite usage .. thx for the answer, but I'm looking for all types of mixed content, not images alone. `css`, `js`, `fonts`, `images`, and more – Kirk Strobeck Feb 10 '14 at 03:13
  • Sorry, I wasn't aware of that requirement. But I like your idea, so I gave you one more answer which (like most other answers here) uses JavaScript, and covers it all: css, js, fonts, images, and anything else you can think of. – Ruud Helderman Feb 12 '14 at 11:33
1

I've a simple answer.

var data64imgs = ['', ''];

$(function () {
    $('[data-index]').each(function () {
        $(this).attr("src",data64imgs[parseInt($(this).data("index"))]);
    });
});

Paste the above code in a separate JS file and reference it where you need it.

Do this:

<img data-index="0" />
<img data-index="1" />

Now automatically, the images will show up.

Here's the Demo

Note if you want additional imgs, then add it to the array and that's it , just use appropriate index.

Amit Joki
  • 58,320
  • 7
  • 77
  • 95
1

From what I understand, you wish to cache various type of resources (like a font, an image, etc.) using a single HTTP request.

So, here's my take on this using jQuery (you may opt for using vanilla JavaScript):

Firstly, you can store all your data-to-cache as string or Boolean or Number in a Associative Array in JavaScript and load this single file as shown below.

var ResourceLibrary = [];
ResourceLibrary["hello"] = "Hello World!";
ResourceLibrary["genderMale"] = true;
ResourceLibrary["username"] = "srvikram13";
ResourceLibrary["isDeveloper"] = true;
ResourceLibrary["firefox"] = "...."
ResourceLibrary["chrome"] = "...."

Now for every element that you wish to apply the cache data, you can give a custom attribute say-data-src as shown below.

<img src='#' data-src='resourceId-firefox' alt='firefox' />
<img src='#' data-src='resourceId-chrome' alt='chrome' />
<p data-src='resourceId-hello'></p><br>
<input id='username' data-src='resourceId-username'/><br>
<input type="radio" data-src="resourceId-genderMale" />Male<br>
<input type="checkbox" data-src="resourceId-isDeveloper" />Developer<br>

Now on document.ready (or onload if using vanilla JS) just loop through each element with this custom attribute; and apply the cached value based on the corresponding resourceId.

$(document).ready(function(){
    $("*").each(function(i, e){
        if($(e).attr("data-src") && $(e).attr("data-src").indexOf("resourceId-") != -1) {
            switch($(e).get(0).tagName) { // Add more case statements to handle various types of elements
                case "INPUT" : 
                    if(!$(e).attr("type")){
                        $(e).val(ResourceLibrary[$(e).attr("data-src").replace("resourceId-", "")]);
                        return;
                    }
                    switch($(e).attr("type").toLowerCase()) {
                        case "radio":
                        case "checkbox":
                            $(e).attr("checked", ResourceLibrary[$(e).attr("data-src").replace("resourceId-", "")])
                    }
                    return;
                case "IMG" :
                    $(e).attr("src", ResourceLibrary[$(e).attr("data-src").replace("resourceId-", "")]);
                    return;
                case "P" :
                    $(e).html(ResourceLibrary[$(e).attr("data-src").replace("resourceId-", "")]);
                   return;
                default:
                   return;
            }
        }
    });
});

Working Demo

Vikram Deshmukh
  • 12,304
  • 4
  • 36
  • 38