2017-11-14 15 views
3

最初のプロジェクトでAngular 4を学び始めました。FormArrayが反応しないフォームで空でも定義されていない場合も表示しない

私は次のような構造でフォームを表示するように反応性のフォームを作成しました:

Form (FormGroup) 
    | 
    --> aggLevels (FormArray) 
     | 
     --> aggregationLevelName (FormControl? I show it in a label) 
     | 
     --> variableDataForLevel (FormArray) 
      | 
      --> variableDataId (FormControl) 
      | 
      --> value (FormControl) 

これは、コンポーネントのHTMLです:

<form [formGroup]="varDataForm" (ngSubmit)="onSubmit()" novalidate> 
    <div formArrayName="aggLevels" 
     *ngFor="let agLevel of varDataForm.get('aggLevels')?.controls; let aggLevelId = index;"> 
     <div [formGroupName]="aggLevelId"> 
      <label>{{agLevel?.get('aggregationLevelName')?.value}}</label> 
      <div formArrayName="variableDataForLevel" 
       *ngFor="let vardata of varDataForm.get('aggLevels')?.controls[0]?.get('variableDataForLevel')?.controls; let rowIndex = index;"> 
       <div [formGroupName]="rowIndex"> 
        <select formControlName="variableDataId"> 
         <option *ngFor="let gs1 of gs1AIs" [value]="gs1.Id">{{gs1.description}}</option> 
        </select> 
        <input formControlName="value" placeholder="Valor"> 
        <div class="error" *ngIf="vardata.get('value').hasError('required') && vardata.get('value').touched"> 
         Obligatorio 
        </div> 
       </div> 
      </div> 
     </div> 
    </div> 
</form> 

どの要素がない場合、ここで私の問題がありますin variableDataForLevel FormArray。私はこのメッセージが表示されます。

ng:///AppModuleShared/VarDataComponent.ngfactory.js:49 ERROR Error: Cannot find control with path: 'aggLevels -> 1 -> variableDataForLevel -> 0'

私はすべての要素の最後に安全なナビゲーション演算子?を追加しようとしました:

varDataForm.get('aggLevels')?.controls[0]?.get('variableDataForLevel')?.controls 

しかし、私はまだ問題を取得します。 FormArray aggLevelsvariableDataForLevelは私がインスタンスを作成したときにインスタンス化されているため、未定義ではありません。

は私が空の場合FormArrayを表示しないために何ができますか?

私は角度を持つ新しいと私はそれを修正する方法についてのいくつかの知識を持っています。

これは、このコンポーネントのコンポーネントのtypescriptですクラスです:

import { Component, Inject } from '@angular/core'; 
import { FormBuilder, FormGroup, FormArray, FormControl, Validators } from '@angular/forms'; 
import { Http, RequestOptions, Headers } from '@angular/http'; 
import { IProductionOrder } from '../../interfaces/productionorder.interface'; 
import { IAggLevel } from '../../interfaces/aglevel.interface'; 
import { IGS1 } from '../../interfaces/gs1.interface'; 
import { IVarData } from '../../interfaces/vardata.interface'; 

@Component({ 
    selector: 'vardata', 
    templateUrl: './vardata.component.html' 
}) 

export class VarDataComponent { 
    public productionorders: IProductionOrder[]; 
    public gs1AIs: IGS1[]; 

    public varDataForm: FormGroup = new FormGroup({}); 
    public selectedProductionOrder: string; 

    constructor(private http: Http, 
     @Inject('BASE_URL') private baseUrl: string, 
     @Inject(FormBuilder) private fb: FormBuilder) { 

     this.selectedProductionOrder = ''; 

     http.get(baseUrl + 'api/ProductionOrder/0').subscribe(result => { 
      this.productionorders = result.json() as IProductionOrder[]; 

      console.log(this.productionorders); 
     }, error => console.error(error)); 
    } 

    productionOrderChanges() { 
     if (this.selectedProductionOrder != '') { 
      let urlVarData: string = this.baseUrl + 'api/VariableData/' + this.selectedProductionOrder; 
      let urlAggLevel: string = this.baseUrl + 'api/AggregationLevelConfiguration/' + this.selectedProductionOrder; 

      this.http.get(urlAggLevel).subscribe(result => { 
       let aggLevels: IAggLevel[] = result.json() as IAggLevel[]; 

       console.log('niveles'); 
       console.log(aggLevels); 

       if ((aggLevels != null) && (aggLevels.length > 0)) { 

        this.http.get(urlVarData).subscribe(result => { 
         let varDatas: IVarData[] = result.json() as IVarData[]; 

         console.log('variable data'); 
         console.log(varDatas); 

         this.varDataForm = this.fb.group({ 
          aggLevels: this.fb.array(this.createForm(aggLevels, varDatas)) 
         }); 

        }, error => console.error(error)); 
       } 

      }, error => console.error(error)); 
     } 
     else 
      this.varDataForm = new FormGroup({}); 
    } 

