angular2
Version:
Angular 2 - a web framework for modern web apps
366 lines • 48 kB
JavaScript
'use strict';var __decorate = (this && this.__decorate) || function (decorators, target, key, desc) {
var c = arguments.length, r = c < 3 ? target : desc === null ? desc = Object.getOwnPropertyDescriptor(target, key) : desc, d;
if (typeof Reflect === "object" && typeof Reflect.decorate === "function") r = Reflect.decorate(decorators, target, key, desc);
else for (var i = decorators.length - 1; i >= 0; i--) if (d = decorators[i]) r = (c < 3 ? d(r) : c > 3 ? d(target, key, r) : d(target, key)) || r;
return c > 3 && r && Object.defineProperty(target, key, r), r;
};
var __metadata = (this && this.__metadata) || function (k, v) {
if (typeof Reflect === "object" && typeof Reflect.metadata === "function") return Reflect.metadata(k, v);
};
var collection_1 = require('angular2/src/facade/collection');
var lang_1 = require('angular2/src/facade/lang');
var exceptions_1 = require('angular2/src/facade/exceptions');
var DefaultKeyValueDifferFactory = (function () {
function DefaultKeyValueDifferFactory() {
}
DefaultKeyValueDifferFactory.prototype.supports = function (obj) { return obj instanceof Map || lang_1.isJsObject(obj); };
DefaultKeyValueDifferFactory.prototype.create = function (cdRef) { return new DefaultKeyValueDiffer(); };
DefaultKeyValueDifferFactory = __decorate([
lang_1.CONST(),
__metadata('design:paramtypes', [])
], DefaultKeyValueDifferFactory);
return DefaultKeyValueDifferFactory;
})();
exports.DefaultKeyValueDifferFactory = DefaultKeyValueDifferFactory;
var DefaultKeyValueDiffer = (function () {
function DefaultKeyValueDiffer() {
this._records = new Map();
this._mapHead = null;
this._previousMapHead = null;
this._changesHead = null;
this._changesTail = null;
this._additionsHead = null;
this._additionsTail = null;
this._removalsHead = null;
this._removalsTail = null;
}
Object.defineProperty(DefaultKeyValueDiffer.prototype, "isDirty", {
get: function () {
return this._additionsHead !== null || this._changesHead !== null ||
this._removalsHead !== null;
},
enumerable: true,
configurable: true
});
DefaultKeyValueDiffer.prototype.forEachItem = function (fn) {
var record;
for (record = this._mapHead; record !== null; record = record._next) {
fn(record);
}
};
DefaultKeyValueDiffer.prototype.forEachPreviousItem = function (fn) {
var record;
for (record = this._previousMapHead; record !== null; record = record._nextPrevious) {
fn(record);
}
};
DefaultKeyValueDiffer.prototype.forEachChangedItem = function (fn) {
var record;
for (record = this._changesHead; record !== null; record = record._nextChanged) {
fn(record);
}
};
DefaultKeyValueDiffer.prototype.forEachAddedItem = function (fn) {
var record;
for (record = this._additionsHead; record !== null; record = record._nextAdded) {
fn(record);
}
};
DefaultKeyValueDiffer.prototype.forEachRemovedItem = function (fn) {
var record;
for (record = this._removalsHead; record !== null; record = record._nextRemoved) {
fn(record);
}
};
DefaultKeyValueDiffer.prototype.diff = function (map) {
if (lang_1.isBlank(map))
map = collection_1.MapWrapper.createFromPairs([]);
if (!(map instanceof Map || lang_1.isJsObject(map))) {
throw new exceptions_1.BaseException("Error trying to diff '" + map + "'");
}
if (this.check(map)) {
return this;
}
else {
return null;
}
};
DefaultKeyValueDiffer.prototype.onDestroy = function () { };
DefaultKeyValueDiffer.prototype.check = function (map) {
var _this = this;
this._reset();
var records = this._records;
var oldSeqRecord = this._mapHead;
var lastOldSeqRecord = null;
var lastNewSeqRecord = null;
var seqChanged = false;
this._forEach(map, function (value, key) {
var newSeqRecord;
if (oldSeqRecord !== null && key === oldSeqRecord.key) {
newSeqRecord = oldSeqRecord;
if (!lang_1.looseIdentical(value, oldSeqRecord.currentValue)) {
oldSeqRecord.previousValue = oldSeqRecord.currentValue;
oldSeqRecord.currentValue = value;
_this._addToChanges(oldSeqRecord);
}
}
else {
seqChanged = true;
if (oldSeqRecord !== null) {
oldSeqRecord._next = null;
_this._removeFromSeq(lastOldSeqRecord, oldSeqRecord);
_this._addToRemovals(oldSeqRecord);
}
if (records.has(key)) {
newSeqRecord = records.get(key);
}
else {
newSeqRecord = new KVChangeRecord(key);
records.set(key, newSeqRecord);
newSeqRecord.currentValue = value;
_this._addToAdditions(newSeqRecord);
}
}
if (seqChanged) {
if (_this._isInRemovals(newSeqRecord)) {
_this._removeFromRemovals(newSeqRecord);
}
if (lastNewSeqRecord == null) {
_this._mapHead = newSeqRecord;
}
else {
lastNewSeqRecord._next = newSeqRecord;
}
}
lastOldSeqRecord = oldSeqRecord;
lastNewSeqRecord = newSeqRecord;
oldSeqRecord = oldSeqRecord === null ? null : oldSeqRecord._next;
});
this._truncate(lastOldSeqRecord, oldSeqRecord);
return this.isDirty;
};
/** @internal */
DefaultKeyValueDiffer.prototype._reset = function () {
if (this.isDirty) {
var record;
// Record the state of the mapping
for (record = this._previousMapHead = this._mapHead; record !== null; record = record._next) {
record._nextPrevious = record._next;
}
for (record = this._changesHead; record !== null; record = record._nextChanged) {
record.previousValue = record.currentValue;
}
for (record = this._additionsHead; record != null; record = record._nextAdded) {
record.previousValue = record.currentValue;
}
// todo(vicb) once assert is supported
// assert(() {
// var r = _changesHead;
// while (r != null) {
// var nextRecord = r._nextChanged;
// r._nextChanged = null;
// r = nextRecord;
// }
//
// r = _additionsHead;
// while (r != null) {
// var nextRecord = r._nextAdded;
// r._nextAdded = null;
// r = nextRecord;
// }
//
// r = _removalsHead;
// while (r != null) {
// var nextRecord = r._nextRemoved;
// r._nextRemoved = null;
// r = nextRecord;
// }
//
// return true;
//});
this._changesHead = this._changesTail = null;
this._additionsHead = this._additionsTail = null;
this._removalsHead = this._removalsTail = null;
}
};
/** @internal */
DefaultKeyValueDiffer.prototype._truncate = function (lastRecord, record) {
while (record !== null) {
if (lastRecord === null) {
this._mapHead = null;
}
else {
lastRecord._next = null;
}
var nextRecord = record._next;
// todo(vicb) assert
// assert((() {
// record._next = null;
// return true;
//}));
this._addToRemovals(record);
lastRecord = record;
record = nextRecord;
}
for (var rec = this._removalsHead; rec !== null; rec = rec._nextRemoved) {
rec.previousValue = rec.currentValue;
rec.currentValue = null;
this._records.delete(rec.key);
}
};
/** @internal */
DefaultKeyValueDiffer.prototype._isInRemovals = function (record) {
return record === this._removalsHead || record._nextRemoved !== null ||
record._prevRemoved !== null;
};
/** @internal */
DefaultKeyValueDiffer.prototype._addToRemovals = function (record) {
// todo(vicb) assert
// assert(record._next == null);
// assert(record._nextAdded == null);
// assert(record._nextChanged == null);
// assert(record._nextRemoved == null);
// assert(record._prevRemoved == null);
if (this._removalsHead === null) {
this._removalsHead = this._removalsTail = record;
}
else {
this._removalsTail._nextRemoved = record;
record._prevRemoved = this._removalsTail;
this._removalsTail = record;
}
};
/** @internal */
DefaultKeyValueDiffer.prototype._removeFromSeq = function (prev, record) {
var next = record._next;
if (prev === null) {
this._mapHead = next;
}
else {
prev._next = next;
}
// todo(vicb) assert
// assert((() {
// record._next = null;
// return true;
//})());
};
/** @internal */
DefaultKeyValueDiffer.prototype._removeFromRemovals = function (record) {
// todo(vicb) assert
// assert(record._next == null);
// assert(record._nextAdded == null);
// assert(record._nextChanged == null);
var prev = record._prevRemoved;
var next = record._nextRemoved;
if (prev === null) {
this._removalsHead = next;
}
else {
prev._nextRemoved = next;
}
if (next === null) {
this._removalsTail = prev;
}
else {
next._prevRemoved = prev;
}
record._prevRemoved = record._nextRemoved = null;
};
/** @internal */
DefaultKeyValueDiffer.prototype._addToAdditions = function (record) {
// todo(vicb): assert
// assert(record._next == null);
// assert(record._nextAdded == null);
// assert(record._nextChanged == null);
// assert(record._nextRemoved == null);
// assert(record._prevRemoved == null);
if (this._additionsHead === null) {
this._additionsHead = this._additionsTail = record;
}
else {
this._additionsTail._nextAdded = record;
this._additionsTail = record;
}
};
/** @internal */
DefaultKeyValueDiffer.prototype._addToChanges = function (record) {
// todo(vicb) assert
// assert(record._nextAdded == null);
// assert(record._nextChanged == null);
// assert(record._nextRemoved == null);
// assert(record._prevRemoved == null);
if (this._changesHead === null) {
this._changesHead = this._changesTail = record;
}
else {
this._changesTail._nextChanged = record;
this._changesTail = record;
}
};
DefaultKeyValueDiffer.prototype.toString = function () {
var items = [];
var previous = [];
var changes = [];
var additions = [];
var removals = [];
var record;
for (record = this._mapHead; record !== null; record = record._next) {
items.push(lang_1.stringify(record));
}
for (record = this._previousMapHead; record !== null; record = record._nextPrevious) {
previous.push(lang_1.stringify(record));
}
for (record = this._changesHead; record !== null; record = record._nextChanged) {
changes.push(lang_1.stringify(record));
}
for (record = this._additionsHead; record !== null; record = record._nextAdded) {
additions.push(lang_1.stringify(record));
}
for (record = this._removalsHead; record !== null; record = record._nextRemoved) {
removals.push(lang_1.stringify(record));
}
return "map: " + items.join(', ') + "\n" + "previous: " + previous.join(', ') + "\n" +
"additions: " + additions.join(', ') + "\n" + "changes: " + changes.join(', ') + "\n" +
"removals: " + removals.join(', ') + "\n";
};
/** @internal */
DefaultKeyValueDiffer.prototype._forEach = function (obj, fn) {
if (obj instanceof Map) {
obj.forEach(fn);
}
else {
collection_1.StringMapWrapper.forEach(obj, fn);
}
};
return DefaultKeyValueDiffer;
})();
exports.DefaultKeyValueDiffer = DefaultKeyValueDiffer;
var KVChangeRecord = (function () {
function KVChangeRecord(key) {
this.key = key;
this.previousValue = null;
this.currentValue = null;
/** @internal */
this._nextPrevious = null;
/** @internal */
this._next = null;
/** @internal */
this._nextAdded = null;
/** @internal */
this._nextRemoved = null;
/** @internal */
this._prevRemoved = null;
/** @internal */
this._nextChanged = null;
}
KVChangeRecord.prototype.toString = function () {
return lang_1.looseIdentical(this.previousValue, this.currentValue) ?
lang_1.stringify(this.key) :
(lang_1.stringify(this.key) + '[' + lang_1.stringify(this.previousValue) + '->' +
lang_1.stringify(this.currentValue) + ']');
};
return KVChangeRecord;
})();
exports.KVChangeRecord = KVChangeRecord;
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"default_keyvalue_differ.js","sourceRoot":"","sources":["angular2/src/core/change_detection/differs/default_keyvalue_differ.ts"],"names":["DefaultKeyValueDifferFactory","DefaultKeyValueDifferFactory.constructor","DefaultKeyValueDifferFactory.supports","DefaultKeyValueDifferFactory.create","DefaultKeyValueDiffer","DefaultKeyValueDiffer.constructor","DefaultKeyValueDiffer.isDirty","DefaultKeyValueDiffer.forEachItem","DefaultKeyValueDiffer.forEachPreviousItem","DefaultKeyValueDiffer.forEachChangedItem","DefaultKeyValueDiffer.forEachAddedItem","DefaultKeyValueDiffer.forEachRemovedItem","DefaultKeyValueDiffer.diff","DefaultKeyValueDiffer.onDestroy","DefaultKeyValueDiffer.check","DefaultKeyValueDiffer._reset","DefaultKeyValueDiffer._truncate","DefaultKeyValueDiffer._isInRemovals","DefaultKeyValueDiffer._addToRemovals","DefaultKeyValueDiffer._removeFromSeq","DefaultKeyValueDiffer._removeFromRemovals","DefaultKeyValueDiffer._addToAdditions","DefaultKeyValueDiffer._addToChanges","DefaultKeyValueDiffer.toString","DefaultKeyValueDiffer._forEach","KVChangeRecord","KVChangeRecord.constructor","KVChangeRecord.toString"],"mappings":"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","sourcesContent":["import {MapWrapper, StringMapWrapper} from 'angular2/src/facade/collection';\nimport {stringify, looseIdentical, isJsObject, CONST, isBlank} from 'angular2/src/facade/lang';\nimport {BaseException} from 'angular2/src/facade/exceptions';\nimport {ChangeDetectorRef} from '../change_detector_ref';\nimport {KeyValueDiffer, KeyValueDifferFactory} from '../differs/keyvalue_differs';\n\n@CONST()\nexport class DefaultKeyValueDifferFactory implements KeyValueDifferFactory {\n  supports(obj: any): boolean { return obj instanceof Map || isJsObject(obj); }\n\n  create(cdRef: ChangeDetectorRef): KeyValueDiffer { return new DefaultKeyValueDiffer(); }\n}\n\nexport class DefaultKeyValueDiffer implements KeyValueDiffer {\n  private _records: Map<any, any> = new Map();\n  private _mapHead: KVChangeRecord = null;\n  private _previousMapHead: KVChangeRecord = null;\n  private _changesHead: KVChangeRecord = null;\n  private _changesTail: KVChangeRecord = null;\n  private _additionsHead: KVChangeRecord = null;\n  private _additionsTail: KVChangeRecord = null;\n  private _removalsHead: KVChangeRecord = null;\n  private _removalsTail: KVChangeRecord = null;\n\n  get isDirty(): boolean {\n    return this._additionsHead !== null || this._changesHead !== null ||\n           this._removalsHead !== null;\n  }\n\n  forEachItem(fn: Function) {\n    var record: KVChangeRecord;\n    for (record = this._mapHead; record !== null; record = record._next) {\n      fn(record);\n    }\n  }\n\n  forEachPreviousItem(fn: Function) {\n    var record: KVChangeRecord;\n    for (record = this._previousMapHead; record !== null; record = record._nextPrevious) {\n      fn(record);\n    }\n  }\n\n  forEachChangedItem(fn: Function) {\n    var record: KVChangeRecord;\n    for (record = this._changesHead; record !== null; record = record._nextChanged) {\n      fn(record);\n    }\n  }\n\n  forEachAddedItem(fn: Function) {\n    var record: KVChangeRecord;\n    for (record = this._additionsHead; record !== null; record = record._nextAdded) {\n      fn(record);\n    }\n  }\n\n  forEachRemovedItem(fn: Function) {\n    var record: KVChangeRecord;\n    for (record = this._removalsHead; record !== null; record = record._nextRemoved) {\n      fn(record);\n    }\n  }\n\n  diff(map: Map<any, any>): any {\n    if (isBlank(map)) map = MapWrapper.createFromPairs([]);\n    if (!(map instanceof Map || isJsObject(map))) {\n      throw new BaseException(`Error trying to diff '${map}'`);\n    }\n\n    if (this.check(map)) {\n      return this;\n    } else {\n      return null;\n    }\n  }\n\n  onDestroy() {}\n\n  check(map: Map<any, any>): boolean {\n    this._reset();\n    var records = this._records;\n    var oldSeqRecord: KVChangeRecord = this._mapHead;\n    var lastOldSeqRecord: KVChangeRecord = null;\n    var lastNewSeqRecord: KVChangeRecord = null;\n    var seqChanged: boolean = false;\n\n    this._forEach(map, (value, key) => {\n      var newSeqRecord;\n      if (oldSeqRecord !== null && key === oldSeqRecord.key) {\n        newSeqRecord = oldSeqRecord;\n        if (!looseIdentical(value, oldSeqRecord.currentValue)) {\n          oldSeqRecord.previousValue = oldSeqRecord.currentValue;\n          oldSeqRecord.currentValue = value;\n          this._addToChanges(oldSeqRecord);\n        }\n      } else {\n        seqChanged = true;\n        if (oldSeqRecord !== null) {\n          oldSeqRecord._next = null;\n          this._removeFromSeq(lastOldSeqRecord, oldSeqRecord);\n          this._addToRemovals(oldSeqRecord);\n        }\n        if (records.has(key)) {\n          newSeqRecord = records.get(key);\n        } else {\n          newSeqRecord = new KVChangeRecord(key);\n          records.set(key, newSeqRecord);\n          newSeqRecord.currentValue = value;\n          this._addToAdditions(newSeqRecord);\n        }\n      }\n\n      if (seqChanged) {\n        if (this._isInRemovals(newSeqRecord)) {\n          this._removeFromRemovals(newSeqRecord);\n        }\n        if (lastNewSeqRecord == null) {\n          this._mapHead = newSeqRecord;\n        } else {\n          lastNewSeqRecord._next = newSeqRecord;\n        }\n      }\n      lastOldSeqRecord = oldSeqRecord;\n      lastNewSeqRecord = newSeqRecord;\n      oldSeqRecord = oldSeqRecord === null ? null : oldSeqRecord._next;\n    });\n    this._truncate(lastOldSeqRecord, oldSeqRecord);\n    return this.isDirty;\n  }\n\n  /** @internal */\n  _reset() {\n    if (this.isDirty) {\n      var record: KVChangeRecord;\n      // Record the state of the mapping\n      for (record = this._previousMapHead = this._mapHead; record !== null; record = record._next) {\n        record._nextPrevious = record._next;\n      }\n\n      for (record = this._changesHead; record !== null; record = record._nextChanged) {\n        record.previousValue = record.currentValue;\n      }\n\n      for (record = this._additionsHead; record != null; record = record._nextAdded) {\n        record.previousValue = record.currentValue;\n      }\n\n      // todo(vicb) once assert is supported\n      // assert(() {\n      //  var r = _changesHead;\n      //  while (r != null) {\n      //    var nextRecord = r._nextChanged;\n      //    r._nextChanged = null;\n      //    r = nextRecord;\n      //  }\n      //\n      //  r = _additionsHead;\n      //  while (r != null) {\n      //    var nextRecord = r._nextAdded;\n      //    r._nextAdded = null;\n      //    r = nextRecord;\n      //  }\n      //\n      //  r = _removalsHead;\n      //  while (r != null) {\n      //    var nextRecord = r._nextRemoved;\n      //    r._nextRemoved = null;\n      //    r = nextRecord;\n      //  }\n      //\n      //  return true;\n      //});\n      this._changesHead = this._changesTail = null;\n      this._additionsHead = this._additionsTail = null;\n      this._removalsHead = this._removalsTail = null;\n    }\n  }\n\n  /** @internal */\n  _truncate(lastRecord: KVChangeRecord, record: KVChangeRecord) {\n    while (record !== null) {\n      if (lastRecord === null) {\n        this._mapHead = null;\n      } else {\n        lastRecord._next = null;\n      }\n      var nextRecord = record._next;\n      // todo(vicb) assert\n      // assert((() {\n      //  record._next = null;\n      //  return true;\n      //}));\n      this._addToRemovals(record);\n      lastRecord = record;\n      record = nextRecord;\n    }\n\n    for (var rec: KVChangeRecord = this._removalsHead; rec !== null; rec = rec._nextRemoved) {\n      rec.previousValue = rec.currentValue;\n      rec.currentValue = null;\n      this._records.delete(rec.key);\n    }\n  }\n\n  /** @internal */\n  _isInRemovals(record: KVChangeRecord) {\n    return record === this._removalsHead || record._nextRemoved !== null ||\n           record._prevRemoved !== null;\n  }\n\n  /** @internal */\n  _addToRemovals(record: KVChangeRecord) {\n    // todo(vicb) assert\n    // assert(record._next == null);\n    // assert(record._nextAdded == null);\n    // assert(record._nextChanged == null);\n    // assert(record._nextRemoved == null);\n    // assert(record._prevRemoved == null);\n    if (this._removalsHead === null) {\n      this._removalsHead = this._removalsTail = record;\n    } else {\n      this._removalsTail._nextRemoved = record;\n      record._prevRemoved = this._removalsTail;\n      this._removalsTail = record;\n    }\n  }\n\n  /** @internal */\n  _removeFromSeq(prev: KVChangeRecord, record: KVChangeRecord) {\n    var next = record._next;\n    if (prev === null) {\n      this._mapHead = next;\n    } else {\n      prev._next = next;\n    }\n    // todo(vicb) assert\n    // assert((() {\n    //  record._next = null;\n    //  return true;\n    //})());\n  }\n\n  /** @internal */\n  _removeFromRemovals(record: KVChangeRecord) {\n    // todo(vicb) assert\n    // assert(record._next == null);\n    // assert(record._nextAdded == null);\n    // assert(record._nextChanged == null);\n\n    var prev = record._prevRemoved;\n    var next = record._nextRemoved;\n    if (prev === null) {\n      this._removalsHead = next;\n    } else {\n      prev._nextRemoved = next;\n    }\n    if (next === null) {\n      this._removalsTail = prev;\n    } else {\n      next._prevRemoved = prev;\n    }\n    record._prevRemoved = record._nextRemoved = null;\n  }\n\n  /** @internal */\n  _addToAdditions(record: KVChangeRecord) {\n    // todo(vicb): assert\n    // assert(record._next == null);\n    // assert(record._nextAdded == null);\n    // assert(record._nextChanged == null);\n    // assert(record._nextRemoved == null);\n    // assert(record._prevRemoved == null);\n    if (this._additionsHead === null) {\n      this._additionsHead = this._additionsTail = record;\n    } else {\n      this._additionsTail._nextAdded = record;\n      this._additionsTail = record;\n    }\n  }\n\n  /** @internal */\n  _addToChanges(record: KVChangeRecord) {\n    // todo(vicb) assert\n    // assert(record._nextAdded == null);\n    // assert(record._nextChanged == null);\n    // assert(record._nextRemoved == null);\n    // assert(record._prevRemoved == null);\n    if (this._changesHead === null) {\n      this._changesHead = this._changesTail = record;\n    } else {\n      this._changesTail._nextChanged = record;\n      this._changesTail = record;\n    }\n  }\n\n  toString(): string {\n    var items = [];\n    var previous = [];\n    var changes = [];\n    var additions = [];\n    var removals = [];\n    var record: KVChangeRecord;\n\n    for (record = this._mapHead; record !== null; record = record._next) {\n      items.push(stringify(record));\n    }\n    for (record = this._previousMapHead; record !== null; record = record._nextPrevious) {\n      previous.push(stringify(record));\n    }\n    for (record = this._changesHead; record !== null; record = record._nextChanged) {\n      changes.push(stringify(record));\n    }\n    for (record = this._additionsHead; record !== null; record = record._nextAdded) {\n      additions.push(stringify(record));\n    }\n    for (record = this._removalsHead; record !== null; record = record._nextRemoved) {\n      removals.push(stringify(record));\n    }\n\n    return \"map: \" + items.join(', ') + \"\\n\" + \"previous: \" + previous.join(', ') + \"\\n\" +\n           \"additions: \" + additions.join(', ') + \"\\n\" + \"changes: \" + changes.join(', ') + \"\\n\" +\n           \"removals: \" + removals.join(', ') + \"\\n\";\n  }\n\n  /** @internal */\n  _forEach(obj, fn: Function) {\n    if (obj instanceof Map) {\n      (<Map<any, any>>obj).forEach(<any>fn);\n    } else {\n      StringMapWrapper.forEach(obj, fn);\n    }\n  }\n}\n\n\nexport class KVChangeRecord {\n  previousValue: any = null;\n  currentValue: any = null;\n\n  /** @internal */\n  _nextPrevious: KVChangeRecord = null;\n  /** @internal */\n  _next: KVChangeRecord = null;\n  /** @internal */\n  _nextAdded: KVChangeRecord = null;\n  /** @internal */\n  _nextRemoved: KVChangeRecord = null;\n  /** @internal */\n  _prevRemoved: KVChangeRecord = null;\n  /** @internal */\n  _nextChanged: KVChangeRecord = null;\n\n  constructor(public key: any) {}\n\n  toString(): string {\n    return looseIdentical(this.previousValue, this.currentValue) ?\n               stringify(this.key) :\n               (stringify(this.key) + '[' + stringify(this.previousValue) + '->' +\n                stringify(this.currentValue) + ']');\n  }\n}\n"]}