Angular.js 更新指令中的 SVG 模板

2024-01-21

不久前我问过“Angular.js 在指令中渲染 SVG 模板 https://stackoverflow.com/questions/19568226/angular-js-rendering-svg-templates-in-directives”,我用 SVG 节点替换了 Angular 在渲染模板时生成的 DOM 节点。我得到了一个回答,但我意识到我丢失了 Angular 的所有数据绑定。

请参阅 Plunkr(点击更新):http://plnkr.co/edit/HjOpqc?p=preview http://plnkr.co/edit/HjOpqc?p=preview

如何用 SVG 节点替换这些 DOM 节点,并保持角度绑定不变?我尝试使用 $compile 使其工作(就像我对常规 html 所做的那样),但它就是不起作用。

code:

var svgNS = 'http://www.w3.org/2000/svg';
app.directive('path', ngSvg('path'));
app.directive('g', ngSvg('g'));

function ngSvg(type) {
  return function($timeout, $compile) {
    return {
      restrict: 'E',
      link: function(scope, el, attr) {
        //skip nodes if they are already svg
        if (el[0].namespaceURI === svgNS) {
          return;
        }

        // I would expect the chunk of code below to work,
        // but it does not with ng-repeat

        // var newAttr = {};
        // _.each(el[0].attributes, function(at) {
        //   newAttr[at.nodeName] = at.value;
        // });

        // var path = makeNode(type, el, newAttr);
        // var parent = path.cloneNode(true);

        // $compile(parent)(scope);

        // var children = el.children();
        // $(parent).append(children);

        // $timeout(function() {
        //   el.replaceWith(parent);
        // })


        // this works for rendering, but does not update the svg elements
        // when update is clicked
        $timeout(function() {
          var newAttr = {};
          _.each(el[0].attributes, function(at) {
            newAttr[at.nodeName] = at.value;
          });

          var path = makeNode(type, el, newAttr);
          var parent = path.cloneNode(true);


          var children = el.children();
          $(parent).append(children);
          el.replaceWith(parent);
        });
      }
    }
  }
}

/* Create a shape node with the given settings. */
function makeNode(name, element, settings) {
  // var ns = 'http://www.w3.org/2000/svg';
  var node = document.createElementNS(svgNS, name);
  for (var attribute in settings) {
    var value = settings[attribute];
    if (value !== null && value !== null && !attribute.match(/\$/) &&
      (typeof value !== 'string' || value !== '')) {
      node.setAttribute(attribute, value);
    }
  }
  return node;
}

这个问题在 Angular 1.3 中得到了解决,这里是一些自定义 svg 指令的实现,其中包含您期望从 Angular 指令中获得的行为。现在对指令声明有特殊要求如下templateNamespace: 'svg'

另请注意,我正在覆盖一些保留的属性,例如,x and height关于<rect/>。要保留对这些的更多控制,您可以利用ng-attr像这样'<rect ng-attr-width="{{ ngWidth }}" />

JSFiddle 链接 http://jsfiddle.net/6etLazLy/

这里有一个习俗<rect/> and <circle/>

app.directive('ngRect', [function () {
    return {
        templateNamespace: 'svg',
        replace: true,
        template: '<rect ng-attr-width="{{ ngWidth }}" ng-attr-height="{{ ngHeight }}" ng-attr-x="{{ ngX }}" ng-attr-y="{{ ngY }}" ng-click="ngRectClick()"/>',
        scope: {
            'ngHeight': '=',
            'ngWidth': '='
        },
        link: function (scope, elem, attrs) {
            scope.ngRectClick = function() {
                console.log(elem);
            }
        }
    }
}]);

app.directive('ngCircle', [function () {
    return {
        templateNamespace: 'svg',
        replace: true,
        template: '<circle ng-attr-cx="{{ ngCx }}" ng-attr-cy="{{ ngCy }}" ng-attr-r="{{ ngR }}" ng-attr-fill="{{ ngFill }}" ng-click="ngCircleClick()"/>',
        scope: {
            'ngCx': '=',
            'ngCy': '=',
            'ngR': '=',
            'ngFill': '='
        },
        link: function (scope, elem, attrs) {
            scope.ngCircleClick = function() {
                console.log(elem);
            }
        }
    }
}]);
本文内容由网友自发贡献,版权归原作者所有,本站不承担相应法律责任。如您发现有涉嫌抄袭侵权的内容,请联系:hwhale#tublm.com(使用前将#替换为@)

Angular.js 更新指令中的 SVG 模板 的相关文章