我在 Safari 上遇到 SVG 的foreignObjects 元素问题。我正在使用 SVG 元素以及我正在使用的 d3.js 库。
我所做的就是在画布上附加一个foreignObject元素,因为与矩形或其他SVG元素不同,不允许将html元素放入其中。现在,经过几天的开发,我注意到 Safari 似乎不允许将 html 元素放入foreignObject 中。更好的是,它允许执行操作,并且根据检查员的说法,所有操作似乎都有效,但是我放入foreignObject 中的内容有一个背景集。
结果是我可以正确地将foreignObject和div拖到画布内,但与Chrome或Firefox不同的是,div的背景停留在画布左上角而不是跟随div。
在下面,我粘贴了附加到画布的样式和外来对象代码的部分
SVG
<foreignObject transform="translate(304,215)rotate(-174.86597769360367)" height="180" width="40" class="dragging">
<div class="new-rect" style="background-image: url(); left:-45px;">
</div>
</foreignObject>
风格
foreignObject{
cursor:pointer;
position:relative;
}
foreignObject div.new-rect{
position: absolute;
top: -8px;
background-repeat: no-repeat;
background-size: auto;
height: 180px;
width: 130px;
}
仅此而已。我认为我正在做 Safari 不喜欢的事情。但问题是..什么?
在
position: "fixed"
内的 div
上设置 foreignObject
修复了我在 Safari 中(以及 iOS 上的 Chrome)中的foreignObject 位置。虽然它似乎没有根据 SVG 的缩放来缩放foreignObject
我发现您需要插入一个
<body>
元素才能使其在 Safari 中正常工作,如下所示:
<foreignObject width="170" height="28">
<body xmlns="http://www.w3.org/1999/xhtml">
<div>my content</div>
</body>
</foreignObject>
从 Safari 15.4 开始,将
display:contents
放在 html 标签上即可将其放置到位。但是您将无法与它进行太多交互,因为包装 div 现在已从 DOM 中删除。违背了foreignObject的全部目的。跟我一起唱歌 Safari 是新的 IE
新解决方案:
请勿在 Safari + 异物中使用以下样式:
position
webkit-transform-style
webkit-backface-visibility
transition
transform
calc
(计算结果可能不符合预期)
opacity
不使用它们也能避免风格问题
旧:
我找到@Quinn的答案,添加一些信息,首先,你需要获取你的SVG转换信息,就像是
<g class = "main-tree" transform="translate(875) scale(1.09)"></g>
。并且您可以使用 d3.zoomTransform
来获取这些数据,let transform = d3.zoomTransform(d3.select("#treesvg").node());
以下内容与@Quinn相同,需要选择
foreignObject
下的根元素,请勿在其他浏览器中执行代码
尝试添加内联
width
和 height
作为 foreignObject
。这似乎适用于 Safari。
<foreignObject style="overflow: visible;" y="0" x="0" width="240" height="160">
<div class="info-panel" xmlns="http://www.w3.org/2000/xmlns/" >
<div class="panel-text" xmlns="http://www.w3.org/2000/xmlns/">
<h1>ForeignObject</h1>
<p>For this to work in Safari try adding in a width and height inline on the foreignObject element. The background will also expand to the more text you add.</p>
</div>
</div>
</foreignObject>
演示 https://codepen.io/chrisgannon/pen/2107d092ec56025bba036934d680f2bc
有点旧,但我希望这能有所帮助:
正如 @do-ic 建议将
position: fixed
添加到 foreignObject
内部的元素可以解决定位问题,但不能解决缩放问题。
为了解决规模问题,我添加了这个:
d3.zoom().on('zoom', event => {
// zoom stuff...
const isSafari = /^((?!chrome|android).)*safari/i.test(navigator.userAgent);
if (isSafari) {
d3.selectAll(NODE_SELECTOR).style('transform', `scale(${transform.k})`);
}
});
foreignObject
内的元素也应该有:transform-origin: 0px 0px
尝试将foreignObjects包装在组元素中。我发现尝试使用转换来设置foreignObjects 的样式在Safari 中效果不佳。此外,大多数 SVG 不能在 Firefox 的画布元素中工作。
<g transform="translate(304,215)rotate(-174.86597769360367)">
<foreignObject>
<div> whatever in here </div>
<foreignObject>
</g>
我在 iOS Safari 中发现了一个关于
foreignObject
的奇怪错误。当我在CSS中为opacity
(在我的例子中它是一个按钮)中的HTML内容设置foreignObject
时,它会将其绝对定位在SVG上。但当我移除opacity
时,它的位置符合预期。我花了几个小时来调试这个问题。
嗨,我在没有视图框的情况下解决了这个行为,并设置了 #svg 高度和宽度尺寸
在我的例子中,我在foreignObject标签内有html内容,并且我的html文本在iPhone的所有浏览器中都没有缩放。 我通过在所有 html 标签前添加 xlink 来修复此文本缩放问题,如下面的示例代码所示。
<svg xmlns="http://www.w3.org/2000/svg"
xmlns:xlink="http://www.w3.org/1999/xlink" viewBox="0 0 900 900"
width="900" height="900">
<image xlink:href="img/chart.png" height="100%" width="100%" />
<foreignObject x="18%" y="1%" width="15%" height="20%">
<xlink:body xmlns="http://www.w3.org/1999/xhtml">
<xlink:table width="100%" height="100%">
<xlink:tr>
<xlink:td id="PlanetH3" align="center" valign="middle">
<text>Your text which is to be scaled.</text>
</xlink:td>
</xlink:tr>
</xlink:table>
</xlink:body>
</foreignObject>
</svg>
请注意,上面的代码示例中的第二行是最重要的,您必须在 svg 打开标记上定义 xlink 命名空间和 viewBox 属性。
谢谢。