我要实现的基本上是一个“对ng重复完成渲染”处理程序。我能够检测它何时完成,但我不知道如何从它触发一个函数。

检查小提琴:http://jsfiddle.net/paulocoelho/BsMqq/3/

JS

var module = angular.module('testApp', [])
    .directive('onFinishRender', function () {
    return {
        restrict: 'A',
        link: function (scope, element, attr) {
            if (scope.$last === true) {
                element.ready(function () {
                    console.log("calling:"+attr.onFinishRender);
                    // CALL TEST HERE!
                });
            }
        }
    }
});

function myC($scope) {
    $scope.ta = [1, 2, 3, 4, 5, 6];
    function test() {
        console.log("test executed");
    }
}

HTML

<div ng-app="testApp" ng-controller="myC">
    <p ng-repeat="t in ta" on-finish-render="test()">{{t}}</p>
</div>

答: 工作小提琴从finishingmove: http://jsfiddle.net/paulocoelho/BsMqq/4/


当前回答

如果你需要在同一个控制器上为不同的ng-repeat调用不同的函数,你可以尝试这样做:

该指令:

var module = angular.module('testApp', [])
    .directive('onFinishRender', function ($timeout) {
    return {
        restrict: 'A',
        link: function (scope, element, attr) {
            if (scope.$last === true) {
            $timeout(function () {
                scope.$emit(attr.broadcasteventname ? attr.broadcasteventname : 'ngRepeatFinished');
            });
            }
        }
    }
});

在你的控制器中,用$on捕获事件:

$scope.$on('ngRepeatBroadcast1', function(ngRepeatFinishedEvent) {
// Do something
});

$scope.$on('ngRepeatBroadcast2', function(ngRepeatFinishedEvent) {
// Do something
});

在模板中使用多个ng-repeat

<div ng-repeat="item in collection1" on-finish-render broadcasteventname="ngRepeatBroadcast1">
    <div>{{item.name}}}<div>
</div>

<div ng-repeat="item in collection2" on-finish-render broadcasteventname="ngRepeatBroadcast2">
    <div>{{item.name}}}<div>
</div>

其他回答

其他解决方案在初始页面加载时工作得很好,但是从控制器调用$timeout是确保在模型更改时调用函数的唯一方法。这里有一个工作小提琴使用$timeout。对于你的例子,它将是:

.controller('myC', function ($scope, $timeout) {
$scope.$watch("ta", function (newValue, oldValue) {
    $timeout(function () {
       test();
    });
});

ngRepeat只会在行内容为新时计算指令,所以如果你从列表中删除项目,onFinishRender将不会触发。例如,尝试在这些fiddles发射中输入过滤器值。

我很惊讶在这个问题的答案中没有找到最简单的解决方案。 你要做的是在你的repeat元素(带有ngRepeat指令的元素)上添加一个ngInit指令,检查$last (ngRepeat在作用域中设置的一个特殊变量,它表明重复元素是列表中的最后一个)。如果$last为真,我们将呈现最后一个元素,并可以调用我们想要的函数。

ng-init="$last && test()"

HTML标记的完整代码如下:

<div ng-app="testApp" ng-controller="myC">
    <p ng-repeat="t in ta" ng-init="$last && test()">{{t}}</p>
</div>

除了你想要调用的作用域函数(在本例中是test),你不需要在应用程序中添加任何额外的JS代码,因为ngInit是由Angular.js提供的。只要确保你的测试函数在作用域中,这样就可以从模板中访问它:

$scope.test = function test() {
    console.log("test executed");
}

如果你不反对使用双重作用域道具,并且你正在编写一个唯一内容是repeat的指令,那么有一个相当简单的解决方案(假设你只关心初始渲染)。在link函数中:

const dereg = scope.$watch('$$childTail.$last', last => {
    if (last) {
        dereg();
        // do yr stuff -- you may still need a $timeout here
    }
});

这对于需要根据呈现列表成员的宽度或高度执行DOM manip的指令是有用的(我认为这是人们会问这个问题的最可能的原因),但它不像已经提出的其他解决方案那样通用。

如果您希望回调(即test())在DOM构造之后执行,但在浏览器呈现之前执行,请使用$evalAsync。这将防止闪烁- ref。

if (scope.$last) {
   scope.$evalAsync(attr.onFinishRender);
}

小提琴。

如果你真的想在渲染后调用回调函数,使用$timeout:

if (scope.$last) {
   $timeout(function() { 
      scope.$eval(attr.onFinishRender);
   });
}

我更喜欢$eval而不是一个事件。对于事件,我们需要知道事件的名称,并为该事件向控制器添加代码。使用$eval,控制器和指令之间的耦合更少。

很简单,我就是这么做的。

.directive('blockOnRender', function ($blockUI) { 返回{ 限制:' A ', 链接:函数(范围,元素,attrs) { If (scope.$first) { (blockUI.blockElement美元(元素).parent ()); } If (scope.$last) { (blockUI.unblockElement美元(元素).parent ()); } } }; })