Skip to content

Commit d277442

Browse files
committed
fix(change_detection): do not coalesce records with different directive indices
1 parent 8681f79 commit d277442

File tree

2 files changed

+44
-13
lines changed

2 files changed

+44
-13
lines changed

modules/angular2/src/change_detection/coalesce.ts

+13-2
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
import {isPresent} from 'angular2/src/facade/lang';
1+
import {isPresent, isBlank} from 'angular2/src/facade/lang';
22
import {List, ListWrapper, Map} from 'angular2/src/facade/collection';
33
import {RecordType, ProtoRecord} from './proto_record';
44

@@ -45,11 +45,22 @@ function _selfRecord(r: ProtoRecord, contextIndex: number, selfIndex: number): P
4545

4646
function _findMatching(r: ProtoRecord, rs: List<ProtoRecord>) {
4747
return ListWrapper.find(rs, (rr) => rr.mode !== RecordType.DIRECTIVE_LIFECYCLE &&
48-
rr.mode === r.mode && rr.funcOrValue === r.funcOrValue &&
48+
_sameDirIndex(rr, r) && rr.mode === r.mode &&
49+
rr.funcOrValue === r.funcOrValue &&
4950
rr.contextIndex === r.contextIndex && rr.name === r.name &&
5051
ListWrapper.equals(rr.args, r.args));
5152
}
5253

54+
function _sameDirIndex(a: ProtoRecord, b: ProtoRecord): boolean {
55+
var di1 = isBlank(a.directiveIndex) ? null : a.directiveIndex.directiveIndex;
56+
var ei1 = isBlank(a.directiveIndex) ? null : a.directiveIndex.elementIndex;
57+
58+
var di2 = isBlank(b.directiveIndex) ? null : b.directiveIndex.directiveIndex;
59+
var ei2 = isBlank(b.directiveIndex) ? null : b.directiveIndex.elementIndex;
60+
61+
return di1 === di2 && ei1 === ei2;
62+
}
63+
5364
function _replaceIndices(r: ProtoRecord, selfIndex: number, indexMap: Map<any, any>) {
5465
var args = ListWrapper.map(r.args, (a) => _map(indexMap, a));
5566
var contextIndex = _map(indexMap, r.contextIndex);

modules/angular2/test/change_detection/coalesce_spec.ts

+31-11
Original file line numberDiff line numberDiff line change
@@ -1,13 +1,21 @@
11
import {ddescribe, describe, it, iit, xit, expect, beforeEach, afterEach} from 'angular2/test_lib';
2+
import {isBlank} from 'angular2/src/facade/lang';
23

34
import {coalesce} from 'angular2/src/change_detection/coalesce';
45
import {RecordType, ProtoRecord} from 'angular2/src/change_detection/proto_record';
6+
import {DirectiveIndex} from 'angular2/src/change_detection/directive_record';
57

68
export function main() {
7-
function r(funcOrValue, args, contextIndex, selfIndex, lastInBinding = false,
8-
mode = RecordType.PROPERTY, name = "name") {
9-
return new ProtoRecord(mode, name, funcOrValue, args, null, contextIndex, null, selfIndex, null,
10-
null, lastInBinding, false);
9+
function r(
10+
funcOrValue, args, contextIndex, selfIndex,
11+
{lastInBinding, mode, name,
12+
directiveIndex}: {lastInBinding?: any, mode?: any, name?: any, directiveIndex?: any} = {}) {
13+
if (isBlank(lastInBinding)) lastInBinding = false;
14+
if (isBlank(mode)) mode = RecordType.PROPERTY;
15+
if (isBlank(name)) name = "name";
16+
if (isBlank(directiveIndex)) directiveIndex = null;
17+
return new ProtoRecord(mode, name, funcOrValue, args, null, contextIndex, directiveIndex,
18+
selfIndex, null, null, lastInBinding, false);
1119
}
1220

1321
describe("change detection - coalesce", () => {
@@ -44,18 +52,18 @@ export function main() {
4452
[r("user1", [], 0, 1), r("user2", [], 0, 2), r("hi", [1], 0, 3), r("hi", [2], 0, 4)]);
4553
});
4654

47-
it("should replace duplicate terminal records with" + " self records", () => {
48-
49-
var rs = coalesce([r("user", [], 0, 1, true), r("user", [], 0, 2, true)]);
55+
it("should replace duplicate terminal records with self records", () => {
56+
var rs = coalesce(
57+
[r("user", [], 0, 1, {lastInBinding: true}), r("user", [], 0, 2, {lastInBinding: true})]);
5058

5159
expect(rs[1]).toEqual(new ProtoRecord(RecordType.SELF, "self", null, [], null, 1, null, 2,
5260
null, null, true, false));
5361
});
5462

5563
it("should not coalesce directive lifecycle records", () => {
5664
var rs = coalesce([
57-
r("onCheck", [], 0, 1, true, RecordType.DIRECTIVE_LIFECYCLE),
58-
r("onCheck", [], 0, 1, true, RecordType.DIRECTIVE_LIFECYCLE)
65+
r("onCheck", [], 0, 1, {mode: RecordType.DIRECTIVE_LIFECYCLE}),
66+
r("onCheck", [], 0, 1, {mode: RecordType.DIRECTIVE_LIFECYCLE})
5967
]);
6068

6169
expect(rs.length).toEqual(2);
@@ -64,10 +72,22 @@ export function main() {
6472
it("should not coalesce protos with different names but same value", () => {
6573
var nullFunc = () => {};
6674
var rs = coalesce([
67-
r(nullFunc, [], 0, 1, false, RecordType.PROPERTY, "foo"),
68-
r(nullFunc, [], 0, 1, false, RecordType.PROPERTY, "bar"),
75+
r(nullFunc, [], 0, 1, {name: "foo"}),
76+
r(nullFunc, [], 0, 1, {name: "bar"}),
6977
]);
7078
expect(rs.length).toEqual(2);
7179
});
80+
81+
it("should not coalesce protos with the same context index but different directive indices",
82+
() => {
83+
var nullFunc = () => {};
84+
var rs = coalesce([
85+
r(nullFunc, [], 0, 1, {directiveIndex: new DirectiveIndex(0, 0)}),
86+
r(nullFunc, [], 0, 1, {directiveIndex: new DirectiveIndex(0, 1)}),
87+
r(nullFunc, [], 0, 1, {directiveIndex: new DirectiveIndex(1, 0)}),
88+
r(nullFunc, [], 0, 1, {directiveIndex: null}),
89+
]);
90+
expect(rs.length).toEqual(4);
91+
});
7292
});
7393
}

0 commit comments

Comments
 (0)