SVG ForeignObject绘制Chrome中的所有其他元素

时间:2014-10-13 20:05:40

标签: javascript google-chrome svg html5-canvas

2016年4月22日更新:没有什么新内容,我只是决定检查一下,并发布快速代码更新。 Firefox仍然按预期执行,IE根本没有执行,Chrome 50.0.2661.87 m的行为仍然与去年相同。下面的小提琴链接和代码已经更新,以反映最新的工作版本(在Firefox中)。

背景 我正在玩一个画布+ HTML元素到PNG。当然,这意味着创建一个临时SVG,它将HTML作为foreignObject托管。

整个事情是一个元素层。我有一个背景,一层元素,画布和另一层元素。您可以在下面的代码段中看到它的样子。

我本可以通过两种方式解决这个问题:

  1. 将所有内容(包括图像)写入单个SVG,该SVG将呈现给画布。

  2. 编写两个SVG,一个用于背景和图像后面的项目,另一个用于图像前面的项目,然后绘制背面SVG,然后是图像,然后是前面的SVG到目标画布

  3. 我选择了选项1,因为它看似简单明了。

    问题: SVG绘制顺序应该遵循DOM顺序,但在Chrome(38& Canary)的情况下,它的行为就像它在后呈现foreignObjects em>它渲染本机对象,完全覆盖本机对象。 (代码在Firefox中按预期工作,在IE11中失败了。)那么谁是正确的?这是Chrome,Firefox中的错误,还是他们都没有正确处理?或者我错过了一些用户错误?

    谢谢!

    function putAnImageInTheCanvas() {
      var svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
      var svgNS = svg.namespaceURI;
      svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
      svg.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink');
      svg.setAttribute('height', '310');
      svg.setAttribute('width', '310');
      svg.setAttribute('version', '1.1');
    
      var svgRect = document.createElementNS(svgNS, 'rect');
      svgRect.setAttribute('x', '125');
      svgRect.setAttribute('y', '25');
      svgRect.setAttribute('height', '250');
      svgRect.setAttribute('width', '50');
      svgRect.setAttribute('fill', 'rgb(0,255,255)');
    
      svg.appendChild(svgRect);
    
      var dataSrc = 'data:image/svg+xml;base64,' + btoa(svg.outerHTML);
    
      var img = document.createElement('img');
      img.setAttribute('src', dataSrc);
    
      var c = document.getElementById('myCanvas');
      var ctx = c.getContext("2d");
      img.addEventListener('load', function() {
        ctx.drawImage(img, 0, 0, 310, 310, 0, 0, 310, 310);
      });
    }
    
    function render(darwBackground) {
      var myCanvas = document.getElementById('myCanvas');
    
      var svg = document.createElementNS('http://www.w3.org/2000/svg', 'svg');
      var svgNS = svg.namespaceURI;
      svg.setAttribute('xmlns', 'http://www.w3.org/2000/svg');
      svg.setAttribute('xmlns:xlink', 'http://www.w3.org/1999/xlink');
      svg.setAttribute('height', '310');
      svg.setAttribute('width', '310');
      svg.setAttribute('version', '1.1');
    
      var background = document.getElementById('main').cloneNode();
      background.setAttribute("xmlns", document.documentElement.namespaceURI);
      var svgFO_BG = document.createElementNS(svgNS, 'foreignObject');
      svgFO_BG.setAttribute('height', '310');
      svgFO_BG.setAttribute('width', '310');
      svgFO_BG.setAttribute('x', '0');
      svgFO_BG.setAttribute('y', '0');
      svgFO_BG.innerHTML = background.outerHTML.replace(/(\r\n|\n|\r|\t|[\s]{2,})/gm, '');
    
      var back = document.getElementById('back').cloneNode(true);
      back.setAttribute("xmlns", document.documentElement.namespaceURI);
      var svgFO_AB = document.createElementNS(svgNS, 'foreignObject');
      svgFO_AB.setAttribute('height', '310');
      svgFO_AB.setAttribute('width', '310');
      svgFO_AB.setAttribute('x', '0');
      svgFO_AB.setAttribute('y', '0');
      svgFO_AB.innerHTML = back.outerHTML.replace(/(\r\n|\n|\r|\t|[\s]{2,})/gm, '');
    
      var front = document.getElementById('front').cloneNode(true);
      front.setAttribute("xmlns", document.documentElement.namespaceURI);
      var svgFO_AA = document.createElementNS(svgNS, 'foreignObject');
      svgFO_AA.setAttribute('height', '310');
      svgFO_AA.setAttribute('width', '310');
      svgFO_AA.setAttribute('x', '0');
      svgFO_AA.setAttribute('y', '0');
      svgFO_AA.innerHTML = front.outerHTML.replace(/(\r\n|\n|\r|\t|[\s]{2,})/gm, '');
    
      var svgImage = document.createElementNS(svgNS, 'image');
      svgImage.setAttribute('xlink:href', myCanvas.toDataURL());
      svgImage.setAttribute('x', '0');
      svgImage.setAttribute('y', '0');
      svgImage.setAttribute('height', '310');
      svgImage.setAttribute('width', '310');
    
      var g = document.createElementNS(svgNS, 'g');
      if (darwBackground) {
        g.appendChild(svgFO_BG);
        svg.appendChild(g);
      }
    
      g = document.createElementNS(svgNS, 'g');
      g.appendChild(svgFO_AB);
      svg.appendChild(g);
    
      g = document.createElementNS(svgNS, 'g');
      g.appendChild(svgImage);
      svg.appendChild(g);
    
      g = document.createElementNS(svgNS, 'g');
      g.appendChild(svgFO_AA);
      svg.appendChild(g);
    
      var data = svg.outerHTML;
    
      document.getElementById('renderOutput').innerHTML = data;
    }
    <input type="button" value="load canvas image" onclick="putAnImageInTheCanvas();" />
    <input type="button" value="render with background" onclick="render(true);" />
    <input type="button" value="render without background" onclick="render(false);" />
    <h2>Preview:</h2>
    <div id="main" style="border: 5px blue solid; width: 300px; height: 300px; background: yellow; position: relative; top: 0; left: 0; overflow: hidden;">
      <canvas id="myCanvas" height="300px" width="300px" style="display: block; position: absolute; top: 0; left: 0; z-index: 1;"></canvas>
      <div id="back" style="position: relative; top: 0; left: 0;">
        <div style=" width: 100px; position: absolute; top: 75px; left: 75px; font-size: 20px; font-family: times; z-index: 0;">
          <div style="background: orange;">BACK</div>
        </div>
      </div>
      <div id="front" style="position: relative; top: 0; left: 0;">
        <div style=" width: 100px; position: absolute; top: 150px; left: 150px; font-size: 20px; font-family: times; z-index: 2;">
          <div style="background: lime;">FRONT</div>
        </div>
      </div>
    </div>
    <h2>Render Result:</h2>
    <div id="renderOutput">
    
    </div>

2 个答案:

答案 0 :(得分:0)

万岁!我不确定何时会发生更改,但截至目前,the Chrome bug mentioned in the comments above已解决,似乎已经解决了该问题。

我的演示现在可以在所有主要浏览器上使用了

  • Chrome(69.0.3497.100)
  • Firefox(62.0.2)
  • 边缘(38.14393.2068.0)

答案 1 :(得分:0)

在Chrome版本76(金丝雀)和正式版本中,这仍然是一个问题。 74.

我使用的解决方法是将画布放置在svg之前而不是其中的foreignObject中,并用position: relative的div包裹整个内容,因此{{1} }可以放置在svg上,使其始终位于画布上方。

包装器div还必须具有一些额外的CSS(在我的情况下为flexbox),以便画布位于svg下方。

See it on CodePen

absolute

这种方法的缺点是,如果svg以100%的比例覆盖了画布,则您将不会访问任何画布鼠标/屏幕事件侦听器。