I have Angular components and first component uses the second one as a directive. They should share the same model object, which is initialized in the first component. How can I pass that model to the second component?
-
2Can you post some code? Generally, you use template local variables for this kind of thing in ng2, but it's hard to say without a little more detail on your part. – jlew Dec 04 '15 at 12:24
-
I found this answer helpful: http://stackoverflow.com/a/31037168/1341825 – theUtherSide May 25 '16 at 00:42
5 Answers
For one-way data binding from parent to child, use the @Input
decorator (as recommended by the style guide) to specify an input property on the child component
@Input() model: any; // instead of any, specify your type
and use template property binding in the parent template
<child [model]="parentModel"></child>
Since you are passing an object (a JavaScript reference type) any changes you make to object properties in the parent or the child component will be reflected in the other component, since both components have a reference to the same object. I show this in the Plunker.
If you reassign the object in the parent component
this.model = someNewModel;
Angular will propagate the new object reference to the child component (automatically, as part of change detection).
The only thing you shouldn't do is reassign the object in the child component. If you do this, the parent will still reference the original object. (If you do need two-way data binding, see https://stackoverflow.com/a/34616530/215945).
@Component({
selector: 'child',
template: `<h3>child</h3>
<div>{{model.prop1}}</div>
<button (click)="updateModel()">update model</button>`
})
class Child {
@Input() model: any; // instead of any, specify your type
updateModel() {
this.model.prop1 += ' child';
}
}
@Component({
selector: 'my-app',
directives: [Child],
template: `
<h3>Parent</h3>
<div>{{parentModel.prop1}}</div>
<button (click)="updateModel()">update model</button>
<child [model]="parentModel"></child>`
})
export class AppComponent {
parentModel = { prop1: '1st prop', prop2: '2nd prop' };
constructor() {}
updateModel() { this.parentModel.prop1 += ' parent'; }
}
Plunker - Angular RC.2

- 1
- 1

- 362,217
- 114
- 495
- 492
-
2you're doing god's work! any advice for sibling components? in my case I have 2 at root level bootstrapped. HeaderComponent has a search input which i want to share with components in the body.. – Sonic Soul Sep 29 '16 at 21:12
-
@SonicSoul, put the data in a JavaScript reference type in the parent, or, if they don't share a parent, put the data in a shared service. For the service approach, you can again used a JavaScript reference type or [use observables](https://angular.io/docs/ts/latest/cookbook/component-communication.html#!#bidirectional-service). – Mark Rajcok Oct 01 '16 at 03:04
-
thanks! I am trying the service route.. didn't think of actually putting the Observable on it – Sonic Soul Oct 01 '16 at 14:59
-
1@Mark we have learned Angular1 with your posts and know again you are teaching us Angular2, you rock :) – benek Jan 05 '17 at 14:15
-
I posted a two way databinding example here : https://stackoverflow.com/questions/31026886/how-do-i-share-data-between-components-in-angular2/45274521#45274521 – sancelot Jul 24 '17 at 07:50
Component 2, the directive component can define a input property (@input
annotation in Typescript). And Component 1 can pass that property to the directive component from template.
See this SO answer How to do inter communication between a master and detail component in Angular2?
and how input is being passed to child components. In your case it is directive.

- 1
- 1

- 42,589
- 12
- 85
- 88
-
Can i use it for non parent-child components? For example
And i want all components in router-outlet have an access to navbar component, is it possible? – EgorkZe Apr 27 '16 at 11:10 -
1@EgorkZe In order to achieve that, the object you are sharing has to be in the common parent of those two components. There's no other way to share an object among sibling components, which is what you described. – Lucio Mollinedo May 20 '16 at 18:48
you could also store your data in an service with an setter and get it over a getter
import { Injectable } from '@angular/core';
@Injectable()
export class StorageService {
public scope: Array<any> | boolean = false;
constructor() {
}
public getScope(): Array<any> | boolean {
return this.scope;
}
public setScope(scope: any): void {
this.scope = scope;
}
}

- 227
- 2
- 2
-
Your answer is too short but it is the right one for no parent-child components. – Benjamin Lucidarme Jul 21 '17 at 08:36
-
This is a great idea. I am so glad I found this answer. It is especially good if you know that your object is pretty much globally used in many components across the page. – Gherman Dec 08 '17 at 13:06
-
getScope() is returning 'undefined' when I access it after setting it from other page. It sets the variable fine, but can't retrieve it back! Any ideas? – kevaljarsania Jul 12 '18 at 05:23
-
1
-
This is the best solution, so much code and time saver! perfect! it works like a star!!!! – Saim Jan 27 '23 at 21:51
From component
import { Component, OnInit, ViewChild} from '@angular/core';
import { HttpClient } from '@angular/common/http';
import { dataService } from "src/app/service/data.service";
@Component( {
selector: 'app-sideWidget',
templateUrl: './sideWidget.html',
styleUrls: ['./linked-widget.component.css']
} )
export class sideWidget{
TableColumnNames: object[];
SelectedtableName: string = "patient";
constructor( private LWTableColumnNames: dataService ) {
}
ngOnInit() {
this.http.post( 'getColumns', this.SelectedtableName )
.subscribe(
( data: object[] ) => {
this.TableColumnNames = data;
this.LWTableColumnNames.refLWTableColumnNames = this.TableColumnNames; //this line of code will pass the value through data service
} );
}
}
DataService
import { Injectable } from '@angular/core';
import { BehaviorSubject, Observable } from 'rxjs';
@Injectable()
export class dataService {
refLWTableColumnNames: object;//creating an object for the data
}
To Component
import { Component, OnInit } from '@angular/core';
import { dataService } from "src/app/service/data.service";
@Component( {
selector: 'app-linked-widget',
templateUrl: './linked-widget.component.html',
styleUrls: ['./linked-widget.component.css']
} )
export class LinkedWidgetComponent implements OnInit {
constructor(private LWTableColumnNames: dataService) { }
ngOnInit() {
console.log(this.LWTableColumnNames.refLWTableColumnNames);
}
createTable(){
console.log(this.LWTableColumnNames.refLWTableColumnNames);// calling the object from another component
}
}

- 1,193
- 17
- 25
Use the output annotation
@Directive({
selector: 'interval-dir',
})
class IntervalDir {
@Output() everySecond = new EventEmitter();
@Output('everyFiveSeconds') five5Secs = new EventEmitter();
constructor() {
setInterval(() => this.everySecond.emit("event"), 1000);
setInterval(() => this.five5Secs.emit("event"), 5000);
}
}
@Component({
selector: 'app',
template: `
<interval-dir (everySecond)="everySecond()" (everyFiveSeconds)="everyFiveSeconds()">
</interval-dir>
`,
directives: [IntervalDir]
})
class App {
everySecond() { console.log('second'); }
everyFiveSeconds() { console.log('five seconds'); }
}
bootstrap(App);

- 1,338
- 12
- 12