Получение ссылки на дочерний компонент в родительском компоненте
В Angular 2 у меня есть компонент, который имеет дочерний компонент. Однако я хочу получить копию этого дочернего компонента для использования в родительском компоненте, для вызова его функций или чего-либо еще.
Я узнал, что могу использовать локальные переменные, и таким образом я смогу использовать компонент в шаблоне. Однако я не только использую его в шаблоне, я хочу использовать его в реальном коде компонента.
Я нашел способ сделать это, вот дочерний код:
//our child
import {Component, OnInit, EventEmitter} from 'angular2/core'
@Component({
selector: 'my-child',
providers: [],
template: '
<div>
<h2>Child</h2>
</div>
',
directives: [],
outputs: ['onInitialized']
})
export class Child implements OnInit{
onInitialized = new EventEmitter<Child>();
constructor() {
this.name = 'Angular2'
}
ngOnInit() {
this.onInitialized.emit(this);
}
}
родитель:
//our root app component
import {Component} from 'angular2/core'
import {Child} from './child'
@Component({
selector: 'my-app',
providers: [],
template: '
<div>
<h2>Hello {{name}}</h2>
<my-child (onInitialized)="func($event)"></my-child>
</div>
',
directives: [Child]
})
export class App {
constructor() {
this.name = 'Angular2'
}
func(e) {
console.log(e)
}
}
Я реализовал это здесь, в этом поршне. Но это похоже на взлом.
Нет ли более простого способа присоединить компонент к переменной в его родительском элементе?
Ответы
Ответ 1
Вы можете использовать ViewChild
<child-tag #varName></child-tag>
@ViewChild('varName') someElement;
ngAfterViewInit() {
someElement...
}
где varName
- переменная шаблона, добавленная к элементу. Кроме того, вы можете сделать запрос по типу компонента или директивы.
Существуют альтернативы, такие как ViewChildren
, ContentChild
, ContentChildren
.
@ViewChildren
также может быть использован в конструкторе.
constructor(@ViewChildren('var1,var2,var3') childQuery:QueryList)
Преимущество в том, что результат доступен раньше.
См. Также http://www.bennadel.com/blog/3041-constructor-vs-property-querylist-injection-in-angular-2-beta-8.htm, чтобы узнать о некоторых преимуществах и недостатках использования конструктора или поля.
Примечание: @Query()
является устаревшим предшественником @ContentChildren()
Обновить
Query
в настоящее время является просто абстрактным базовым классом. Я не нашел, используется ли он вообще https://github.com/angular/angular/blob/2.1.x/modules/@angular/core/src/metadata/di.ts#L145
Ответ 2
Вам необходимо использовать декоратор @ViewChild
для ссылки на дочерний компонент из родительского путем внедрения:
import { Component, ViewChild } from 'angular2/core';
(...)
@Component({
selector: 'my-app',
template: '
<h1>My First Angular 2 App</h1>
<child></child>
<button (click)="submit()">Submit</button>
',
directives:[App]
})
export class AppComponent {
@ViewChild(Child) child:Child;
(...)
someOtherMethod() {
this.searchBar.someMethod();
}
}
Вот обновленный план: http://plnkr.co/edit/mrVK2j3hJQ04n8vlXLXt?p=preview.
Вы можете заметить, что декоратор параметров @Query
также может быть использован:
export class AppComponent {
constructor(@Query(Child) children:QueryList<Child>) {
this.childcmp = children.first();
}
(...)
}
Ответ 3
На самом деле вы можете пойти с ViewChild API
...
parent.ts
<button (click)="clicked()">click</button>
export class App {
@ViewChild(Child) vc:Child;
constructor() {
this.name = 'Angular2'
}
func(e) {
console.log(e)
}
clicked(){
this.vc.getName();
}
}
child.ts
export class Child implements OnInit{
onInitialized = new EventEmitter<Child>();
...
...
getName()
{
console.log('called by vc')
console.log(this.name);
}
}