Some notes on the stuff I struggeled with learning Angular 2 (beta0)
- If your component uses a subcomponent, declared in a single file like the file below, declare MySubcomponent first.
@Component
@View { directives: [MySubComponent]}
export class MyComponent {}
@Component
@View
export class MySubComponent {} // This should go first
Try changing your injections from constructor(private myService: MyService)
to @Inject(forwardRef(() => MyService)) private myService: MyService
.
See also: (http://blog.thoughtram.io/angular/2015/09/03/forward-references-in-angular-2.html)
The ng-content
directive allows you to insert content inside a (custom) component very easily. This is similar to but much more powerful than Angular 1 translusion.
Using 'my-component'
<my-component>
Some content
</my-comonent>
The 'my-component' view template:
<div> Something something </div>
<ng-content></ng-content> <!-- This is where 'Some content' will appear -->
<div> The end </div>
Use the select
property on ng-content
to determine which part of the provided content should go where.
Using 'my-component'
<my-component>
<div class="first-paragraph">Angular 2 is cool...</div>
<div class="second-paragraph">...but not always documented.</div>
</my-comonent>
The 'my-component' view template:
<div> Something something </div>
<p>
<ng-content select=".first-paragraph"></ng-content> <!-- This is where 'Angular 2 is cool...' will appear -->
</p>
<p>
<ng-content select=".second-paragraph"></ng-content> <!-- ...but not always documented.' will appear -->
</p>
<div> The end </div>
Angular 2 by default has only one-way databinding. The flow from controller to view is by data-flow, view to controller by events. Now what if you want to invoke a method on a subcomponent? How can you get a reference to a subcomponent? One way is to let the subcomponent fire an event when it is constructed, passing itself as a reference.
The subcomponent - a modal dialog
@Component({...})
@View({...})
export class Modal implements OnInit {
/**
* Fires an event when the modal is ready with a pointer to the modal.
*/
@Output('loaded') loadedEmitter: EventEmitter<Modal> = new EventEmitter<Modal>();
onInit() {
this.loadedEmitter.next(this);
}
}
The view template of App.ts
<modal (loaded)="modalLoaded($event)"></modal>
<div>Other content of the App view</div>
App.ts, getting a reference to the modal
@Component({...})
@View({...})
export class App {
modalLoaded(modal: Modal) {
this.modal = modal;
}
}
TODO: maybe you need the content children, see: https://angular.io/docs/ts/latest/api/core/ContentChildren-var.html and also http://stackoverflow.com/questions/34326745/whats-the-difference-between-viewchild-and-contentchild
Suppose you're making a wizard component with child pages and you need to refer the children from the parent component.
<wizard>
<wizard-page [label]="page1"></wizard-page>
<wizard-page [label]="page2"></wizard-page>
</wizard>
The wizard needs to be able to do something like:
this.activePage = wizardPages[0];
wizardPages[0].isValid();
[...]
@Component({
selector: 'wizard'
})
@View({
directives: [WizardPage]
})
export class Wizard implements AfterViewInit {
// This is the key: viewChildrew will contain all the wizard pages.
@ViewChildren(WizardPage) viewChildren: QueryList<WizardPage>;
// Viewchildren (the WizardPages) are available now.
ngAfterViewInit() {
// viewChild is updated after the view has been initialized
for(let child of this.viewChildren.toArray()) {
console.log(child);
}
}
}
@Component({
selector: 'wizard-page'
})
export class WizardPage implements OnInit {
[...]
}
Suppose you have a SearchComponent that has one input element (search term input box). You want to give focus to the search term input box every time the component is displayed (also when navigating away from and back to the search component through the router).
The way to do this is to inject an ElementRef as a View and handle the event in the AfterViewInit callback method.
PS if you are using inheritance (ie abstract base classes) make sure to inject the child in the component and not in the abstract base class.
export class BaseSearchComponent {
}
@Component({...})
@View({...})
export class SearchComponent extends BaseSearchComponent implements AfterViewInit {
@ViewChild('zoekTermInput') zoekTermInput: ElementRef;
ngAfterViewInit() {
this.zoekTermInput.nativeElement.focus();
}
}
The template should look like this :
<input type="text" #zoekTermInput
...
autofocus
autocomplete="off">