在指令中使用 ng-transclude 进行 ng-repeat

IT技术 javascript angularjs
2021-02-21 19:23:10

我想在内容更改时创建一个具有自定义行为的列表。我尝试为此创建一个指令,但我对如何将 ng-transclude 与 ng-repeat 指令结合起来有点迷茫。有人可以让我走上正轨吗?

网址:

<div ng-app="myApp">
  <div ng-controller="ctrl">
    <mylist items="myItem in items">
       <span class="etc">{{myItem}}</span>
    </mylist>
  </div>
</div>

Javascript:

angular.module('myApp', [])    

.controller('ctrl', function ($scope) {
  $scope.items = ['one', 'two', 'three'];
})    

.directive('mylist', function () {
  return {
    restrict:'E',
    transclude: 'element',
    replace: true,
    scope: true,
    template: [
      '<ul>',
        '<li ng-repeat="WhatGoesHere in items" ng-transclude></li>',
      '</ul>'
    ].join(''),
    link: function (scope, element, attr) {
      var parts = attr.items.split(' in ');
      var itemPart = parts[0];
      var itemsPart = parts[1];
      scope.$watch(itemsPart, function (value) {
        scope.items = value; 
      });      
    }
  }
});

我有一部分在这里工作

编辑:

标准:

  • 项目的模板必须在视图中定义,而不是在指令中定义,并且它必须能够访问子作用域中的项目属性。理想情况下,我想像在 ng-repeat 指令中那样定义它
  • 该指令必须有权访问该列表,以便我可以设置适当的监视和更改内容。如果可能的话,我希望能够轻松访问生成的 DOM 项目(我也可以用它element[0].querySelectorAll('ul>li')或其他东西来做,它只需要在 Chrome 上工作)。
  • 如果可能的话,我想重用 ng-repeat 指令中的逻辑,因为它已经做了很多我想要的。最好我不想复制代码。我只是想增强它的行为,而不是改变它
5个回答

自己解决了这个问题:

我可以在编译步骤 ( jsfiddle ) 中通过在编译ng-repeat模板时添加属性并将我的属性内容提供给它来完成。

网址:

<div ng-app="myApp">
  <div ng-controller="ctrl">
    <mylist element="myItem in items">{{myItem}}</mylist>
  </div>
</div>

Javascript:

var myApp = angular.module('myApp', [])

.controller('ctrl', function ($scope) {
  $scope.items = ['one', 'two', 'three'];
})

.directive('mylist', function ($parse) {
  return {
    restrict:'E',
    transclude: 'element',
    replace: true,
    scope: true,
    template: [
      '<ul>',
      '<li ng-transclude></li>',
      '</ul>'
    ].join(''),
    compile: function (tElement, tAttrs, transclude) {
      var rpt = document.createAttribute('ng-repeat');
      rpt.nodeValue = tAttrs.element;
      tElement[0].children[0].attributes.setNamedItem(rpt);
      return function (scope, element, attr) {
        var rhs = attr.element.split(' in ')[1];
        scope.items = $parse(rhs)(scope);
        console.log(scope.items);
      }        
    }
  }
});
这在当前的角度版本中不再起作用。请参阅此讨论:github.com/angular/angular.js/issues/7874
2021-04-24 19:23:10
items在父作用域中声明。您是否使用隔离范围?scope: {}代替scope: true
2021-05-01 19:23:10
这很棒!有一件事对我不起作用:在编译的返回函数中,$parse(rhs)(scope)总是返回undefined,但如果我这样做,我可以找到正确的变量$parse(rhs)(scope.$parent)你的为什么在子作用域上工作——没有items在父级中声明?
2021-05-03 19:23:10
@ulilicht 我认为就是答案
2021-05-17 19:23:10

实现此目的的替代方法如下。

索引.html:

<html ng-app='myApp'>

<head>
    <title>AngularJS Transclude within Repeat Within Directive</title>
    <script src="//ajax.googleapis.com/ajax/libs/angularjs/1.2.18/angular.min.js"></script>
    <script src='index.js'></script>
</head>

<body ng-controller='myController'>
    <people>Hello {{person.name}}</people>
    <button name="button" ng-click="changeRob()">Change Rob</button>
</body>
</html>

索引.js:

var myApp = angular.module( 'myApp', [] );

myApp.controller( 'myController', function( $scope ) {
    $scope.people = [
        { name: 'Rob'  },
        { name: 'Alex' },
        { name: 'John' }
    ];

    $scope.changeRob = function() {
        $scope.people[0].name = 'Lowe';
    }
});

myApp.directive( 'people', function() {
    return {
        restrict: 'E',

        transclude: true,
        template: '<div ng-repeat="person in people" transcope></div>',
    }
});

myApp.directive( 'transcope', function() {
    return {
        link: function( $scope, $element, $attrs, controller, $transclude ) {
            if ( !$transclude ) {
                throw minErr( 'ngTransclude' )( 'orphan',
                    'Illegal use of ngTransclude directive in the template! ' +
                    'No parent directive that requires a transclusion found. ' +
                    'Element: {0}',
                    startingTag( $element ));
            }
            var innerScope = $scope.$new();

            $transclude( innerScope, function( clone ) {
                $element.empty();
                $element.append( clone );
                $element.on( '$destroy', function() {
                    innerScope.$destroy();
                });
            });
        }
    };
}); 

在这个类似的 plunker 中查看它的运行情况基于这个漫长的 Github 问题讨论

不幸的是,其他答案不适用于最新版本的 angular(我检查过1.4),所以我认为分享我发现的这个 jsbin 有好处

var app = angular.module('app', [])
  .controller('TestCtrl', function($scope) {
    $scope.myRecords = ['foo', 'bar', 'baz'];
  });

