UNPKG

angular2

Version:

Angular 2 - a web framework for modern web apps

189 lines 25.6 kB
'use strict';"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 core_1 = require('angular2/core'); var lang_1 = require('angular2/src/facade/lang'); var exceptions_1 = require("../../facade/exceptions"); /** * The `NgFor` directive instantiates a template once per item from an iterable. The context for * each instantiated template inherits from the outer context with the given loop variable set * to the current item from the iterable. * * ### Local Variables * * `NgFor` provides several exported values that can be aliased to local variables: * * * `index` will be set to the current loop iteration for each template context. * * `first` will be set to a boolean value indicating whether the item is the first one in the * iteration. * * `last` will be set to a boolean value indicating whether the item is the last one in the * iteration. * * `even` will be set to a boolean value indicating whether this item has an even index. * * `odd` will be set to a boolean value indicating whether this item has an odd index. * * ### Change Propagation * * When the contents of the iterator changes, `NgFor` makes the corresponding changes to the DOM: * * * When an item is added, a new instance of the template is added to the DOM. * * When an item is removed, its template instance is removed from the DOM. * * When items are reordered, their respective templates are reordered in the DOM. * * Otherwise, the DOM element for that item will remain the same. * * Angular uses object identity to track insertions and deletions within the iterator and reproduce * those changes in the DOM. This has important implications for animations and any stateful * controls * (such as `<input>` elements which accept user input) that are present. Inserted rows can be * animated in, deleted rows can be animated out, and unchanged rows retain any unsaved state such * as user input. * * It is possible for the identities of elements in the iterator to change while the data does not. * This can happen, for example, if the iterator produced from an RPC to the server, and that * RPC is re-run. Even if the data hasn't changed, the second response will produce objects with * different identities, and Angular will tear down the entire DOM and rebuild it (as if all old * elements were deleted and all new elements inserted). This is an expensive operation and should * be avoided if possible. * * ### Syntax * * - `<li *ngFor="let item of items; #i = index">...</li>` * - `<li template="ngFor #item of items; #i = index">...</li>` * - `<template ngFor #item [ngForOf]="items" #i="index"><li>...</li></template>` * * ### Example * * See a [live demo](http://plnkr.co/edit/KVuXxDp0qinGDyo307QW?p=preview) for a more detailed * example. */ var NgFor = (function () { function NgFor(_viewContainer, _templateRef, _iterableDiffers, _cdr) { this._viewContainer = _viewContainer; this._templateRef = _templateRef; this._iterableDiffers = _iterableDiffers; this._cdr = _cdr; } Object.defineProperty(NgFor.prototype, "ngForOf", { set: function (value) { this._ngForOf = value; if (lang_1.isBlank(this._differ) && lang_1.isPresent(value)) { try { this._differ = this._iterableDiffers.find(value).create(this._cdr, this._ngForTrackBy); } catch (e) { throw new exceptions_1.BaseException("Cannot find a differ supporting object '" + value + "' of type '" + lang_1.getTypeNameForDebugging(value) + "'. NgFor only supports binding to Iterables such as Arrays."); } } }, enumerable: true, configurable: true }); Object.defineProperty(NgFor.prototype, "ngForTemplate", { set: function (value) { if (lang_1.isPresent(value)) { this._templateRef = value; } }, enumerable: true, configurable: true }); Object.defineProperty(NgFor.prototype, "ngForTrackBy", { set: function (value) { this._ngForTrackBy = value; }, enumerable: true, configurable: true }); NgFor.prototype.ngDoCheck = function () { if (lang_1.isPresent(this._differ)) { var changes = this._differ.diff(this._ngForOf); if (lang_1.isPresent(changes)) this._applyChanges(changes); } }; NgFor.prototype._applyChanges = function (changes) { var _this = this; // TODO(rado): check if change detection can produce a change record that is // easier to consume than current. var recordViewTuples = []; changes.forEachRemovedItem(function (removedRecord) { return recordViewTuples.push(new RecordViewTuple(removedRecord, null)); }); changes.forEachMovedItem(function (movedRecord) { return recordViewTuples.push(new RecordViewTuple(movedRecord, null)); }); var insertTuples = this._bulkRemove(recordViewTuples); changes.forEachAddedItem(function (addedRecord) { return insertTuples.push(new RecordViewTuple(addedRecord, null)); }); this._bulkInsert(insertTuples); for (var i = 0; i < insertTuples.length; i++) { this._perViewChange(insertTuples[i].view, insertTuples[i].record); } for (var i = 0, ilen = this._viewContainer.length; i < ilen; i++) { var viewRef = this._viewContainer.get(i); viewRef.setLocal('first', i === 0); viewRef.setLocal('last', i === ilen - 1); } changes.forEachIdentityChange(function (record) { var viewRef = _this._viewContainer.get(record.currentIndex); viewRef.setLocal('\$implicit', record.item); }); }; NgFor.prototype._perViewChange = function (view, record) { view.setLocal('\$implicit', record.item); view.setLocal('index', record.currentIndex); view.setLocal('even', (record.currentIndex % 2 == 0)); view.setLocal('odd', (record.currentIndex % 2 == 1)); }; NgFor.prototype._bulkRemove = function (tuples) { tuples.sort(function (a, b) { return a.record.previousIndex - b.record.previousIndex; }); var movedTuples = []; for (var i = tuples.length - 1; i >= 0; i--) { var tuple = tuples[i]; // separate moved views from removed views. if (lang_1.isPresent(tuple.record.currentIndex)) { tuple.view = this._viewContainer.detach(tuple.record.previousIndex); movedTuples.push(tuple); } else { this._viewContainer.remove(tuple.record.previousIndex); } } return movedTuples; }; NgFor.prototype._bulkInsert = function (tuples) { tuples.sort(function (a, b) { return a.record.currentIndex - b.record.currentIndex; }); for (var i = 0; i < tuples.length; i++) { var tuple = tuples[i]; if (lang_1.isPresent(tuple.view)) { this._viewContainer.insert(tuple.view, tuple.record.currentIndex); } else { tuple.view = this._viewContainer.createEmbeddedView(this._templateRef, tuple.record.currentIndex); } } return tuples; }; NgFor = __decorate([ core_1.Directive({ selector: '[ngFor][ngForOf]', inputs: ['ngForTrackBy', 'ngForOf', 'ngForTemplate'] }), __metadata('design:paramtypes', [core_1.ViewContainerRef, core_1.TemplateRef, core_1.IterableDiffers, core_1.ChangeDetectorRef]) ], NgFor); return NgFor; }()); exports.NgFor = NgFor; var RecordViewTuple = (function () { function RecordViewTuple(record, view) { this.record = record; this.view = view; } return RecordViewTuple; }()); //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng_for.js","sourceRoot":"","sources":["diffing_plugin_wrapper-output_path-BRJer1J9.tmp/angular2/src/common/directives/ng_for.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  DoCheck,\n  Directive,\n  ChangeDetectorRef,\n  IterableDiffer,\n  IterableDiffers,\n  ViewContainerRef,\n  TemplateRef,\n  EmbeddedViewRef,\n  TrackByFn\n} from 'angular2/core';\nimport {isPresent, isBlank, stringify, getTypeNameForDebugging} from 'angular2/src/facade/lang';\nimport {\n  DefaultIterableDiffer,\n  CollectionChangeRecord\n} from \"../../core/change_detection/differs/default_iterable_differ\";\nimport {BaseException} from \"../../facade/exceptions\";\n\n/**\n * The `NgFor` directive instantiates a template once per item from an iterable. The context for\n * each instantiated template inherits from the outer context with the given loop variable set\n * to the current item from the iterable.\n *\n * ### Local Variables\n *\n * `NgFor` provides several exported values that can be aliased to local variables:\n *\n * * `index` will be set to the current loop iteration for each template context.\n * * `first` will be set to a boolean value indicating whether the item is the first one in the\n *   iteration.\n * * `last` will be set to a boolean value indicating whether the item is the last one in the\n *   iteration.\n * * `even` will be set to a boolean value indicating whether this item has an even index.\n * * `odd` will be set to a boolean value indicating whether this item has an odd index.\n *\n * ### Change Propagation\n *\n * When the contents of the iterator changes, `NgFor` makes the corresponding changes to the DOM:\n *\n * * When an item is added, a new instance of the template is added to the DOM.\n * * When an item is removed, its template instance is removed from the DOM.\n * * When items are reordered, their respective templates are reordered in the DOM.\n * * Otherwise, the DOM element for that item will remain the same.\n *\n * Angular uses object identity to track insertions and deletions within the iterator and reproduce\n * those changes in the DOM. This has important implications for animations and any stateful\n * controls\n * (such as `<input>` elements which accept user input) that are present. Inserted rows can be\n * animated in, deleted rows can be animated out, and unchanged rows retain any unsaved state such\n * as user input.\n *\n * It is possible for the identities of elements in the iterator to change while the data does not.\n * This can happen, for example, if the iterator produced from an RPC to the server, and that\n * RPC is re-run. Even if the data hasn't changed, the second response will produce objects with\n * different identities, and Angular will tear down the entire DOM and rebuild it (as if all old\n * elements were deleted and all new elements inserted). This is an expensive operation and should\n * be avoided if possible.\n *\n * ### Syntax\n *\n * - `<li *ngFor=\"let item of items; #i = index\">...</li>`\n * - `<li template=\"ngFor #item of items; #i = index\">...</li>`\n * - `<template ngFor #item [ngForOf]=\"items\" #i=\"index\"><li>...</li></template>`\n *\n * ### Example\n *\n * See a [live demo](http://plnkr.co/edit/KVuXxDp0qinGDyo307QW?p=preview) for a more detailed\n * example.\n */\n@Directive({selector: '[ngFor][ngForOf]', inputs: ['ngForTrackBy', 'ngForOf', 'ngForTemplate']})\nexport class NgFor implements DoCheck {\n  /** @internal */\n  _ngForOf: any;\n  /** @internal */\n  _ngForTrackBy: TrackByFn;\n  private _differ: IterableDiffer;\n\n  constructor(private _viewContainer: ViewContainerRef, private _templateRef: TemplateRef,\n              private _iterableDiffers: IterableDiffers, private _cdr: ChangeDetectorRef) {}\n\n  set ngForOf(value: any) {\n    this._ngForOf = value;\n    if (isBlank(this._differ) && isPresent(value)) {\n      try {\n        this._differ = this._iterableDiffers.find(value).create(this._cdr, this._ngForTrackBy);\n      } catch (e) {\n        throw new BaseException(\n            `Cannot find a differ supporting object '${value}' of type '${getTypeNameForDebugging(value)}'. NgFor only supports binding to Iterables such as Arrays.`);\n      }\n    }\n  }\n\n  set ngForTemplate(value: TemplateRef) {\n    if (isPresent(value)) {\n      this._templateRef = value;\n    }\n  }\n\n  set ngForTrackBy(value: TrackByFn) { this._ngForTrackBy = value; }\n\n  ngDoCheck() {\n    if (isPresent(this._differ)) {\n      var changes = this._differ.diff(this._ngForOf);\n      if (isPresent(changes)) this._applyChanges(changes);\n    }\n  }\n\n  private _applyChanges(changes: DefaultIterableDiffer) {\n    // TODO(rado): check if change detection can produce a change record that is\n    // easier to consume than current.\n    var recordViewTuples: RecordViewTuple[] = [];\n    changes.forEachRemovedItem((removedRecord: CollectionChangeRecord) =>\n                                   recordViewTuples.push(new RecordViewTuple(removedRecord, null)));\n\n    changes.forEachMovedItem((movedRecord: CollectionChangeRecord) =>\n                                 recordViewTuples.push(new RecordViewTuple(movedRecord, null)));\n\n    var insertTuples = this._bulkRemove(recordViewTuples);\n\n    changes.forEachAddedItem((addedRecord: CollectionChangeRecord) =>\n                                 insertTuples.push(new RecordViewTuple(addedRecord, null)));\n\n    this._bulkInsert(insertTuples);\n\n    for (var i = 0; i < insertTuples.length; i++) {\n      this._perViewChange(insertTuples[i].view, insertTuples[i].record);\n    }\n\n    for (var i = 0, ilen = this._viewContainer.length; i < ilen; i++) {\n      var viewRef = <EmbeddedViewRef>this._viewContainer.get(i);\n      viewRef.setLocal('first', i === 0);\n      viewRef.setLocal('last', i === ilen - 1);\n    }\n\n    changes.forEachIdentityChange((record) => {\n      var viewRef = <EmbeddedViewRef>this._viewContainer.get(record.currentIndex);\n      viewRef.setLocal('\\$implicit', record.item);\n    });\n  }\n\n  private _perViewChange(view: EmbeddedViewRef, record: CollectionChangeRecord) {\n    view.setLocal('\\$implicit', record.item);\n    view.setLocal('index', record.currentIndex);\n    view.setLocal('even', (record.currentIndex % 2 == 0));\n    view.setLocal('odd', (record.currentIndex % 2 == 1));\n  }\n\n  private _bulkRemove(tuples: RecordViewTuple[]): RecordViewTuple[] {\n    tuples.sort((a: RecordViewTuple, b: RecordViewTuple) =>\n                    a.record.previousIndex - b.record.previousIndex);\n    var movedTuples: RecordViewTuple[] = [];\n    for (var i = tuples.length - 1; i >= 0; i--) {\n      var tuple = tuples[i];\n      // separate moved views from removed views.\n      if (isPresent(tuple.record.currentIndex)) {\n        tuple.view = <EmbeddedViewRef>this._viewContainer.detach(tuple.record.previousIndex);\n        movedTuples.push(tuple);\n      } else {\n        this._viewContainer.remove(tuple.record.previousIndex);\n      }\n    }\n    return movedTuples;\n  }\n\n  private _bulkInsert(tuples: RecordViewTuple[]): RecordViewTuple[] {\n    tuples.sort((a, b) => a.record.currentIndex - b.record.currentIndex);\n    for (var i = 0; i < tuples.length; i++) {\n      var tuple = tuples[i];\n      if (isPresent(tuple.view)) {\n        this._viewContainer.insert(tuple.view, tuple.record.currentIndex);\n      } else {\n        tuple.view =\n            this._viewContainer.createEmbeddedView(this._templateRef, tuple.record.currentIndex);\n      }\n    }\n    return tuples;\n  }\n}\n\nclass RecordViewTuple {\n  view: EmbeddedViewRef;\n  record: any;\n  constructor(record: any, view: EmbeddedViewRef) {\n    this.record = record;\n    this.view = view;\n  }\n}\n"]}