我为自定义表单验证器创建了可注射服务:
import { Injectable } from '@angular/core'
import { FormControl } from '@angular/forms'
import { Observable, of, timer } from 'rxjs'
import { map, switchMap } from 'rxjs/operators'
@Injectable({
providedIn: 'root'
})
export class IsValidNicknameService {
validate = (time: number = 500) => {
return (input: FormControl) => {
return timer(time).pipe(
switchMap(() => this.isValidNickname(input.value)),
map(isValid => {
return isValid ? null : { shouldNotStartWithA: true }
})
)
}
}
private isValidNickname(value: string): Observable<boolean> {
return this.checkIfFirstLetterIsA(value)
}
private checkIfFirstLetterIsA(value: string): Observable<boolean> {
const firstCharacter = value.toLowerCase().charAt(0)
if (firstCharacter === 'a' || firstCharacter === 'à' || firstCharacter === 'ä' || firstCharacter === 'á' || firstCharacter === 'ã') {
return of(false)
} else {
return of(true)
}
}
}
然后我在控制器中这样称呼它:
import { IsValidNicknameService } from './src/Core/Services/isvalidnickname.service'
createNicknamesGroup(): any {
return new FormGroup({
buildingRoom: new FormControl(this.mockBuildingRooms[0], Validators.required),
nickname: new FormControl('', Validators.required, this.isValidNicknameService.validate())
})
}
它有效,但是我觉得这不是一个好方法。有没有更好,更简洁的方法来实现这一目标?
答案 0 :(得分:1)
这里没有正确或错误的方法-做任何适合您的事情。
从您的代码开始-您没有任何服务依赖关系,因此您的服务可以轻松转换为一堆函数。使用普通函数的一个优点是可以轻松地用单元测试覆盖它们。
通常来说,如果需要通过Angular的DI向构造函数中注入某些依赖项,则需要使用@Injectable()
类。但这也可以通过将必要的依赖项传递给您的验证器higher order function来避免:
# my-custom.validator.ts
export const setupRequiredValidator(minLength: number, someCheckerService: CheckerService): ValidatorFn => {
return (control: AbstractControl) => {
return control.value.length >= minLength && someCheckerService.check(control);
}
}
# my.component.ts
constructor(private someCheckerService: CheckerService) {
}
createNicknamesGroup(): FormGroup {
return new FormGroup({
buildingRoom: new FormControl(this.mockBuildingRooms[0], Validators.required),
nickname: new FormControl('', [Validators.required, setupRequiredValidator(5, this.someCheckerService)])
}) // an example of synchronous validator
}
您还可以创建@Injectable()
验证程序,不要将其称为Service
-没有人会对您发誓。
# my-custom.validator.ts
@Injectable({
providedIn: 'root'
})
export class MyCustomValidator {
constructor(private router: Router) {}
validateName = (control: AbstractControl) => {
// try to do only one validation per method if there are no dependencies between the validators.
// It's easier to combine and tests them afterwards
return control.value ? null : {error: 'text'};
}
validateEmail = (control: AbstractControl) => {
return control.value ? Validators.email(control) : null;
}
validateSurnameIfTheNameExists = (control: AbstractControl) => {
return control.parent.get('name').valid ? null : this.router.navigate(['/error']);
}
}