Here are my notes from getting to know React Redux a lot better.
- This document covers what react-redux is all about and suggests using react-redux-starter-kit.
- Next we'll build the Todo app from the manual with react-redux-starter-kit and redux-cli.
- After that we'll use redux-saga to manage our asynchronous side effects.
- Then we'll make it work with a JSONAPI server for persisting todos to the server.
I've been a web developer for more than a decade. I've seen JavaScript take a massive leap forward several times over the years. First with Prototype.js then jQuery and more recently with Angular and Ember. JavaScript has been maturing slowly for what seems like forever. But these days everyone is excited about the possibilities of ES6. And the premier way to build apps in with ES6 is React-Redux.
Getting into the Redux ecosystem can be daunting as there are many new tools to learn, even if you've been trying to keep up. It's reminiscent of learning SCSS for the first time where suddenly there's a build step where there wasn't one before. The days of writing HTML, CSS and JavaScript without a build tool are long gone. Now in order to create even a simple brochure site you should be using a build tool or you're doing it wrong. You can get a feel for how complex it's starting to get by reading the "State of the Art JavaScript in 2016" article from February. It basically lays out all of the tools a modern web developer should be using and, not surprisingly, those are all the exact same tools you'll see in use in a React-Redux application.
Anyone who got started on Angular back in the early 1.x days can attest that the biggest hurdle was... getting it started. The "best" way to work with Angular was to use build tools but Angular didn't ship with a CLI or a starter kit. So getting started was really hard! Eventually there was a good Yeoman generator for Angular available. This made it dead simple to understand all of the core concepts of Angular and start applying them without having to get lost in the underlying tooling.
In the early Angular days JavaScript build tools were still an insane mess. How messy? Read this epic anti-history of requireJS from the man who accidentally inspired the official requireJS history. In Angular 1.x you were still bending over backwards to overcome the weirdest part of working with JavaScript: everything ran in the global scope! One could argue that the vast majority of Angular 1.x was written to overcome that one messy hurdle. It was barely a step up from the IIFE that were the best practice in the jQuery days.
With ES6 and Babel we're now able to use the new import
syntax for JavaScript modules. It encapsulates all of your code, similar in concept to an IFFE, and gives you tight control over how you include and export code in your project. It makes JavaScript feel like a modern language. It brings the power of Node and NPM to browser development. With Webpack it gets even better. If you've got the right setup you can focus on your code and trust that everything will just work.
Ember takes all of that a huge step further into the future with their impressive ember-cli tool. Used with Ember Data and JSONAPI (especially with a Rails 5 API) Ember is a powerful and productive framework. Ember shares a number of core concepts and tooling with React-Redux, including a focus on components and using Babel to bring ES6 modules to the development life cycle. The biggest problem with Ember is that it makes it enormously difficult to use NPM packages. They have their reasons for this decision, but it's not a restriction that exists in a typical React-Redux App.
If you want to read about how hard it can be to manage your JavaScript code in a modular fashion check out this epic tome on AMD, CommonJS and ES6 Modules.
ES6 has arrived. Now we're supposed to call it ES2015 and we're to expect an ES2016 this year, aka ES7. Babel is helping the JavaScript community avoid the huge debacle that was Python 3 (seriously, skim these: 1, 2, 3 and realize just how well-executed the ES6 roll-out has been). Python 3 was different enough from Python 2 that it was nearly impossible to adopt. The old code wouldn't work on the new engine. Many people in the Python community simply chose not to upgrade. You could find similar trepidation within the Ruby community when they were upgrading from 1.8 to 1.9 and 2.x. That problem eventually lead to the wide adoption of RVM. These days if you're not using RVM to manage multiple versions of Ruby on your machine you're doing it wrong. There's even a Node clone of RVM called NVM that manages multiple versions of Node for those that need it (you probably don't).
Upgrading a language can be excruciatingly painful for a community of developers. We all remember the pains of trying to support IE6, IE7, IE8, IE9... isn't this new version of JS going to break everything?
Node makes things different. Since Node was overtaken by the io.js community, things at Node have really started to change. Upgrading versions of Node is starting to become as easy as upgrading Chrome. In the same way that Chrome has to run code that will run on a ton of different browsers, Node is pretty forgiving of old code. If you didn't know, Node runs the same JavaScript engine as Chrome.
The differences between ES5 and ES6 are huge and most browsers don't support ES6 very well yet (although Chrome and Node are very close now). In browser-land, the complicated matrix of (un)supported features makes the versioning problems of Python and Ruby seem laughably simple. How does it work?
Like SCSS before it, Babel completely side-steps the problem: it converts your shiny new ES6 code to plain-old JavaScript first. The magic is that it uses a transpiler to re-write your code using a standardized polyfill (your build setup usually applies the babel-polyfill for you). Check out the massive polyfill, core-js, that makes this all possible. Under the hood Babel rewrites all of your code to be compatible with core-js. Meanwhile, core-js defines a standard way to translate cutting edge JavaScript to the common version of JavaScript. You can see the concept of polyfills showing up all over the JavaScript community. Most of the new ES6 functionality documented in the MDN manual includes the polyfill in case you want to use that functionality in an environment that doesn't support it natively yet. Check out the Array.prototype.forEach
polyfill in the MDN manual. Babel is basically autoprefixer for your JavaScript code. Babel isn't just smoothing over language compatibility issues for browsers. You can even use babel to write npm modules.
If you haven't seen it yet, check out Bourbon for SCSS. And you should start getting to know ParseCSS because you'll be seeing more of this kind of thing.
If you're not using Babel you're doing it wrong.
Note: Yes, TypeScript exists. Don't use it.
Note: Babel is highly configurable and is smart enough only to "transpile" what's needed for your target system. If you're using Babel for Node-only development you can use something like babel-preset-node6 which only transforms code that Node 6 doesn't yet support.
With the change over to ES2015 and beyond, JavaScript development is making a dramatic shift towards modernization. A React-Redux application takes full advantage of that new power. Often then best way to do something in Redux is to simply write ES6-styled JavaScript in a smart way. Redux makes older OOP frameworks like Angular and Ember seem obsolete because ES6 makes JavaScript much easier to write -- with ES6 you don't need a framework to help you fight the deficiencies in the language. In many cases the conventions-based-approach of React and Redux can completely replace the need for more complicated OOP frameworks. Of course the downside is that Redux development is mostly about conventions, not frameworks. There is no massive framework that holds your hand through the whole process. Instead, Redux gets out of your way and forces you to get to know a new way to think about writing apps.
React are Redux are based on functional programming (FP) techniques (also called reactive programming) while Angular and Ember are based on object-oriented programming (OOP) techniques.
If you didn't realize it already, reactive programming is nothing new, but it is fairly new to JavaScript. The biggest hurdle is rearranging your thinking away from OOP frameworks like Ember and embracing functional programming instead. Bear in mind it's possible that React-Redux is getting it all wrong.
If you're new to Redux and want to get started there's really no better place than the react-redux-starter-kit in conjunction with the redux-cli tool. If you've been digging into Redux and noticing that it's missing an app framework and a CLI, then you just found it! Although it's deceptively named "starter kit", it's the missing app framework and CLI for React-Redux projects. The biggest advantage of the starter kit is that it wires up the router and bootstraps your application better than you would all by yourself. It sets up Webpack and Babel for you and shows you a great way to organize your code. From there it's easy to start putting your app together.
Having the hard work of getting Webpack and Babel working up with all of the best practices is a massive time saver. You're not likely to need to mess with the defaults until much later in your dev process. This allows you to get to the hard work of building your app without getting lost in the tooling.
Because the Redux ecosystem is heavily inspired by the Node ecosystem (a React-Redux app uses NPM to manage packages) there isn't ever going to be a full framework like there is for Ember. Instead, building a React-Redux app is an exercise in assembling the right tools for your application. Sadly, beyond the foundations in the starter-kit you're going to have to research to find each of those little tools and stitch them together yourself. Much of what you need is easy to find on NPM. In practice this level of control actually becomes a blessing as your app matures. However, for starting out it can be daunting.
If you're completely new to the Node ecosystem you might be worried about things like how all of your code fits together. It becomes clearer with practice. You might enjoy reading about ES6 modules in depth. The short version is that you don't have to worry about your code colliding with other code. If you're looking at a file that uses import
then you know exactly what code is going to run in that file. If you look at the export
declarations in a file then you'll know exactly what will be available when you import it. When you need a package, install it with NPM.
One thing to note is that the files in your project are all imported relative to each other. This is extremely powerful because it allows you to organize your code in any way you want. The only restriction is that if you want your code to run you have to import it somewhere. It's hard to really grasp how powerful it is to have no restrictions on how your organize your code. There's no wrong place to put your files.
- Don't bother with Bower unless it's the only way to get something you need (sometimes bower packages include code that npm packages don't). For the most part a Bower package can be installed with NPM.
- Don't bother with alternative package managers.
- Don't worry about code that works in Node vs. The Browser. If you're trying to write universal code that runs in the browser and on the server remember this simple rule: all JavaScript code is universal (with very few exceptions). When in doubt, use functions you can find on MDN.
- The one big exception is, duh, the Node-specific APIs! The Node-specific APIs usually won't work in the browser because they're for working with the file system or the operating system (that's a security issue in browsers). However, the vast majority of what Node offers is powered by V8, the same JavaScript engine used in Chrome. Note: If you're constantly running into issues where your code isn't working in a browser because it's Node-specific... you are probably doing something very wrong.
- If you're trying to evaluate if a package is likely to work for your project a good rule of thumb is "does it work with the file system or not?" That's usually a good filter. Some NPM packages are clearly designed to work on the server and they're usually easy to spot.
What you may not realize is that you must watch the videos before doing anything with React-Redux. You have to watch the videos before you can even read the manual. If you haven't watched the videos you shouldn't be reading this right now!
-
Watch Getting Started with Redux - By the creator of Redux, Dan Abramov. These videos are more essential than you may realize. It's nearly impossible to understand Redux without watching them.
-
Read the Official Redux documentation - Also by the creator of Redux. Written in a similar style to the videos and covers much of the same code. After watching the videos you will want to try to build the example Todo app. You can do this by reading the code in the manual.
-
Read Starting out with react-redux-starter-kit - Blog post about how to use the react-reduc-starter-kit. It covers all of the basics of the project layout. If you skim the write-up on fractal project structures you'll see many of the same concepts. The starter-kit example shows an example of using Redux-thunk to use a remote API. It's worth getting to know that example to better understand the problem that middleware is trying to solve.
-
Quit thinking about universal/isomorphic and native apps for now. - You're getting ahead of yourself. Once you get the hang of Redux the other stuff is easy and already solved for you. Seriously. If you have a fully functioning React-Redux application it can be made "universal" in a matter of minutes if it isn't already. Porting it to React Native will be different than what you're imagining if you've never tried. Regardless, these are things that can wait until later.
-
Read the Airbnb JavaScript Style Guide - This style guide is a nice primer on the day-to-day ES6 features you need to use and how to use them. It's possible to lint your code using the Airbnb rules which -- because of how well it's documented -- can help you casually learn about best practices. However, the starter kit uses the standard rules for the linter and you should too. The biggest difference is that the standard rules don't use semicolons.
-
Build the Todo example - The manual links to a complete Todo app built to match what's documented in the manual. Because the Redux API is so transparent, much of the concepts only make sense once you start to write code. This means that, unlike other frameworks where you can read the API for hours, Redux can be best learned by actually writing some code.
The Redux Todo tutorial currently stops at "use middleware to solve your async needs." There are dozens of middleware solutions that try to solve the problem of asynchronous actions in a variety of different ways. The classic redux-thunk middleware is just the simplistic beginning. Once you start building an app you're going to want to move away from the plain Redux you learned in the Todo tutorial and pick the middleware that fits your working style.
The biggest missing piece from the starter kit is a strong opinion about middleware. Don't get slowed down if you don't understand middleware right now. What you need to know is that you will use someone else's middleware, you likely won't write your own. The Redux community is coalescing around redux-sagas as the middleware of choice. Each project is free to make its own choice in this regard and Redux seems to have left this gap wide open on purpose. You'll likely feel extremely opinionated yourself once you get started on your second app.
When we get there you'll want to choose between redux-saga, redux-effects, redux-side-effects, and redux-loop. Unless of course you're in love with redux-thunk.
We're going to be using redux-saga and you should too. But there are other options for Redux effects middleware if the Sagas middleware isn't right for you.
It's important to get a picture of how a typical starter-kit app is structured because it makes it clear how to apply the core principles of React-Redux. At it's core React-Redux is a marriage of components to a store (React deals with components, Redux deals with the store). Most people get a little lost at this point so we'll try to dive into a quick example starting with a plain React component (see below).
There's a good write-up about "smart" and "dumb" components, by the creator of Redux. In short, a "smart" component knows about the outside world. A "dumb" component only cares about itself.
Note: It's now passé to refer to presentational components as "dumb." It's not cool to pass judgment, they're just components, there's no need to be rude about it ;) You actually have to know about words like "dumb" and "smart" and "duck" because they're still used all over the Redux ecosystem. We need to know the old terms while we wait for the documentation to catch up to trends.
- components (dumb)
- containers (smart)
- modules (duck)
- constants
- actions
- reducers
Note: Everything we're showing here is from the Todo example in the Redux manual... although slightly rewritten to show how experienced developers apply those core concepts. You'll be less confused if you've studied that tutorial.
-
Components should be simple templates - A component uses React
-
Containers connect a component to the store - A container uses React-Redux
-
The Store is immutable by convention - The store is Redux
-
Modules control specific parts of the store - A module is where you'll interact with middleware
A dumb component is essentially a plain template. The code below should look familiar from the components/Link.js
file in the classic React-Redux Todo example. The redux manual calls it a presentational component because all it does is render itself. It doesn't concern itself with were a variable came from.
Read the code below and ask yourself "where do active
and onClick
come from?" If you're able to follow along you'll note that those properties "come from the outside." The Link
component below is "dumb" because it doesn't care how those properties are defined, it just tries to use them to render its template.
import React, { PropTypes } from 'react'
const Link = ({ active, children, onClick }) => {
if (active) {
return <span>{children}</span>
}
return (
<a href="#"
onClick={e => {
e.preventDefault()
onClick()
}}
>
{children}
</a>
)
}
Link.propTypes = {
active: PropTypes.bool.isRequired,
children: PropTypes.node.isRequired,
onClick: PropTypes.func.isRequired
}
export default Link
- The
Link
component is a stateless functional component. You can define these as simple functions that accept aprops
argument. Here we're destructuring the props argument intoactive
,children
, andonClick
. We're using a fat-arrow function to be cool.
const Link = ({ active, children, onClick }) => {
//...
}
- We're using the
active
prop to return aspan
or ana[href]
.
//...
// return a span if active is truthy
if (active) {
return <span>{children}</span>
}
// otherwise return an a[href]
return <a href="#">{children}</a>
//...
- The
onClick
attribute of thea[href]
calls theonclick()
property on theLink
. Theonclick()
prop must be passed in from the outside.
// the onClick prop comes from the outside
const Link = ({ active, children, onClick }) => {
//...
// React allows JSX elements to declare onCLick attributes
// @see https://facebook.github.io/react/docs/events.html#supported-events
return (
<a href="#"
onClick={e => {
e.preventDefault()
onClick() // <-- we call onClick prop here
}}
>
{children}
</a>
)
}
- If you're defining a component that only needs to return a template you can use parenthesis to wrap your JSX template. This allows for multiline templates to be used with a one-line fat-arrow function with an implied return.
const SimpleLink = ({ onClick, children }) => (
<a href="#"
onClick={e => {
e.preventDefault()
onClick()
}}
>
{children}
</a>
)
Note: Starting in React 0.14 it's common practice to define components as stateless functional components using a fat-arrow function.
Note: You may want to read up on how to use destructuring to simulate named parameters.
Below is the simplest example of a component. Notice how the component itself is just one line.
import React, { PropTypes } from 'react'
// create a stateless functional component
// - use a fat-arrow function
// - use descructing to pull name out of props
// - return a JSX template
const DumbComponent = ({ name }) => <div>How dumb am {name}?</div>
// specify the properties you need
// (you can leave off `.isRequired` if it's not required)
DumbComponent.propTypes = {
name: PropTypes.string.isRequired
}
// export the component so someone can put it in a page somewhere
export default DumbComponent
Note: PropTypes is explained in the massive code block at the top of the React manual page for reusable components.
Note: You should only need to use the fancier ES6 class syntax (introduced in React 0.13) in special cases. There's a great write-up on the Babel blog on the ES6 way to write React components but it doesn't cover the newer stateless functional component syntax.
import React, { Component, PropTypes } from 'react'
// create a class component
// - use an es6 class
class DumbComponent extends Component {
// commonly you are using a class
// to fire an action when the component mounts
// or hook up DOM events that are hard to express inline
// @see https://gist.github.com/koistya/934a4e452b61017ad611
componentDidMount() {
this.props.fetchInitialState()
}
// you need to define a render function
render() {
// props are available as a property of this
const { name } = this.props
return <div>How dumb am {name}?</div>
}
}
// specify the properties you need
// it's still prettier to define them down here
// @see https://babeljs.io/blog/2015/06/07/react-on-es6-plus
DumbComponent.propTypes = {
name: PropTypes.string.isRequired,
fetchInitialState: PropTypes.func.isRequired
}
// export the component so someone can put it in a page somewhere
export default DumbComponent
Note: If you're trying to read about React, ignore how they use the old React.createClass()
syntax in the manual. Never use something like React.createClass()
unless, inexplicably, you're not allowed to use ES6. But really... you should be using ES6 no matter what by this point. There is no reason not to be on the ES6 train.
If dumb components don't know where the data comes from, who does?! The answer is "smart" components. The Redux manual calls them container components. A container component is the bridge between React and Redux. The library that connects them is called React-Redux. Because a container component is gluing together two different things it needs to map the concepts of one to the other. A container component maps the concepts for interacting with a React component with the concepts for interacting with the Redux store.
A container hooks a component into Redux. React, even without Redux, has several advanced methods that allow components to deeply manage their own state. If you've read a React tutorial you've probably read about the component state. Forget about that! Redux stores the state for you. Redux enforces a strict data flow in order to know precisely when components need to update. The container can dispatch actions and read from the store. The container handles events from the component and passes data to it. The interface for this is deceptively simple and best explained with some code. The code below should look familiar from the containers/FilterLink.js
in the classic React-Redux Todo example.
import { connect } from 'react-redux'
import { setVisibilityFilter } from '../redux/modules/todos'
import Link from '../components/Link'
const mapStateToProps = (state, ownProps) => {
return {
active: ownProps.filter === state.visibilityFilter
}
}
const mapDispatchToProps = (dispatch, ownProps) => {
return {
onClick: () => {
dispatch(setVisibilityFilter(ownProps.filter))
}
}
}
const FilterLink = connect(
mapStateToProps,
mapDispatchToProps
)(Link)
export default FilterLink
It's important to go over the steps that are involved here.
FilterLink
is a container component. It uses theconnect()
function to map values from the Reduxstate
anddispatch()
to properties on theLink
component. There's a reason the two arguments are "map state" and "map dispatch".
// connect() is from react-redux
// @see https://github.com/reactjs/react-redux
const FilterLink = connect(
mapStateToProps,
mapDispatchToProps
)(Link)
-
mapStateToProps
andmapDispatchToProps
are callback functions that are called from withinconnect()
. Under the hood they are using thesubscribe()
method from Redux andcomponentDidMount()
andsetProps()
in React. You may want to check out what the connect function does. Whatever the magic that goes on under the hood it's important to know thatconnect()
ensures that when the state changes the component will be rendered with the updated props from the store. -
The
FilterLink
container is wrapping theLink
component. A container's entire purpose is to wrap a component in order to connect it to the Redux store. To do that the container will map values from the state to properties on the component. It also maps dispatched actions to functions on the component.
const FilterLink = connect(
mapStateToProps,
mapDispatchToProps
)(Link) // <-- Look! The Link component is being passed to the FilterLink container
mapStateToProps
is where you can pull values out of the state and define them as properties on the connected component. If you look at the example, the wrapped component will be receiving anactive
property that returns true if the container'sfilter
prop is equal to thevisibilityFilter
from the state. The example is designed to show that the container can have props, likeownProps.filter
that the wrapped component never knows about.
// state comes from Redux
// ownProps comes from the container
// <FilterLink hello={true} /> becomes ownProps.hello
const mapStateToProps = (state, ownProps) => {
return {
// we're comparing a prop with a value from the state
// you can read any value from the state that you'd like
// it's common to use selectors when you read from the state
// @see https://github.com/reactjs/reselect
active: ownProps.filter === state.visibilityFilter
}
}
mapDispatchToProps
is where you can create functions that dispatch actions to the store. In Redux there is a strict separation of concerns. In another framework you'd be tempted to have your template actions immediately do something. But in Redux we dispatch every action using a strict flow. This makes it easier to reuse functionality and to creatively mix and match functionality once you get the hang of things.
// dispatch comes from Redux
// ownProps comes from the container
// <FilterLink hello={true} /> becomes ownProps.hello
const mapDispatchToProps = (dispatch, ownProps) => {
return {
// here we define the onClick prop for the component
onClick: () => {
// when we click in the component we dispatch an action to Redux
// here we're using the setVisibilityFilter action creator
// same as dispatch({ type: 'SET_VISIBILITY_FILTER', payload: ownProps.filter })
dispatch( setVisibilityFilter(ownProps.filter) )
}
}
}
A container maps values from the Redux store to properties on a React component. It reads from the store when a component is first loaded and after the store has been updated. In order to closely monitor when the store has been updated Redux only allows you to dispatch actions. From a container it is not really possible to know what an action does because Redux separates that functionality into reducers.
If the concept of reducers seems totally foreign right now then you completely understand why smart components only dispatch actions and read from the store. Writing to the store is a task for reducers.
Containers are all about reading from the store and dispatching actions to Redux.
If you've been reading up on React and Redux you've probably come across libraries such as Immutable.js. Forget about it. Redux completely replaces the need for a library like Immutable.js. You can still use it with Redux (and many people do) but Immutable simply enforces what Redux defines by convention. Hint: If you're worried about your state being mutated in unexpected ways you should be writing better tests.
There are a great number of fascinating advantages that surface from the simple decision of Redux to store all of the application state in a single immutable object. Redux is a methodology and minimal toolkit for working with the store. This is the next part of React-Redux that can seem very daunting. Redux's simplicity becomes it's great power. Utilizing barebones JavaScript and good conventions is the spirit of Redux. There is no need for something like Immutable.js when you're doing Redux correctly.
Everything uses the same store. If you put something in the store, a container can read from it. There are no restrictions of which part of the store you can write to or read from. At it's base level it might feel a little disorganized or even insecure. But the store is what you make of it. We'll see that people commonly organize things into modules and use some basic Redux functionality that makes things a little less scary.
We'll see later that there are some conventions emerging about how to maintain data that you recieved from an API. Those patterns can be stamped into other parts of your app because of how well Redux encapsulates functionality.
In the Redux world it's now popular to group constants, action creators and reducers into a single file called a "module" or a "duck." It's actually fine to do it the old way and break them into separate constants/
, actions/
and reducers/
folders if you want. Either style make sense for different use cases. For most people starting out it's easier to put all of those things together in a modules/
folder. If you run into a case where modules need to re-use functionality from each other then you've just discovered why the Redux manual suggests keeping them separate.
This starts make more sense with some code. The following is a combination of the actions.js
and the reducers.js
files in the classic React-Redux Todo example. The action creator and reducer have been rewritten using redux-actions for simplicity. For brevity we're only looking at the visibilityFilter
example from the link component we're exploring. A lot changed here but it will make sense in a second.
import { combineReducers } from 'redux'
import { createAction, handleActions } from 'redux-actions'
// ...for brevity
import todos from './todos'
// Action Types (Constants)
export const SET_VISIBILITY_FILTER = 'SET_VISIBILITY_FILTER'
// Action Creators
export const setVisibilityFilter = createAction(SET_VISIBILITY_FILTER)
// Reducers
export const visibilityFilter = handleActions({
[SET_VISIBILITY_FILTER]: (state, { payload }) => payload
}, 'SHOW_ALL')
export default combineReducers({
todos,
visibilityFilter
});
Let's dig into what's going on here:
- We have an action creator named
setVisibilityFilter()
. In theFilterLink
container we use our action creator function to create an action object that we dispatch when someone clicks theLink
. An action creator is a simple function that returns an object with atype
and apayload
.
// create an action creator
// @see https://github.com/acdlite/redux-actions#createactiontype-payloadcreator--identity-metacreator
export const setVisibilityFilter = createAction(SET_VISIBILITY_FILTER)
// action creators are super simple
console.log( setVisibilityFilter )
/* -->
function(payload) {
return { type: SET_VISIBILITY_FILTER, payload }
}
*/
- Action creators are easy to write by hand but using
createAction()
makes it even easier. By default the function creates an action creator that accepts a payload. The vast majority of the time this is all an action needs to do -- marry a payload to an action type. You can see that theFilterLink
container sends theownProps.filter
payload to thesetVisibilityFilter(payload)
action creator. Compare this to the long-hand version in the Todos example.
// some people call an action creator an action
// but actions are what action creators return
// an action is just an object with a type and a payload
const action = setVisibilityFilter('some string')
console.log(action)
// --> { type: 'SET_VISIBILITY_FILTER', payload: 'some string' }
- The action actually gets dispatched from the container. Redux uses this
dispatch(action)
construct to allow any container to call an action while still maintaining control over the when the store gets updated. If you're used to MVC you could think of a component as the view, the container as the controller and the actionCreators as the model. Well... sort of. Action creators cover a portion of that you normally get with a model. We'll be rounding this out more as we go along.
// from the FilterLink container
const mapDispatchToProps = (dispatch, ownProps) => {
return {
onClick: () => {
// we have to dispatch our actions manually
dispatch(setVisibilityFilter(ownProps.filter))
}
}
}
- After an action is dispatched it is handled by a reducer. In the example you can see that the
handleActions(map, initialState)
function creates a reducer that can handle an action namedSET_VISIBILITY_FILTER
. Probably the single ugliest thing about Redux is the reliance on all-caps constants. It's absolutely awful to have these long descriptive strings screaming at you. If you can squint and look past them, reducers start to get really simple.
export const visibilityFilter = handleActions({
[SET_VISIBILITY_FILTER]: (state, { payload }) => payload
}, 'SHOW_ALL')
// a reducer is really simple
console.log(visibilityFilter);
/* -->
function(state = 'SHOW_ALL', action) {
switch(action.type) {
case SET_VISIBILITY_FILTER: return action.payload
default: return state
}
}
*/
- Our reducer for
SET_VISIBILITY_FILTER
accepts thestate
and theaction
and returns the new state. In this case the state forvisibilityFilter
is just a simple string. We're returning whatever was passed in as the payload as the new state. What's really subtle is thatcombineReducers()
is what's pulling out the part of the state that thevisibilityFilter(state, action)
reducer cares about. This makes it possible for our reducer to simply return theaction.payload
as the new state instead of trying to store it in thestate.visibilityFilter
property. That's slightly confusing because where you store something in the state determines how read something. Not explicitly stating how our state is stored is actually a hidden power of Redux. The confusing part simply goes away once you've worked with it for a while.
// a reducer can be incredibly simple
// here it's just a fat arrow function
// we're blindly returning the payload as the new state
(state, { payload }) => payload
combineReducers()
is a key part of how the Redux store can allow every component to share a state without causing massive collisions. A reducer is in charge of managing the state that's passed to it.combineReducers()
calls each reducer it is passed with only the part of the state matching its key. In this case the reducer created bycombineReducers()
will passstate.visibilityFilter
to thevisibilityFilter(state, action)
function. That clever tick ensures that the reducer won't accidentally overwrite the wrong part of the state. This is also the beginning of some of the magic reusability that Redux enables. Once you begin to master reducers you will be able to mix and match them all you want. The Redux authors refer to this as composability (it's what thecompose()
function is for).
// how we wrote it in the example above
export default combineReducers({
todos,
visibilityFilter
});
// a toy example replacing combineReducers
// check out the real deal: https://github.com/reactjs/redux/blob/master/src/combineReducers.js
export default function(state, action) {
// we have an object whose keys match the name of a reducer function
const reducers = {
todos,
visibilityFilter
}
// we loop through the keys
// calling a function for each key
Object.keys(reducers).map(key => {
// capture the reducer and subState by key
const reducer = reducers[key]
const subState = state[key]
// return a new state
// replace the key with the new sub state from the reducer
return {
...state,
[key]: reducer(subState, action)
}
})
}
Note: If you're noticing that the use of ALL_CAPS
constants is annoying you are not alone. For now we'll stick with them but we'll quickly see that you don't use them for very much once you start building things.
- Next we'll build the Todo app from the manual with react-redux-starter-kit and redux-cli.
- We'll use Redux-saga to manage our asynchronous side effects.
- We'll make it work with a JSONAPI server for persisting todos to the database.