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

Commit 6c6a408

Browse files
hannahhowardpetebacondarwin
authored andcommitted
fix ($compile): keep prototype properties for template URL directives
Previously, if a directive definition object was defined with methods like `compile` provided on the prototype rather than the instance, the Angular compiler failed to use these methods when the directive had a `templateURL`. This change ensures that these prototypical methods are not lost. This enables developers to define their directives using "classes" such as in CoffeeScript or ES6. Closes #10926
1 parent 0d42426 commit 6c6a408

File tree

2 files changed

+50
-2
lines changed

2 files changed

+50
-2
lines changed

src/ng/compile.js

+1-2
Original file line numberDiff line numberDiff line change
@@ -2155,8 +2155,7 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
21552155
afterTemplateChildLinkFn,
21562156
beforeTemplateCompileNode = $compileNode[0],
21572157
origAsyncDirective = directives.shift(),
2158-
// The fact that we have to copy and patch the directive seems wrong!
2159-
derivedSyncDirective = extend({}, origAsyncDirective, {
2158+
derivedSyncDirective = inherit(origAsyncDirective, {
21602159
templateUrl: null, transclude: null, replace: null, $$originalDirective: origAsyncDirective
21612160
}),
21622161
templateUrl = (isFunction(origAsyncDirective.templateUrl))

test/ng/compileSpec.js

+49
Original file line numberDiff line numberDiff line change
@@ -1109,6 +1109,29 @@ describe('$compile', function() {
11091109
expect(element.find('p').text()).toBe('Hello, world!');
11101110
});
11111111
});
1112+
1113+
it('should keep prototype properties on directive', function() {
1114+
module(function() {
1115+
function DirectiveClass() {
1116+
this.restrict = 'E';
1117+
this.template = "<p>{{value}}</p>";
1118+
}
1119+
1120+
DirectiveClass.prototype.compile = function() {
1121+
return function(scope, element, attrs) {
1122+
scope.value = "Test Value";
1123+
};
1124+
};
1125+
1126+
directive('templateUrlWithPrototype', valueFn(new DirectiveClass()));
1127+
});
1128+
1129+
inject(function($compile, $rootScope) {
1130+
element = $compile('<template-url-with-prototype><template-url-with-prototype>')($rootScope);
1131+
$rootScope.$digest();
1132+
expect(element.find("p")[0].innerHTML).toEqual("Test Value");
1133+
});
1134+
});
11121135
});
11131136

11141137

@@ -2028,6 +2051,32 @@ describe('$compile', function() {
20282051
});
20292052
});
20302053

2054+
it('should keep prototype properties on sync version of async directive', function() {
2055+
module(function() {
2056+
function DirectiveClass() {
2057+
this.restrict = 'E';
2058+
this.templateUrl = "test.html";
2059+
}
2060+
2061+
DirectiveClass.prototype.compile = function() {
2062+
return function(scope, element, attrs) {
2063+
scope.value = "Test Value";
2064+
};
2065+
};
2066+
2067+
directive('templateUrlWithPrototype', valueFn(new DirectiveClass()));
2068+
});
2069+
2070+
inject(function($compile, $rootScope, $httpBackend) {
2071+
$httpBackend.whenGET('test.html').
2072+
respond('<p>{{value}}</p>');
2073+
element = $compile('<template-url-with-prototype><template-url-with-prototype>')($rootScope);
2074+
$httpBackend.flush();
2075+
$rootScope.$digest();
2076+
expect(element.find("p")[0].innerHTML).toEqual("Test Value");
2077+
});
2078+
});
2079+
20312080
});
20322081

20332082

0 commit comments

Comments
 (0)