feat(ChangeDetector): implement enabling/disabling watch group

This commit is contained in:
vsavkin 2014-11-18 17:26:38 -08:00
parent 41856ad3f0
commit 862c6412c4
5 changed files with 309 additions and 228 deletions

View File

@ -13,7 +13,7 @@ export class ChangeDetector {
detectChanges():int { detectChanges():int {
var count:int = 0; var count:int = 0;
for (var record = this._rootWatchGroup.headEnabledRecord; for (var record = this._rootWatchGroup.findFirstEnabledRecord();
isPresent(record); isPresent(record);
record = record.nextEnabled) { record = record.nextEnabled) {
if (record.check()) { if (record.check()) {

View File

@ -1,5 +1,5 @@
import {ProtoWatchGroup, WatchGroup} from './watch_group'; import {ProtoWatchGroup, WatchGroup} from './watch_group';
import {FIELD, isPresent, int, StringWrapper, FunctionWrapper, BaseException} from 'facade/lang'; import {FIELD, isPresent, isBlank, int, StringWrapper, FunctionWrapper, BaseException} from 'facade/lang';
import {ListWrapper, MapWrapper} from 'facade/collection'; import {ListWrapper, MapWrapper} from 'facade/collection';
import {ClosureMap} from 'change_detection/parser/closure_map'; import {ClosureMap} from 'change_detection/parser/closure_map';
@ -105,6 +105,11 @@ export class Record {
this.funcOrValue = null; this.funcOrValue = null;
this.args = null; this.args = null;
if (isBlank(protoRecord)) {
this.mode = MODE_STATE_MARKER;
return;
}
var type = protoRecord.recordType; var type = protoRecord.recordType;
if (type === PROTO_RECORD_CONST) { if (type === PROTO_RECORD_CONST) {
this.mode = MODE_STATE_CONST; this.mode = MODE_STATE_CONST;
@ -135,6 +140,12 @@ export class Record {
} }
} }
static createMarker(wg:WatchGroup) {
var r = new Record(wg, null, null);
r.disabled = true;
return r;
}
check():boolean { check():boolean {
this.previousValue = this.currentValue; this.previousValue = this.currentValue;
this.currentValue = this._calculateNewValue(); this.currentValue = this._calculateNewValue();
@ -200,7 +211,13 @@ export class Record {
updateContext(value) { updateContext(value) {
this.context = value; this.context = value;
this.watchGroup.enableRecord(this); if (! this.isMarkerRecord) {
this.watchGroup.enableRecord(this);
}
}
get isMarkerRecord() {
return isBlank(this.protoRecord);
} }
} }

View File

@ -77,110 +77,158 @@ export class WatchGroup {
@FIELD('final dispatcher:WatchGroupDispatcher') @FIELD('final dispatcher:WatchGroupDispatcher')
@FIELD('final headRecord:Record') @FIELD('final headRecord:Record')
@FIELD('final tailRecord:Record') @FIELD('final tailRecord:Record')
@FIELD('final disabled:boolean')
// TODO(rado): the type annotation should be dispatcher:WatchGroupDispatcher. // TODO(rado): the type annotation should be dispatcher:WatchGroupDispatcher.
// but @Implements is not ready yet. // but @Implements is not ready yet.
constructor(protoWatchGroup:ProtoWatchGroup, dispatcher) { constructor(protoWatchGroup:ProtoWatchGroup, dispatcher) {
this.protoWatchGroup = protoWatchGroup; this.protoWatchGroup = protoWatchGroup;
this.dispatcher = dispatcher; this.dispatcher = dispatcher;
this.headRecord = null;
this.tailRecord = null;
this.headEnabledRecord = null;
this.tailEnabledRecord = null;
this.context = null;
this.childHead = null; this.disabled = false;
this.childTail = null;
this.next = null; this.headRecord = Record.createMarker(this);
this.prev = null; this.tailRecord = Record.createMarker(this);
this.headRecord.next = this.tailRecord;
this.tailRecord.prev = this.headRecord;
} }
/// addRecord must be called before addChild /// addRecord assumes that all records are enabled
addRecord(record:Record) { addRecord(record:Record) {
if (isPresent(this.tailRecord)) { var lastRecord = this.tailRecord.prev;
this.tailRecord.next = record;
this.tailRecord.nextEnabled = record;
record.prev = this.tailRecord;
record.prevEnabled = this.tailRecord;
this.tailRecord = this.tailEnabledRecord = record;
} else { lastRecord.next = record;
this.headRecord = this.tailRecord = record; lastRecord.nextEnabled = record;
this.headEnabledRecord = this.tailEnabledRecord = record;
record.prev = lastRecord;
record.prevEnabled = lastRecord;
record.next = this.tailRecord;
this.tailRecord.prev = record;
}
addChild(child:WatchGroup) {
var lastRecord = this.tailRecord.prev;
var lastEnabledRecord = this.findLastEnabledRecord();
var firstEnabledChildRecord = child.findFirstEnabledRecord();
lastRecord.next = child.headRecord;
child.headRecord.prev = lastRecord;
child.tailRecord.next = this.tailRecord;
this.tailRecord.prev = child.tailRecord;
if (isPresent(lastEnabledRecord) && isPresent(firstEnabledChildRecord)) {
lastEnabledRecord.nextEnabled = firstEnabledChildRecord;
firstEnabledChildRecord.prevEnabled = lastEnabledRecord;
} }
} }
removeChild(child:WatchGroup) {
var firstEnabledChildRecord = child.findFirstEnabledRecord();
var lastEnabledChildRecord = child.findLastEnabledRecord();
var next = child.tailRecord.next;
var prev = child.headRecord.prev;
next.prev = prev;
prev.next = next;
var nextEnabled = lastEnabledChildRecord.nextEnabled;
var prevEnabled = firstEnabledChildRecord.prevEnabled;
if (isPresent(nextEnabled)) nextEnabled.prev = prevEnabled;
if (isPresent(prevEnabled)) prevEnabled.next = nextEnabled;
}
findFirstEnabledRecord() {
return this._nextEnabled(this.headRecord);
}
findLastEnabledRecord() {
return this._prevEnabled(this.tailRecord);
}
disableRecord(record:Record) { disableRecord(record:Record) {
var prev = record.prevEnabled; var prevEnabled = record.prevEnabled;
var next = record.nextEnabled; var nextEnabled = record.nextEnabled;
if (isPresent(prevEnabled)) prevEnabled.nextEnabled = nextEnabled;
if (isPresent(nextEnabled)) nextEnabled.prevEnabled = prevEnabled;
record.disabled = true; record.disabled = true;
if (isPresent(prev)) {
prev.nextEnabled = next;
} else {
this.headEnabledRecord = next;
}
if (isPresent(next)) {
next.prevEnabled = prev;
} else {
this.tailEnabledRecord = prev;
}
} }
enableRecord(record:Record) { enableRecord(record:Record) {
if (!record.disabled) return; if (!record.disabled) return;
var prev = record.prev; var prevEnabled = this._prevEnabled(record);
while (prev != null && prev.disabled) prev = prev.prev; var nextEnabled = this._nextEnabled(record);
var next = record.next; record.prevEnabled = prevEnabled;
while (next != null && next.disabled) next = next.next; record.nextEnabled = nextEnabled;
if (isPresent(prevEnabled)) prevEnabled.nextEnabled = record;
if (isPresent(nextEnabled)) nextEnabled.prevEnabled = record;
record.disabled = false; record.disabled = false;
record.prevEnabled = prev;
record.nextEnabled = next;
if (isPresent(prev)) {
prev.nextEnabled = record;
} else {
this.headEnabledRecord = record;
}
if (isPresent(next)) {
next.prevEnabled = record;
} else {
this.tailEnabledRecord = record;
}
} }
addChild(child:WatchGroup) { disableGroup(child:WatchGroup) {
if (isBlank(this.childTail)) { var firstEnabledChildRecord = child.findFirstEnabledRecord();
this.childHead = this.childTail = child; var lastEnabledChildRecord = child.findLastEnabledRecord();
} else {
this.childTail.next = child; var nextEnabled = lastEnabledChildRecord.nextEnabled;
child.prev = this.childTail; var prevEnabled = firstEnabledChildRecord.prevEnabled;
this.childTail = child;
} if (isPresent(nextEnabled)) nextEnabled.prevEnabled = prevEnabled;
this._attachRecordsFromWatchGroup(child); if (isPresent(prevEnabled)) prevEnabled.nextEnabled = nextEnabled;
child.disabled = true;
} }
_attachRecordsFromWatchGroup(child:WatchGroup) { enableGroup(child:WatchGroup) {
if (isPresent(this.tailRecord)) { var prevEnabledRecord = this._prevEnabled(child.headRecord);
if (isPresent(child.headRecord)) { var nextEnabledRecord = this._nextEnabled(child.tailRecord);
this.tailRecord.next = child.headRecord;
this.tailRecord.nextEnabled = child.headRecord;
child.headRecord.prev = this.tailRecord; var firstEnabledChildRecord = child.findFirstEnabledRecord();
child.headRecord.prevEnabled = this.tailRecord; var lastEnabledChildRecord = child.findLastEnabledRecord();
if (isPresent(firstEnabledChildRecord) && isPresent(prevEnabledRecord)){
firstEnabledChildRecord.prevEnabled = prevEnabledRecord;
prevEnabledRecord.nextEnabled = firstEnabledChildRecord;
}
if (isPresent(lastEnabledChildRecord) && isPresent(nextEnabledRecord)){
lastEnabledChildRecord.nextEnabled = nextEnabledRecord;
nextEnabledRecord.prevEnabled = lastEnabledChildRecord;
}
child.disabled = false;
}
_nextEnabled(record:Record) {
record = record.next;
while (isPresent(record) && record !== this.tailRecord && record.disabled) {
if (record.isMarkerRecord && record.watchGroup.disabled) {
record = record.watchGroup.tailRecord.next;
} else {
record = record.next;
} }
} else {
this.headRecord = child.headRecord;
this.headEnabledRecord = child.headEnabledRecord;
} }
return record === this.tailRecord ? null : record;
}
this.tailRecord = child.tailRecord; _prevEnabled(record:Record) {
this.tailEnabledRecord = child.tailEnabledRecord; record = record.prev;
while (isPresent(record) && record !== this.headRecord && record.disabled) {
if (record.isMarkerRecord && record.watchGroup.disabled) {
record = record.watchGroup.headRecord.prev;
} else {
record = record.prev;
}
}
return record === this.headRecord ? null : record;
} }
/** /**

View File

@ -1,5 +1,6 @@
import {ddescribe, describe, it, iit, xit, expect} from 'test_lib/test_lib'; import {ddescribe, describe, it, iit, xit, expect} from 'test_lib/test_lib';
import {isPresent} from 'facade/lang';
import {List, ListWrapper, MapWrapper} from 'facade/collection'; import {List, ListWrapper, MapWrapper} from 'facade/collection';
import {Parser} from 'change_detection/parser/parser'; import {Parser} from 'change_detection/parser/parser';
import {Lexer} from 'change_detection/parser/lexer'; import {Lexer} from 'change_detection/parser/lexer';

View File

@ -1,6 +1,7 @@
import {ddescribe, describe, it, iit, xit, expect} from 'test_lib/test_lib'; import {ddescribe, describe, it, iit, xit, expect, beforeEach} from 'test_lib/test_lib';
import {List, ListWrapper, MapWrapper} from 'facade/collection'; import {List, ListWrapper, MapWrapper} from 'facade/collection';
import {isPresent} from 'facade/lang';
import {Parser} from 'change_detection/parser/parser'; import {Parser} from 'change_detection/parser/parser';
import {Lexer} from 'change_detection/parser/lexer'; import {Lexer} from 'change_detection/parser/lexer';
import {ClosureMap} from 'change_detection/parser/closure_map'; import {ClosureMap} from 'change_detection/parser/closure_map';
@ -11,180 +12,125 @@ import {
WatchGroup, WatchGroup,
WatchGroupDispatcher, WatchGroupDispatcher,
ProtoRecord ProtoRecord
} from 'change_detection/change_detector'; } from 'change_detection/change_detector';
import {Record} from 'change_detection/record'; import {Record} from 'change_detection/record';
export function main() { export function main() {
function humanize(wg:WatchGroup, names:List) {
var lookupName = (item) =>
ListWrapper.last(
ListWrapper.find(names, (pair) => pair[0] === item));
var res = [];
var record = wg.findFirstEnabledRecord();
while (isPresent(record)) {
ListWrapper.push(res, lookupName(record));
record = record.nextEnabled;
}
return res;
}
function createRecord(wg) { function createRecord(wg) {
return new Record(wg, new ProtoRecord(null, null, null, null, null), null); return new Record(wg, new ProtoRecord(null, null, null, null, null), null);
} }
describe('watch group', () => { describe('watch group', () => {
describe("adding records", () => { it("should add records", () => {
it("should add a record", () => { var wg = new WatchGroup(null, null);
var wg = new WatchGroup(null, null); var record1 = createRecord(wg);
var record = createRecord(wg); var record2 = createRecord(wg);
wg.addRecord(record); wg.addRecord(record1);
wg.addRecord(record2);
expect(wg.headRecord).toBe(record); expect(humanize(wg, [
expect(wg.tailRecord).toBe(record); [record1, 'record1'],
expect(wg.headEnabledRecord).toBe(record); [record2, 'record2']
expect(wg.tailEnabledRecord).toBe(record); ])).toEqual(['record1', 'record2']);
});
it("should add multiple records", () => {
var wg = new WatchGroup(null, null);
var record1 = createRecord(wg);
var record2 = createRecord(wg);
wg.addRecord(record1);
wg.addRecord(record2);
expect(wg.headRecord).toBe(record1);
expect(wg.tailRecord).toBe(record2);
expect(wg.headEnabledRecord).toBe(record1);
expect(wg.tailEnabledRecord).toBe(record2);
expect(record1.next).toBe(record2);
expect(record2.prev).toBe(record1);
});
}); });
describe("adding children", () => { describe("adding/removing child groups", () => {
it("should add child watch group", () => { var parent, child1, child2;
var parent = new WatchGroup(null, null); var childRecord1, childRecord2;
var child1 = new WatchGroup(null, null); var recordNames;
var child2 = new WatchGroup(null, null);
beforeEach(() => {
parent = new WatchGroup(null, null);
child1 = new WatchGroup(null, null);
childRecord1 = createRecord(child1);
child1.addRecord(childRecord1);
child2 = new WatchGroup(null, null);
childRecord2 = createRecord(child2);
child2.addRecord(childRecord2);
recordNames = [
[childRecord1, 'record1'],
[childRecord2, 'record2'],
];
});
it("should add child groups", () => {
parent.addChild(child1); parent.addChild(child1);
parent.addChild(child2); parent.addChild(child2);
expect(parent.childHead).toBe(child1); expect(humanize(parent, recordNames)).toEqual(['record1', 'record2']);
expect(parent.childTail).toBe(child2);
expect(child1.next).toBe(child2);
expect(child2.prev).toBe(child1);
}); });
it("should link all records", () => { it("should remove children", () => {
var parent = new WatchGroup(null, null); parent.addChild(child1);
var parentRecord = createRecord(parent); parent.addChild(child2);
parent.addRecord(parentRecord);
var child = new WatchGroup(null, null); parent.removeChild(child1);
var childRecord = createRecord(child);
child.addRecord(childRecord);
parent.addChild(child); expect(humanize(parent, recordNames)).toEqual(['record2']);
expect(parent.headRecord).toBe(parentRecord); parent.removeChild(child2);
expect(parent.tailRecord).toBe(childRecord);
expect(parent.headEnabledRecord).toBe(parentRecord); expect(humanize(parent, recordNames)).toEqual([]);
expect(parent.tailEnabledRecord).toBe(childRecord);
expect(parentRecord.next).toBe(childRecord);
expect(childRecord.prev).toBe(parentRecord);
});
it("should work when parent has no records", () => {
var parent = new WatchGroup(null, null);
var child = new WatchGroup(null, null);
var childRecord = createRecord(child);
child.addRecord(childRecord);
parent.addChild(child);
expect(parent.headRecord).toBe(childRecord);
expect(parent.tailRecord).toBe(childRecord);
expect(parent.headEnabledRecord).toBe(childRecord);
expect(parent.tailEnabledRecord).toBe(childRecord);
});
it("should work when parent has no records and first child has no records", () => {
var parent = new WatchGroup(null, null);
var firstChild = new WatchGroup(null, null);
parent.addChild(firstChild);
var child = new WatchGroup(null, null);
var childRecord = createRecord(child);
child.addRecord(childRecord);
parent.addChild(child);
expect(parent.headRecord).toBe(childRecord);
expect(parent.tailRecord).toBe(childRecord);
expect(parent.headEnabledRecord).toBe(childRecord);
expect(parent.tailEnabledRecord).toBe(childRecord);
});
it("should work when second child has no records", () => {
var parent = new WatchGroup(null, null);
var firstChild = new WatchGroup(null, null);
var childRecord = createRecord(firstChild);
firstChild.addRecord(childRecord);
parent.addChild(firstChild);
var secondChild = new WatchGroup(null, null);
parent.addChild(secondChild);
expect(parent.childHead).toBe(firstChild);
expect(parent.childTail).toBe(secondChild);
});
// todo: vsavkin: enable after refactoring addChild
xit("should update head and tail of the parent when disabling the only record" +
"of the child", () => {
var parent = new WatchGroup(null, null);
var child = new WatchGroup(null, null);
var record = createRecord(child);
child.addRecord(record);
parent.addChild(child);
child.disableRecord(record);
expect(parent.headRecord).toBeNull();
expect(parent.tailRecord).toBeNull();
}); });
}); });
describe("enabling/disabling records", () => { describe("enabling/disabling records", () => {
var wg;
var record1, record2, record3, record4;
var recordNames;
beforeEach(() => {
wg = new WatchGroup(null, null);
record1 = createRecord(wg);
record2 = createRecord(wg);
record3 = createRecord(wg);
record4 = createRecord(wg);
recordNames = [
[record1, 'record1'],
[record2, 'record2'],
[record3, 'record3'],
[record4, 'record4']
];
});
it("should disable a single record", () => { it("should disable a single record", () => {
var wg = new WatchGroup(null, null); wg.addRecord(record1);
var record = createRecord(wg);
wg.addRecord(record);
wg.disableRecord(record); wg.disableRecord(record1);
expect(wg.headEnabledRecord).toBeNull(); expect(humanize(wg, recordNames)).toEqual([]);
expect(wg.tailEnabledRecord).toBeNull();
}); });
it("should enable a single record", () => { it("should enable a single record", () => {
var wg = new WatchGroup(null, null); wg.addRecord(record1);
var record = createRecord(wg); wg.disableRecord(record1);
wg.addRecord(record);
wg.disableRecord(record);
wg.enableRecord(record); wg.enableRecord(record1);
expect(wg.headEnabledRecord).toBe(record); expect(humanize(wg, recordNames)).toEqual(['record1']);
expect(wg.tailEnabledRecord).toBe(record);
}); });
it("should disable a record", () => { it("should disable a record", () => {
var wg = new WatchGroup(null, null);
var record1 = createRecord(wg);
var record2 = createRecord(wg);
var record3 = createRecord(wg);
var record4 = createRecord(wg);
wg.addRecord(record1); wg.addRecord(record1);
wg.addRecord(record2); wg.addRecord(record2);
wg.addRecord(record3); wg.addRecord(record3);
@ -194,20 +140,12 @@ export function main() {
wg.disableRecord(record3); wg.disableRecord(record3);
expect(record2.disabled).toBeTruthy(); expect(record2.disabled).toBeTruthy();
expect(record3.disabled).toBeTruthy();
expect(wg.headEnabledRecord).toBe(record1); expect(humanize(wg, recordNames)).toEqual(['record1', 'record4']);
expect(wg.tailEnabledRecord).toBe(record4);
expect(record1.nextEnabled).toBe(record4);
expect(record4.prevEnabled).toBe(record1);
}); });
it("should enable a record", () => { it("should enable a record", () => {
var wg = new WatchGroup(null, null);
var record1 = createRecord(wg);
var record2 = createRecord(wg);
var record3 = createRecord(wg);
var record4 = createRecord(wg);
wg.addRecord(record1); wg.addRecord(record1);
wg.addRecord(record2); wg.addRecord(record2);
wg.addRecord(record3); wg.addRecord(record3);
@ -218,11 +156,88 @@ export function main() {
wg.enableRecord(record2); wg.enableRecord(record2);
wg.enableRecord(record3); wg.enableRecord(record3);
expect(record1.nextEnabled).toBe(record2); expect(humanize(wg, recordNames)).toEqual(['record1', 'record2', 'record3', 'record4']);
expect(record2.nextEnabled).toBe(record3);
expect(record3.nextEnabled).toBe(record4);
expect(record4.prevEnabled).toBe(record3);
}); });
}) });
describe("enabling/disabling child groups", () => {
var child1, child2, child3, child4;
var record1, record2, record3, record4;
var recordNames;
beforeEach(() => {
child1 = new WatchGroup(null, null);
record1 = createRecord(child1);
child1.addRecord(record1);
child2 = new WatchGroup(null, null);
record2 = createRecord(child2);
child2.addRecord(record2);
child3 = new WatchGroup(null, null);
record3 = createRecord(child3);
child3.addRecord(record3);
child4 = new WatchGroup(null, null);
record4 = createRecord(child4);
child4.addRecord(record4);
recordNames = [
[record1, 'record1'],
[record2, 'record2'],
[record3, 'record3'],
[record4, 'record4']
];
});
it("should disable a single watch group", () => {
var parent = new WatchGroup(null, null);
parent.addChild(child1);
parent.disableGroup(child1);
expect(humanize(parent, recordNames)).toEqual([]);
});
it("should enable a single watch group", () => {
var parent = new WatchGroup(null, null);
parent.addChild(child1);
parent.disableGroup(child1);
parent.enableGroup(child1);
expect(humanize(parent, recordNames)).toEqual(['record1']);
});
it("should disable a watch group", () => {
var parent = new WatchGroup(null, null);
parent.addChild(child1);
parent.addChild(child2);
parent.addChild(child3);
parent.addChild(child4);
parent.disableGroup(child2);
parent.disableGroup(child3);
expect(humanize(parent, recordNames)).toEqual(['record1', 'record4']);
});
it("should enable a watch group", () => {
var parent = new WatchGroup(null, null);
parent.addChild(child1);
parent.addChild(child2);
parent.addChild(child3);
parent.addChild(child4);
parent.disableGroup(child2);
parent.disableGroup(child3);
parent.enableGroup(child2);
parent.enableGroup(child3);
expect(humanize(parent, recordNames)).toEqual([
'record1', 'record2', 'record3', 'record4'
]);
});
});
}); });
} }