85 votes

Angular 4 appelle la méthode parent dans un composant enfant

Je veux appeler la méthode parent (deletePhone) dans un composant enfant dans Angular 4. Comment puis-je le faire correctement ?

mon composant parent ressemble à :

export class ContactInfo implements OnInit {
    phoneForm: FormGroup;
    phones: Phone[];

    constructor(private fb: FormBuilder,
            private userService: UserService) {
    }

    ngOnInit() {
        this.userService.getDataPhones().subscribe(
            phones => {
                this.phones = phones;
            });

        this.phoneForm = this.fb.group({
            phone: ['', [Validators.pattern(PHONE_PATTERN)]]
        });
    }

    deletePhone(phone: Phone) {
        this.userService.deleteUserPhone(phone)
            .subscribe(res => {
                let index = this.phones.indexOf(phone);
                if (index > -1) {
                    this.phones.splice(index, 1);
                }
        });
    }
}

2 votes

Avez-vous envisagé d'utiliser un service pour ce type de fonctionnalité ?

165voto

Günter Zöchbauer Points 21340
import { Output, EventEmitter } from '@angular/core'; 

...

class ChildComponent {
  @Output() someEvent = new EventEmitter<string>();

  callParent(): void {
    this.someEvent.next('somePhone');
  }
}

Sur ContactInfo Le modèle de l'entreprise

<child-component (someEvent)="deletePhone($event)"

3 votes

Vous pouvez ajouter le type prévu comme "new EventEmitter<boolean>()" :)

13 votes

Pour les débutants, vous devez ajouter "import { Output, EventEmitter } from '@angular/core'".

9 votes

Ce ne sont pas seulement les débutants, La réponse est incomplète. La réponse devrait être éditée pour inclure l'addendum de Sergey.

23voto

Nitin Jadhav Points 124

Cela a fonctionné pour moi (exemple tiré des documents officiels) :

https://angular.io/api/core/EventEmitter#examples

Enfant :

@Component({
  selector: 'zippy',
  template: `
  <div class="zippy">
    <div (click)="toggle()">Toggle</div>
    <div [hidden]="!visible">
      <ng-content></ng-content>
    </div>
 </div>`})
export class Zippy {
  visible: boolean = true;
  @Output() open: EventEmitter<any> = new EventEmitter();
  @Output() close: EventEmitter<any> = new EventEmitter();

  toggle() {
    this.visible = !this.visible;
    if (this.visible) {
      this.open.emit(null); //emit event here
    } else {
      this.close.emit(null);
    }
  }
}

Parent :

<zippy (open)="onOpen($event)" (close)="onClose($event)"></zippy>

10voto

Je n'aime pas le code passe-partout comme @Output(). J'ai trouvé une autre solution, il suffit de passer l'objet avec un nombre quelconque de fonctions anonymes.

import { Component, OnInit } from '@angular/core';

@Component({
  selector: 'app-parent',
  styleUrls: ['./parent.component.css'],
  template: `
  <app-child [parentApi]="getParentApi()"></app-child>
`,
})
export class ParentComponent implements OnInit {

  getParentApi(): ParentComponentApi {
    return {
      callParentMethod: (name) => {
        this.parentMethod(name)
      }
    }
  }

  constructor() { }

  ngOnInit() {
  }

  parentMethod(name: string) {
    console.log(`Hello ${name} from parent`)
  }

}

export interface ParentComponentApi {
  callParentMethod: (string) => void
}

Et enfant :

import { Component, OnInit, Input } from '@angular/core';
import { ParentComponentApi } from '../parent/parent.component';

@Component({
  selector: 'app-child',
  template: `<button (click)="callParent()">call parent</button>`,
  styleUrls: ['./child.component.css']
})
export class ChildComponent implements OnInit {

  @Input() parentApi: ParentComponentApi

  constructor() { }

  callParent() {
    this.parentApi.callParentMethod("child")
  }

  ngOnInit() {

  }

}

Je pense que c'est assez sûr de faire ça comme ça, non ?

4 votes

J'aime l'idée de ce système. Malheureusement, il ne fonctionne pas dans Angular 7. Il y a une erreur d'analyse des modèles.

1 votes

Je préfère cette solution, qui fonctionne bien pour moi dans Angular 10.

0 votes

@Output n'est pas un modèle passe-partout, si vous envisagez @Input . Ils ont tous deux des aspects différents dans la relation parent-enfant. Nous ne pouvons pas les remplacer l'un par l'autre.

3voto

sully Points 11

C'est plus simple que vous ne le pensez. La clé est de passer une méthode parent à une propriété enfant @Input . Testez-le en ligne

Composante parentale

@Component({
  selector: 'my-app',
  template: `
    <h1>Parent:</h1>
    <p>Parent counting: {{this.count}}</p>

    <child-comp [childInc]="this.inc"></child-comp>
  `
})
export class AppComponent {
  name = "I'm Parent";

  count = 0;

  inc = () => {
    this.count++;
  }
}

Composante enfant

@Component({
  selector: 'child-comp',
  template: `
  <h1>Child:</h1> 
  <button (click)="childInc()">Click me!</button>
  `
})
export class ChildComponent  {
  name = "I'm child"

  @Input() childInc: () => void
}

J'ai utilisé inc = () => {...} notion de parent, qui peut se souvenir de la bonne this . Si vous utilisez le inc(){...} vous devez alors lier le parent this como [childInc]="this.inc.bind(this)" .

Prograide.com

Prograide est une communauté de développeurs qui cherche à élargir la connaissance de la programmation au-delà de l'anglais.
Pour cela nous avons les plus grands doutes résolus en français et vous pouvez aussi poser vos propres questions ou résoudre celles des autres.

Powered by:

X