简体   繁体   English

SVG foreignObject 在 Safari 上无法正常工作

[英]SVG foreignObject not working properly on Safari

I have a problem with SVG's foreignObjects elements on Safari. I am working with SVG elements and what I am using d3.js library.我在 Safari 上遇到 SVG 的 foreignObjects 元素问题。我正在使用 SVG 元素以及我正在使用的 d3.js 库。

What I did is to append on canvas a foreignObject element because differently from rect or other SVG elements didn't allow to put html elements inside.我所做的是 append 上的 canvas 一个 foreignObject 元素,因为与 rect 或其他 SVG 元素不同,不允许将 html 元素放入其中。 Now, after some days of development I noticed that it seems that Safari doesn't allow to put html elements inside a foreignObject.现在,经过几天的开发,我注意到 Safari 似乎不允许将 html 元素放入 foreignObject 中。 Better, it allows to do and all seems to work according to the inspector, but the I put inside the foreignObject has a background set.更好的是,它允许做并且根据检查员似乎一切正常,但我放在 foreignObject 中的有一个背景集。

The result is that I can correctly drag the foreignObject and the div inside inside the canvas but differently from Chrome or Firefox, the div's background stay on the canvas top left corner not following the div.结果是我可以正确地将 foreignObject 和 div 拖到 canvas 内部,但与 Chrome 或 Firefox 不同的是,div 的背景位于 canvas 的左上角,不跟随 div。

Here below I paste the portion of the style and foreign object code I append to the canvas下面我把样式和国外object代码我append的部分贴到canvas

SVG 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>

Style风格

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(移动和桌面)上背景远离它的 div

That's all.就这样。 I think I am doing something that Safari doesn't like.我想我正在做 Safari 不喜欢的事情。 But question is.. what?但问题是……什么?

Setting position: "fixed" on the div inside foreignObject fixed the foreignObject position for me in Safari (as well as on Chrome on iOS).设置position: "fixed"foreignObject内的div上为我在Safari(以及iOS 上的Chrome)中固定了foreignObject 的位置。 It doesn't seem to scale the foreignObject according to the scaling of the SVG though虽然它似乎没有根据 SVG 的缩放比例来缩放外来对象

I've found that you need to insert a <body> element to get it work in Safari, like this:我发现你需要插入一个<body>元素才能让它在 Safari 中工作,就像这样:

<foreignObject width="170" height="28">
    <body xmlns="http://www.w3.org/1999/xhtml">
        <div>my content</div>
    </body>
</foreignObject>

You can use <image> element in SVG instead of <foreignObject> .您可以在 SVG 中使用<image>元素而不是<foreignObject> This can solve the scale and position issue easily to avoid <foreignObject> bug on Safari这可以轻松解决比例和位置问题,以避免 Safari 上的<foreignObject>错误

Remember to change the src attribute to href.请记住将 src 属性更改为 href。

Try wrapping your foreignObjects in group elements.尝试将您的 foreignObjects 包装在组元素中。 I found that trying to style foreignObjects with transforms doesn't work well in Safari.我发现在 Safari 中尝试使用转换来设置 foreignObjects 的样式效果不佳。 Also most SVG's don't work in canvas elements in Firefox.此外,大多数 SVG 在 Firefox 中的画布元素中不起作用。

<g transform="translate(304,215)rotate(-174.86597769360367)">
    <foreignObject>
        <div> whatever in here </div>
    <foreignObject>
</g>

Try adding in an inline width and height for the foreignObject .尝试为foreignObject添加内联widthheight This seems to work on Safari.这似乎适用于 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>

Demo https://codepen.io/chrisgannon/pen/2107d092ec56025bba036934d680f2bc演示https://codepen.io/chrisgannon/pen/2107d092ec56025bba036934d680f2bc

I found a weird bug with iOS Safari regarding foreignObject .我在 iOS Safari 中发现了一个关于foreignObject的奇怪错误。 When I set the opacity in CSS for the HTML content within the foreignObject (it was a button in my case) it would position it absolutely over the SVG.当我在 CSS 中为foreignObject的 HTML 内容设置opacity时(在我的例子中它是一个按钮),它会将它绝对定位在 SVG 上。 But it positioned as expected when I removed the opacity .但是当我删除opacity时它按预期定位。 Took me hours to debug this issue.我花了几个小时来调试这个问题。

Putting display:contents on the html tag inside puts it into place as of Safari 15.4.从 Safari 15.4 开始,将display:contents放在里面的 html 标签上。 But you won't be able to interact much with it since the wrapping div is now nuked from the DOM.但是您将无法与它进行太多交互,因为包装 div 现在已从 DOM 中删除。 Defeating the whole purpose of foreignObject.打败了 foreignObject 的全部目的。 Sing with me Safari is the new IE跟我唱Safari是新的IE

A bit old, but I hope this could help:有点旧,但我希望这会有所帮助:

As @do-ic suggested adding position: fixed to the element inside foreignObject solves the positioning, but not the scale.正如@do-ic 建议的那样添加position: fixed to the element inside foreignObject解决了定位问题,但不是比例问题。 To solve the scale I added this:为了解决规模,我添加了这个:

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})`);
    }
});

Also the element inside foreignObject should have: transform-origin: 0px 0px foreignObject中的元素也应该有: transform-origin: 0px 0px

Hi i solved this behavior without viewbox and set a #svg height and width dimension嗨,我在没有 viewbox 的情况下解决了这种行为,并设置了#svg 高度和宽度尺寸

声明:本站的技术帖子网页,遵循CC BY-SA 4.0协议,如果您需要转载,请注明本站网址或者原文地址。任何问题请咨询:yoyou2525@163.com.

 
粤ICP备18138465号  © 2020-2024 STACKOOM.COM