ng-repeat完成后调用函数


249

我要实现的基本上是“重复完成渲染”处理程序。我能够检测到何时完成,但无法弄清楚如何从中触发功能。

检查小提琴: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>

:从finishmove的工作提琴:http : //jsfiddle.net/paulocoelho/BsMqq/4/


出于好奇,element.ready()这段代码的目的是什么?我的意思是..是您拥有某种jQuery插件,还是应该在元素准备就绪时触发它?
gion_13 2014年

1
可以使用内建指令(例如ng-init)来做到这一点
Semiomant 2015年


Answers:


400
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.onFinishRender);
                });
            }
        }
    }
});

请注意,我没有使用.ready(),而是将其包装在中$timeout$timeout确保在ng重复的元素真正完成渲染后$timeout执行(因为会在当前摘要周期的末尾执行- 并且也会在$apply内部调用,与不同setTimeout)。因此,ng-repeat完成后,我们$emit将向外部作用域(同级作用域和父作用域)发出一个事件。

然后在您的控制器中,可以使用以下命令进行捕获$on

$scope.$on('ngRepeatFinished', function(ngRepeatFinishedEvent) {
    //you also get the actual event object
    //do stuff, execute functions -- whatever...
});

使用html看起来像这样:

<div ng-repeat="item in items" on-finish-render="ngRepeatFinished">
    <div>{{item.name}}}<div>
</div>

10
+1,但我会在使用事件之前使用$ eval -减少耦合。请参阅我的答案以获取更多详细信息。
Mark Rajcok

1
@PigalevPavel我认为您与混淆$timeout(基本上是setTimeout+ Angular的$scope.$apply()setInterval$timeout每个if条件只会执行一次,并且会在下一个$digest周期开始时执行。有关JavaScript超时的更多信息,请参见:ejohn.org/blog/how-javascript-timers-work
holographic-principle

1
@finishingmove也许我听不懂:)但请参阅我在另一个ng-repeat中有ng-repeat。我想确定所有这些何时完成。当我将您的脚本与$ timeout一起用于父级ng-repeat时,一切正常。但是,如果我不使用$ timeout,则会在ng-repeat子项完成之前得到响应。我想知道为什么吗?并且我可以确定,如果我将您的脚本与$ timeout一起用于父级ng-repeat,那么当所有ng-repeats完成后,我总是会得到响应吗?
Pigalev Pavel

1
为什么要使用setTimeout()0延迟之类的东西是另一个问题,尽管我必须说,我从未在任何地方遇到过有关浏览器事件队列的实际规范,而单线程和的存在暗示了什么setTimeout()
rakslice 2013年

1
感谢您的回答。我有一个问题,为什么我们需要分配on-finish-render="ngRepeatFinished"?当我分配时on-finish-render="helloworld",它的工作原理相同。
Arnaud 2015年

89

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

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

小提琴

如果您确实要在渲染后调用回调,请使用$ timeout:

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

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


支票$last做什么?在文档中找不到它。它被删除了吗?
Erik Aigner 2013年

2
$last如果ng-repeat在元素上激活了@ErikAigner,则在作用域上定义。
Mark Rajcok 2013年

看来您的小提琴正在吸收不必要的东西$timeout
bbodenmiller 2014年

1
大。从性能的角度来看,发射/广播的成本更高,这应该是公认的答案。
Florin Vistig

29

到目前为止,给出的答案只会在第一次ng-repeat呈现时起作用,但是,如果您具有dynamic ng-repeat,则意味着您将要添加/删除/过滤项目,并且每次都需要通知您ng-repeat渲染后,这些解决方案将无法为您服务。

因此,如果您需要每次ng-repeat都收到重新渲染的通知,而不仅仅是第一次通知我,我已经找到了一种方法,这确实很“ hacky”,但是如果您知道自己是什么,它就可以正常工作在做。$filterng-repeat 使用其他任何方法之前,请先$filter使用此方法:

.filter('ngRepeatFinish', function($timeout){
    return function(data){
        var me = this;
        var flagProperty = '__finishedRendering__';
        if(!data[flagProperty]){
            Object.defineProperty(
                data, 
                flagProperty, 
                {enumerable:false, configurable:true, writable: false, value:{}});
            $timeout(function(){
                    delete data[flagProperty];                        
                    me.$emit('ngRepeatFinished');
                },0,false);                
        }
        return data;
    };
})

每次渲染时都会$emit调用此事件。ngRepeatFinishedng-repeat

如何使用它:

<li ng-repeat="item in (items|ngRepeatFinish) | filter:{name:namedFiltered}" >

ngRepeatFinish过滤器需要直接应用于ArrayObject在你的定义$scope,你可以申请之后其它过滤器。

如何不使用它:

<li ng-repeat="item in (items | filter:{name:namedFiltered}) | ngRepeatFinish" >

不要先应用其他过滤器,然后再应用ngRepeatFinish过滤器。

我什么时候应该使用这个?

如果要在列表完成渲染后将某些css样式应用于DOM中,因为您需要考虑已由重新渲染的DOM元素的新尺寸ng-repeat。(顺便说一句:这类操作应在指令内完成)

