Angular2 modèle de formulaire imbriqué

C'est juste de la folie , on dirait qu'il n'y a aucun moyen d'avoir une forme l'un de ses entrées est dans un composant enfant .

j'ai lu tous les blogs et tutoriels et tout , aucun moyen de résoudre cela .

le problème est quand un composant enfant va avoir n'importe quel type de directives de forme ( ngModel , ngModelGroup ou n'importe quoi d'autre ..) , il l'habitude de travailler.

ce n'est qu'un problème dans les template driven les formes

C'est le plunker:

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

@Component({
  selector: 'child-form-component',
  template: ` 
  <fieldset ngModelGroup="address">
    <div>
      <label>Street:</label>
      <input type="text" name="street" ngModel>
    </div>
    <div>
      <label>Zip:</label>
      <input type="text" name="zip" ngModel>
    </div>
    <div>
      <label>City:</label>
      <input type="text" name="city" ngModel>
    </div>
  </fieldset>`
})

export class childFormComponent{


}

@Component({
  selector: 'form-component',
  directives:[childFormComponent],
  template: `
    <form #form="ngForm" (ngSubmit)="submit(form.value)">
      <fieldset ngModelGroup="name">
        <div>
          <label>Firstname:</label>
          <input type="text" name="firstname" ngModel>
        </div>
        <div>
          <label>Lastname:</label>
          <input type="text" name="lastname" ngModel>
        </div>
      </fieldset>

      <child-form-component></child-form-component>

      <button type="submit">Submit</button>
    </form>

    <pre>
      {{form.value | json}}
    </pre>

    <h4>Submitted</h4>
    <pre>    
      {{value | json }}
    </pre>
  `
})
export class FormComponent {

  value: any;

  submit(form) {
    this.value = form; 
  }
}
33
demandé sur Jacob Stamm 2016-08-31 09:27:49

5 réponses

une solution simple est de fournir ControlContainerviewProviders tableau de votre composant enfant comme:

import { ControlContainer, NgForm } from '@angular/forms';

@Component({
 ...,
 viewProviders: [ { provide: ControlContainer, useExisting: NgForm } ]
})
export class ChildComponent {}

Stackblitz Exemple

lisez aussi cet article qui explique pourquoi cela fonctionne.

mise à Jour

si vous cherchez imbriqués model driven forme alors voici le similaire approche:

@Component({
  selector: 'my-form-child',
  template: `<input formControlName="age">`,
  viewProviders: [
    {
      provide: ControlContainer,
      useExisting: FormGroupDirective
    }
  ]
})
export class ChildComponent {
  constructor(private parent: FormGroupDirective) {}

  ngOnInit() {
    this.parent.form.addControl('age', new FormControl('', Validators.required))
  }
}

Ng-exécuter l'Exemple

26
répondu yurzui 2018-03-02 06:22:49

Lecture à travers un tas de liés github [1] [2], je n'ai pas trouvé de moyen simple de faire angulaire ajouter un enfant Component's des contrôles à un parent ngForm (certaines personnes les appellent aussi des formes imbriquées, des entrées imbriquées ou des contrôles complexes).

alors ce que je vais montrer ici est un contournement que fonctionne pour moi, à l'aide d'une ngForm directives pour les parents et les enfants. Ce n'est pas parfait, mais ça me rapproche assez. que je me suis arrêté là.

je déclare mon childFormComponent avec un ngForm directive (à savoir un formulaire html, balise, seule la directive):

<fieldset ngForm="addressFieldsForm" #addressFieldsForm="ngForm">
  <div class="form-group">
    <label for="email">Email</label>
    <input type="email" class="form-control" [(ngModel)]="model.email" name="email" #email="ngModel" required placeholder="Email">
  </div>
  ...

le composant expose alors le addressFieldsForm comme une propriété, et s'exporte aussi comme un variable de référence du modèle:

