我尝试实施角度ng-repeat指令,但我不明白为什么这段代码不能正常工作。
.directive("myRepeat", function() {
return {
transclude: "element",
priority: 1000,
compile: function(tElem, tAttrs) {
var myLoop = tAttrs.myRepeat,
match = myLoop.match(/^\s*(.+)+in\s+(.*?)\s*(\s+track\s+by\s+(.+)\s*)?$/),
indexString = match[1],
collectionString = match[2],
parent = tElem.parent();
return function($scope, iElem, iAttrs, controller, $transclude) {
$scope.$watchCollection(collectionString, function(newCollection) {
var i, block, elements = [];
// check if elements have already been rendered
if (elements.length) {
// if so remove them from DOM, and destroy their scope
for (i = 0; i < elements.length; i++) {
elements[i].el.remove();
elements[i].scope.$destroy();
}
elements = [];
}
for (i = 0; i < newCollection.length; i++) {
$transclude(function(clone, scope) {
scope[indexString] = newCollection[i];
parent.append(clone);
block = {};
block.el = clone;
block.scope = scope;
elements.push(block);
});
}
});
}
}
}
})
和HTML片段
<ul ng-controller="MyCtrl">
<li my-repeat="city in cities">{{city.name}}</li>
</ul>
我的问题是LI元素呈现正常,但它们不包含城市名称。请解释我为什么会这样。我理解如何在原始情况下进行工作转换,当我们使用带有ng-transclude的元素的模板时,在我们的指令定义中指定transclude:true,但我不明白如何使用transclude:&#34; element& #34 ;. 附:对不起我的英语不好。我是初学者:)
答案 0 :(得分:1)
我注意到当我将indexString写入控制台时,它的索引不正确。更改:match = myLoop.match(/^\s*(.+)+in\s+(.*?)\s*(\s+track\s+by\s+(.+)\s*)?$/)
至match = myLoop.split(' ')
适用于我的完整代码:
var app = angular.module('app', []);
app.controller("MyCtrl", function($scope){
$scope.cities = [{
name:'a'
}, {name: 'b'},
{name: 'c'}]
})
app.directive("myRepeat", function() {
return {
transclude: "element",
priority: 1000,
compile: function(tElem, tAttrs) {
var myLoop = tAttrs.myRepeat,
match = myLoop.split(' '),
indexString = match[0],
collectionString = match[2],
parent = tElem.parent();
console.log("match: " + match);
console.log("index string: " + indexString);
console.log("coll: " + collectionString);
var elements = [];
return function($scope, iElem, iAttrs, controller, $transclude) {
$scope.$watchCollection(collectionString, function(newCollection) {
var i;
// check if elements have already been rendered
if (elements.length) {
// if so remove them from DOM, and destroy their scope
for (i = 0; i < elements.length; i++) {
elements[i].el.remove();
elements[i].scope.$destroy();
}
elements = [];
}
for (i = 0; i < newCollection.length; i++) {
$transclude(function(clone, scope) {
scope[indexString] = newCollection[i];
parent.append(clone);
block = {};
block.el = clone;
block.scope = scope;
elements.push(block);
});
}
});
}
}
}
})