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

Commit c90abf0

Browse files
committed
Changed the angular.compile(element)(scope[, cloneAttachNode])
1 parent cdc093a commit c90abf0

10 files changed

+57
-44
lines changed

CHANGELOG.md

+1-1
Original file line numberDiff line numberDiff line change
@@ -12,7 +12,7 @@
1212
recommended way to deal with initializing scope is to put it in the root constructor controller.
1313
To migrate simply remove the call to $init() and move any code you had before $init() to the
1414
root controller.
15-
- Change API angular.compile(..) to angular.compile(element)([scope], [element/true])
15+
- Change API angular.compile(..) to angular.compile(element)([scope], [cloneAttachFn])
1616

1717

1818
<a name="0.9.11"><a/>

docs/angular.element.ngdoc

-4
Original file line numberDiff line numberDiff line change
@@ -44,10 +44,6 @@ raw DOM references.
4444
version:
4545

4646
- `scope()` - retrieves the current angular scope of the element.
47-
- `cloneNode()` - Clones the current node, ensuring identical structure. This is important since
48-
the `clone()` method implemented by jQuery under some circumstances changes the DOM
49-
structure, which then prevents proper application of compiled template to the cloned node.
50-
__Always use `cloneNode()` when cloning previously compiled templates.__
5147

5248
@param {string|DOMElement} element HTML string or DOMElement to be wrapped into jQuery.
5349
@returns {Object} jQuery object.

src/Angular.js

