Skip to content
This repository was archived by the owner on Apr 12, 2024. It is now read-only.

Commit de59ca7

Browse files
zbjornsonpetebacondarwin
authored andcommitted
feat($compile): backport $doCheck
Backuport ngDoCheck from Angular 2. Closes #14656
1 parent fa3214c commit de59ca7

File tree

2 files changed

+67
-1
lines changed

2 files changed

+67
-1
lines changed

src/ng/compile.js

+22-1
Original file line numberDiff line numberDiff line change
@@ -300,6 +300,12 @@
300300
* `changesObj` is a hash whose keys are the names of the bound properties that have changed, and the values are an
301301
* object of the form `{ currentValue, previousValue, isFirstChange() }`. Use this hook to trigger updates within a
302302
* component such as cloning the bound value to prevent accidental mutation of the outer value.
303+
* * `$doCheck()` - Called on each turn of the digest cycle. Provides an opportunity to detect and act on
304+
* changes. Any actions that you wish to take in response to the changes that you detect must be
305+
* invoked from this hook; implementing this has no effect on when `$onChanges` is called. For example, this hook
306+
* could be useful if you wish to perform a deep equality check, or to check a Date object, changes to which would not
307+
* be detected by Angular's change detector and thus not trigger `$onChanges`. This hook is invoked with no arguments;
308+
* if detecting changes, you must store the previous value(s) for comparison to the current values.
303309
* * `$onDestroy()` - Called on a controller when its containing scope is destroyed. Use this hook for releasing
304310
* external resources, watches and event handlers. Note that components have their `$onDestroy()` hooks called in
305311
* the same order as the `$scope.$broadcast` events are triggered, which is top down. This means that parent
@@ -2512,6 +2518,12 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
25122518
$exceptionHandler(e);
25132519
}
25142520
}
2521+
if (isFunction(controllerInstance.$doCheck)) {
2522+
controllerInstance.$doCheck();
2523+
}
2524+
if (isFunction(controllerInstance.$doCheck)) {
2525+
controllerInstance.$doCheck();
2526+
}
25152527
if (isFunction(controllerInstance.$onDestroy)) {
25162528
controllerScope.$on('$destroy', function callOnDestroyHook() {
25172529
controllerInstance.$onDestroy();
@@ -3158,7 +3170,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
31583170
forEach(bindings, function initializeBinding(definition, scopeName) {
31593171
var attrName = definition.attrName,
31603172
optional = definition.optional,
3161-
mode = definition.mode, // @, =, or &
3173+
mode = definition.mode, // @, =, <, or &
31623174
lastValue,
31633175
parentGet, parentSet, compare, removeWatch;
31643176

@@ -3270,6 +3282,11 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
32703282
}
32713283
});
32723284

3285+
if (isFunction(destination.$doCheck)) {
3286+
var doCheckWatch = scope.$watch(triggerDoCheckHook);
3287+
removeWatchCollection.push(doCheckWatch);
3288+
}
3289+
32733290
function recordChanges(key, currentValue, previousValue) {
32743291
if (isFunction(destination.$onChanges) && currentValue !== previousValue) {
32753292
// If we have not already scheduled the top level onChangesQueue handler then do so now
@@ -3297,6 +3314,10 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
32973314
changes = undefined;
32983315
}
32993316

3317+
function triggerDoCheckHook() {
3318+
destination.$doCheck();
3319+
}
3320+
33003321
return {
33013322
initialChanges: initialChanges,
33023323
removeWatches: removeWatchCollection.length && function removeWatches() {

test/ng/compileSpec.js

+45
Original file line numberDiff line numberDiff line change
@@ -3816,6 +3816,51 @@ describe('$compile', function() {
38163816
});
38173817
});
38183818

3819+
describe('$doCheck', function() {
3820+
it('should call `$doCheck`, if provided, for each digest cycle, after $onChanges and $onInit', function() {
3821+
var log = [];
3822+
3823+
function TestController() { }
3824+
TestController.prototype.$doCheck = function() { log.push('$doCheck'); };
3825+
TestController.prototype.$onChanges = function() { log.push('$onChanges'); };
3826+
TestController.prototype.$onInit = function() { log.push('$onInit'); };
3827+
3828+
angular.module('my', [])
3829+
.component('dcc', {
3830+
controller: TestController,
3831+
bindings: { 'prop1': '<' }
3832+
});
3833+
3834+
module('my');
3835+
inject(function($compile, $rootScope) {
3836+
element = $compile('<dcc prop1="val"></dcc>')($rootScope);
3837+
expect(log).toEqual([
3838+
'$onChanges',
3839+
'$onInit',
3840+
'$doCheck'
3841+
]);
3842+
3843+
// Clear log
3844+
log = [];
3845+
3846+
$rootScope.$apply();
3847+
expect(log).toEqual([
3848+
'$doCheck',
3849+
'$doCheck'
3850+
]);
3851+
3852+
// Clear log
3853+
log = [];
3854+
3855+
$rootScope.$apply('val = 2');
3856+
expect(log).toEqual([
3857+
'$doCheck',
3858+
'$onChanges',
3859+
'$doCheck'
3860+
]);
3861+
});
3862+
});
3863+
});
38193864

38203865
describe('$onChanges', function() {
38213866

0 commit comments

Comments
 (0)