2015-02-06 13:19:47 -08:00
|
|
|
import {ABSTRACT, CONST, normalizeBlank, isPresent} from 'angular2/src/facade/lang';
|
|
|
|
import {ListWrapper, List} from 'angular2/src/facade/collection';
|
2014-11-21 21:19:23 -08:00
|
|
|
|
2015-03-14 00:00:42 +00:00
|
|
|
// type StringMap = {[idx: string]: string};
|
|
|
|
|
2015-03-13 21:54:19 +00:00
|
|
|
/**
|
2015-03-17 19:22:13 +00:00
|
|
|
* Directives allow you to attach behavior to the DOM elements.
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* Directive is an abstract concept, instead use concrete directives such as: [Component], [Decorator] or [Viewport].
|
2015-03-17 19:22:13 +00:00
|
|
|
* @publicModule angular2/angular2
|
2015-03-13 21:54:19 +00:00
|
|
|
*/
|
2014-11-21 21:19:23 -08:00
|
|
|
@ABSTRACT()
|
|
|
|
export class Directive {
|
2015-03-14 00:00:42 +00:00
|
|
|
/**
|
2015-03-17 19:22:13 +00:00
|
|
|
* The CSS selector that triggers the instantiation of a directive.
|
|
|
|
*
|
|
|
|
* Angular only allows directives to trigger on CSS selectors that do not cross element boundaries.
|
2015-03-14 00:00:42 +00:00
|
|
|
* The supported selectors are:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* - `element-name` select by element name.
|
|
|
|
* - `.class` select by class name.
|
|
|
|
* - `[attribute]` select by attribute name.
|
|
|
|
* - `[attribute=value]` select by attribute name and value.
|
|
|
|
* - `:not(sub_selector)` select only if the element does not match the `sub_selector`.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* ## Example
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* Suppose we have a directive with an `input[type=text]` selector.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* And the following HTML:
|
|
|
|
*
|
2015-03-17 19:22:13 +00:00
|
|
|
* ```html
|
|
|
|
* <form>
|
|
|
|
* <input type="text">
|
|
|
|
* <input type="radio">
|
|
|
|
* <form>
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* The directive would only be instantiated on the `<input type="text">` element.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
*/
|
|
|
|
selector:string;
|
2015-03-17 19:22:13 +00:00
|
|
|
|
2015-03-14 00:00:42 +00:00
|
|
|
/**
|
|
|
|
* Enumerates the set of properties that accept data binding for a directive.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* The `bind` property defines a set of `directiveProperty` to `bindingProperty` key-value pairs:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* - `directiveProperty` specifies the component property where the value is written.
|
2015-03-17 19:22:13 +00:00
|
|
|
* - `bindingProperty` specifies the DOM property where the value is read from.
|
|
|
|
*
|
|
|
|
* You can include [Pipes] when specifying a `bindingProperty` to allow for data transformation and structural
|
2015-03-14 00:00:42 +00:00
|
|
|
* change detection of the value.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* ## Syntax
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Directive({
|
|
|
|
* bind: {
|
|
|
|
* 'directiveProperty1': 'bindingProperty1',
|
|
|
|
* 'directiveProperty2': 'bindingProperty2 | pipe1 | ...',
|
|
|
|
* ...
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
*
|
|
|
|
* ## Basic Property Binding:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Decorator({
|
|
|
|
* selector: '[tooltip]',
|
|
|
|
* bind: {
|
|
|
|
* 'tooltipText': 'tooltip'
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* class Tooltip {
|
|
|
|
* set tooltipText(text) {
|
|
|
|
* // This will get called every time the 'tooltip' binding changes with the new value.
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* As used in this example:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```html
|
|
|
|
* <div [tooltip]="someExpression">
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* Whenever the `someExpression` expression changes, the `bind` declaration instructs Angular to update the
|
2015-03-14 00:00:42 +00:00
|
|
|
* `Tooltip`'s `tooltipText` property.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* Similarly in this example:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```html
|
|
|
|
* <div tooltip="Some Text">
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* The `Tooltip`'s `tooltipText` property gets initialized to the `Some Text` literal.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
*
|
|
|
|
* ## Bindings With Pipes:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Decorator({
|
|
|
|
* selector: '[class-set]',
|
|
|
|
* bind: {
|
|
|
|
* 'classChanges': 'classSet | keyValDiff'
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* class ClassSet {
|
|
|
|
* set classChanges(changes:KeyValueChanges) {
|
|
|
|
* // This will get called every time the `class-set` expressions changes its structure.
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* As used in this example:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```html
|
|
|
|
* <div [class-set]="someExpression">
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* In the above example, the `ClassSet` uses the `keyValDiff` [Pipe] for watching structural changes. This means that
|
2015-03-17 19:22:13 +00:00
|
|
|
* the `classChanges` setter gets invoked if the expression changes to a different reference, or if the
|
2015-03-14 00:00:42 +00:00
|
|
|
* structure of the expression changes. (Shallow property watching of the object)
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* NOTE: The `someExpression` can also contain its own [Pipe]s. In this case, the two pipes compose as if they were
|
|
|
|
* inlined.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
*/
|
|
|
|
bind:any; // StringMap
|
2015-03-17 19:22:13 +00:00
|
|
|
|
2015-03-14 00:00:42 +00:00
|
|
|
/**
|
|
|
|
* Specifies which DOM events the directive listens to and what the action should be.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-16 22:46:05 +01:00
|
|
|
* The `events` property defines a set of `event` to `method` key-value pairs:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-16 22:46:05 +01:00
|
|
|
* - `event1` specifies the DOM event that the directive listens to.
|
|
|
|
* - `onMethod1` specifies the method to execute when the event occurs.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* ## Syntax
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Directive({
|
|
|
|
* events: {
|
2015-03-16 22:46:05 +01:00
|
|
|
* 'event1': 'onMethod1',
|
2015-03-17 19:22:13 +00:00
|
|
|
* ...
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
|
|
|
* ## Basic Event Binding:
|
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Decorator({
|
|
|
|
* selector: 'input',
|
2015-03-16 22:46:05 +01:00
|
|
|
* events: {
|
2015-03-17 19:22:13 +00:00
|
|
|
* 'change': 'onChange'
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* class InputDecorator {
|
|
|
|
* onChange(event:Event) {
|
|
|
|
* // invoked whenever the DOM element fires the 'change' event.
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
*/
|
|
|
|
events:any; // StringMap
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Specifies a set of lifecycle events in which the directive participates.
|
|
|
|
*/
|
|
|
|
lifecycle:any; //List<LifecycleEvent>
|
2015-03-17 19:22:13 +00:00
|
|
|
|
2014-11-21 21:19:23 -08:00
|
|
|
@CONST()
|
|
|
|
constructor({
|
|
|
|
selector,
|
|
|
|
bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle
|
2014-11-21 21:19:23 -08:00
|
|
|
}:{
|
|
|
|
selector:string,
|
|
|
|
bind:any,
|
2015-03-06 15:44:59 +01:00
|
|
|
events: any,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle:List
|
2014-11-21 21:19:23 -08:00
|
|
|
}={})
|
|
|
|
{
|
|
|
|
this.selector = selector;
|
|
|
|
this.bind = bind;
|
2015-03-06 15:44:59 +01:00
|
|
|
this.events = events;
|
2015-01-13 16:17:43 -08:00
|
|
|
this.lifecycle = lifecycle;
|
2014-11-21 21:19:23 -08:00
|
|
|
}
|
2015-02-06 13:19:47 -08:00
|
|
|
|
|
|
|
hasLifecycleHook(hook:string):boolean {
|
|
|
|
return isPresent(this.lifecycle) ? ListWrapper.contains(this.lifecycle, hook) : false;
|
|
|
|
}
|
2014-11-21 21:19:23 -08:00
|
|
|
}
|
|
|
|
|
2015-03-17 19:22:13 +00:00
|
|
|
/**
|
|
|
|
* @publicModule angular2/angular2
|
|
|
|
*/
|
2014-11-21 21:19:23 -08:00
|
|
|
export class Component extends Directive {
|
|
|
|
//TODO: vsavkin: uncomment it once the issue with defining fields in a sublass works
|
2015-03-11 16:48:57 -07:00
|
|
|
services:any; //List;
|
2014-11-21 21:19:23 -08:00
|
|
|
|
2015-01-13 16:17:43 -08:00
|
|
|
@CONST()
|
2014-11-21 21:19:23 -08:00
|
|
|
constructor({
|
|
|
|
selector,
|
|
|
|
bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events,
|
2015-03-11 16:48:57 -07:00
|
|
|
services,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle
|
2014-11-21 21:19:23 -08:00
|
|
|
}:{
|
2015-01-13 16:17:43 -08:00
|
|
|
selector:String,
|
2014-11-21 21:19:23 -08:00
|
|
|
bind:Object,
|
2015-03-06 15:44:59 +01:00
|
|
|
events:Object,
|
2015-03-11 16:48:57 -07:00
|
|
|
services:List,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle:List
|
|
|
|
}={})
|
2014-11-21 21:19:23 -08:00
|
|
|
{
|
|
|
|
super({
|
|
|
|
selector: selector,
|
|
|
|
bind: bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events: events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle: lifecycle
|
|
|
|
});
|
2014-11-21 21:19:23 -08:00
|
|
|
|
2015-03-11 16:48:57 -07:00
|
|
|
this.services = services;
|
2014-11-21 21:19:23 -08:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-13 11:33:57 -07:00
|
|
|
/**
|
|
|
|
* @publicModule angular2/angular2
|
|
|
|
*/
|
|
|
|
export class DynamicComponent extends Directive {
|
|
|
|
services:any; //List;
|
|
|
|
|
|
|
|
@CONST()
|
|
|
|
constructor({
|
|
|
|
selector,
|
|
|
|
bind,
|
|
|
|
events,
|
|
|
|
services,
|
|
|
|
lifecycle
|
|
|
|
}:{
|
|
|
|
selector:string,
|
2015-03-13 11:34:20 -07:00
|
|
|
bind:Object,
|
|
|
|
events:Object,
|
2015-03-13 11:33:57 -07:00
|
|
|
services:List,
|
|
|
|
lifecycle:List
|
|
|
|
}={}) {
|
|
|
|
super({
|
|
|
|
selector: selector,
|
|
|
|
bind: bind,
|
|
|
|
events: events,
|
|
|
|
lifecycle: lifecycle
|
|
|
|
});
|
|
|
|
|
|
|
|
this.services = services;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-17 19:22:13 +00:00
|
|
|
/**
|
|
|
|
* @publicModule angular2/angular2
|
|
|
|
*/
|
2014-11-21 21:19:23 -08:00
|
|
|
export class Decorator extends Directive {
|
2015-01-07 17:32:46 +01:00
|
|
|
compileChildren: boolean;
|
2014-11-21 21:19:23 -08:00
|
|
|
@CONST()
|
|
|
|
constructor({
|
|
|
|
selector,
|
|
|
|
bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle,
|
|
|
|
compileChildren = true,
|
2014-11-21 21:19:23 -08:00
|
|
|
}:{
|
|
|
|
selector:string,
|
|
|
|
bind:any,
|
2015-03-06 15:44:59 +01:00
|
|
|
events:any,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle:List,
|
2015-01-07 18:20:29 +01:00
|
|
|
compileChildren:boolean
|
2014-11-21 21:19:23 -08:00
|
|
|
}={})
|
|
|
|
{
|
2015-01-07 18:20:29 +01:00
|
|
|
this.compileChildren = compileChildren;
|
2014-11-21 21:19:23 -08:00
|
|
|
super({
|
|
|
|
selector: selector,
|
|
|
|
bind: bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events: events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle: lifecycle
|
2014-11-21 21:19:23 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2015-03-17 19:22:13 +00:00
|
|
|
/**
|
|
|
|
* @publicModule angular2/angular2
|
|
|
|
*/
|
2015-02-12 11:54:22 +01:00
|
|
|
export class Viewport extends Directive {
|
2014-11-21 21:19:23 -08:00
|
|
|
@CONST()
|
|
|
|
constructor({
|
|
|
|
selector,
|
|
|
|
bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle
|
2014-11-21 21:19:23 -08:00
|
|
|
}:{
|
|
|
|
selector:string,
|
|
|
|
bind:any,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle:List
|
2014-11-21 21:19:23 -08:00
|
|
|
}={})
|
|
|
|
{
|
|
|
|
super({
|
|
|
|
selector: selector,
|
|
|
|
bind: bind,
|
2015-03-06 15:44:59 +01:00
|
|
|
events: events,
|
2015-01-13 16:17:43 -08:00
|
|
|
lifecycle: lifecycle
|
2014-11-21 21:19:23 -08:00
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
2015-01-13 16:17:43 -08:00
|
|
|
|
2015-03-14 00:00:42 +00:00
|
|
|
//TODO(misko): turn into LifecycleEvent class once we switch to TypeScript;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Specify that a directive should be notified whenever a [View] that contains it is destroyed.
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* ## Example
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Decorator({
|
|
|
|
* ...,
|
|
|
|
* lifecycle: [ onDestroy ]
|
|
|
|
* })
|
|
|
|
* class ClassSet implements OnDestroy {
|
|
|
|
* onDestroy() {
|
|
|
|
* // invoked to notify directive of the containing view destruction.
|
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
* @publicModule angular2/angular2
|
2015-03-14 00:00:42 +00:00
|
|
|
*/
|
2015-02-06 13:19:47 -08:00
|
|
|
export const onDestroy = "onDestroy";
|
2015-03-14 00:00:42 +00:00
|
|
|
|
|
|
|
|
|
|
|
/**
|
2015-03-17 19:22:13 +00:00
|
|
|
* Specify that a directive should be notified when any of its bindings have changed.
|
|
|
|
*
|
2015-03-14 00:00:42 +00:00
|
|
|
* ## Example:
|
2015-03-17 19:22:13 +00:00
|
|
|
*
|
|
|
|
* ```
|
|
|
|
* @Decorator({
|
|
|
|
* selector: '[class-set]',
|
|
|
|
* bind: {
|
|
|
|
* 'propA': 'propA'
|
|
|
|
* 'propB': 'propB'
|
|
|
|
* }
|
|
|
|
* })
|
|
|
|
* class ClassSet {
|
|
|
|
* propA;
|
|
|
|
* propB;
|
|
|
|
* onChange(changes:{[idx: string, PropertyUpdate]}) {
|
|
|
|
* // This will get called after any of the properties have been updated.
|
|
|
|
* if (changes['propA']) {
|
|
|
|
* // if propA was updated
|
|
|
|
* }
|
|
|
|
* if (changes['propA']) {
|
|
|
|
* // if propB was updated
|
2015-03-14 00:00:42 +00:00
|
|
|
* }
|
2015-03-17 19:22:13 +00:00
|
|
|
* }
|
|
|
|
* }
|
|
|
|
* ```
|
|
|
|
* @publicModule angular2/angular2
|
2015-03-14 00:00:42 +00:00
|
|
|
*/
|
2015-02-06 13:19:47 -08:00
|
|
|
export const onChange = "onChange";
|