gpt4 book ai didi

javascript - SVG ForeignObject 覆盖 Chrome 中的所有其他元素

转载 作者:搜寻专家 更新时间:2023-11-01 04:43:21 24 4
gpt4 key购买 nike

2016 年 4 月 22 日更新:没什么新鲜事,我只是决定检查一下,然后发布快速代码更新。 Firefox 仍然按预期运行,IE 完全没有运行,而 Chrome 50.0.2661.87 m 仍然与去年一样运行。下面的 fiddle 链接和代码已更新以反射(reflect)最新的工作版本(在 Firefox 中)。

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

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

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

  1. 将所有内容(包括图像)写入单个 SVG,然后将其渲染到 Canvas 上。

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

我选择了选项 1,因为它看起来简单明了。

问题:SVG 绘制顺序应该遵循 DOM 顺序,但在 Chrome(38 和 Canary)的情况下,它的行为就像是在之后 它渲染原生对象,完全覆盖原生对象。 (该代码在 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>

最佳答案

万岁!我不确定变化是什么时候发生的,但截至目前,the Chrome bug mentioned in the comments above has been resolved, 好像解决了问题。

我的演示现在适用于所有主流浏览器:

  • Chrome (69.0.3497.100)
  • 火狐 (62.0.2)
  • 边缘 (38.14393.2068.0)

关于javascript - SVG ForeignObject 覆盖 Chrome 中的所有其他元素,我们在Stack Overflow上找到一个类似的问题: https://stackoverflow.com/questions/26348038/

24 4 0
Copyright 2021 - 2024 cfsdn All Rights Reserved 蜀ICP备2022000587号
广告合作:1813099741@qq.com 6ren.com