调用子组件的方法


77

我有一个像这样的嵌套子组件:

<app-main>
    <child-component />
</app-main>

我的appMain组件需要在子组件上调用一个方法。

如何在子组件上调用方法?

Answers:


96

您可以使用来获得对元素的引用

@ViewChild('childComponent') child;

其中childComponent是一个模板变量<some-elem #childComponent>`或

@ViewChild(ComponentType) child;

哪里ComponentType是组件或指令的类型,然后是inngAfterViewInit或事件处理程序调用child.someFunc()

ngAfterViewInit() {
  console.log(this.child);
}

另请参阅获取模板中的元素


1
我认为语法是@ViewChild(ComponentName) child;,根据文档,直到ngAfterViewInit()生命周期钩子才可用。
Mark Rajcok

我尝试过这个。传递字符串时,它将使用该名称查找模板变量,否则将使用传递类型的组件或指令。文档中的“选择器”有点误导。
君特Zöchbauer

1
感谢更新。我查看了ViewChild源代码,它说:“除了后代,支持与QueryMetadata相同的查询参数。” 该QueryMetadata API文档解释了所有受支持的不同类型“选择”的。(现在发现这些东西
真是

52

父母和孩子可以通过数据绑定进行通信。

例:

@Component({
    selector: 'child-component',
    inputs: ['bar'],
    template: `"{{ bar }}" in child, counter  {{ n }}`
})
class ChildComponent{
    constructor () {
        this.n = 0;
    }
    inc () {
        this.n++;
    }
}

@Component({
    selector: 'my-app',
    template: `
        <child-component #f [bar]="bar"></child-component><br>
        <button (click)="f.inc()">call child func</button>
        <button (click)="bar = 'different'">change parent var</button>
    `,
    directives: [ChildComponent]
})
class AppComponent {
    constructor () {
        this.bar = 'parent var';
    }
}

bootstrap(AppComponent);  

演示版

#f创建对子组件的引用,并且可以在模板中使用或传递给函数。父级的数据可以通过[ ]绑定传递。


我做错了。当我声明像#f这样的变量时,#f引用dom元素,而当我执行f.someMethodCall()时,由于该方法未在该元素上定义,我得到一个错误。你知道出什么事了吗
PerHornshøj-Schierbeck15年

2
@PerHornshøj-Schierbeck,如果元素上没有组件/指令,则本地模板变量(例如#f)是指元素本身。也许那是你的问题。
Mark Rajcok

@PerHornshøj-Schierbeck变量必须在使用前声明(您不能child-component在示例中放在模板的最后)
JanZahradník16

22

成为儿子

@Component({
    // configuration
    template: `{{data}}`,
    // more configuration
})
export class Son {

  data: number = 3;

  constructor() { }

  updateData(data:number) {
    this.data = data;
  }

}

有父亲的成分

@Component({
    // configuration
})
export class Parent {
  @ViewChild(Son) mySon: Son;

  incrementSonBy5() {
    this.mySon.updateData(this.mySon.data + 5);
  }
}

在父亲的模板中

<son></son>
<button (click)="incrementSonBy5()">Increment son by 5</button>

此解决方案仅适用<son></son>于父模板中的一个实例。如果您有多个实例,则只能在模板的第一个实例中使用。


4
您可以使用@ViewChildren()它们来全部获取。
君特Zöchbauer

这欺骗了我!我的智识不会采用子组件方法,这使我相信它是不可见的!+1
斯卡比亚

1

访问子组件的最佳方法是@ViewChild

假设您的示例中的AppMainComponent带有嵌套的ChildComponent。

// app-main.component.ts
import { Component } from '@angular/core';

@Component({
    selector: 'app-main',
    template: `
        <child-component />
    `
})

export class AppMainComponent {}

您想从ChildComponent调用clear方法。

// child.component.ts
import { Component } from '@angular/core';

@Component({
    selector: 'child-component',
    template: '{{ greeting }}'
})

class ChildComponent {
    greeting: String = 'Hello World!';

    clear() {
        this.greeting = null;
    }
}

您可以通过导入ChildComponent类,ViewChild装饰器并在其中将组件的类作为查询传递来完成此操作。这样,您将可以访问存储在自定义变量中的ChildComponent接口。这是一个例子:

// app-main.component.ts
import { Component, ViewChild } from '@angular/core';
import { ChildComponent } from './components/child/child.component';

@Component({
    selector: 'app-main',
    template: `
        <child-component />
    `
})

class ChildComponent {
    @ViewChild(ChildComponent)
    child: ChildComponent;

    clearChild() {
        this.child.clear();
    }
}

注意!子视图仅在ngAfterViewInit之后可用

Angular初始化组件的视图和子视图之后响应。在第一个ngAfterContentChecked()之后调用一次。仅组件挂钩。

如果要自动执行方法,则需要在此生命周期挂钩中执行。

您还可以得到一个QueryList通过子组件ViewChildren装饰。

import { Component, ViewChildren, QueryList } from '@angular/core';
import { ChildComponent } from './components/child/child.component';

...

@ViewChildren(ChildComponent)
children: QueryList<ChildComponent>;

QueryList可能非常有用,例如您可以订阅子项更改。

也可以创建模板参考变量,并通过ViewChild装饰器对其进行访问。

By using our site, you acknowledge that you have read and understand our Cookie Policy and Privacy Policy.
Licensed under cc by-sa 3.0 with attribution required.