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

Commit 7617c08

Browse files
committed
fix($compile): do not add <span> elements to root text nodes
BREAKING CHANGE: Text nodes at the root of transcluded content will no longer be wrapped into <span> elements. If there is a need for this <span> element to be present, then this should be added to the content to be transcluded. Before: ```html <div directive-that-will-transclude-the-content> I expect this content to e wrapped </div> ``` After: ```html <div directive-that-will-transclude-the-content> <span>I expect this content to e wrapped</span> </div> ```
1 parent a021a37 commit 7617c08

File tree

3 files changed

+53
-34
lines changed

3 files changed

+53
-34
lines changed

src/ng/compile.js

-13
Original file line numberDiff line numberDiff line change
@@ -1515,19 +1515,6 @@ function $CompileProvider($provide, $$sanitizeUriProvider) {
15151515
// modify it.
15161516
$compileNodes = jqLite($compileNodes);
15171517
}
1518-
1519-
var NOT_EMPTY = /\S+/;
1520-
1521-
// We can not compile top level text elements since text nodes can be merged and we will
1522-
// not be able to attach scope data to them, so we will wrap them in <span>
1523-
for (var i = 0, len = $compileNodes.length; i < len; i++) {
1524-
var domNode = $compileNodes[i];
1525-
1526-
if (domNode.nodeType === NODE_TYPE_TEXT && domNode.nodeValue.match(NOT_EMPTY) /* non-empty */) {
1527-
jqLiteWrapNode(domNode, $compileNodes[i] = document.createElement('span'));
1528-
}
1529-
}
1530-
15311518
var compositeLinkFn =
15321519
compileNodes($compileNodes, transcludeFn, $compileNodes,
15331520
maxPriority, ignoreDirective, previousCompileContext);

src/ng/directive/ngTransclude.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@
4747
* <div ng-controller="ExampleController">
4848
* <input ng-model="title" aria-label="title"> <br/>
4949
* <textarea ng-model="text" aria-label="text"></textarea> <br/>
50-
* <pane title="{{title}}">{{text}}</pane>
50+
* <pane title="{{title}}"><span>{{text}}</span></pane>
5151
* </div>
5252
* </file>
5353
* <file name="protractor.js" type="protractor">

test/ng/compileSpec.js

+52-20
Original file line numberDiff line numberDiff line change
@@ -372,27 +372,28 @@ describe('$compile', function() {
372372
expect($document.scope()).toBe($rootScope);
373373
}));
374374

375-
it('should wrap root text nodes in spans', inject(function($compile, $rootScope) {
376-
element = jqLite('<div>A&lt;a&gt;B&lt;/a&gt;C</div>');
377-
var text = element.contents();
378-
expect(text[0].nodeName).toEqual('#text');
379-
text = $compile(text)($rootScope);
380-
expect(text[0].nodeName).toEqual('SPAN');
381-
expect(element.find('span').text()).toEqual('A<a>B</a>C');
382-
}));
383-
384375

