form.valid always true angular

2019-08-24 19:54发布

问题:

I am working on angular form validation after user clicks on submit button, I am setting the required validations to empty fields on submit and also the error messages as 'required' showing in the form but after that the form is getting submitted. I have searched for solutions but did not get any one working.

component.html:

<form [formGroup]="companyForm" #formDir="ngForm" >
      <!-- company details -->

        <ng-container formGroupName="company_details">
          <div *ngSwitchCase="'company_details'">
            <div fxLayout="row wrap">
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>company name</mat-label>
                  <input matInput formControlName="name">
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>display name</mat-label>
                  <input matInput formControlName= "display_name">
                </mat-form-field>
              </div>
            </div>
          </div>
        </ng-container>
        <ng-container formGroupName="company_settings">
          <div *ngSwitchCase="'company_details'">
            <div fxLayout="row wrap">
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>website</mat-label>
                  <input matInput formControlName="website">
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                  <!-- <input matInput placeholder="FISCAL START" value="January 2018"> -->
                  <mat-label>fiscal start</mat-label>
                  <mat-select formControlName="fiscal_start">
                    <mat-option *ngFor="let fl of FiscalList | keyvalue" [value]="fl.key">
                      {{fl.value}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>

              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>company contact</mat-label>
                  <input matInput  formControlName="contact_number">
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>faxe</mat-label>
                  <input matInput formControlName="fax">
                </mat-form-field>
              </div>
              <div fxFlex="100">
                <h2 class="text-uppercase font-weight-900 font-size-18">Description</h2>
                <div fxfill fusePerfectScrollbar>
                  <quill-editor [style.display]="'inline-block'" [style.height]="'200px'" #description
                    formControlName="description">
                  </quill-editor>
                </div>
              </div>
            </div>
          </div>
          <div *ngSwitchCase="'logos'">
            <div class="page-layout simple">
              <div class="logo-bg">
                <div fxLayout="row wrap" fxFill>
                  <div fxFlex="50" class="px-24 pb-24 text-center">
                    <h4 class="font-size-16 text-bold text-center mb-0">Company Logo </h4>
                    <span class="text-normal">500px X 300px</span>
                    <div class="file-upload-bg p-12">
                      <div class="text-center h-160">
                        <!-- <img src={{logo_path}} style="max-width:100%;height: 130px;"> -->
                        <img [src]="url" style="max-width:100%;height: auto;">
                      </div>
                      <div class="file-upload text-center">
                        <label for="file"> Choose file </label>
                        <input type="file" accept="image/*" id="file" formControlName="company_logo" 
                          (change)="onSelectFile($event)">
                      </div>
                    </div>

                  </div>

                  <div fxFlex="50" class="px-24 pb-24 text-center">
                    <h4 class="font-size-16 text-bold text-center mb-0">Social Media Logo </h4>
                    <span class="text-normal">500px X 300px</span>
                    <div class="file-upload-bg p-12">
                      <div class="text-center h-160">
                         <img [src]="url1" style="max-width:100%;height: auto;">
                      </div>
                      <div class="file-upload text-center">
                        <label for="file1"> Choose file </label>
                        <input type="file" accept="image/*" id="file1" formControlName="social_media_logo_path"  (change)="onSelectFile1($event)">
                      </div>
                    </div>
                  </div>
                </div>
              </div>
            </div>
          </div>
          <div *ngSwitchCase="'locale_information'">
            <div fxLayout="row wrap">
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>Currency</mat-label>
                  <mat-select formControlName="currency">
                    <mat-option *ngFor="let currency of currencyList | keyvalue" [value]="currency.key">
                      {{currency.value}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>Date Format</mat-label>
                  <mat-select formControlName="date_format">
                    <mat-option *ngFor="let df of defaultDateFormats | keyvalue" [value]="df.key">
                      {{df.value}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
              <div fxFlex="50"> 
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>Timezone</mat-label>
                  <mat-select formControlName="timezone">
                    <mat-option *ngFor="let tz of timezone_list[userList.country_id]" [value]="tz.id">
                      {{tz.name}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>Time Format</mat-label>
                  <mat-select formControlName="timeformat">
                    <mat-option *ngFor="let tf of timeformat | keyvalue" [value]="tf.key">
                      {{tf.value}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
            </div>
          </div>
          <div *ngSwitchCase="'resume_inbox_config'">
            <mat-checkbox formControlName="resume_inbox_non_user" (change)="changeSelect($event)" > 
              All non users to send resumes
            </mat-checkbox>
            <br>
            <mat-form-field appearance="outline" *ngIf="companyForm.get('company_settings').get('resume_inbox_non_user').value" class="example-full-width w-40-p mt-12">
              <mat-label>Candidate Owner</mat-label>
              <mat-select formControlName="resume_inbox_contact_owner" multiple>
                <mat-option *ngFor="let eachc of candidateList" [value]="eachc">{{eachc.name}}</mat-option>
              </mat-select>
              <mat-error *ngIf="(companyForm.get('company_settings').get('resume_inbox_non_user').value && companyForm.get('company_settings').get('resume_inbox_contact_owner').errors?.required) ">
                Candidate Owner is required
              </mat-error>
            </mat-form-field>
          </div>
        </ng-container>

        <div *ngSwitchCase="'billing_address'">
          <ng-container formGroupName="billing" >
            <!-- <form> -->
            <div fxLayout="row wrap" class="pl-28">
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>street</mat-label>
                  <input matInput formControlName="street">
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>city</mat-label>
                  <input matInput formControlName="city">
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>Country</mat-label>
                  <mat-select formControlName="country">
                    <mat-option value="">-- Select Country --</mat-option>
                    <mat-option *ngFor="let country of objectValues(masterdatalist.countriesList)" [value]="country.id">
                      {{country.name}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="w-80-p">
                  <mat-label>State</mat-label>
                  <mat-select formControlName="state">
                    <mat-option *ngFor="let state of masterdatalist.countriesStates[companyForm.get('billing').get('country').value]" [value]="state.id">
                      {{state.name}}
                    </mat-option>
                  </mat-select>
                </mat-form-field>
              </div>
              <div fxFlex="50">
                <mat-form-field appearance="outline" class="example-full-width w-80-p">
                    <mat-label>zip</mat-label>
                  <input matInput formControlName="zip">
                </mat-form-field>
              </div>
            </div>
          </ng-container>
        </div>

        <div *ngSwitchCase="'ip_restrictions'">
          <div class="mb-20" fxLayout="row wrap" fxLayoutAlign="start center">
            <span class="mr-12">
              Applicable to:
            </span>
            <div *ngFor="let user of companyForm.get('applicable_to').controls; let i = index" >
              <mat-checkbox class="mr-32" *ngIf="i === 0" [formControl]="user" [checked]="user.value === 1 ? true : false ">Recruitment Activities</mat-checkbox>
              <mat-error *ngIf="(user.touched || user.untouched) && !user.valid">
                Required
              </mat-error>
              <mat-checkbox [formControl]="user" *ngIf="i === 1" [checked]="user.value === 1 ? true : false " (change)="checkRecruitment($event);">HR Management</mat-checkbox>
            </div>
          </div>
          <div formArrayName="company_ip_addresses">
            <div *ngFor="let item of companyForm.get('company_ip_addresses').controls; let i=index; let isFirst = first">
              <ng-container [formGroupName]="i">

              <div fxLayout="row wrap" fxLayoutAlign="space-around center">
                <div fxFlex="30">
                  <mat-form-field appearance="outline" class="example-full-width" class="w-90-p">
                      <mat-label>start ip</mat-label>
                    <input matInput formControlName="start_ip" (blur)="checkStartEndIP($event.target.value,i,'start_ip')" >

                    <mat-error *ngIf="(companyForm.get('company_ip_addresses').at(i).controls.start_ip.touched  || companyForm.get('company_ip_addresses').at(i).controls.start_ip.untouched) && companyForm.get('company_ip_addresses').at(i).controls.end_ip.valid && !companyForm.get('company_ip_addresses').at(i).controls.start_ip.value ">
                      Required Start IP
                    </mat-error>
                    <mat-error *ngIf="((companyForm.get('company_ip_addresses').at(i).controls.end_ip.dirty || companyForm.get('company_ip_addresses').at(i).controls.end_ip.touched) && companyForm.get('company_ip_addresses').at(i).controls.start_ip.errors?.pattern)">
                        Invalid Start IP
                    </mat-error>
                  </mat-form-field>
                </div>


                <div fxFlex="30">
                  <mat-form-field appearance="outline" class="example-full-width" class="w-90-p">
                      <mat-label>End Ip</mat-label>
                    <input matInput formControlName= "end_ip" (blur)="checkStartEndIP($event.target.value,i,'end_ip')">
                  <mat-error *ngIf="(companyForm.get('company_ip_addresses').at(i).controls.end_ip.touched  || companyForm.get('company_ip_addresses').at(i).controls.end_ip.untouched) && companyForm.get('company_ip_addresses').at(i).controls.start_ip.valid && !companyForm.get('company_ip_addresses').at(i).controls.end_ip.value ">
                      Required End IP
                  </mat-error>
                  <mat-error *ngIf="((companyForm.get('company_ip_addresses').at(i).controls.end_ip.dirty || companyForm.get('company_ip_addresses').at(i).controls.end_ip.touched) && companyForm.get('company_ip_addresses').at(i).controls.end_ip.errors?.pattern)">
                    Invalid End IP
                  </mat-error>     
                </mat-form-field>  

                </div>


                <div fxFlex="30">
                  <div *ngIf="isFirst;then addfield else delete">
                  </div>
                  <ng-template #addfield>
                    <button mat-icon-button class="text-center" (click)="addItem()">
                      <mat-icon class="font-weight-900">add</mat-icon>
                    </button>
                  </ng-template>
                  <ng-template #delete>
                    <button mat-icon-button class="text-center" (click)="deletefield(i)">
                      <mat-icon class="font-weight-900 red-500-fg">delete</mat-icon>
                    </button>
                  </ng-template>
                </div>
                <ng-template #addfield>
                  <button mat-icon-button class="text-center" (click)="addItem()">
                    <mat-icon class="font-weight-900">add</mat-icon>
                  </button>
                </ng-template>
                <ng-template #delete>
                  <button mat-icon-button class="text-center" (click)="deletefield(i)">
                    <mat-icon class="font-weight-900 red-500-fg">delete</mat-icon>
                  </button>
                </ng-template>
              </div>
            </ng-container>
            </div>
          </div>
        </div>
      <ng-container formGroupName="login">
        <div *ngSwitchCase="'login'">
          <div fxLayout="row wrap">
            <div fxFlex="50" fxFlexOffset="25" class="px-48">
              <div class="text-center">
                <h2 class="font-size-26 text-bold">Hello</h2>
                <p class="font-size-14">Please login to your account</p>
              </div>
              <div>
                <mat-form-field appearance="outline" class="w-100-p">
                    <mat-label>email</mat-label>
                  <input matInput formControlName="login_email" [errorStateMatcher]="matcher">
                  <mat-icon matSuffix class="grey-700-fg">email</mat-icon>
                </mat-form-field>

                <mat-form-field appearance="outline" class="w-100-p">
                    <mat-label>Password</mat-label>
                  <input matInput formControlName="login_password" [type]="hide ? 'password' : 'text'">
                  <mat-icon matSuffix (click)="hide = !hide" class="grey-700-fg cursor-pointer">{{hide ? 'visibility_off'
                    :'visibility'}}</mat-icon>
                </mat-form-field>
                <div class="text-right cursor-pointer">
                  <a>forgot password? </a>
                </div>
                <div class="text-center mt-12">
                  <button mat-flat-button color="primary"
                    class="px-60 py-4 font-size-14 text-bold btn-rounded">Login</button>

                </div>

              </div>


              <div class="mt-60 text-center">
                <button mat-flat-button class="font-size-14 text-bold grey-400-bg mr-20 px-12"
                  (click)="urlDialog()">URL</button>
                <button mat-flat-button class="font-size-14 text-bold grey-400-bg" (click)="embedDialog()">Embed
                  Code</button>
              </div>
            </div>
          </div>
        </div>
      </ng-container>

    </form>

component.ts:

ip_addresses_arr: any;

this.companyForm = this.fb.group({
id: [],
company_details: this.fb.group({
  name: [],
  display_name: []
}),
company_settings: this.fb.group({
  id: [],
  website: [],
  fiscal_start: [],
  contact_number: [],
  fax: [],
  description: [],
  company_logo: [],
  social_media_logo_path: [],
  logo_path: [],
  currency: [],
  date_format: [],
  timezone: [],
  timeformat: [],
  resume_inbox_non_user: [],
  resume_inbox_contact_owner: [''],
  logo: [''],
  original_logo: [''],
  social_media_logo: ['']
}),
billing: this.fb.group({
  id: [],
  street: [],
  city: [],
  country: [],
  state: [],
  zip: []
}),
company_ip_addresses: this.fb.array([this.fb.group({
  id: 0,
  created_by: 0,
  modified_by: 0,
  start_ip: new FormControl('', Validators.pattern("^([0-9]{1,3})[.]([0-9]{1,3})[.]([0-9]{1,3})[.]([0-9]{1,3})$")),
  end_ip: new FormControl('', Validators.pattern("^([0-9]{1,3})[.]([0-9]{1,3})[.]([0-9]{1,3})[.]([0-9]{1,3})$"))
})]),
  applicable_to: this.buildUserTypes(),
    login: this.fb.group({
    login_email: [''],
    login_password: ['']
  }),
});

My form validation on submitting the form is:

saveCompanyInfo() {
  this.ip_addresses_arr = this.companyForm.get('company_ip_addresses') as FormArray;
 this.ip_addresses_arr.controls.forEach(eachfield => {
  console.log("start_ip value - ",eachfield.value['start_ip']);
  console.log("end_ip value - ",eachfield.value['end_ip']);
  if( eachfield.value['start_ip'] != "" && (eachfield.value['end_ip'] == null || eachfield.value['end_ip'] == "" ) ){
    console.log("coming inside end_ip is empty");
    eachfield.get('end_ip').setValidators([Validators.required]);
    eachfield.get('end_ip').updateValueAndValidity();
    eachfield.get('end_ip').markAsTouched();
    console.log("eachfield.get('end_ip') after setErrors - ",eachfield.get('end_ip'));
  }else if( eachfield.value['end_ip'] != "" && (eachfield.value['start_ip'] == null || eachfield.value['start_ip'] == "" ) ){
    console.log("coming inside start_ip is empty");
    eachfield.get('start_ip').setValidators([Validators.required]);
    eachfield.get('start_ip').updateValueAndValidity();
    eachfield.get('start_ip').markAsTouched();
    console.log("eachfield.get('start_ip') after setErrors - ",eachfield.get('start_ip'));
  }else{
    console.log("coming to else");
    eachfield.get('start_ip').clearValidators();
    eachfield.get('end_ip').clearValidators();
  }
 });

 console.log("this.companyForm.valid - ",this.companyForm.valid);

  if (this.companyForm.valid) {
    -----------------
    -----------------
  }
}

My browser console is:

eachfield.get('start_ip') after setErrors
FormControl {...}
 asyncValidator: null
 dirty: (...)
 disabled: (...)
 enabled: (...)
 errors: {required: true}
 --------

 this.companyForm.valid -  true

As you can see above my formcontrol set to "required: true" and also the my form.valid condition is setting to 'true' and the form is getting submitted automatically after clicking on submit button. Any help would be appreciated. Thanks.