1

我正在尝试动态修改嵌入在 html 页面中的 SVG 文档,因此我可以在某些事件发生时添加装饰器(例如,按下按钮)。

为此,我首先在“onload”期间将装饰器图像插入 SVG 的“defs”元素,然后在事件发生期间将“use”元素添加到 SVG 组中。该代码似乎在 onload 期间添加了(在 Firefox 中)图像元素,并在事件发生时添加了 use 元素,但不显示装饰器图像。如果我保留同一个 SVG 文档,它会被所有浏览器正确显示。

让我向您展示一个简化的代码。这是一个 jsfiddle,感谢 Phrogz:http: //jsfiddle.net/ewYkp/3/

想象一下这个 html 页面:

 <!DOCTYPE html>
 <html>
 <head>
  <title>Dynamic Modification of SVG demo</title>
  <meta content="text/html;charset=utf-8" http-equiv="Content-Type">
  <script language="javascript">
   var svgns = "http://www.w3.org/2000/svg";
   function setOnLoad()
   {
    var svg = document.getElementById("SVG_IMAGE").contentDocument;
    defs = svg.getElementsByTagName("defs");
    def1 = defs[0];
    var imageNote = document.SVG_IMAGE.contentDocument.createElementNS(svgns,"image");
    imageNote.setAttribute("width","22");
    imageNote.setAttribute("height","22");
    imageNote.setAttribute("id","noteImage");
     imageNote.setAttribute("xlink:href","");
     def1.appendChild(imageNote);
    };

    function decorateSVG() {
     var svg = document.getElementById("SVG_IMAGE");
     var dElement = svg.contentDocument.getElementById("group1");
 var useNote = svg.contentDocument.createElementNS(svgns,"use");
 useNote.setAttribute("x","150");
 useNote.setAttribute("y","150");
 useNote.setAttribute("xlink:href","#noteImage");
 dElement.appendChild(useNote);
    };
   </script>
 </head>
 <body>
   <h1>Dynamic Modification of SVG demo - embed svg file with SVG image</h1>
   <p> A  yellow circle that was embeded using the svg "object" tag</p>
    <object id="SVG_IMAGE" preserveAspectRatio="xMidYMid meet"  data="basic_shapes_circle1.svg" width="400" y="0" x="0" type="image/svg+xml" onload="setOnLoad()">
    </object>
   <p>
    <button onclick="decorateSVG('circle')">Decorate circle </button>
   </p>
  <hr>
 </body>
</html> 

以及以下 svg 文档:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg viewBox="0 0 1000 1000" xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink">
 <defs>
  <!-- A circle of radius 200 -->
  <circle id="s1" cx="200" cy="200" r="200" fill="yellow" stroke="black" stroke-width="3"/>
 </defs>
 <g id="group1">
    <use x="0" y="0" xlink:href="#s1"/>
 </g>
</svg>

代码(据说)结果是这样的:

<?xml version="1.0" encoding="UTF-8" standalone="no"?>
<svg viewBox="0 0 1000 1000" xmlns="http://www.w3.org/2000/svg" version="1.1" xmlns:xlink="http://www.w3.org/1999/xlink">
 <defs>
  <!-- A circle of radius 200 -->
  <circle id="s1" cx="200" cy="200" r="200" fill="yellow" stroke="black" stroke-width="3"/>
  <image width="22" height="22" id="noteImage" xlink:href=""/>
</defs>
<g id="group1">
  <use x="0" y="0" xlink:href="#s1"/>
  <use x="150" y="150" xlink:href="#noteImage"/>
 </g>
</svg>

如果将 SVG 文档保存到文件中,则它可以在任何浏览器中正确显示。但是,在内存中修改时它似乎不起作用,我不知道为什么。有任何想法吗?

在此先感谢您的帮助。

4

1 回答 1

4

问题是您没有xlink:href正确设置属性。你正在这样做:

someElement.setAttribute( "xlink:href", "…" );

这样做会创建一个名为“xlink:href”(无效名称)且没有命名空间的属性。相反,您想使用:

someElement.setAttributeNS( "http://www.w3.org/1999/xlink", "href", "…");

这是一个工作演示,表明无论您将动态创建的图像直接添加到组中,还是将其放在<defs>部分中并通过 a 引用它,这都有效<use>

演示:http: //jsfiddle.net/ewYkp/4/


顺便说一句,为了方便和节省你的手腕,我推荐一个这样的小功能:

function createOn( dad, name, attrs, text ){
  var svg = dad.ownerSVGElement, doc = dad.ownerDocument;
  var ns = createOn.$NAMESPACES;
  var defaultNS = svg.namespaceURI;
  if (!ns){
    ns = createOn.$NAMESPACES = {};
    for (var a=svg.attributes,i=a.length;i--;) if (a[i].prefix=='xmlns') ns[a[i].localName] = a[i].nodeValue;
  }
  var p = name.split(':');
  var el = p[1] ? doc.createElementNS(ns[p[0]],p[1]) : doc.createElementNS(defaultNS,name);
  for (var a in attrs){
    p = a.split(':');
    if (p[1]) el.setAttributeNS(ns[p[0]],p[1],attrs[a]);
    else      el.setAttributeNS(null,a,attrs[a]);
  }
  if (text) el.appendChild(doc.createTextNode(text));
  return dad.appendChild(el);
}

像这样使用它:

var defs = svgDoc.querySelector('defs');
var img  = createOn(defs,'image',{
  x:100, y:150,
  width:22, height:22,
  id:'noteImage', 'xlink:href':data
});

只要前缀与所属文档上声明的命名空间匹配,它将自动查找元素名称或属性名称的命名空间前缀。这是一个使用它的更新演示:

演示:http: //jsfiddle.net/ewYkp/6/

于 2013-10-29T15:20:38.397 回答