app.directive('myDirective', function($compile) {
  var template = '<div id="inner-transclude" ng-repeat="record in records"></div>';

  return {
    scope: {
      records: '='
    },
    restrict: 'A',
    compile: function(ele) {
      var transclude = ele.html();
      ele.html('');

      return function(scope, elem) {
        var tpl = angular.element(template);
        tpl.append(transclude);

        $compile(tpl)(scope);

        elem.append(tpl);
      };
    }
  };
});
<script src="https://ajax.googleapis.com/ajax/libs/angularjs/1.4.0/angular.js"></script>


<div ng-app="app" ng-controller="TestCtrl">
  <div my-directive records="myRecords">
    ?: {{record}}
  </div>

</div>

我不得不做很多额外的工作,但这个答案看起来基本上对我有用。
2021-05-06 19:23:10
这对我来说非常有效,也很简单……谢谢!
2021-05-07 19:23:10

转置不是必需的,因为items包含我们渲染模板所需的内容。换句话说,元素内部没有任何东西——即 <mylist>nothing new here we need to transclude</mylist>. 看来 Angular 也会为我们做 $watching。

.directive('mylist', function () {
  return {
    restrict:'E',
    replace: true,
    scope: true,
    template: [
      '<ul>',
      '<li ng-repeat="myItem in items">{{myItem}}</li>',
      '</ul>'
    ].join('')
  }
});

HTML:

<mylist></mylist>

小提琴

请注意,创建新范围是可选的,因此您可以注释掉这一行:

//scope: true,

更新:您可以选择创建一个隔离范围:

scope: { items: '='},

HTML:

<mylist items=items></mylist>

小提琴

更新 2:基于 Jan 提供的附加信息:

项目的模板必须在视图中定义...我想重用ng-repeat指令中的逻辑

好的,让我们把它全部放在视图中,并使用 ng-repeat:

<ul mylist>
  <li ng-repeat="myItem in items">
    <span class="etc">{{myItem}}</span>
   </li>
</ul>

它 [指令] 必须有权访问子作用域中的项属性...该指令必须有权访问列表,以便我可以设置适当的监视并更改内容

按照您的原始小提琴,我们将使用普通的子作用域(即子作用域将原型继承自父作用域):scope: true,这将确保指令可以访问在控制器范围内定义的属性,例如,items

访问生成的 DOM 项

指令的链接函数有一个element参数。所以在上面的 HTML 中, element 将被设置为<ul>元素。所以我们可以访问所有的 DOM 元素。例如,element.find('li')element.children()在下面引用的小提琴中,我有 $watchitems数组。$watch 回调可以访问element,因此您可以访问生成的 DOM 项。回调记录element.children()到控制台。

小提琴

总之,要将自定义行为添加到列表中,只需将指令放入 ul 或 ol 上即可。

里面还有什么——只是文本,还是其他 $scope 属性?
2021-04-26 19:23:10
您可以将您需要的内容放入指令的模板中,而不是嵌入吗?如果没有,我将需要查看您尝试嵌入的内容的示例,以进一步帮助您。此外,您可以items使用$watch 观察整个属性,但将第三个参数设置为 true 以进行“深度检查”/对象相等性,而不仅仅是引用检查: scope.$watch('items', function(newVal) { ... }, true);
2021-04-26 19:23:10
我可以将视图逻辑放在指令中,也可以将 DOM 操作放在控制器中,这都是不好的做法。我需要一个模仿 ng-repeat 指令的指令,但我想添加额外的行为。我的用例在这里:github.com/Janpotoms/google-reader-notifier/blob/master/app/...在里面寻找#grn-items-list。但这没关系。我想要一个可重用的组件
2021-04-27 19:23:10
你不明白,我里面的项目模板<mylist></mylist>比我这里给出的例子复杂得多
2021-05-20 19:23:10
看,我在现实生活中的列表是一个包含大对象的列表,这些项目包含很多范围属性。我只是简化了我的例子。我想要一个指令,它在具有隔离作用域的 ul>li 结构上充当 ng-repeat,这样我就可以观察 items 属性并在它更改时在列表上执行操作。例如,加载更多项目,更改滚动行为等
2021-05-21 19:23:10

我遇到了同样的问题,最终添加了一些代码,ng-transclude以便它允许观察和接受来自使用父级的自定义数据。

  • 这样嵌入的数据可以访问祖父范围
  • 并且还可以访问给它的 ng-repeat 数据。

用法

祖父母.js

<div>{{ $ctrl.listName }}</div.

<my-list items="$ctrl.movies">
   <div>From context: {{ name }}</div>
   <div>From grandparent: {{ $ctrl.listName }}</div>
</my-list>

Parent.js (MyList)

<li ng-repeat="item in $ctrl.items track by item.id">
  <cr-transclude context="item"></cr-transclude>
</li>

ng-transclude 的代码更改

return function ngTranscludePostLink(
   ...
  ) {
  let context = null;
  let childScope = null;
  ...
  $scope.$watch($attrs.context, (newVal, oldVal) => {
    context = newVal;
    updateScope(childScope, context);
  });
  ...
  $transclude(ngTranscludeCloneAttachFn, null, slotName);
  ...
  function ngTranscludeCloneAttachFn(clone, transcludedScope) {
     ...                                 
     $element.append(clone);
     childScope = transcludedScope;
     updateScope(childScope, context);
     ...
  }
  ...
  function updateScope(scope, varsHash) {
    if (!scope || !varsHash) {
      return;
    }
    angular.extend(scope, varsHash);
  }
}

完整代码在Github

可以在Codesandbox 中看到一个工作示例