试图将SVG绘制到画布,为什么我的SVG XML会被截断?

时间:2015-10-01 19:50:46

标签: javascript canvas d3.js svg internet-explorer-11

我真正想做的就是将我喜欢的动态创建的SVG放到PDF中,目前通过jsPDF。 addSVG不起作用,所以我试图将它们转换为PNG,以便尝试使用addImage。

这是在IE11(客户要求)。

如果我这样做:

var lsvg = d3.select("#nowhere2").node().parentNode.innerHTML;
console.log(lsvg);

可能有三分之一的SVG出现在控制台中。 Odder,它在事物的中间截断,所以没有结束标记或其他:

<div id="nowhere2"><svg xmlns="http://www.w3.org/2000/svg" width="50px" height="800px"><defs><pattern id="oaghm" patternUnits="userSpaceOnUse" width="30" height="30"><path stroke="#343434" stroke-linecap="square" stroke-width="80" d="M 0 30 l 30 -30 M -7.5 7.5 l 15 -15 M 22.5 37.5 l 15 -15" shape-rendering="auto" /></pattern></defs><rect style="fill: url(#oaghm);" stroke="black" x="10" y="20" width="10" height="10" /><defs><pattern id="zpdff" patternUnits="userSpaceOnUse" width="4" height="4"><path stroke="#343434" stroke-linecap="square" stroke-width="1" d="M 0 4 l 4 -4 M -1 1 l 2 -2 M 3 5 l 2 -2" shape-rendering="auto" /></pattern></defs><rect style="fill: url(#zpdff);" stroke="black" x="10" y="40" width="10" height="10" /><defs><pattern id="dyxwi" patternUnits="userSpaceOnUse" width="10" height="10"><path stroke="#343434" stroke-linecap="square" stroke-width="2" d="M 0 10 l 10 -10 M -2.5 2.5 l 5 -5 M 7.5 12.5 l 5 -5" shape-rendering="auto" /><path stroke="#343434" stroke-linecap="square" stroke-width="2" d

当然,这意味着我的DataURI已经损坏,虽然Chrome有勇气尝试(仅用于测试,必须使用IE11):



Odder仍然,如果我只是将lsvg对象附加回页面,它就会很好地呈现。

var lsvg = d3.select("#nowhere2").node().parentNode.innerHTML;
$("#nowhere").append(lsvg);

如果我在追加后尝试在控制台中检查它,我会得到相同的截断字符串。当然,这让我相信它是一个时间/异步问题,但是我已经尝试过每一种我能想到的方法来解决这个问题,包括设置绘制到画布然后保存-to-png函数作为我初始SVG绘制函数的回调函数。我每次都得到同样奇怪的截断字符串。

这是我的画布(由于其他问题在html中声明):

<canvas id="lcanvas" width="50" height="800"></canvas>

研究让我相信它可能是画布/ SVG尺寸不匹配但似乎并非如此(我得到的SVG显示的宽度/高度相同,见上文)。另外,如果这是真的,它会在画布甚至涉及之前截断初始字符串赋值似乎很奇怪。

我获得完整DataURI的一个辉煌时刻:



所以我明确地使用DataURI测试了它,换句话说,不用担心绘图是否已经完成。我做了:

limg.src = "";
console.log(limg.src);

在控制台中是一样的。炮轰。截断。串。这是我的代码:

        var lsvg = d3.select("#nowhere2").node().parentNode.innerHTML;
        console.log(lsvg);

        var limg = new Image;
        var lcontext = canl.getContext('2d');

        limg.src = 'data:image/svg+xml;base64,' + btoa(lsvg);

        console.log(limg.src);
        lcontext.load = limg.addEventListener("load", function () {
            //console.log(limg.src);
            //This fun bit of code brought to you courtesy of IE11.
            try {
                lcontext.drawImage(limg, 0, 0);
            }
            catch (err) {
                setTimeout(lcontext.drawImage(limg, 0, 0), 0);
            };
            console.log(canl.toDataURL('image/png'));
        });

我留下了画布可见,所以我可以看到可能会发生什么样的恶作剧。它会很好地从文件中提取图像。另外,从技术上讲,我在这里得到一张图片,它只是空白和空白。

我不想从文件中提取,因为这会污染画布,我在canl.toDataURL上收到安全错误。

我尝试了各种获取lsvg位,XMLSerialize等的方法;所有我得到的是一个更短的头(没有div元素)和一个更长的尾巴之前它被截断在看起来完全相同的字符数之后。就像往常一样,在事物的中间,所以仍然没有结束标签,只是在中游突然结束。

要清楚,这不是裁剪的SVG。这是一个剪辑的&#39; SVG。开始标记,没有结束标记,破坏图像,除非您是Chrome并且无论如何都尝试渲染它(IE甚至不打扰)。在画布涉及之前,SVG就会被破坏,甚至在显式赋值后也独立于绘图函数。

