2015-10-26 09:50:51 -07:00
|
|
|
import {isPresent, isString, StringWrapper, isBlank, isArray} from 'angular2/src/core/facade/lang';
|
2015-09-02 16:43:39 -07:00
|
|
|
import {DoCheck, OnDestroy} from 'angular2/lifecycle_hooks';
|
2015-09-03 22:01:36 -07:00
|
|
|
import {Directive} from 'angular2/src/core/metadata';
|
2015-10-02 07:37:23 -07:00
|
|
|
import {ElementRef} from 'angular2/src/core/linker';
|
2015-07-31 12:23:50 -07:00
|
|
|
import {
|
|
|
|
IterableDiffer,
|
|
|
|
IterableDiffers,
|
2015-09-03 22:01:36 -07:00
|
|
|
KeyValueDiffer,
|
2015-07-31 12:23:50 -07:00
|
|
|
KeyValueDiffers
|
2015-09-03 22:01:36 -07:00
|
|
|
} from 'angular2/src/core/change_detection';
|
|
|
|
import {Renderer} from 'angular2/src/core/render';
|
2015-10-07 09:09:43 -07:00
|
|
|
import {StringMapWrapper, isListLikeIterable} from 'angular2/src/core/facade/collection';
|
2015-03-26 17:51:08 +01: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
|
|
|
*
|
|
|
|
* ```
|
2015-10-11 07:41:19 -07:00
|
|
|
* import {Component, NgClass} from 'angular2/angular2';
|
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: `
|
|
|
|
* <div class="button" [ng-class]="{active: isOn, disabled: isDisabled}"
|
|
|
|
* (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;
|
|
|
|
* }
|
|
|
|
* `]
|
|
|
|
* directives: [NgClass]
|
|
|
|
* })
|
|
|
|
* class ToggleButton {
|
|
|
|
* isOn = false;
|
|
|
|
* isDisabled = false;
|
|
|
|
*
|
|
|
|
* toggle(newState) {
|
|
|
|
* if (!this.isDisabled) {
|
|
|
|
* this.isOn = newState;
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* }
|
2015-06-19 09:41:58 +02:00
|
|
|
* ```
|
|
|
|
*/
|
2015-09-30 20:59:23 -07:00
|
|
|
@Directive({selector: '[ng-class]', inputs: ['rawClass: ng-class', 'initialClasses: class']})
|
2015-08-31 18:32:32 -07:00
|
|
|
export class NgClass implements DoCheck, OnDestroy {
|
2015-07-31 12:23:50 -07:00
|
|
|
private _differ: any;
|
|
|
|
private _mode: string;
|
2015-08-10 12:25:46 +02:00
|
|
|
private _initialClasses = [];
|
|
|
|
private _rawClass;
|
2015-06-18 15:40:12 -07:00
|
|
|
|
2015-07-31 12:23:50 -07:00
|
|
|
constructor(private _iterableDiffers: IterableDiffers, private _keyValueDiffers: KeyValueDiffers,
|
2015-10-06 19:39:44 -07:00
|
|
|
private _ngEl: ElementRef, private _renderer: Renderer) {}
|
2015-03-26 17:51:08 +01:00
|
|
|
|
2015-08-10 12:25:46 +02:00
|
|
|
set initialClasses(v) {
|
|
|
|
this._applyInitialClasses(true);
|
|
|
|
this._initialClasses = isPresent(v) && isString(v) ? v.split(' ') : [];
|
|
|
|
this._applyInitialClasses(false);
|
|
|
|
this._applyClasses(this._rawClass, false);
|
|
|
|
}
|
|
|
|
|
2015-06-18 15:40:12 -07:00
|
|
|
set rawClass(v) {
|
2015-06-21 11:54:21 +02:00
|
|
|
this._cleanupClasses(this._rawClass);
|
2015-06-18 15:40:12 -07:00
|
|
|
|
2015-06-21 11:54:21 +02:00
|
|
|
if (isString(v)) {
|
|
|
|
v = v.split(' ');
|
|
|
|
}
|
|
|
|
|
|
|
|
this._rawClass = v;
|
2015-07-31 12:23:50 -07:00
|
|
|
if (isPresent(v)) {
|
|
|
|
if (isListLikeIterable(v)) {
|
|
|
|
this._differ = this._iterableDiffers.find(v).create(null);
|
|
|
|
this._mode = 'iterable';
|
|
|
|
} else {
|
|
|
|
this._differ = this._keyValueDiffers.find(v).create(null);
|
|
|
|
this._mode = 'keyValue';
|
|
|
|
}
|
2015-08-10 12:25:46 +02:00
|
|
|
} else {
|
|
|
|
this._differ = null;
|
2015-07-31 12:23:50 -07:00
|
|
|
}
|
2015-03-26 17:51:08 +01:00
|
|
|
}
|
|
|
|
|
2015-08-27 21:19:56 -07:00
|
|
|
doCheck(): void {
|
2015-07-31 12:23:50 -07:00
|
|
|
if (isPresent(this._differ)) {
|
|
|
|
var changes = this._differ.diff(this._rawClass);
|
|
|
|
if (isPresent(changes)) {
|
|
|
|
if (this._mode == 'iterable') {
|
|
|
|
this._applyIterableChanges(changes);
|
|
|
|
} else {
|
|
|
|
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
|
|
|
|
2015-07-23 16:22:14 -07:00
|
|
|
onDestroy(): void { this._cleanupClasses(this._rawClass); }
|
|
|
|
|
2015-06-21 11:54:21 +02:00
|
|
|
private _cleanupClasses(rawClassVal): 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 {
|
|
|
|
changes.forEachAddedItem((record) => { this._toggleClass(record.key, record.currentValue); });
|
|
|
|
changes.forEachChangedItem((record) => { this._toggleClass(record.key, record.currentValue); });
|
|
|
|
changes.forEachRemovedItem((record) => {
|
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 {
|
|
|
|
changes.forEachAddedItem((record) => { this._toggleClass(record.item, true); });
|
|
|
|
changes.forEachRemovedItem((record) => { 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) {
|
2015-10-07 09:09:43 -07:00
|
|
|
this._initialClasses.forEach(className => this._toggleClass(className, !isCleanup));
|
2015-08-10 12:25:46 +02:00
|
|
|
}
|
|
|
|
|
2015-10-26 09:50:51 -07:00
|
|
|
private _applyClasses(rawClassVal: string[] | Set<string>| {[key: string]: string},
|
|
|
|
isCleanup: boolean) {
|
2015-08-10 12:25:46 +02:00
|
|
|
if (isPresent(rawClassVal)) {
|
2015-10-26 09:50:51 -07:00
|
|
|
if (isArray(rawClassVal)) {
|
2015-10-07 09:09:43 -07:00
|
|
|
(<string[]>rawClassVal).forEach(className => this._toggleClass(className, !isCleanup));
|
2015-10-26 09:50:51 -07:00
|
|
|
} else if (rawClassVal instanceof Set) {
|
|
|
|
(<Set<string>>rawClassVal).forEach(className => this._toggleClass(className, !isCleanup));
|
2015-08-10 12:25:46 +02:00
|
|
|
} else {
|
2015-10-08 22:44:58 -07:00
|
|
|
StringMapWrapper.forEach(<{[k: string]: string}>rawClassVal, (expVal, className) => {
|
2015-08-10 12:25:46 +02:00
|
|
|
if (expVal) this._toggleClass(className, !isCleanup);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-06-21 11:54:21 +02:00
|
|
|
private _toggleClass(className: string, enabled): void {
|
2015-09-14 14:11:28 +02:00
|
|
|
className = className.trim();
|
|
|
|
if (className.length > 0) {
|
|
|
|
this._renderer.setElementClass(this._ngEl, className, enabled);
|
|
|
|
}
|
2015-06-21 11:54:21 +02:00
|
|
|
}
|
2015-03-26 17:51:08 +01:00
|
|
|
}
|