通常,当我在响应式表单中进行复杂的验证时,我会为那些相互依赖的控件定义一个 formGroup。
这在上面的 senario 中是不可能的,我们有3 steps = 3 groups
和依赖的 3 个字段firstUnique, secondUnique, thirdUnique
。
<form [formGroup]="myForm">
<mat-horizontal-stepper formArrayName="formArray" #stepper>
<mat-step formGroupName="0" [stepControl]="formArray?.get([0])" errorMessage="Name is required.">
<ng-template matStepLabel>Fill out your name</ng-template>
<mat-form-field>
<input matInput placeholder="Last name, First name" formControlName="firstCtrl" required>
</mat-form-field>
<mat-form-field>
<input matInput placeholder="UNIQUE1" formControlName="firstUnique" required>
</mat-form-field>
<div>
<button mat-button matStepperNext>Next</button>
</div>
</mat-step>
<mat-step formGroupName="1" [stepControl]="formArray?.get([1])" errorMessage="Address is required.">
<ng-template matStepLabel>Fill out your address</ng-template>
<mat-form-field>
<input matInput placeholder="Address" formControlName="secondCtrl" required>
</mat-form-field>
<mat-form-field>
<input matInput placeholder="UNIQUE2" formControlName="secondUnique" required>
</mat-form-field>
<div>
<button mat-button matStepperPrevious>Back</button>
<button mat-button matStepperNext>Next</button>
</div>
</mat-step>
<mat-step formGroupName="2" [stepControl]="formArray?.get([2])" errorMessage="Error!">
<ng-template matStepLabel>Done</ng-template>
You are now done.
<div>
<mat-form-field>
<input matInput placeholder="UNIQUE3" formControlName="thirdUnique" required>
</mat-form-field>
<button mat-button matStepperPrevious>Back</button>
<button mat-button (click)="stepper.reset()">Reset</button>
</div>
</mat-step>
</mat-horizontal-stepper>
我使用描述的技术SO_answer和Material_docs
到目前为止,我的解决方案正在运行,但我对此并不满意:
- 启动
Unique Validation
时运行一千次(30-40 次)(hacky) - 在整个步进器中任何输入的每次更改都会触发。(这是因为我必须将它添加到整个 formGroup)。
Unique Validation
一个简单的任务,因为这 3 个输入字段需要是唯一的,这已经成为一个样板和复杂的混乱。(请注意
function Unique(arr: string[])
)当正确的步骤被
UNIQUE Validator
或 STEP 再次有效时,不会调用 STEPPER-VALIDATION。(例如:firstUnique = “a”,secondUnique “b”,thirdUnique = “a”(再次))
我的表格
this.myForm = this._formBuilder.group({
formArray:
this._formBuilder.array([
this._formBuilder.group({
firstCtrl: [''],
firstUnique: [''],
}),
this._formBuilder.group({
secondCtrl: [''],
secondUnique: [''],
}),
this._formBuilder.group({
thirdUnique: [''],
})
])
}, {
validator: [Unique(['0;firstUnique', '1;secondUnique', '2;thirdUnique'])]
});
独特的验证者乐趣
function Unique(arr: string[]) {
const validKey = "uniqueValid";
return (formGroup: FormGroup) => {
const myValues =
arr.map(path => {
const s = path.split(';');
return (<FormArray>formGroup.get('formArray'))
.controls[parseInt(s[0])]
.controls[s[1]];
});
const myKeys = arr.map(path => path.split(';')[1] )
const obj = {};
myKeys.forEach(function (k, i) {
obj[k] = myValues[i];
})
myKeys.forEach((item, index) => {
debugger
console.log('unique validation function runs')
const control = obj[item];
const tmp = myKeys.slice();
tmp.splice(index,1);
const ans = tmp
.filter( el => obj[item].value === obj[el].value)
if ( ans.length && control.value ) {
const err = {}
err[validKey] = `identicial to: ${ans.join(', ')}`
control.setErrors(err);
} else if ( obj[item].errors && !obj[item].errors[validKey] ) {
return;
} else {
control.setErrors(null);
}
})
}