Glory to Russia
Glory to Russia

Reputation: 18712

How to set the background image of FabricJS canvas to image from a web service?

I have following piece of code, which is executed when the user presses a button. It

  1. connects to a web service,
  2. gets an image from it,
  3. stores it in myDataURL variable and
  4. displays it on the canvas.

Here's the code:

function downloadButtonPressed() {
    var username = $("input#username").val();
    var password = $("input#password").val();
    $.ajax({
        type: "GET",
        url: "http://myserver/myapp/map",
        headers: {
            "Authorization": "Basic " + btoa(username + ":" + password),
          },
        crossDomain: true,
        xhrFields: {
                withCredentials: true
            },
        success: function (data, status, xhr) {
            alert("success");
        }
    }).fail(function ($xhr) {
        console.log("response: " + $xhr);
        var myDataURL = "data:image/png;base64," + $xhr.responseText;
        var canvas = document.getElementById('canvas');
        var context = canvas.getContext('2d');
        var img = new Image();
        img.onload = function() {
            context.drawImage(this, 0, 0, canvas.width, canvas.height);
        };
        img.src = myDataURL;
    });
}

And here's what the browser window looks like, after I press the download button:

Screenshot

Now I want to make that image a backgrond image of the canvas, so that I can add other objects (e. g. rectagles, other images) on top of it.

How can I do it?

The official FabricJS tutorial recommends this:

var myDataURL = "data:image/png;base64," + $xhr.responseText;
var canvas = document.getElementById('canvas');
var context = canvas.getContext('2d');
var img = new Image();
img.onload = function() {
    // context.drawImage(this, 0, 0, canvas.width, canvas.height);
};
img.src = myDataURL;
canvas.setBackgroundImage(myDataURL, canvas.renderAll.bind(canvas));

But then I get the following error:

Screenshot

Update 1 (02.12.2015 13:23 MSK): Regardless of which method I use, the background image disappears as soon as I click on it after download.

Here is a video, which shows, what I mean.

Update 2 (02.12.2015 13:28 MSK):

Here's the current version of code:

Option 1:

function ($xhr) {
    var myDataURL = "data:image/png;base64," + $xhr;
    var canvas = new fabric.Canvas('canvas');
    var img = new Image();
    img.onload = function() {
        var f_img = new fabric.Image(img);
        canvas.setBackgroundImage(f_img);
        canvas.renderAll();
    };
    img.src = myDataURL;
}

Option 2:

function ($xhr) {
    var myDataURL = "data:image/png;base64," + $xhr;
    var canvas = new fabric.Canvas('canvas');
    canvas.setBackgroundImage(myDataURL, canvas.renderAll.bind(canvas));
}

Update 3 (02.12.2015 14:07 MSK):

Same problem occurs, if I try to add a rectangle to the canvas using the following code.

function rectButtonPressed() {
    var canvas = new fabric.Canvas('canvas');
    var rect = new fabric.Rect({
      left: 100,
      top: 100,
      fill: 'red',
      width: 20,
      height: 20
    });
    canvas.add(rect);
}

First, it appears, but when I click on the canvas, it becomes empty again.

Upvotes: 2

Views: 10971

Answers (1)

AndreaBogazzi
AndreaBogazzi

Reputation: 14731

Main problem of your code is that you are working with the canvas element and not the fabricJS canvas.

To use canvas.renderAll() function you have to initialize a fabric.Canvas object.

Check the snippet:

var myDataURL = "";

var canvas = new fabric.Canvas('canvas');
var img = new Image();
img.onload = function() {
    // this is syncronous
    var f_img = new fabric.Image(img);
    canvas.setBackgroundImage(f_img);
    canvas.renderAll();
};
img.src = myDataURL;
<script src="http://cdnjs.cloudflare.com/ajax/libs/fabric.js/1.5.0/fabric.min.js"></script>
<canvas id='canvas' width="400" height="400" style="border:#000 1px solid;"></canvas>

Also if you do not need the img element for some reason, you can just do as you stated before:

var canvas = new fabric.Canvas('canvas');
canvas.setBackgroundImage(myDataUrl, canvas.renderAll.bind(canvas));

That is shorter and nicer.

Upvotes: 7

Related Questions