I am new to angular2
<parent>
<router-outlet><router-outlet>
</parent>
I have a button in parent component, if I click that button it should call a method in the child component (which is loaded in router outlet.)
Is there any way to call the child component (in router outlet) method from parent ?
There are basically two ways to display a component's template: As a nested component or as a routing target.
Nested Component
If you use a nested component, then the components are considered to have a "parent component/child component" relationship.
The Html for the parent component would then look like this:
<div>
<child></child>
</div>
Where "child" is the selector of the child component. You can then communicate between the two using @Input and @Output properties.
For more information on this technique when you have nested components, check out the docs here: https://angular.io/guide/component-interaction
Routing Target
If you use a component as a routing target by displaying it in a <router-outlet>
, then there is no "parent component" and "child component" relationship.
The best way to communicate between components in this case is to use a service.
I have a blog post about creating a service here: https://blogs.msmvps.com/deborahk/build-a-simple-angular-service-to-share-data/
Resources
If you are new to Angular, you may save your self a bunch of time and frustration by working through a tutorial or online course. That will introduce all of the basic concepts to get you on your way with Angular quickly.
You can work through the "Tour of Heroes" tutorial here: https://angular.io/tutorial
Or watch a course such as this one: https://app.pluralsight.com/library/courses/angular-2-getting-started-update
Or this one: https://app.pluralsight.com/library/courses/angular-2-first-look/table-of-contents
(You can sign up for a free week.)
With child in router-outlet
, you can use ContentChild
to be able to call a method in the child. So...
import { ContentChild } from '@angular/core';
in your parent:
@ContentChild(ChildComponent)
private childComponent: ChildComponent;
and on your click event do:
this.childComponent.doSomething()
Also you need to add your child component in the providers array in parent:
@Component({
selector: 'parent',
...
providers: [ChildComponent]
})
I found two ways to achieve this:
1. Injecting the main component into children
You can add an event to your main component, inject the main component to your child components and subscribe to the event. See the plunk that illustrates this. But, now your children have a dependency on your main component. This may be not good.
main component
executeAction: EventEmitter<any> = new EventEmitter();
constructor() { }
performAction() {
this.executeAction.emit();
}
child
constructor(private appComponent: AppComponent) {
this.executeAction = this.executeAction.bind(this);
eventSubscriber(appComponent.executeAction, this.executeAction);
}
ngOnDestroy(): void {
eventSubscriber(this.appComponent.executeAction, this.executeAction, true);
}
executeAction() {
alert('1');
}
2. Implementing a service
The best solution here and as described in Parent and children communicate via a service is to create a service that will be an additional layer between the main component and children. In this way, you will be independent from the main component implementation. See the plunk that illustrates this approach.
service
subscription = new Subject();
executeAction() {
this.subscription.next();
}
main component
constructor(private appService: AppService) { }
performAction() {
this.appService.executeAction();
}
child
constructor(private appService: AppService) {
this.executeAction = this.executeAction.bind(this);
eventSubscriber(appService.subscription, this.executeAction);
}
ngOnDestroy(): void {
eventSubscriber(this.appService.subscription, this.executeAction, true);
}
executeAction() {
alert('1');
}
If you love us? You can donate to us via Paypal or buy me a coffee so we can maintain and grow! Thank you!
Donate Us With