2016-06-23 09:47:54 -07:00
/ * *
* @license
* Copyright Google Inc . All Rights Reserved .
*
* Use of this source code is governed by an MIT - style license that can be
* found in the LICENSE file at https : //angular.io/license
* /
2017-01-25 13:45:06 -08:00
import { ChangeDetectorRef , Directive , DoCheck , EmbeddedViewRef , Input , IterableChangeRecord , IterableChanges , IterableDiffer , IterableDiffers , NgIterable , OnChanges , SimpleChanges , TemplateRef , TrackByFunction , ViewContainerRef , forwardRef , isDevMode } from '@angular/core' ;
2016-06-08 16:38:52 -07:00
2016-09-08 19:25:25 -07:00
import { getTypeNameForDebugging } from '../facade/lang' ;
2014-12-05 17:44:00 -08:00
2017-01-25 13:45:06 -08:00
export class NgForOfRow < T > {
constructor ( public $implicit : T , public index : number , public count : number ) { }
2016-04-28 14:00:31 -07:00
get first ( ) : boolean { return this . index === 0 ; }
get last ( ) : boolean { return this . index === this . count - 1 ; }
get even ( ) : boolean { return this . index % 2 === 0 ; }
get odd ( ) : boolean { return ! this . even ; }
}
2015-03-31 22:47:11 +00:00
/ * *
2017-01-25 13:45:06 -08:00
* The ` NgForOf ` 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
*
2016-04-09 16:16:14 -04:00
* # # # Local Variables
2015-10-12 21:38:32 +00:00
*
2017-01-25 13:45:06 -08:00
* ` NgForOf ` provides several exported values that can be aliased to local variables :
2015-10-12 21:38:32 +00:00
*
* * ` index ` will be set to the current loop iteration for each template context .
2016-01-16 16:37:44 +01:00
* * ` first ` will be set to a boolean value indicating whether the item is the first one in the
* iteration .
2015-10-12 21:38:32 +00:00
* * ` 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 .
*
2016-04-09 16:16:14 -04:00
* # # # Change Propagation
2015-04-06 11:47:38 +02:00
*
2017-01-25 13:45:06 -08:00
* When the contents of the iterator changes , ` NgForOf ` 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
2017-01-25 13:45:06 -08:00
* 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
*
2017-01-25 13:45:06 -08:00
* To customize the default tracking algorithm , ` NgForOf ` supports ` trackBy ` option .
2016-08-17 11:39:22 +09:00
* ` trackBy ` takes a function which has two arguments : ` index ` and ` item ` .
* If ` trackBy ` is given , Angular tracks changes by the return value of the function .
*
2016-04-09 16:16:14 -04:00
* # # # Syntax
2015-04-06 11:47:38 +02:00
*
2016-08-17 11:39:22 +09:00
* - ` <li *ngFor="let item of items; let i = index; trackBy: trackByFn">...</li> `
* - ` <li template="ngFor let item of items; let i = index; trackBy: trackByFn">...</li> `
*
* With ` <template> ` element :
*
* ` ` `
* < template ngFor let - item [ ngForOf ] = "items" let-i = "index" [ ngForTrackBy ] = "trackByFn" >
* < 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 .
2016-05-27 11:24:05 -07:00
*
* @stable
2015-03-31 22:47:11 +00:00
* /
2017-01-25 13:45:06 -08:00
@Directive ( {
selector : '[ngFor][ngForOf]' ,
providers : [ { provide : forwardRef ( ( ) = > NgFor ) , useExisting : forwardRef ( ( ) = > NgForOf ) } ]
} )
export class NgForOf < T > implements DoCheck ,
OnChanges {
@Input ( ) ngForOf : NgIterable < T > ;
2016-12-21 03:18:24 +03:00
@Input ( )
2017-01-25 13:45:06 -08:00
set ngForTrackBy ( fn : TrackByFunction < T > ) {
2017-01-03 15:14:30 -08:00
if ( isDevMode ( ) && fn != null && typeof fn !== 'function' ) {
// TODO(vicb): use a log service once there is a public one available
if ( < any > console && < any > console . warn ) {
console . warn (
` trackBy must be a function, but received ${ JSON . stringify ( fn ) } . ` +
` See https://angular.io/docs/ts/latest/api/common/index/NgFor-directive.html#!#change-propagation for more information. ` ) ;
}
2016-12-21 03:18:24 +03:00
}
this . _trackByFn = fn ;
}
2017-01-25 13:45:06 -08:00
get ngForTrackBy ( ) : TrackByFunction < T > { return this . _trackByFn ; }
2016-07-16 10:35:31 -07:00
2017-01-25 13:45:06 -08:00
private _differ : IterableDiffer < T > = null ;
private _trackByFn : TrackByFunction < T > ;
2015-06-18 15:40:12 -07:00
2016-06-08 16:38:52 -07:00
constructor (
2017-01-25 13:45:06 -08:00
private _viewContainer : ViewContainerRef , private _template : TemplateRef < NgForOfRow < T > > ,
2016-09-08 19:25:25 -07:00
private _differs : IterableDiffers , private _cdr : ChangeDetectorRef ) { }
2015-06-18 15:40:12 -07:00
2016-07-16 10:35:31 -07:00
@Input ( )
2017-01-25 13:45:06 -08:00
set ngForTemplate ( value : TemplateRef < NgForOfRow < T > > ) {
// TODO(TS2.1): make TemplateRef<Partial<NgForRowOf<T>>> once we move to TS v2.1
// The current type is too restrictive; a template that just uses index, for example,
// should be acceptable.
2016-09-08 19:25:25 -07:00
if ( value ) {
this . _template = value ;
2015-11-06 16:34:41 +01:00
}
}
2015-10-09 12:04:10 -07:00
2016-07-16 10:35:31 -07:00
ngOnChanges ( changes : SimpleChanges ) : void {
if ( 'ngForOf' in changes ) {
// React on ngForOf changes only once all inputs have been initialized
const value = changes [ 'ngForOf' ] . currentValue ;
2016-09-08 19:25:25 -07:00
if ( ! this . _differ && value ) {
2016-07-16 10:35:31 -07:00
try {
2016-09-08 19:25:25 -07:00
this . _differ = this . _differs . find ( value ) . create ( this . _cdr , this . ngForTrackBy ) ;
2016-07-16 10:35:31 -07:00
} catch ( e ) {
2016-08-25 00:50:16 -07:00
throw new Error (
2016-07-16 10:35:31 -07:00
` Cannot find a differ supporting object ' ${ value } ' of type ' ${ getTypeNameForDebugging ( value ) } '. NgFor only supports binding to Iterables such as Arrays. ` ) ;
}
}
}
}
2016-02-01 18:31:26 -08:00
2016-12-21 03:18:24 +03:00
ngDoCheck ( ) : void {
2016-09-08 19:25:25 -07:00
if ( this . _differ ) {
2016-07-16 10:35:31 -07:00
const changes = this . _differ . diff ( this . ngForOf ) ;
2016-09-08 19:25:25 -07:00
if ( changes ) this . _applyChanges ( changes ) ;
2015-07-31 12:23:50 -07:00
}
2014-12-05 17:44:00 -08:00
}
2015-02-12 14:56:41 -08:00
2017-01-25 13:45:06 -08:00
private _applyChanges ( changes : IterableChanges < T > ) {
const insertTuples : RecordViewTuple < T > [ ] = [ ] ;
2016-08-01 11:09:52 -07:00
changes . forEachOperation (
2016-10-27 20:28:11 +02:00
( item : IterableChangeRecord < any > , adjustedPreviousIndex : number , currentIndex : number ) = > {
2016-08-01 11:09:52 -07:00
if ( item . previousIndex == null ) {
2016-09-08 19:25:25 -07:00
const view = this . _viewContainer . createEmbeddedView (
2017-01-25 13:45:06 -08:00
this . _template , new NgForOfRow ( null , null , null ) , currentIndex ) ;
2016-09-08 19:25:25 -07:00
const tuple = new RecordViewTuple ( item , view ) ;
2016-08-01 11:09:52 -07:00
insertTuples . push ( tuple ) ;
} else if ( currentIndex == null ) {
this . _viewContainer . remove ( adjustedPreviousIndex ) ;
} else {
2016-09-08 19:25:25 -07:00
const view = this . _viewContainer . get ( adjustedPreviousIndex ) ;
2016-08-01 11:09:52 -07:00
this . _viewContainer . move ( view , currentIndex ) ;
2017-01-25 13:45:06 -08:00
const tuple = new RecordViewTuple ( item , < EmbeddedViewRef < NgForOfRow < T > > > view ) ;
2016-08-01 11:09:52 -07:00
insertTuples . push ( tuple ) ;
}
} ) ;
2014-12-05 17:44:00 -08:00
2016-07-16 10:35:31 -07:00
for ( let 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
2016-07-16 10:35:31 -07:00
for ( let i = 0 , ilen = this . _viewContainer . length ; i < ilen ; i ++ ) {
2017-01-25 13:45:06 -08:00
const viewRef = < EmbeddedViewRef < NgForOfRow < T > > > this . _viewContainer . get ( i ) ;
2016-04-28 14:00:31 -07:00
viewRef . context . index = i ;
viewRef . context . count = ilen ;
2015-09-04 16:59:04 +02:00
}
2016-02-05 16:32:24 -08:00
2016-07-16 10:35:31 -07:00
changes . forEachIdentityChange ( ( record : any ) = > {
2017-01-25 13:45:06 -08:00
const viewRef = < EmbeddedViewRef < NgForOfRow < T > > > this . _viewContainer . get ( record . currentIndex ) ;
2016-04-28 14:00:31 -07:00
viewRef . context . $implicit = record . item ;
2016-02-05 16:32:24 -08:00
} ) ;
2014-12-05 17:44:00 -08:00
}
2017-01-25 13:45:06 -08:00
private _perViewChange ( view : EmbeddedViewRef < NgForOfRow < T > > , record : IterableChangeRecord < any > ) {
2016-04-28 14:00:31 -07:00
view . context . $implicit = record . item ;
2014-12-05 17:44:00 -08:00
}
}
2017-01-25 13:45:06 -08:00
class RecordViewTuple < T > {
constructor ( public record : any , public view : EmbeddedViewRef < NgForOfRow < T > > ) { }
2014-12-05 17:44:00 -08:00
}
2017-01-25 13:45:06 -08: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 .
*
* # # # 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 .
*
* To customize the default tracking algorithm , ` NgFor ` supports ` trackBy ` option .
* ` trackBy ` takes a function which has two arguments : ` index ` and ` item ` .
* If ` trackBy ` is given , Angular tracks changes by the return value of the function .
*
* # # # Syntax
*
* - ` <li *ngFor="let item of items; let i = index; trackBy: trackByFn">...</li> `
* - ` <li template="ngFor let item of items; let i = index; trackBy: trackByFn">...</li> `
*
* With ` <template> ` element :
*
* ` ` `
* < template ngFor let - item [ ngForOf ] = "items" let-i = "index" [ ngForTrackBy ] = "trackByFn" >
* < li > . . . < / li >
* < / template >
* ` ` `
*
* # # # Example
*
* See a [ live demo ] ( http : //plnkr.co/edit/KVuXxDp0qinGDyo307QW?p=preview) for a more detailed
* example .
*
* @deprecated v4 . 0.0 - Use ` NgForOf<T> ` instead .
* /
export class NgFor extends NgForOf < any > { }