feat(change_detector): notify directives on property changes

This commit is contained in:
vsavkin 2014-12-02 17:09:46 -08:00
parent 5bdefee6c9
commit 847cefcb7b
9 changed files with 320 additions and 62 deletions

View File

@ -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];

View File

@ -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) {

View File

@ -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) {

View File

@ -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) {

View File

@ -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', () => {

View File

@ -0,0 +1,5 @@
export class OnChange {
onChange(changes) {
throw "not implemented";
}
}

View File

@ -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

View File

@ -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';

View File

@ -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';
}; };