使用foreignObject将svg导出为png/tiff



我使用https://github.com/jgraph/drawio(mxgraph(。我使用AngularJS作为前端,使用NodeJS/Express作为后端。这个想法是做一个图表设计,然后能够导出/打印它等等。我能够与后端交互并绘制图表,但在打印/渲染方面遇到了问题
问题是生成的svg包含QR和条形码的外来对象。当图表在前端渲染/显示时,我需要生成一个png/tiff以便将其发送到打印机,但nodejs无法渲染foreignObject元素。我在node上测试了canvg、sharp,但不支持foreignObject(根据github上的一些问题(。

这是一个例子svg:

<svg width='600' height='500'>
<g xmlns="http://www.w3.org/2000/svg" fill="#464445" font-family="Lucida Console" pointer-events="none" text-anchor="middle" font-size="72px">
<text x="350" y="135">ABCDEF12345</text>
</g>
<g xmlns="http://www.w3.org/2000/svg">
<switch>
<foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility">
<div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 130px; margin-left: 300px;">
<div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: #774400; ">
<div style="display: inline-block; font-size: 11px; font-family: Arial, Helvetica; color: rgb(119, 68, 0); line-height: 1.2; pointer-events: none; white-space: nowrap;">
<img src="" />
</div>
</div>
</div>
</foreignObject>
<text x="300" y="133" fill="#774400" font-family="Arial,Helvetica" font-size="11px" text-anchor="middle">[Object]</text>
</switch>
</g>
<g xmlns="http://www.w3.org/2000/svg">
<switch>
<foreignObject style="overflow: visible; text-align: left;" pointer-events="none" width="100%" height="100%" requiredFeatures="http://www.w3.org/TR/SVG11/feature#Extensibility">
<div xmlns="http://www.w3.org/1999/xhtml" style="display: flex; align-items: unsafe center; justify-content: unsafe center; width: 1px; height: 1px; padding-top: 340px; margin-left: 450px;">
<div style="box-sizing: border-box; font-size: 0px; text-align: center;" data-drawio-colors="color: #774400; ">
<div style="display: inline-block; font-size: 11px; font-family: Arial, Helvetica; color: rgb(119, 68, 0); line-height: 1.2; pointer-events: none; white-space: nowrap;">
<div title="Diagram Title">
<canvas style="display: none;" width="200" height="200"></canvas>
<img style="display: block;" src="" />
</div>
</div>
</div>
</div>
</foreignObject>
<text x="450" y="343" fill="#774400" font-family="Arial,Helvetica" font-size="11px" text-anchor="middle">[Object]</text>
</switch>
</g>
</svg>

应该是这样的:测试svg。使用sharp/canvg时,不显示QR和条形码,而是在最终的png/tiff上显示[Object]。甚至尝试了C#SVG NuGet和htm2canvas在后端进行渲染,也遇到了同样的问题。

我看过一些关于如何用将canvas.toDataURL渲染为png的帖子,但我在文档DOM上没有对象,因为mxgraph的工作方式不同,并且只有最后一个可用的字符串。关于如何删除foreignObjects或如何手动修改svg,有什么想法吗?

可以";创建";一个画布元素,并在里面渲染svg,然后导出它?

所以,在@ccprog回答后,我意识到我需要做什么。我很难创建一个新的画布,复制子元素属性,再次渲染,使用库来做这件事,但没有成功。有了这个答案,想法很简单:获取img src,创建新的子项,删除foreignObject。我对x,y坐标有一些问题,因为图像没有居中(mxgraph使用顶部填充和左侧边距(,代码最终是这个

var newXml = xmlCanvas.root;
for(var i = 0; i < newXml.children.length; i++){
var child = newXml.children[i];
if(child.tagName == 'g' && child.children[0].tagName == "switch"){
var imgChild = child.getElementsByTagName("img")[0];
var foreignObjectChild = child.getElementsByTagName("foreignObject")[0];
var textChild = child.getElementsByTagName("text")[0];
var image = document.createElement("image");
image.setAttribute("href", imgChild.src);
let marginLeft = Number(foreignObjectChild.firstChild.style.marginLeft.replace("px", ""));
image.setAttribute("x", marginLeft - imgChild.naturalWidth/2);
let paddingTop = Number(foreignObjectChild.firstChild.style.paddingTop.replace("px", ""));
image.setAttribute("y", paddingTop - imgChild.naturalHeight/2);
image.setAttribute("xmlns", "http://www.w3.org/2000/svg");
newXml.children[i].firstChild.appendChild(image);
newXml.children[i].firstChild.removeChild(foreignObjectChild);
newXml.children[i].firstChild.removeChild(textChild);
}
}

这个例子有点硬编码,但它有效。不是javascript专家,但这可能可以用更好的方式进行优化/重写。

我遇到的一个问题是,最终的svg具有标签"<a0:图像";以及"<a1:图像";等等,但随后我使用regex并删除de a0/a1/aN。不知道为什么,可能是我如何创建/附加子元素的一些错误。

最新更新