Taming The State in React
Taming The State in React
Taming The State in React
Robin Wieruch
This book is for sale at http://leanpub.com/taming-the-state-in-react
This is a Leanpub book. Leanpub empowers authors and publishers with the Lean Publishing
process. Lean Publishing is the act of publishing an in-progress ebook using lightweight tools and
many iterations to get reader feedback, pivot until you have the right book and build traction once
you do.
Introduction . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . i
About the Author . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . iii
Requirements . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . iv
React . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . iv
Editor and Terminal . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . iv
Node and NPM . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . v
FAQ . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . vi
How to read the Book . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . viii
Redux . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 45
CONTENTS
Basics in Redux . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 47
Action(s) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 47
Reducer(s) . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 48
Store . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 52
Hands On: Redux Standalone . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 53
Advanced Actions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 56
Minimum Action Payload . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 56
Action Type . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 57
Action Creator . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 58
Optional Payload . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 59
Payload Structure . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 60
Hands On: Redux Standalone with advanced Actions . . . . . . . . . . . . . . . . . . 61
Advanced Reducers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 65
Initial State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 65
Nested Data Structures . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 66
Combined Reducer . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 67
Clarification for Initial State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 70
Nested Reducers . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 71
Hands On: Redux Standalone with advanced Reducers . . . . . . . . . . . . . . . . . 72
Redux in React . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 75
Connecting the State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 76
Hands On: Bootstrap React App with Redux . . . . . . . . . . . . . . . . . . . . . . . 76
Hands On: Naive Todo with React and Redux . . . . . . . . . . . . . . . . . . . . . . 81
Connecting the State, but Sophisticated . . . . . . . . . . . . . . . . . . . . . . . . . . . 85
Hands On: Sophisticated Todo with React and Redux . . . . . . . . . . . . . . . . . . 86
Hands On: Connecting State Everywhere . . . . . . . . . . . . . . . . . . . . . . . . . 88
MobX . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 217
Introduction . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 218
Observable State . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 219
Autorun . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 219
Actions . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 220
Computed Values . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . . 223
issues in the first generation of their existence in the world of SPAs. Eventually, they solved these
issues in a future iteration when other SPA solutions appeared at the scene.
The second generation of SPA solutions, among them libraries like React and Vue, focused only on
smaller parts of the application. They focused on the view layer. It was up to the engineer to decide on
further libraries as solutions for specific problems. That’s what made React such a powerful library
in the first place1 , because everyone could decide to extend the application with libraries that solve
specific yet small problems.
Nowadays, a ton of articles and libraries try to solve the issue of state management. It is difficult to
find a consistent source of truth to learn state management in modern applications. Even though,
solutions like React have their own state management implementation for local state in components,
there are more external solutions coming as libraries such as Redux and MobX that establish
sophisticated state management.
Still, it lacks one guide to navigate through all these different solutions to make the differences and
benefits clear. Quite often, the guides miss the point of teaching the problem first. In addition, instead
of showing the minimal approach, they try to fix the problem of state management by using over-
engineered approaches. But it can be so much simpler. It only needs one resource to guide through
state management in modern applications in a consistent and constructive way. That’s the mission
of this book.
If you want to learn something, you have to do it step by step. Trying to solve each atomic problem
after the next one. Don’t apply everything at once. Understand the problem and solve it. That’s my
attempt with this book: It doesn’t only teach Redux in React, but state management in modern
applications. It goes beyond the documentation of state management libraries, but applies the
learnings in real world applications in the book.
These are the heroes of the book: Local State (in React), Redux and MobX. It wouldn’t have been
possible to write the book without the innovators behind these solutions: Dan Abramov2 , Andrew
Clark3 and Michel Weststrate4 . I guess, I can thank them in the name of the community for their
efforts to make state management in modern applications a consistent and enjoyable experience.
1
https://www.robinwieruch.de/reasons-why-i-moved-from-angular-to-react/
2
https://twitter.com/dan_abramov
3
https://twitter.com/acdlite
4
https://twitter.com/mweststrate
Introduction iii
Requirements
What are the requirements to read the book? First of all, you should be familiar with the basics of
web development. You should know how to use HTML, CSS and JavaScript. Perhaps it makes sense
to know the term API7 too, because you will use APIs in the book. In addition, I encourage you to
join the official Slack Group8 for the book to get help or to help others.
React
The book uses React as library to teach modern state management. It is a perfect choice for
demonstrating and learning state management in modern applications. Because React is only a view
layer, it is up to you to decide how to deal with the state in your application. The state management
layer is exchangeable.
After all, it’s not necessary to be a React developer in order to learn about state management in
modern applications. If you are developing with another SPA framework, such as Angular, or view
layer library, such as Vue, all these things about state management taught in this book can still
be applied in your applications. The state management solutions are agnostic to frameworks and
libraries.
Still, since the book uses React for the sake of teaching state management in a proper context, if you
are not familiar with React or need to have a refresher on the topic, I encourage you to read the
precedent book: The Road to learn React9 . It is for free and should enable everyone to learn React.
However, you can decide to pay something to support the project.
Even though the book is for free, people with lacking education have no access to these resources in
the first place. They have to be educated in the English language to be enabled to access it. The Road
to learn React attempts to support education in the developing world10 on an occasionally basis, but
it is a tough undertaking since the book itself is pay what you want.
In addition, the Road to learn React teaches you to make the transition from JavaScript ES5 to
JavaScript ES6. After having read the Road to learn React, you should possess all the knowledge to
read this book. It builds up on the React book perfectly.
substitute most of the tools for other operating system. There is a ton of articles out there that will
show you how to setup a web development environment in a more elaborated way for your OS.
Optionally, you can use git and GitHub on your own, while conducting the exercises in the book, to
keep your projects and the progress in repositories on GitHub. There exists a little guide12 on how
to use these tools. But once again, it is not mandatory for the book and can be overwhelming when
learning everything from scratch. So you can skip it if you are a newcomer in web development to
focus on the essential parts taught in this book.
Command Line
node --version
*v8.2.1
npm --version
*v5.3.0
If you read the Road to learn React, you should be familiar with the setup already. The book gives
you a short introduction into the npm ecosystem on the command line, too. So if you are not familiar
with this, once again you can pick up the Road to learn React book.
12
https://www.robinwieruch.de/git-essential-commands/
13
https://nodejs.org/en/
Introduction vi
FAQ
How to get updates? I have two channels where I share updates about my content. Either you can
subscribe to updates by email14 or follow me on Twitter15 . Regardless of the channel, my objective
is to only share qualitative content. You will never receive any spam. Once you get the update that
the book has changed, you can download the new version of it.
How to get access to the source code projects and screencasts series? If you have bought one
of the extended packages that give you access to the source code projects and screencast series, you
should find these on your course dashboard16 . If you have bought the course somewhere else than
on the official Road to React17 course platform, you need to create an account on this platform, go
to the Admin page and reach out to me with one of the email templates. Afterward I can enroll you
to the course. If you haven’t bought one of the extended packages, you can reach out any time to
upgrade your content to access the source code projects and screencast series.
Why is the book not for free? I often share content for free (blog18 , ebook19 ). I believe information
should be free and everyone should be able to learn from it. However, at some point I think people are
able to be professionals in these areas and can afford to pay for the more advanced content. Whereas
learning React is an entry point to a whole ecosystem and to first professional opportunities, Redux
and MobX are advanced solutions that are used in larger applications most often in companies. At
this point, it makes sense to pay for educating to keep up with the industry standards.
What should I do when I cannot afford to pay for the book? If you cannot afford the book but
want to learn about the topic, you can reach out to me. It could be that you are still a student or that
the book would be too expensive in your country. In addition, I want to support any cause to improve
the diversity in our culture of developers. If you belong to a minority or are in an organization that
supports diversity, please reach out to me.
How to support the project? If you believe in the content that I create, you can support me20 .
Furthermore, I would be grateful if you spread the word about this book after you read it and enjoyed
reading it. Furthermore, I would love to have you as my Patron on Patreon21 .
Is there a money back guarantee? Yes, there is 100% money back guarantee for two months if you
don’t think it’s a good fit. Please reach out to me to get a refund.
Can I help to improve the content? Yes, I would love to hear your feedback. You can simply open
an issue on GitHub22 . These can be improvements technical wise yet also about the written word. I
14
https://www.getrevue.co/profile/rwieruch
15
https://twitter.com/rwieruch
16
https://roadtoreact.com/my-courses
17
https://roadtoreact.com/
18
https://www.robinwieruch.de/
19
https://www.robinwieruch.de/the-road-to-learn-react/
20
https://www.robinwieruch.de/about/
21
https://www.patreon.com/rwieruch
22
https://github.com/rwieruch/taming-the-state-in-react
Introduction vii
am no native speaker that’s why any help is appreciated. You can open pull requests on the GitHub
page as well.
How can I get help while reading the book? The book has a Slack Group23 for people who are
reading the book. You can join the channel to get help or to help others. After all, helping others can
internalize your learnings, too. If there is no one out to help you, you can always reach out to me.
Why does it use React to teach about state? Nowadays, state management is often used in modern
applications. These applications are built with solutions like React, Angular and Vue. In order to
teach about state management, it makes sense to apply it in a real world context such as React. I
picked React, because it has only a slim API and a good learning curve. It is only the view layer with
local state management. You can learn about it in one of my other books: the Road to learn React24 .
However, you don’t necessarily need to apply your learnings about state management in React. You
can apply these learnings in your Angular, Vue or React Native application, too.
What is the ratio between learning Redux and MobX? Even though MobX becomes more of a
popular alternative to Redux, the book teaches you more about Redux than MobX. However, in the
future I want to extend the content and depending on the popularities adjust the chapters.
What’s your motivation behind the book? I want to teach about this topic in a consistent way.
You often find material online that doesn’t receive any updates or only teaches a small part of a
topic. When you learn something new, people struggle to find consistent and up-to-date resources
to learn from. I want to give you this consistent and up-to-date learning experience. In addition, I
hope I can support minorities with my projects by giving them the content for free or by having
other impacts25 . In addition, in the recent time, I found myself fulfilled when teaching others about
programming. It’s a meaningful activity for me that I prefer over any other 9 to 5 job at any company.
That’s why I hope to pursue this path in the future.
23
https://slack-the-road-to-learn-react.wieruch.com/
24
https://www.robinwieruch.de/the-road-to-learn-react/
25
https://www.robinwieruch.de/giving-back-by-learning-react/
Introduction viii
The React library is used as view layer library in this book to demonstrate the usage of state
management in modern applications. However, fortunately, a view layer is exchangeable. If you
are learning state management to apply it to another view layer library, such as Vue, you can try to
substitute React with Vue on your own. The book doesn’t want to dictate the view layer library or
framework. If you are brave, you can apply the learnings in your very own application where you
want to introduce state management while reading the book.
Another hint is to make notes while you read the book. You can write down questions where the
book doesn’t give you an answer and look them up afterward. Or you can write down your learnings
to internalize them. That is how I do it when I read a book. Last but not least, if you write down
feedback about the book, you can send me your notes afterward. I am highly interested to improve
the book all the time to keep the quality up.
It should be obvious by now that you will have the best outcome of this book by having a laptop on
your side. You can join the Slack Group28 to get help from others or to help others yourself. When
having a laptop on your side, you can directly apply your new learnings and confront yourself with
the hands-on chapters. As mentioned before, no one learned something by just reading a book.
28
https://slack-the-road-to-learn-react.wieruch.com/
Local State Management
This chapter will guide you through state management in React without taking any external
state management library into account. You will revisit state management in React with only
this.setState() and this.state. It enables you to build medium sized applications without
complicating and over-engineering it. Not every application needs external state management in
the first place.
By revisiting the topic of local state management in React, you will get to know how to use the
local state in a React application. The chapter guides you through important topics when dealing
with state in React: controlled components, unidirectional data flow and asynchronous state. It will
teach you all these necessary topics before diving into state management with an external state
management library.
After revisiting the local state, you will get to know best practices and patterns to scale your state
management when using only local state. Even though you are not dependent on external state
management solutions yet, you can use a handful of those techniques to scale state management
with external libraries, too, which are described later on in this book.
At the end of the chapter, you will get to know the limits of local state management in React. The
topic itself is highly discussed in the community as you will learn in one of the final lessons of this
chapter. The chapter concludes in the problem of local state management to give you a motivation
to dive into one of the external state management solutions.
Local State Management 2
Definitions
Before we dive into state management, this chapter gives you general definitions and definitions for
state managements to build up a common vocabulary for state management for this book. It should
help you to follow the book effortlessly when reading it without leaving space for confusion.
Pure Functions
Pure functions is a concept from the functional programming paradigm. It says that a pure function
always returns the same output if given the same input. There is no layer in between that could alter
the output on the way when the input doesn’t change. The layer in between, that could possibly alter
the output, is called side-effect. Thus, pure functions have no side-effects. Two major benefits of
these pure functions are predictability and testability.
Immutability
Immutability is a concept of functional programming, too. It says that a data structure is immutable
when it cannot be changed. When there is the need to modify the immutable data structure, for
instance an object, you would always return a new object. Rather than altering the object at hand,
you would create a new object based on the old object and the modification. The old and new object
would have their own instances.
Immutable data structures have the benefit of predictability. For instance, when sharing an object
through the whole application, it could lead to bugs when altering the object directly, because every
stakeholder has a reference to this potentially altered object. It would be unpredictable what happens
when an object changes and a handful of stakeholders are dependent on this object. In a growing
application, it is difficult to oversee the places where the object is currently used by its reference.
The antagonist of immutability is called mutability. It says that an object can be modified.
State
State is a broad word in modern applications. When speaking about application state, it could be
anything that needs to live and be modified in the browser. It could be an entity state that was
retrieved from a backend application or a view state in the application, for instance, when toggling
a popup to show additional information.
I will refer to the former one as entity state and to the latter one as view state. Entity state is data
retrieved from the backend. It could be a list of authors or the user object describing the user that
is currently logged in to the application. View state, on the other hand, doesn’t need to be stored in
the backend. It is used when you open up a modal or switch a box from preview to edit mode.
When speaking about managing the state, meaning initializing, modifying and deleting state, it will
be coined under the umbrella term of state management. Yet, state management is a much broader
Local State Management 3
topic. While the mentioned actions are low-level operations, almost implementation details, the
architecture, best practices and patterns around state management stay abstract. State management
involves all these topics to keep your application state durable.
State can be an atomic object or one large aggregated object. When speaking about the view state,
that only determines whether a popup is open or closed, it is an atomic state object. When the
whole application state can be derived from on aggregated object, which includes all the atomic
state objects, it is called a global state object. Often, a global state object implies that it is accessible
from everywhere, but that’s not necessarily the case.
The state itself can be differentiated into local state and sophisticated state. The management of
this state is called local state management and sophisticated state management.
Local State
The naming local state is widely accepted in the web development community. Another term might
be internal component state.
Local state is bound to components. It lives in the view layer. It is not stored somewhere else. That’s
why it is called local state because it is colocated to the component.
In React, the local state is embraced by using this.state and this.setState(). But it can have
a different implementation and usage in other view layer or SPA solutions. The book explains and
showcases the local state in React before diving into sophisticated state management with external
libraries.
Sophisticated State
I cannot say that it is widely agreed on to call it sophisticated state in the web development
community. However, at some point you need a term to distinguish it from local state. That’s why
I often refer to it as sophisticated state. In other resources, you might find it referred to as external
state, because it lives outside of the local component or outside of the view layer.
Most often, sophisticated state is outsourced to libraries that are library or framework agnostic and
thus agnostic to the view layer. But most often they provide a bridge to access and modify state
from the view layer. When using only local state in a scaling application, you will allocate too much
state along your components in the view layer. However, at some point you want to separate these
concerns. That’s when sophisticated state comes into play.
Two libraries that are known for handling sophisticated state are called Redux and MobX. Both
libraries will be explained, discussed and showcased in this book.
Local State Management 4
Visibility of State
Since local state is only bound to the component instance, only the component itself is aware of
these properties being state. However, the component can share the state to its child components. In
React, the child components are unaware of these properties being state in their parent component.
They only receive these properties as props.
On the other hand, sophisticated state is often globally accessible. In theory, the state can be accessed
by each component. Often, it is not best practice to give every component access to the global state,
thus it is up to the developer to bridge only selected components to the global state object. All other
components stay unaware of the state and only receive properties as props to act on them.
Local State Management 5
Code Playground
this.state = {
counter: 0
};
}
render() {
return (
<div>
<p>{this.state.counter}</p>
</div>
);
}
}
The example shows a Counter component that has a counter property in the local state object. It
is defined with a value of 0 when the component gets instantiated by its constructor. In addition,
the counter property from the local state object is used in the render method of the component to
display its current value.
There is no state manipulation in place yet. Before you start to manipulate your state, you should
know that you are never allowed to alter the state directly: this.state.counter = 1. That would
be a direct mutation. Instead, you have to use the React component API to change the state explicitly
by using the this.setState() method.
Local State Management 6
Code Playground
this.onIncrement = this.onIncrement.bind(this);
this.onDecrement = this.onDecrement.bind(this);
}
onIncrement() {
this.setState({
counter: this.state.counter + 1
});
}
onDecrement() {
this.setState({
counter: this.state.counter - 1
});
}
render() {
...
}
}
The class methods can be used in the render() method to trigger the local state changes.
Code Playground
render() {
return (
<div>
<p>{this.state.counter}</p>
<button type="button" onClick={this.onIncrement}>
Local State Management 7
Increment
</button>
<button type="button" onClick={this.onDecrement}>
Decrement
</button>
</div>
);
}
}
Now ,the button onClick handler should invoke the class methods to alter the state by either
incrementing or decrementing the counter value.
The update functionality with this.setState() is performing a shallow merge of objects. What
does a shallow merge mean? Imagine you had the following state in your component.
Code Playground
this.state = {
authors: [...],
articles: [...],
};
When updating the state only partly, for instance the authors by doing the following, the articles
are left intact
Code Playground
this.setState({
authors: [
{ name: 'Robin', id: '1' }
]
});
It only updates the authors without touching the articles. That’s a shallow merge. It simplifies
the local state management by not always keeping an eye on all properties in the local state.
On the other hand, functional stateless components have no state, because, as the name implies,
they are only functions and thus, they are stateless. In a stateless component state can only be passed
as props from a parent component. In addition, callback functions could be passed down to alter the
state in the parent component. A functional stateless component for the Counter example could look
like the following:
Code Playground
function CounterPresenter(props) {
return (
<div>
<p>{props.counter}</p>
<button type="button" onClick={props.onIncrement}>
Increment
</button>
<button type="button" onClick={props.onDecrement}>
Decrement
</button>
</div>
);
}
Now only the props from the parent component would be used in this functional stateless
component. The counter prop would be displayed and the two callback functions, onIncrement()
and onDecrement() would be used for the buttons. However, the functional stateless component is
not aware whether the passed properties are state, props or some other derived properties. The origin
of the props doesn’t need to be in the parent component after all, it could be somewhere higher up
the component tree. The parent component would only pass the properties or derived properties
along the way. In addition, the component is unaware of what the callback functions are doing. It
doesn’t know that these alter the local state of the parent component.
After all, the callback functions in the stateless component would make it possible to alter the state
somewhere above in one of the parent components. Once the state was manipulated, the new state
flows down as props into the child component again. The new counter prop would be displayed
correctly, because the render method of the child component runs again with the incoming changed
props.
The example shows how local state can traverse down from one component to the component tree.
To make the example with the functional stateless component complete, let’s quickly show what
a potential parent component, that manages the local state, would look like. It is a React ES6 class
component in order to be stateful.
Local State Management 9
Code Playground
this.state = {
counter: 0
};
this.onIncrement = this.onIncrement.bind(this);
this.onDecrement = this.onDecrement.bind(this);
}
onIncrement() {
this.setState({
counter: this.state.counter + 1
});
}
onDecrement() {
this.setState({
counter: this.state.counter - 1
});
}
render() {
return <CounterPresenter
counter={this.state.counter}
onIncrement={this.onIncrement}
onDecrement={this.onDecrement}
/>
}
}
It is not by accident that the suffixes in the naming of both Counter components are Container
and Presenter. It is called the container and presentational component pattern29 . It is most often
applied in React, but could live in other component centred libraries and frameworks, too. If you have
never heard about it, I recommend reading the referenced article. It is a widely used pattern, where
29
https://medium.com/@dan_abramov/smart-and-dumb-components-7ca2f9a7c7d0
Local State Management 10
the container component deals with “How things work” and the presenter component deals with
“How things look”. In this case, the container component cares about the state while the presenter
component only displays the counter value and provides a handful of click handler yet without
knowing that these click handlers manipulate the state.
Container components are the ideal candidates to manage state while the presenter components
only display it and act on callback functions. You will encounter these container components more
often in the book, when dealing with the concepts of higher order components, that could potentially
manage local state, and connected components.
• Are the properties passed from the parent component? If yes, the likelihood is high that they
aren’t state. Though it is possible to save props as state, there are little use cases. It should be
avoided to save props as state. Use them as props as they are.
• Are the properties unchanged over time? If yes, they don’t need to be stateful, because they
don’t get modified.
• Are the properties derivable from local state or props? If yes, you don’t need them as state,
because you can derive them. If you allocated extra state, the state has to be managed and can
get out of sync when you miss to derive the new properties at some point.
30
https://facebook.github.io/react/docs/thinking-in-react.html
Local State Management 11
Form State
A common use case in applications is to use HTML forms. For instance, you might need to retrieve
user information like a name or credit card number or submit a search query to an external API.
Forms are used everywhere in web applications.
There are two ways to use forms in React. You can use the ref attribute or local state. It is
recommended to use the local state approach, because the ref attribute is reserved for only a few
use cases. If you want to read about these use cases when using the ref attribute, I encourage you to
read the following article: When to use Ref on a DOM node in React31 .
The following code snippet is a quick demonstration on how form state can be used by using the
ref attribute. Afterward, the code snippet will get refactored to use the local state which is the best
practice anyway.
Code Playground
this.onSubmit = this.onSubmit.bind(this);
}
onSubmit(event) {
const { value } = this.input;
event.preventDefault();
}
render() {
return (
<form onSubmit={this.onSubmit}>
<input
ref={node => this.input = node}
type="text"
/>
31
https://www.robinwieruch.de/react-ref-attribute-dom-node/
Local State Management 12
<button type="submit">
Search
</button>
</form>
);
}
}
The value from the input node is retrieved by using the reference to the DOM node. It happens in
the onSubmit() method. The reference is created by using the ref attribute in the render() method.
Now let’s see how to make use of local state to embrace best practices rather than using the reserved
ref attribute.
Code Playground
this.state = {
query: ''
};
this.onChange = this.onChange.bind(this);
this.onSubmit = this.onSubmit.bind(this);
}
onChange(event) {
const { value } = event.target;
this.setState({
query: value
});
}
onSubmit(event) {
const { query } = this.state;
this.props.onSearch(query);
event.preventDefault();
}
render() {
return (
<form onSubmit={this.onSubmit}>
<input
onChange={this.onChange}
type="text"
/>
<button type="submit">
Search
</button>
</form>
);
}
}
You don’t need to make use of the ref attribute anymore. You can solve the problem by using local
state only. The example demonstrates it with only one input field yet it can be used with multiple
input fields, too. You would only need to allocate more properties in the local state.
Controlled Components
The previous example of using form state with local state has one flaw. It doesn’t make use of
controlled components. Naturally, a HTML input field holds its own state. When you enter a value
into the input field, the DOM node knows about the value.
However, the value lives in your local state, too. You have it in both, the native DOM node state and
local state. But you want to make use of a single source of truth. It is a best practice to overwrite the
native DOM node state by using the value attribute and the value from the local state.
Let’s consider the previous example again. The input field had no value attribute assigned. By
using the native value attribute and passing the local state as value, you convert an uncontrolled
component to a controlled component.
Local State Management 14
Code Playground
...
render() {
return (
<form onSubmit={this.onSubmit}>
<input
value={this.state.query}
onChange={this.onChange}
type="text"
/>
<button type="submit">
Search
</button>
</form>
);
}
}
Now the value comes from the local state as single source of truth. It cannot get out of sync with
the native DOM node state.
the updated state. The state flows back to the input field to make it a controlled component. The
loop is closed. A new loop can be triggered by typing something into the input field again.
The unidirectional data flow makes state management predictable and maintainable. The best
practice already spread to other state libraries, view layer libraries and single page application
solutions. In the previous generation of SPAs, most often other mechanics were used.
For instance, in Angular 1.x you had to use two-way data binding in a model-view-controller (MVC)
architecture. That means, once you changed the value in the view, let’s say in an input field by
typing something, the value got changed in the controller. But it worked vice versa, too. Once you
had changed the value in the controller programmatically, the view, to be more specific the input
field, displayed the new value.
You might wonder: What’s the problem with this approach? Why is everybody using unidirectional
data flow instead of bidirectional data flow now?
React embraces unidirectional data flow. In the past, frameworks like Angular 1.x embraced
bidirectional data flow. It was known as two-way data binding. It was one of the reasons that
made Angular popular in the first place. But it failed in this particular area, too. Especially, in my
opinion, this particular flaw led a lot of people to switch to React. But at this point I don’t want
to get too opinionated. So why did the bidirectional data flow fail? Why is everyone adopting the
unidirectional data flow?
The three advantages in unidirectional data flow over bidirectional data flow are predicability,
maintainability and performance.
Predicability: In a scaling application, state management needs to stay predictable. When you alter
your state, it should be clear which components care about it. It should also be clear who alters
the state in the first place. In an unidirectional data flow one stakeholder alters the state, the state
gets stored, and the state flows down from one place, for instance a stateful component, to all child
components that are interested in the state.
Maintainability: When collaborating in a team on a scaling application, one requirement of state
management is predictability. Humans are not capable to keep track of a growing bidirectional data
flow. It is a limitation by nature. That’s why the state management stays more maintainable when
it is predictable. Otherwise, when people cannot reason about the state, they introduce inefficient
state handling.
But maintainability doesn’t come without any cost in a unidirectional data flow. Even though the
state is predictable, it often needs to be refactored thoughtfully. In a later chapter, you will read
about those refactorings such as lifting state or higher order components for local state.
Performance: In a unidirectional data flow, the state flows down the component tree. All compo-
nents that depend on the state have the chance to re-render. Contrary to a bidirectional data flow,
it is not always clear who has to update according to state changes. The state flows in too many
Local State Management 16
directions. The model layer depends on the view layer and the view layer depends on the model
layer. It’s a vice versa dependency that leads to performance issues in the update lifecycle.
These three advantages show the benefits of using a unidirectional data flow over an bidirectional
data flow. That’s why so many state management solutions thrive for the former one nowadays.
Local State Management 17
Lifting State
In a scaling application, you will notice that you pass a lot of state down to child components as
props. These props are often passed down multiple component levels. That’s how state is shared
vertically. Yet, the other way around, you will notice that more components need to share the same
state. That’s how state is shared horizontally. These two scaling issues, sharing state vertically and
horizontally, are common in local state management. Therefore you can lift the state up and down
keeping your local state architecture maintainable. Lifting the state prevents to share too much or
too less state in your component tree. Basically, it is a refactoring that you have to do once in a while
to keep your components maintainable and focused on only consuming the state that they need to
consume.
In order to experience up and down lifting of local state, the following chapter will demonstrate it
with two examples. The first example that demonstrates the uplifting of state is called: “Search a
List”-example. The second example that demonstrates the downlifting of state is called “Archive in
a List”-example.
The “Search a List”-example has three components. Two sibling components, a Search component
and a List component, that are used in an overarching SearchableList component.
First, the Search component:
Code Playground
this.state = {
query: ''
};
this.onChange = this.onChange.bind(this);
}
onChange(event) {
Local State Management 18
this.setState({
query: value
});
}
render() {
return (
<div>
{this.props.children} <input
type="text"
value={this.state.query}
onChange={this.onChange}
/>
</div>
);
}
}
Code Playground
Code Playground
While the Search component is a stateful ES6 class component, the List component is only
a stateless functional component. The parent component that combines the List and Search
components into a SearchableList component is a stateless functional component too.
However, the example doesn’t work. The Search component knows about the query that could be
used to filter the list, but the List component doesn’t know about it. You have to lift the state of
the Search component up to the SearchableList to make the query state accessible for the List
component in order to filter the list. You want to share the query state in both List component and
Search component.
In order to lift the state up, the SearchableList becomes a stateful component. You have to refactor
it to an React ES6 class component. On the other hand, you can refactor the Search component to a
functional stateless component, because it doesn’t need to be stateful anymore. The stateful parent
component takes care about its whole state. In other cases the Search component might stay as a
stateful ES6 class component, because it still manages own state. But not in this example.
First, the Search component:
Code Playground
Code Playground
this.state = {
query: ''
};
this.onChange = this.onChange.bind(this);
}
onChange(event) {
const { value } = event.target;
this.setState({
query: value
});
}
render() {
const { list } = this.props;
const { query } = this.state;
return (
<div>
<Search
query={query}
onChange={this.onChange}
>
Search List:
</Search>
<List list={(list || []).filter(byQuery(query))} />
</div>
);
}
}
function byQuery(query) {
return function(item) {
Local State Management 21
return !query ||
item.name.toLowerCase().includes(query.toLowerCase());
}
}
After you have lifted the state up, the parent component takes care about the local state management.
Both child components don’t need to take care about it anymore. You have lifted the state up to share
the local state across the child components. The list gets filtered by the search query before it reaches
the List component.
Let’s get to the second example: the “Archive in a List”-example. It builds up on the previous example,
but this time the List component has the extended functionality to archive an item in the list.
First, the List component:
Code Playground
Code Playground
this.state = {
query: '',
archivedItems: []
};
this.onChange = this.onChange.bind(this);
this.onArchive = this.onArchive.bind(this);
}
...
onArchive(id) {
const { archivedItems } = this.state;
this.setState({
archivedItems: [...archivedItems, id]
});
}
render() {
const { list } = this.props;
const { query, archivedItems } = this.state;
return (
<div>
...
<List
list={filteredList}
onArchive={this.onArchive}
/>
</div>
Local State Management 23
);
}
}
...
function byArchived(archivedItems) {
return function(item) {
return !archivedItems.includes(item.id);
}
}
The previous example was extended to facilitate the archiving of items in a list. Now, the List
component receives all the necessary properties: an onArchive callback and the list, filtered by query
and archivedItems.
You might see already the flaw. The SearchableList takes care about the archiving functionality.
However, it doesn’t need the functionality itself. It only passes all the state to the List component
as props. It manages the state on behalf of the List component. In a scaling application it would
make sense to lift the state down to the List component. Even though the List component becomes
a stateful component afterward, it is step in the right direction keeping the local state maintainable
in the long run.
First, the List component:
Code Playground
this.state = {
archivedItems: []
};
this.onArchive = this.onArchive.bind(this);
}
onArchive(id) {
const { archivedItems } = this.state;
this.setState({
Local State Management 24
render() {
const { list } = this.props;
const { archivedItems } = this.state;
return (
<ul>
{filteredList.map(item =>
<li key={item.id}>
<span>
{item.name}
</span>
<span>
<button
type="button"
onClick={() => this.onArchive(item.id)}
>
Archive
</button>
</span>
</li>
)}
</ul>
);
}
}
Code Playground
import React from 'react';
this.state = {
query: ''
};
this.onChange = this.onChange.bind(this);
}
...
render() {
const { list } = this.props;
const { query } = this.state;
return (
<div>
...
<List list={filteredList} />
</div>
);
}
}
Now, you have seen both variations of lifting state: lifting state up and lifting state down.
In the first example, the “Search a List”-example, the state had to be lifted up to share the query
property in two child components. The Search component had to manipulate the state by using
a callback, but also had to use the query to be a controlled component. On the other hand, the
‘SearchableList’ component had to filter the list by using the query property. Another solution would
have been to pass down the query property to the List component and let the component deal with
the filtering itself.
In the second example, the “Archive in a List”-example, the state could be lifted down to keep the
state maintainable in the long run. The parent component shouldn’t be concerned about state that
Local State Management 26
isn’t used by the parent component itself and isn’t shared across multiple child components. Because
only one child component cared about the archived items, it was a clean code refactoring to lift the
state down.
In conclusion, lifting state allows you to keep your local state management maintainable. Lifting
state should be used to give components access to all the state they need, but not to more state
than they need. Sometimes you have to refactor components from a functional stateless component
to a React ES6 class component or vice versa. It’s not always possible, because a component that
could become possibly a stateless functional component could still have other stateful properties.
Functional State
In all recent chapters, there is a mistake in using this.setState(). It is important to know that
this.setState() is executed asynchronously. React batches all the state updates. It executes them
after each other for performance optimization. Thus this.setState() comes in two versions.
In its first version, the this.setState() method takes an object to update the state. As explained
in a previous chapter, the merging of the object is a shallow merge. For instance, when updating
authors in a state object of authors and articles, the articles stay intact. The previous examples
have already used this approach.
Code Playground
this.setState({
...
});
In its second version, the this.setState() method takes a function. The function has the previous
state and props in the function signature.
Code Playground
So, what’s the flaw in using this.setState() with an object? In several examples in the last
chapters, the state was updated based on the previous state or props. However, this.setState()
executes asynchronously. Thus the state or props that are used to perform the update could be stale
at this point in time. It could lead to bugs in your local state management, because you would update
the state based on stale properties. When using the functional approach to update the local state, the
state and props at the time of execution are used when this.setState() performs asynchronously.
Let’s revisit one of the previous examples:
Local State Management 27
Code Playground
this.state = {
counter: 0
};
this.onIncrement = this.onIncrement.bind(this);
this.onDecrement = this.onDecrement.bind(this);
}
onIncrement() {
this.setState({
counter: this.state.counter + 1
});
}
onDecrement() {
this.setState({
counter: this.state.counter - 1
});
}
render() {
return <CounterPresenter
counter={this.state.counter}
onIncrement={this.onIncrement}
onDecrement={this.onDecrement}
/>
}
}
Executing one of the class methods, onIncrement() or onDecrement(), multiple times could lead
to a bug. Because both methods depend on the previous state, it could use a stale state when the
asynchronous update wasn’t executed but the method invoked another time.
Local State Management 28
Code Playground
It becomes even more error prone when multiple functions that use this.setState() depend on
the previous state. You can refactor the example to use the functional state updating approach.
Code Playground
onIncrement() {
this.setState(prevState => ({
counter: prevState.counter + 1
}));
}
onDecrement() {
this.setState(prevState => ({
counter: prevState.counter - 1
}));
}
render() {
...
}
}
The functional approach opens up two more benefits. First, the function that updates the state is a
pure function. There are no side-effects. The function always will return the same output when
given the same input. It makes it predictable and uses the benefits of functional programming.
Second, since the function is pure, it can be tested easily in an unit test and independently from the
component. It gives you the opportunity to test your local state updates. You only have to extract
the function from the component.
Local State Management 29
Code Playground
import React from 'react';
onIncrement() {
this.setState(incrementUpdate);
}
onDecrement() {
this.setState(decrementUpdate);
}
render() {
...
}
}
Now, you could test the pure functions. After all you might wonder, when to use the object and
when to use the function in this.setState()? The recommended rules of thumb are:
• Always use this.setState() with a function when you depend on previous state or props.
• Only use this.setState() with an object when you don’t depend on previous properties.
• In case of uncertainty, default to use this.setState() with a function.
why not use it to manage the local state of a component? Let’s revisit an adjusted example of the
“Archive in a List”-example.
Code Playground
import React from 'react';
this.state = {
archivedItems: []
};
this.onArchive = this.onArchive.bind(this);
}
onArchive(id) {
const { archivedItems } = this.state;
this.setState({
archivedItems: [...archivedItems, id]
});
}
render() {
const { list } = this.props;
const { archivedItems } = this.state;
return (
<ul>
{filteredList.map(item =>
<li key={item.id}>
<span>
{item.name}
</span>
<span>
<button
type="button"
onClick={() => onArchive(item.id)}
Local State Management 31
>
Archive
</button>
</span>
</li>
)}
</ul>
);
}
}
function byArchived(archivedItems) {
return function(item) {
return !archivedItems.includes(item.id);
};
}
The ArchiveableList has two purposes. On the one hand, it is a pure presenter that shows the items
in a list. On the other hand, it is stateful container that keeps track of the archived items. Therefore,
you could split it up into representation and logic thus into presentational and container component.
However, another approach could be to transfer the logic, mainly the local state management, into a
higher order component. Higher order components are reusable and thus the local state management
could become reusable.
Code Playground
function byArchived(archivedItems) {
return function(item) {
return !archivedItems.includes(item.id);
};
}
function withArchive(Component) {
class WithArchive extends React.Component {
constructor(props) {
super(props);
this.state = {
archivedItems: []
};
Local State Management 32
this.onArchive = this.onArchive.bind(this);
}
onArchive(id) {
const { archivedItems } = this.state;
this.setState({
archivedItems: [...archivedItems, id]
});
}
render() {
const { list } = this.props;
const { archivedItems } = this.state;
return <Component
list={filteredList}
onArchive={this.onArchive}
/>
}
}
return WithArchive;
}
In return the List component would only display the list and receives a function in its props to
archive an item.
Code Playground
type="button"
onClick={() => onArchive(item.id)}
>
Archive
</button>
</span>
</li>
)}
</ul>
);
}
Now you can compose the list facilitating component with the functionality to archive items in a
list.
Code Playground
The List component would only display the items. The ability to archive an item in the List
component would be opt-in with a higher order component called withArchive. In addition, the
HOC can be reused in other List components too for managing the state of archived items. After
all, higher order components are great to extract local state management from components and to
reuse the local state management in other components.
component tree. In addition, the provider pattern is later on used in sophisticated state management
libraries to glue the state layer (Redux, MobX) to the (React) view layer.
There are two things you have to know about React before you can implement your own provider
pattern in React: children and context.
When you have already learned to use plain React, you should know about React’s children. Basically
they enable you to nest JSX into each other same as you would nest HTML tags into each other.
Code Playground
class App extends Component {
render() {
return (
<Search
query={this.state.query}
onChange={this.onChange}
>
Search the List
</Search>
);
}
}
The component would be displayed as an input field with a label next to it that says “Search the
List”.
The second requirement before implementing the provider pattern in React is React’s context. React’s
context is not highly advertised. It is even discouraged to use it. The team behind React keeps it open
if the API of the context in React changes in the future.
Local State Management 35
Nevertheless, the context in React is a powerful feature. Do you remember the last time when
you had to pass props several components down your component tree? In plain React, you can
be confronted often with this issue. It can happen that a couple of these props are even mandatory
for each child component. Thus you would need to pass the props down to each child component.
In return, this would clutter every component passing down these props.
When these props become mandatory, React’s context gives you a way out of this mess. Instead of
passing down the props explicitly down each component, you can hide props, that are necessary for
each component, in the React context object and pass them implicitly down to each component. The
React context object traverses invisible down the component tree. When a component needs access
to the context object, it can access it.
But you shouldn’t overdo it with React’s context. So what are use cases for this approach? For
instance, your application could have a configurable colored theme. Each component should be
colored depending on the configuration. The configuration is fetched once from your server, but
now you want to make this implicitly accessible for all components. Therefore you could use React’s
context to give every component access to the colored theme.
How is React’s context provided and consumed? Imagine you would have component A as root
component that provides the context and component C as one of the child components that consumes
the context. But in between is component D. The application has a colored theme that can be used to
style your components. Thus, you want to make the colored theme available for every component
via the React context without passing it as mandatory props through each component.
In your A component you provide the context. It is a hardcoded colored theme property in this case,
but it can be anything from component state to component props. Component A display component
D but makes the context available to all its children.
Code Playground
render() {
return <D />;
}
}
A.childContextTypes = {
coloredTheme: PropTypes.string
};
Local State Management 36
In your component C, somewhere below component D, you could consume the context object. Notice
that component A doesn’t need to pass down anything via component D in the props.
Code Playground
C.contextTypes = {
coloredTheme: PropTypes.string
};
By using the colored theme property from this.context, the component can derive its style. That
way every component that needs to be styled according to the colored theme could get the necessary
information from React’s context object. You can read more about React’s context in the official
documentation33 .
Both functionalities of React, context and children, are necessary to implement the provider pattern
in React. These were the basics. Now you are able to implement it. Basically for the provider pattern
there needs to be one part in the pattern that makes the properties accessible in the context and
another part where components consume the context.
Let’s start with the former: a Provider component.
33
https://facebook.github.io/react/docs/context.html
Local State Management 37
Code Playground
class ThemeProvider extends React.Component {
getChildContext() {
return {
coloredTheme: this.props.coloredTheme
};
}
render() {
return <div>{this.props.children}</div>;
}
}
ThemeProvider.childContextTypes = {
coloredTheme: PropTypes.string
};
The Provider component only sets the colored theme from the incoming props. In addition, it only
renders its children and doesn’t add anything to the JSX.
After declaring the Provider component, you can use it anywhere in your React component tree. It
makes most sense to use it at the top of your component hierarchy to make the context, the colored
theme, accessible to everyone.
Code Playground
const coloredTheme = "green";
// hardcoded theme
// however, imagine the configuration is located somewhere else
// and would be different for every user of your application
// it would need to be fetched first
// and varies depending on the app user
ReactDOM.render(
<ThemeProvider coloredTheme={coloredTheme}>
<App />
</ThemeProvider>,
document.getElementById('app')
);
Now, every child component can consume the colored theme provided by the ThemeProvider
component. It doesn’t need to be the direct child, in this case the App component, but any component
down the component tree.
Local State Management 38
Code Playground
Paragraph.contextTypes = {
coloredTheme: PropTypes.string
};
That’s basically it for the provider pattern. You have the Provider component that makes properties
accessible in React’s context and components that consume the context. How does this relate to
state management? Basically the provider pattern is often used, when using a sophisticated state
management solution that makes the state object(s) accessible in your view layer via React’s context.
The whole state can be accessed in each component. Perhaps you will never implement the provider
pattern on your own, but you will most likely use it from a external library when you use a
sophisticated state management solution such as Redux or MobX later on. So keep it in mind.
Local State Management 39
Persistence in State
State in applications is often not persistent. When your application starts, there is often an initial
state. The state updates when the user interacts with the application or data arrives from a backend
application. However, you might wonder whether there is a way to persist the state? The question
applies to both, local state management and sophisticated state management later on.
The obvious answer to this question would be to implement a backend application with a database
to persist the state. Extracting the state from your application is called dehydrating state. Now,
every time your application bootstraps, you would retrieve the state from the backend that keeps
it in a database. Once the state arrives in the response asynchronously, you would rehydrate state
into your application.
While the dehydration of the state could happen any time your application is running, the
rehydration would take place when your components mount. The best place to do it in React would
be the componentDidMount() lifecycle method. Take for example the ArchiveableList component.
It could retrieve all the already archived unique identifiers of items on componentDidMount() and
rehydrate it to the local state.
Code Playground
onArchive(id) {
...
}
componentDidMount() {
fetch('path/to/archived/items')
.then(response => response.json())
.then(archivedItems => this.setState(rehydrateArchivedItems(archivedItems)\
));
}
render() {
...
}
}
Local State Management 40
function rehydrateArchivedItems(archivedItems) {
return function(prevState) {
return {
archivedItems: [
...prevState.archivedItems,
...archivedItems
]
};
};
}
Now, every time the component initializes, the persistent archived items will get rehydrated into
the application state.
The dehydration could happen anytime, but to avoid inconsistencies, in the example of archived
items, the dehydration would take place when an item gets archived. It is a usual request to the
backend to save the item as being archived.
The rehydration and dehydration of state are most often unconscious steps in modern applications.
It is common sense to retrieve all the necessary data from the backend when your application
bootstraps and to update the data when something has changed. But you can keep the rehydration
and dehydration of state in mind to keep your application state in sync with your backend data as
single source of truth.
Local Storage
Is there a more lightweight solution compared to a backend application? You could use the native
browser API. To be more specific, most of the modern browser have a storage functionality to persist
data. It is the lightweight version of a database that is used in the browser. Of course, it is only visible
to the user of the browser and cannot be distributed to other users.
Modern browsers have access to the local storage34 and session storage35 . Both work the same, but
there is one difference in their functionalities. While the local storage keeps the data even when the
browser is closed, the session storage expires once the browser closes. Both storages work the same
by using key value pairs.
34
https://developer.mozilla.org/en/docs/Web/API/Window/localStorage
35
https://developer.mozilla.org/en/docs/Web/API/Window/sessionStorage
Local State Management 41
Code Playground
In the end, you can apply them the same way as you did in the previous ArchiveableList component
that used the backend request to retrieve the data. Only that the ArchiveableList component would
use the storage instead of the backend to retrieve the state. If you are keen to explore the usage with
the local storage in React, you can read more about it in this article36 .
Caching in State
The local state, later on the sophisticated state as well, can be used as a cache for your application.
A cache would make recurring requests to retrieve data from a backend redundant, because they
would return the same data as before and the data is already cached in the state.
Imagine your application has an interface to search for popular stories on a news platform. The news
platform has an open API that you can use to retrieve those popular stories. Your own application
that consumes the API of this platform would only have a search field to search for popular stories
from the platform and a list to display the stories once you have searched for them.
Next, imagine you made your first request searching popular stories about “React”. You are not
satisfied with the search result, because you wanted to be more specific, and search again for “React
Local State”. Still, no satisfying search result. Now the search result for “React Local State” is visible
in your application. Next, you want to head back to search for “React” stories again. Your application
makes a third request to retrieve the “React” stories from the platform API. In a perfect world, the
application would know that you have already searched for “React” stories before. That’s where
caching comes into play. The third request could have been avoided if the application had cached
the search results.
Such a fluctuant cache solution is not difficult to implement with a local state. Bear in mind that
it would work with a sophisticated state management layer that is explained later in the book, too.
When searching for the stories, you already have a unique identifier which you can use as a key in an
36
https://www.robinwieruch.de/local-storage-react/
Local State Management 42
object to store the search result in the local state. The unique identifier is your search term. It would
be either “React” or “React Local State” considering the previous example. The value corresponding
to the key would be the search result. In the example, it would be the popular stories about “React”
and “React Local Storage”. After all, your cache object in the local state might be similar to this:
Code Playground
this.state = {
...
searchCache: {
React: [...],
ReactLocalState: [...],
}
}
Every time your application performs a search request, the key value pair in the cache object in your
local state would be filled. Before you make a new request, the cache would be checked whether the
search term was already available as a key. If the key is available, the request would be suppressed
and the cache result would be used instead. If the key is not available, a request would be made.
After the request succeeded, the search term would be saved as key and the search result would be
saved as value for the key in the local state.
The book doesn’t give you an in-depth implementation of the cache solution. If you did read the
Road to learn React37 , you will already know how to implement such a cache in plain React with
local state. In one of its lessons, the book uses a cache in a more elaborated way to cache paginated
search results efficiently in the local state.
37
https://www.robinwieruch.de/the-road-to-learn-react/
Local State Management 43
in a modern application. A user can interact with the View in order to trigger an Action. An Action
would encapsulate all the necessary information to update the state in the Store(s). The Dispatcher
on the way delegates the Actions to the Store(s). The updated state would be propagated to the Views
again to update them.
The data flow goes in one direction. A View can trigger an Action, that goes through the Dispatcher
and Store, and would change the View eventually when the state in the Store changed. The
unidirectional data flow is enclosed in this loop. Then again, a View can trigger another Action.
Since Facebook introduced the Flux architecture, the View was associated with React.
You can read more about the Flux architecture39 on the official website. There you will find a video
about its introduction at a conference40 too. If you are interested about the origins of Redux, I highly
recommend reading and watching the material.
After all, Redux became the successor library of the Flux architecture. Even though there were a
bunch of solutions around the Flux architecture, Redux managed to succeed. But why did it succeed?
Dan Abramov41 and Andrew Clark42 are the creators of Redux. It was introduced by Dan Abramov
at React Europe43 in 2015. However, the talk by Dan doesn’t introduce Redux per se. Instead, the
talk introduced a problem that Dan Abramov faced that led to implementing Redux. I don’t want
to foreclose the content of the talk, that’s why I encourage you to watch the video yourself. If you
are keen to learn Redux, you should dive into the problem that was solved by it.
Nevertheless, one year later, again at React Europe, Dan Abramov reflected on the journey of Redux
and its success. He mentioned a few things that had made Redux successful in his opinion.
Redux was developed to solve a problem. The problem was explained by Dan Abramov one year
earlier when he introduced Redux. It was not yet another library. It was a library that solved a
problem. Time Traveling and Hot Reloading were the stress test for Redux.
The constraints of Redux were another key factor to its success. Redux managed to shield away the
problem with a simple API and a thoughtful way to solve the problem of state management itself.
You can watch the talk44 too. I highly recommend it. Either you watch it right now or after the next
chapter that introduces you to the basics of Redux.
39
https://facebook.github.io/flux/
40
https://youtu.be/nYkdrAPrdcw?list=PLb0IAmt7-GS188xDYE-u1ShQmFFGbrk0v
41
https://twitter.com/dan_abramov
42
https://twitter.com/acdlite
43
https://www.youtube.com/watch?v=xsSnOQynTHs
44
https://www.youtube.com/watch?v=uvAXVMwHJXU
Redux 47
Basics in Redux
On the official Redux website45 it says: “Redux is a predictable state container for JavaScript apps.”.
It can be used standalone or in connection with with libraries, like React and Angular, to manage
state in JavaScript applications.
Redux adopted a handful of constraints from the Flux architecture but not all of them. It has Actions
that encapsulate information about the state update. It has a Store to save the state, too. However, the
Store is a singleton. Thus, there are not multiple Stores like there used to be in the Flux architecture.
In addition, there is no single Dispatcher. Instead, Redux uses multiple Reducers. Basically, Reducers
pick up the information from Actions and “reduce” it to a new state that is saved in the Store. When
state in the Store is changed, the View can act on this by subscribing to the Store.
Concept Playground
Why is it called Redux? Because it combines the two words Reducer and Flux. The abstract picture
should be imaginable now. The state doesn’t live in the View anymore, it is only connected to
the View. What does connected mean? It is connected on two ends, because it is part of the
unidirectional data flow. One end is responsible to trigger an Action to update the state, the second
end is responsible to receive the state from the Store. The View can update according to state changes
and can trigger state changes.
The View, in this case, would be React, but Redux could be used with any other library or standalone.
After all, it is only a state management container.
Action(s)
An action in Redux is a JavaScript object. It has a type and an optional payload. The type is often
referred to as action type. While the type is a string literal, the payload can be anything.
In the beginning, your playground to get to know Redux will be a Todo application. For instance,
the following action in this application can be used to add a new todo item:
45
http://redux.js.org/
Redux 48
Code Playground
{
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux', completed: false },
}
Executing an action is called to dispatch in Redux. You can dispatch an action to alter the state in
the Redux store. You only dispatch when you want to change the state. The dispatch of an action
can be triggered in your view layer. It could be as simple as a click on a button.
In addition, the payload in a Redux action is not mandatory. You can define actions that have only
an action type. That subject will be revisited later in the book.
So once an action is dispatched, it will come by all reducers in Redux.
Reducer(s)
A reducer is the next part in the chain of the unidirectional data flow. The view dispatches an action
and the action object, with action type and optional payload, will pass through all reducers.
What’s a reducer? A reducer is a pure function. It always produces the same output when the input
stays the same. It has no side-effects, thus it is only an input/output operation.
A reducer has two inputs: state and action. The state is always the whole state object from the Redux
store. The action is the dispatched action with a type and an optional payload. The reducer reduces
- that explains the naming - the previous state and incoming action to a new state.
Code Playground
Apart from the functional programming principle, namely that a reducer is a pure function without
side-effects, it also embraces immutable data structures. It always returns a newState object without
mutating the incoming state object. Thus, the following reducer, where the state of the Todo
application is a list of todos, is not an allowed reducer function:
Code Playground
function(state, action) {
return state.push(action.todo);
}
It would mutate the previous state instead of returning a new state object. The following is allowed
because it keeps the previous state intact:
Redux 49
Code Playground
By using the JavaScript built-in concat functionality46 , the state and thus the list of todos is
concatenated to another item. The other item is the newly added todo from the action. You might
wonder if this embraces immutability now. Yes it does, because concat always returns a new array
without mutating the old array. The data structure stays immutable. You will learn more about how
to keep your data structures immutable later on.
But what about the action type? Right now, only the payload is used to produce a new state but
the action type is ignored.
When an action object arrives at the reducers, the action type can be evaluated. Only when a reducer
cares about the action type, it will produce a new state. Otherwise, it simply returns the previous
state. In JavaScript, a switch case can help to evaluate different action types or to return the previous
state on default.
Imagine your Todo application would have a second action that toggles a Todo to either completed
or incomplete.
Code Playground
{
type: 'TODO_TOGGLE',
todo: { id: '0' },
}
The reducer would have to act on two actions now: TODO_ADD and ‘TODO_TOGGLE’. By using a
switch case statement, you can branch into different cases. If there is not such a case, you return the
unchanged state by default.
46
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Array/concat
Redux 50
Code Playground
The book already discussed the TODO_ADD action type and its functionality. It simply concats a new
todo item to the previous list of todo items. But what about the TODO_TOGGLE functionality?
Code Playground
In the example, the built-in JavaScript functionality map is used to map over the state, the list of
todos, to either return the intact todo or return the toggled todo. The toggled todo is identified by
its id.
The JavaScript built-in functionality map47 always returns a new array. It doesn’t mutate the
previous state and thus the state of todos stays immutable and can be returned as new state.
47
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Array/map
Redux 51
But isn’t the toggled todo mutated? No, because Object.assign() returns a new object without
mutating the old object. Object.assign() merges all given objects from the former to the latter into
each other. If a former object shares the same property as a latter object, the property of the latter
object will be used. Thus, the completed property of the updated todo item will be the negated state
of the old todo item.
Notice that these functionalities, actions and reducer, are plain JavaScript. There is no function from
the Redux library involved by now. There is no hidden library magic. It is plain JavaScript with
functional programming principles in mind.
One last basic step about the reducer: It has grown in size that makes it less maintainable. In order
to keep reducers tidy, most often the different switch case branches are extracted as pure functions:
Code Playground
The Todo application has two actions and one reducer by now. One last part in the Redux setup is
missing: the Store.
Redux 52
Store
So far, the Todo application has a way to trigger state updates (actions) and a way to reduce the
previous state and action to a new state (reducer). But no one is responsible to glue these parts
together.
It is the Redux store. The store holds one global state object. There are no multiple stores and no
multiple states. The store is only one instance in your application. In addition, it is the first library
dependency you encounter when using Redux.
In JavaScript ES6 you can use the import statement to get the functionality to create the store object.
Code Playground
Now you can use it to create a store singleton instance. The createStore function takes one
mandatory argument: a reducer. You already defined a reducer in the Reducer chapter which adds
and completes todo items.
Code Playground
In addition, the createStore takes a second optional argument: the initial state. In the case of the
Todo application, the reducer operated on a list of todos as state. The list of todo items should be
initialized as an empty array or pre-filled array with todos. If it wasn’t initialized, the reducer would
fail because it would operate on undefined property.
Code Playground
In another chapter, the book will showcase another way to initialize state. Therefore, you would use
the reducer instead of the store.
Now you have a store instance that knows about the reducer. The Redux setup is done. However,
now you want to interact with the store. You want to dispatch actions, get the state from the store
and listen to updates of the state in the store. So how to dispatch an action?
Redux 53
Code Playground
store.dispatch({
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux', completed: false },
});
Code Playground
store.getState();
How to subscribe (and unsubscribe) to the store in order to listen for updates?
Code Playground
That’s it. The Redux store has only a slim API to access the state, update it and listen for updates.
Concept Playground
Let’s apply these learnings. You can either use your own project where you have JavaScript,
JavaScript ES6 features enabled and Redux at your disposal. Or you can open up the following
JS Bin: Redux Playground48 .
Let’s apply the learnings about actions, reducers, and the store from the last chapter together. First,
you can define your reducer that deals with adding and toggling todo items:
48
https://jsbin.com/zukogaj/2/edit?html,js,console
Redux 54
Code Playground
Second, you can initialize the Redux store that uses the reducer and an initial state. In the JS Bin
you have Redux available as global variable.
Code Playground
If you are in your own project, you might be able to import the createStore:
Code Playground
Code Playground
store.dispatch({
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux', completed: false },
});
That’s it. You have set up all parts of Redux and interacted with it by using an action. You can
retrieve the state by getting it from the store.
Code Playground
console.log(store.getState());
But rather than outputting it manually, you can subscribe a callback function to the store to output
the latest state. Make sure to subscribe to the store before dispatching your actions in order to get
the output.
Code Playground
Now, whenever you dispatch an action, after the state was updated, the store subscription should
become active. Don’t forget to unsubscribe eventually.
Code Playground
unsubscribe();
Advanced Actions
You have learned about actions in a previous chapter. However, there are more fine grained details
that I want to cover in this chapter. The same applies for reducers. Both will be covered in the
following chapters.
Therefore, it would be a requirement that you feel confident with the learnings from the previous
chapter. Not all of the following learnings are mandatory to write applications in Redux, but they
teach best practices and common usage patterns. In an advanced application, you would want to
know about these topics.
Code Playground
{
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux', completed: false },
}
As you can see, the completed property is defined as false. In addition, you saw that the action and
reducer from the previous chapter did work under these circumstances. However, a rule of thumb
in Redux is to keep the action payload to a minimum.
In the example, when you want to add a todo in a Todo application, it would need at least the
unique identifier and a name of a todo. But the completed property is unnecessary. The assumption
is that every todo that is added to the store will be incomplete. It wouldn’t make sense in a puristic
Todo application to add completed todos, would it? Therefore, not the action would take care of the
property but the reducer.
Instead of simply passing the whole todo object into the list of todos in your reducer:
Code Playground
Code Playground
Code Playground
{
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux' },
}
Now, you only defined the necessary payload in the action. Nevertheless, if at some point the Todo
application decides to add uncompleted todos in the first place, you can add it in the action again
and leave it out in the reducer. Ultimately, keeping the payload in the action to a minimum is a best
practice in Redux.
Action Type
Actions get evaluated in reducers by action type. The action type is the glue between both parts even
though actions and reducers can be defined independently. To make the application more robust,
you should extract the action type as a variable. Otherwise, you can run into typos where an action
never reaches a reducer because you misspelled it.
Code Playground
const action = {
type: TODO_ADD,
todo: { id: '0', name: 'learn redux' },
};
const toggleTodoAction = {
type: TODO_TOGGLE,
todo: { id: '0' },
};
Redux 58
There is another benefit in extracting the action type as variable. Because action, reducer and action
type are loosely coupled, you can define them in separate files. You would only need to import the
action type to use them only in specific actions and reducers. After all, action types could be used
in multiple reducers. This use case will be covered in another chapter when it comes to advanced
reducers.
Action Creator
Action creators add another layer on top, which often leads to confusion when learning Redux.
Action creators are not mandatory, but they are convenient to use.
So far, you have dispatched an action as plain action object:
Code Playground
store.dispatch({
type: TODO_ADD,
todo: { id: '0', name: 'learn redux' },
});
Action creators encapsulate the action with its action type and optional payload in a reusable
function. In addition, they give you the flexibility to pass any payload. After all, they are only
pure functions which return an object.
Redux 59
Code Playground
Now, you can use it by invoking the function in your dispatch method:
Code Playground
Action creators return a plain action. Once again, it is not mandatory to use them, but it adds
convenience and makes your code more readable in the long run. In addition, you can test action
creators independently as functions. Last but not least, these action creators stay reusable because
they are functions.
Optional Payload
In the book, it was mentioned earlier that actions don’t need to have a payload. Only the action type
is required.
For instance, imagine you want to login into your Todo application. Therefore, you need to open up
a modal where you can enter your credentials: email and password. You wouldn’t need a payload
for your action in order to open a modal. You only need to signalize that the modal state should be
stored as open by dispatching an action.
Code Playground
{
type: 'LOGIN_MODAL_OPEN',
}
A reducer would take care of it and set the state of a isLoginModalOpen property to true. While it
is good to know that the payload is not mandatory in actions, the last example can lead to a bad
practice. Because you already know that you would need a second action to close the modal again.
Redux 60
Code Playground
{
type: 'LOGIN_MODAL_CLOSE',
}
A reducer would set the isLoginModalOpen property in the state to false. That’s verbose, because
you already need two actions to alter only one property in the state.
By planning your actions thoughtfully, you avoid these bad practices and keep your actions on
a higher level of abstraction. If you used the optional payload for the action, you could solve
login scenario in only one action instead of two actions. The isLoginModalOpen property would
be dynamically passed in the action rather than being hardcoded in a reducer.
Code Playground
{
type: 'LOGIN_MODAL_TOGGLE',
isLoginModalOpen: true,
}
By using an action creator, the payload can be passed in as arguments and thus stays flexible.
Code Playground
function doToggleLoginModal(open) {
return {
type: 'LOGIN_MODAL_TOGGLE',
isLoginModalOpen: open,
};
}
In Redux, actions should always try to stay on an abstract level rather than on a concrete level.
Otherwise, you will end up with duplications and verbose actions. However, don’t worry too much
about it for now. This will be explained in more detail in another chapter in this book that is about
commands and events.
Payload Structure
Again you will encounter a best practice that is not mandatory in Redux. So far, the payload was
dumped without much thought in the actions. Now imagine an action that has a larger payload than
a simple todo. For instance, the action payload should clarify to whom the todo is assigned.
Redux 61
Code Playground
{
type: 'TODO_ADD_ASSIGNED',
todo: { id: '0', name: 'learn redux' },
assignedTo: { id: '99' name: 'Robin' },
}
The properties would add up horizontally, but mask the one most important property: the type.
Therefore, you should treat action type and payload on the same level, but nest the payload itself
one level deeper as the two abstract properties.
Code Playground
{
type: 'TODO_ADD_ASSIGNED',
payload: {
todo: { id: '0', name: 'learn redux' },
assignedTo: { id: '99' name: 'Robin' },
},
}
The refactoring ensures that type and payload are visible on first glance. As said, it is not mandatory
to do so and often adds more complexity. But in larger applications it can keep your action creators
readable.
50
https://jsbin.com/kopohur/28/edit?html,js,console
Redux 62
Code Playground
...
store.dispatch({
type: 'TODO_ADD',
todo: { id: '0', name: 'learn redux' },
});
store.dispatch({
type: 'TODO_ADD',
todo: { id: '1', name: 'learn mobx' },
});
The next step is the extraction of the action type from the actions and reducer. It should be defined
as a variable and can be replaced in the reducer.
Code Playground
Code Playground
store.dispatch({
type: TODO_ADD,
todo: { id: '0', name: 'learn redux' },
});
store.dispatch({
type: 'TODO_ADD',
todo: { id: '1', name: 'learn mobx' },
});
store.dispatch({
type: TODO_TOGGLE,
todo: { id: '0' },
});
In the next step, you can introduce action creators to your Todo application. First, you can define
them:
Code Playground
function doToggleTodo(id) {
return {
type: TODO_TOGGLE,
todo: { id },
};
}
Code Playground
There were two more advanced topics about actions in this chapter: optional payload and payload
structure. The first topic wouldn’t apply in the current application. Every action has to have a
payload. The second topic could be applied. However, the payload is small and thus doesn’t need to
be deeply restructured with a payload property.
The final Todo application can be found in this JS Bin51 . You can do further experiments with it
before you continue with the next chapter.
51
https://jsbin.com/kopohur/29/edit?html,js,console
Redux 65
Advanced Reducers
Apart from the advanced topics about actions, there is more to know about reducers, too. Again, not
everything is mandatory, but you should at least know about the following things to embrace best
practices, common usage patterns and practices on scaling your state architecture.
Initial State
So far, you have provided your store with an initial state. It was an empty list of todos.
Code Playground
const store = createStore(reducer, []);
That’s the initial state for the whole Redux store. However, you can apply the initial state on a more
fine-grained level. Before you dispatch your first action, the Redux store will initialize by running
through all reducers once. You can try it by removing all of your dispatches in the editor and add a
console.log() in the reducer. You will see that it runs with an initializing action once, even though
there is no dispatched action.
The initializing action, that is received in the reducer, is accompanied by the initial state that is
specified in the createStore() function. However, if you leave out the initial state in the store
initialization, the incoming state in the reducer will be undefined.
Code Playground
const store = createStore(reducer);
That’s where you can opt-in to specify initial state on a fine-grained level. If the incoming state is
undefined, you can default with a JavaScript ES6 default parameter52 to a default initial state.
Code Playground
function reducer(state = [], action) {
switch(action.type) {
case TODO_ADD : {
return applyAddTodo(state, action);
}
case TODO_TOGGLE : {
return applyToggleTodo(state, action);
}
default : return state;
}
}
52
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Functions/Default_parameters
Redux 66
The initial state will be the same as before, but you defined it on a more fine-grained level. Later
on, that will help you when you specify more than one reducer and your state object becomes more
complex.
Code Playground
const initialState = {
todos: [],
};
const store = createStore(reducer, initialState);
Now you can use the space horizontally in your initialState object. It might grow to the following
at some point:
Code Playground
const initialState = {
currentUser: null,
todos: [],
filter: 'SHOW_ALL',
};
If you get back to your Todo application, you will have to adjust your reducer. The reducer deals
with a todos list as state, but now it is a complex global state object.
Redux 67
Code Playground
function reducer(state, action) {
switch(action.type) {
case TODO_ADD : {
return applyAddTodo(state, action);
}
case TODO_TOGGLE : {
return applyToggleTodo(state, action);
}
default : return state;
}
}
Those nested data structures are fine in Redux, but you want to avoid deeply nested data
structures. As you can see, it adds complexity to create your new state object. Another chapter
later on in the book will pick up this topic. It will showcase how you can avoid deeply nested data
structures by using a neat helper library with the name normalizr.
Combined Reducer
The next chapter is crucial to understand the principles of scaling state by using substates in Redux.
You heard about multiple reducers, but haven’t used them yet. A reducer can grow horizontally by
using action types, but it doesn’t scale at some point. You want to split it up into two reducers or
introduce another reducer right from the beginning. Imagine you had a reducer that served your
todos, but you want to have a reducer for the filter state for your todos eventually. Another use
case could be to have a reducer for the currentUser that is logged in into your Todo application.
Redux 68
You can already see a pattern on how to separate your reducers. It is usually by domain like todos,
filter, or user. A todo reducer might be responsible to add, remove, edit and complete todos. A
filter reducer is responsible to manage the filter state. A user reducer cares about user entities that
could be the currentUser who is logged in in your application or a list of users who are assigned
to todos. That’s where you could again split up the user reducer to a currentUser reducer and
a assignedUsers reducer. You can imagine how this approach, introducing reducers by domain,
scales very well.
Let’s enter combined reducers to enable you using multiple reducers. Redux gives you a helper to
combine multiple reducers into one root reducer: combineReducers(). The function takes an object
as input that has a state as property name and a reducer as value.
Code Playground
Afterward, the rootReducer can be used to initialize the Redux store instead of the single todo
reducer.
Code Playground
That’s it for the initialization of the Redux store with combined reducers. But what about the
reducers themselves? There is already one reducer that cares about the todos.
Code Playground
The filterReducer is not there yet. It would look something like the following:
Redux 69
Code Playground
Now, there comes the important clue. The combineReducers() function introduces an intermediate
state layer for the global state object. The global state object, when using the combined reducers as
shown before, would look like the following:
Code Playground
{
todoState: ...,
filterState: ...,
}
The property keys for the intermediate layer are those defined in the combineReducers() function.
However, in your reducers the incoming state is not the global state object anymore. It is their
defined substate from the combinedReducers() function. The todoReducer doesn’t know anything
about the filterState and the filterReducer doesn’t know anything about the todoState.
The filterReducer and todoReducer can use the JavaScript ES6 default parameter to define their
initial state.
Code Playground
Now your apply function in the reducers have to operate on these substates again.
Code Playground
This chapter seems like a lot of hassle. But it is crucial knowledge to split up your global state into
substates. These substates can be managed by their reducers who only operate on these substates.
In addition, each reducer can be responsible to define the initial substate.
One plain reducer: When using only one plain reducer, the initial state in createStore() dominates
the initial state in the reducer. The initial state in the reducer only works when the incoming initial
state is undefined because then it can apply a default state. But the initial state is already defined
in createStore() and thus utilized by the reducer.
Combined reducers: When using combined reducers, you can embrace a more nuanced usage of
the state initialization. The initial state object that is used for the createStore() function doesn’t
have to include all substates that are introduced by the combineReducers() function. Thus, when a
substate is undefined, the reducer can define the default substate. Otherwise, the default substate
from the createStore() is used.
Nested Reducers
By now, you know two things about scaling reducers in a growing application that demand
sophisticated state management:
These steps are used to scale the reducers horizontally (even though combined reducers add at least
one vertical level). A reducer operates on the global state or on a substate when using combined
reducers. However, you can use nested reducers to introduce vertically clearer levels of substate.
Take for example the todoReducer that operates on a list of todos. From a technical perspective, a list
of todos has todo entities. So why not introduce a nested reducer that deals with the todo substate
as entities?
Code Playground
switch(action.type) {
case TODO_ADD : {
return applyAddTodo(state, action);
}
case TODO_TOGGLE : {
return applyToggleTodo(state, action);
}
default : return state;
}
}
You can use nested reducers to introduce clearer boundaries in substates. In addition, they can be
reused. You might run into cases where you can reuse a nested reducer somewhere else.
While nested reducers can give you a better picture on your state, they can add more levels of
complexity for your state, too. You should follow the practice of not nesting your state too deeply
in the first place. Then you won’t run into nested reducers often.
53
https://jsbin.com/kopohur/29/edit?html,js,console
Redux 73
Code Playground
The reducer has an initial state. There is no action yet that serves the reducer though. You can add
a simple action creator that only gets a string as filter. You will experience later on, how this can be
used in a real application.
Code Playground
function doSetFilter(filter) {
return {
type: FILTER_SET,
filter,
};
}
Second, you can rename your first reducer to todoReducer and give it an initial state of an empty
list of todos.
Code Playground
The initial state isn’t initialized in the createStore() function. It is initialized on a more fine-grained
level in the reducers. When you recap the last lessons learned from the advanced reducers chapter,
you will notice that you spared the back and forth with the initial state. Now, the todoReducer still
operates on the todos substate and the new filterReducer operates on the filter substate. As third
and last step you you have to combine both reducers to get this intermediate layer of substates.
In the JS Bin you have Redux available as global variable to get the combineReducer function.
Otherwise, you could import it with JavaScript ES6.
Code Playground
Code Playground
When you run your application again, everything should work. You can now use the filterReducer
as well.
Code Playground
The Todo application favors initial state in reducers over initial state in createStore(). In addition,
it will not use a nested todoEntityReducer for the sake of keeping the reducer hierarchy simple for
now. The nested data structures are achieved implicitly by using combined reducers.
The final Todo application can be found in this JS Bin54 . You can do further experiments with it
before continuing with the next chapter.
54
https://jsbin.com/kopohur/30/edit?html,js,console
Redux in React
In the last chapters, you got to know plain Redux. It helps you to manage a predictable state object.
However, you want to use this state object in an application eventually. It can be any JavaScript
application that has to deal with state management. The principle of Redux could be deployed to
any programming language to manage a state object.
State management in single page applications (SPAs) is one of these use cases where Redux can
be applied. These applications are usually built with a framework (Angular) or view layer library
(React, Vue), but most often these solutions lack of a sophisticated state management solution. That’s
where Redux comes into play. The book focuses on React, but you can apply the learnings to other
solutions, such as Angular and Vue, too.
The following scenarios could live without Redux in the first place, because they wouldn’t run
into state management issues with using only local state. But for the sake of demonstrating Redux
in React, they will omit local state management and apply sophisticated state management with
Redux.
Redux in React 76
Concept Playground
How can dispatch(), subscribe() and getState() be accessed in a React view layer? Basically,
the view layer has to be able to dispatch actions on the one end, while it has to listen to updates
from the store, in order to update itself, on the other end. All three functionalities are accessible as
methods on the Redux store.
Command Line
Now you can bootstrap your React application with create-react-app, navigate into the folder, and
start it:
Redux in React 77
Command Line
create-react-app taming-the-state-todo-app
cd taming-the-state-todo-app
npm start
If you haven’t used create-react-app before, I recommend you to read up the basics in the official
documentation55 . Basically, your src/ folder has several files. You will not use the src/App.js file in
this application, but only the src/index.js file. Open up your editor and adjust your src/index.js file
to the following:
src/index.js
function TodoApp() {
return <div>Todo App</div>;
}
Now, when you start your application again with npm start, you should see the displayed “Todo
App” string from the TodoAddcomponent. Before you continue to build a React application now, let’s
hook in all of the Redux code that you have written in the previous chapters. First, install Redux in
your application.
Command Line: /
Second, re-use the Redux code from the previous chapters in your src/index.js file. You start at the top
to import the two Redux functionalities that you have used so far. They belong next to the imports
that are already there:
55
https://github.com/facebookincubator/create-react-app
Redux in React 78
src/index.js
Now, in between of your imports and your React code, you introduce your Redux functionalities.
First, the action types:
src/index.js
// action types
src/index.js
// reducers
const todos = [
{ id: '0', name: 'learn redux' },
{ id: '1', name: 'learn mobx' },
];
return state.concat(todo);
}
src/index.js
// action creators
function doToggleTodo(id) {
return {
type: TODO_TOGGLE,
todo: { id },
};
}
Redux in React 80
function doSetFilter(filter) {
return {
type: FILTER_SET,
filter,
};
}
And last but not least, the creation of the store with the combined reducers:
src/index.js
// store
After that, your React code follows. It should already be there in the same file.
src/index.js
// view layer
function TodoApp() {
return <div>Todo App</div>;
}
The bootstrapping is done. The bootstrapped application can be found in a GitHub repository56 . Now
you have a running React application and a Redux store. But they don’t work together yet. The next
step is to wire both together.
56
https://github.com/rwieruch/taming-the-state-todo-app/tree/0.0.0
Redux in React 81
src/index.js
src/index.js
src/index.js
Notice that none of these components is aware of Redux. They simply display todos and use a
callback function to toggle todo items. Now, in the last step, you wire together Redux and React.
You can use the initialized store in your root component where React hooks into HTML.
src/index.js
ReactDOM.render(
<TodoApp
todos={store.getState().todoState}
onToggleTodo={id => store.dispatch(doToggleTodo(id))}
/>,
document.getElementById('root')
);
The store does two things: it makes state accessible and exposes functionalities to alter the state.
The todos props are passed down to the TodoApp by retrieving them from the store. In addition, a
onToggleTodo property is passed down which is a function. This function is a higher order function
that wraps the dispatching of an action that is created by its action creator. However, the TodoApp
component is completely unaware of the todos being retrieved from the Redux store or of the
onToggleTodo() being a dispatched action on the Redux store. These passed properties are simple
props for the TodoApp. You can start your application again with npm start. The todos should be
displayed but not updated yet.
What about the update mechanism? When an action is dispatched, someone needs to subscribe to
the Redux store. In a naive approach, you can do the following to force a view update. First, wrap
your React root into a function.
Redux in React 83
src/index.js
function render() {
ReactDOM.render(
<TodoApp
todos={store.getState().todoState}
onToggleTodo={id => store.dispatch(doToggleTodo(id))}
/>,
document.getElementById('root')
);
}
Second, you can pass the function to the subscribe() method of the Redux store. And last but not
least, you have to invoke the function one time for the initial render of your component tree.
src/index.js
function render() {
ReactDOM.render(
<TodoApp
todos={store.getState().todoState}
onToggleTodo={id => store.dispatch(doToggleTodo(id))}
/>,
document.getElementById('root')
);
}
store.subscribe(render);
render();
The Todo application should display the todos and update the completed state once you toggle it.
The final application of this approach can be found in a GitHub repository57 .
The approach showcased how you can wire up your React component tree with the Redux store. The
components don’t need to be aware of the Redux store at all, but the root component is. In addition,
everything is re-rendered when the global state in the Redux store updates.
Even though the previous approach is pragmatic and shows a simplified version of how to wire
up all these things, it is naive. Why is that? In a real application you want to avoid the following
practices:
57
https://github.com/rwieruch/taming-the-state-todo-app/tree/1.0.1
Redux in React 84
• re-render the whole component tree: You want to re-render only the components that are
affected by the global state updated. Otherwise, you will run into performance issues in a
scaling application.
• using the store instance: You want to avoid to operate directly on the Redux store instance.
The store should be injected somehow into your component tree to make it accessible for
components that need to have access to the store.
• making the store globally available: The store shouldn’t be globally accessible by every
component. In the previous example only the React root component uses it, but who prevents
you from using it directly in your TodoItem component to dispatch an action?
Fortunately, there exists a library that takes care of these things and gives you a bridge from the
Redux to the React world.
Redux in React 85
Code Playground
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
document.getElementById('root')
);
After you have done this, every child component in the whole component tree has an implicit access
to the store. Thus, every component is able to dispatch actions and to listen to updates in order to
re-render. But not every component has to listen to updates. How does this work without passing
the store as props to each child component? It uses the provider pattern that you got to know in a
previous chapter. Under the hood it uses the React context API:
“In some cases, you want to pass data through the component tree without having to pass the props
down manually at every level. You can do this directly in React with the powerful “context” API.”
That was part one to use Redux in React. Second, you can use a higher component that is called
connect from the react-redux library. It makes the Redux store functionality dispatch and the state
from the store itself available to the enhanced component.
Code Playground
function Component(props) {
...
}
Code Playground
connect([mapStateToProps], [mapDispatchToProps], [mergeProps], [options])(...);
Usually, you will only use two of them: mapStateToProps() and mapDispatchToProps(). You will
learn about the other two arguments, mergeProps() and options, later in this book.
mapStateToProps(state, [props]) ⇒ derivedProps: It is a function that can be passed to the
connect HOC. If it is passed, the input component of the connect HOC will subscribe to updates
from the Redux store. Thus, it means that every time the store subscription notices an update, the
mapStateToProps() function will run. The mapStateToProps() function itself has two arguments in
its function signature: the global state object and optionally the props from the parent component.
The function returns an object that is derived from the global state and optionally from the props
from the parent component. The returned object will be merged into the remaining props that come
as input in the connected component when it is used.
mapDispatchToProps(dispatch, [props]): It is a function (or object) that can be passed to the
connect HOC. Whereas mapStateToProps() gives access to the global state, mapDispatchToProps()
gives access to the dispatch method of the store. It makes it possible to dispatch actions but passes
down only plain functions that wire up the dispatching in a higher order function. After all, it
makes it possible to pass functions down to the input component of the connect HOC to alter the
state. Optionally, you can use the incoming props to wrap those into the dispatched action.
That is a lot of knowledge to digest. Both functions, mapStateToProps() and mapDispatchToProps(),
can be intimidating at the beginning. In addition, they are used in a foreign higher order component.
However, they only give you access to the state and to the dispatch method of the store.
Concept Playground
View -> (mapDispatchToProps) -> Action -> Reducer(s) -> Store -> (mapStateToProp\
s) -> View
You will see in the following examples that these functions don’t need to be intimidating at all.
Command Line: /
npm install --save react-redux
Second, instead of wrapping the React root component into the render() function and subscribing
it to the store.subscribe() method, you will use the plain React root component again but use the
Provider component given by react-redux.
Redux in React 87
src/index.js
...
ReactDOM.render(
<Provider store={store}>
<TodoApp />
</Provider>,
document.getElementById('root')
);
It uses the plain TodoApp component. The component still expects todos and onToggleTodo as props.
But it doesn’t have these props. Let’s use the connect higher order component to expose these to
the TodoApp component. The TodoApp component will become a connected TodoApp component.
src/index.js
...
ReactDOM.render(
<Provider store={store}>
<ConnectedTodoApp />
</Provider>,
document.getElementById('root')
);
Now, only the connections, mapStateToProps() and mapDispatchToProps() are missing. They are
quite similar to the naive React with Redux version.
Redux in React 88
src/index.js
function mapStateToProps(state) {
return {
todos: state.todoState,
};
}
function mapDispatchToProps(dispatch) {
return {
onToggleTodo: id => dispatch(doToggleTodo(id)),
};
}
...
59
https://github.com/rwieruch/taming-the-state-todo-app/tree/2.0.0
Redux in React 89
src/index.js
const ConnectedTodoList = connect(mapStateToProps)(TodoList);
const ConnectedTodoItem = connect(null, mapDispatchToProps)(TodoItem);
ReactDOM.render(
<Provider store={store}>
<TodoApp />
</Provider>,
document.getElementById('root')
);
Now you wouldn’t need to pass the onToggleTodo() props through the TodoApp component and
TodoList component anymore. The same applies for the todos that don’t need to get passed through
the TodoApp component.
src/index.js
function TodoApp() {
return <ConnectedTodoList />;
}
Middleware in Redux
In Redux, you can use a middleware. Every dispatched action in Redux flows through this
middleware. You can opt-in a specific feature in between dispatching an action and the moment
it reaches the reducer.
There are useful libraries out there to opt-in feature into your Redux middleware. In the following,
you will get to know one of them: redux-logger61 . When you use it, it doesn’t change anything in
your application in production. But it will make your life easier as developer when dealing with
Redux in development. What does it do? It simply logs the actions in your developer console with
console.log(). As a developer, it gives you clarity on which action is dispatched and how the
previous and the new state are structured.
But where to apply the middleware? The stores can be initialized with a middleware. The
createStore() functionality from Redux takes as third argument an enhancer. The redux library
comes with one of these enhancers: applyMiddleware().
Code Playground
Now, when using redux-logger, you can pass a logger instance to the applyMiddleware() function.
Code Playground
61
https://github.com/evgenyrodionov/redux-logger
Redux State Structure and Retrieval 92
That’s it. Now every action should be visible in your developer console when dispatching them. You
don’t have to take care of this anymore. Your state changes become more predictable as developer.
The applyMiddleware() functionality takes any number of middleware: applyMiddleware(firstMiddleware,
secondMiddleware, ...);. The action will flow through all middleware before it reaches the
reducers. Sometimes, you have to make sure to apply them in the correct order. For instance, the
redux-logger middleware must be last in the middleware chain in order to output the correct actions
and states.
Nevertheless, that’s only the redux-logger middleware. On your way to implement Redux appli-
cations, you will surely find out more about useful features that can be used with a middleware.
Most of these features are already taken care of in libraries that you will find published on npm.
For instance, asynchronous actions in Redux are possible by using the Redux middleware. These
asynchronous actions will be explained later in this book.
Redux State Structure and Retrieval 93
Immutable State
Redux embraces an immutable state. Your reducers will always return a new state object. You will
never mutate the incoming state. Therefore, you might have to get used to different JavaScript
functionalities and syntax to embrace immutable data structures.
So far, you have used built-in JavaScript functionalities to keep your data structures immutable.
Such as array.map() and array.concat(item) for arrays or Object.assign() for objects. All of
these functionalities return new instances of arrays or objects. Often, you have to read the official
JavaScript documentation to make sure that they return a new instance of the array or object.
Otherwise, you would violate the constraints of Redux because you would mutate the previous
instance.
But it doesn’t end here. You should know about your tools to keep data structures immutable
in JavaScript. There are a handful of third-party libraries that can support you in keeping them
immutable.
• immutable.js62
• mori.js63
• seamless-immutable.js64
• baobab.js65
But they come with three drawbacks. First, they add another layer of complexity to your application.
Second, you have to learn yet another library. And third, you have to dehydrate and rehydrate your
data, because most of these libraries wrap your vanilla JavaScript objects and arrays into a library
specific immutable data object and array. It is an immutable data object/array in your Redux store,
but once you want to use it in React you have to transform it into a plain JavaScript object/array.
Personally I would recommend to use such libraries only in two scenarios:
• You are not comfortable to keep your data structures immutable with JavaScript ES5,
JavaScript ES6 and beyond.
• You want to improve the performance of immutable data structures when using huge amounts
of data.
If both statements are false, I would advice you to stick to plain JavaScript. As you have seen, the
built-in JavaScript functionalities already help a lot. In JavaScript ES6 and beyond you get one more
functionality to keep your data structures immutable: spread operators66 . Spreading an object or
array into a new object or new array always gives you a new object or new array.
Do you recall how you added a new todo item or how you completed a todo item in your reducers?
62
https://github.com/facebook/immutable-js
63
https://github.com/swannodette/mori
64
https://github.com/rtfeldman/seamless-immutable
65
https://github.com/Yomguithereal/baobab
66
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Operators/Spread_operator
Redux State Structure and Retrieval 94
Code Playground
// adding todo
const todo = Object.assign({}, action.todo, { completed: false });
const newTodos = todos.concat(todo);
// toggling todo
const newTodos = todos.map(todo =>
todo.id === action.todo.id
? Object.assign({}, todo, { completed: !todo.completed })
: todo
);
If you added more JavaScript ES6, you could keep it even more concise.
Code Playground
// adding todo
const todo = { ...action.todo, completed: false };
const newTodos = [ ...todos, todo ];
// toggling todo
const newTodos = todos.map(todo =>
todo.id === action.todo.id
? { ...todo, completed: !todo.completed }
: todo
);
JavaScript gives you enough tools to keep your data structures immutable. There is no need to use
a third-party library except for the two mentioned use cases. However, there might be a third use
case where such library would help: deeply nested data structures in Redux that need to be kept
immutable. It is true that it becomes more difficult to keep data structures immutable when they are
deeply nested. But, as mentioned earlier in the book, it is bad practice to have deeply nested data
structures in Redux in the first place. That’s where the next chapter of the book comes into play that
can be used to keep your data structures flat.
Redux State Structure and Retrieval 95
Normalized State
A best practice in Redux is a flat state structure. You don’t want to maintain an immutable state
when it is deeply nested. It becomes tedious and unreadable even with spread operators. But often
you don’t have control over your data structure, because it comes from a backend application via
its API. When having a deeply nested data structure, you have two options:
You should prefer the second option. You only deal with the problem once and all subsequent parts in
your application will be grateful for it. Let’s run through one scenario to illustrate the normalization
of data. Imagine you have a deeply nested data structure:
Code Playground
const todos = [
{
id: '0',
name: 'create redux',
completed: true,
assignedTo: {
id: '99',
name: 'Dan Abramov',
},
},
{
id: '1',
name: 'create mobx',
completed: true,
assignedTo: {
id: '77',
name: 'Michel Weststrate',
},
}
];
Both library creators, Dan Abramov and Michel Weststrate, did a great job: they created popular
libraries for state management. The first option, as mentioned, would be to save the todos as they
are in the store. The todos themselves would have the deeply nested information of the assigned
user. Now, if an action wanted to correct an assigned user, the reducer would have to deal with the
deeply nested data structure. Let’s add Andrew Clark to the creators of Redux.
Redux State Structure and Retrieval 96
Code Playground
store.dispatch({
type: ASSIGNED_TO_CHANGE,
payload: {
todoId: '0',
name: 'Dan Abramov and Andrew Clark',
},
});
...
The further you have to reach into a deeply nested data structure, the more you have to be careful
to keep your data structure immutable. Each level of nested data adds more tedious work of
maintaining it. Therefore, you should use a library called normalizr67 . The library uses schema
definitions to transform deeply nested data structures into dictionaries that have entities and a
corresponding list of ids.
67
https://github.com/paularmstrong/normalizr
Redux State Structure and Retrieval 97
What would that look like? Let’s take the previous list of todo items as example. First, you would
have to define schemas for your entities only once:
Code Playground
Code Playground
Code Playground
{
entities: {
assignedTo: {
77: {
id: "77",
name: "Michel Weststrate"
},
99: {
id: "99",
name: "Dan Abramov"
}
},
Redux State Structure and Retrieval 98
todo: {
0: {
assignedTo: "99",
completed: true,
id: "0",
name: "create redux"
},
1: {
assignedTo: "77",
completed: true,
id: "1",
name: "create mobx"
}
}
},
result: ["0", "1"]
}
The deeply nested data became a flat data structure grouped by entities. Each entity can reference
another entity by its id. Now you could have one reducer that stores and deals with the assignedTo
entities and one reducer that deals with the todoentities. The data structure is flat and grouped by
entities and thus easier to access and to manipulate. The basic approach would be to normalize data
structure first before you send it as a payload in a action to their reducers.
There is another benefit in normalizing your data. When your data is denormalized, entities are
often duplicated in your nested data structure. Imagine the following object with todos.
Code Playground
const todos = [
{
id: '0',
name: 'write a book',
completed: true,
assignedTo: {
id: '55',
name: 'Robin Wieruch',
},
},
{
id: '1',
name: 'self-publish a book',
completed: true,
Redux State Structure and Retrieval 99
assignedTo: {
id: '55',
name: 'Robin Wieruch',
},
}
];
If you store such denormalized data in your Redux store, you will likely run into an issue. What’s
the problem with the denormalized data structure? Imagine you want to update the name property
Robin Wieruch of all assignedTo properties. You would have to run through all todos in order to
update all assignedTo properties with the id 55. The problem: There is no single source of truth.
You will likely forget to update an entity and run into a stale state eventually. Therefore, the best
practice is to store your state normalized so that each entity can be a single source of truth. There
will be no duplication of entities and thus no stale state when updating the single source of truth.
Each todo will reference to the updated assignedTo entity:
Code Playground
{
entities: {
assignedTo: {
55: {
id: "55",
name: "Robin Wieruch"
}
},
todo: {
0: {
assignedTo: "55",
completed: true,
id: "0",
name: "write a book"
},
1: {
assignedTo: "55",
completed: true,
id: "1",
name: "call it taming the state"
}
}
},
result: ["0", "1"]
}
Redux State Structure and Retrieval 100
In conclusion, normalizing your state has two benefits. It keeps your state flat and thus easier
manageable with immutable data structures. In addition, it groups entities to single sources of truth
without any duplications. When you normalize your state, you will automatically get groupings of
entities that could lead to their own reducers managing them.
There exists yet another benefit when normalizing your state. It is about denormalization: how do
component retrieve the normalized state? You will learn it in the next chapter.
Redux State Structure and Retrieval 101
Selectors
In Redux, there is the concept of selectors to retrieve derived properties from your state. A selector
is a function that takes the state as argument and returns a substate or derived properties of it. It can
be that they only return a substate of your global state or that they already preprocess your state to
return derived properties. The function can take optional arguments to support the selection of the
derived properties.
Plain Selectors
Selectors usually follow the same syntax. The mandatory argument of a selector is the state from
where it has to select from. There can be optional arguments that are in a supportive role to select
the substate or the derived properties.
Code Playground
Selectors are not mandatory. When thinking about all the parts in Redux, only the action(s), the
reducer(s) and the Redux store are a binding requirement. Similar to action creators, selectors can
be used to achieve an improved developer experience in a Redux architecture. What does a selector
look like? It is a plain function, as mentioned, that could live anywhere. However, you would use it,
when using Redux in React, in your mapStateToProps() function.
Instead of retrieving the state explicitly:
Code Playground
function mapStateToProps(state) {
return {
todos: state.todoState,
};
}
Code Playground
function getTodos(state) {
return state.todoState;
}
function mapStateToProps(state) {
return {
todos: getTodos(state),
};
}
It is similar to the action and reducer concept. Instead of manipulating the state directly in the Redux
store, you will use action(s) and reducer(s) to alter it indirectly. The same applies for selectors that
don’t retrieve the derived properties directly but indirectly from the global state.
Why is that an advantage? There are several benefits. A selector can be reused. You will run into
cases where you select the derived properties more often. That’s always a good sign to use a function
in the first place. In addition, selectors can be tested separately. They are pure functions and thus
an easily testable part in the architecture. Last but not least, deriving properties from state can
become a complex undertaking in a scaling application. As mentioned, a selector could get optional
arguments to derive more sophisticated properties from the state. The selector function itself would
become more complex, but it would be encapsulated in one function rather than, for instance in
React and Redux, in a mapStateToProps() function.
Code Playground
{
entities: ...
ids: ...
}
For instance, in a real work application it would look like the following:
Redux State Structure and Retrieval 103
Code Playground
// state
[
{ id: '0', name: 'learn redux' },
{ id: '1', name: 'learn mobx' },
]
// normalized state
{
entities: {
0: {
id: '0',
name: 'learn redux',
},
1: {
id: '1',
name: 'learn redux',
},
},
ids: ['0', '1'],
}
If you recall the Redux in React chapter, there you passed the list of todos from the TodoList
component, because it is a connected component, down to the whole component tree. How would
you solve this with the normalized state from above?
Assuming that the reducer stored the state in a normalized immutable data structure, you would
only pass the list of todo ids to your TodoList component. Because this component manages the
list and not the entities themselves, it makes perfect sense that it only gets the list with references
to the entities.
Code Playground
function getTodosAsIds(state) {
return state.todoState.ids;
}
function mapStateToProps(state) {
return {
todosAsIds: getTodosAsIds(state),
};
}
Now the ConnectedTodoItem component, that already passes the onToggleTodo() handler via the
mapDispatchToProps() function to its plain TodoItem component, would select the todo entity
matching to the incoming todoId property.
Code Playground
function getTodoAsEntity(state, id) {
return state.todoState.entities[id];
}
function mapDispatchToProps(dispatch) {
return {
onToggleTodo: id => dispatch(doToggleTodo(id)),
};
}
The TodoItem component itself would stay the same. It still gets the todo item and the onTog-
gleTodo() handler as arguments. In addition, you can see two more concepts that were explained
earlier. First, the selector grows in complexity because it gets optional arguments to select derived
properties from the state. Second, the mapStateToProps() function makes use of the incoming props
from the TodoList component that uses the ConnectedTodoItem component.
Redux State Structure and Retrieval 105
As you can see, the normalized state requires to use more connected components. More components
are responsible to select their needed derived properties. But in a growing application, following this
pattern can make it easier to reason about it. You only pass properties that are really necessary to
the component. In the last case, the TodoList component only cares about a list of references yet
the TodoItem component itself cares about the entity that is selected by using the reference passed
down by the TodoList component.
There exists another way to denormalize your normalized state when using normalizr. The previous
scenario allowed you to only pass the minimum of properties to the components. Each component
was responsible to select its state. In this scenario, you will denormalize your state in one component
while the other components don’t need to care about it. You will use the defined schema again to
reverse the normalization.
Code Playground
import { denormalize, schema } from 'normalizr';
function getTodos(state) {
const entities = state.todoState.entities;
const ids = state.todoState.ids;
return denormalize(ids, [ todoSchema ], entities);
}
function mapStateToProps(state) {
return {
todos: getTodos(state),
};
}
In this scenario, the whole normalized data structure gets denormalized in the selector. You will
have the whole list of todos in your TodoList component. The TodoItem component wouldn’t need
to take care about the denormalization.
Reselect
When using selectors in a scaling application, you should consider a library called reselect68 that
provides you with advanced selectors. Basically, it uses the same concept of plain selectors as you
have learned before. But it comes with two improvements.
A plain selector has one constraint:
• “Selectors can compute derived data, allowing Redux to store the minimal possible state.”
There are two more constraints when using selectors from the reselect library:
• “Selectors are efficient. A selector is not recomputed unless one of its arguments change.”
• “Selectors are composable. They can be used as input to other selectors.”
Selectors are pure functions without any side-effects. The output doesn’t change when the input
stays the same. Therefore, when a function is called twice and its arguments didn’t change, it returns
the same output. This proposition is used in reselect’s selectors. It is called memoization. A selector
doesn’t need to compute everything again when its input didn’t change. It will simply return the
same output. In a scaling application this can have a performance impacts.
Another benefit, when using reselect, is the ability to compose selectors. It supports the case of
implementing reusable selectors that only solve one problem. Afterward they can be composed in a
functional programming style.
The book will not dive deeper into reselect. When learning Redux it is good to know about these
advanced selectors, but you are fine by using plain selectors in the beginning. If you cannot stay put,
you can read up the example usages in the official GitHub repository69 and apply in your projects
while reading the book.
68
https://github.com/reactjs/reselect
69
https://github.com/reactjs/reselect
Redux State Structure and Retrieval 107
Command Line: /
src/index.js
...
// store
70
https://github.com/rwieruch/taming-the-state-todo-app/tree/3.0.0
71
https://github.com/evgenyrodionov/redux-logger
Redux State Structure and Retrieval 108
When you start your Todo application now, you should see the output of the logger in the developer
console. The Todo application with the middleware using redux-logger can be found in the GitHub
repository72 .
The second part is to use spread operators instead of the Object.assign() function to keep an
immutable data structure. You can apply it in your reducer functions:
src/index.js
The application should work the same as before. The source code can be found in the GitHub
repository73 .
In the third part of the refactoring, you will use a normalized state structure. Therefore, you can
install the neat library normalizr74 .
Command Line: /
Let’s have a look at the initial state for the todoReducer. You could make up your own initial state.
For instance, what about completing all coding examples in this book?
72
https://github.com/rwieruch/taming-the-state-todo-app/tree/4.0.0
73
https://github.com/rwieruch/taming-the-state-todo-app/tree/5.0.0
74
https://github.com/paularmstrong/normalizr
Redux State Structure and Retrieval 109
src/index.js
const todos = [
{ id: '1', name: 'Hands On: Redux Standalone with advanced Actions' },
{ id: '2', name: 'Hands On: Redux Standalone with advanced Reducers' },
{ id: '3', name: 'Hands On: Bootstrap App with Redux' },
{ id: '4', name: 'Hands On: Naive Todo with React and Redux' },
{ id: '5', name: 'Hands On: Sophisticated Todo with React and Redux' },
{ id: '6', name: 'Hands On: Connecting State Everywhere' },
{ id: '7', name: 'Hands On: Todo with advanced Redux' },
{ id: '8', name: 'Hands On: Todo but more Features' },
{ id: '9', name: 'Hands On: Todo with Notifications' },
{ id: '10', name: 'Hands On: Hacker News with Redux' },
];
You can use normalizr to normalize this data structure. First, you have to define a schema:
src/index.js
// schemas
Second, you can use the schema to normalize your initial todos and use them as default parameter
in your todoReducer.
Redux State Structure and Retrieval 110
src/index.js
// reducers
const todos = [
...
];
Third, your todoReducer needs to handle the normalized state structure. It has to deal with entities
and a result (list of ids). You can output the normalized todos even though the Todo application
crashes when you attempt to start it.
src/index.js
src/index.js
It operates on entities and ids. Last but not least, when connecting Redux with React, the
components need to be aware of the normalized data structure. First, the connection between store
and components:
src/index.js
function mapStateToPropsList(state) {
return {
todosAsIds: state.todoState.ids,
};
}
function mapDispatchToPropsItem(dispatch) {
return {
onToggleTodo: id => dispatch(doToggleTodo(id)),
};
}
Second, the TodoList component receives only the todosAsIds and the TodoItem receives the todo
entity.
Redux State Structure and Retrieval 112
src/index.js
function TodoApp() {
return <ConnectedTodoList />;
}
The application should work again. Start it and play around with it. You can find the source code in
the GitHub repository75 .
In the fourth and last part of the refactoring you are going to use selectors. This refactoring is fairly
straight forward. You have to extract the parts that operate on the state in your mapStateToProps()
functions to selector functions. First, define the selector functions:
src/index.js
// selectors
function getTodosAsIds(state) {
return state.todoState.ids;
}
Second, you can use these functions instead of operating on the state directly in your mapStateTo-
Props() functions:
75
https://github.com/rwieruch/taming-the-state-todo-app/tree/6.0.0
Redux State Structure and Retrieval 113
src/index.js
function mapStateToPropsList(state) {
return {
todosAsIds: getTodosAsIds(state),
};
}
The Todo application should work with selectors now. You can find it in the GitHub repository76 .
76
https://github.com/rwieruch/taming-the-state-todo-app/tree/7.0.0
Redux State Structure and Retrieval 114
src/index.js
this.state = {
value: '',
};
this.onCreateTodo = this.onCreateTodo.bind(this);
this.onChangeName = this.onChangeName.bind(this);
}
onChangeName(event) {
this.setState({ value: event.target.value });
}
onCreateTodo(event) {
this.props.onAddTodo(this.state.value);
this.setState({ value: '' });
event.preventDefault();
}
render() {
return (
<div>
<form onSubmit={this.onCreateTodo}>
<input
type="text"
placeholder="Add Todo..."
value={this.state.value}
onChange={this.onChangeName}
/>
<button type="submit">Add</button>
</form>
</div>
Redux State Structure and Retrieval 115
);
}
}
Notice again that the component is completely unaware of Redux. It only updates its local value
state. When the form gets submitted, it uses the internal value state for the onAddTodo() callback
function that’s accessible in the props object. The component doesn’t know whether the callback
function updates the local state of a parent component or the Redux store. Next, you can use the
connected version of this component in the TodoApp component.
src/index.js
function TodoApp() {
return (
<div>
<ConnectedTodoCreate />
<ConnectedTodoList />
</div>
);
}
The last step is to wire the React component to the Redux store by making it a connected component.
src/index.js
function mapDispatchToPropsCreate(dispatch) {
return {
onAddTodo: name => dispatch(doAddTodo(uuid(), name)),
};
}
It uses the mapDispatchToPropsCreate() function to get access to the dispatch method of the Redux
store. The doAddTodo() action creator takes the name of the todo item, coming from the TodoCreate
component, and generates a unique identifier with the uuid() function. The uuid() function is a
neat little helper function that comes from the uuid77 library. You have to install it and import it to
your Todo application:
77
https://github.com/kelektiv/node-uuid
Redux State Structure and Retrieval 116
Command Line: /
npm install --save uuid
src/index.js
import React from 'react';
import ReactDOM from 'react-dom';
import { applyMiddleware, combineReducers, createStore } from 'redux';
import { Provider, connect } from 'react-redux';
import { createLogger } from 'redux-logger';
import { schema, normalize } from 'normalizr';
import uuid from 'uuid/v4';
import './index.css';
You can try to create a todo item in your Todo application now. It should work. Next you want to
make use of your filter functionality to filter by completeness status of a todo item. First, you have
to add a Filter component.
src/index.js
function Filter({ onSetFilter }) {
return (
<div>
Show
<button
type="button"
onClick={() => onSetFilter('SHOW_ALL')}>
All</button>
<button
type="button"
onClick={() => onSetFilter('SHOW_COMPLETED')}>
Completed</button>
<button
type="button"
onClick={() => onSetFilter('SHOW_INCOMPLETED')}>
Incompleted</button>
</div>
);
}
The Filter component only receives a callback function. Again it doesn’t know anything about the
state management that is happening above. The callback function is used in different buttons to set
specific filter types. You can use the connected component in the TodoApp component again.
Redux State Structure and Retrieval 117
src/index.js
function TodoApp() {
return (
<div>
<ConnectedFilter />
<ConnectedTodoCreate />
<ConnectedTodoList />
</div>
);
}
Last but not least, you have to connect the Filter component to actually use it in the TodoApp
component. It dispatched the doSetFilter action creator by passing the filter type from the
underlying buttons in the Filter component.
src/index.js
function mapDispatchToPropsFilter(dispatch) {
return {
onSetFilter: filterType => dispatch(doSetFilter(filterType)),
};
}
When you start your Todo application now, you will see that the filterState will change once
you click on one of your filter buttons. But nothing happens to your displayed todos. They are not
filtered, because in your selector you select the whole list of todos. The next step would be to adjust
the selector to only select the todos in the list that are matching the filter. First, you can define filter
functions that match todos according to their completed state.
src/index.js
// filters
const VISIBILITY_FILTERS = {
SHOW_COMPLETED: item => item.completed,
SHOW_INCOMPLETED: item => !item.completed,
SHOW_ALL: item => true,
};
Second, can use your selector to only select the todos matching a filter. You already have all selectors
in place. But you need to adjust one to filter the todos according to the filterState.
Redux State Structure and Retrieval 118
src/index.js
// selectors
function getTodosAsIds(state) {
return state.todoState.ids
.map(id => state.todoState.entities[id])
.filter(VISIBILITY_FILTERS[state.filterState])
.map(todo => todo.id);
}
Since your state is normalized, your have to map through all your ids to get a list of todos, filter
them by filterState, and map them back to ‘ids’. Your filter functionality should work now. Start
your application and try it.
You can find the final application in the GitHub repository78 . It applies all the learnings about the
Redux middleware, immutable and normalized data structures and selectors.
78
https://github.com/rwieruch/taming-the-state-todo-app/tree/8.0.0
Asynchronous Redux
In the book, you have only used synchronous actions so far. There is no delay of the action
dispatching involved. Yet, sometimes you want to delay an action. The example can be a simple one:
Imagine you want to have a notification for your application user when a todo item was created.
The notification should hide after one second. The first action would show the notification and set
a isShowingNotification property to true. If the isShowing status of the notification resides in the
store, you would need a way to delay a second action to hide the notification again. In a simple
scenario it would look like the following:
Code Playground
There is nothing against a simple setTimeout() function in your application. It can be used in the
context of Redux and React. Sometimes it is easier to remember the basics in JavaScript than trying
to apply yet another library to fix a problem. Since you know about actions creators, the next step
could be to extract these actions into according action creators and action types.
Code Playground
function doShowNotification(text) {
return {
type: NOTIFICATION_SHOW,
text,
};
}
function doHideNotification() {
return {
type: NOTIFICATION_HIDE,
};
}
Asynchronous Redux 120
store.dispatch(doShowNotification('Todo created.'));
setTimeout(() => {
store.dispatch(doHideNotification());
}, 1000);
There are two problems in a growing application now. First, you would have to duplicate the logic
of the delayed action every time. Second, once your application dispatches multiple notifications at
various places at the same time, the first running action that hides the notifications would hide all
of them. The solution would be to give each notification an identifier. Both problems can be solved
by extracting the functionality into a function.
Code Playground
// actions
const NOTIFICATION_SHOW = 'NOTIFICATION_SHOW';
const NOTIFICATION_HIDE = 'NOTIFICATION_HIDE';
function doHideNotification(id) {
return {
type: NOTIFICATION_HIDE,
id,
};
}
// extracted functionality
let naiveId = 0;
function showNotificationWithDelay(dispatch, text) {
dispatch(doShowNotification(text, naiveId));
setTimeout(() => {
dispatch(doHideNotification(naiveId));
}, 1000);
naiveId++;
}
Asynchronous Redux 121
// usage
showNotificationWithDelay(store.dispatch, 'Todo created.');
Now each notification could be identified in the reducer and thus be either shown or hidden. The
extracted function gets control over the dispatch() method from the Redux store, because it needs
to dispatch a delayed action.
Why not passing the Redux store instead? Usually, you want to avoid to pass the store around
directly. You have encountered the same reasoning in the book when weaving the Redux store for
the first time into your React application. You want to make the functionalities of the store available,
but not the entire store itself. That’s why you only have the dispatch() method and not the entire
store in your mapDispatchToProps() function when using react-redux. A connected component
does never have access to the store directly and thus no other functionalities should have direct
access to it.
The pattern from above suffices for simple Redux applications that need a delayed action. However,
in scaling applications it has a drawback. The approach creates two types of action creators. While
there are synchronous action creators that can be dispatched directly, there are pseudo asynchronous
action creators, too. These pseudo asynchronous action creators cannot be dispatched directly but
have to accept the dispatch method as argument. Wouldn’t it be great to use both types of actions
the same without worrying to pass around the dispatch method and without worrying about
asynchronous or synchronous action creators?
Asynchronous Redux 122
Redux Thunk
The previous question led Dan Abramov, the creator of Redux, thinking about a general pattern to
the problem of asynchronous actions. He came up with the library called redux-thunk79 to legitimize
the concept. Synchronous and asynchronous action creators should be dispatched in a similar way
from a Redux store. It is used as middleware in your Redux store.
Code Playground
...
Basically, it creates a middleware for your actions creators. In this middleware, you are enabled
to dispatch asynchronous actions. Apart from dispatching objects, you can dispatch functions with
Redux Thunk. You always dispatched objects as actions before in this book. An action itself is an
object and an action creator returns an action object.
Code Playground
However, now you can pass the dispatch method a function, too. The function will always give you
access to the dispatch method again.
79
https://github.com/gaearon/redux-thunk
Asynchronous Redux 123
Code Playground
naiveId++;
});
The dispatch method of the Redux store when using Redux Thunk will differentiate between passed
objects and functions. The passed function is called a thunk. You can dispatch as many actions
synchronously and asynchronously as you want in a thunk. When a thunk is growing and handles
complex business logic at some point in your application, it is called a fat thunk. You can extract
a thunk function as an asynchronous action creator, that is a higher order function and returns the
thunk function, and can be dispatched the same way as an synchronous action creator.
Code Playground
let naiveId = 0;
function showNotificationWithDelay(text) {
return function (dispatch) {
dispatch(doShowNotification(text, naiveId));
setTimeout(() => {
dispatch(doHideNotification(naiveId));
}, 1000);
naiveId++;
};
}
store.dispatch(showNotificationWithDelay('Todo created.'));
It is similar to the solution without Redux Thunk. However, this time you don’t have to pass around
the dispatch method and instead have access to it in the returned thunk function. Now, when using
it in a React component, the component still only executes a callback function that it receives
via its props. The connected component then dispatches an action, regardless of the action being
synchronously or asynchronously, in the mapDispatchToProps() function.
That are the basics of Redux Thunk. There are a few more things that are good to know about:
Asynchronous Redux 124
• getState(): A thunk function gives you the getState() method of the Redux store as second
argument: function (dispatch, getState). However, you should generally avoid it. It’s best
practice to pass all necessary state to the action creator instead of retrieving it in the thunk.
• Promises: Thunks work great in combination with promises. You can return a promise from
your thunk and use it, for instance, to wait for its completion: store.dispatch(showNotificationWithDelay(
created.')).then(...).
• Recursive Thunks: The dispatch method in a thunk can again be used to dispatch an
asynchronous action. Thus, you can apply the thunk pattern recursively.
src/index.js
You don’t need to create a new action type. Instead, you can reuse the action you already have to
add todos. When a todo gets created, the notification reducer will store a new notification about
the created todo item. Second, you have to include the reducer in your combined reducer to make
it accessible to the Redux store.
80
https://github.com/rwieruch/taming-the-state-todo-app/tree/8.0.0
Asynchronous Redux 125
src/index.js
The Redux part is done. It is only a reducer that you need to include in the Redux store. The action
gets reused. Third, you have to implement a React component that displays all of your notifications.
src/index.js
Fourth, you can include the connected version of the Notifications component in your TodoApp
component.
src/index.js
function TodoApp() {
return (
<div>
<ConnectedFilter />
<ConnectedTodoCreate />
<ConnectedTodoList />
<ConnectedNotifications />
</div>
);
}
Last but not least, you have to wire up React and Redux in the connected ConnectedNotifications
component.
Asynchronous Redux 126
src/index.js
The only thing left is to implement the missing selector getNotifications(). Since the notifications
in the Redux store are saved as an object, you have to use a helper function to convert it into an
array. It is good to extract the helper function earlier on, because you might need such functionalities
more often and shouldn’t couple it to the domain of notifications.
src/index.js
function getNotifications(state) {
return getArrayOfObject(state.notificationState);
}
function getArrayOfObject(object) {
return Object.keys(object).map(key => object[key]);
}
The first part of the “Hands On” chapter is done. You should see a notification in your Todo
application once you create a todo item. The second part will implement a NOTIFICATION_HIDE
action and use it in the notificationReducer to remove the notification from the state. First, you
have to introduce the action type:
src/index.js
Second, you can implement an action creator that uses the action type. It will hide (remove) the
notification by id, because they are stored by id:
Asynchronous Redux 127
src/index.js
function doHideNotification(id) {
return {
type: NOTIFICATION_HIDE,
id
};
}
Third, you can capture it in the notificationReducer. The JavaScript destructuring functionality
can be used to omit a property from an object. You can simply omit the notification and return the
remaining object.
src/index.js
function notificationReducer(state = {}, action) {
switch(action.type) {
case TODO_ADD : {
return applySetNotifyAboutAddTodo(state, action);
}
case NOTIFICATION_HIDE : {
return applyRemoveNotification(state, action);
}
default : return state;
}
}
That was the second part of the “Hands On” chapter that introduced the hiding notification
functionality. The third and last part of the “Hands On” chapter will introduce asynchronous actions
to hide a notification after a couple of seconds. As mentioned earlier, you wouldn’t need a library to
solve this problem. You could simply built on the JavaScript timeout functionality. But for the sake
of learning about asynchronous actions, you will use Redux Thunk. It’s up to you to exchange it
with another asynchronous actions library afterward for the sake of learning about the alternatives.
First, you have to install the redux-thunk81 on the command line:
81
https://github.com/gaearon/redux-thunk
Asynchronous Redux 128
Command Line: /
src/index.js
...
import thunk from 'redux-thunk';
...
src/index.js
The application should still work. When using Redux Thunk, you can dispatch action objects as
before. However, now you can dispatch thunks (functions), too. Rather than dispatching an action
object that only creates a todo item, you can dispatch a thunk function that creates a todo item
and hides the notification about the creation after a couple of seconds. You have two plain actions
creators, doAddTodo() and doHideNotification(), already in place. You only have to reuse it in
your thunk function.
src/index.js
setTimeout(function () {
dispatch(doHideNotification(id));
}, 5000);
}
}
In the last step, you have to use the doAddTodoWithNotification() rather than the doAddTodo()
action creator when connecting Redux and React in your TodoCreate component.
Asynchronous Redux 129
src/index.js
function mapDispatchToPropsCreate(dispatch) {
return {
onAddTodo: name => dispatch(doAddTodoWithNotification(uuid(), name)),
};
}
That’s it. Your notifications should work and hide after five seconds. Basically, you have built the
foundation for a notification system in your Todo application. You can use it for other actions, too.
The project can be found in the GitHub repository82 .
82
https://github.com/rwieruch/taming-the-state-todo-app/tree/9.0.0
Asynchronous Redux 130
Redux Saga
Redux Saga83 is the most popular asynchronous actions library for Redux. “The mental model is
that a saga is like a separate thread in your application that’s solely responsible for side effects.”
Basically, it outsources the impure operations into separate threads. These threads can be started,
paused or cancelled with plain Redux actions from your core application. Thereby, threads in Redux
Saga make it simple to keep your side-effects away from your core application. However, threads
can dispatch actions and have access to the state though.
Redux Saga uses JavaScript ES6 Generators84 as underlying technology. That’s why the code reads
like synchronous code. You avoid to have callbacks. The advantage over Redux Thunk is that your
actions stay pure and thus they can be tested well.
Apart from Redux Thunk and Redux Sage, there are other side-effect libraries for Redux. If you want
to try out observables in JavaScript, you could give Redux Observable85 a shot. It builds up on RxJS,
a generic library for reactive programming. If you are interested in another library that uses reactive
programming principles, too, you can try Redux Cycles86 .
In conclusion, as you can see, all these libraries, Redux Saga, Redux Observable and Redux Cycles,
make use of different techniques in JavaScript. You can give them a shot to experiment with recent
JavaScript technologies: generators or observables. The whole ecosystem around asynchronous
actions is a great playground to try new things in JavaScript.
83
https://github.com/redux-saga/redux-saga
84
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Global_Objects/Generator
85
https://github.com/redux-observable/redux-observable
86
https://github.com/cyclejs-community/redux-cycles
Asynchronous Redux 131
Command Line: /
The action you have used before with a thunk becomes a pure action creator now. It will be used to
trigger the saga thread.
src/index.js
...
Now you can introduce your first saga that listens on this particular action, because the action is
solely used to trigger the saga thread.
src/index.js
...
// sagas
function* watchAddTodoWithNotification() {
yield takeEvery(TODO_ADD_WITH_NOTIFICATION, ...);
}
Asynchronous Redux 132
Most often you will find one part of the saga watching incoming actions and evaluating them. If
an evaluation applies truthfully, it will often call another generator function that handles the side-
effect. That way, you can keep your side-effect watcher maintainable and don’t clutter them with
business logic. In the previous example a takeEvery() effect of Redux Saga is used to handle every
action with the specified action type. Yet there are other effects in Redux Saga such as takeLatest()
which only takes the last of the incoming actions by action type.
src/index.js
function* watchAddTodoWithNotification() {
yield takeEvery(TODO_ADD_WITH_NOTIFICATION, handleAddTodoWithNotification);
}
function* handleAddTodoWithNotification(action) {
const { todo } = action;
const { id, name } = todo;
yield put(doAddTodo(id, name));
yield delay(5000);
yield put(doHideNotification(id));
}
As you can see, in JavaScript generators you use the yield statement to execute asynchronous code
synchronously. Only when the function after the yield resolves, the code will execute the next line
of code. Redux Saga comes with a helper such as delay() that can be used to delay the execution
by an amount of time. It would be the same as using setTimeout() in JavaScript, but the delay()
helper makes it more concise when using JavaScript generators and can be used in a synchronous
way when using yield.
Now, you only have to exchange your middleware in your Redux store from using Redux Thunk to
Redux Saga.
src/index.js
...
notificationState: notificationReducer,
});
saga.run(watchAddTodoWithNotification);
That’s it. You Todo application should run with Redux Saga instead of Redux Thunk now. The final
application can be found in the GitHub repository87 . In the future, it is up to you to decide on an
asynchronous actions library when using Redux. Is it Redux Thunk or Redux Saga? Or do you
decide to try something new with Redux Observable or Redux Cycles? The Redux ecosystem is full
of amenities to try cutting edge JavaScript features such as generators or observables.
87
https://github.com/rwieruch/taming-the-state-todo-app/tree/10.0.0
Redux Patterns, Techniques and Best
Practices
There are several patterns and best practices that you can apply in a Redux application. I will go
through a handful of them to point you in the right direction. However, the evolving patterns and
best practices around the ecosystem are changing all the time. You will want to read more about
these topics on your own.
Redux Patterns, Techniques and Best Practices 135
Code Playground
// JavaScript ES5
function higherOrderFunction() {
return function () {
...
};
}
// JavaScript ES6
const higherOrderFunction = () => { ... };
It’s a higher order function that is much more readable in JavaScript ES6. You will find yourself
using higher order functions more often when programming in a functional style. It will happen
that you not only use one higher order function, but a higher order function that returns a higher
order function that returns a function. Again it becomes easier to read when using JavaScript ES6.
Code Playground
// JavaScript ES5
function higherOrderFunction() {
return function () {
return function () {
...
}
};
}
Redux Patterns, Techniques and Best Practices 136
// JavaScript ES6
const higherOrderFunction = () => () => { ... };
Code Playground
// action creator
const doAddTodo = (id, name) => ({
type: ADD_TODO,
todo: { id, name },
});
// selector
const getTodos = (state) => state.todos;
The JavaScript community shifts in the direction of functional programming and embraces more
than ever this style. Functional programming let’s you write more predictable code by embracing
pure functions without side-effects and immutable data structures. JavaScript ES6 and beyond make
it easier and more readable to write in a functional style.
Redux Patterns, Techniques and Best Practices 137
Naming Conventions
In Redux, you have a handful of different types of functions. You have action creators, selectors and
reducers. It is always good to name them accordingly to their type. Other developers will have an
easier time identifying the function type. Just following a naming convention for your functions,
you can give yourself and others a valuable information about the function itself.
Personally, I follow this naming convention with Redux functions. It uses prefixes for each function
type:
In the previous chapters, the example code always used this naming convention. In addition, it
clarifies things when using higher order functions. Remember the mapDispatchToProps() function
when connecting Redux to React?
Code Playground
The functions themselves become more concise by using JavaScript ES6 arrow functions. But there
is another clue that makes proper naming so powerful. Solely from a naming perspective, you
can see that the mapStateToProps() and mapDispatchToProps() functions transform the returned
properties from the Redux world to the React world. The connected component doesn’t know about
selectors or actions creators anymore. As you can see, that is already expressed in the transformed
props and functions. They are named todos and onToggleTodo. There are no remains from the
Redux world, from a technical perspective but also from a pure naming perspective. That’s powerful,
because your underlying components are Redux agnostic.
So far, the topic was only about function naming. But there is another part in Redux that can be
named properly: action types. Consider the following action type names:
Redux Patterns, Techniques and Best Practices 138
Code Playground
Most cultures read from left to right. That’s conveyed in programming, too. So which action type
naming makes more sense? Is it the verb or the subject? You can decide on your own, but become
clear about a consistent naming convention for your action types. Personally, I find it easier to scan
when I have the subject first. When using Redux Logger in a scaling application where a handful
actions can be dispatched at once, I find it easier to scan by subject than by verb.
You can even go one step further and apply the subject as domain prefix for your action types.
Code Playground
These are only opinionated naming conventions for those types of functions and constants in Redux.
You can come up with your own. But do yourself and your fellow developers a favor and reach an
agreement first and then apply them consistently through your code base.
Redux Patterns, Techniques and Best Practices 139
Code Playground
function doCompleteTodo(id) {
return {
type: TODO_COMPLETE,
todo: { id },
};
}
Now, imagine that there should be a measuring of the progress of the Todo application user. The
progress will always start at zero when the user opens the application. When a todo gets completed,
the progress should increase by one. A potentially easy solution could be counting all completed
todo items. However, since there could be completed todo items already, and you want to measure
the completed todo items in this session, the solution would not suffice. The solution could be a
second reducer that counts the completed todos in this session.
Code Playground
function progressReducer(state = 0, action) {
switch(action.type) {
case TODO_COMPLETE : {
return state++;
}
default : return state;
}
}
The counter will increment when a todo got completed. Now, you can easily measure the progress
of the user. Suddenly, you have a 1:2 relationship between action and reducer. Nobody forces you
not to couple action and reducer in a 1:1 relationship, but it always makes sense to be creative in
this manner. What would happen otherwise? Regarding the progress measurement issue, you might
come up with a second action type and couple it to the previous reducer:
Code Playground
function doTrackProgress() {
return {
type: PROGRESS_TRACK,
};
}
Code Playground
dispatch(doCompleteTodo('0'));
dispatch(doTrackProgress());
But that would miss the point in Redux. You would want to come up with these commonalities
to make your actions more abstract and be used by multiple reducers. My rule of thumb for this:
Approach your actions as concrete actions with a 1:1 relationship to their reducers, but keep yourself
always open to reuse them as more abstract actions in other reducers.
Redux Patterns, Techniques and Best Practices 142
Folder Organization
Eventually, your Redux application grows and you cannot manage everything - reducers, action
creators, selectors, store and view - in one file. You will have to split up the files. Fortunately,
JavaScript ES6 brings import88 and export89 statements to distribute functionalities in files. If you
are not familiar with these, you should read about them.
In this chapter, I want to show you two approaches to organize your folder and files in a Redux
application. The first approach, the technical folder organization, is used in smaller applications.
Once your application scales and more than one team in your organization is working on the project,
you can consider the feature folder organization. In addition, you will learn about best practices
for your file and folder structure in this chapter.
• the application is managed by only one person or one team, thus has less conflict potential
when working on the same code base
• the application is small from a lines of code perspective and can be managed by one person
or one team
In conclusion, it depends on the size of the team and the size of the code base. Now, how to separate
the files? They get separated by their technical aspects:
Folder Organization
-app
--reducers
--actions creators
--selectors
--store
--constants
--components
The reducers, action creators, selectors and store should be clear. In these folders you have all the
different aspects of Redux. In the components folder you have your view layer. When using React,
that would be the place where you will find your React components. In the constants folder you can
have any constants, but also the action types of Redux. These can be imported in the action creators
and reducers. An elaborated folder/file organization split by technical aspects might look like the
following:
88
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Statements/import
89
https://developer.mozilla.org/en/docs/Web/JavaScript/Reference/Statements/export
Redux Patterns, Techniques and Best Practices 143
Folder Organization
-app
--reducers
---todoReducer.js
---filterReducer.js
---notificationReducer.js
--actions creators
---filters.js
---todos.js
---notifications.js
--selectors
---filters.js
---todos.js
---notifications.js
--store
---store.js
--constants
---actionTypes.js
--components
---TodoList.js
---TodoItem.js
---Filter.js
---Notifications.js
What are the advantages and disadvantages of this approach? The most important advantage is that
reducers and action creators are not coupled. They are loosely arranged in their folders. It embraces
the notion of Redux to capture any action in any reducer. Reducers and action creators are not in a
1:1 relationship. In addition, all Redux functionalities are reachable from a top level. None of these
functionalities are hidden in a lower level and thus less accessible. This approach embraces the event
pattern. A disadvantage of this approach, hence the two requirements, is that it doesn’t scale well.
Each technical folder will grow endlessly. There are no constraints except for the separation by type.
It can become messy after you have introduced tons of reducers, action creators and selectors.
Folder Organization
-app
--todo
--filter
--notification
--store
Folder Organization
-app
--todo
---TodoList.js
---TodoItem.js
---reducer.js
---actionCreators.js
---selectors.js
--filter
---Filter.js
---reducer.js
---actionCreators.js
---selectors.js
--notification
---Notifications.js
---reducer.js
---actionCreators.js
---selectors.js
--store
---store.js
This approach, separating by features, is way more flexible than the previous approach. It gives you
more freedom to arrange your folders and files. When using this approach, there are more ways to
accomplish it. You don’t necessarily have to follow the example above.
What are the advantages and disadvantages of this approach? It has the same advantages and
disadvantages as the technical folder organization but negated. Instead of making action creators
and reducers accessible on a top level, they are hidden in a feature folder. In a scaling application
with multiple teams, other teams will most likely not reuse your action creators and reducers but
implement their own. Another disadvantage is that it groups action creators and reducers in a 1:1
relationship which goes against the overarching idea of Redux. You embrace a command pattern
instead of an event pattern. The advantage on the other side, and that’s why most teams in a scaling
Redux Patterns, Techniques and Best Practices 145
application are using this approach, is that it grows well. Teams can work on separate feature folders
and don’t run into conflicts. Still, they can follow the overarching state management flow, when
using a middleware library like redux-logger.
Even though the feature folder organization bears a lot of pitfalls by embracing the command
pattern, it is often used in scaling applications with several development teams. Therefore, I can
give one advice: Make your action creators, reducers and selectors accessible to everyone so that they
can be reused. It can happen by documentation, word of mouth or another variation of folder/file
organization.
Ducks
In Redux, there exists another concept called ducks. It relates to the organization of action creators,
action types and reducers as tuples. The ducks concept bundles these tuples into self contained
modules. Often, these modules end up being only one file. The official ducks pattern has a bunch of
guidelines which you can read up in the GitHub repository90 . However, you wouldn’t need to apply
all of these. For instance, in the Todo application a duck file for the filter domain might look like the
following:
Code Playground
function doSetFilter(filter) {
return {
type: FILTER_SET,
filter,
};
}
90
https://github.com/erikras/ducks-modular-redux
Redux Patterns, Techniques and Best Practices 146
export {
doSetFilter,
};
The drawbacks of the ducks concept are similar to the feature folder approach. You couple actions
and reducers, hence no one will embrace to capture actions in multiple reducers. As long as action
and reducer are coupled, the ducks concept makes sense. Otherwise, it shouldn’t be applied too often.
Instead, you should embrace the idea of Redux to keep your reducers and action creators accessible
on a top level.
Redux Patterns, Techniques and Best Practices 147
Testing
The book will not dive deeply into the topic of testing, but it shouldn’t be unmentioned. Testing your
code in programming is essential and should be seen as mandatory. You want to keep the quality
of your code high and an assurance that everything works. However, testing your code can often
be tedious. You have to setup, mock or spy things before you can finally start to test it. Or you you
have to cover a ton of edge cases in your one huge code block. But I can give you comfort by saying
that this is not the case when testing state management done with Redux. I will show you how you
can easily test the necessary parts, keep your efforts low and stay lazy.
Perhaps you have heard about the testing pyramid. There are end-to-end tests, integration tests and
unit tests. If you are not familiar with those, the book gives you a quick and basic overview. A unit
test is used to test an isolated and small block of code. It can be a single function that is tested by an
unit test. However, sometimes the units work well in isolation yet don’t work in combination with
other units. They need to be tested as a group as units. That’s where integration tests can help out by
covering whether units work well together. Last but not least, an end-to-end test is the simulation
of a real user scenario. It could be an automated setup in a browser simulating the login flow of an
user in a web application. While unit tests are fast and easy to write and to maintain, end-to-end
tests are the opposite of this spectrum.
How many tests do I need of each type? You want to have many unit tests to cover your isolated
functions. After that, you can have several integration tests to cover that the most important
functions work in combination as expected. Last but not least, you might want to have only a few
end-to-end tests to simulate critical scenarios in your web application. That’s it for the general
excursion in the world of testing. Now, how does it apply to state management with Redux?
Redux embraces the functional programming style. Your functions are pure and you don’t have to
worry about any side-effects. A function always returns the same output for the same input. Such
functions are easy to test, because you only have to give them an input and expect the output because
there is a no side-effect guarantee. That’s the perfect fit for unit tests, isn’t it? In conclusion, it makes
state management testing when build with Redux a pleasure.
In Redux, you have different groups of functions: action creators, reducers, selectors. For each of
these groups, you can see a pattern for their input and output. These can be applied to a test pattern
which can be used as blueprint for a unit test for each group of functions.
Input Pattern:
• action creators can have an optional input that becomes their optional payload
• selectors can have an optional input that supports them to select the substate
• reducers will always receive a previous state and action
Output Pattern:
• action creators will always return an object with a type and optional payload
Redux Patterns, Techniques and Best Practices 148
Test Pattern:
• when invoking an action creator, the correct return object should be expected
• when invoking a selector, the correct substate should be expected
• when invoking a reducer, the correct new state should be expected
How does that apply in code? The book will show it in pseudo code, because it will not make any
assumption about your testing libraries. Yet it should be sufficient to pick up these patterns for each
group of functions (action creators, reducers, selectors) to apply them in your unit tests.
Action Creators:
Code Playground
expect(action).to.equal(expectedAction);
Selectors:
Code Playground
expect(substate).to.equal(expectedSubstate)
Reducers:
Redux Patterns, Techniques and Best Practices 149
Code Playground
expect(newState).to.equal(expectedNewState);
These test patterns will always stay the same for their aspects. You only have to fill in the blanks. You
can even give yourself an easier time and setup automated code snippets for your editor of choice.
For instance, typing “rts” (abbr. for “redux test selector”) gives you the blueprint for a selector test.
The other two snippets could be “rtr” (redux test reducer) and “rta” (redux test action). After that,
you only have to fill in the remaining things.
These test patterns for state management show you how simple testing becomes when working with
the clear constraints of a library like Redux. Everything behaves the same, it is predictable, and thus
can be tested every time the in the same way. When setting up automated code snippets, you will
save yourself a lot of time, yet have a great test coverage for your whole state management. You can
even go one step further and apply test-driven development91 (TDD) which basically means you test
before you implement.
There is another neat helper that can ensure that your state stays immutable. Because you never
know if you accidentally mutate your state even though it is forbidden in Redux. I guess, there is
a handful of libraries around this topic, but I use deep-freeze92 in my tests to ensure that the state
(and even actions) doesn’t get mutated.
Code Playground
deepFreeze(previousState);
91
https://en.wikipedia.org/wiki/Test-driven_development
92
https://github.com/substack/deep-freeze
Redux Patterns, Techniques and Best Practices 150
expect(newState).to.equal(expectedNewState);
That’s it for testing your different aspects when using Redux. It can be accomplished by using
unit tests. You could apply integration tests, too, for instance to test an action creator and reducer
altogether. After all, you have a blueprint for testing these functions all the time at your hand and
there is no excuse anymore to not test your code.
Redux Patterns, Techniques and Best Practices 151
Error Handling
The topic of error handling is rarely touched in programming. Often, the topic is avoided by the
community and it is hard to find a common sense about it. This chapter gives you basic guidance
on how you could provide error handling in your Redux application.
Error handling is often involved when making requests to an API. You have learned about
asynchronous actions in Redux that can be used for these kind of side-effects. But there was no
saying about error handling in those side-effects so far. How to catch the errors and how to make
them visible for your application end-user?
Basically, an error in an application can be represented as a state. That’s why the topic is discussed
in a state management book in the first place. For instance, imagine that you get your todo items
from a server request. You would have an API on the server-side that exposes these todo items.
Once you fetch these todo items from the API, you would have to deal with error handling, because
a request could always fail. The following request returns a JavaScript promise. The fetch can be
either successfully resolved in a then() method or yields an error in a catch() method.
Code Playground
fetch('my/todos/api').then(function(response) {
return response.json();
}).then(function(todos) {
// do something with todos
}).catch(function(error) {
// do something with error
});
When using Redux asynchronous actions, the request could live in a Redux Thunk.
Code Playground
function getTodos(dispatch) {
fetch('my/todos/api').then(function(response) {
return response.json();
}).then(function(todos) {
// do something with todos
}).catch(function(error) {
// do something with error
});
}
Now, it would be up to you to store either the todos or the error as state in your Redux store. You
could have two potential actions:
Redux Patterns, Techniques and Best Practices 152
Code Playground
These could be used in your Redux Thunk to store both potential outcomes:
Code Playground
function getTodos(dispatch) {
fetch('my/todos/api').then(function(response) {
return response.json();
}).then(function(todos) {
dispatch({ type: TODOS_FETCH_SUCCESS, todos });
}).catch(function(error) {
dispatch({ type: TODOS_FETCH_ERROR, error });
});
}
The todo reducer would have to deal with both actions now. One that stores the todo items and one
that stores the error object.
Code Playground
const initialState = {
todos: [],
error: null,
};
...
Redux Patterns, Techniques and Best Practices 153
That’s it basically for the state management part. Whereas the applyFetchTodosError() function
would set the error object in the state, the applyFetchTodosSuccess() function would set the list
of todos. In addition, the success function would have to reset the error property in the state to null
again, because imagine you would do a second request after the first request has failed. When the
second request was successful, you would want to store the todo items but reset the error state.
In your view layer, depending on the todo state, you could decide whether to show an error message,
because there is an error object in the todo state, or to show the list of todos. When there is an error
message displayed, you could provide your end-user with a button to try fetching the todos again.
When the second request is successful, the error object is set to null and instead the todo items are
set in the state. The view layer could display the list of todo items now.
After all, there is no magic behind error handling in Redux. Whenever an error occurs, you would
store it in your state. When the view layer notices an error in the state, it could use conditional
rendering to show an error message instead of the assumed result.
(React in) Redux FAQ
I intend to grow this section organically to answer frequently asked questions to the best of
my knowledge. These are questions that come up often when discussing Redux standalone or as
complementing part of React.
(React in) Redux FAQ 155
Concept Playground
The chain is connected to the view layer by something (e.g. react-redux with mapStateToProps() and
mapDispatchToProps()) that enables you to write connected components. These components have
access to the Redux store. They are used to receive state or to alter the state. They are a specialized
case of a container component in the presenter and container pattern when using components.
Concept Playground
View -> (mapDispatchToProps) -> Action -> Reducer(s) -> Store -> (mapStateToProp\
s) -> View
All other components are not aware of any local or sophisticated state management solution. They
only receive props, except they have their own local state management (such as this.state and
this.setState() in React).
State can be received directly by operating on the state object or indirectly by selecting it with
selectors.
Code Playground
State can be altered by dispatching an action directly or by using action creators that return an
action object.
Redux State as Architecture 159
Code Playground
In order to keep your state predictable and manageable in the reducers, you can apply techniques for
an improved state structure. You can normalize your state to have always a single source of truth.
That means you don’t have to operate on duplicated entities, but only on one reference of the entity.
In addition, it keeps the state flat. It is easier to manage only by using spread operators.
Around these practical usages, you have learned several supporting techniques. There are tons
of opinionated ways to organize your folders and files. The book showcased two of the main
approaches, but they vary in their execution from developer to developer, team to team or company
to company. Nevertheless, you should always bear in mind to keep Redux at a top level. It is not
used to manage the state of one single component. Instead it is used to wire dedicated components
to the store in order to enable them to alter and to retrieve the state from it.
Coupling actions and reducers is fine, but always think twice when adding another action type. For
instance, perhaps a action type could be reused in another reducer. When reusing action types, you
avoid to end up with fat thunks when using Redux thunk. Instead of dispatching several actions,
your thunk could dispatch only one abstract action that is reused in more than one reducer.
You have learned that you can plan your state management ahead. There are use cases where
local state makes more sense than sophisticated state. Both can be used and should be used in a
scaling application. By combining local state to the native local storage of the browser, you can
give the user of your application an improved UX. In addition, you can plan the state ahead too.
Think about view state and entity state and where it should live in your application. You can give
your reducers difference domains as their ownership such as todoReducer, filterReducer and
notificationReducer. However, once you have planned your state management and state, don’t
stick to it. When growing your application, always revisit those things to apply refactorings. That
will help you to keep your state manageable, maintainable and predictable in the long run.
Redux State as Architecture 160
Command Line
create-react-app react-redux-hackernews
After the project was created for you, you can navigate into the project folder, open your editor and
start the application.
Command Line
cd react-redux-hackernews
npm start
In your browser it should show the defaults that come with create-create-app.
Command Line: /
cd src
rm logo.svg App.js App.test.js App.css
Even the App component is removed, because you’ll organize it in folders instead of in the top
level src/ folder. Now, from the src/ folder, create the folders for an organized folder structure by a
technical separation.
93
https://news.ycombinator.com/
94
https://hn.algolia.com/api
95
https://www.robinwieruch.de/the-road-to-learn-react/
96
https://github.com/facebookincubator/create-react-app
Redux State as Architecture 161
Folder Structure
-src/
--actions/
--api/
--components/
--constants/
--reducers/
--sagas/
--selectors/
--store/
--index.css
--index.js
Navigate in the components/ folder and create the following files for your independent components.
These are not all components yet. You will create more of them on your own for this application.
cd components
touch App.js Stories.js Story.js App.css Stories.css Story.css
You can continue this way and create the remaining files to end up with the following folder
structure.
Folder Structure
-src/
--actions/
--api/
--components/
---App.js
---App.css
---Stories.js
---Stories.css
---Story.js
---Story.css
Redux State as Architecture 162
--constants/
---actionTypes.js
--reducers/
---index.js
--sagas/
---index.js
--selectors/
--store/
---index.js
--index.css
--index.js
Now you have your foundation of folders and files for your React and Redux application. Except for
the specific component files that you already have, everything else can be used as a blueprint, your
own boilerplate, for any application using React and Redux. But only if it is separated by technical
concerns. In a growing application, you might want to separate your folders by feature. You can find
this part of the chapter in the GitHub repository97 .
src/index.js
import React from 'react';
import ReactDOM from 'react-dom';
import App from './components/App';
import './index.css';
In the next step, you can come up with sample data that can be used in the React components. The
sample data becomes the input of the App component. At a later point in time, this data will get
fetched from the Hacker News API.
97
https://github.com/rwieruch/react-redux-hackernews/tree/d5ab6a77653ee641d339c0a6a91c8444eff3f699
Redux State as Architecture 163
src/index.js
...
const stories = [
{
title: 'React',
url: 'https://facebook.github.io/react/',
author: 'Jordan Walke',
num_comments: 3,
points: 4,
objectID: 0,
}, {
title: 'Redux',
url: 'https://github.com/reactjs/redux',
author: 'Dan Abramov, Andrew Clark',
num_comments: 2,
points: 5,
objectID: 1,
},
];
ReactDOM.render(
<App stories={stories} />,
document.getElementById('root')
);
The three components, App, Stories and Story, are not defined yet but you have already created
the files. Let’s define them component by component. First, the App receives the sample stories from
above as props and its only responsibility is to render the Stories component and to pass over the
stories as props.
src/components/App.js
Second, the Stories component receives the stories as props and renders for each story a Story
component. You want to default to an empty array that the Stories component doesn’t crash when
the list of stories is null.
src/components/Stories.js
Third, the Story component renders a few properties of the story object. The story object gets
already destructured from the props in the function signature. Furthermore the story object gets
destructured as well.
src/components/Story.js
return (
<div className="story">
<span>
<a href={url}>{title}</a>
</span>
<span>{author}</span>
<span>{num_comments}</span>
<span>{points}</span>
</div>
);
}
You can start your application again with npm start on the command line. Both sample stories
should be displayed in plain React. You can find this part of the chapter in the GitHub repository98 .
src/index.css
body {
color: #222;
background: #f4f4f4;
font: 400 14px CoreSans, Arial,sans-serif;
}
a {
color: #222;
}
a:hover {
text-decoration: underline;
}
ul, li {
98
https://github.com/rwieruch/react-redux-hackernews/tree/f5843d2a06033cd045e6d0427993e30e289031a7
Redux State as Architecture 166
list-style: none;
padding: 0;
margin: 0;
}
input {
padding: 10px;
border-radius: 5px;
outline: none;
margin-right: 10px;
border: 1px solid #dddddd;
}
button {
padding: 10px;
border-radius: 5px;
border: 1px solid #dddddd;
background: transparent;
color: #808080;
cursor: pointer;
}
button:hover {
color: #222;
}
.button-inline {
border-width: 0;
background: transparent;
color: inherit;
text-align: inherit;
-webkit-font-smoothing: inherit;
padding: 0;
font-size: inherit;
cursor: pointer;
}
.button-active {
border-radius: 0;
border-bottom: 1px solid #38BB6C;
}
Redux State as Architecture 167
*:focus {
outline: none;
}
src/components/App.css
.app {
margin: 20px;
}
.interactions, .error {
text-align: center;
}
src/components/Stories.css
.stories {
margin: 20px 0;
}
.stories-header {
display: flex;
line-height: 24px;
font-size: 16px;
padding: 0 10px;
justify-content: space-between;
}
And last but not least, the Story component will get styled too:
Redux State as Architecture 168
src/components/Story.css
.story {
display: flex;
line-height: 24px;
white-space: nowrap;
margin: 10px 0;
padding: 10px;
background: #ffffff;
border: 1px solid #e3e3e3;
}
When you start your application again, it seems more organized by its styling. But there is still
something missing for displaying the stories properly. The columns for each story should be aligned
and perhaps there should be a heading for each column. First, you can define an object to describe
the columns.
src/components/Stories.js
const COLUMNS = {
title: {
label: 'Title',
width: '40%',
},
author: {
label: 'Author',
width: '30%',
},
comments: {
label: 'Comments',
width: '10%',
},
Redux State as Architecture 169
points: {
label: 'Points',
width: '10%',
},
archive: {
width: '10%',
},
};
The last column with the archive property name will not be used yet, but will be used in a later point
in time. Second, you can pass this object to your Story component. Still the Stories component has
the object to use it later on for the column headings.
src/components/Stories.js
The Story component can use it to style each displaying property of the story. It uses inline style to
define the width of each column.
src/components/Story.js
...
return (
<div className="story">
<span style={{ width: columns.title.width }}>
<a href={url}>{title}</a>
</span>
Redux State as Architecture 170
Last but not least, you can use the COLUMNS object to give your Stories component matching header
columns. That’s why the COLUMNS object got defined in the Stories component in the first place.
Now, rather than doing it manually, as in the Story component, you will map over the object
dynamically to render the header columns. Since it is an object, you have to turn it into an array of
the property names first, and then access the object by its mapped keys again.
src/components/Stories.js
)}
</div>
You can extract the header columns as its own StoriesHeader component to keep your component
well arranged.
src/components/Stories.js
const Stories = ({ stories }) =>
<div className="stories">
<StoriesHeader columns={COLUMNS} />
In this part, you have applied styling for your application and components. It should be in a
representable state now. You can find this part of the chapter in the GitHub repository99 .
src/index.js
...
ReactDOM.render(
<App stories={stories} onArchive={() => {}} />,
document.getElementById('root')
);
Second, you can pass it through your App and Stories components. These components don’t use
the function but only pass it to the Story component. You might already notice that this could be a
potential refactoring later on, because the function gets passed from the root component through a
few components only to reach a leaf component.
src/components/App.js
src/components/Stories.js
Finally, you can use it in your Story component in a onClick handler of a button. The story objectID
will be passed in the handler to identify the archived story.
Redux State as Architecture 173
src/components/Story.js
return (
<div className="story">
...
<span style={{ width: columns.archive.width }}>
<button
type="button"
className="button-inline"
onClick={() => onArchive(objectID)}
>
Archive
</button>
</span>
</div>
);
}
A refactoring that you could already do would be to extract the button as a reusable component.
src/components/Story.js
return (
<div className="story">
...
<span style={{ width: columns.archive.width }}>
<ButtonInline onClick={() => onArchive(objectID)}>
Archive
</ButtonInline>
</span>
Redux State as Architecture 174
</div>
);
}
const ButtonInline = ({
onClick,
type = 'button',
children
}) =>
<button
type={type}
className="button-inline"
onClick={onClick}
>
{children}
</button>
You can make even another more abstract Button component that doesn’t share the button-inline
CSS class.
src/components/Story.js
...
const ButtonInline = ({
onClick,
type = 'button',
children
}) =>
<Button
type={type}
className="button-inline"
onClick={onClick}
>
{children}
</Button>
const Button = ({
onClick,
className,
type = 'button',
children
}) =>
Redux State as Architecture 175
<button
type={type}
className={className}
onClick={onClick}
>
{children}
</button>
Both button components should be extracted to a new file called src/components/Button.js, but
exported so that at least the ButtonInline component can be reused in the Story component. You
can find this part of the chapter in the GitHub repository100 . Now, when you start your application
again, the button to archive a story is there. But it doesn’t work because it only receives a no-op
(empty function) as property from your root component. You will later on introduce a Redux action
that can be dispatched to archive a story.
Command Line
npm install --save redux
Second, in the root entry point of React, you can import the Redux store. The store is not defined
yet. Instead of using the sample stories, you will use the stories that are stored in the Redux store.
Taken that the store only saves a list of stories as state, you can simply get the root state of the store
and assume that it is the list of stories.
src/index.js
import React from 'react';
import ReactDOM from 'react-dom';
import App from './components/App';
import store from './store';
import './index.css';
ReactDOM.render(
<App stories={store.getState()} onArchive={() => {}} />,
document.getElementById('root')
);
100
https://github.com/rwieruch/react-redux-hackernews/tree/55de13475aa9c2424b0fc00ce95dd4c5474c0473
Redux State as Architecture 176
Third, you have to create your Redux store instance in a separate file. It already takes a reducer that
is not implemented yet. You will implement it in the next step.
src/store/index.js
import { createStore } from 'redux';
import storyReducer from '../reducers/story';
Fourth, in your src/reducers/ folder you can create your first reducer: storyReducer. It can have the
sample stories as initial state.
src/reducers/story.js
const INITIAL_STATE = [
{
title: 'React',
url: 'https://facebook.github.io/react/',
author: 'Jordan Walke',
num_comments: 3,
points: 4,
objectID: 0,
}, {
title: 'Redux',
url: 'https://github.com/reactjs/redux',
author: 'Dan Abramov, Andrew Clark',
num_comments: 2,
points: 5,
objectID: 1,
},
];
Your application should work when you start it. It is using the state from the Redux store that
is initialized in the storyReducer, because it is the only reducer in your application. There are no
actions yet and no action is captured in the reducer yet. Even though there was no action dispatched,
you can see that the Redux store runs through all its defined reducers to initialize its initial state in the
store. The state gets visible through the Stories and Story components, because it is passed down
from the React root entry point. You can find this part of the chapter in the GitHub repository101 .
src/reducers/archive.js
You will implement the action to archive a story in a second. First, the Redux store in its instantiation
needs to get both reducers now. It has to get the combined reducer. Let’s pretend that the store can
import the combined reducer from the entry file, the reducers/index.js, without worrying about the
combining of the reducers yet.
101
https://github.com/rwieruch/react-redux-hackernews/tree/5aafb21595541c21db778ad8825c97403e44b963
Redux State as Architecture 178
src/store/index.js
Next you can combine both reducers in the file that is used by the Redux store to import the
rootReducer.
src/reducers/index.js
Since your state is sliced up into two substates now, you have to adjust how you retrieve the stories
from your store with the intermediate storyState. This is a crucial step, because it shows how a
combined reducer slices up your state into substates.
src/index.js
ReactDOM.render(
<App
stories={store.getState().storyState}
onArchive={() => {}}
/>,
document.getElementById('root')
);
Redux State as Architecture 179
The application should show up the same stories as before when you start it. You can find this part
of the chapter in the GitHub repository102 . However, there is still no state manipulation happening,
because no actions are involved yet. Finally in the next part you will dispatch your first action to
archive a story.
src/reducers/archive.js
The action type is already outsourced in a different file. This way it can be reused when dispatching
the action from the Redux store.
102
https://github.com/rwieruch/react-redux-hackernews/tree/f6d436fdfdab19296e473fbe7243690e830c1c2b
Redux State as Architecture 180
src/constants/actionTypes.js
Last but not least, you can import the action type and dispatch the action in your root component
where you had the empty function before.
src/reducers/archive.js
ReactDOM.render(
<App
stories={store.getState().storyState}
onArchive={id => store.dispatch({ type: STORY_ARCHIVE, id })}
/>,
document.getElementById('root')
);
Now you dispatch the action directly without an action creator. You can find this part of the chapter
in the GitHub repository103 . When you start your application, it should still work, but nothing
happens when you archive a story. The archived stories are not yet evaluated in the component tree.
The stories prop that is passed to the App component still uses all the stories from the storyState.
103
https://github.com/rwieruch/react-redux-hackernews/tree/5ddbcc2fa8269d615763770a49e7675c5f02d173
Redux State as Architecture 181
src/selectors/story.js
export {
getReadableStories,
};
The selector makes heavily use of JavaScript ES6 arrow functions, JavaScript ES6 destructuring and a
higher order function: isNotArchived(). If you are not used to JavaScript ES6, don’t feel intimidated
by it. It is only a way to express these functions more concise. In plain JavaScript ES5 it would look
like the following:
src/selectors/story.js
function isNotArchived(archivedIds) {
return function (story) {
return archivedIds.indexOf(story.objectID) === -1;
};
}
export {
getReadableStories,
};
Last but not least, you can use the selector to compute the not archived stories instead of retrieving
the whole list of stories from the store directly.
Redux State as Architecture 182
src/index.js
ReactDOM.render(
<App
stories={getReadableStories(store.getState())}
onArchive={id => store.dispatch({ type: STORY_ARCHIVE, id })}
/>,
document.getElementById('root')
);
You can find this part of the chapter in the GitHub repository104 . When you start your application,
nothing happens again when you archive a story. Even though you are using the readable stories
now. That’s because there is no re-rendering of the view in place to update it.
src/index.js
...
function render() {
ReactDOM.render(
<App
stories={getReadableStories(store.getState())}
onArchive={id => store.dispatch({ type: STORY_ARCHIVE, id })}
/>,
document.getElementById('root')
104
https://github.com/rwieruch/react-redux-hackernews/tree/5e3338d3ffff924b7a12eccb691365fd11cb5aed
Redux State as Architecture 183
);
}
store.subscribe(render);
render();
Now the components will re-render once you archive a story, because the state in the Redux
store updates and the subscription will run to render again the whole component tree. In addition,
you render the component only once when the application starts. Congratulations, you dispatched
your first action, selected derived properties from the state and updated your component tree by
subscribing it to the Redux store. That took longer as expected, didn’t it? However, now most of the
Redux and React infrastructure is in place to be more efficient when introducing new features. You
can find this part of the chapter in the GitHub repository105 .
Command Line
src/store/index.js
applyMiddleware(logger)
);
That’s it. Every time you dispatch an action now, for instance when archiving a story, you will see
the logging in the developer console in your browser. You can find this part of the chapter in the
GitHub repository107 .
src/actions/archive.js
export {
doArchiveStory,
};
Second, you can use it in your root component. Instead of dispatching the action object directly, you
can create an action by using its action creator.
107
https://github.com/rwieruch/react-redux-hackernews/tree/652e6419e2a872ba2d1dd65465006b13f0799c4f
Redux State as Architecture 185
src/index.js
function render() {
ReactDOM.render(
<App
stories={getReadableStories(store.getState())}
onArchive={id => store.dispatch(doArchiveStory(id))}
/>,
document.getElementById('root')
);
}
...
The application should operate as before when you start it. But this time you have used an action
creator rather than dispatching an action object directly. You can find this part of the chapter in the
GitHub repository108 .
Command Line
You can use the Provider component, which makes the Redux store available to all components
below, in your React root entry point.
108
https://github.com/rwieruch/react-redux-hackernews/tree/4cc5e995d63fd935a2e335b0a4946a1811c04202
109
https://github.com/reactjs/react-redux
Redux State as Architecture 186
src/index.js
ReactDOM.render(
<Provider store={store}>
<App />
</Provider>,
document.getElementById('root')
);
Notice that the render method isn’t used in a Redux store subscription anymore. No one subscribes
to the Redux store and the App component isn’t receiving any props. In addition, the App component
is only rendering a component and doesn’t pass any props.
src/components/App.js
But who gives the props to the Stories component then? This component is the first component
that needs to know about the list of stories, because it has to display it. The solution is to upgrade the
Stories component to a connected component. It should be connected to the state layer. So, instead
of only exporting the plain Stories component:
Redux State as Architecture 187
src/components/Stories.js
...
You can export the connected component that has access to the Redux store:
src/components/Stories.js
...
The Stories component is a connected component now and is the only component that has access
to the Redux store. It receives the stories from the state in mapStateToProps() and a function that
triggers the dispatching of an action to archive a story in mapDispatchToProps(). The application
should work again, but this time with a sophisticated connection between Redux and React. You can
find this part of the chapter in the GitHub repository110 .
function? It is not used in the Stories component, but only in the Story component and only passed
via the Stories component. Thus you could lift the connection partly. The stories would stay in
the Stories component, but the onArchive() function could live in the Story component.
First, you remove the onArchive() function for the Stories component and remove the mapDis-
patchToProps() as well. It will be used later on in the Story component.
src/components/Stories.js
...
...
Now you can connect the Story component instead. You would have two connected components
afterward.
Redux State as Architecture 189
src/components/Story.js
...
With this refactoring step in your mind, you can always lift your connections to the Redux store
from your view layer depending on the needs of the components. Does the component need state
from the Redux store? Does the component need to alter the state in the Redux store via dispatching
an action? You are in full control of where you want to use connected components and where you
want to keep your components as presenter components. You can find this part of the chapter in the
GitHub repository111 .
Command Line: /
First, you can introduce a root saga in your entry point file to sagas. It can be similar seen to the
combined root reducer, because in the end the Redux store expects one saga for its creation. Basically
the root saga watches all saga triggering actions by using effects such as takeEvery().
111
https://github.com/rwieruch/react-redux-hackernews/tree/779d52fc85ecfbaf5a821cbbae384aac962e76a7
112
https://hn.algolia.com/api
Redux State as Architecture 190
src/sagas/index.js
function *watchAll() {
yield all([
takeEvery(STORIES_FETCH, handleFetchStories),
])
}
Second, the root saga can be used in the Redux store middleware when initializing the saga
middleware. It is used in the middleware, but also needs to be run in a saga.run() method.
src/store/index.js
saga.run(rootSaga);
Third, you can introduce the new action type in your constants that will trigger the saga. However,
you can already introduce a second action type that will later on - when the request succeeds - add
the stories in your storyReducer to the Redux store. Basically you have one action to trigger the
side-effect that is handled with Redux Saga and one action that stores the result of the side-effect in
the Redux store.
Redux State as Architecture 191
src/constants/actionTypes.js
Fourth, you can implement the story saga that encapsulates the API request. It uses the native fetch
API of the browser ti retrieve the stories from the Hacker News endpoint. In your handleFetchSto-
ries() generator function, that is used in your too saga, you can use the yield statement to write
asynchronous code as it would be synchronous code. As long as the promise from the Hacker News
request doesn’t resolve, the next line of code after the yield state will not be evaluated. When you
finally have the result from the API request, you can use the put() effect to dispatch an action.
src/sagas/story.js
function* handleFetchStories(action) {
const { query } = action;
const result = yield call(fetchStories, query);
yield put(doAddStories(result.hits));
}
export {
handleFetchStories,
};
In the fifth step, you need to define both actions creators: the first one that triggers the side-effect to
fetch stories by a search term and the second one that adds the fetched stories to your Redux store.
Redux State as Architecture 192
src/actions/story.js
import {
STORIES_ADD,
STORIES_FETCH,
} from '../constants/actionTypes';
export {
doAddStories,
doFetchStories,
};
Only the second action needs to be intercepted in your storyReducer to store the stories. The first
action is only used to trigger the saga in your root saga. Don’t forget to remove the sample stories
in your reducers.
src/reducers/story.js
Now, everything is setup from a Redux and Redux Saga perspective. As last step, only one component
from the view layer needs to trigger the STORIES_FETCH action. This action is intercepted in the saga,
fetches the stories in a side-effect, and stores them in the Redux store with the STORIES_ADD action.
Therefore, in your App component, you can introduce the new SearchStories component.
src/components/App.js
The SearchStories component will be a connected component. The next step is to implement that
component. First, you start with a plain React component that has a form, input field and button.
src/components/SearchStories.js
this.state = {
query: '',
};
}
Redux State as Architecture 194
render() {
return (
<form onSubmit={this.onSubmit}>
<input
type="text"
value={this.state.query}
onChange={this.onChange}
/>
<Button type="submit">
Search
</Button>
</form>
);
}
}
There are two missing class methods: onChange() and onSubmit(). Let’s introduce them to make
the component complete.
src/components/SearchStories.js
...
this.onChange = this.onChange.bind(this);
this.onSubmit = this.onSubmit.bind(this);
}
onSubmit(event) {
const { query } = this.state;
if (query) {
this.props.onFetchStories(query)
this.setState(applyQueryState(''));
Redux State as Architecture 195
event.preventDefault();
}
onChange(event) {
const { value } = event.target;
this.setState(applyQueryState(value));
}
render() {
...
}
}
The component should work on its own now. It only receives one function from the outside via its
props: onFetchStories(). This function will dispatch an action to trigger the saga that fetches the
stories from the Hacker News platform. You would have to connect the SearchStories component
to make the dispatch functionality available.
src/components/SearchStories.js
...
Start your application again and try to search for stories such as “React” or “Redux”. It should work
now. The connect component dispatches an action that triggers the saga. The side-effect of the
Redux State as Architecture 196
saga is the fetching process of the stories by search term from the Hacker News API. Once the
request succeeds, another action gets dispatched and captured in the storyReducer to finally store
the stories. You can find this part of the chapter in the GitHub repository113 .
src/sagas/story.js
import { call, put } from 'redux-saga/effects';
import { doAddStories } from '../actions/story';
import { fetchStories } from '../api/story';
function* handleFetchStories(action) {
const { query } = action;
const result = yield call(fetchStories, query);
yield put(doAddStories(result.hits));
}
export {
handleFetchStories,
};
src/api/story.js
const HN_BASE_URL = 'http://hn.algolia.com/api/v1/search?query=';
export {
fetchStories,
};
Great, you have separated the API functionality from the saga. You can find this part of the chapter
in the GitHub repository114 .
113
https://github.com/rwieruch/react-redux-hackernews/tree/94efe051bd639aeedce402a33af5acb20397f9f2
114
https://github.com/rwieruch/react-redux-hackernews/tree/b6a6e59af71613471a50c9366c4c4e107e00b66f
Redux State as Architecture 197
src/constants/actionTypes.js
In the second step, you would need an action creator that keeps an error object in its payload and
can be caught in a reducer later on.
src/actions/story.js
import {
STORIES_ADD,
STORIES_FETCH,
STORIES_FETCH_ERROR,
} from '../constants/actionTypes';
...
export {
doAddStories,
doFetchStories,
doFetchErrorStories,
};
The action can be triggered now in your story saga. Redux Saga uses try and catch statements for
error handling. Every time you would get an error in your try block, you would end up in the catch
block to do something with the error object. In this case, you can dispatch your new action to save
an error state in your Redux store.
Redux State as Architecture 198
src/sagas/story.js
function* handleFetchStories(action) {
const { query } = action;
try {
const result = yield call(fetchStories, query);
yield put(doAddStories(result.hits));
} catch (error) {
yield put(doFetchErrorStories(error));
}
}
export {
handleFetchStories,
};
Last but not least, a reducer needs to deal with the new action type. The best place to keep it would
be next to the stories. The story reducer keeps only a list of stories so far, but you could change it to
manage a complex object that holds the list of stories and an error object.
src/reducers/story.js
const INITIAL_STATE = {
stories: [],
error: null,
};
}
default : return state;
}
}
Now you could introduce the second action type in the reducer. It would allocate the error object in
the state but keeps the list of stories empty.
src/reducers/story.js
import {
STORIES_ADD,
STORIES_FETCH_ERROR,
} from '../constants/actionTypes';
...
In your story selector, you would have to change the structure of the story state. The story state
isn’t anymore a mere list of stories but a complex object with a list of stories and an error object.
In addition, you could add a second selector to select the error object. It will be used later on in a
component.
Redux State as Architecture 200
src/selectors/story.js
...
export {
getReadableStories,
getFetchError,
};
Last but not least, in your component you could retrieve the error object in your connect higher
order component and display with React’s conditional rendering115 an error message when an error
occurs.
src/components/Stories.js
...
import {
getReadableStories,
getFetchError,
} from '../selectors/story';
...
...
115
https://www.robinwieruch.de/conditional-rendering-react/
Redux State as Architecture 201
...
In your browser in the developer console, you can simulate being offline. You can try it and see that
an error message shows up when searching for stories. When you go online again and search for
stories, the error message should disappear. Instead a list of stories displays again. You can find this
part of the chapter in the GitHub repository116 .
src/reducers/story.test.js
const action = {
type: 'STORIES_ADD',
stories,
};
116
https://github.com/rwieruch/react-redux-hackernews/tree/a1f6a885357a891b5e94ade90728a1f2d3d1dbb9
117
https://facebook.github.io/jest/
Redux State as Architecture 202
expect(newState).toEqual(expectedNewState);;
});
});
Basically you created the necessary inputs for your reducer and the expected output. Then you can
compare both in your expectation. It depends on your test philosophy whether you create the action
hard coded in the file or import your action creator that you already have in your application. In
this case, a hard coded action was used.
In order to verify that your previous state isn’t mutated when creating the new state, because Redux
embraces immutable data structures, you could use a neat helper library that freezes your state.
Command Line: /
src/reducers/story.test.js
const action = {
type: 'STORIES_ADD',
stories,
};
deepFreeze(previousState);
const newState = storyReducer(previousState, action);
Redux State as Architecture 203
expect(newState).toEqual(expectedNewState);;
});
});
Now every time you would mutate accidentally your previous state an error would show up. It is up
to you to add two more tests for the story reducer. One test could verify that an error object is set
when an error occurs and another test that verifies that the error object is set to null when stories
are successfully added to the state.
Second, you can add a test for your selectors. Let’s demonstrate it with your story selector. Since
the selector function is a pure function again, you can easily test it with an input and an expected
output. You would have to define your global state and use the selector the retrieve an expected
substate.
src/selectors/story.test.js
import { getReadableStories } from './story';
expect(readableStories).toEqual(expectedReadableStories);;
});
});
That’s it. Your Redux state is a combination of the storyState and the archiveState. When both
are defined, you already have your global state. The selector is used to retrieve a substate from the
global state. Thus you would only have to check if all the readable stories that were not archived
are retrieved by the selector.
Third, you can add a test for your action creators. An action creator only gets a payload and returns
an action object. The expected action object can be tested.
Redux State as Architecture 204
src/actions/story.test.js
const expectedAction = {
type: 'STORIES_ADD',
stories,
};
const action = doAddStories(stories);
expect(action).toEqual(expectedAction);;
});
});
As you can see, testing reducers, selectors and action creators follows always a similar pattern. Due
to the functions being pure functions, you can focus on the input and output of these functions. In
the previous examples all three test cases were strictly decoupled. However, you could also decide
to import your action creator in your reducer test avoid creating a hard coded action. You can find
this part of the chapter in the GitHub repository118 .
Final Words
Implementing this application could go on infinitely. I would have plenty of features in my head that
I would want to add to it. What about you? Can you imagine to continue building this application?
From a technical perspective, things that were taught in this book, everything is set up to give you
the perfect starting point. However, there were more topics in this book that you could apply. For
instance, you could normalize your incoming stories from the API before they reach the Redux store.
The following list should give you an idea about potential next steps:
• Normalize: The data that comes from the Hacker News API could be normalized before it
reaches the reducer and finally the Redux store. You could use the normalizr library that
was introduced earlier in the book. It might be not necessary yet to normalize your state,
but in a growing application you would normalize your data eventually. The data would be
normalized between fetching the data and sending it via an action creator to the reducers.
118
https://github.com/rwieruch/react-redux-hackernews/tree/d1fcb31b7a1b1602069718941844d08c21583607
Redux State as Architecture 205
• Local State: So far you have only used Redux. But what about mixing local state into the
application? Could you imagine an use case for it? For instance, you would be able to
distinguish between readable and archived stories in your application. There could be a
toggle, that is true or false in your Stories component as local state, that decides whether
the component shows readable or archived stories. Depending on the toggle in your view
layer, you would retrieve either readable or archived stories via selectors from your Redux
store and display them.
• React Router: Similar to the previous step, using a toggle to show archived and readable stories,
you could add a view layer Router to display these different stories on two routes. It could be
React Router when using React as your view layer. All of this is possible, because fortunately
you don’t delete stories when archiving them from your Redux store, but keep a list of archived
stories in a separate substate.
• Paginated Data: The response from the Hacker News API doesn’t only return the list of stories.
It returns a paginated list of stories with a page property. You could use the page property to
fetch more stories with the same search term. The list component in React could be a paginated
list119 or infinite scroll list120 .
• Caching: You could cache the incoming data from the Hacker News API in your Redux store.
It could be cached by search term. When you search for a search term twice, the Redux store
could be used, when a result by search term is already in place. Otherwise a request to the
Hacker News API would be made. In the Road to learn React121 readers create a cache in
React’s local state. However, the same can be done in a Redux store.
• Local Storage: You already keep track of your archived stories in the Redux store. You could
introduce the native local storage of the browser, as you have seen in the plain React chapters,
to keep this state persistent. When a user loads the application, there could be a lookup in the
local storage for archived stories. If there are archived stories, they could be rehydrated into
the Redux store. When a story gets archived, it would be dehydrated into the local storage too.
That way you would keep the list of archived stories in your Redux store and local storage in
sync, but would add a persistent layer to it when an user closes your application and comes
back later to it.
As you can see, there are a multitude of features you could implement or techniques you could make
use of. Be curious and apply these on your own. After you come up with your own implementations,
I am keen to see them. Feel free to reach out to me on Twitter122 .
119
https://www.robinwieruch.de/react-paginated-list/
120
https://www.robinwieruch.de/react-infinite-scroll/
121
https://www.robinwieruch.de/the-road-to-learn-react/
122
https://twitter.com/rwieruch
Redux Ecosystem Outline
After learning the basics and advanced techniques in Redux and applying them on your own in an
application, you are ready to explore the Redux ecosystem. The Redux ecosystem is huge and cannot
be covered in one book. However, this chapter attempts to outline different paths you can take to
explore the world of Redux. Apart from outlining these different paths, a couple of topics will be
revisited as well to give you a richer toolset when using Redux.
Before you are left alone with the last chapter covering Redux, I want to make you aware of this
repository123 by Mark Erikson. It is a categorized list of Redux related add-ons, libraries and articles.
If you get stuck at some point, want to find a solution for your problem or are just curious about the
ecosystem, check out the repository. Otherwise, I encourage you to join the official Slack Group124
for further recommendations.
123
https://github.com/markerikson/redux-ecosystem-links
124
https://slack-the-road-to-learn-react.wieruch.com/
Redux Ecosystem Outline 207
Redux DevTools
The Redux DevTools are essential for many developers when implementing Redux applications. It
improves the Redux development workflow by offering a rich set of features such as inspecting the
state and action payload, time traveling and realtime optimizations.
How does it work? Basically, you have two choices to use the Redux DevTools. Either you integrate
it directly into your project by using its node package with npm or you install the official browser
extension. While the former comes with an implementation setup in your application, the latter can
simply be installed for your browser without changing your implementation.
The most obvious feature is to inspect actions and state. Rather than using the redux-logger125 , you
can use the Redux DevTools to get insights into these information. You can follow each state change
by inspecting the action and the state.
Another great feature is the possibility to time travel. In Redux you dispatch actions and travel
from one state to another state. The Redux DevTools enable you to travel back in time by reverting
actions. For instance, that way you wouldn’t need to reload your browser anymore to follow a set of
actions to get to a specific application state. You could simply alter the actions in between by using
the Redux DevTools. You can trace back what action led to which state.
In addition, you can persist your Redux state when doing page reloads with Redux DevTools. That
way, you don’t need to perform all the necessary actions to get to a specific state anymore. You simply
reload the page and keep the same application state. This enables you to debug your application when
having one specific application state.
However, there are more neat features that you might enjoy while developing a Redux application.
You can find all information about the Redux DevTools in the official repository126 .
125
https://github.com/evgenyrodionov/redux-logger
126
https://github.com/gaearon/redux-devtools
Redux Ecosystem Outline 208
Connect Revisited
In one of the previous chapters, you have connected your view layer to your state layer with react-
redux127 . There you have used the provider pattern in React to make the state accessible to your
entire view layer.
The connect higher order components enabled you to wire the Redux store to any component. The
most often used two arguments are mapStateToProps() and mapDispatchToProps() for the connect
higher order component. While the former gives access to the state, the latter gives access to actions
to be dispatched for manipulating the state.
However, connect has two more optional arguments that shouldn’t stay uncovered in this book.
The third argument is called mergeProps(). As arguments it gets the result from mapStateTo-
Props(), mapDispatchToProps() and the parent props: mergeProps(stateProps, dispatchProps,
ownProps). The function returns props as an object to the wrapped component. Basically, it gives
you an intermediate layer to mix up stateProps and dispatchProps before they reach the wrapped
component. However, it is rarely used. Often, when mixing up state and actions in this layer, it is
associated with a bad state architecture. You should ask yourself if something else can be changed
to avoid this intermediate layer.
The fourth argument is called options. It is an object to configure the connect higher order com-
ponent. It comes with these additional properties: pure, areStatesEqual(), areOwnPropsEqual(),
areMergedPropsEqual(). How does it work altogether? When the first argument, the pure property,
is set to true, the connect higher order component will avoid re-rendering the view and avoids
the calls to its arguments mapStateToProps(), mapDispatchToProps() and mergeProps(). But only
when the equality checks of areStatesEqual(), areOwnPropsEqual(), areMergedPropsEqual()
remain equal based on their respective equality checks. These equality checks are performed on
the previous state and props and updated state and props. These equality checks can be modified
in the options areStatesEqual, areOwnPropsEqual, areMergedPropsEqual. Otherwise they have a
default equality check.
After all, the options are a pure performance optimization. It is rarely used when developing Redux
applications. Basically, you can set the pure property to true to avoid re-renderings and other
argument evaluations of the connect higher order component. But it comes with certain default
equality checks that can be configured. In addition, the underlying assumption is that the wrapped
component is a pure component and doesn’t rely on any other side-effect data.
If you want to read up the connect higher order component again, you can checkout the official
repository of react-redux128 and look for the connect chapter.
127
https://github.com/reactjs/react-redux
128
https://github.com/reactjs/react-redux
Redux Ecosystem Outline 209
Code Playground
The doAddTodo() is an action creator. It uses the specified action type ‘TODO_ADD’. When using it,
you can pass a payload when needed. It becomes automatically allocated under a payload property.
Code Playground
const action = doAddTodo({ id: '0', name: 'learn redux', completed: false });
// action: {
// type: 'TODO_ADD',
// payload: {
// id: '0',
// name: 'learn redux',
// completed: false
// }
// }
The handleAction() method is a utility for reducers. It aligns action types with reducers whereas
no switch case statement is needed anymore. It takes the action type as argument and a reducer
function for handling the incoming action. As third argument, it takes an initial state.
129
https://github.com/acdlite/redux-actions
Redux Ecosystem Outline 210
Code Playground
The two methods createAction() and handleAction() have sibling methods for using, creating,
and handling multiple actions too: createActions() and handleActions(). Especially when defin-
ing a reducer, it makes sense to map multiple action types to multiple handlers.
Code Playground
As you can see, it is far more concise than defining reducers in plain JavaScript.
Code Playground
The drawback when using the library is that it hides how Redux works with plain JavaScript. It can
be difficult for newcomers to grasp what’s going on when using such utility libraries from the very
beginning without understanding how actions and reducers in Redux work.
Redux Ecosystem Outline 211
The library is only a small utility belt for Redux, yet a lot of people are using it. You can read up
everything about it in the official documentation130 .
130
https://github.com/acdlite/redux-actions
Redux Ecosystem Outline 212
Typed Redux
JavaScript by nature is an untyped language. You will often encounter bugs in your career that could
have been prevented by type safety. In Redux, type safety can make a lot of sense, because you can
define exactly what kind of types go into your actions, reducers or state. You could define that an
action that creates a todo item would have the property name with the type String and the property
completed with the type Boolean. Every time you pass a wrong typed value for these properties to
create a todo item, you would get an error on compile time of your application. You wouldn’t wait
until your application runs to figure out that you have passed a wrong value to your action. There
wouldn’t be a runtime exception when you can already cover these bugs during compile time.
Typed JavaScript can be a verbose solution when working on short living or simple projects. But
when working in a large code base, where code needs to be kept maintainable, it is advisable to use
a type checker. It makes refactorings easier and adds a bunch of benefits to the developer experience
due to editor and IDE integrations.
There exist two major solutions gradually using JavaScript as a typed language: Flow (Facebook)
and TypeScript (Microsoft). While the former has its biggest impact in the React community, the
latter is well adopted amongst other frameworks and libraries.
What would a type checker like Flow look like when using in Redux? For instance, in a todo reducer
the state could be defined by a type:
Code Playground
type Todo = {
id: string,
name: string,
completed: boolean,
};
Now, whenever an action leads to a state that is not defined by its type definition, you would get
an error on compile time. In addition, you could use plugins for your editor or IDE to give you the
Redux Ecosystem Outline 215
early feedback that something is wrong with your action or reducer. As the previous example has
shown type safety for reducers, you could apply the same for your action creators and selectors.
Everything can be type checked. You can read more about Flow on its official site137 .
137
https://flow.org/
Redux Ecosystem Outline 216
Server-side Redux
Server-side rendering is used to render the initial page load from a server. Every further user
interaction is done on the client-side. For instance, it is beneficial for SEO, because when a web
crawler visits your website, it can retrieve the whole application without bothering to execute
JavaScript on the client-side. It retrieves the whole application with its initial state. The initial
state can already be data that is fetched from a database. In React, but also in other single page
applications, there are solutions to deal with server-side rendering. However, introducing server-
side rendering comes with a handful of challenges. One of these challenges is state management.
When the initial page is rendered by the server-side, the initial state must be sent as a response to
the client as well. The client in return would use the initial state. For instance, imagine you want
to load data from a database before you send the response from the server to the client. Somehow
you would have to put this data into the response next to your server-side rendered application.
Afterward, the client can use the response to render the application and would already have the
initial state that comes from a database. If the data wasn’t sent along in the initial server request,
the client would have to fetch it again.
In Redux, you can initialize a Redux store anywhere. You can initialize it on a client-side to access
and manipulate the state, but also on the server-side to provide your application with an initial state.
The initial state would be put in the Redux store before the server-sided response is send to the client
application. But how does it work? The Redux store on the client-side is a singleton. There is only
one instance of the Redux store. On the server-side, the Redux store isn’t a singleton. Every time a
server-side request is made, it would initialize a new instance of the Redux store. The Redux store
can be filled with an initial state before the server-side response is sent to a client.
Server-side rendering and state management open up a whole new topic. That’s why the book
doesn’t cover the topic but only points you in the right direction. You can read more about the
topic in the official Redux documentation138 .
138
http://redux.js.org/docs/recipes/ServerRendering.html
MobX
The next chapters of the book will dive into an alternative library that provides a state management
solution: MobX. However, the book will not allocate the same space as for Redux and thus not deeply
dive into the topic. Because MobX doesn’t follow an opinionated way of how to structure your state
management, it is difficult to tackle it from all angles. There are several ways on where to put your
state and how to update it. The book shows only a few opinionated ways, but doesn’t showcase all
of them.
MobX139 advertises itself as simple yet scalable state management library. It was created and
introduced by Michel Weststrate140 and heavily used, thus battle tested, in his own company. MobX
is an alternative to Redux for state management. It grows in popularity even though only a fraction
of people uses it as a state management alternative to Redux. In a later chapter, you can read about
the differences between both libraries for state management, because you may want to make an
informed decision on whether you should use Redux or MobX to scale your state management.
The library uses heavily JavaScript decorators that are not widely adopted and supported by
browsers yet. But they are not mandatory and you can avoid using them with plain functions. You
can find these plain functions in the official documentation141 . However, this book will showcase
the usage of these decorators, because it is another exciting way of using JavaScript.
Along the way of the following chapters you can decide to opt-in any time the MobX + React
DevTools142 . You can install the node package with npm and place the DevTools component that
comes from the library somewhere between your components.
139
https://mobx.js.org/
140
https://twitter.com/mweststrate
141
https://mobx.js.org/
142
https://github.com/mobxjs/mobx-react-devtools
MobX 218
Introduction
MobX is often used in applications that have a view layer such as React. Thus the state, similar to
Redux, needs to be connected to the view. It needs to be connected in a way that the state can be
updated and the updated state flows back into the view.
Concept Playground
The schema can be elaborated to give more detail about MobX and its parts.
Concept Playground
View -> (Actions) -> State -> (Computed Values) -> Reactions -> View
It doesn’t need to be necessarily the view layer, but when using MobX in an application with
components, most likely the view will either mutate the state directly or use a MobX action to mutate
it. It can be as simple as a onClick handler in a component that triggers the mutation. However, the
mutation could also be triggered by a side-effect (e.g. scheduled event).
The state in MobX isn’t immutable, thus you can mutate the state directly. Actions in MobX can
be used to mutate the state too, but they are not mandatory. You are allowed to mutate the state
directly. There is no opinionated way around how to update the state. You have to come up with
your own best practice.
In MobX the state becomes observable. Thus, when the state changes, the application reacts to the
changes with so called reactions. The part of your application that uses these reactions becomes
reactive. For instance, a MobX reaction can be as simple as a view layer update. The view layer in
MobX becomes reactive by using reactions. It will update when the state in MobX updates.
In between of an observable MobX state and MobX reactions are computed values. These are
not mandatory, similar to the MobX Actions, but add another fine-grained layer into your state.
Computed values are derived properties from the state or from other computed values. Apart from
the MobX state itself, they are consumed by reactions too. When using computed values, you can
keep the state itself in a simple structure. Yet you can derive more complex properties from the state
with computed values that are used in reactions too. Computed values evaluate lazily when used in
reactions when the state changes. They don’t necessarily update every time the state changes, but
only when they are consumed in a reaction that updates the view.
These are basically all parts in MobX. The state can be mutated directly or by using a MobX action.
Reactions observe these state changes and consume the state itself or computed values from the state
or other computed values. Both ends, actions and reactions, can simply be connected to a view layer
such as React. The connection can happen in a straight forward way or with a bridging library as
you will experience it in the following chapters.
MobX 219
Observable State
The state in MobX can be everything from JavaScript primitives to complex objects, arrays or only
references over to classes that encapsulate the state. Any of these properties can be made observable
by MobX. When the state changes, all the reactions, for instance the reaction of the view layer, will
run to re-render the view. State in MobX isn’t managed in one global state object. It is managed in
multiple states that are most of the time called stores or states.
Code Playground
class TodoStore {
@observable todos = [];
}
Keep in mind that it doesn’t need to be managed in a store instance that comes from a JavaScript
class. It can be a plain list of todos. The way of using stores to manage your MobX state is already
opinionated. Since there are a couple of different ways on where to store your state in MobX, the
book will teach the straight forward way of managing it in stores. In the end, stores enable you to
manage a predictable state where every store can be kept responsible for its own substate.
The state in MobX can be mutated directly without actions:
Code Playground
That means as well, that the store instances can leak into the view layer and an onClick handler
could mutate the state directly in the store.
State and view layer can be coupled very closely in MobX. In comparison to Redux, it doesn’t need
to use explicit actions to update the state indirectly. You will get to know more about MobX actions
in a later chapter.
Autorun
The autorun functionality in MobX is not often seen. It is similar to the subscription() method
of the Redux store. It is always called when the observable state in MobX changes and once in the
beginning when the MobX state initializes. Similar to the subscription() method of the Redux
MobX 220
store, it is later on used to make the view layer reactive with MobX. The autorun function is only
one way to produce a reaction on MobX.
However, you can use it to experiment with your state updates while learning MobX. You can simply
add it to your TodoStore example.
Code Playground
class TodoStore {
@observable todos = [];
}
It will run the first time when the state initializes, but then every time again when the observable
state updates. You can open the MobX Playground143 to experiment with it.
Actions
As mentioned, the state can be mutated directly in MobX. But a handful of people would argue that
it is a bad practice. It couples the state mutation to close to the view layer when you start to mutate
the state directly in an onClick handler. Therefore, you can use MobX actions to decouple the state
update and keep your state updates at one place.
Code Playground
class TodoStore {
@observable todos = [];
@action addTodo(todo) {
this.todos.push(todo);
}
}
143
https://jsbin.com/vawugugugu/4/edit?js,console
MobX 221
However, MobX is not opinionated about the way you update your state. You can use actions or
mutate the state directly without an action.
Code Playground
class TodoStore {
@observable todos = [];
addTodo(todo) {
this.todos.push(todo);
}
}
In order to enforce state updates with actions, you can opt-in the useStrict() functionality by
MobX. After you have set it to true, every state update needs to happen via an action. This way you
enforce the decoupling of state mutation and view with actions.
Code Playground
useStrict(true);
class TodoStore {
@observable todos = [];
@action addTodo(todo) {
this.todos.push(todo);
}
}
You can test the MobX action and the useStrict() function in the MobX Playground144 .
In addition, it makes always sense to think thoughtfully about your actions. In the previous case,
every call of addTodo() would lead all relying reactions to run. That’s why the autorun function
runs every time you add a todo item. So how would you accomplish to add multiple todo items at
once without triggering reactions for every todo item? You could have another action that takes an
array of todo items.
Code Playground
class TodoStore {
@observable todos = [];
@action addTodo(todo) {
this.todos.push(todo);
}
@action addTodos(todos) {
todos.forEach(todo => this.addTodo(todo));
}
}
todoStore.addTodos([
{ id: '0', name: 'learn redux', completed: true },
{ id: '1', name: 'learn mobx', completed: false },
]);
That way, the relying reactions only evaluate once after the action got called. You can find the
necessary code to play around with in the MobX Playground145 .
144
https://jsbin.com/qazazajusa/1/edit?js,console
145
https://jsbin.com/loyuser/10/edit?js,console
MobX 223
Computed Values
Computed values are derived properties from the state or other computed values. They have no
side-effects and thus are pure functions. The computed values help you to keep your state structure
simple yet can derive complex properties from it. For instance, when you would filter a list of todos
for their completed property, you could compute the values of uncompleted todo items.
Code Playground
class TodoStore {
@observable todos = [];
@action addTodo(todo) {
this.todos.push(todo);
}
The computation happens reactively when the state has changed and a reaction asks for it. Thus,
these computed values are at your disposal, apart from the state itself, for your view layer later
on. In addition, they don’t compute actively every time but rather only compute reactively when a
reaction demands it. You can experiment with it in the MobX Playground146 .
146
https://jsbin.com/didenujipi/3/edit?js,console
MobX in React
The basics in MobX should be clear by now. The state in MobX is mutable and can be mutated
directly, by actions too or only by actions, and not directly, when using the strict mode. When
scaling your state management in MobX, you would keep it in multiple yet manageable stores to
keep it maintainable. These stores can expose actions and computed values, but most important they
make their properties observable. All of these facts already give you an opinionated way of how to
store state (e.g. with stores) and how to update the state (e.g. explicit actions with strict mode).
However, you can decide to use a different opinionated approach.
Now, every time an observable property in a store changes, the autorun function of MobX runs.
The autorun makes it possible to bridge the MobX state updates over to other environments. For
instance, it can be used in a view layer, such as React, to re-render it every time the state changes.
MobX and React match very well. Both libraries solve their own problem, but can be used together
to build a sophisticated scaling application. The React view layer can receive the state from MobX,
but also can mutate the state. It can connect to both ends: getting state and mutating it.
When you start to introduce React to your MobX Playground, you could begin to display the list of
todo items from your todoStore.
Code Playground
ReactDOM.render(
<TodoList todoStore={todoStore} />,
document.getElementById('app')
);
MobX in React 225
However, when you update your MobX store nothing happens. The view layer is not notified about
any state updates, because these happen outside of React. You can use the autorun function of MobX
to introduce a naive re-rendering of the view layer.
Code Playground
function render() {
ReactDOM.render(
<TodoList todoStore={todoStore} />,
document.getElementById('app')
);
}
autorun(render);
Now you have one initial rendering of the view layer, because the autorun is running once initially,
and successive renderings when the MobX state updates. You can play around with it in the MobX
Playground147 .
There exists a neat library that bridges from MobX to React: mobx-react148 . It spares you to use the
autorun reaction in order to re-render the view layer. Instead it uses a observer decorator, that uses
the autorun function under the hood, to produce a reaction. The reaction simply flushes the update
to a React component to re-render it. It makes your React view layer reactive and re-renders it when
the observable state in MobX has changed.
Code Playground
...
@observer
class TodoList extends React.Component {
render() {
return (
<div>
{this.props.todoStore.todos.map(todo =>
<div key={todo.id}>
{todo.name}
</div>
)}
</div>
147
https://jsbin.com/vobicoqifu/1/edit?html,js,output
148
https://github.com/mobxjs/mobx-react
MobX in React 226
);
}
}
ReactDOM.render(
<TodoList todoStore={todoStore} />,
document.getElementById('app')
);
...
Again you can play around with it in the MobX Playground149 . If you want to use the TodoList
component as functional component, you can use the observer as well.
Code Playground
Local State
As mentioned before, MobX is not opinionated about how to store your state and how to update it. It
goes so far, that you can even exchange your local state in React, this.state and this.setState(),
with MobX. The case can be demonstrated by introducing a component that adds a todo item to the
list of todo items from the previous example.
Code Playground
ReactDOM.render(
<div>
<TodoAdd todoStore={todoStore} />
<TodoList todoStore={todoStore} />
</div>,
document.getElementById('app')
);
The TodoAdd component only renders an input field to capture the name of the todo item and a
button to create the todo item.
Code Playground
@observer
class TodoAdd extends React.Component {
render() {
return (
<div>
<input
type="text"
value={this.input}
onChange={this.onChange}
/>
<button
type="button"
onClick={this.onSubmit}
>Add Todo</button>
</div>
);
}
}
The two handlers class methods are missing. The onChange handler can be an action itself to update
a internally managed value of the input field.
MobX in React 228
Code Playground
@observer
class TodoAdd extends React.Component {
render() {
...
}
}
This way, the input property is not allocated in the local state of React, but in the observable state
of MobX. The observer decorator makes sure that the component stays reactive to its observed
properties. The onSubmit handler finally creates the todo item yet alters the local state of the
component again, because it has to reset the input value and increments the identifier.
Code Playground
@observer
class TodoAdd extends React.Component {
this.id++;
this.input = '';
}
render() {
...
}
}
MobX is able to take over the local state of React. You wouldn’t need to use this.state and
this.setState() anymore. The previous example can be found in the MobX Playground151 . Again
you experience that MobX isn’t opinionated about the way the state is managed. You can have the
state management encapsulated in a store class or couple it next to a component as local state. It can
make React local state obsolete. But should it? That is your own decision.
151
https://jsbin.com/sonate/1/edit?html,js,output
MobX in React 230
Scaling Reactions
Each component can be decorated with an observer to be reactive to observable state changes in
MobX. When introducing a new component to display a todo item, you can decorate it as well. This
TodoItem component receives the todo property, but also the todoStore in order to complete a todo
item.
Code Playground
Notice that the TodoItem is a functional stateless component. In addition, in order to complete a
todo item, you would have to introduce the toggleCompleted action in the TodoStore.
Code Playground
class TodoStore {
@observable todos = [];
...
@action toggleCompleted(todo) {
todo.completed = !todo.completed;
}
}
Code Playground
@observer
class TodoList extends React.Component {
render() {
return (
<div>
{this.props.todoStore.todos.map(todo =>
<TodoItem
todoStore={this.props.todoStore}
todo={todo}
key={todo.id}
/>
)}
</div>
);
}
};
In the running application you should be able to complete a todo item. The benefit of splitting up one
reactive component into multiple reactive components can be seen when adding two console.log()
statements.
Code Playground
const TodoItem = observer(({ todo, todoStore }) => {
console.log('TodoItem: ' + todo.name);
return (
...
);
});
@observer
class TodoList extends React.Component {
console.log('TodoList');
render() {
...
}
};
You can open up the application in the MobX Playground152 . When you add a todo item, you
will get the console.log() outputs for the TodoList component and only the newly created
152
https://jsbin.com/sonate/6/edit?js,console,output
MobX in React 232
TodoItem. When you complete a todo item, you will only get the console.log() of the completing
TodoItem component. The reactive component only updates when their observable state changes.
Everything else doesn’t update, because the observer decorator implements under the hood the
shouldComponentUpdate() lifecycle method of React to prevent the component from updating when
nothing has changed. You can read more about optimizing MobX performance in React in the official
documentation153 .
153
https://mobx.js.org/best/react-performance.html
MobX in React 233
Inject Stores
So far, the application passes down the store from the React entry point via props to its child
components. They are already passed down more than one layer. However, the store(s) could be
used directly in the components by using them directly (when accessible in the file). They are only
observable state. Since MobX is not opinionated about where to put state, the observable state, in this
case stores, could live anywhere. But as mentioned, the book tries to give an opinionated approach
as best practice.
The mobx-react154 library provides you with two helpers to pass the observable state implicitly down
to the components (via React‘s context) rather than passing them through every component layer
explicitly.
The first helper is the Provider component that passes down all the necessary observable states
down.
Code Playground
...
You can use it in the React entry point to wrap your component tree. In addition, you can pass it any
observable state that should be passed down. In this case, the observable state is the store instance.
Code Playground
...
ReactDOM.render(
<Provider todoStore={todoStore}>
<div>
<TodoAdd />
<TodoList />
</div>
</Provider>,
document.getElementById('app')
);
154
https://github.com/mobxjs/mobx-react
MobX in React 234
Code Playground
<Provider
storeOne={storeOne}
storeTwo={storeTwo}
anyOtherState={anyOtherState}
>
...
</Provider>
The second helper from the library is the inject decorator. You can use it for any component down
your component tree that is wrapped somewhere above by the Provider component. It retrieves the
provided observable state from React’s context as props.
Code Playground
...
@inject('todoStore') @observer
class TodoAdd extends React.Component {
...
}
The TodoAdd component already has access to the todoStore now. You can add the injection to the
other components too. It can be used as function for functional stateless components.
Code Playground
</button>
</div>
));
The TodoList component doesn’t need to manually pass down the todoStore anymore. The
TodoItem already accesses it via its inject helper.
Code Playground
@inject('todoStore') @observer
class TodoList extends React.Component {
render() {
return (
<div>
{this.props.todoStore.todos.map(todo =>
<TodoItem
todo={todo}
key={todo.id}
/>
)}
</div>
);
}
};
Every component can access the observable state, that is passed to the Provider component, with
the inject decorator. This way you keep a clear separation of state and view layer. You can access
the project in the MobX Playground155 again.
155
https://jsbin.com/sonate/7/edit?js,output
MobX in React 236
Advanced MobX
MobX is not opinionated. Thus it gives you a handful of tools to accomplish your on way of
mastering state management. It would be sufficient to use the basics of MobX to introduce state
management in your application. But there are more tools hidden in MobX that this chapter is
going to point out. It addition, this chapter should give you a couple more pillars to understand and
use MobX successfully in your own way.
Other Reactions
You have encountered two reactions by now: autorun and observer. The observer produces a reaction
because it uses autorun under the hood. It is only used in the mobx-react package. Thus, both
functions are used to create reactions based on observable state changes. While the autorun function
can be used to re-render naively the UI, it can also used for broader domains. The observer decorator
is solely used to make a view-layer reactive.
However, MobX comes with more reactions. The book will not go too much into detail here, but it
does no harm to be aware of other options too. The MobX when156 is another function that produces
a reaction. It is based on predicates and effects. A given predicate runs as long as it returns true.
When it returns true, the effect is called. After that the autorunner is disposed. The when function
returns a disposer to cancel the autorunner prematurely, so before an effect can be called.
Code Playground
class TodoStore {
@observable todos = [];
constructor() {
when(
// once (predicate)...
() => this.hasCompleteTodos,
// ... then (effect)
() => this.celebrateAccomplishment()
);
}
celebrateAccomplishment() {
console.log('First todo completed, celebrate it!');
}
}
todoStore.todos.push({ id: '0', name: 'finish the book', completed: false });
todoStore.todos.push({ id: '1', name: 'learn redux', completed: true });
todoStore.todos.push({ id: '2', name: 'learn mobx basics', completed: true });
todoStore.todos.push({ id: '3', name: 'learn mobx', completed: false });
So how many times does the reaction run? You can take your guess first and afterward open the
MobX Playground157 to experience the reaction yourself. Basically the when triggers its effect when
the predicate returns true. But it only triggers once. As you can see, two todo items that are completed
are added. However, the celebrateAccomplishment() method only runs once. This way MobX
allows you to use its reactions to trigger side-effects. You could trigger anything ranging from an
animation to an API call.
Another function in MobX, the reaction function158 itself, can be used to produce MobX reactions
too. It is a fine-grained version of autorun. Whereas autorun will always run when the observable
state has changed, the reaction function only runs when a particular given observable state has
changed.
Code Playground
class TodoStore {
@observable todos = [];
constructor() {
reaction(
157
https://jsbin.com/loyuser/4/edit?js,console
158
https://mobx.js.org/refguide/reaction.html
MobX in React 238
() => this.completeTodos.length,
sizeCompleteTodos => console.log(sizeCompleteTodos + " todos completed!")
);
}
todoStore.todos.push({ id: '0', name: 'finish the book', completed: false });
todoStore.todos.push({ id: '1', name: 'learn redux', completed: true });
todoStore.todos.push({ id: '2', name: 'learn mobx basics', completed: true });
todoStore.todos.push({ id: '3', name: 'learn mobx', completed: false });
How many times does the reaction run? First you can have a guess, afterward you can confirm it
by trying it in the MobX Playground159 .
Now you have seen two more functions in MobX that produce reactions: when and reaction.
Whereas the when function only runs once an effect when the predicate returns null, the reaction
runs every time when a particular observable state has changed. You can use both to trigger side-
effects, such as an API call.
Be Opinionated
MobX gives you all the tools that are needed to manage state in modern JavaScript application.
However, it doesn’t give you an opinionated way of doing it. This way you have all the freedom to
manage your state yet it can be difficult to follow best practices or to align a team on one philosophy.
That’s why it is important to find your own opinionated way of doing things in MobX. You have to
align on one opinionated way to manage your state.
The chapters before have shown you that observable state in MobX can be far away managed in
stores yet it could be used in the local state of the view layer too. Should MobX be used instead of
this.state and this.setState() in React? Be clear about how close you want to keep your MobX
state to your view layer.
Another thing you should have an opinion about is how you update your observable state. Do you
mutate the state directly in your view? Going this path would lead to coupling your state closer to
159
https://jsbin.com/loyuser/5/edit?js,console
MobX in React 239
your view layer. On the other hand, you could use explicit MobX actions. It would keep your state
mutation at one place. You can make them even mandatory by using the useStrict() functionality
provided by MobX. That way, every state mutation would have to go through an explicit action. No
direct mutations of the state would be allowed anymore. Recommendation: You should make your
state mutations as explicit as possible with actions and useStrict().
When using MobX to complement your view layer, you would need to decide on how to pass your
state around. You can simply allocate your state next to your components, import it directly from
another file using JavaScript ES6 import and export statements, pass it down explicitly (e.g. in React
with props) or pass it down implicitly from your root component with inject() function and the
Provider component. You should avoid to mix up these things and follow one opinionated way.
Recommendation: You should use the inject() function and Provider component to make your
state implicitly accessible to your view layer.
Last but not least, you would need to align on a state structure. Observable state in MobX can be
anything. It can be primitives, it can be objects or arrays but it can also be store instances derived
from JavaScript classes. Without mixing up everything, you would need to align on a proper state
architecture. The approach to manage your state in stores, as shown in the previous chapters, gives
you a maintainable way to manage your state for specific domains. In addition, you are able to
keep actions, computed values and even reactions such as autorun, reaction and when in your store.
Recommendation: You should use JavaScript classes to manage your state in stores. That way your
state management stays maintainable by domain related stores as stakeholders.
As you can see, there are a handful of decisions to make on how to use MobX. It gives you all the
freedom to decide your own way of doing things, but after all you have to establish the opinionated
way yourself and stay disciplined with it.
MobX in React 240
Alternative to Redux?
After all, is MobX a viable alternative to Redux? It depends all on yourself. Both solutions are
different in their philosophy, their underlying mechanics and in their usage. Whereas Redux gives
you one opinionated way of managing your state, MobX gives you only the tools to manage your
state but not the way of how to do things. Redux has a large community, a vibrant ecosystem of
libraries and great selection of best practices. MobX is a younger library compared to Redux, but
gives you a different approach of managing your state.
The defining powers of MobX come from its reactive nature. As you have seen when you connected
your view layer to MobX with observers, only the reactive components updated that relied on an
observable state change. Everything else stayed untouched. In a large scale application, it can keep
your view layer updates to a minimum when using MobX the right way.
In MobX you don’t need to normalize your state. You can work with references and keep your state
nested. It stays simple to update your state with mutations not worrying about immutability. On
the other hand, you have to be cautious on how close you couple your state to your view layer. In
the end, when the state is too close to your view layer, it could end up the same way as for the first
generation of single page applications were two-way data binding became a mess.
If you want to read more about the differences of Redux and MobX, I recommend you to check
out the following article: Redux or MobX: An attempt to dissolve the Confusion160 . After that you
might come to a more informed decision on what you want to use for state management in your
own application.
160
https://www.robinwieruch.de/redux-mobx-confusion/
Last but not Least
Well, if I haven’t lost you by now, you can read a bit further to make it to the end of this book. The
last chapters of this book have the objective to inspire you to apply your learnings. You will read
about further learning paths after you read the book, a life hack to improve your learning experience
and other people in the (React and) Redux ecosystem that you might want to follow for inspirations.
So far, the book has taught you different approaches of state management. Whether you are using
React, an alternative view layer library or a sophisticated SPA library; most of them will come with a
built-in solution to deal with local state. The book has shown you React’s local state management and
demonstrated approaches to scale it in plain React applications. Afterward, you learned extensively
about Redux as sophisticated state management library. It can be used in combination with any view
layer or SPA library. The book has taught you how to use it in React applications, too. As alternative
to Redux, you read about MobX as sophisticated state management library. It comes with its own
advantages and disadvantages. After all, Redux and MobX give you two different approaches to
opt-in state management to your application. However, you should never forget about your local
state management solution to keep your state coupled to your components rather than exposing it
in your global state.
Last but not Least 242
You can keep yourself educated in this area by reading articles about it. I write a lot about these
topics on my personal website. You can give it a shot and dig into some of the topics. One of it
gives you a list of useful tips to learn React and Redux165 . Two more great external resources are the
repositories by Mark Erikson: react-redux-links166 and redux-ecosystem-links167 . I would argue that
you will find any solution to a problem in them. In addition, you will dive deeper into the techniques
and tools in Redux.
If you want to go beyond Redux, you can give Relay Modern168 or Apollo Client169 a shot. Both enable
you to build GraphQL170 consuming client application. By using Relay Modern or Apollo Client, you
can give your application state management in your client API layer rather than application layer.
I am sure you will find more educating material on them if you search for it. In addition, I have
more upcoming tutorials on those technologies if you want to keep an eye on them. Similar to the
platforms that expose REST APIs, you could for instance look up platforms that expose a GraphQL
API. In your own client application you could use Relay Modern to consume the API. Perhaps there
already is a tutorial that explains how to consume these GraphQL APIs.
As you can see, there are endless possibilities to apply your learnings. Don’t hesitate and jump into
coding. I am curious what you come up with, so please reach out to me.
165
https://www.robinwieruch.de/tips-to-learn-react-redux/
166
https://github.com/markerikson/react-redux-links
167
https://github.com/markerikson/redux-ecosystem-links
168
https://facebook.github.io/relay/docs/relay-modern.html
169
https://github.com/apollographql/apollo-client
170
http://graphql.org/
Last but not Least 244
• 5% Lecture
• 10% Reading (Uuups!)
• 20% Audiovisual (Phew… I hope you have purchased the Screencasts too)
• 30% Demonstration (Check!)
• 50% Discussion (Did I mention that there is a Slack Group171 ?)
• 75% Practice by Doing (I hope you did all practical chapters on your own! If you have
purchased the Source Code as well, use these to internalize your learnings.)
• 90% Teach Others (You can help others in the Slack Group172 when they struggle!)
Perhaps you recall it from somewhere, but it is always refreshing to see it again. In the beginning of
the book, I told you that nobody became perfect by reading a book. You have to apply your learnings.
I hope that I arranged the book in a way, enabling you with all the techniques and exercises along
the way, that you can learn form these experiences. In addition, I gave you some more learning paths
to practice by doing in the last chapter.
Finally, let’s get to the one secret about learning that I mentioned. It is the bottom item in the pyramid
that has the biggest return of investment: Teaching Others. Personally, I made the same experience
when I started to blog about my experiences in web development, answered questions on Quora,
Reddit and Stack Overflow, and wrote a book(s). You have to dive deep into a topic in order to teach
it to others. You learn about the little nuances and you dive deep into these topics because you want
to teach them the right way. It’s no shallow learning experience, because you get to know every
detail in order to explain it precisely. You will learn tons of stuff that you didn’t know before. And
most importantly, you will internalize these things by teaching it to others.
After all, you can’t know everything. No one is an expert in everything. I challenge myself too,
by trying to teach others about web development. I get great feedback from people, positive and
negative, that I can apply to grow myself. You can do it as well. You can become better by challenging
yourself, teach something to others and grow.
So here is my quest for you after you read this book. I am sure that you have a friend, coworker
or perhaps someone you know only online, from Stack Overflow or Reddit, who is keen to learn
about state management in modern applications with React, Redux and MobX. Schedule a get-
together with this person and teach him/her state management in modern applications (when using
171
https://slack-the-road-to-learn-react.wieruch.com/
172
https://slack-the-road-to-learn-react.wieruch.com/
Last but not Least 245
React). You can take this book as guidance. After all, teaching others is a win-win situation. Both
participants, mentor and student, will grow from it. So my advice for you: Become a mentor, teach
others and grow.
Last but not Least 246
Acknowledgements
Foremost, I want to acknowledge the work of the people who provide us with the solutions to build
modern applications nowadays. The true heroes are the people behind the tools we use every day
in software development. And these solutions are mostly built in the free time of these people. Just
give yourself one minute and think about all the libraries you use in your own applications.
I want to thank Dan Abramov173 and Andrew Clark174 for open sourcing Redux. A whole community
has gathered behind this library and I must admit that it’s a great community. In addition, I’d like
to thank Michel Weststrate175 for providing an alternative. It keeps the whole ecosystem in balance
if there is more than one solution. These solutions can learn from each other or provide different
approaches to one problem. But it always helps to think out of the box and solve problems from
different perspectives.
Again, I want to thank Dan Abramov for guiding a whole generation of JavaScript developers. You
work on solutions, such as create-react-app, that make a developer’s life easier when getting started
in React. You work closely with the community, listening to their pains and provide solutions for
those problems. Moreover, you encourage people to contribute in the ecosystem. Each day, there
are more contributors for react176 , create-react-app177 and redux178 . You give people a platform to
share their knowledge. For instance, that’s how you encouraged me to write about my experiences
working with React and Redux. Without you sharing the content people are writing, often it would
never reach a broader audience. You drive people to come up with things that other people could use
to solve their problems. The book wouldn’t have happened without Dan sharing my content over
the last year.
I want to thank Mark Erikson179 for his perpetual desire to help others in the world of React and
Redux. He is the keeper of the great lists (react-redux-links180 ) and (redux-ecosystem-links181 ) that
you definitely need to check out. There is not a day passing by that I wouldn’t see a helpful comment
of Mark on Reddit or Twitter about React or Redux. In addition, he is one of the many contributors
in open source who shape the libraries to become a great place for newcomers.
I want to thank Christopher Chedeau182 for his talk at React Europe 2016183 about being successful
in open source. It had a lasting impact on me and was at the time when I published my first blog
post. Thank you for your work with the community.
I want to thank my people at Small Improvements who are great mentors and always a true source
173
https://twitter.com/dan_abramov
174
https://twitter.com/acdlite
175
https://twitter.com/mweststrate
176
https://github.com/facebook/react
177
https://github.com/facebookincubator/create-react-app
178
https://github.com/reactjs/redux
179
https://twitter.com/acemarke
180
https://github.com/markerikson/react-redux-links
181
https://github.com/markerikson/redux-ecosystem-links
182
https://twitter.com/Vjeux
183
https://www.youtube.com/watch?v=nRF0OVQL9Nw
Last but not Least 247
of inspiration. They supported me to make the leap educating others about the things I do in my
daily work. If you ever wanted to work at a company with a great working culture, you should
definitely consider Small Improvements184 . I will always remember a coworker saying: “It doesn’t
feel like work. Every day it is like coming to a place to work with friends on a great project.”
I want to thank Per Fragemann, CEO of Small Improvements, for the chance to work, taking
responsibilities and grow at Small Improvements. He sees opportunities in people, believes in a
sustainable company culture, and gives people all they need to strive. His values taught me a lot.
A special thanks goes to Charisse Ysabel de Torres185 who contributed the awesome cover for the
book in her free time. She was keen to help me with it and in a few brainstorm and several sketching
sessions, she came up with an amazing illustration. I couldn’t have done it without you. Thank you
for being a source of inspiration, a great coworker and friend at Small Improvements.
In the end, I want to thank my girlfriend Liesa186 for always being supportive. Writing a book can
be an enduring battle. There were several weekends, mornings and evenings where I just sat down
to write this book. She prevented me from getting into burnout and managed a lot in my life. Now,
after the book is written, I hope we can have a few more relaxed weekends again. Liesa is doing
most of my marketing efforts. I wouldn’t be able to these things on my own as a software engineer.
So if you are looking for someone to advertise your stuff, reach out to her!
Last but not least, I want to thank the React community. It’s a highly creative and innovative yet
friendly place that makes it possible for everyone to build effortless applications. The community
is supportive and welcoming to everyone. It provides every newcomer with useful resources to get
started. I hope I can contribute to this community as well by writing about these things. One of
my objective is to broaden the diversity among developers by providing free learning material to
minorities. Reach out to me, if you are from an underrepresented group, to get a free copy of this
book.
Thank you.
184
https://www.small-improvements.com/
185
https://dribbble.com/charisseysabel
186
https://www.iamliesa.com/
Last but not Least 248
Thank You
Foremost, I want to thank you for reading the book or taking the full blown course. My biggest hope
is that you had a great learning experience with the material. You should feel empowered now to
build your own application that uses any kind of state management. Reach out to me, if you have
any kind of feedback. I strive to go more in the direction of education which is why I depend on
your feedback.
You can visit my website187 to find more topics about software and web development. If you like to
get any updates, feel free to subscribe to them188 . The updates will only be quality content and never
spam. In addition, I can recommend to read again about the further learning paths from one of the
recent chapters. Otherwise grab a friend of yours or join the Slack Group to teach others about state
management.
In the end, if you liked the learning experience, I hope you will recommend the book to other people.
Share it if you liked it. Just think about people in your life who want to learn more about this topic,
regardless of using React, React Native, Angular or Vue, and share the book with them. I believe,
developers need to align on this topic in order to get to the next level of modern applications.
Thank you for reading the book. Robin
187
https://www.robinwieruch.de/
188
https://www.getrevue.co/profile/rwieruch
Last but not Least 249
Copyright