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

Commit e235f20

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

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
@@ -2499,6 +2505,12 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
24992505
$exceptionHandler(e);
25002506
}
25012507
}
2508+
if (isFunction(controllerInstance.$doCheck)) {
2509+
controllerInstance.$doCheck();
2510+
}
2511+
if (isFunction(controllerInstance.$doCheck)) {
2512+
controllerInstance.$doCheck();
2513+
}
25022514
if (isFunction(controllerInstance.$onDestroy)) {
25032515
controllerScope.$on('$destroy', function callOnDestroyHook() {
25042516
controllerInstance.$onDestroy();
@@ -3151,7 +3163,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
31513163
forEach(bindings, function initializeBinding(definition, scopeName) {
31523164
var attrName = definition.attrName,
31533165
optional = definition.optional,
3154-
mode = definition.mode, // @, =, or &
3166+
mode = definition.mode, // @, =, <, or &
31553167
lastValue,
31563168
parentGet, parentSet, compare, removeWatch;
31573169

@@ -3263,6 +3275,11 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
32633275
}
32643276
});
32653277

3278+
if (isFunction(destination.$doCheck)) {
3279+
var doCheckWatch = scope.$watch(triggerDoCheckHook);
3280+
removeWatchCollection.push(doCheckWatch);
3281+
}
3282+
32663283
function recordChanges(key, currentValue, previousValue) {
32673284
if (isFunction(destination.$onChanges) && currentValue !== previousValue) {
32683285
// If we have not already scheduled the top level onChangesQueue handler then do so now
@@ -3290,6 +3307,10 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
32903307
changes = undefined;
32913308
}
32923309

3310+
function triggerDoCheckHook() {
3311+
destination.$doCheck();
3312+
}
3313+
32933314
return {
32943315
initialChanges: initialChanges,
32953316
removeWatches: removeWatchCollection.length && function removeWatches() {

test/ng/compileSpec.js

+45
Original file line numberDiff line numberDiff line change
@@ -3828,6 +3828,51 @@ describe('$compile', function() {
38283828
});
38293829
});
38303830

3831+
describe('$doCheck', function() {
3832+
it('should call `$doCheck`, if provided, for each digest cycle, after $onChanges and $onInit', function() {
3833+
var log = [];
3834+
3835+
function TestController() { }
3836+
TestController.prototype.$doCheck = function() { log.push('$doCheck'); };
3837+
TestController.prototype.$onChanges = function() { log.push('$onChanges'); };
3838+
TestController.prototype.$onInit = function() { log.push('$onInit'); };
3839+
3840+
angular.module('my', [])
3841+
.component('dcc', {
3842+
controller: TestController,
3843+
bindings: { 'prop1': '<' }
3844+
});
3845+
3846+
module('my');
3847+
inject(function($compile, $rootScope) {
3848+
element = $compile('<dcc prop1="val"></dcc>')($rootScope);
3849+
expect(log).toEqual([
3850+
'$onChanges',
3851+
'$onInit',
3852+
'$doCheck'
3853+
]);
3854+
3855+
// Clear log
3856+
log = [];
3857+
3858+
$rootScope.$apply();
3859+
expect(log).toEqual([
3860+
'$doCheck',
3861+
'$doCheck'
3862+
]);
3863+
3864+
// Clear log
3865+
log = [];
3866+
3867+
$rootScope.$apply('val = 2');
3868+
expect(log).toEqual([
3869+
'$doCheck',
3870+
'$onChanges',
3871+
'$doCheck'
3872+
]);
3873+
});
3874+
});
3875+
});
38313876

38323877
describe('$onChanges', function() {
38333878

0 commit comments

Comments
 (0)