@Component({
  selector: 'mst-address-fields',
  templateUrl: './address-fields.component.html',
  styleUrls: ['./address-fields.component.scss'],
  exportAs: 'mstAddressFields'
})
export class AddressFieldsComponent implements OnInit {
  @ViewChild('addressFieldsForm') public form: NgForm;
  ....

Le formulaire parent peut alors utiliser le formulaire enfant composant comme ceci:

  <form (ngSubmit)="saveAddress()" #ngFormAddress="ngForm" action="#">
    <fieldset>
      <mst-address-fields [model]="model" #addressFields="mstAddressFields"></mst-address-fields>
      <div class="form-group form-buttons">
        <button class="btn btn-primary" type="submit" [disabled]="!ngFormAddress.valid || !addressFields.form.valid">Save</button>
      </div>
    </fieldset>
  </form>

Notez que le bouton soumettre explicitement vérifie l'état valide à la fois sur les ngFormAddress et addressFields formulaire. De cette façon, je peux au moins raisonnablement composer des formes complexes, même s'il y a quelques boilerplate.

19
répondu Johannes Rudolph 2017-04-14 12:29:13

une Autre solution possible:

@Directive({
    selector: '[provide-parent-form]',
    providers: [
        {
            provide: ControlContainer,
            useFactory: function (form: NgForm) {
                return form;
            },
            deps: [NgForm]
        }
    ]
})
export class ProvideParentForm {}

il suffit de placer cette directive dans un composant enfant quelque part en haut de la hiérarchie des noeuds (avant tout ngModel).

Comment ça marche: NgModel qualifie recherche de dépendances de la forme parent avec @Host (). Ainsi, une forme d'un composant parent n'est pas visible à NgModel dans un composant enfant. Mais nous pouvons l'injecter et le fournir à l'intérieur d'un composant enfant en utilisant le code démontré ci-dessus.

4
répondu Artem Andreev 2017-07-25 15:39:55

officiel docs: This directive can only be used as a child of NgForm.

donc je pense que vous pouvez essayer d'envelopper votre composant enfant dans différent ngForm, et attendre en composant parent résultat @Output du composant enfant. Laissez-moi savoir si vous avez besoin de plus de précisions.

mise à jour: Plunker avec quelques modifications, j'ai converti la forme enfant en forme de modèle, parce qu'il n'y a aucun moyen d'écouter sur la forme de forme de la mise à jour avant qu'elle ne soit soumise.

2
répondu Mikki 2016-08-31 08:44:09

j'ai créé une solution en utilisant une directive et un service. Une fois que vous les ajoutez à votre module, le seul autre changement de code que vous devez faire est au niveau du formulaire dans les modèles. Cela fonctionne avec des champs de forme dynamiquement ajoutés et AOT. Il prend également en charge plusieurs formulaires non reliés sur une page. Voici la plunker: plunker.

utilise cette directive:

import { Directive, Input } from '@angular/core';
import { NgForm } from '@angular/forms';
import { NestedFormService } from './nested-form.service';

@Directive({
    selector: '[nestedForm]',
    exportAs: 'nestedForm'   
})
export class NestedFormDirective {    
    @Input('nestedForm') ngForm: NgForm;
    @Input() nestedGroup: string;
       
    public get valid() {
        return this.formService.isValid(this.nestedGroup);
    }

    public get dirty() {
        return this.formService.isDirty(this.nestedGroup);
    }

    public get touched() {
        return this.formService.isTouched(this.nestedGroup);
    }
    
    constructor(      
        private formService: NestedFormService
    ) { 
        
    }

    ngOnInit() {   
        this.formService.register(this.ngForm, this.nestedGroup);
    }

    ngOnDestroy() {
        this.formService.unregister(this.ngForm, this.nestedGroup);
    } 

    reset() {
        this.formService.reset(this.nestedGroup);
    }
}

Et ce service:

import { Injectable } from '@angular/core';
import { NgForm } from '@angular/forms';

@Injectable()
export class NestedFormService {

    _groups: { [key: string] : NgForm[] } = {};
      
    register(form: NgForm, group: string = null) {           
        if (form) {
            group = this._getGroupName(group);
            let forms = this._getGroup(group);        
            if (forms.indexOf(form) === -1) {
                forms.push(form);
                this._groups[group] = forms;
            }
        }
    }

