在项目中,我们有时候往往需要动表单的验证做动态的规划。比如在一个注册界面中同步注册两种用户,但两种用户的输入项却不是相同的。
教师的话,要求输入工号:
文章图片
学生用户的话,则要求输入学号:
文章图片
我们把这种情景,称为动态的表单验证。
在上述表中校验中,我们要求:
- 工号与学号互不干扰。
- 选择教师类型时,只判断工号是否已经输入。
- 选择学生类型时,则只判断学号是否已经输入。
- 使用跨字段验证器。
- 订阅用户类型,将用户类型发生变化时,重置
工号
或学号
的验证规则。 - 订阅用户类型,将用户类型发生变化时,在
fromGroup
中添加或移除工号
,学号
FromControl。
Anguar的官方给出在在跨字段验证器的使用示例,该思想是在
FromGroup
上添加一个验证器,然后在该验证器中获取FormControl
的值,在根据具体的情况来进行验证。优点:
- 官方示例,学习成本低。
- 直接将验证放到了验证器中,逻辑清晰。
- 验证器不会对获取
FromGroup
的值产生影响。
- 无法在
FormControl
直接定义验证条件,不直观。
FromControl
提供了clearValidators()
来清空验证器,以及setValidators()
来设置验证器,所以我们可以订阅用户类型是否发生变化,在发生变化时,根据情况清空交叉字段的验证器,然后再重新对其验证器进行设置。优点:
- 为动态地添加异步验证器提供了一种新的思路
- 验证规则不直观。
- 代码量大。
FromGroup
提供的removeControl()
使得我们可以移除其中的FormControl
,利用该机制我们可以订阅用户类型发生变化后,根据情况来移除、添加相应的FormControl
,从而达到动态验证表单的目的。示例代码 C 层:
export class AppComponent implements OnInit {
name = 'Angular ' + VERSION.major;
formGroup = new FormGroup({});
// 学号
studentNoFormControl = new FormControl(null, Validators.required);
// 工号
teachterNoFormControl = new FormControl(null, Validators.required);
// 用户类型
typeFormControl = new FormControl(null, Validators.required);
ngOnInit(): void {
this.formGroup.addControl('name', new FormControl('', Validators.required));
this.formGroup.addControl('type', this.typeFormControl);
// 订阅类型的变化,从而决定在formGroup中添加学号还是工号FormControl
this.typeFormControl.valueChanges.subscribe((type) => {
if (type === 0) {
this.formGroup.removeControl('studentNo');
this.formGroup.addControl('teacherNo', this.teachterNoFormControl);
} else {
this.formGroup.removeControl('teacherNo');
this.formGroup.addControl('studentNo', this.studentNoFormControl);
}
});
// 初始化用户类型为教师
this.typeFormControl.setValue(0);
}onSubmit(): void {
alert('submit');
}/**
* 显示学号或是工号的input
*/
showStudent(): boolean {
return this.typeFormControl.value =https://www.it610.com/article/== 1;
}
}
V 层:
表单动态验证示例
{{ formGroup.invalid | json }}
{{ formGroup.get('type').value | json }}
优点:
- 直接在
FormControl
是设置验证器,代码直观。
- 在后续对
FormGroup
获取相关的值的操作中,需要对FormGroup
是否有值来进行判断,容易产生在undefined
上调用value
的错误。(这可以使用直接获取FromControl
的值的方法来规避)。