2015-09-02 16:43:39 -07:00
|
|
|
import {DoCheck} from 'angular2/lifecycle_hooks';
|
2015-09-03 22:01:36 -07:00
|
|
|
import {Directive} from 'angular2/src/core/metadata';
|
|
|
|
import {
|
|
|
|
ChangeDetectorRef,
|
|
|
|
IterableDiffer,
|
|
|
|
IterableDiffers
|
|
|
|
} from 'angular2/src/core/change_detection';
|
2015-10-02 07:37:23 -07:00
|
|
|
import {ViewContainerRef, TemplateRef, ViewRef} from 'angular2/src/core/linker';
|
2015-08-20 14:28:25 -07:00
|
|
|
import {isPresent, isBlank} from 'angular2/src/core/facade/lang';
|
2014-12-05 17:44:00 -08:00
|
|
|
|
2015-03-31 22:47:11 +00:00
|
|
|
/**
|
2015-05-26 21:53:59 -07:00
|
|
|
* 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.
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-10-12 21:38:32 +00:00
|
|
|
* # 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.
|
|
|
|
* * `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
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-05-26 21:53:59 -07:00
|
|
|
* When the contents of the iterator changes, `NgFor` makes the corresponding changes to the DOM:
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
|
|
|
* * 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.
|
2015-10-12 21:38:32 +00:00
|
|
|
* * Otherwise, the DOM element for that item will remain the same.
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-10-12 21:38:32 +00:00
|
|
|
* 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.
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-10-12 21:38:32 +00:00
|
|
|
* 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.
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-10-12 21:38:32 +00:00
|
|
|
* # Syntax
|
2015-04-06 11:47:38 +02:00
|
|
|
*
|
2015-05-11 17:04:55 -07:00
|
|
|
* - `<li *ng-for="#item of items; #i = index">...</li>`
|
2015-05-16 08:43:07 +02:00
|
|
|
* - `<li template="ng-for #item of items; #i = index">...</li>`
|
2015-06-21 12:15:30 +02:00
|
|
|
* - `<template ng-for #item [ng-for-of]="items" #i="index"><li>...</li></template>`
|
2015-10-12 21:38:32 +00:00
|
|
|
*
|
|
|
|
* ### Example
|
|
|
|
*
|
|
|
|
* See a [live demo](http://plnkr.co/edit/KVuXxDp0qinGDyo307QW?p=preview) for a more detailed
|
|
|
|
* example.
|
2015-03-31 22:47:11 +00:00
|
|
|
*/
|
2015-10-09 12:04:10 -07:00
|
|
|
@Directive({selector: '[ng-for][ng-for-of]', inputs: ['ngForOf', 'ngForTemplate']})
|
2015-08-31 18:32:32 -07:00
|
|
|
export class NgFor implements DoCheck {
|
2015-10-09 17:21:25 -07:00
|
|
|
/** @internal */
|
2015-06-18 15:40:12 -07:00
|
|
|
_ngForOf: any;
|
2015-07-31 12:23:50 -07:00
|
|
|
private _differ: IterableDiffer;
|
2015-06-18 15:40:12 -07:00
|
|
|
|
2015-09-04 15:09:02 -07:00
|
|
|
constructor(private _viewContainer: ViewContainerRef, private _templateRef: TemplateRef,
|
|
|
|
private _iterableDiffers: IterableDiffers, private _cdr: ChangeDetectorRef) {}
|
2015-06-18 15:40:12 -07:00
|
|
|
|
|
|
|
set ngForOf(value: any) {
|
|
|
|
this._ngForOf = value;
|
2015-07-31 12:23:50 -07:00
|
|
|
if (isBlank(this._differ) && isPresent(value)) {
|
2015-09-04 15:09:02 -07:00
|
|
|
this._differ = this._iterableDiffers.find(value).create(this._cdr);
|
2015-07-31 12:23:50 -07:00
|
|
|
}
|
2015-06-18 15:40:12 -07:00
|
|
|
}
|
|
|
|
|
2015-10-09 12:04:10 -07:00
|
|
|
set ngForTemplate(value: TemplateRef) { this._templateRef = value; }
|
|
|
|
|
2015-08-27 21:19:56 -07:00
|
|
|
doCheck() {
|
2015-07-31 12:23:50 -07:00
|
|
|
if (isPresent(this._differ)) {
|
|
|
|
var changes = this._differ.diff(this._ngForOf);
|
|
|
|
if (isPresent(changes)) this._applyChanges(changes);
|
|
|
|
}
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
2015-02-12 14:56:41 -08:00
|
|
|
|
2015-06-18 15:40:12 -07:00
|
|
|
private _applyChanges(changes) {
|
2014-12-05 17:44:00 -08:00
|
|
|
// TODO(rado): check if change detection can produce a change record that is
|
|
|
|
// easier to consume than current.
|
|
|
|
var recordViewTuples = [];
|
2015-06-17 11:17:21 -07:00
|
|
|
changes.forEachRemovedItem((removedRecord) =>
|
|
|
|
recordViewTuples.push(new RecordViewTuple(removedRecord, null)));
|
2014-12-05 17:44:00 -08:00
|
|
|
|
2015-06-17 11:17:21 -07:00
|
|
|
changes.forEachMovedItem((movedRecord) =>
|
|
|
|
recordViewTuples.push(new RecordViewTuple(movedRecord, null)));
|
2014-12-05 17:44:00 -08:00
|
|
|
|
2015-09-14 21:09:40 -07:00
|
|
|
var insertTuples = this._bulkRemove(recordViewTuples);
|
2014-12-05 17:44:00 -08:00
|
|
|
|
2015-06-17 11:17:21 -07:00
|
|
|
changes.forEachAddedItem((addedRecord) =>
|
|
|
|
insertTuples.push(new RecordViewTuple(addedRecord, null)));
|
2014-12-05 17:44:00 -08:00
|
|
|
|
2015-09-14 21:09:40 -07:00
|
|
|
this._bulkInsert(insertTuples);
|
2014-12-05 17:44:00 -08:00
|
|
|
|
|
|
|
for (var i = 0; i < insertTuples.length; i++) {
|
2015-06-18 15:40:12 -07:00
|
|
|
this._perViewChange(insertTuples[i].view, insertTuples[i].record);
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
2015-09-04 16:59:04 +02:00
|
|
|
|
2015-09-04 15:09:02 -07:00
|
|
|
for (var i = 0, ilen = this._viewContainer.length; i < ilen; i++) {
|
|
|
|
this._viewContainer.get(i).setLocal('last', i === ilen - 1);
|
2015-09-04 16:59:04 +02:00
|
|
|
}
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
|
|
|
|
2015-06-18 15:40:12 -07:00
|
|
|
private _perViewChange(view, record) {
|
2015-01-27 22:34:25 -08:00
|
|
|
view.setLocal('\$implicit', record.item);
|
2015-01-28 00:42:08 +01:00
|
|
|
view.setLocal('index', record.currentIndex);
|
2015-09-14 14:07:59 -07:00
|
|
|
view.setLocal('even', (record.currentIndex % 2 == 0));
|
|
|
|
view.setLocal('odd', (record.currentIndex % 2 == 1));
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
|
|
|
|
2015-09-14 21:09:40 -07:00
|
|
|
private _bulkRemove(tuples: RecordViewTuple[]): RecordViewTuple[] {
|
2014-12-05 17:44:00 -08:00
|
|
|
tuples.sort((a, b) => a.record.previousIndex - b.record.previousIndex);
|
|
|
|
var movedTuples = [];
|
|
|
|
for (var i = tuples.length - 1; i >= 0; i--) {
|
|
|
|
var tuple = tuples[i];
|
2015-01-15 13:03:50 -08:00
|
|
|
// separate moved views from removed views.
|
2014-12-05 17:44:00 -08:00
|
|
|
if (isPresent(tuple.record.currentIndex)) {
|
2015-09-14 21:09:40 -07:00
|
|
|
tuple.view = this._viewContainer.detach(tuple.record.previousIndex);
|
2015-06-17 11:17:21 -07:00
|
|
|
movedTuples.push(tuple);
|
2015-01-15 13:03:50 -08:00
|
|
|
} else {
|
2015-09-14 21:09:40 -07:00
|
|
|
this._viewContainer.remove(tuple.record.previousIndex);
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return movedTuples;
|
|
|
|
}
|
|
|
|
|
2015-09-14 21:09:40 -07:00
|
|
|
private _bulkInsert(tuples: RecordViewTuple[]): RecordViewTuple[] {
|
2014-12-05 17:44:00 -08:00
|
|
|
tuples.sort((a, b) => a.record.currentIndex - b.record.currentIndex);
|
|
|
|
for (var i = 0; i < tuples.length; i++) {
|
|
|
|
var tuple = tuples[i];
|
|
|
|
if (isPresent(tuple.view)) {
|
2015-09-14 21:09:40 -07:00
|
|
|
this._viewContainer.insert(tuple.view, tuple.record.currentIndex);
|
2014-12-05 17:44:00 -08:00
|
|
|
} else {
|
2015-09-14 21:09:40 -07:00
|
|
|
tuple.view =
|
|
|
|
this._viewContainer.createEmbeddedView(this._templateRef, tuple.record.currentIndex);
|
2014-12-05 17:44:00 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return tuples;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-09-14 21:09:40 -07:00
|
|
|
class RecordViewTuple {
|
2015-04-28 11:20:01 -07:00
|
|
|
view: ViewRef;
|
2014-12-05 17:44:00 -08:00
|
|
|
record: any;
|
|
|
|
constructor(record, view) {
|
|
|
|
this.record = record;
|
|
|
|
this.view = view;
|
|
|
|
}
|
|
|
|
}
|