    unregister(form: NgForm, group: string = null) {        
        if (form) {
            group = this._getGroupName(group);
            let forms = this._getGroup(group);
            let i = forms.indexOf(form);
            if (i > -1) {
                forms.splice(i, 1);
                this._groups[group] = forms;
            }
        }
    }

    isValid(group: string = null) : boolean {   
        group = this._getGroupName(group);         
        let forms = this._getGroup(group);
       
        for(let i = 0; i < forms.length; i++) {
            if (forms[i].invalid)
                return false;
        }
        return true;
    } 

    isDirty(group: string = null) : boolean {   
        group = this._getGroupName(group);         
        let forms = this._getGroup(group);
       
        for(let i = 0; i < forms.length; i++) {
            if (forms[i].dirty)
                return true;
        }
        return false;
    } 

    isTouched(group: string = null) : boolean {   
        group = this._getGroupName(group);         
        let forms = this._getGroup(group);
       
        for(let i = 0; i < forms.length; i++) {
            if (forms[i].touched)
                return true;
        }
        return false;
    } 

    reset(group: string = null) {
        group = this._getGroupName(group);         
        let forms = this._getGroup(group);
       
        for(let i = 0; i < forms.length; i++) {
            forms[i].onReset();
        }
    }

    _getGroupName(name: string) : string {
        return name || '_default';
    }

    _getGroup(name: string) : NgForm[] {        
        return this._groups[name] || [];
    }          
}

à utilisez la directive dans un composant parent avec un formulaire:

import { Component, Input } from '@angular/core';
import { Person } from './person.model';

@Component({
    selector: 'parent-form',
    template: `  
        <div class="parent-box">

            <!--
            ngForm                        Declare Angular Form directive
            #theForm="ngForm"             Assign the Angular form to a variable that can be used in the template
            [nestedForm]="theForm"        Declare the NestedForm directive and pass in the Angular form variable as an argument
            #myForm="nestedForm"          Assign the NestedForm directive to a variable that can be used in the template
            [nestedGroup]="model.group"   Pass a group name to the NestedForm directive so you can have multiple forms on the same page (optional).
            -->

            <form 
                ngForm                  
                #theForm="ngForm" 
                [nestedForm]="theForm"
                #myForm="nestedForm" 
                [nestedGroup]="model.group">        

                <h3>Parent Component</h3> 
                <div class="pad-bottom">
                    <span *ngIf="myForm.valid" class="label label-success">Valid</span>
                    <span *ngIf="!myForm.valid" class="label label-danger">Not Valid</span>
                    <span *ngIf="myForm.dirty" class="label label-warning">Dirty</span>    
                    <span *ngIf="myForm.touched" class="label label-info">Touched</span>    
                </div> 

                <div class="form-group" [class.hasError]="firstName.invalid">
                    <label>First Name</label>
                    <input type="text" id="firstName" name="firstName" [(ngModel)]="model.firstName" #firstName="ngModel" class="form-control" required />
                </div>

                <child-form [model]="model"></child-form>
               
                <div>
                    <button type="button" class="btn btn-default" (click)="myForm.reset()">Reset</button>
                </div>
            </form>   
        </div>
    `
})
export class ParentForm {   
    
    model = new Person();
   
}

Puis dans un composant enfant:

import { Component, Input } from '@angular/core';
import { Person } from './person.model';

@Component({
    selector: 'child-form',
    template: `  
        <div ngForm #theForm="ngForm" [nestedForm]="theForm" [nestedGroup]="model.group" class="child-box">
            <h3>Child Component</h3>
            <div class="form-group" [class.hasError]="lastName.invalid">
                <label>Last Name</label>
                <input type="text" id="lastName" name="lastName" [(ngModel)]="model.lastName" #lastName="ngModel" class="form-control" required />
            </div>
        </div>  
    `
})
export class ChildForm {    
    @Input() model: Person;
      
}
1
répondu nuckolhead 2017-09-15 14:23:13