147

I want to change background dynamically in JS and my set of images is in base64 encoded. I try:

document.getElementById("bg_image").style.backgroundImage = 
   "url('http://amigo.com/300107-2853.jpg')"; 

with perfect result,

yet I fail to do the same with:

document.getElementById("bg_image").style.backgroundImage = 
   "url('...')";

nor

document.getElementById("bg_image").style.backgroundImage = 
   "...";

Is there any way to do it?

Igor Savinkin
  • 5,669
  • 8
  • 37
  • 69
  • 2
    `url('` should work, my problem was that the ActionScript dataURL actually had newlines, and I had to `replace(/\n/g, '')` – neaumusic Nov 16 '17 at 07:47
  • `yourimageelement.setAttribute('style', 'background-image:url('+ yourbase64data +')');` – Hanna Rose Mar 15 '23 at 09:57

10 Answers10

158

I tried to do the same as you, but apparently the backgroundImage doesn't work with encoded data. As an alternative, I suggest to use CSS classes and the change between those classes.

If you are generating the data "on the fly" you can load the CSS files dynamically.

function change() {
  if (document.getElementById("test").className == "backgroundA") {
    document.getElementById("test").className = "backgroundB";
    document.getElementById("test2").className = "backgroundA";
  } else {
    document.getElementById("test").className = "backgroundA";
    document.getElementById("test2").className = "backgroundB";
  }
}

btn.onclick = change;
.backgroundA {
  background-image: url("");
}

.backgroundB {
  background-image: url("");
}
<div id="test" height="20px" class="backgroundA">
  div test 1
</div>
<div id="test2" name="test2" height="20px" class="backgroundB">
  div test2
</div>
<input type="button" id="btn" />

I fiddled it here, press the button and it will switch the divs' backgrounds: http://jsfiddle.net/egorbatik/fFQC6/

Jeremy Thille
  • 26,047
  • 12
  • 43
  • 63
Ezequiel Gorbatik
  • 2,475
  • 1
  • 16
  • 19
  • thank you, yet i work with big quantity of elements, so it's not practical to exchange classes. – Igor Savinkin Jun 13 '13 at 19:02
  • 1
    @IgorSavinkin and whomever reads this in the future. If you are changing lots of elements to the same background simultaneously, you can use css to cascade the rendering, by putting a class against each element that you want to be changed, but changing the class of a common ancestor element. For example, the css would be `#ancestor.backgroundA .Change{background-image:...}` and `#ancestor.backgroundB .Change{background-image...}`, where `#ancestor` is the id of the common ancestor, and `.Change` is the class applied to all elements to which to apply one of the backgrounds. – Patanjali Nov 22 '16 at 03:35
  • It's funny to see everyone voting this answer as the best before even thinking if it would be a good practice to create a css with a million classes... or even better: a css containing a million base64 encoded images (tip: just think of a folder with a million maybe HD, images in it). But yeah, this works for a few images. – gabriel garcia Jun 06 '20 at 11:35
  • Also, how would you solve the problem of adding new images to your system available images? is this approach even scalable? – gabriel garcia Jun 06 '20 at 11:38
48

Had the same problem with base64. For anyone in the future with the same problem:

url = "...";

This would work executed from console, but not from within a script:

$img.css("background-image", "url('" + url + "')");

But after playing with it a bit, I came up with this:

var img = new Image();
img.src = url;
$img.css("background-image", "url('" + img.src + "')");

No idea why it works with a proxy image, but it does. Tested on Firefox Dev 37 and Chrome 40.

Hope it helps someone.

EDIT

Investigated a little bit further. It appears that sometimes base64 encoding (at least in my case) breaks with CSS because of line breaks present in the encoded value (in my case value was generated dynamically by ActionScript).

Simply using e.g.:

$img.css("background-image", "url('" + url.replace(/(\r\n|\n|\r)/gm, "") + "')");

works too, and even seems to be faster by a few ms than using a proxy image.

lutogniew
  • 693
  • 6
  • 9
36

Try this, I have got success response ..it's working

$("#divId").css("background-image", "url('data:image/png;base64," + base64String + "')");
gihansalith
  • 1,770
  • 5
  • 17
  • 21
  • Seems if `data:image/png;base64` is long, and the string define as jquery variable, then does not work. But works if string defined as php variable like PHClaus example.... Just info about what i tested – Andris Feb 08 '15 at 18:29
13

Adding this trick to gabriel garcia's following solution -

var img = 'data:image/png;base64, ...'; //place ur base64 encoded img here
document.body.style.backgroundImage = 'url(' + img + ')';

However, in my case this wasn't working. Moving url into the img variable did the trick. SO the final code looked like this

var img = "url('data:image/png;base64, "+base64Data + "')";
document.body.style.backgroundImage = img; 
Tunaki
  • 132,869
  • 46
  • 340
  • 423
Amit Kumar Rai
  • 169
  • 1
  • 4
11

I tried this (and worked for me):

var img = 'data:image/png;base64, ...'; //place ur base64 encoded img here
document.body.style.backgroundImage = 'url(\'' + img + '\')';

ES6 syntax:

let img = 'data:image/png;base64, ...'
document.body.style.backgroundImage = `url('${img}')`

A bit better:

let setBackground = src => {
    this.style.backgroundImage = `url('${src}')`
};

let node = nodeIGotFromDOM, img = imageBase64EncodedFromMyGF;
setBackground.call(node, img);
gabriel garcia
  • 368
  • 3
  • 17
4

This is the correct way to make a pure call. No CSS.

<div style='background:url()repeat-x center;'></div>
3

I think this will help, Base64 is addressed by CSS in a different way, you should set the data type of the image to base64, this will help the CSS to change the base64 to image and display it to the user. and you can use this from javascript by assigning the background image using the jquery script, it will automatically change the base64 to mage and display it

url = "data:image;base64,"+data.replace(/(\r\n|\n|\r)/gm, "");
$("body").css("background-image", "url('" + url.replace(/(\r\n|\n|\r)/gm, "") + "')");
Ian Elvister
  • 357
  • 3
  • 9
  • Code-only answers are discouraged. Please click on [edit] and add some words summarising how your code addresses the question, or perhaps explain how your answer differs from the previous answer/answers. I can't see any real difference between this and one of the previously posted answers. – Nick Jan 22 '19 at 01:27
3

Do not use quotes inside url when using data URIs.

li {
  background:
    url()
    no-repeat
    left center;
  padding: 5px 0 5px 25px;
}
Wellington
  • 151
  • 1
  • 3
  • Is it against CSS / data URIs to use quotes? Your link doesn't mention the issue. – gth Apr 11 '22 at 08:25
  • [Apparently it should work too](https://developer.mozilla.org/en-US/docs/Web/CSS/url), but for some reason, the **data URI** only worked without quotes for me. – Wellington Apr 12 '22 at 16:08
2

What I usually do, is to store the full string into a variable first, like so:

<?php
$img_id = '...';
?>

Then, where I want either JS to do something with that variable:

<script type="text/javascript">
document.getElementById("img_id").backgroundImage="url('<?php echo $img_id; ?>')";
</script>

You could reference the same variable via PHP directly using something like:

<img src="<?php echo $img_id; ?>">

Works for me ;)

0

In my case, it was just because I didn't set the height and width.

But there is another issue.

The background image could be removed using

element.style.backgroundImage=""

but couldn't be set using

element.style.backgroundImage="some base64 data"

Jquery works fine.

stlebeax
  • 1
  • 3