Skip to content

Thay đổi giao diện với State và useState Hook (phần 3)

Thực hành tạo, mở, đóng Modal sử dụng useState, State Lifting, Conditional Rendering, Children Props.
vị trí đặt modalIsVisible và state lifting
sử dụng childern để chèn nội component khác vào modal
condition rendering để mở và đóng modal
conditional rendering để thay đổi trạng thái nút
Bài này chốt lại toàn bộ những gì đã học về useState bằng một ví dụ thực tế: Modal — component xuất hiện ở khắp mọi nơi trong production, và là bài tập tổng hợp hoàn hảo cho state lifting, conditional rendering, và children props.
Mục tiêu cuối bài: click “New Post” → modal mở ra chứa form → submit hoặc click backdrop/nút X → modal đóng lại.

State đặt ở đâu?

Câu hỏi đầu tiên cần trả lời trước khi viết code: isModalOpen đặt trong component nào?
Muốn biết, phải hiểu rõ cấu trúc của các component.

Cấu trúc component

Nút “New Post” nằm trong PostList.
Nút X và backdrop nằm trong Modal.
NewPost chỉ là form thuần — không biết gì về modal.
Áp dụng quy tắc từ bài trước: state đặt ở component thấp nhất có thể, nhưng phải đủ cao để tất cả component cần dùng đều có thể nhận qua props.
Những component nào cần biết modal đang mở hay đóng?
PostList — cần biết để render nút “New Post” hay “Đang mở…”
Modal — cần biết để render hay không render
NewPost — không cần biết, chỉ lo render form
PostList là cha chung của cả Modal lẫn nút “New Post” — nên isModalOpen đặt ở đây.

Children Props và render cái gì bên trong Modal

Trước khi viết Modal, cần hiểu children props — một loại props đặc biệt trong React.
Thông thường bạn truyền data qua props có tên cụ thể: title, author, onSubmit… Nhưng Modal không biết trước bên trong nó sẽ chứa gì — lúc thì NewPost, lúc thì form đăng nhập, lúc thì hộp thoại xác nhận xóa.
children giải quyết điều đó: bất cứ thứ gì bạn đặt giữa thẻ mở và thẻ đóng của component sẽ được truyền vào dưới dạng props.children.
Modal không cần biết nội dung bên trong là gì — nó chỉ lo phần backdrop, khung trắng, và nút đóng. Nội dung là việc của component cha quyết định.

Viết Modal component

onClose được dùng ở hai chỗ: backdrop và nút X — cả hai đều chỉ gọi một hàm duy nhất từ cha truyền xuống, không tự quản lý state.

Conditional Rendering cho Modal đóng mở

isModalOpen là boolean — dùng nó để quyết định render Modal hay không. Có 3 cách phổ biến:

Cách 1 — Tách ra biến trước khi return

Dễ đọc nhất khi nội dung phức tạp:

Cách 2 — Short-circuit &&

Gọn, phù hợp khi chỉ có một nhánh (true thì render, false thì không):

Cách 3 — Ternary operator

Dùng khi cần render khác nhau ở cả hai trạng thái:
Trong thực tế, cách 2 phổ biến nhất cho modal vì đơn giản. Cách 1 hữu ích khi logic phức tạp hơn.

Conditional Rendering cho nút

Nút “New Post” cũng thay đổi trạng thái tùy theo modal đang mở hay đóng — dùng ternary để đổi text và disabled:

PostList hoàn chỉnh

Ghép tất cả lại:

Tóm tắt

Bài này kết hợp đủ các khái niệm đã học:
useState — quản lý isModalOpen, form state, và list state.
State Lifting — tất cả state và handler đặt trong PostList, truyền xuống ModalNewPost qua props.
Children PropsModal nhận nội dung linh hoạt qua props.children, không cần biết bên trong là gì.
Conditional Rendering — 3 cách: biến modalContent, &&, ternary — tùy độ phức tạp mà chọn.
Bước tiếp theo: khi ứng dụng lớn hơn và nhiều component cùng cần truy cập một state (ví dụ: trạng thái đăng nhập, theme sáng/tối), truyền props qua nhiều tầng trở nên rườm rà — đó là lúc cần đến Context API.
Want to print your doc?
This is not the way.
Try clicking the ··· in the right corner or using a keyboard shortcut (
CtrlP
) instead.