新增 reducer
sliceTodos 內新增 reducer,狀態管理器物件,物件內有各種方法來更新 state。
- state: initialState 內的值
- action: 傳入的參數會放在 action.payload
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22
| import { createSlice } from "@reduxjs/toolkit";
export const todoSlice = createSlice({ name: "todos", initialState: [ { id: 1, text: "這是一段話(副本) Slice", }, ], reducers: { createTodo(state, action) { state.push(action.payload); }, }, });
export const { createTodo } = todoSlice.actions; export default todoSlice.reducer;
|
state 無法被賦值
1 2 3 4 5 6
| reducers: { removeTodo(state, action) { const newTodos = state.filter((todo) => todo.id !== action.payload); state = newTodos; }, },
|
上面寫法是錯誤的 因為 state 無法直接被賦值
要賦值的話要用 return 寫法
1 2 3 4 5 6
| reducers: { removeTodo(state, action) { const newTodos = state.filter((todo) => todo.id !== action.payload); return newTodos; }, },
|
從 store 匯出
從 store.js 匯出 todoReducer
1 2 3 4 5 6 7 8 9
| import { configureStore } from "@reduxjs/toolkit"; import todoReducer from "./slice/todosSlice";
export const store = configureStore({ reducer: { todos: todoReducer, }, });
|
元件內使用方法
在元件內將 createTodo 方法引入,並在 dispatch 內帶入 createTodo,來更新 todoSlice 的狀態。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26
| import { useSelector, useDispatch } from "react-redux"; import { createTodo } from "./slice/todosSlice";
const initState = { id: "", text: "", };
function TodoList() { const todos = useSelector((state) => { return state.todos; }); const [newTodoText, setNewTodoText] = useState(""); const [editState, setEditState] = useState(initState); const dispatch = useDispatch();
function addTodo() { dispatch( createTodo({ id: todos.length + 1, text: newTodoText, }) ); setNewTodoText(""); } }
|
非同步方法撰寫
slice 中的 reducers 無法處理非同步邏輯,如果需要處理非同步邏輯,可以使用 createAsyncThunk。
1 2
| import { createSlice, createAsyncThunk } from "@reduxjs/toolkit";
|
- 建立 reducer,用來將狀態寫入 state (只能處理同步邏輯)。
1 2 3 4 5 6 7 8 9 10 11 12
| export const cartSlice = createSlice({ name: 'cart', initialState: { carts: [], }, reducers: { setCarts(state, action) { state.carts = action.payload; }, }, });
|
- 用 createAsyncThunk 建立非同步函式,並 export 出去給其他元件使用。
createAsyncThunk 會帶入兩個參數
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16
|
export const getCartList = createAsyncThunk( 'cart/getCartList', async (payload, { dispatch }) => { try { const res = await axios.get(`/carts`); dispatch(setCarts(res.data)); } catch (error) { console.log(error); } }, );
export const { setCarts } = cartSlice.actions; export default cartSlice.reducer;
|
- 在其他元件中使用 createAsyncMessage 方法
1 2 3 4 5 6 7 8 9 10 11 12 13
| import { useSelector, useDispatch } from 'react-redux'; import { getCartList } from '../../slice/cartSlice';
export default function CartPage() { const carts = useSelector((state) => state.cart.carts); const dispatch = useDispatch();
useEffect(() => { dispatch(getCartList()); }, []) }
|
loading 狀態處理
想在打api時處理 loading 狀態的話可以在 slice 增加 status 狀態,並在 extraReducers 裡用 isAnyOf 監聽所有非同步函式的狀態,當狀態為 pending 時將 status 設為 ‘loading’,狀態為 fulfilled 在設為 ‘success’。
1 2 3 4 5 6 7 8 9 10 11 12 13 14 15 16 17 18 19 20 21 22 23 24 25 26 27 28 29 30
| export const cartSlice = createSlice({ name: 'cart', initialState: { carts: [], status: 'idle', }, reducers: { setCarts(state, action) { state.carts = action.payload; }, }, extraReducers: (builder) => { builder.addMatcher( isAnyOf( getCartList.pending, ), (state) => { state.status = 'loading'; }, ); builder.addMatcher( isAnyOf( getCartList.fulfilled, ), (state) => { state.status = 'success'; }, ); }, });
|