angular-forms
Compare original and translation side by side
🇺🇸
Original
English🇨🇳
Translation
Chinese@angular/forms
@angular/forms
Version: Angular 21 (2025)
Tags: Forms, Validation, Reactive Forms, Template-Driven
API Changes
API变更
This section documents recent version-specific API changes.
-
NEW: Signal-based form controls — Angular 19+ signal integration for form controls
-
NEW: Functional validators — Prefer functional validators over class-based validators
-
NEW:— New method to mark all controls as touched
AbstractControl.markAllAsTouched() -
NEW:option — Create non-nullable FormControls
nonNullable -
NEW: Typed forms — Full TypeScript support for form groups and arrays
-
DEPRECATED: Class-based validators — Migrate to functional validators
本节记录了近期对应版本的API改动。
-
新增:基于Signal的表单控件 —— Angular 19+ 版本为表单控件集成了Signal支持
-
新增:函数式验证器 —— 优先使用函数式验证器而非基于类的验证器
-
新增:—— 用于将所有控件标记为已触碰的新方法
AbstractControl.markAllAsTouched() -
新增:选项 —— 用于创建非空的FormControls
nonNullable -
新增:类型化表单 —— 为表单组和表单数组提供完整的TypeScript支持
-
已弃用:基于类的验证器 —— 请迁移至函数式验证器
Best Practices
最佳实践
- Use Reactive Forms for complex forms — Better structure, validation, and testability
ts
import { FormBuilder, Validators, ReactiveFormsModule } from '@angular/forms';
@Component({
standalone: true,
imports: [ReactiveFormsModule],
template: `
<form [formGroup]="form" (ngSubmit)="onSubmit()">
<input formControlName="email" />
<div *ngIf="form.controls.email.invalid && form.controls.email.touched">
Email is required
</div>
</form>
`
})
export class LoginComponent {
private fb = inject(FormBuilder);
form = this.fb.nonNullable.group({
email: ['', [Validators.required, Validators.email]],
password: ['', [Validators.required, Validators.minLength(8)]]
});
onSubmit() {
if (this.form.valid) {
console.log(this.form.getRawValue());
}
}
}- Use FormBuilder for cleaner code
ts
form = this.fb.group({
name: ['', Validators.required],
email: ['', [Validators.required, Validators.email]],
address: this.fb.group({
street: [''],
city: ['']
})
});- Use custom validators for complex logic
ts
// Functional validator (preferred)
const forbiddenNameValidator = (control: AbstractControl): ValidationErrors | null => {
const forbidden = ['admin', 'root', 'superuser'];
return forbidden.includes(control.value) ? { forbiddenName: true } : null;
};
// Usage
name: ['', forbiddenNameValidator]- Use async validators for server-side validation
ts
email: ['',
[Validators.email],
[asyncValidator, { debounceTime: 300 }]
]- Use FormArray for dynamic form fields
ts
const form = this.fb.group({
emails: this.fb.array([
this.fb.control('')
])
});
get emails(): FormArray {
return this.form.get('emails') as FormArray;
}
addEmail() {
this.emails.push(this.fb.control(''));
}- Use option for performance
updateOn
ts
// Only validate on blur
email: ['', { updateOn: 'blur' }]
// Only validate on submit
password: ['', { updateOn: 'submit' }]- Show validation errors conditionally
ts
<input formControlName="email"
[class.is-invalid]="email.invalid && (email.dirty || email.touched)" />- Use disable() for conditionally disabled fields
ts
// Disable based on another control
this.form.statusChanges.subscribe(() => {
if (this.form.value.isAdmin) {
this.form.controls.permissions.enable();
} else {
this.form.controls.permissions.disable();
}
});- Reset form properly
ts
// Reset with default values
this.form.reset({ email: '', name: 'Default' });
// Reset and clear validators
this.form.reset();
this.form.clearValidators();- 复杂表单使用响应式表单 —— 结构更清晰、验证更方便、可测试性更强
ts
import { FormBuilder, Validators, ReactiveFormsModule } from '@angular/forms';
@Component({
standalone: true,
imports: [ReactiveFormsModule],
template: `
<form [formGroup]="form" (ngSubmit)="onSubmit()">
<input formControlName="email" />
<div *ngIf="form.controls.email.invalid && form.controls.email.touched">
Email is required
</div>
</form>
`
})
export class LoginComponent {
private fb = inject(FormBuilder);
form = this.fb.nonNullable.group({
email: ['', [Validators.required, Validators.email]],
password: ['', [Validators.required, Validators.minLength(8)]]
});
onSubmit() {
if (this.form.valid) {
console.log(this.form.getRawValue());
}
}
}- 使用FormBuilder简化代码
ts
form = this.fb.group({
name: ['', Validators.required],
email: ['', [Validators.required, Validators.email]],
address: this.fb.group({
street: [''],
city: ['']
})
});- 复杂逻辑使用自定义验证器
ts
// 函数式验证器(推荐使用)
const forbiddenNameValidator = (control: AbstractControl): ValidationErrors | null => {
const forbidden = ['admin', 'root', 'superuser'];
return forbidden.includes(control.value) ? { forbiddenName: true } : null;
};
// 使用示例
name: ['', forbiddenNameValidator]- 服务端验证使用异步验证器
ts
email: ['',
[Validators.email],
[asyncValidator, { debounceTime: 300 }]
]- 动态表单字段使用FormArray
ts
const form = this.fb.group({
emails: this.fb.array([
this.fb.control('')
])
});
get emails(): FormArray {
return this.form.get('emails') as FormArray;
}
addEmail() {
this.emails.push(this.fb.control(''));
}- 使用选项优化性能
updateOn
ts
// 仅在失焦时验证
email: ['', { updateOn: 'blur' }]
// 仅在提交时验证
password: ['', { updateOn: 'submit' }]- 条件性展示验证错误
ts
<input formControlName="email"
[class.is-invalid]="email.invalid && (email.dirty || email.touched)" />- 使用disable()实现字段的条件禁用
ts
// 根据其他控件的状态切换禁用状态
this.form.statusChanges.subscribe(() => {
if (this.form.value.isAdmin) {
this.form.controls.permissions.enable();
} else {
this.form.controls.permissions.disable();
}
});- 正确重置表单
ts
// 重置为默认值
this.form.reset({ email: '', name: 'Default' });
// 重置并清空验证规则
this.form.reset();
this.form.clearValidators();