+10-10
Original file line numberDiff line numberDiff line change
@@ -803,17 +803,18 @@ function merge(src, dst) {
803803
</pre>
804804
*
805805
* @param {string|DOMElement} element Element or HTML to compile into a template function.
806-
* @returns {function([scope][, element])} a template function which is used to bind element
806+
* @returns {function([scope][, cloneAttachFn])} a template function which is used to bind element
807807
* and scope. Where:
808808
*
809809
* * `scope` - {@link angular.scope scope} A scope to bind to. If none specified, then a new
810810
* root scope is created.
811-
* * `element` - {@link angular.element element} Element to use as the template. If none
812-
* specified then reuse the element from `angular.compile(element)`. If `true`
813-
* then clone the `angular.compile(element)`. The element must be either the same
814-
* element as `angular.compile(element)` or an identical clone to
815-
* `angular.compile(element)`. Using an element with differnt structure will cause
816-
* unpredictable behavior.
811+
* * `cloneAttachFn` - If `cloneAttachFn` is provided, then the link function will clone the
812+
* `template` and call the `cloneAttachFn` allowing the caller to attach the
813+
* clonned elements to the DOM at the approriate place. The `cloneAttachFn` is
814+
* called as: <br/> `cloneAttachFn(clonedElement, scope)`:
815+
*
816+
* * `clonedElement` - is a clone of the originale `element` passed into the compiler.
817+
* * `scope` - is the current scope with which the linking function is working with.
817818
*
818819
* Calling the template function returns object: `{scope:?, view:?}`, where:
819820
*
@@ -1006,7 +1007,7 @@ function toKeyValue(obj) {
10061007
function angularInit(config){
10071008
if (config.autobind) {
10081009
// TODO default to the source of angular.js
1009-
var scope = compile(window.document)(null, createScope({'$config':config})),
1010+
var scope = compile(window.document)(createScope({'$config':config})).scope,
10101011
$browser = scope.$service('$browser');
10111012

10121013
if (config.css)
@@ -1048,8 +1049,7 @@ function bindJQuery(){
10481049
if (jQuery) {
10491050
jqLite = jQuery;
10501051
extend(jQuery.fn, {
1051-
scope: JQLitePrototype.scope,
1052-
cloneNode: JQLitePrototype.cloneNode
1052+
scope: JQLitePrototype.scope
10531053
});
10541054
} else {
10551055
jqLite = jqLiteWrap;

src/Compiler.js

+9-6
Original file line numberDiff line numberDiff line change
@@ -93,14 +93,17 @@ Compiler.prototype = {
9393
}
9494
}
9595
template = this.templatize(templateElement, index, 0) || new Template();
96-
return function(scope, element){
97-
scope = scope || createScope();
98-
element = element === true
99-
? templateElement.cloneNode()
100-
: (element ? jqLite(element) : templateElement);
96+
return function(scope, cloneConnectFn){
97+
// important!!: we must call our jqLite.clone() since the jQuery one is trying to be smart
98+
// and sometimes changes the structure of the DOM.
99+
var element = cloneConnectFn
100+
? JQLitePrototype.clone.call(templateElement) // IMPORTAN!!!
101+
: templateElement;
102+
scope = scope || createScope();
101103
element.data($$scope, scope);
102-
template.attach(element, scope);
103104
scope.$element = element;
105+
(cloneConnectFn||noop)(element, scope);
106+
template.attach(element, scope);
104107
scope.$eval();
105108
return {scope:scope, view:element};
106109
};

src/jqLite.js

+6-3
Original file line numberDiff line numberDiff line change
@@ -72,7 +72,7 @@ function JQLite(element) {
7272
}
7373
}
7474

75-
function JQLiteCloneNode(element) {
75+
function JQLiteClone(element) {
7676
return element.cloneNode(true);
7777
}
7878

@@ -370,12 +370,15 @@ forEach({
370370
return element.parentNode || null;
371371
},
372372

373+
next: function(element) {
374+
return element.nextSibling;
375+
},
376+
373377
find: function(element, selector) {
374378
return element.getElementsByTagName(selector);
375379
},
376380

377-
clone: JQLiteCloneNode,
378-
cloneNode: JQLiteCloneNode
381+
clone: JQLiteClone
379382
}, function(fn, name){
380383
/**
381384
* chaining functions

src/widgets.js

+15-14
Original file line numberDiff line numberDiff line change
@@ -790,10 +790,10 @@ var ngSwitch = angularWidget('ng:switch', function (element){
790790
forEach(cases, function(switchCase){
791791
if (!found && switchCase.when(childScope, value)) {
792792
found = true;
793-
var caseElement = switchCase.element.cloneNode();
794-
element.append(caseElement);
795793
childScope.$tryEval(switchCase.change, element);
796-
switchCase.template(childScope, caseElement);
794+
switchCase.template(childScope, function(caseElement){
795+
element.append(caseElement);
796+
});
797797
}
798798
});
799799
});
@@ -886,11 +886,11 @@ angularWidget('a', function() {
886886
</doc:scenario>
887887
</doc:example>
888888
*/
889-
angularWidget("@ng:repeat", function(expression, element){
889+
angularWidget('@ng:repeat', function(expression, element){
890890
element.removeAttr('ng:repeat');
891-
element.replaceWith(jqLite("<!-- ng:repeat: " + expression + " --!>"));
891+
element.replaceWith(jqLite('<!-- ng:repeat: ' + expression + ' --!>'));
892892
var linker = this.compile(element);
893-
return function(reference){
893+
return function(iterStartElement){
894894
var match = expression.match(/^\s*(.+)\s+in\s+(.*)\s*$/),
895895
lhs, rhs, valueIdent, keyIdent;
896896
if (! match) {
@@ -910,10 +910,9 @@ angularWidget("@ng:repeat", function(expression, element){
910910
var children = [], currentScope = this;
911911
this.$onEval(function(){
912912
var index = 0,
913-
cloneElement,
914913
childCount = children.length,
915-
lastElement = reference,
916-
collection = this.$tryEval(rhs, reference),
914+
lastIterElement = iterStartElement,
915+
collection = this.$tryEval(rhs, iterStartElement),
917916
is_array = isArray(collection),
918917
collectionLength = 0,
919918
childScope,
@@ -934,6 +933,7 @@ angularWidget("@ng:repeat", function(expression, element){
934933
childScope = children[index];
935934
childScope[valueIdent] = collection[key];
936935
if (keyIdent) childScope[keyIdent] = key;
936+
lastIterElement = childScope.$element;
937937
} else {
938938
// grow children
939939
childScope = createScope(currentScope);
@@ -943,21 +943,22 @@ angularWidget("@ng:repeat", function(expression, element){
943943
childScope.$position = index == 0
944944
? 'first'
945945
: (index == collectionLength - 1 ? 'last' : 'middle');
946-
lastElement.after(cloneElement = element.cloneNode());
947-
cloneElement.attr('ng:repeat-index', index);
948-
linker(childScope, cloneElement);
949946
children.push(childScope);
947+
linker(childScope, function(clone){
948+
clone.attr('ng:repeat-index', index);
949+
lastIterElement.after(clone);
950+
lastIterElement = clone;
951+
});
950952
}
951953
childScope.$eval();
952-
lastElement = childScope.$element;
953954
index ++;
954955
}
955956
}
956957
// shrink children
957958
while(children.length > index) {
958959
children.pop().$element.remove();
959960
}
960-
}, reference);
961+
}, iterStartElement);
961962
};
962963
});
963964

test/AngularSpec.js

+5-3
Original file line numberDiff line numberDiff line change
@@ -369,8 +369,10 @@ describe('angular', function(){
369369
var scope = angular.scope();
370370
var template = jqLite('<div>{{greeting = "hello world"}}</div>');
371371
var templateFn = angular.compile(template);
372-
var templateClone = template.cloneNode();
373-
mvc = templateFn(scope, templateClone);
372+
var templateClone = template.clone();
373+
mvc = templateFn(scope, function(clone){
374+
templateClone = clone;
375+
});
374376
expect(template.text()).toEqual('');
375377
expect(mvc.view.text()).toEqual('hello world');
376378
expect(mvc.view).toEqual(templateClone);
@@ -380,7 +382,7 @@ describe('angular', function(){
380382
it('should link to cloned node and create scope', function(){
381383
var scope = angular.scope();
382384
var template = jqLite('<div>{{greeting = "hello world"}}</div>');
383-
mvc = angular.compile(template)(scope, true);
385+
mvc = angular.compile(template)(scope, noop);
384386
expect(template.text()).toEqual('');
385387
expect(mvc.view.text()).toEqual('hello world');
386388
expect(mvc.scope.greeting).toEqual('hello world');

test/BinderSpec.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -417,7 +417,7 @@ describe('Binder', function(){
417417
});
418418

419419
it('BindClassEvenOdd', function(){
420-
var x = this.compile('<div><div ng:repeat="i in [0,1]" ng:class-even="\'e\'" ng:class-odd="\'o\'"/></div>');
420+
var x = this.compile('<div><div ng:repeat="i in [0,1]" ng:class-even="\'e\'" ng:class-odd="\'o\'"></div></div>');
421421
x.scope.$eval();
422422
var d1 = jqLite(x.view[0].childNodes[1]);
423423
var d2 = jqLite(x.view[0].childNodes[2]);

test/CompilerSpec.js

+2-2
Original file line numberDiff line numberDiff line change
@@ -47,7 +47,7 @@ describe('compiler', function(){
4747
};
4848
var template = compiler.compile(e);
4949
expect(log).toEqual("found");
50-
scope = template(angular.scope(), e).scope;
50+
scope = template(angular.scope()).scope;
5151
expect(e.hasClass('ng-directive')).toEqual(true);
5252
expect(log).toEqual("found:init");
5353
});
@@ -84,7 +84,7 @@ describe('compiler', function(){
8484
var template = this.compile(element);
8585
return function(marker) {
8686
this.$onEval(function() {
87-
marker.after(template(angular.scope(), true).view);
87+
marker.after(template(angular.scope(), noop).view);
8888
});
8989
};
9090
};

test/jqLiteSpec.js

+8
Original file line numberDiff line numberDiff line change
@@ -300,6 +300,14 @@ describe('jqLite', function(){
300300
expect(element.parent().length).toEqual(0);
301301
});
302302
});
303+
describe('next', function(){
304+
it('should return next sibling', function(){
305+
var element = jqLite('<div><b>b</b><i>i</i></div>');
306+
var b = element.find('b');
307+
var i = element.find('i');
308+
expect(b.next()).toJqEqual([i]);
309+
});
310+
});
303311
describe('find', function(){
304312
it('should find child by name', function(){
305313
var root = jqLite('<div><div>text</div></div>');

0 commit comments

Comments
 (0)