Điều kiện bắt buộc
Vì dụ hôm nay sẽ hơi khác ví dụ từ các bài trước, tuy vẫn có đôi chút quen thuộc.
Mình sẽ build một form đăng kí user - registerForm
bao gồm:
- Username textbox
- Không được bỏ trống, có độ dài từ 6 đến 32 ký tự, chỉ chứa ký tự alphabet.
- Giả sử như có username: trungvo, tieppt, chautran đã tồn tại trong hệ thống. Khi người dùng nhập một trong ba user kể trên thì sẽ ko cho phép đăng kí.
- Password textbox
- Không được bỏ trống, có độ dài từ 6 đến 32 ký tự, chỉ chưa các ký tự alphabet, digit, và phải chứa ít nhất một ký tự đặc biệt trong list:
!@#\$%^&\*
- Retype password
- Cùng yêu cầu như password kể trên.
- Để đảm bảo chắc chắn user nhập đúng password. Giá trị của textbox này phải giống hệt giá trị textbox password ở trên.
Mình sẽ setup form như ở dưới nhé.
- Tạo register component dùng CLI
ng g c register
- Update route để chỉ đến component này
diff
const routes: Routes = [
{
path: "sign-in",
component: SignInComponent
},
{
path: "sign-in-rf",
component: SignInRfComponent
},
+ {
+ path: "register",
+ component: RegisterComponent
+ },
{
path: "",
redirectTo: "register",
pathMatch: "full"
}
];
- Setup registerForm
ts
const PASSWORD_PATTERN = /^(?=.*[!@#$%^&*]+)[a-z0-9!@#$%^&*]{6,32}$/;
this.registerForm = this._fb.group({
username: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(/^[a-z]{6,32}$/i),
]),
],
password: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(PASSWORD_PATTERN),
]),
],
confirmPassword: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(PASSWORD_PATTERN),
]),
],
});
Phần HTML của form sẽ có dạng như sau.
html
<div class="container">
<form
class="register-form"
[formGroup]="registerForm"
autocomplete="off"
(ngSubmit)="submitForm()"
>
<h2>Register</h2>
<div class="row-control">
<mat-form-field appearance="outline">
<mat-label>Username</mat-label>
<input matInput placeholder="Username" formControlName="username" />
</mat-form-field>
<pre>{{ registerForm.get("username")?.errors | json }}</pre>
</div>
<div class="row-control">
<mat-form-field appearance="outline">
<mat-label>Password</mat-label>
<input
type="password"
matInput
placeholder="Password"
formControlName="password"
/>
</mat-form-field>
<pre>{{ registerForm.get("password")?.errors | json }}</pre>
</div>
<div class="row-control">
<mat-form-field appearance="outline">
<mat-label>Confirm Password</mat-label>
<input
type="password"
matInput
placeholder="Confirm Password"
formControlName="confirmPassword"
/>
</mat-form-field>
<pre>{{ registerForm.get("confirmPassword")?.errors | json }}</pre>
</div>
<div class="row-control row-actions">
<button
mat-raised-button
color="primary"
type="submit"
[disabled]="registerForm.invalid"
>
Register
</button>
</div>
<pre>{{ registerForm.value | json }}</pre>
</form>
</div>
Custom Validators
Dựa vào requirements, ta cần viết 2 custom validator:
- Async validator để gọi API check xem username đã tồn tại trong hệ thống hay chưa
- Sync validator để check xem password type lần hai có trùng khớp với password đầu tiên hay ko.
1. Async Validator để validate username
Nhắc lại một chút về Async Validator. Đây là các validate function sẽ trả về Promise hoặc Observable. Ví dụ như bạn muốn validate xem username nhập vào đã có trong hệ thống hay chưa. Thông thường bắt buộc bạn phải gửi một yêu cầu lên server để làm việc này, HTTP request thường sẽ trả về Promise/Observable.
Vì ko có API nên mình sẽ mock một hàm để check username và sẽ trả về false
nếu như input là một trong 3 giá trị: trungvo, tieppt, chautran
, nếu ko sẽ trả lại true. Mỗi khi validateUsername
dc gọi mình cũng sẽ in ra console một dòng text Trigger API call
để tiện cho việc demo ở dưới.
ts
validateUsername(username: string): Observable<boolean> {
console.log("Trigger API call");
let existedUsers = ["trungvo", "tieppt", "chautran"];
let isValid = existedUsers.every(x => x !== username);
return of(isValid).pipe(delay(1000));
}
Giờ mình sẽ tiến hành viết custom async validator để validate username trùng khớp. Để viết đc một async validator, các bạn có hai lựa chọn:
- Viết một function, nhận vào là một
AbstractControl
và output trả về dạngPromise<ValidationErrors | null> | Observable<ValidationErrors | null>
. Cả function sẽ có signature dạngvalidate(control: AbstractControl): Promise<ValidationErrors | null> | Observable<ValidationErrors | null>
. - Implement interface [AsyncValidator][asyncinterface], trong đó có định nghĩa sẵn là bạn phải implement function
validate
.
validateUserNameFromAPI
Mình sẽ làm theo cách thứ nhất. Nếu làm theo cách thứ hai phải truyền thêm một số thông tin như API service vào contructor khi khởi tạo async validator.
- Nếu API trả về là
true
, thì hàmvalidateUserNameFromAPI
sẽ trả về null, tức là input này hoàn toàn ko có lỗi gì. - Còn nếu API trả về
false
, thì hàmvalidateUserNameFromAPI
sẽ trả về một object với data là bất cứ gì bạn muốn, nhưng thông tin nên có giá trị một chút để sau này còn dùng hiển thị thông báo lỗi cho user chẳng hạn
ts
const validateUserNameFromApi = (api: ApiService) => {
return (control: AbstractControl): Observable<ValidationErrors | null> => {
return api.validateUsername(control.value).pipe(
map((isValid: boolean) => {
return isValid ? null : { usernameDuplicated: true };
})
);
};
};
Sau khi viết xong function, chúng ta cần config control để sử dụng validator đó.
ts
this.registerForm = this._fb.group({
username: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(/^[a-z]{6,32}$/i),
]),
validateUserNameFromApi(this._api),
],
});
Kết quả như hình dưới. Khi điền đủ 6 kí tự alpha, khi đó username đã pass toàn bộ sync validator thì async validator sẽ đc trigger ngay sau khi bạn điền kí tự thứ 6. Sau đó, mỗi kí tự khi được nhập từ bàn phím vào sẽ trigger một API lên server để validate, như ta đã thấy có console.log "Trigger API call".
validateUserNameFromAPIDebounce
Bạn có thấy screenshot có điểm nào quen quen ko? Use case khi điền vào searchbox ko trigger API call ngay lập tức mà chỉ call API nếu như giữa hai keystroke cách nhau một khoảng thời gian, thường là 300ms
. Chúng ta cũng có thể implement behavior đó tương tự như dùng async validator. Sửa đoạn code ở trên có dùng timer như ở dưới:
ts
const validateUserNameFromApiDebounce = (api: ApiService) => {
return (control: AbstractControl): Observable<ValidationErrors | null> => {
return timer(300).pipe(
switchMap(() =>
api.validateUsername(control.value).pipe(
map((isValid) => {
if (isValid) {
return null;
}
return {
usernameDuplicated: true,
};
})
)
)
);
};
};
Sau đó config control để dùng validateUserNameFromAPIDebounce
.
ts
this.registerForm = this._fb.group({
username: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(/^[a-z]{6,32}$/i),
]),
validateUserNameFromAPIDebounce(this._api),
],
});
Kết quả đây, sau khi đã type chautr
đủ 6 kí tự. Mình sẽ type ba lăng nhăng và sau khi dừng ko type nữa thì API sẽ được trigger sau 300ms, chứ ko phải sau mỗi keystroke.
Chú ý!
Angular doesn't wait for async validators to complete before firing ngSubmit. So the form may be invalid if the validators have not resolved.
Ở form Register, sau khi mình đã điền đúng hai password và giờ mình mới điền username. Trong khoảng thời gian delay 1 giây khi async validator đang đợi kết quả kiểm trả từ API thì nút Register đã được enable trở lại. Nếu user nhanh tay bấm Register thì ngSubmit vẫn dc trigger và bạn vẫn sẽ thực hiện phần phần code cho submit form. Sau 1 giây sau khi đã validate xong, nếu username bị trùng lặp thì nút Register mới lại đc vô hiệu hóa.
Để chứng minh thì mình sẽ add thêm hàm onSubmit
:
ts
submitForm() {
console.log("Submit form leh");
}
Có thể thấy là mình vẫn bấm được nút Register trong khi đang validate username 😂 Để fix lỗi này thì mình có tham khảo [một câu trả lời trên stackoveflow][stack].
Ý tưởng là thay vì ngSubmit
sẽ trigger thẳng hàm submit
, mình sẽ tạo ra một Subject
tên là formSubmit$
và handle chỉ khi nào status của form chuyển thành VALID
thì formSubmit$
mới emit một value, từ đó mới call hàm submitForm
.
ts
this.formSubmit$
.pipe(
tap(() => this.registerForm.markAsDirty()),
switchMap(() =>
this.registerForm.statusChanges.pipe(
startWith(this.registerForm.status),
filter((status) => status !== "PENDING"),
take(1)
)
),
filter((status) => status === "VALID"),
tap(() => {
this.submitForm();
})
)
.subscribe();
html
<form
class="register-form"
[formGroup]="registerForm"
autocomplete="off"
(ngSubmit)="formSubmit$.next()"
></form>
Test thôi anh em. Như trong hình thì trong khoảng thời gian đang validate mà nút Register đc enable thì có bấm nút cũng sẽ ko trigger hàm submitForm
và ko có console hiện ra.
2. Bonus: Validate confirm password
Use case để validate confirm password trùng với password thì chúng ta chỉ cần viết một hàm custom validator đơn giản hơn, nhưng hàm này vì cần value của 2 controls nên mình sẽ apply validator này cho formGroup
nhé. Code của function validateControlsValue
sẽ như sau:
ts
const validateMatchedControlsValue = (
firstControlName: string,
secondControlName: string
) => {
return function (formGroup: FormGroup): ValidationErrors | null {
const { value: firstControlValue } = formGroup.get(
firstControlName
) as AbstractControl;
const { value: secondControlValue } = formGroup.get(
secondControlName
) as AbstractControl;
return firstControlValue === secondControlValue
? null
: {
valueNotMatch: {
firstControlValue,
secondControlValue,
},
};
};
};
- Mình tạo ra hàm
validateControlsValue
và truyền vào tên của 2 controls. Function này sẽ return lại một function làm nhiệm vụ validate. - Function validate sẽ nhận vào một formGroup và get value từ hai control.
- Nếu hai control này có giá trị giống nhau thì return null, tức là ko có lỗi. Nếu ko sẽ return một object thông báo lỗi để dựa vào đó ta có thể hiển thị lên UI.
Sau đó mình apply validator này vào form group.
ts
this.registerForm = this._fb.group(
{
password: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(PASSWORD_PATTERN)
])
],
confirmPassword: [
"",
Validators.compose([
Validators.required,
Validators.minLength(6),
Validators.pattern(PASSWORD_PATTERN)
])
]
},
{
validators: validateControlsValue("password", "confirmPassword")
}
);
})
Kết quả đây anh em ei. Nếu hai password ko giống nhau thì form sẽ có errors. Phần UI hiển thị đẹp đẽ như nào thì để tùy anh em 😍
Lời kết
Trong bài này chúng ta đã tìm hiểu về async validator với reactive form. Anh em chú ý mấy điểm này:
- Muốn viết async validator thì theo cú pháp
validate(control: AbstractControl): Promise<ValidationErrors | null> | Observable<ValidationErrors | null>
- Angular sẽ ko chờ async validator hoàn thành rồi mới submit form nên phải thật cẩn thận trong một số trường hợp.