Redux thật là đơn giản (Phần 2)

September 22, 2017 (6y ago)

Redux với React Native

Bài trước Nguyên Lý của Redux các bạn đã nắm được cơ bản phần lý thuyết về Redux và dùng cho việc gì, sau đây chúng ta sẽ đi vào áp dụng nó với React hoặc React Native, ở đây mình chọn React Native vì nó vẫn đang còn khá HOT :D (Thực ra React hay React Native thì đều dùng Redux là y hệt nhau)

Đầu tiên các bạn vào link:https://facebook.github.io/react-native/docs/getting-started.html để làm theo và tạo 1 project với React Native.

Lưu ý: Máy phải được cài sẵn yarn (https://yarnpkg.com/en/) hoặc dùng npm sau khi đã cài node, vì nếu cài yarn thì nó sẽ đồng thời cài node luôn cho bạn.

Phần 1: Tạo giao diện màn hình với React Native

Sau khi install xong, tạo thêm thư mục src -> file Main.js như sau:

Ở đây mình sẽ minh họa tạo chức năng hết sức đơn giản là Bấm — Tăng — Giảm 1 số dùng React Native + Redux có màn hình hiển thị lên như sau:

Main.js

Trong thư mục components ta tạo thêm 1 component có tên Button.js, thực tế bạn không cần phải viết hẳn 1 component chỉ render mỗi cái Button như này, do mình muốn tách nhỏ component nên tạm lấy ví dụ viết 1 component con riêng ra. (trong component này mình có dùng react-native-elements cài đặt nó qua yarn bằng lệnh sau:yarn add react-native-elements

Component để hiển thị text number:

Okay tới đây coi như xong giao diện, như vậy bạn đã có 1 app counter khá đơn giản :D và hoàn toàn có thể làm dùng React Native đơn giản hơn nhiều, nhưng để hiểu về Redux thì chúng ta nên bắt đầu từ việc đơn giản đã.

Phần 2: Cài đặt Redux

Step1: install Redux, React-Redux dependencies

yarn add redux react-redux

Step 2: Tạo cấu trúc thư mục project như hình:

3 nhân tố của Redux là actions, reducersstore tương ứng với 3 thư mục trong project.

actions:

reducers:

Store:

tạo 1 file index.js trong thư mục store. (Mình có config thêm redux-devtools-extension để tiện debug và xem store khi redux hoạt động trên extention của chrome).

Step 3: connect React component với store của Redux

Việc cần làm là làm sao để kết nối toàn bộ components của app với store của Redux.

Thư viện react-redux đã cung cấp 1 thằng có tên Provider để làm cầu nối cho React và Redux, chúng ta chỉ việc bọc nó bao ngoài root component của React và truyền 1 tham số duy nhất là store vào (store đã được tạo ở thư mục store và được import vào file này).

Step 4: Gọi 1 action từ 1 component và map data từ store ra View

Quay lại bài trước như đã biết thì flow của redux:

action → reducer → store → View

3 bước đã thực hiện xong ở trên, việc còn lại là store và View tương tác như nào, câu hỏi đặt ra là :

Nhiệm vụ quan trọng này được thực hiện bởi hàm connect() trong react-redux.

Hàm connect() có 2 tham số:

Để đơn giản và ngắn gọn hơn ta sẽ bỏ đi hàm mapDispatchToProps(dispatch) và thay bằng việc truyền trực tiếp actions vào hàm connect() cuối cùng sẽ là:

connnect(mapStateToProps, actions)(COMPONENT)

Quay trở lại code React Native ở trên, ở phần View ta có:

main.js

Giải thích:

Tiếp theo là hiển thị ra View ở component child.js

Ở file này ta vừa gán thuộc tính counter : state.counter lúc này counter cũng trở thành 1 props của component Child → gọi show ra View ta chỉ cần gọi như thông thường <Text>{this.props.counter}</Text>

Như vậy là 1 app counter đơn giản bằng React Native và Redux đã thực hiện xong tóm lại flow như sau:

1. View gồm 2 button IncreaseDecrease và 1 component hiển thị number

2. Khi Button được click → dispatch() tới 1 action creator có tên counterIncrease()

3. counterIncrease() sẽ tạo ra 1 Object (Trong redux action phải là 1 plain object có thuộc tính là type và payload, type là bắt buộc) ở đây chỉ có 1 thuộc tính {type:"INCREASE"} sau đó nó truyền tới counterReducer() để xử lý.

4. counterReducer(state, action) => kiểm tra xem action kiểu type = "increase" trả ra 1 state mới : state + 1` (state của redux là immutable)

5 ở component View(Child) hiển thị number ta sẽ dùng hàm mapStateToProps(state) đẻ nhận state là counter rồi update vào View.

→ App đã chạy ngon lành :D

Giả sử bây giờ yêu cầu bài toán có thay đổi chút như sau:

Bình thường ở bài toán trên ta bấm nút thì ngay lập tức action đáp trả kết quả, nhưng trong thực tế có nhiều bài toán ko lập tức có thay đổi luôn điển hình như call api tới server để fetch data, thì phải mất 1 lúc kết quả mới được trả về. Hay như bài toán bấm nút thì sau 1s mới thay đổi → thì những điều này người ta gọi đó là side-effect .

Như đã biết thì Redux yêu cầu 1 object được trả ra phải là Plain object: {type: ACTION_TYPE, payload:params} và reducer phải là pure function,hay nói cách khác là hoàn toàn ko có side-effect.

Okay có lẽ bài thứ 2 về redux đến đây là dài rồi :))

Mình sẽ giải quyết vấn đề side-effect ở bài sau dùng Redux-middleware

Hẹn gặp lại các bạn về bài cuối của Redux.

Source code ví dụ trên: check out branch lesson_2

https://github.com/tridungbk2010/react-native-class