UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

366 lines 48 kB
'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"]}