在 Angularjs 指令中包含 SVG 模板

IT技术 javascript svg angularjs
2021-02-21 02:10:10
<div ng-app="testrectApp">
<svg>
    <rect height="10" width="10" style="fill: #00ff00" />
    <testrect />
</svg>
</div>

这是我的指示

module.directive('testrect', function() {
    return {
        restrict: 'E',
        template: '<rect top="20" left="20" height="10" width="10" style="fill: #ff00ff" />',
        replace: true
    };
});

但这就是元素在浏览器中最终的样子。填充两次不是错字。

<rect top="20" left="20" height="10" width="10" style="fill: #ff00ff;fill: #ff00ff"></rect>

这是一个 jsfiddle http://jsfiddle.net/RG2CF/

我试图做的事情是不可能的还是我做错了什么?

谢谢

编辑:我应该补充一点,这个问题可能与模板 svg rect 命名空间为 xhtml 而不是 svg 的事实有关,但我不确定如何将它或命名空间强制返回到 svg,如果确实如此解决方案。

5个回答

templateNamespace您可以设置一个属性svg

module.directive('testrect', function() {
    return {
        restrict: 'E',
        templateNamespace: 'svg',
        template: '<rect .../>',
        replace: true
    };
});

这是文档链接

这是现在的正确答案。我会更新我的答案以指向你的
2021-05-07 02:10:10

Angular 1.3 更新!

我在下面的原始答案充其量是一个巨大的黑客。你真正应该做的是更新到 Angular 1.3 并templateNamespace:'svg'在你的指令中设置属性。下面以@Josef Pfleger 回答为例:在 Angularjs 指令中包含 SVG 模板


旧答案

如果将来有人遇到这种情况,我可以创建一个与 svg 模板一起使用的编译函数。它现在有点难看,因此非常感谢重构帮助。

现场示例:http : //plnkr.co/edit/DN2M3M?p=preview

app.service('svgService', function(){

    var xmlns = "http://www.w3.org/2000/svg";
    var compileNode = function(angularElement){
      var rawElement = angularElement[0];

      //new lines have no localName
      if(!rawElement.localName){
        var text = document.createTextNode(rawElement.wholeText);
        return angular.element(text);
      }
      var replacement = document.createElementNS(xmlns,rawElement.localName);
      var children = angularElement.children();

      angular.forEach(children, function (value) {
        var newChildNode = compileNode(angular.element(value));
        replacement.appendChild(newChildNode[0]);
      });

      if(rawElement.localName === 'text'){
        replacement.textContent = rawElement.innerText;
      }

      var attributes = rawElement.attributes;
      for (var i = 0; i < attributes.length ; i++) {
        replacement.setAttribute(attributes[i].name, attributes[i].value);
      }

      angularElement.replaceWith(replacement);

      return angular.element(replacement);
    };

    this.compile = function(elem, attrs, transclude) {
      compileNode(elem);

      return function postLink(scope, elem,attrs,controller){
//        console.log('link called');
      };
    }
  })

用例:

app.directive('ngRectAndText', function(svgService) {
  return {
    restrict: 'E',
    template: 
    '<g>'
    + '<rect x="140" y="150" width="25" height="25" fill="red"></rect>'
    + '<text x="140" y="150">Hi There</text>'
    + '</g>'
    ,
    replace: true,
    compile: svgService.compile
  };
});
伙计,经过 3 或 4 个小时的谷歌搜索如何创建 svg 元素后,您的服务是第一个对我有用的代码。谢谢伙计,你是个救星!
2021-04-22 02:10:10
现在有一个templateNamespace您可以设置属性svg(请参阅我的答案)。
2021-04-22 02:10:10
这很棒,但我发现它不适用于 defs 部分中的模式。它们被插入并渲染结束,但渲染看起来与原始渲染非常不同
2021-04-28 02:10:10
感谢这段代码!到目前为止,我发现了一个问题,对于<text>元素,您没有在replacement. 我会尝试修改代码,但我对DOM了解不多。
2021-05-07 02:10:10
再一个:如果我复制并粘贴生成的 svg 并将整个内容再次插入某个 div,它看起来像预期的那样,即使在 IE 中也是如此。
2021-05-17 02:10:10

