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
|
|
|
|
2015-06-19 09:41:58 +02:00
|
|
|
/**
|
2015-09-22 16:05:58 +02:00
|
|
|
* The `NgClass` directive conditionally adds and removes CSS classes on an HTML element based on
|
|
|
|
* an expression's evaluation result.
|
2015-06-19 09:41:58 +02:00
|
|
|
*
|
2015-09-22 16:05:58 +02:00
|
|
|
* The result of an expression evaluation is interpreted differently depending on type of
|
|
|
|
* the expression evaluation result:
|
|
|
|
* - `string` - all the CSS classes listed in a string (space delimited) are added
|
|
|
|
* - `Array` - all the CSS classes (Array elements) are added
|
|
|
|
* - `Object` - each key corresponds to a CSS class name while values are interpreted as expressions
|
|
|
|
* evaluating to `Boolean`. If a given expression evaluates to `true` a corresponding CSS class
|
|
|
|
* is added - otherwise it is removed.
|
2015-06-19 09:41:58 +02:00
|
|
|
*
|
2015-09-22 16:05:58 +02:00
|
|
|
* While the `NgClass` directive can interpret expressions evaluating to `string`, `Array`
|
|
|
|
* or `Object`, the `Object`-based version is the most often used and has an advantage of keeping
|
|
|
|
* all the CSS class names in a template.
|
|
|
|
*
|
|
|
|
* ### Example ([live demo](http://plnkr.co/edit/a4YdtmWywhJ33uqfpPPn?p=preview)):
|
2015-06-19 09:41:58 +02:00
|
|
|
*
|
|
|
|
* ```
|
2016-04-28 17:50:03 -07:00
|
|
|
* import {Component} from '@angular/core';
|
|
|
|
* import {NgClass} from '@angular/common';
|
2015-09-22 16:05:58 +02:00
|
|
|
*
|
|
|
|
* @Component({
|
2015-10-11 07:41:19 -07:00
|
|
|
* selector: 'toggle-button',
|
|
|
|
* inputs: ['isDisabled'],
|
2015-09-22 16:05:58 +02:00
|
|
|
* template: `
|
2015-11-23 16:02:19 -08:00
|
|
|
* <div class="button" [ngClass]="{active: isOn, disabled: isDisabled}"
|
2015-09-22 16:05:58 +02:00
|
|
|
* (click)="toggle(!isOn)">
|
|
|
|
* Click me!
|
|
|
|
* </div>`,
|
|
|
|
* styles: [`
|
|
|
|
* .button {
|
|
|
|
* width: 120px;
|
|
|
|
* border: medium solid black;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* .active {
|
|
|
|
* background-color: red;
|
|
|
|
* }
|
|
|
|
*
|
|
|
|
* .disabled {
|
|
|
|
* color: gray;
|
|
|
|
* border: medium solid gray;
|
|
|
|
* }
|
2016-05-26 18:50:59 +02:00
|
|
|
* `],
|
2015-09-22 16:05:58 +02:00
|
|
|
* directives: [NgClass]
|
|
|
|
* })
|
|
|
|
* class ToggleButton {
|
|
|
|
* isOn = false;
|
|
|
|
* isDisabled = false;
|
|
|
|
*
|
|
|
|
* toggle(newState) {
|
|
|
|
* if (!this.isDisabled) {
|
|
|
|
* this.isOn = newState;
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* }
|
2015-06-19 09:41:58 +02:00
|
|
|
* ```
|
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
|
|
|
}
|