如何将 FabricJS canvas 的背景图片设置为来自 Web 服务的图片?

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

我有以下代码,当用户按下按钮时执行。它

  1. 连接到网络服务,
  2. 从中获取图像,
  3. 将其存储在 myDataURL 变量中并且
  4. 显示在 canvas。

代码如下:

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;
    });
}

这是浏览器 window 在我按下下载按钮后的样子:

现在我想将该图像设为 canvas 的背景图像,以便我可以在其上添加其他对象(例如矩形、其他图像)。

我该怎么做?

官方FabricJS tutorial推荐这个:

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));

但随后出现以下错误:

Update 1 (02.12.2015 13:23 MSK):无论我用哪种方法,下载后一点击背景图就消失了。

Here 是一个视频,它显示了我的意思。

更新 2(02.12.2015 13:28 MSK):

这是代码的当前版本:

选项 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;
}

选项 2:

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

更新 3(02.12.2015 14:07 MSK):

如果我尝试使用以下代码向 canvas 添加一个矩形,则会出现同样的问题。

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);
}

首先,它出现了,但是当我点击 canvas 时,它又变空了。

您的代码的主要问题是您使用的是 canvas 元素而不是 fabricJS canvas。

要使用 canvas.renderAll() 函数,您必须初始化一个 fabric.Canvas 对象。

检查代码段:

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>

此外,如果您出于某种原因不需要 img 元素,您可以按照您之前的说明进行操作:

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

更短更好。