385-
it('should not wrap root whitespace text nodes in spans', function() {
376+
it('should not wrap root text nodes in spans', function() {
386377
element = jqLite(
387-
'<div> <div>A</div>\n ' + // The spaces and newlines here should not get wrapped
388-
'<div>B</div>C\t\n ' + // The "C", tabs and spaces here will be wrapped
378+
'<div> <div>A</div>\n ' +
379+
'<div>B</div>C\t\n ' +
389380
'</div>');
390381
$compile(element.contents())($rootScope);
391382
var spans = element.find('span');
392-
expect(spans.length).toEqual(1);
393-
expect(spans.text().indexOf('C')).toEqual(0);
383+
expect(spans.length).toEqual(0);
394384
});
395385

386+
387+
it('should be able to compile text nodes at the root', inject(function($rootScope) {
388+
element = jqLite('<div>Name: {{name}}<br />\nColor: {{color}}</div>');
389+
$rootScope.name = 'Lucas';
390+
$rootScope.color = 'blue';
391+
$compile(element.contents())($rootScope);
392+
$rootScope.$digest();
393+
expect(element.text()).toEqual('Name: Lucas\nColor: blue');
394+
}));
395+
396+
396397
it('should not leak memory when there are top level empty text nodes', function() {
397398
// We compile the contents of element (i.e. not element itself)
398399
// Then delete these contents and check the cache has been reset to zero
@@ -6595,8 +6596,8 @@ describe('$compile', function() {
65956596
$rootScope.x = 'root';
65966597
$rootScope.$apply();
65976598
expect(element.text()).toEqual('W:iso-1-2;T:root-2-3;');
6598-
expect(jqLite(element.find('span')[0]).text()).toEqual('T:root-2-3');
6599-
expect(jqLite(element.find('span')[1]).text()).toEqual(';');
6599+
expect(jqLite(jqLite(element.find('li')[1]).contents()[0]).text()).toEqual('T:root-2-3');
6600+
expect(jqLite(element.find('span')[0]).text()).toEqual(';');
66006601
});
66016602
});
66026603

@@ -6634,6 +6635,37 @@ describe('$compile', function() {
66346635
});
66356636

66366637

6638+
it('should not merge text elements from transcluded content', function() {
6639+
module(function() {
6640+
directive('foo', valueFn({
6641+
transclude: 'content',
6642+
template: '<div>This is before {{before}}. </div>',
6643+
link: function(scope, element, attr, ctrls, $transclude) {
6644+
var futureParent = element.children().eq(0);
6645+
$transclude(function(clone) {
6646+
futureParent.append(clone);
6647+
}, futureParent);
6648+
},
6649+
scope: true
6650+
}));
6651+
});
6652+
inject(function($rootScope, $compile) {
6653+
element = $compile('<div><div foo>This is after {{after}}</div></div>')($rootScope);
6654+
$rootScope.before = "BEFORE";
6655+
$rootScope.after = "AFTER";
6656+
$rootScope.$apply();
6657+
expect(element.text()).toEqual('This is before BEFORE. This is after AFTER');
6658+
6659+
$rootScope.before = "Not-Before";
6660+
$rootScope.after = "AfTeR";
6661+
$rootScope.$$childHead.before = "BeFoRe";
6662+
$rootScope.$$childHead.after = "Not-After";
6663+
$rootScope.$apply();
6664+
expect(element.text()).toEqual('This is before BeFoRe. This is after AfTeR');
6665+
});
6666+
});
6667+
6668+
66376669
it('should only allow one content transclusion per element', function() {
66386670
module(function() {
66396671
directive('first', valueFn({
@@ -6932,11 +6964,11 @@ describe('$compile', function() {
69326964
transclude: true,
69336965
replace: true,
69346966
scope: true,
6935-
template: '<div><span>I:{{$$transcluded}}</span><div ng-transclude></div></div>'
6967+
template: '<div><span>I:{{$$transcluded}}</span><span ng-transclude></span></div>'
69366968
};
69376969
});
69386970
});
6939-
inject(function(log, $rootScope, $compile) {
6971+
inject(function($rootScope, $compile) {
69406972
element = $compile('<div><div trans>T:{{$$transcluded}}</div></div>')($rootScope);
69416973
$rootScope.$apply();
69426974
expect(jqLite(element.find('span')[0]).text()).toEqual('I:');
@@ -6955,10 +6987,10 @@ describe('$compile', function() {
69556987
};
69566988
});
69576989
});
6958-
inject(function(log, $rootScope, $compile) {
6990+
inject(function($rootScope, $compile) {
69596991
element = $compile('<div trans>unicorn!</div>')($rootScope);
69606992
$rootScope.$apply();
6961-
expect(sortedHtml(element.html())).toEqual('<div ng-transclude=""><span>unicorn!</span></div>');
6993+
expect(sortedHtml(element.html())).toEqual('<div ng-transclude="">unicorn!</div>');
69626994
});
69636995
});
69646996

0 commit comments

Comments
 (0)