为 SVG 使用 innerHTML shim(又名“innerSVG”)提供了这个难题的另一部分,作为 Jason More 富有洞察力的答案的替代方案。我觉得这种 innerHTML/innerSVG 方法更优雅,因为它不需要像 Jason 的回答那样的特殊编译函数,但它至少有一个缺点:如果在指令上设置了“replace: true”,它就不起作用。(此处讨论了innerSVG 方法:在SVG/XML 中是否有一些innerHTML 替换?

// Inspiration for this derived from a shim known as "innerSVG" which adds
// an "innerHTML" attribute to SVGElement:
Object.defineProperty(SVGElement.prototype, 'innerHTML', {
  get: function() {
    // TBD!
  },
  set: function(markup) {
    // 1. Remove all children
    while (this.firstChild) {
      this.removeChild(this.firstChild);
    }

    // 2. Parse the SVG
    var doc = new DOMParser().parseFromString(
        '<svg xmlns="http://www.w3.org/2000/svg" xmlns:xlink="http://www.w3.org/1999/xlink">'
        + markup
        + '</svg>',
        'application/xml'
    );

    // 3. Import the parsed SVG and grab the childNodes
    var contents = this.ownerDocument.importNode(doc.documentElement, true).childNodes;

    // 4. Append the childNodes to this node
    // Note: the following for loop won't work because as items are appended,
    // they are removed from contents (you'd end up with only every other
    // element getting appended and the other half remaining still in the
    // original document tree):
    //for (var i = 0; i < contents.length; i++) {
    //  this.appendChild(contents[i]);
    //}
    while (contents.length) {
      this.appendChild(contents[0]);
    }
  },
  enumerable: false,
  configurable: true
});

这是一个 Plunker 尝试:http ://plnkr.co/edit/nEqfbbvpUCOVC1XbTKdQ?p=preview

这样做的原因是:SVG 元素位于 XML 命名空间http://www.w3.org/2000/svg 中,因此,它们必须使用 DOMParser() 解析或使用 createElementNS() 实例化(如在 Jason 的回答中),两者都可以适当地设置命名空间。如果这个使用 SVG 的过程像 HTML 一样简单就好了,但不幸的是,它们的 DOM 结构略有不同,这导致需要提供我们自己的 innerHTML shim。现在,如果我们也可以使用“replace: true”来处理它,那就太好了!

编辑:更新了 appendChild() 循环以解决在循环运行时更改内容的方法。

目前 SVG 标签不支持动态添加标签,至少在 Chrome 中是这样。它甚至不显示新的 rect,即使您直接使用 DOM 或 JQuery.append() 添加它也不会显示。看看只是尝试用 JQuery 来做

// this doesn't even work right. Slightly different error than yours.
$(function() {
    $('svg').append('<rect top="20" left="20" height="10" width="10" style="fill: #ff00ff" />');
});

我的猜测是,无论您以这种方式做什么,您都会看到一些非常疯狂的行为。这似乎是浏览器的问题,而不是 Angular。我建议将其报告给 Chromium 项目

编辑:

如果您以编程方式通过 DOM 100% 添加它,它看起来可能会起作用:How to make Chrome redraw SVG dynamic added content?

我也遇到了 Firefox 的问题。你真的发现这在你的 Firefox 版本中有效吗?
2021-05-08 02:10:10
如果您希望将来可以使用它,请联系whatwg.org和/或w3.org/html/wg因为根据当前的规范,上面的 rect 将成为 DOM 中的 HTMLUnknownElement。
2021-05-16 02:10:10
是的,我在 FF 中也遇到了问题。
2021-05-18 02:10:10

angular 在创建元素时不注意命名空间(尽管元素似乎是用父元素的 namepsace 克隆的),因此<svg>如果没有自定义编译函数,内部的新指令将无法工作。

像下面这样的事情可以通过创建 dom 元素本身来解决这个问题。为此,您需要将xmlns="http://www.w3.org/2000/svg"的根元素作为属性,template并且您的模板必须是有效的 XML(只有一个根元素)。

directiveGenerator = function($compile) {
  var ret, template;
  template = "<g xmlns=\"http://www.w3.org/2000/svg\"> +
             "<rect top=\"20\" left=\"20\" height=\"10\" width=\"10\" style=\"fill: #ff00ff\" />" +
             "</g>";
  ret = {
    restrict: 'E',
    compile: function(elm, attrs, transclude) {
      var templateElms;
      templateElms = (new DOMParser).parseFromString(template, 'application/xml');
      elm.replaceWith(templateElms.documentElement);
      return function(scope, elm, attrs) {
        // Your link function
      };
    }
  };
  return ret;
};

angular.module('svg.testrect', []).directive('testrec', directiveGenerator);