在处理ngRepeatFinished事件的函数中不可以做什么:

  • 不要$scope.$apply在该函数中执行,否则您将Angular陷入Angular无法检测到的无限循环中。

  • 不要使用它来更改$scope属性,因为这些更改直到下一个$digest循环才会反映在您的视图中,并且由于您无法执行更改,因此$scope.$apply它们将毫无用处。

“但是过滤器不应该那样使用!!”

不,他们不是,这是一个hack,如果您不喜欢它,请不要使用它。如果您知道完成同一件事的更好方法,请告诉我。

总结

这是一种技巧,以错误的方式使用它是危险的,仅在ng-repeat完成渲染后才将其用于应用样式,并且不会有任何问题。


Thx 4小费。无论如何,一个带有工作示例的小伙子将不胜感激。
holographix

2
不幸的是,这至少对最新的AngularJS 1.3.7无效。因此,我发现了另一种解决方法,而不是最好的解决方法,但是如果这对您来说必不可少的话,它将起作用。我所做的是每当我添加/更改/删除元素时,我总是将另一个虚拟元素添加到列表的末尾,因此,由于$last元素也已更改,因此ngRepeatFinish通过检查$last现在的简单指令将起作用。一旦ngRepeatFinish被调用,我就删除了虚拟物品。(我也用CSS隐藏了它,所以它不会短暂出现)
darklow

这个技巧不错,但并不完美;每次在活动中启动过滤器时,都会发送5次:-/
Sjeiti,2015年

下面的by Mark Rajcok完美地完成了ng-repeat的每一次重新渲染(例如,由于模型更改)。因此,不需要此解决方案。
Dzhu 2015年

1
@Josep你是对的-当项目被拼接/未移位时(即数组发生突变),它不起作用。我以前的测试可能是使用已重新分配的数组(使用sugarjs),因此它每次都有效...感谢您指出这一点
Dzhu 2015年

10

如果需要在同一控制器上为不同的ng-repeats调用不同的函数,则可以尝试如下操作:

指令:

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>

5

其他解决方案可以在初始页面加载时正常工作,但是从控制器调用$ timeout是确保在模型更改时调用函数的唯一方法。这是使用$ timeout 的有效提琴。以您的示例为例:

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

ngRepeat仅在行内容为新内容时才评估指令,因此,如果从列表中删除项目,则不会触发onFinishRender。例如,尝试在这些小提琴进入过滤器值EMIT


同样,当模型改变小提琴
John Harley

2
是什么ta$scope.$watch("ta",...
穆罕默德·阿德·扎希德

4

如果您不反对使用双美元范围的道具,并且正在编写仅重复内容的指令,则有一个非常简单的解决方案(假设您只关心初始渲染)。在链接功能中:

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

这对于以下情况很有用:您有一条指令需要根据呈现列表成员的宽度或高度进行DOM操作(我认为这是最可能有人问这个问题的原因),但它不是通用的作为已提出的其他解决方案。


1

过滤后的ngRepeat可以解决此问题,而Mutation事件可以解决此问题,但不建议使用(不立即替换)。

然后我想到了另一个简单的方法:

app.directive('filtered',function($timeout) {
    return {
        restrict: 'A',link: function (scope,element,attr) {
            var elm = element[0]
                ,nodePrototype = Node.prototype
                ,timeout
                ,slice = Array.prototype.slice
            ;

            elm.insertBefore = alt.bind(null,nodePrototype.insertBefore);
            elm.removeChild = alt.bind(null,nodePrototype.removeChild);

            function alt(fn){
                fn.apply(elm,slice.call(arguments,1));
                timeout&&$timeout.cancel(timeout);
                timeout = $timeout(altDone);
            }

            function altDone(){
                timeout = null;
                console.log('Filtered! ...fire an event or something');
            }
        }
    };
});

这会钩住父元素的Node.prototype方法,并带有一勾的$ timeout来监视连续的修改。

它的工作原理基本上是正确的,但是我确实遇到了altDone被调用两次的情况。

再次...将此指令添加到ngRepeat 的级。


到目前为止,这是效果最好的,但不是完美的。例如,我从70项增加到68项,但没有触发。
盖伊2015年

1
可行的方法是在上述代码中添加appendChild(因为我只使用了insertBeforeremoveChild)。
斯吉蒂2015年

1

很简单,这就是我的方法。

.directive('blockOnRender', function ($blockUI) {
    return {
        restrict: 'A',
        link: function (scope, element, attrs) {
            
            if (scope.$first) {
                $blockUI.blockElement($(element).parent());
            }
            if (scope.$last) {
                $blockUI.unblockElement($(element).parent());
            }
        }
    };
})


如果您拥有自己的$ blockUI服务,则此代码当然有效。
CPhelefu


0

我很惊讶没有看到这个问题的答案中最简单的解决方案。您要执行的操作是ngInit在重复元素(带有ngRepeat伪指令的元素)上添加指令以进行检查$last(范围中设置的特殊变量,ngRepeat该变量指示重复元素是列表中的最后一个)。如果$last为true,则渲染最后一个元素,然后可以调用所需的函数。

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由于ngInit要由Angular.js提供,因此除了要调用的作用域函数(在本例中为)之外,您的应用程序中不需要任何其他JS代码。只要确保您的test功能在作用域中,就可以从模板中对其进行访问:

$scope.test = function test() {
    console.log("test executed");
}
By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.