- vừa được xem lúc

Redux Toolkit & Redux Saga

0 0 32

Người đăng: Prince Norin

Theo Viblo Asia

Getting Start

Over the years the javascript world has evolved greatly, new libraries and frameworks pop up and existing ones have improve a lot and that is especially true to React as well. Among the vast majority of library, redux is a popular one that developer usually used along side when they develop a React application. Many others utility and devtool libraries also being used, which makes setting up a React project a very daunting task. As the pattern grow very common, a community has bundle all those together and thus redux-toolkit was borned.

Redux Toolkit

Install Package

To get start with redux toolkit, first lets install it

$ npx create-react-app app-name --template redux # javascript
$ npx create-react-app app-name --template redux-typescript # typescript
$ # or using yarn
$ yarn create react-app --template redux
$ yarn create react-app --template redux-typescript

Or add it into existing project

$ npm install @reduxjs/toolkit # or using yarn
$ yarn add @reduxjs/toolkit

The Basic

Lets start from our redux store in src/app/store.js. It will looks like this

import { configureStore } from '@reduxjs/toolkit'; export const store = configureStore({ reducer: {},
});

Lets add a todo feature into our app, start from todoReducer

// src/features/todo/todoSlice.js
import { createSlice } from '@reduxjs/toolkit'; const initialState = { todos: [], status: 'idle'
}; export const todoSlice = createSlice({ name: 'todo', initialState, reducers: { addTodo: (state, action) => { state.todos.push(action.payload); }, removeTodo: (state, action) => { const index = state.todos.findIndex((todo) => todo.id === action.payload.id); if (index !== -1) { state.todos.splice(index, 1); } } }
}); export const { addTodo, removeTodo } = todoSlice.actions; export const selectTodo = (state) => state.todo; export default todoSlice.reducer;

The slice is a place where your reducer and action that will cause redux's state to change live. createSlice takes the name of the slice of the state tree, initial state value and reducer function to define how the state will update. These reducer function will become action creator function to be used inside of component.

Also notice that redux require that we write all updates immutably, but our reducer function mutate the state directly that's because createSlice API used Immer object internally.

Then in src/app/store.js add the following lines

import { configureStore } from '@reduxjs/toolkit';
import counterReducer from '../features/todo/todoSlice' // new line export default configureStore({ reducer: { todo: todoSlice // new line },
});

Using in Component

Let's take a look at how to use it in component. The code is simple first we use useSelector to select the todo from a slice of our state tree by using selector function import from our previous file. Then in the component we dispatch addTodo and removeTodo in respond to Add todo and Delete button click respectively. When addTodo and removeTodo are dispatch our reducer function define in todoSlice with the same name will be trigger and update our state tree accordingly.

// src/features/TodoList.jsx
import React, { useState } from 'react';
import { useDispatch, useSelector } from 'react-redux';
import { selectTodo, addTodo, removeTodo } from './todoSlice'; function Todo({ todo, onDelete }) { const onClick = (event) => { event.preventDefault(); onDelete(todo.id); }; return ( <li> <span>{todo.title}</span> <a href="#" onClick={onClick}>Delete</a> </li> );
} export default function TodoList() { const dispatch = useDispatch(); const { todos } = useSelector(selectTodo); const [todo, setTodo] = useState({ id: 1, title: '' }); const onChange = (event) => { const { value } = event.target; setTodo({ ...todo, title: value }); }; const onClick = () => { dispatch(addTodo(todo)); setTodo({ id: todo.id + 1, title: '' }); }; const onDelete = (id) => { dispatch(removeTodo({ id })); }; return ( <div> <input type="text" onChange={onChange} value={todo.title} /> <button onClick={onClick}>Add todo</button> <ul> {todos.map((todo) => ( <Todo key={todo.id} todo={todo} onDelete={onDelete} /> ))} </ul> </div> );
}

Async Action

Redux Toolkit come with async action handling by default using redux-thunk. To create an async action creator we use createAsyncThunk like follow.

