feat(change_detector): notify directives on property changes
This commit is contained in:
parent
5bdefee6c9
commit
847cefcb7b
|
@ -1,25 +1,70 @@
|
||||||
import {ProtoRecordRange, RecordRange} from './record_range';
|
import {ProtoRecordRange, RecordRange} from './record_range';
|
||||||
import {ProtoRecord, Record} from './record';
|
import {ProtoRecord, Record} from './record';
|
||||||
import {FIELD, int, isPresent} from 'facade/lang';
|
import {int, isPresent, isBlank} from 'facade/lang';
|
||||||
|
import {ListWrapper, List} from 'facade/collection';
|
||||||
|
|
||||||
export * from './record';
|
export * from './record';
|
||||||
export * from './record_range'
|
export * from './record_range'
|
||||||
|
|
||||||
export class ChangeDetector {
|
export class ChangeDetector {
|
||||||
_rootRecordRange:RecordRange;
|
_rootRecordRange:RecordRange;
|
||||||
|
|
||||||
constructor(recordRange:RecordRange) {
|
constructor(recordRange:RecordRange) {
|
||||||
this._rootRecordRange = recordRange;
|
this._rootRecordRange = recordRange;
|
||||||
}
|
}
|
||||||
|
|
||||||
detectChanges():int {
|
detectChanges():int {
|
||||||
var count:int = 0;
|
var count = 0;
|
||||||
for (var record = this._rootRecordRange.findFirstEnabledRecord();
|
var updatedRecords = null;
|
||||||
isPresent(record);
|
var record = this._rootRecordRange.findFirstEnabledRecord();
|
||||||
record = record.nextEnabled) {
|
|
||||||
|
while (isPresent(record)) {
|
||||||
|
var currentRange = record.recordRange;
|
||||||
|
var currentGroup = record.groupMemento();
|
||||||
|
|
||||||
|
var nextEnabled = record.nextEnabled;
|
||||||
|
var nextRange = isPresent(nextEnabled) ? nextEnabled.recordRange : null;
|
||||||
|
var nextGroup = isPresent(nextEnabled) ? nextEnabled.groupMemento() : null;
|
||||||
|
|
||||||
if (record.check()) {
|
if (record.check()) {
|
||||||
count++;
|
count ++;
|
||||||
|
if (record.terminatesExpression()) {
|
||||||
|
updatedRecords = this._addRecord(updatedRecords, record);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if (this._shouldNotifyDispatcher(currentRange, nextRange, currentGroup, nextGroup, updatedRecords)) {
|
||||||
|
currentRange.dispatcher.onRecordChange(currentGroup, updatedRecords);
|
||||||
|
updatedRecords = null;
|
||||||
|
}
|
||||||
|
|
||||||
|
record = record.nextEnabled;
|
||||||
}
|
}
|
||||||
|
|
||||||
return count;
|
return count;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_groupChanged(currentRange, nextRange, currentGroup, nextGroup) {
|
||||||
|
return currentRange != nextRange || currentGroup != nextGroup;
|
||||||
|
}
|
||||||
|
|
||||||
|
_shouldNotifyDispatcher(currentRange, nextRange, currentGroup, nextGroup, updatedRecords) {
|
||||||
|
return this._groupChanged(currentRange, nextRange, currentGroup, nextGroup) && isPresent(updatedRecords);
|
||||||
|
}
|
||||||
|
|
||||||
|
_addRecord(updatedRecords:List, record:Record) {
|
||||||
|
if (isBlank(updatedRecords)) {
|
||||||
|
updatedRecords = _singleElementList;
|
||||||
|
updatedRecords[0] = record;
|
||||||
|
|
||||||
|
} else if (updatedRecords === _singleElementList) {
|
||||||
|
updatedRecords = [_singleElementList[0], record];
|
||||||
|
|
||||||
|
} else {
|
||||||
|
ListWrapper.push(updatedRecords, record);
|
||||||
|
}
|
||||||
|
return updatedRecords;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _singleElementList = [null];
|
||||||
|
|
|
@ -34,15 +34,19 @@ export class ProtoRecord {
|
||||||
arity:int;
|
arity:int;
|
||||||
name:string;
|
name:string;
|
||||||
dest:any;
|
dest:any;
|
||||||
|
groupMemento:any;
|
||||||
|
|
||||||
next:ProtoRecord;
|
next:ProtoRecord;
|
||||||
|
|
||||||
recordInConstruction:Record;
|
recordInConstruction:Record;
|
||||||
|
|
||||||
constructor(recordRange:ProtoRecordRange,
|
constructor(recordRange:ProtoRecordRange,
|
||||||
mode:int,
|
mode:int,
|
||||||
funcOrValue,
|
funcOrValue,
|
||||||
arity:int,
|
arity:int,
|
||||||
name:string,
|
name:string,
|
||||||
dest) {
|
dest,
|
||||||
|
groupMemento) {
|
||||||
|
|
||||||
this.recordRange = recordRange;
|
this.recordRange = recordRange;
|
||||||
this._mode = mode;
|
this._mode = mode;
|
||||||
|
@ -50,6 +54,7 @@ export class ProtoRecord {
|
||||||
this.arity = arity;
|
this.arity = arity;
|
||||||
this.name = name;
|
this.name = name;
|
||||||
this.dest = dest;
|
this.dest = dest;
|
||||||
|
this.groupMemento = groupMemento;
|
||||||
|
|
||||||
this.next = null;
|
this.next = null;
|
||||||
// The concrete Record instantiated from this ProtoRecord
|
// The concrete Record instantiated from this ProtoRecord
|
||||||
|
@ -190,21 +195,20 @@ export class Record {
|
||||||
|
|
||||||
check():boolean {
|
check():boolean {
|
||||||
if (this.isCollection) {
|
if (this.isCollection) {
|
||||||
var changed = this._checkCollection();
|
return this._checkCollection();
|
||||||
if (changed) {
|
|
||||||
this._notifyDispatcher();
|
|
||||||
return true;
|
|
||||||
}
|
|
||||||
return false;
|
|
||||||
} else {
|
} else {
|
||||||
this.previousValue = this.currentValue;
|
return this._checkSingleRecord();
|
||||||
this.currentValue = this._calculateNewValue();
|
|
||||||
if (isSame(this.previousValue, this.currentValue)) return false;
|
|
||||||
this._updateDestination();
|
|
||||||
return true;
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_checkSingleRecord():boolean {
|
||||||
|
this.previousValue = this.currentValue;
|
||||||
|
this.currentValue = this._calculateNewValue();
|
||||||
|
if (isSame(this.previousValue, this.currentValue)) return false;
|
||||||
|
this._updateDestination();
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
|
||||||
_updateDestination() {
|
_updateDestination() {
|
||||||
// todo(vicb): compute this info only once in ctor ? (add a bit in mode not to grow the mem req)
|
// todo(vicb): compute this info only once in ctor ? (add a bit in mode not to grow the mem req)
|
||||||
if (this.dest instanceof Record) {
|
if (this.dest instanceof Record) {
|
||||||
|
@ -213,15 +217,9 @@ export class Record {
|
||||||
} else {
|
} else {
|
||||||
this.dest.updateContext(this.currentValue);
|
this.dest.updateContext(this.currentValue);
|
||||||
}
|
}
|
||||||
} else {
|
|
||||||
this._notifyDispatcher();
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
_notifyDispatcher() {
|
|
||||||
this.recordRange.dispatcher.onRecordChange(this, this.protoRecord.dest);
|
|
||||||
}
|
|
||||||
|
|
||||||
// return whether the content has changed
|
// return whether the content has changed
|
||||||
_checkCollection():boolean {
|
_checkCollection():boolean {
|
||||||
switch(this.type) {
|
switch(this.type) {
|
||||||
|
@ -307,9 +305,21 @@ export class Record {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
terminatesExpression():boolean {
|
||||||
|
return !(this.dest instanceof Record);
|
||||||
|
}
|
||||||
|
|
||||||
get isMarkerRecord() {
|
get isMarkerRecord() {
|
||||||
return this.type == RECORD_TYPE_MARKER;
|
return this.type == RECORD_TYPE_MARKER;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
expressionMemento() {
|
||||||
|
return this.protoRecord.dest;
|
||||||
|
}
|
||||||
|
|
||||||
|
groupMemento() {
|
||||||
|
return isPresent(this.protoRecord) ? this.protoRecord.groupMemento : null;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
function isSame(a, b) {
|
function isSame(a, b) {
|
||||||
|
|
|
@ -45,21 +45,24 @@ export class ProtoRecordRange {
|
||||||
* Parses [ast] into [ProtoRecord]s and adds them to [ProtoRecordRange].
|
* Parses [ast] into [ProtoRecord]s and adds them to [ProtoRecordRange].
|
||||||
*
|
*
|
||||||
* @param ast The expression to watch
|
* @param ast The expression to watch
|
||||||
* @param memento an opaque object which will be passed to WatchGroupDispatcher on
|
* @param expressionMemento an opaque object which will be passed to WatchGroupDispatcher on
|
||||||
* detecting a change.
|
* detecting a change.
|
||||||
* @param content Wether to watch collection content (true) or reference (false, default)
|
* @param content Wether to watch collection content (true) or reference (false, default)
|
||||||
*/
|
*/
|
||||||
addRecordsFromAST(ast:AST,
|
addRecordsFromAST(ast:AST,
|
||||||
memento,
|
expressionMemento,
|
||||||
|
groupMemento,
|
||||||
content:boolean = false)
|
content:boolean = false)
|
||||||
{
|
{
|
||||||
if (this.recordCreator === null) {
|
if (this.recordCreator === null) {
|
||||||
this.recordCreator = new ProtoRecordCreator(this);
|
this.recordCreator = new ProtoRecordCreator(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
if (content) {
|
if (content) {
|
||||||
ast = new Collection(ast);
|
ast = new Collection(ast);
|
||||||
}
|
}
|
||||||
this.recordCreator.createRecordsFromAST(ast, memento);
|
|
||||||
|
this.recordCreator.createRecordsFromAST(ast, expressionMemento, groupMemento);
|
||||||
}
|
}
|
||||||
|
|
||||||
// TODO(rado): the type annotation should be dispatcher:WatchGroupDispatcher.
|
// TODO(rado): the type annotation should be dispatcher:WatchGroupDispatcher.
|
||||||
|
@ -85,6 +88,8 @@ export class ProtoRecordRange {
|
||||||
for (var proto = this.recordCreator.headRecord; proto != null; proto = proto.next) {
|
for (var proto = this.recordCreator.headRecord; proto != null; proto = proto.next) {
|
||||||
if (proto.dest instanceof Destination) {
|
if (proto.dest instanceof Destination) {
|
||||||
proto.recordInConstruction.dest = proto.dest.record.recordInConstruction;
|
proto.recordInConstruction.dest = proto.dest.record.recordInConstruction;
|
||||||
|
} else {
|
||||||
|
proto.recordInConstruction.dest = proto.dest;
|
||||||
}
|
}
|
||||||
proto.recordInConstruction = null;
|
proto.recordInConstruction = null;
|
||||||
}
|
}
|
||||||
|
@ -378,6 +383,8 @@ class ProtoRecordCreator {
|
||||||
protoRecordRange:ProtoRecordRange;
|
protoRecordRange:ProtoRecordRange;
|
||||||
headRecord:ProtoRecord;
|
headRecord:ProtoRecord;
|
||||||
tailRecord:ProtoRecord;
|
tailRecord:ProtoRecord;
|
||||||
|
groupMemento:any;
|
||||||
|
|
||||||
constructor(protoRecordRange) {
|
constructor(protoRecordRange) {
|
||||||
this.protoRecordRange = protoRecordRange;
|
this.protoRecordRange = protoRecordRange;
|
||||||
this.headRecord = null;
|
this.headRecord = null;
|
||||||
|
@ -491,12 +498,13 @@ class ProtoRecordCreator {
|
||||||
|
|
||||||
visitTemplateBindings(ast, dest) {this._unsupported();}
|
visitTemplateBindings(ast, dest) {this._unsupported();}
|
||||||
|
|
||||||
createRecordsFromAST(ast:AST, memento){
|
createRecordsFromAST(ast:AST, expressionMemento:any, groupMemento:any){
|
||||||
ast.visit(this, memento);
|
this.groupMemento = groupMemento;
|
||||||
|
ast.visit(this, expressionMemento);
|
||||||
}
|
}
|
||||||
|
|
||||||
construct(recordType, funcOrValue, arity, name, dest) {
|
construct(recordType, funcOrValue, arity, name, dest) {
|
||||||
return new ProtoRecord(this.protoRecordRange, recordType, funcOrValue, arity, name, dest);
|
return new ProtoRecord(this.protoRecordRange, recordType, funcOrValue, arity, name, dest, this.groupMemento);
|
||||||
}
|
}
|
||||||
|
|
||||||
add(protoRecord:ProtoRecord) {
|
add(protoRecord:ProtoRecord) {
|
||||||
|
|
|
@ -26,7 +26,7 @@ export function main() {
|
||||||
function createChangeDetector(memo:string, exp:string, context = null, formatters = null,
|
function createChangeDetector(memo:string, exp:string, context = null, formatters = null,
|
||||||
content = false) {
|
content = false) {
|
||||||
var prr = new ProtoRecordRange();
|
var prr = new ProtoRecordRange();
|
||||||
prr.addRecordsFromAST(ast(exp), memo, content);
|
prr.addRecordsFromAST(ast(exp), memo, memo, content);
|
||||||
|
|
||||||
var dispatcher = new LoggingDispatcher();
|
var dispatcher = new LoggingDispatcher();
|
||||||
var rr = prr.instantiate(dispatcher, formatters);
|
var rr = prr.instantiate(dispatcher, formatters);
|
||||||
|
@ -97,21 +97,21 @@ export function main() {
|
||||||
it("should support literal array", () => {
|
it("should support literal array", () => {
|
||||||
var c = createChangeDetector('array', '[1,2]');
|
var c = createChangeDetector('array', '[1,2]');
|
||||||
c["changeDetector"].detectChanges();
|
c["changeDetector"].detectChanges();
|
||||||
expect(c["dispatcher"].loggedValues).toEqual([[1,2]]);
|
expect(c["dispatcher"].loggedValues).toEqual([[[1,2]]]);
|
||||||
|
|
||||||
c = createChangeDetector('array', '[1,a]', new TestData(2));
|
c = createChangeDetector('array', '[1,a]', new TestData(2));
|
||||||
c["changeDetector"].detectChanges();
|
c["changeDetector"].detectChanges();
|
||||||
expect(c["dispatcher"].loggedValues).toEqual([[1,2]]);
|
expect(c["dispatcher"].loggedValues).toEqual([[[1,2]]]);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should support literal maps", () => {
|
it("should support literal maps", () => {
|
||||||
var c = createChangeDetector('map', '{z:1}');
|
var c = createChangeDetector('map', '{z:1}');
|
||||||
c["changeDetector"].detectChanges();
|
c["changeDetector"].detectChanges();
|
||||||
expect(MapWrapper.get(c["dispatcher"].loggedValues[0], 'z')).toEqual(1);
|
expect(MapWrapper.get(c["dispatcher"].loggedValues[0][0], 'z')).toEqual(1);
|
||||||
|
|
||||||
c = createChangeDetector('map', '{z:a}', new TestData(1));
|
c = createChangeDetector('map', '{z:a}', new TestData(1));
|
||||||
c["changeDetector"].detectChanges();
|
c["changeDetector"].detectChanges();
|
||||||
expect(MapWrapper.get(c["dispatcher"].loggedValues[0], 'z')).toEqual(1);
|
expect(MapWrapper.get(c["dispatcher"].loggedValues[0][0], 'z')).toEqual(1);
|
||||||
});
|
});
|
||||||
|
|
||||||
it("should support binary operations", () => {
|
it("should support binary operations", () => {
|
||||||
|
@ -242,6 +242,7 @@ export function main() {
|
||||||
|
|
||||||
context.a = [0];
|
context.a = [0];
|
||||||
cd.detectChanges();
|
cd.detectChanges();
|
||||||
|
|
||||||
expect(dsp.log).toEqual(["a=" +
|
expect(dsp.log).toEqual(["a=" +
|
||||||
arrayChangesAsString({
|
arrayChangesAsString({
|
||||||
collection: ['0[null->0]'],
|
collection: ['0[null->0]'],
|
||||||
|
@ -343,10 +344,69 @@ export function main() {
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe("onGroupChange", () => {
|
||||||
|
it("should notify the dispatcher when a group of records changes", () => {
|
||||||
|
var prr = new ProtoRecordRange();
|
||||||
|
prr.addRecordsFromAST(ast("1 + 2"), "memo", 1);
|
||||||
|
prr.addRecordsFromAST(ast("10 + 20"), "memo", 1);
|
||||||
|
prr.addRecordsFromAST(ast("100 + 200"), "memo2", 2);
|
||||||
|
|
||||||
|
var dispatcher = new LoggingDispatcher();
|
||||||
|
var rr = prr.instantiate(dispatcher, null);
|
||||||
|
|
||||||
|
var cd = new ChangeDetector(rr);
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
expect(dispatcher.loggedValues).toEqual([[3, 30], [300]]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should update every instance of a group individually", () => {
|
||||||
|
var prr = new ProtoRecordRange();
|
||||||
|
prr.addRecordsFromAST(ast("1 + 2"), "memo", "memo");
|
||||||
|
|
||||||
|
var dispatcher = new LoggingDispatcher();
|
||||||
|
var rr = new RecordRange(null, dispatcher);
|
||||||
|
rr.addRange(prr.instantiate(dispatcher, null));
|
||||||
|
rr.addRange(prr.instantiate(dispatcher, null));
|
||||||
|
|
||||||
|
var cd = new ChangeDetector(rr);
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
expect(dispatcher.loggedValues).toEqual([[3], [3]]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it("should notify the dispatcher before switching to the next group", () => {
|
||||||
|
var prr = new ProtoRecordRange();
|
||||||
|
prr.addRecordsFromAST(ast("a()"), "a", 1);
|
||||||
|
prr.addRecordsFromAST(ast("b()"), "b", 2);
|
||||||
|
prr.addRecordsFromAST(ast("c()"), "b", 2);
|
||||||
|
|
||||||
|
var dispatcher = new LoggingDispatcher();
|
||||||
|
var rr = prr.instantiate(dispatcher, null);
|
||||||
|
|
||||||
|
var tr = new TestRecord();
|
||||||
|
tr.a = () => {dispatcher.logValue('InvokeA'); return 'a'};
|
||||||
|
tr.b = () => {dispatcher.logValue('InvokeB'); return 'b'};
|
||||||
|
tr.c = () => {dispatcher.logValue('InvokeC'); return 'c'};
|
||||||
|
rr.setContext(tr);
|
||||||
|
|
||||||
|
var cd = new ChangeDetector(rr);
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
expect(dispatcher.loggedValues).toEqual(['InvokeA', ['a'], 'InvokeB', 'InvokeC', ['b', 'c']]);
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class TestRecord {
|
||||||
|
a;
|
||||||
|
b;
|
||||||
|
c;
|
||||||
|
}
|
||||||
|
|
||||||
class Person {
|
class Person {
|
||||||
name:string;
|
name:string;
|
||||||
address:Address;
|
address:Address;
|
||||||
|
@ -387,6 +447,7 @@ class TestData {
|
||||||
class LoggingDispatcher extends WatchGroupDispatcher {
|
class LoggingDispatcher extends WatchGroupDispatcher {
|
||||||
log:List;
|
log:List;
|
||||||
loggedValues:List;
|
loggedValues:List;
|
||||||
|
|
||||||
constructor() {
|
constructor() {
|
||||||
this.log = null;
|
this.log = null;
|
||||||
this.loggedValues = null;
|
this.loggedValues = null;
|
||||||
|
@ -398,9 +459,17 @@ class LoggingDispatcher extends WatchGroupDispatcher {
|
||||||
this.loggedValues = ListWrapper.create();
|
this.loggedValues = ListWrapper.create();
|
||||||
}
|
}
|
||||||
|
|
||||||
onRecordChange(record:Record, context) {
|
logValue(value) {
|
||||||
ListWrapper.push(this.loggedValues, record.currentValue);
|
ListWrapper.push(this.loggedValues, value);
|
||||||
ListWrapper.push(this.log, context + '=' + this._asString(record.currentValue));
|
}
|
||||||
|
|
||||||
|
onRecordChange(group, records:List) {
|
||||||
|
var value = records[0].currentValue;
|
||||||
|
var dest = records[0].protoRecord.dest;
|
||||||
|
ListWrapper.push(this.log, dest + '=' + this._asString(value));
|
||||||
|
|
||||||
|
var values = ListWrapper.map(records, (r) => r.currentValue);
|
||||||
|
ListWrapper.push(this.loggedValues, values);
|
||||||
}
|
}
|
||||||
|
|
||||||
_asString(value) {
|
_asString(value) {
|
||||||
|
|
|
@ -46,7 +46,7 @@ export function main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
function createRecord(rr) {
|
function createRecord(rr) {
|
||||||
return new Record(rr, new ProtoRecord(null, 0, null, null, null, null), null);
|
return new Record(rr, new ProtoRecord(null, 0, null, null, null, null, null), null);
|
||||||
}
|
}
|
||||||
|
|
||||||
describe('record range', () => {
|
describe('record range', () => {
|
||||||
|
|
|
@ -0,0 +1,5 @@
|
||||||
|
export class OnChange {
|
||||||
|
onChange(changes) {
|
||||||
|
throw "not implemented";
|
||||||
|
}
|
||||||
|
}
|
|
@ -1,5 +1,5 @@
|
||||||
import {DOM, Element, Node, Text, DocumentFragment, TemplateElement} from 'facade/dom';
|
import {DOM, Element, Node, Text, DocumentFragment, TemplateElement} from 'facade/dom';
|
||||||
import {ListWrapper, MapWrapper, List} from 'facade/collection';
|
import {ListWrapper, MapWrapper, StringMapWrapper, List} from 'facade/collection';
|
||||||
import {ProtoRecordRange, RecordRange, WatchGroupDispatcher} from 'change_detection/record_range';
|
import {ProtoRecordRange, RecordRange, WatchGroupDispatcher} from 'change_detection/record_range';
|
||||||
import {Record} from 'change_detection/record';
|
import {Record} from 'change_detection/record';
|
||||||
import {AST} from 'change_detection/parser/ast';
|
import {AST} from 'change_detection/parser/ast';
|
||||||
|
@ -12,6 +12,7 @@ import {FIELD, IMPLEMENTS, int, isPresent, isBlank} from 'facade/lang';
|
||||||
import {Injector} from 'di/di';
|
import {Injector} from 'di/di';
|
||||||
import {NgElement} from 'core/dom/element';
|
import {NgElement} from 'core/dom/element';
|
||||||
import {ViewPort} from './viewport';
|
import {ViewPort} from './viewport';
|
||||||
|
import {OnChange} from './interfaces';
|
||||||
|
|
||||||
const NG_BINDING_CLASS = 'ng-binding';
|
const NG_BINDING_CLASS = 'ng-binding';
|
||||||
|
|
||||||
|
@ -47,22 +48,55 @@ export class View {
|
||||||
this.viewPorts = null;
|
this.viewPorts = null;
|
||||||
}
|
}
|
||||||
|
|
||||||
onRecordChange(record:Record, target) {
|
onRecordChange(groupMemento, records:List<Record>) {
|
||||||
|
this._invokeMementoForRecords(records);
|
||||||
|
if (groupMemento instanceof DirectivePropertyGroupMemento) {
|
||||||
|
this._notifyDirectiveAboutChanges(groupMemento, records);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_invokeMementoForRecords(records:List<Record>) {
|
||||||
|
for(var i = 0; i < records.length; ++i) {
|
||||||
|
this._invokeMementoFor(records[i]);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
_notifyDirectiveAboutChanges(groupMemento, records:List<Record>) {
|
||||||
|
var dir = groupMemento.directive(this.elementInjectors);
|
||||||
|
if (dir instanceof OnChange) {
|
||||||
|
dir.onChange(this._collectChanges(records));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
// dispatch to element injector or text nodes based on context
|
// dispatch to element injector or text nodes based on context
|
||||||
if (target instanceof DirectivePropertyMemento) {
|
_invokeMementoFor(record:Record) {
|
||||||
|
var memento = record.expressionMemento();
|
||||||
|
if (memento instanceof DirectivePropertyMemento) {
|
||||||
// we know that it is DirectivePropertyMemento
|
// we know that it is DirectivePropertyMemento
|
||||||
var directiveMemento:DirectivePropertyMemento = target;
|
var directiveMemento:DirectivePropertyMemento = memento;
|
||||||
directiveMemento.invoke(record, this.elementInjectors);
|
directiveMemento.invoke(record, this.elementInjectors);
|
||||||
} else if (target instanceof ElementPropertyMemento) {
|
|
||||||
var elementMemento:ElementPropertyMemento = target;
|
} else if (memento instanceof ElementPropertyMemento) {
|
||||||
|
var elementMemento:ElementPropertyMemento = memento;
|
||||||
elementMemento.invoke(record, this.bindElements);
|
elementMemento.invoke(record, this.bindElements);
|
||||||
|
|
||||||
} else {
|
} else {
|
||||||
// we know it refers to _textNodes.
|
// we know it refers to _textNodes.
|
||||||
var textNodeIndex:number = target;
|
var textNodeIndex:number = memento;
|
||||||
DOM.setText(this.textNodes[textNodeIndex], record.currentValue);
|
DOM.setText(this.textNodes[textNodeIndex], record.currentValue);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
_collectChanges(records:List) {
|
||||||
|
var changes = StringMapWrapper.create();
|
||||||
|
for(var i = 0; i < records.length; ++i) {
|
||||||
|
var record = records[i];
|
||||||
|
var propertyUpdate = new PropertyUpdate(record.currentValue, record.previousValue);
|
||||||
|
StringMapWrapper.set(changes, record.expressionMemento()._setterName, propertyUpdate);
|
||||||
|
}
|
||||||
|
return changes;
|
||||||
|
}
|
||||||
|
|
||||||
addViewPort(viewPort: ViewPort) {
|
addViewPort(viewPort: ViewPort) {
|
||||||
if (isBlank(this.viewPorts)) this.viewPorts = [];
|
if (isBlank(this.viewPorts)) this.viewPorts = [];
|
||||||
ListWrapper.push(this.viewPorts, viewPort);
|
ListWrapper.push(this.viewPorts, viewPort);
|
||||||
|
@ -163,7 +197,8 @@ export class ProtoView {
|
||||||
elBinder.textNodeIndices = ListWrapper.create();
|
elBinder.textNodeIndices = ListWrapper.create();
|
||||||
}
|
}
|
||||||
ListWrapper.push(elBinder.textNodeIndices, indexInParent);
|
ListWrapper.push(elBinder.textNodeIndices, indexInParent);
|
||||||
this.protoRecordRange.addRecordsFromAST(expression, this.textNodesWithBindingCount++);
|
var memento = this.textNodesWithBindingCount++;
|
||||||
|
this.protoRecordRange.addRecordsFromAST(expression, memento, memento);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -175,12 +210,8 @@ export class ProtoView {
|
||||||
elBinder.hasElementPropertyBindings = true;
|
elBinder.hasElementPropertyBindings = true;
|
||||||
this.elementsWithBindingCount++;
|
this.elementsWithBindingCount++;
|
||||||
}
|
}
|
||||||
this.protoRecordRange.addRecordsFromAST(expression,
|
var memento = new ElementPropertyMemento(this.elementsWithBindingCount-1, propertyName);
|
||||||
new ElementPropertyMemento(
|
this.protoRecordRange.addRecordsFromAST(expression, memento, memento);
|
||||||
this.elementsWithBindingCount-1,
|
|
||||||
propertyName
|
|
||||||
)
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
@ -202,15 +233,15 @@ export class ProtoView {
|
||||||
expression:AST,
|
expression:AST,
|
||||||
setterName:string,
|
setterName:string,
|
||||||
setter:SetterFn) {
|
setter:SetterFn) {
|
||||||
this.protoRecordRange.addRecordsFromAST(
|
|
||||||
expression,
|
var expMemento = new DirectivePropertyMemento(
|
||||||
new DirectivePropertyMemento(
|
this.elementBinders.length-1,
|
||||||
this.elementBinders.length-1,
|
directiveIndex,
|
||||||
directiveIndex,
|
setterName,
|
||||||
setterName,
|
setter
|
||||||
setter
|
|
||||||
)
|
|
||||||
);
|
);
|
||||||
|
var groupMemento = DirectivePropertyGroupMemento.get(expMemento);
|
||||||
|
this.protoRecordRange.addRecordsFromAST(expression, expMemento, groupMemento, false);
|
||||||
}
|
}
|
||||||
|
|
||||||
static _createElementInjectors(elements, binders, hostElementInjector) {
|
static _createElementInjectors(elements, binders, hostElementInjector) {
|
||||||
|
@ -363,6 +394,43 @@ export class DirectivePropertyMemento {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var _groups = MapWrapper.create();
|
||||||
|
|
||||||
|
class DirectivePropertyGroupMemento {
|
||||||
|
_elementInjectorIndex:number;
|
||||||
|
_directiveIndex:number;
|
||||||
|
|
||||||
|
constructor(elementInjectorIndex:number, directiveIndex:number) {
|
||||||
|
this._elementInjectorIndex = elementInjectorIndex;
|
||||||
|
this._directiveIndex = directiveIndex;
|
||||||
|
}
|
||||||
|
|
||||||
|
static get(memento:DirectivePropertyMemento) {
|
||||||
|
var elementInjectorIndex = memento._elementInjectorIndex;
|
||||||
|
var directiveIndex = memento._directiveIndex;
|
||||||
|
var id = elementInjectorIndex * 100 + directiveIndex;
|
||||||
|
|
||||||
|
if (! MapWrapper.contains(_groups, id)) {
|
||||||
|
return MapWrapper.set(_groups, id, new DirectivePropertyGroupMemento(elementInjectorIndex, directiveIndex));
|
||||||
|
}
|
||||||
|
return MapWrapper.get(_groups, id);
|
||||||
|
}
|
||||||
|
|
||||||
|
directive(elementInjectors:List<ElementInjector>) {
|
||||||
|
var elementInjector:ElementInjector = elementInjectors[this._elementInjectorIndex];
|
||||||
|
return elementInjector.getAtIndex(this._directiveIndex);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
class PropertyUpdate {
|
||||||
|
currentValue;
|
||||||
|
previousValue;
|
||||||
|
|
||||||
|
constructor(currentValue, previousValue) {
|
||||||
|
this.currentValue = currentValue;
|
||||||
|
this.previousValue = previousValue;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
//TODO(tbosch): I don't like to have done be called from a different place than notify
|
//TODO(tbosch): I don't like to have done be called from a different place than notify
|
||||||
|
|
|
@ -2,6 +2,7 @@
|
||||||
* Define public API for Angular here
|
* Define public API for Angular here
|
||||||
*/
|
*/
|
||||||
export * from './annotations/annotations';
|
export * from './annotations/annotations';
|
||||||
|
export * from './compiler/interfaces';
|
||||||
export * from './annotations/template_config';
|
export * from './annotations/template_config';
|
||||||
|
|
||||||
export * from './application';
|
export * from './application';
|
||||||
|
|
|
@ -3,6 +3,7 @@ import {ProtoView, ElementPropertyMemento, DirectivePropertyMemento} from 'core/
|
||||||
import {ProtoElementInjector, ElementInjector} from 'core/compiler/element_injector';
|
import {ProtoElementInjector, ElementInjector} from 'core/compiler/element_injector';
|
||||||
import {DirectiveMetadataReader} from 'core/compiler/directive_metadata_reader';
|
import {DirectiveMetadataReader} from 'core/compiler/directive_metadata_reader';
|
||||||
import {Component, Decorator, Template} from 'core/annotations/annotations';
|
import {Component, Decorator, Template} from 'core/annotations/annotations';
|
||||||
|
import {OnChange} from 'core/core';
|
||||||
import {ProtoRecordRange} from 'change_detection/record_range';
|
import {ProtoRecordRange} from 'change_detection/record_range';
|
||||||
import {ChangeDetector} from 'change_detection/change_detector';
|
import {ChangeDetector} from 'change_detection/change_detector';
|
||||||
import {TemplateConfig} from 'core/annotations/template_config';
|
import {TemplateConfig} from 'core/annotations/template_config';
|
||||||
|
@ -281,7 +282,7 @@ export function main() {
|
||||||
expect(view.bindElements[0].id).toEqual('buz');
|
expect(view.bindElements[0].id).toEqual('buz');
|
||||||
});
|
});
|
||||||
|
|
||||||
it('should consume directive watch expression change.', () => {
|
it('should consume directive watch expression change', () => {
|
||||||
var pv = new ProtoView(createElement('<div class="ng-binding"></div>'),
|
var pv = new ProtoView(createElement('<div class="ng-binding"></div>'),
|
||||||
new ProtoRecordRange());
|
new ProtoRecordRange());
|
||||||
pv.bindElement(new ProtoElementInjector(null, 0, [SomeDirective]));
|
pv.bindElement(new ProtoElementInjector(null, 0, [SomeDirective]));
|
||||||
|
@ -292,6 +293,43 @@ export function main() {
|
||||||
cd.detectChanges();
|
cd.detectChanges();
|
||||||
expect(view.elementInjectors[0].get(SomeDirective).prop).toEqual('buz');
|
expect(view.elementInjectors[0].get(SomeDirective).prop).toEqual('buz');
|
||||||
});
|
});
|
||||||
|
|
||||||
|
it('should notify a directive about changes after all its properties have been set', () => {
|
||||||
|
var pv = new ProtoView(createElement('<div class="ng-binding"></div>'),
|
||||||
|
new ProtoRecordRange());
|
||||||
|
|
||||||
|
pv.bindElement(new ProtoElementInjector(null, 0, [DirectiveImplementingOnChange]));
|
||||||
|
pv.bindDirectiveProperty( 0, parser.parseBinding('a').ast, 'a', reflector.setter('a'));
|
||||||
|
pv.bindDirectiveProperty( 0, parser.parseBinding('b').ast, 'b', reflector.setter('b'));
|
||||||
|
createView(pv);
|
||||||
|
|
||||||
|
ctx.a = 100;
|
||||||
|
ctx.b = 200;
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
var directive = view.elementInjectors[0].get(DirectiveImplementingOnChange);
|
||||||
|
expect(directive.c).toEqual(300);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should provide a map of updated properties', () => {
|
||||||
|
var pv = new ProtoView(createElement('<div class="ng-binding"></div>'),
|
||||||
|
new ProtoRecordRange());
|
||||||
|
|
||||||
|
pv.bindElement(new ProtoElementInjector(null, 0, [DirectiveImplementingOnChange]));
|
||||||
|
pv.bindDirectiveProperty( 0, parser.parseBinding('a').ast, 'a', reflector.setter('a'));
|
||||||
|
pv.bindDirectiveProperty( 0, parser.parseBinding('b').ast, 'b', reflector.setter('b'));
|
||||||
|
createView(pv);
|
||||||
|
ctx.a = 0;
|
||||||
|
ctx.b = 0;
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
ctx.a = 100;
|
||||||
|
cd.detectChanges();
|
||||||
|
|
||||||
|
var directive = view.elementInjectors[0].get(DirectiveImplementingOnChange);
|
||||||
|
expect(directive.changes["a"].currentValue).toEqual(100);
|
||||||
|
expect(directive.changes["b"]).not.toBeDefined();
|
||||||
|
});
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
@ -324,6 +362,18 @@ class SomeDirective {
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
class DirectiveImplementingOnChange extends OnChange {
|
||||||
|
a;
|
||||||
|
b;
|
||||||
|
c;
|
||||||
|
changes;
|
||||||
|
|
||||||
|
onChange(changes) {
|
||||||
|
this.c = this.a + this.b;
|
||||||
|
this.changes = changes;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
class SomeService {}
|
class SomeService {}
|
||||||
|
|
||||||
@Component({
|
@Component({
|
||||||
|
@ -368,6 +418,8 @@ class AnotherDirective {
|
||||||
|
|
||||||
class MyEvaluationContext {
|
class MyEvaluationContext {
|
||||||
foo:string;
|
foo:string;
|
||||||
|
a;
|
||||||
|
b;
|
||||||
constructor() {
|
constructor() {
|
||||||
this.foo = 'bar';
|
this.foo = 'bar';
|
||||||
};
|
};
|
||||||
|
|
Loading…
Reference in New Issue