Directives allow you to attach behavior to elements in the DOM.
Directives are classes which get instantiated as a response to a particular DOM structure. By controlling the DOM structure, what directives are imported, and their selectors, the developer can use the composition pattern to get a desirable application behavior.
Directives are the cornerstone of an Angular application. We use Directives to break complex problems into smaller more reusable components. Directives allow the developer to turn HTML into a DSL and then control the application assembly process.
A directive consists of a single directive annotation and a controller class. When the directive's selector
matches elements in the DOM, the following steps occur:
- For each directive, the ElementInjector attempts to resolve the directive's constructor arguments.
- Angular instantiates directives for each matched element using ElementInjector in a depth-first order, as declared in the HTML.
Here is a trivial example of a tooltip decorator. The directive will log a tooltip into the console on every time mouse enters a region:
@Directive({
selector: '[tooltip]', | CSS Selector which triggers the decorator
inputs: [ | List which properties need to be bound
'text: tooltip' | - DOM element tooltip property should be
], | mapped to the directive text property.
host: { | List which events need to be mapped.
(mouseover): 'show()' | - Invoke the show() method every time
} | the mouseover event is fired.
}) |
class Form { | Directive controller class, instantiated
| when CSS matches.
text:string; | text property on the Directive Controller.
|
show(event) { | Show method which implements the show action.
console.log(this.text); |
}
}
Example of usage:
<span tooltip="Tooltip text goes here.">Some text here.</span>
The developer of an application can now freely use the tooltip
attribute wherever the behavior is needed. The code above has taught the browser a new reusable and declarative behavior.
Notice that data binding will work with this decorator with no further effort as shown below.
<span tooltip="Greetings {{user}}!">Some text here.</span>
NOTE: Angular applications do not have a main method. Instead they have a root Component. Dependency Injection then assembles the directives into a working Angular application.
A component is a directive which uses shadow DOM to create encapsulate visual behavior. Components are typically used to create UI widgets or to break up the application into smaller components.
- Only one component can be present per DOM element.
- Component's CSS selectors usually trigger on element names. (Best practice)
- Component has its own shadow view which is attached to the element as a Shadow DOM.
- Shadow view context is the component instance. (i.e. template expressions are evaluated against the component instance.)
Each Angular component requires a single @Component
and at least one @View
annotation. The @Component
annotation specifies when a component is instantiated, and which properties and hostListeners it binds to.
When a component is instantiated, Angular
- Creates a shadow DOM for the component.
- Loads the selected template into the shadow DOM.
- Creates a child Injector which is configured with the appInjector for the Component.
Example of a component:
@Component({ | Component annotation
selector: 'pane', | CSS selector on <pane> element
inputs: [ | List which property need to be bound
'title', | - title mapped to component title
'open' | - open attribute mapped to component open property
], |
}) |
@View({ | View annotation
templateUrl: 'pane.html' | - URL of template HTML
}) |
class Pane { | Component controller class
title:string; | - title property
open:boolean;
constructor() {
this.title = '';
this.open = true;
}
// Public API
toggle() => this.open = !this.open;
open() => this.open = true;
close() => this.open = false;
}
pane.html
:
<div class="outer">
<h1>{{title}}</h1>
<div class="inner" [hidden]="!open">
<content></content>
</div>
</div>
pane.css
:
.outer, .inner { border: 1px solid blue;}
.h1 {background-color: blue;}
Example of usage:
<pane #pane title="Example Title" open="true">
Some text to wrap.
</pane>
<button (click)="pane.toggle()">toggle</button>