A couple weeks ago I wrote a post about Using Redux with Classes and Hooks. In this post I will take that same example app I made and rewrite it using Redux Toolkit.
RTK(Redux Toolkit) is a new way to write Redux. It takes away the somewhat annoying parts of Redux, like scattered file structure, store configuration and cumbersome boilerplate.
RTK provides us with some cool functions and utilities that take care a lot of repetitive code for us.
Since this is a brief starter example, I highly recommend taking a look at the official documentation, which I think is very well written.
Here is the finished code. If you want to see the 'vanilla' version go to the master branch.
Let's start
Prerequisites
- Have worked with Redux and understand the basics of it.
Goals
- Convert a 'vanilla Redux' app into a 'RTK' app.
First some brief explanation and then we will jump to the code.
To install RTK run
npm install --save @reduxjs/toolkit
configureStore
In a typical Redux app we set our store with createStore
. That's fine but most times you are copy pasting code which is prone to bugs. configureStore
takes care of that. It also combines our slice reducers and any kind of middleware we might have.
It uses Redux Thunk by default and enables the use of the Redux DevTools.
Since we have some middleware applied by default if we want to apply more than that we have to define all of them explicitly. The default one's will no longer apply. What we can do in this case is to use the getDefaultMiddleware
like so.
const store = configureStore({
reducer: rootReducer,
middleware: [...getDefaultMiddleware(), our middleware]
})
createAction
In regular Redux we create an action creator that returns an action type we declared somewhere else and a payload.
createAction
is a helper function that simplifies the whole process.
Regular Redux
import { ADD_SONG } from "./types" // somewhere else
export const addSong = song => {
return {
type: ADD_SONG,
payload: song,
}
}
with createAction
export const addSong = createAction("ADD_SONG")
createReduce
In regular Redux the norm is looping our action types in a switch statement and returning the specified function that updates our state. createReducer
simplifies the process yet again. It takes two arguments. The first is an initial state and the second is an object with all of our action. It also uses by default the Immer library that allows us to write immutable code with mutable like syntax.(Seems like you are mutating the state but you are not).
Regular Redux
export default function(state = initialState, action) {
switch (action.type) {
case ADD_SONG:
return {
songs: [action.payload, ...state.songs],
}
default:
return state
}
}
With createReducer
export default createReducer(initialState, {
[addSong]: (state, action) => {
state.push(action.payload)
},
}
createSlice
If we want to take things to the next level we can use createSlice
that somewhat combines createActions
and
createReducer
. It's basically a function that accepts an initial state, a reducer object(with functions) and a 'slice' name. Actions are being created for us automaftically.
So let do the rewrite and see how do we apply the above.
First we are going to do this with createAction
and createReducer
and then with createSlice
.
We are going to move all of our actions in the songReducers
file to keep things tidy. You can play around with the folder structure to find what fits you best.
import { createAction } from "@reduxjs/toolkit"
// Actions
export const addSong = createAction("ADD_SONG")
export const removeSong = createAction("DELETE_SONG")
export const editSong = createAction("EDIT_SONG")
export const updateSong = createAction("UPDATE_SONG", function prepare(
title,
index
) {
return {
payload: {
title,
index,
},
}
})
export const cancelEdit = createAction("CANCEL_EDIT")
Two things to note here.
- We don't have to import any action types.
- We don't need to type the payload since is returned implicitly.
You will notice that updateSong
has some additional stuff going on.
Many times you will find yourself wanting to pass more than one parameter or add additional logic to your actions. You can do just that with the prepare
function. In our case we want to have two parameters. A title and index.
Now lets rewrite our reducer.
We import our createReducer
. We pass it our initial state and a object with all of our actions like so. No switch statements. Just the name of our action will do.
const initialState = [
{ title: "I love redux", editing: false },
{ title: "The redux song", editing: false },
{ title: "Run to the redux hill", editing: false },
]
// Reducer
export default createReducer(initialState, {
[addSong]: (state, action) => {
state.push(action.payload)
},
[removeSong]: (state, action) => {
state.splice(action.payload, 1)
},
[editSong]: (state, action) =>
state.map((song, i) =>
i === action.payload
? { ...song, editing: true }
: { ...song, editing: false }
),
[updateSong]: (state, action) =>
state.map((song, i) =>
i === action.payload.index
? { ...song, title: action.payload.title, editing: false }
: song
),
[cancelEdit]: (state, action) =>
state.map((song, i) =>
i === action.payload ? { ...song, editing: false } : song
),
})
Now we can do this with createSlice
which will make things even more tidy and compact.
createSlice
will take an initial state, an object of reducers and a slice name. It will generate the actions automatically with the same name as the reducer.
Again you can play around with the folder structure and the naming conventions.
You can check out the ducks way for bundling your reducers and actions.
Here I've created a folder named features and a file named songSlice
. Don't forget to import it in your index
file in the reducer folder.
It will look something like this.
Note that it seems like I'm mutating the state directly, but I'm not.
import { createSlice } from "@reduxjs/toolkit"
const songSlice = createSlice({
name: "songs",
initialState: [
{ title: "I love redux", editing: false },
{ title: "The redux song", editing: false },
{ title: "Run to the redux hill", editing: false },
],
reducers: {
addSong: (state, action) => {
state.push(action.payload)
},
removeSong: (state, action) => {
state.splice(action.payload, 1)
},
editSong: (state, action) => {
const song = state[action.payload]
song.editing = true
},
updateSong: {
reducer(state, action) {
const { title, index } = action.payload
const song = state[index]
song.title = title
song.editing = false
},
prepare(title, index) {
return { payload: { title, index } }
},
},
cancelEdit: (state, action) => {
const song = state[action.payload]
song.editing = false
},
},
})
export const {
addSong,
removeSong,
editSong,
updateSong,
cancelEdit,
} = songSlice.actions
export default songSlice.reducer
That's was it. Hope you liked it.
I think RTK is a great step for Redux and I'm curious to see how it's going to evolve in the future.
Special thanks to Mark Erikson for the corrections and feedback on Twitter
Top comments (0)