2015-09-03 22:01:36 -07:00
|
|
|
import {
|
2015-11-18 15:55:43 -08:00
|
|
|
DoCheck,
|
|
|
|
Directive,
|
2015-09-03 22:01:36 -07:00
|
|
|
ChangeDetectorRef,
|
|
|
|
IterableDiffer,
|
2015-11-18 15:55:43 -08:00
|
|
|
IterableDiffers,
|
|
|
|
ViewContainerRef,
|
|
|
|
TemplateRef,
|
2016-02-01 18:31:26 -08:00
|
|
|
EmbeddedViewRef,
|
|
|
|
TrackByFn
|
2015-11-18 15:55:43 -08:00
|
|
|
} from 'angular2/core';
|
2015-11-06 17:34:07 -08:00
|
|
|
import {isPresent, isBlank} from 'angular2/src/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-11-23 16:02:19 -08:00
|
|
|
* - `<li *ngFor="#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>`
|
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
|
|
|
*/
|
2016-02-01 18:31:26 -08:00
|
|
|
@Directive({selector: '[ngFor][ngForOf]', inputs: ['ngForTrackBy', '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;
|
2016-02-01 18:31:26 -08:00
|
|
|
_ngForTrackBy: TrackByFn;
|
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)) {
|
2016-02-01 18:31:26 -08:00
|
|
|
this._differ = this._iterableDiffers.find(value).create(this._cdr, this._ngForTrackBy);
|
2015-07-31 12:23:50 -07:00
|
|
|
}
|
2015-06-18 15:40:12 -07:00
|
|
|
}
|
|
|
|
|
2015-11-06 16:34:41 +01:00
|
|
|
set ngForTemplate(value: TemplateRef) {
|
|
|
|
if (isPresent(value)) {
|
|
|
|
this._templateRef = value;
|
|
|
|
}
|
|
|
|
}
|
2015-10-09 12:04:10 -07:00
|
|
|
|
2016-02-01 18:31:26 -08:00
|
|
|
set ngForTrackBy(value: TrackByFn) { this._ngForTrackBy = value; }
|
|
|
|
|
refactor(lifecycle): prefix lifecycle methods with "ng"
BREAKING CHANGE:
Previously, components that would implement lifecycle interfaces would include methods
like "onChanges" or "afterViewInit." Given that components were at risk of using such
names without realizing that Angular would call the methods at different points of
the component lifecycle. This change adds an "ng" prefix to all lifecycle hook methods,
far reducing the risk of an accidental name collision.
To fix, just rename these methods:
* onInit
* onDestroy
* doCheck
* onChanges
* afterContentInit
* afterContentChecked
* afterViewInit
* afterViewChecked
* _Router Hooks_
* onActivate
* onReuse
* onDeactivate
* canReuse
* canDeactivate
To:
* ngOnInit,
* ngOnDestroy,
* ngDoCheck,
* ngOnChanges,
* ngAfterContentInit,
* ngAfterContentChecked,
* ngAfterViewInit,
* ngAfterViewChecked
* _Router Hooks_
* routerOnActivate
* routerOnReuse
* routerOnDeactivate
* routerCanReuse
* routerCanDeactivate
The names of lifecycle interfaces and enums have not changed, though interfaces
have been updated to reflect the new method names.
Closes #5036
2015-11-16 17:04:36 -08:00
|
|
|
ngDoCheck() {
|
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++) {
|
2015-12-02 10:35:51 -08:00
|
|
|
var viewRef = <EmbeddedViewRef>this._viewContainer.get(i);
|
|
|
|
viewRef.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-12-02 10:35:51 -08:00
|
|
|
view: EmbeddedViewRef;
|
2014-12-05 17:44:00 -08:00
|
|
|
record: any;
|
|
|
|
constructor(record, view) {
|
|
|
|
this.record = record;
|
|
|
|
this.view = view;
|
|
|
|
}
|
|
|
|
}
|