哦,甜蜜的正义,我有一个小提琴! https://jsfiddle.net/94xhyv6t/在Chrome和IE中试用,然后查看控制台。在Chrome中,如果您点击该链接,您将获得一个XML页面,但您可以在那里看到所有SVG元素,在IE中它会被截断。

很高兴知道我并非完全疯了。

只是因为我知道有人会提到它,这里没有父DIV节点:https://jsfiddle.net/94xhyv6t/1/同样的问题。

编辑:注意:这个小提琴显示了问题:https://jsfiddle.net/94xhyv6t/1/。将它加载到Chrome中,一切正常。在IE11中加载它,一切都被截断了。我删除了所有其他代码,这个小提琴只显示了SVG字符串生成组件以及它在IE中是如何被截断的。我已经尝试了每个https://jsfiddle.net/94xhyv6t/4/的XMLSerializer,同样的问题。也许有一种解决方法?

现在进行最后编辑:控制台只是截断显示,否则字符串就可以了。接受的答案有一个完整的小提示示例,显示如何将SVG放入画布,但是你不能在IE11中做canvas.toDataURL(),所以这不是一个合适的转换方法。我使用canvg作为解决方法。

1 个答案:

答案 0 :(得分:1)

不知道您的问题究竟是什么,但使用

var data = new XMLSerializer().serializeToString(document.querySelector('svg'));
img.src = 'data:image/svg+xml; charset=utf8, ' + encodeURIComponent(data);

适用于FF和Chrome。

&#13;
&#13;
var n = 20, // number of layers
    m = 200, // number of samples per layer
    stack = d3.layout.stack().offset("wiggle"),
    layers0 = stack(d3.range(n).map(function() { return bumpLayer(m); })),
    layers1 = stack(d3.range(n).map(function() { return bumpLayer(m); }));

var width = 960,
    height = 500;

var x = d3.scale.linear()
    .domain([0, m - 1])
    .range([0, width]);

var y = d3.scale.linear()
    .domain([0, d3.max(layers0.concat(layers1), function(layer) { return d3.max(layer, function(d) { return d.y0 + d.y; }); })])
    .range([height, 0]);

var color = d3.scale.linear()
    .range(["#aad", "#556"]);

var area = d3.svg.area()
    .x(function(d) { return x(d.x); })
    .y0(function(d) { return y(d.y0); })
    .y1(function(d) { return y(d.y0 + d.y); });

var svg = d3.select("#worm").append("svg")
    .attr("width", width)
    .attr("height", height);

svg.selectAll("path")
    .data(layers0)
  .enter().append("path")
    .attr("d", area)
    .style("fill", function() { return color(Math.random()); });

function transition() {
  d3.selectAll("path")
      .data(function() {
        var d = layers1;
        layers1 = layers0;
        return layers0 = d;
      })
    .transition()
      .duration(2500)
      .attr("d", area);
}

// Inspired by Lee Byron's test data generator.
function bumpLayer(n) {

  function bump(a) {
    var x = 1 / (.1 + Math.random()),
        y = 2 * Math.random() - .5,
        z = 10 / (.1 + Math.random());
    for (var i = 0; i < n; i++) {
      var w = (i / n - y) * z;
      a[i] += x * Math.exp(-w * w);
    }
  }

  var a = [], i;
  for (i = 0; i < n; ++i) a[i] = 0;
  for (i = 0; i < 5; ++i) bump(a);
  return a.map(function(d, i) { return {x: i, y: Math.max(0, d)}; });
}
var lsvg = d3.select("#worm").node().parentNode.innerHTML;
console.log(lsvg);
var img = new Image;

console.log(img.src);
document.body.appendChild(img);
img.onload = function(){
	var canvas = document.createElement('canvas');
// IE seems to not set width and height for svg in img tag...
    canvas.width = this.width||960;
    canvas.height = this.height||500;
	canvas.getContext('2d').drawImage(this, 0,0);
    document.body.appendChild(canvas);
}
img.src = 'data:image/svg+xml; charset=utf8, ' + encodeURIComponent(new XMLSerializer().serializeToString(document.querySelector('svg')));
&#13;
svg{border: 1px solid green}
img{border: 1px solid blue}
canvas{border: 1px solid red}
&#13;
<script src="https://cdnjs.cloudflare.com/ajax/libs/d3/3.4.11/d3.min.js"></script>
<div id="worm"></div>
&#13;
&#13;
&#13;

请注意,我在img标记的onload事件中添加了画布内容。

编辑:我发现IE没有设置<img>标签中加载的svg的宽度和高度属性,因此您必须对其进行硬编码或从{{1元素