    createForm(aggLevels: IAggLevel[], varDatas: IVarData[]): any[] { 
     let array: any[] = []; 

     for (let level of aggLevels) { 
      let group: FormGroup; 
      let varDataForLevel: IVarData[] = 
       varDatas.filter(v => v.aggregationLevelConfigurationId == level.aggregationLevelConfigurationId); 

      group = this.createFormGroupForLevel(level, varDataForLevel); 
      array.push(group); 
     } 

     return array; 
    } 

    createFormGroupForLevel(level: IAggLevel, varDatas: IVarData[]): FormGroup { 
     let group: FormGroup; 

     group = this.fb.group({ 
      aggregationLevelName: level.name, 
      variableDataForLevel: this.fb.array(this.createVarDataControls(varDatas)) 
     }); 

     return group; 
    } 

    createVarDataControls(varData: IVarData[]): any[] { 
     let array: any[] = []; 

     for (let vData of varData) { 
      let group: FormGroup; 

      group = this.fb.group({ 
       variableDataId: vData.variableDataId, 
       value: vData.value 
      }); 

      array.push(group); 
     } 

     return array; 
    } 



    disableSubmit() { 
     return (!this.varDataForm.valid); 
    } 

    onSubmit() { 
    } 
} 

私は、問題はここにあると思う:

createVarDataControls(varData: IVarData[]): any[] { 
    let array: any[] = []; 

    for (let vData of varData) { 
     let group: FormGroup; 

     group = this.fb.group({ 
      variableDataId: vData.variableDataId, 
      value: vData.value 
     }); 

     array.push(group); 
    } 

    return array; 
} 

varDataが空の場合、それはAny[]を返しますが、私はしないでくださいフォームに空の何かを表示したくないので、これがエラーであると思います。

を編集しました。varDataが空の場合はFormGroupを返します。これは私が継続させるが、私はそれは私の質問に答えるとは思わない:それで

createVarDataControls(varData: IVarData[]): any[] { 
    let array: any[] = []; 

    if ((varData == null) || (varData.length == 0)) { 
     let group: FormGroup; 

     group = this.fb.group({ 
      variableDataId: '', 
      value: '' 
     }); 

     array.push(group); 
    } 
    else { 
     for (let vData of varData) { 
      let group: FormGroup; 

      group = this.fb.group({ 
       variableDataId: vData.variableDataId, 
       value: vData.value 
      }); 

      array.push(group); 
     } 
    } 

    return array; 
} 

は、私はすべての例外を得ることはありませんが、それは2つの空のコントロールを示しており、私はそれを望んでいません。これは、問題を特定するのに役立ちます。

+0

コンポーネントでフォームとコントロールを初期化するコードを追加してください。ありがとう – amal

+0

@amal私は使用しているTypescriptクラスで質問を更新しました。 – VansFannel

答えて

1

私は問題を発見しました。私はそれが誰かを助けることができるなら私の解決策を望む。二FormArrayループでは

<div formArrayName="variableDataForLevel" 
       *ngFor="let vardata of varDataForm.get('aggLevels')?.controls[0]?.get('variableDataForLevel')?.controls; let rowIndex = index;"> 

しかし、正しいものは次のとおりです。

<div formArrayName="variableDataForLevel" 
     *ngFor="let vardata of agLevel.get('variableDataForLevel')?.controls; let rowIndex = index;"> 

私はagLevelvarDataForm.get('aggLevels')?.controls[0]?を変更しました。

そしてまた、私はそれに表示する変数データが​​存在しない場合にundefinedを返すためにメソッドcreateVarDataControlsを変更しました:

createVarDataControls(varData: IVarData[]): any[] | undefined { 
    let array: any[] = []; 

    if ((varData == null) || (varData.length == 0)) { 
     return undefined; 
    } 
    else { 
     for (let vData of varData) { 
      let group: FormGroup; 

      group = this.fb.group({ 
       aggregationLevelConfigurationId: vData.aggregationLevelConfigurationId, 
       productionOrderId: vData.productionOrderId, 
       variableDataId: vData.variableDataId, 
       value: vData.value 
      }); 

      array.push(group); 
     } 

     return array; 
    } 
} 

しようとすると、障害が解決策を見つけるまで!

関連する問題