Skip to content

Commit 69bbb16

Browse files
committed
fix($compile): prevent double attr interpolation w/ templateUrl
This fixes the issue that caused two attr interpolation observers to be registered for the same attribute as a result of isolate scope definition with attr (@) property for this attribute. Duplicate observers would then fight with each other updating the model. The issue occured only when this directive was used in a repeater because that's when we clone the template node which caused the two observers to point to two different sets of $attr instances. Closes angular#1166, angular#836
1 parent be50e07 commit 69bbb16

File tree

2 files changed

+34
-1
lines changed

2 files changed

+34
-1
lines changed

src/ng/compile.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -899,7 +899,7 @@ function $CompileProvider($provide) {
899899
origAsyncDirective = directives.shift(),
900900
// The fact that we have to copy and patch the directive seems wrong!
901901
derivedSyncDirective = extend({}, origAsyncDirective, {
902-
controller: null, templateUrl: null, transclude: null
902+
controller: null, templateUrl: null, transclude: null, scope: null
903903
});
904904

905905
$compileNode.html('');

test/ng/compileSpec.js

+33
Original file line numberDiff line numberDiff line change
@@ -1048,6 +1048,39 @@ describe('$compile', function() {
10481048
expect($exceptionHandler.errors).toEqual([]);
10491049
});
10501050
});
1051+
1052+
1053+
it('should resume delayed compilation without duplicates when in a repeater', function() {
1054+
// this is a test for a regression
1055+
// scope creation, isolate watcher setup, controller instantiation, etc should happen
1056+
// only once even if we are dealing with delayed compilation of a node due to templateUrl
1057+
// and the template node is in a repeater
1058+
1059+
var controllerSpy = jasmine.createSpy('controller');
1060+
1061+
module(function($compileProvider) {
1062+
$compileProvider.directive('delayed', valueFn({
1063+
controller: controllerSpy,
1064+
templateUrl: 'delayed.html',
1065+
scope: {
1066+
title: '@'
1067+
}
1068+
}));
1069+
});
1070+
1071+
inject(function($templateCache, $compile, $rootScope) {
1072+
$rootScope.coolTitle = 'boom!';
1073+
$templateCache.put('delayed.html', '<div>{{title}}</div>');
1074+
element = $compile(
1075+
'<div><div ng-repeat="i in [1,2]"><div delayed title="{{coolTitle + i}}"></div>|</div></div>'
1076+
)($rootScope);
1077+
1078+
$rootScope.$apply();
1079+
1080+
expect(controllerSpy.callCount).toBe(2);
1081+
expect(element.text()).toBe('boom!1|boom!2|');
1082+
});
1083+
});
10511084
});
10521085

10531086

0 commit comments

Comments
 (0)