Получение ссылки на дочерний компонент в родительском компоненте

В 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);
  }
}