| 
									
										
										
										
											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
 | 
					
						
							|  |  |  |  */ | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-26 15:20:27 -07:00
										 |  |  | import {CollectionChangeRecord, Directive, DoCheck, ElementRef, Input, IterableDiffer, IterableDiffers, KeyValueChangeRecord, KeyValueDiffer, KeyValueDiffers, Renderer} from '@angular/core'; | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  | import {isListLikeIterable} from '../facade/collection'; | 
					
						
							|  |  |  | import {isPresent} from '../facade/lang'; | 
					
						
							| 
									
										
										
										
											2016-07-07 16:35:13 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-06-19 09:41:58 +02:00
										 |  |  | /** | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  * @ngModule CommonModule | 
					
						
							| 
									
										
										
										
											2015-09-22 16:05:58 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  * @whatItDoes Adds and removes CSS classes on an HTML element. | 
					
						
							| 
									
										
										
										
											2015-06-19 09:41:58 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  * @howToUse | 
					
						
							| 
									
										
										
										
											2015-06-19 09:41:58 +02:00
										 |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  *     <some-element [ngClass]="'first second'">...</some-element> | 
					
						
							| 
									
										
										
										
											2015-09-22 16:05:58 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  *     <some-element [ngClass]="['first', 'second']">...</some-element> | 
					
						
							| 
									
										
										
										
											2015-09-22 16:05:58 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  *     <some-element [ngClass]="{'first': true, 'second': true, 'third': false}">...</some-element> | 
					
						
							| 
									
										
										
										
											2015-09-22 16:05:58 +02:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  *     <some-element [ngClass]="stringExp|arrayExp|objExp">...</some-element> | 
					
						
							| 
									
										
										
										
											2015-06-19 09:41:58 +02:00
										 |  |  |  * ```
 | 
					
						
							| 
									
										
										
										
											2016-05-27 11:24:05 -07:00
										 |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-09-12 10:20:33 -07:00
										 |  |  |  * @description | 
					
						
							|  |  |  |  * | 
					
						
							|  |  |  |  * The CSS classes are updated as follow depending on the type of the expression evaluation: | 
					
						
							|  |  |  |  * - `string` - the CSS classes listed in a string (space delimited) are added, | 
					
						
							|  |  |  |  * - `Array` - the CSS classes (Array elements) are added, | 
					
						
							|  |  |  |  * - `Object` - keys are CSS class names that get added when the expression given in the value | 
					
						
							|  |  |  |  *              evaluates to a truthy value, otherwise class are removed. | 
					
						
							|  |  |  |  * | 
					
						
							| 
									
										
										
										
											2016-05-27 11:24:05 -07:00
										 |  |  |  * @stable | 
					
						
							| 
									
										
										
										
											2015-06-19 09:41:58 +02:00
										 |  |  |  */ | 
					
						
							| 
									
										
										
										
											2016-07-07 16:35:13 -07:00
										 |  |  | @Directive({selector: '[ngClass]'}) | 
					
						
							| 
									
										
										
										
											2016-07-26 15:20:27 -07:00
										 |  |  | export class NgClass implements DoCheck { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |   private _iterableDiffer: IterableDiffer; | 
					
						
							|  |  |  |   private _keyValueDiffer: KeyValueDiffer; | 
					
						
							|  |  |  |   private _initialClasses: string[] = []; | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |   private _rawClass: string[]|Set<string>|{[klass: string]: any}; | 
					
						
							| 
									
										
										
										
											2015-06-18 15:40:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |   constructor( | 
					
						
							|  |  |  |       private _iterableDiffers: IterableDiffers, private _keyValueDiffers: KeyValueDiffers, | 
					
						
							|  |  |  |       private _ngEl: ElementRef, private _renderer: Renderer) {} | 
					
						
							| 
									
										
										
										
											2015-03-26 17:51:08 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 16:35:13 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |   @Input('class') | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |   set klass(v: string) { | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |     this._applyInitialClasses(true); | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |     this._initialClasses = typeof v === 'string' ? v.split(/\s+/) : []; | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |     this._applyInitialClasses(false); | 
					
						
							|  |  |  |     this._applyClasses(this._rawClass, false); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-07-07 16:35:13 -07:00
										 |  |  |   @Input() | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |   set ngClass(v: string|string[]|Set<string>|{[klass: string]: any}) { | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  |     this._cleanupClasses(this._rawClass); | 
					
						
							| 
									
										
										
										
											2015-06-18 15:40:12 -07:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     this._iterableDiffer = null; | 
					
						
							|  |  |  |     this._keyValueDiffer = null; | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  | 
 | 
					
						
							|  |  |  |     this._rawClass = typeof v === 'string' ? v.split(/\s+/) : v; | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  |     if (this._rawClass) { | 
					
						
							|  |  |  |       if (isListLikeIterable(this._rawClass)) { | 
					
						
							|  |  |  |         this._iterableDiffer = this._iterableDiffers.find(this._rawClass).create(null); | 
					
						
							| 
									
										
										
										
											2015-07-31 12:23:50 -07:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         this._keyValueDiffer = this._keyValueDiffers.find(this._rawClass).create(null); | 
					
						
							| 
									
										
										
										
											2015-07-31 12:23:50 -07:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-03-26 17:51:08 +01:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
											  
											
												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(): void { | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |     if (this._iterableDiffer) { | 
					
						
							|  |  |  |       const changes = this._iterableDiffer.diff(this._rawClass); | 
					
						
							|  |  |  |       if (changes) { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |         this._applyIterableChanges(changes); | 
					
						
							|  |  |  |       } | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |     } else if (this._keyValueDiffer) { | 
					
						
							|  |  |  |       const changes = this._keyValueDiffer.diff(this._rawClass); | 
					
						
							|  |  |  |       if (changes) { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |         this._applyKeyValueChanges(changes); | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-06-18 15:40:12 -07:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-06-18 15:01:19 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |   private _cleanupClasses(rawClassVal: string[]|Set<string>|{[klass: string]: any}): void { | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |     this._applyClasses(rawClassVal, true); | 
					
						
							|  |  |  |     this._applyInitialClasses(false); | 
					
						
							| 
									
										
										
										
											2015-03-26 17:51:08 +01:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-31 12:23:50 -07:00
										 |  |  |   private _applyKeyValueChanges(changes: any): void { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     changes.forEachAddedItem( | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         (record: KeyValueChangeRecord) => this._toggleClass(record.key, record.currentValue)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     changes.forEachChangedItem( | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         (record: KeyValueChangeRecord) => this._toggleClass(record.key, record.currentValue)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     changes.forEachRemovedItem((record: KeyValueChangeRecord) => { | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  |       if (record.previousValue) { | 
					
						
							|  |  |  |         this._toggleClass(record.key, false); | 
					
						
							|  |  |  |       } | 
					
						
							|  |  |  |     }); | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-07-31 12:23:50 -07:00
										 |  |  |   private _applyIterableChanges(changes: any): void { | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     changes.forEachAddedItem( | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         (record: CollectionChangeRecord) => this._toggleClass(record.item, true)); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-02-20 08:52:51 -08:00
										 |  |  |     changes.forEachRemovedItem( | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         (record: CollectionChangeRecord) => this._toggleClass(record.item, false)); | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |   private _applyInitialClasses(isCleanup: boolean) { | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |     this._initialClasses.forEach(klass => this._toggleClass(klass, !isCleanup)); | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-06-08 16:38:52 -07:00
										 |  |  |   private _applyClasses( | 
					
						
							|  |  |  |       rawClassVal: string[]|Set<string>|{[key: string]: any}, isCleanup: boolean) { | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |     if (rawClassVal) { | 
					
						
							|  |  |  |       if (Array.isArray(rawClassVal) || rawClassVal instanceof Set) { | 
					
						
							|  |  |  |         (<any>rawClassVal).forEach((klass: string) => this._toggleClass(klass, !isCleanup)); | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |       } else { | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |         Object.keys(rawClassVal).forEach(klass => { | 
					
						
							|  |  |  |           if (isPresent(rawClassVal[klass])) this._toggleClass(klass, !isCleanup); | 
					
						
							|  |  |  |         }); | 
					
						
							| 
									
										
										
										
											2015-08-10 12:25:46 +02:00
										 |  |  |       } | 
					
						
							|  |  |  |     } | 
					
						
							|  |  |  |   } | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2016-09-09 12:03:51 -07:00
										 |  |  |   private _toggleClass(klass: string, enabled: boolean): void { | 
					
						
							|  |  |  |     klass = klass.trim(); | 
					
						
							|  |  |  |     if (klass) { | 
					
						
							|  |  |  |       klass.split(/\s+/g).forEach( | 
					
						
							|  |  |  |           klass => { this._renderer.setElementClass(this._ngEl.nativeElement, klass, enabled); }); | 
					
						
							| 
									
										
										
										
											2015-09-14 14:11:28 +02:00
										 |  |  |     } | 
					
						
							| 
									
										
										
										
											2015-06-21 11:54:21 +02:00
										 |  |  |   } | 
					
						
							| 
									
										
										
										
											2015-03-26 17:51:08 +01:00
										 |  |  | } |