// src/features/todo/todoSlice.js
import { createSlice, createAsyncThunk } from '@reduxjs/toolkit'; function addTodoAPI(payload) { return new Promise((resolve, reject) => { setTimeout(() => resolve(payload), 1000); });
} export const addTodoAsync = createAsyncThunk( 'todo/addTodoAsync', async (payload) => { const data = await addTodoAPI(payload); return data; }
); export const todoSlice = createSlice({ ..., extraReducers: (builder) => { builder .addCase(addTodoAsync.pending, (state) => { state.status = 'pending'; }) .addCase(addTodoAsync.fulfilled, (state, action) => { state.status = 'idle'; state.todos.push(action.payload); }); }
});

createAsyncThunk will return a function with three actions: pending, fulfilled and rejected. These actions are correspond to each state of the API. In our example above we mimic the delay of response from server by using setTimeout.

Using Redux Saga

If you prefer handling async action using redux-saga then we need to modify and create a custom middleware when create our store like follow

// src/app/store.js
import createSagaMiddleware from 'redux-saga';
import { configureStore, getDefaultMiddleware } from '@reduxjs/toolkit';
import counterReducer from '../features/todo/todoSlice';
import rootSaga from './saga'; // disalbe thunk and add redux-saga middleware
const sagaMiddleware = createSagaMiddleware();
const middleware = [...getDefaultMiddleware({ thunk: false }), sagaMiddleware]; export default configureStore({ reducer: { todo: todoSlice }, middleware
}); sagaMiddleware.run(rootSaga); // src/app/saga.js
import { all } from 'redux-saga/effects';
import { todoSaga } from '../features/todo/todoSlice'; export default function* rootSaga() { yield all([todoSaga()]);
}

Then in todoSlice.js

import { call, put, takeLatest } from 'redux-saga/effects';
import { createAction } from '@reduxjs/toolkit'; export const addTodoAsync = createAction('todo/addTodoAsync'); function* addTodoSaga(action) { const data = yield call(addTodoAPI, action.payload); yield put(addTodo(data));
} export function* todoSaga() { yield takeLatest(addTodoAsync, addTodoSaga);
} // rest of the code

Conclusion

In this article we've learned the basic of how to use redux toolkit as well as configuring it to our need by switch some of the core library, redux-thunk with redux-saga. I hope you find this is helpful.

Bình luận

Bài viết tương tự

- vừa được xem lúc

Tìm hiểu về Redux Thunk

Chào mọi người, nếu bạn là người đã biết về React và đang làm quen với Redux chắc hẳn bạn đang rất mơ hồ về các khái niệm cơ bản của Redux như dispatch, store, action creator,... bạn còn đang vật lộn với đống document của Redux để hiểu những khái niệm đó và bạn nghe ai đó trong team nói về Redux Thu

0 0 400

- vừa được xem lúc

[React] Giới thiệu tổng quát về Redux Toolkit

1. Redux Toolkit (RTK) là gì và tại sao lại có nó. . .

0 0 6.6k

- vừa được xem lúc

Uống Pepsi code Vue đi - Uống Cocacola code React nha ;)

. (Nguồn ảnh: Internet). Chào các bạn, chào các bạn. Let's go . 1.

0 0 148

- vừa được xem lúc

Cài đặt taillwind css cho dự án React

Trong bài viết cùng mình tìm hiểu cách cài đặt tailwind css cho một dự án React sẵn có. .

0 0 147

- vừa được xem lúc

Formik vs React Hook Form (Phần 1)

Các lập trình viên Front End đều làm việc rất nhiều với form cùng sự phức tạp của ứng dụng. Do vậy chúng ta cần những thư viện form mạnh mẽ hỗ trợ quản lý các form state, form validation... Thành phần module. Formik bao gồm có 9 dependencies khác. . React Hook Form thì không có.

0 0 372

- vừa được xem lúc

Hướng dẫn React Redux cho người mới bắt đầu - Phần 1

Lời mởi đầu. Chào các bạn, ở thời điểm thực hiện bài viết này mình cũng là một người đang bắt đầu tìm hiểu và học với ReactJs và Redux, trong quá trình tìm hiểu đọc các tài liệu về thư viện này mình có tìm được một bài hướng dẫn khá hay nên đã quyết định chia sẻ với mọi người .

0 0 282