For more information on this course, click here
Build a functioning "Would you Rather" application using React.js, Redux, Firebase, and Immutable.js. The end result of this project can be found HERE.
The goal here is to give you just enough guidance for you to struggle without drowning. Note that the steps below are just suggestions. The ideal situation is you look at the completed project, then you build it. However, if you're not up for such things, feel free to follow the (vague by design) steps below. If you get stuck, all steps have coinciding branches for you to reference as a last case scenario.
- Head over HERE and play around with the final project. Think about how you would separate your different components and functionality.
Before I ever start a React app, no matter how complex, I always create a HelloWorld component just to make sure that I've tied everything together properly. I don't expect you to have all this memorized, but do your best to wire up everything by yourself. If you do get stuck you can refer to the 'step1' branch.
- Create a new project or fork this repository
- npm install the dependencies you'll need (for a basic HelloWorld app). Include ES6+.
npm install --save react react-dom
npm install --save-dev babel-core babel-loader babel-preset-es2015 babel-preset-react babel-preset-stage-0 css-loader html-webpack-plugin style-loader webpack webpack-dev-server
- Create and configure your .babelrc file
- Create and configure your webpack.config.js file
- Create an app directory and in your app directory create and configure your index.html file
- In your app directory create and configure your index.js file to render a HelloWorld component
- Start webpack and make sure everything is working
- Add a
production
command to your scripts property in package.json which runswebpack -p
- Add a
start
command to your scripts property in package.json which runswebpack-dev-server
- Run
npm run start
from your terminal then checklocalhost:8080
to make sure everything is rendering correctly
If you followed my example above, our webpack configurations are super minimal. Let's go ahead and beef them up a bit by adding CSS Module source maps, path file resolving, source maps, Hot Module Replacement, and a production build.
- We're going to need babel-preset-react-hmre, so npm install that as a dev dependency.
- In order for us to use ES+ features in our webpack.config.js file we need to change the name to webpack.config.babel.js. Do that now.
- As we did in the videos, we're going to create a different configuration object for both production and development then merge them together based on if we're in production mode or not. First, utilize
process.env.npm_lifecycle_event
to create a variable to know if we're in 'production' mode or not. Remember,process.env.npm_lifecycle_event
will tell us what command was ran in order to start our server. If we runnpm run production
, thenprocess.env.npm_lifecycle_event
will be production. - Let your .babelrc file know if you're in production by utilizing process.env.BABEL_ENV.
- If the previous step is done correctly you can now set up different presets in your .babelrc file based on which mode we're running in. Head over to your .babelrc file and make it so when we're in development mode (or we ran
npm run start
to start our server), then we set the "react-hmre" preset. Don't worry if you don't have all of this memorized, you shouldn't, if you do copy/paste a lot of this configuration code, just make sure you know what's going on. - Back in your webpack.config.babel.js file create a productionPlugin variable which uses Webpack's DefinePlugin property to set
process.env.NODE_ENV
to production. This is critical to have fast performing code in production. - Create three variables,
base
,developmentConfig
, andproductionConfig
. Base has all of the config properties that are shared with withdevelopmentConfig
andproductionConfig
, and the others have that config settings based on the specific build we're running (prod vs development). - Now, we need to export an object that combines our base variable with either
developmentConfig
orproductionConfig
. Use Object.assign to do this. - Run both
npm run start
andnpm run production
to make sure everything is still working correctly (your app should still just render 'Hello World!'. Remember,npm run start
just starts a local server which your files will be served from.npm run production
should create a/dist
folder.
Now that our webpack config is set up properly and we have our HelloWorld app, let's go ahead and tie in some basic routing with React Router.
- We're going to use React Router to handle our routing, run
npm install --save react-router
in your project. - Inside of your
app
folder, create acontainers
folder. - Inside that folder create an
index.js
file, aMain
folder, and aHome
folder. - Inside your
Main
folder create aMainContainer.js
file and inside yourHome
folder create aHomeContainer.js
file. - Now, we're going to use our
index.js
file we made to make our imports easier just like we did in the video. Head over to yourindex.js
file and add the proper exports (or MainContainer.js and HomeContainer.js) - Now inside of both
HomeContainer.js
andMainContainer.js
create a React component which just renders a string or 'Home' or 'Main'. - Now let's set up some basic routing. Inside your
app
folder create aconfig
folder and inside of that create aroutes.js
file. - Inside that routes file create declarative Routes which will render
MainContainer
as the main parent route (/) andHomeContainer
as theIndexRoute
. - Now our initial routes are set up, we need to render those when we call ReactDOM.render. Header over to your
app/index.js
file and instead of renderingMain
, render your Routes. - If you refresh your view you should now see 'Main'. Great! But we should also be seeing 'Home' as well. The reason we're not is because we're not rendering any children components inside of Main. Remember, MainContainer is our main parent route. When we transition to different URLs, different children routes are going to becoming active. We need to render those children routes. Head over to
MainContainer.js
and instead of just rendering 'Main', we want to render 'Main' as well as any children routes that are passed to it (this.props.children
). Make those changes now. - Reload your app and now you should see both Main and Home since our HomeContainer component is our IndexRoute (which becomes active when no other routes 'path' match the URL, in this case we have no other routes, so HomeContainer is always active).
##Step 4: Main and Home Styling If you checkout the final solution, you'll notice the basic home page. Our routing is set up so now let's just style the HomeContainer and MainContainer components
- Create a
components
folder in yourapp
folder - Create a
Home
folder in your components folder and create aHome.js
file inside thatHome
folder - Format and Style your Home component
- Add styling/format to your Home component
- Create an
index.js
file inside your components folder and export Home.js for easier imports - Now in your
HomeContainer
you need to render yourHome
component rather than the text 'Home' - Right now your App should look like this
Our navigation bar is going to be fundamental to our application. Even though all the pieces that it entails won't be set up, in this section you'll go ahead and set up the navigation assuming you'll tie in the rest of the pieces later.
- In your components folder create a Navigation component which takes in an
isAuthed
property. - Looking at the finished example you'll notice that the navbar changes based on if you're logged in or not. On the left side if you're logged in you'll see a 'Home' link and if you're not you'll see Home and Authenticate. On the right if you're logged in you'll see a button to create a new question as well as a logout button.
- Using React Router's
Link
component, create your Navigation component using the following paths as your Links,/
for home,/logout
, and/auth
. Even though these routes aren't created yet, we'll do that later. - Instead of having a Button that pops open a Modal, for now just write 'Modal'
- Style your navigation component appropriately.
- Inside of MainContainer import your newly created Navigation component and render it initially passing in
false
forisAuthed
(test if it works) then pass intrue
and test that your UI is changing based on theisAuthed
prop.
Now that we're starting to get into authentication, we're going to start to have state in our application. Before we do that, it's a good idea to have a general idea of what the shape of your application's state is going to look like. This may not be practical in larger applications, but we're going to map out the shape of Firebase, Redux, our Reducers and Actions creators before we continue to work on any future part of the app. If done correctly, this activity will be hugely beneficial when your start to actually build your app.
- Head back over to the finished app here and really play around with it. Think of all of the different pieces of state that are living and changing in the app.
- Create a reduxSchema.js file in your root directory
- Fill out this file to be a representation of what your full state tree will be once everything is tied up. You're not going to actually use this file, but you will refer to it when you're building your reducers and actions creators.
- Remember if you're struggling, refer to the
step6
branch, just don't use it as a crutch.
Now we're going to do exactly what we did in Step 6, but instead we're going to do it for Firebase. Reflect on the difference we talked about between a Firebase schema and a Redux Schema.
- Create a firebaseSchema.js file and add a representation of what your Firebase schema will look like. Again this will just be for reference only. Don't be afraid to screw it up the first time. It usually takes a few attempts to get it right.
Now that we've designed both our Firebase and our Redux state, let's go ahead and hook up Redux as well as create our first users module with Redux to keep track of all of the user's state as well as the currently authed user.
- Create the following path under your
app
folder,redux/modules
- Inside of
modules
create ausers.js
file. - Create a users reducer which for now just returns the initial state under the default switch case.
- Inside of
modules
create an index.js file. This file will export all of our reducers. For now, just export the users reducer which you just created. - Now head over to your
app/index.js
in order to hook up Redux. - Instead of waiting until later to hook up Redux dev tools as well as React Router Redux, we're going to do it all now.
npm install --save react-router-redux redux-thunk redux react-redux
- First thing we need to do is import the neccessary properties. This can get a bit hairy so I'll offer some more help through the next few steps.
-
import { createStore, applyMiddleware, compose, combineReducers } from 'redux' import { Provider } from 'react-redux' import thunk from 'redux-thunk' import { routerReducer, syncHistoryWithStore } from 'react-router-redux' import * as reducers from 'redux/modules'
- Now that we have everything we need, let's go ahead and create our store.
const store = createStore(combineReducers({...reducers, routing: routerReducer}), compose(
applyMiddleware(thunk),
window.devToolsExtension ? window.devToolsExtension() : (f) => f
))
- Just a refesher of what's going on above, we use combineReducers in order to be able to merge React Router Redux's
routerReducer
with our own reducers. We also use applyMiddleware to make it possible for us to use Redux Thunks in our application. - Next, we need to pass
syncHistoryWithStore
but React Router's hashHistory and our store we created in order to create a new history for us. Don't forget to import hashHistory. - Now, instead of just rendering our routes, we want to render Redux's Provider passint it our store we created.
- Nested inside of the Provider is going to be our Routes, however, we need to pass our new history that we created a few steps ago to our Routes to use rather than hashHistory. Head over to
routes.js
and instead of returning routes, return a getRoutes function which takes in an empty (for now)checkAuth
function as well as ourhistory
variable. - Now back in the
index.js
file invokegetRoutes
as a child ofProvider
passing it a blank function which just returns true and thehistory
variable we created earlier. - Head over HERE and download the Redux Dev Tools. Once that's downloaded load up the app and open up the 'Redux' devtools tab. Click on "State" and you should be able to see our
users
state as well asrouting
from React Router Redux. - Phew! That was a LOT. But now we're in a great position to create a very nice React/Redux application. Again, memorizing all of the steps isn't as important as understanding them all. If you cheat and look at my code, cool. But if you cheat and look at my code while blindly copy/pasting, not cool. If you have a specific question, I'm always available in the Slack channel.
Now that Redux is set up properly, we want to create some action creators that we can eventually invoke once our user authenticates. When the user is browsing the app we're going to rely on Redux to keep track of it that user is authenticated or not.
- Inside of your users.js Redux module, create an
authUser
action creator which sets anisAuthed
property on your state to true and saves the authed usersuid
to anauthedId
property on the state. - Create an action creator called
unauthUser
which changesisAuthed
on the users state to false and resetsauthedId
back to an empty string. - Now that we have this
isAuthed
variable living in our state, head over to MainContainer.js and connect your MainContainer component so instead of passing intrue
to the component, we pass in the true value (which by default should be false.) - Right now your App should look like this
In order to make our app able to have Facebook authentication, there are a few steps we need do take.
- If you haven't already head over to firebase.google.com and make an account
- Once you do that go ahead and make a new project and call it whatever you would like.
- Once your project is created head to your projects dashboard and click on "Auth" side panel button.
- From the "Authentication" page click on "SIGN-IN-METHOD" and then click on Facebook.
- Notice we'll need an APP ID as well as an App Secret. We'll get these from Facebook. In the mean time, copy the url that is at the bottom in the gray box it should look something like this "https://YOUR-PROJECT-XXXX.firebaseapp.com/__/auth/handler".
- Now head over to Facebook's Developer's Site and either sign up or hover over the "My Apps" section in the top right and select "Add a New App" then select Website.
- Enter in a name for your app and click "Create a new Facebook App"
- Enter your contact email and choose a category.
- (Pass the spam filter if needed)
- Now when your app is approved in the top right hand corner click on "Skip Quick Start"
- Now on the left hand sidebar click on "Add Product"
- Then click "Get Started" under the Facebook Auth section.
- Now if you still have that URL we copied from Firebase go ahead and paste that in the section titled "Valid OAuth redirect URIs" then click "Save Changes" in the lower right.
- Now head back to your "Dashboard" and then copy your App ID and App Secret and paste them in the appropriate sections under the URL we went to in the Firebase dashboard earlier.
- If you followed everything correctly your app should now be able to use Facebook authentication.
- Inside your
config
folder create a constants.js file. - Here is where we're going to initialize firebase. But before we do that, go ahead and
npm install --save firebase
- Now back in your Firebase console select "Add Firebase to your web app" and copy both the config object as well as the firebase.initializeApp invocation.
- Inside of your constants.js file import firebase and initialize your app.
- export two variables from this file.
export const ref = firebase.database().ref()
andconst firebaseAuth = firebase.auth
- We'll be importing those variables later to interact with our Firebase database as well as our Firebase Auth module.
Now that our Firebase is initialized, let's go ahead and create some helper methods to assist us with authentication.
- Inside of
app
create ahelpers
folder then inside of that create aauth.js
file. - Create and export as default a function called
auth
which usesfirebaseAuth().signInWithPopup
to auth with Firebase. - Create and export a function called
logout
which unauthenticates from Firebase. - Create an export a function called saveUser which takes in a user and saves that user to
users/AUTHED-USERS-ID/info
then returns the user. - From this point on, all of the weird Webpack/Firebase/etc setup issues are over. So by design these instructions will get less and less detailed. Remember if you get stuck, try to find the solution on your own. If you can't, check out the branch which corresponds with the step. If you're still stuck, ask for help in the Slack channel.
Now is our moment we've been prepping for. We're going to authenticate with Firebase and save the user (as well as their authed state), into Redux.
- Even though we have our
authUser
action creator, that isn't doing anything more than just returning an object. Instead, create an action creator calledfetchAndHandleAuthedUser
which is in charge of authenticating the user, saving the users data into Firebase, then authenticating the user (while saving it) to Redux. To do this step you'll leverage the functions we created last step in auth.js as well as a few new action creators you'll make. - Once you've finished this action creator update your users Reducer so it correctly modifies the state based on the action creators that were invoked.
- Now you should have all of your authentication logic encapsulated inside of
fetchAndHandleAuthedUser
, but now we need to actually create the UI which will eventually run that function. - Create a FacebookAuthButton stateless functional component which takes in an
isFetchin
prop as well as anonAuth
prop. - Now create an
Authenticate
stateless functional component that is going to be the UI for the/auth
route. It may be a good idea to use the FacebookAuthButton we created in the last step. - Once you've finished that component, you should have a good understanding of what Authenticate needs in order to properly function. So the only step now is to create an AuthenticateContainer which is hooked up to Redux and passes Authenticate everything that it needs. Make sure once you authenticate you redirect the user to the '/results' route, even though it doesn't exist yet.
- Now the last step is we need to add the AuthenticateContainer we just created to our routes.js file.
- Go ahead and test your code out now. Fix any errors you have and make sure you can route to the /auth route when you click the "Authenticate" button on the home screen.
- If you've been following my code you should see that the FacebookAuth button show's "Loading". Why is this? If you look at user.js the initial state that was set has
isFetching
as true.isFetching
is what the FacebookAuthButton uses to know if it should show 'Loading' or 'Login with Facebook'. - We can fix this with a check in our MainContainer. First, inside of users.js create an action creator which changes
isFetching
to false and update the user's Reducer as well. - Now, import the action creator you just made into MainContainer, don't forget to use
bindActionCreators
or just dispatch the action creator itself. - Now temporarily, call your function which changes
isFetching
to false inside of MainContainer'scomponentDidMount
method. - Now when MainContainer mounts,
isFetching
should be changed to true which means that the FacebookAuthButton will display the correct text. To test this, head over to your/auth
route. If you're uncomfortable with this last step check out the code. If you've understood everything until this point you're doing extremely well. Once you get comfortable with action creators and how they're imported into container components, managing your app state becomes a breeze. Our app is going to get more functionality, but there aren't really a whole lot of "new" things from here on out.
Your app should be in the state where authentication works fine but once you're done authenticating you get an error that says "Warning: [react-router] Location "/results" did not match any routes". This is obviously because we haven't set up that route yet. Let's do that now.
- Create a stateless functional component called Results which just (for now) renders the text "Results"
- Create a container component which renders the Results component we created in the previous step.
- Set up a route so when you go to
/results
you get the container component created in the previous step. - Your app should be at the point where you can authenticate, and when you do, you get redirected to the
/results
view and the navigation bar changes to be what it should look like when you're authenticated.
Currently there are two issues with our authentication set up. The first one is that if you hit refresh, our Redux state gets reset which means isAuthed
changes to false. The second is related in that we shouldn't be able to access the /results
route unless we're authenticated. Let's tackle problem #1 first.
- Firebase has a
.onAuthStateChanged
method which we can use in order to listen to any authentication changes (or initializations) our app goes through. So what we can do is when MainContainer mounts, set up this listener so that whenever we auth, we make sure Redux knows about it. At this point we've already set up the action creator we need (authUser
) so it's just a matter of hooking it up. - Head over to
MainContainer.js
and when it mounts, invokefirebaseAuth().onAuthStateChanged
and pass it a callback. This callback will receive one parameter which is either null or the authed user. If it's the authed user, format it and save it to Redux. If not, then we want to removeFetchingUser (which should already be there). - Great so now we're checking and setting up an authentication listener when our app boots up, but you'll notice there's a weird lag in the UI as Firebase sets up that listener. There are a lot of ways to fix this but a simple one for now is to make it so if
isFetching
in Redux'susers
module is true, then just haveMainContainer
render null. If not, then render the UI. This works because we know that when we callfetchingUserSuccess
after ouronAuthStateChanged
callback runs, thenisFetching
will get switched the false and the correct UI will render. - Now when you hit refresh, if you've already authenticated with Facebook before your app should save the authed user in Redux and then redirect you to the
/results
page.
So far so good. But now we want to set up some authentication checks with React Router. Many of our Routes we don't want users to access unless they're verified (and vice versa). To do this we're going to leverage React Router's onEnter
property. The good news is we've done the hardest part when we set up our checkAuth
function inside of our index.js
page. No we just need to fill it out with whatever route logic checks we need then add it to the routes we want to check. The idea is that whatever route we attatch this function to will get called whenever a user enters this route.
- By leveraging
store.getState()
go ahead and inside ofcheckAuth
grabisAuthed
andisFetching
off theusers
state in Redux. - First thing we want to do is if
isFetching
is still true, just return. The reason we need this is because we don't want to do any route redirects if Firebase is still figuring out if the user is authed or not. - Next, grab the next path name by using
nextState.location.pathname
wherenextState
is the first argument that will be passed tocheckAuth
. - First check if the next path name is
/
(our index route) or if it's/auth
, if it is andisAuthed
is true, go ahead and callreplace
(second argument to checkAuth) passing is/results
. Basically what this is saying is that if the user is already authenticated, we don't want them to be able to go to the home page or the page to authenticate but instead take them to/results
. - Next, if the path name isn't
/
or/auth
and the user is not authenticated, then take them to/auth
. This protects any routes that aren't/
and/auth
and will redirect the user to/auth
instead. - Now head over to
routes.js
wheregetRoutes
is defined and addcheckAuth
as a prop on all of the routes you want to runcheckAuth
logic on. I have it on/auth
,/results
, and the IndexRoute. - Now if everything is working you should only be able to access specific routes based on your auth state.
Next step is to make it so you can post a new "wouldYouRather" question. To do this, let's first start with our modal Redux module. As always, it's a good idea to start with Redux when you're implementing a feature as the most important aspect of your application is usually how your app's state will change with the new feature.
- Create a
modal.js
file inredux/modules
. If you're using anindex.js
file for easier imports make sure you addmodal.js
to that file. - Create and implement three action creators,
openModal
,closeModal
, andupdateDecisionText
with their accompanying constants and switch statements in the reducer. - Create, impelment, and export a Redux Thunk action creator called
saveAndCloseModal
. - If you haven't already finish
modal.js
by adding some initial state to your reducer and anything else you think you'll need for this module. - If everything worked you should now see your initial
modal
state in Redux dev tools.
Now that our Modal Redux module is set up, we just need to implement the UI so we can actually submit a new decision. The only tip for this one is that I use react-modal
from NPM. We've done all of these steps a few times now so start to venture out more on your own. Check the branch if you get stuck.
At this point you should be able to submit decisions to Firebase and have them persisted. The problem now is we want to go ahead and fetch those decisions, set a listener so we'll be aware if they change, then show those decisions to the view in list on the '/results' view. In this section we'll build our the decisions modules which will handle all of our decisions state. We'll also include a Thunk function which will make the request to Firebase and set up the listener.
Again, I'll tell you the action creators (and Thunks) that I created, and you can choose to follow my lead or create your own path. The biggest thing to remember is that at this point, really understand the shape of your decisions state and understand how that state can be modified.If you understand those two things, creating the action creators which modify the state will be easier.
Implement the following action creators (and Thunks) with their accompanying constants and items in the decisions reducer.
- settingDecisionsListener
- settingDecisionsListenerError
- settingDecisionsListenerSuccess
- setAndHandleDecisionsListener (Thunk for ^ three action creators)
- addDecision
- fetchAndHandleSingleDecision (Thunk for addDecision)
One thing you may have noticed is that it will be a good idea to know if you're already listening to certain endpoints or not. You can keep this state anywhere you like, but to make future changes easier, I like to keep listeners in their own Redux module. To follow my lead, create a listeners
modules which has an addListener
function, then when you add the decisions
listener in decisions.js
, also add that to listeners
so we know what we're listening to.
Now your state tree should look like this ↓. Notice we haven't actually fetched any decisions yet (or set any listeners). That's the next step.
Now that we have the ability to fetch decisions (and update our Redux state), now let's actually do that once the user lands on /results
and we haven't fetched those already.
- In your Results component update the component to take in in the following props, and neatly shows the Results to the view.
isFetching
,decisions
,error
. If you're following along with my code you'll notice that I installedreact-loader
to show a nice spinner when the data is loading. Feel free to do that as well. - Now that the component is built you should be thinking that we need some way to get data to this component - which means we should create a container component which is connected to Redux. Do that now and pass Results the data it needs.
- If you're feeling fancy make it so you only fetch decisions if they either haven't been fetched already or they're "stale" meaning it's beena while since they've been fetched.
- Also if you're feeling extra fancy go ahead and make it so that when you fetch a new decision it takes the author of that decision and caches that user under Redux.
- Now your app and state tree should look similar to this, notice we've now set a listener as well.
You'll notice that on the final version on the /results
view the UI shows a green line on the left with a checkmark on the right of the Result if the user has taken that decision or a red line (and an empty cirlce) if they haven't. In order to implement that we need to go ahead and fetch every decision that the user has made when the app loads.
- Head back to your Results component and add a new prop called
decisionsMade
. This prop will be all of the decisions the user has made. Now, in order to know if they've made a particular decision or not, we can look ifprops.decisionsMade[decisionId]
is truthy or not. Update the UI to show the checkmark/circle as well as the red or green line. - Because we're now assuming Results is going to receive
decisionsMade
, we actually need to pass those in. Head over to your ResultsContainer and getdecisionsMade
which we'll put on each users object in theusers
module. So again, each user will eventually have adecisionsMade
property. Make sure you passdecisionsMade
down to Results. - At this point all we need to do is update the authed users
decisionsMade
property. We'll do this right when the MainComponent mounts. - Head over to your
users
module and create a new Thunk function calledfetchAndAddUsersMadeDecisions
and fill it out to make the appropriate request. - Once that's done go ahead and add it to
fetchAndHandleAuthedUser
so that when a user authenticated, we'll fetch their made decisions. - Now, head over to MainContainer and also call
fetchAndHandleAuthedUser
when the component mounts so we'll have their made decisions on a refresh. - At this point your app should look like this,
We're so close! The second to last thing we need to do is make the Decisions clickable. Obviously the whole point of Would You Rather is that you can choose between two decisions. I'm going to leave this one entirely up to you. Here's the end result for the (what I'm calling) /decision
view. If you get stuck, there is a branch.
Last but not least, we need to hook up our Logout
button to actually unauthenticate us. At this point you should have the hang of this. I'm going to leave this one up to you as well. Check out the branch if you get stuck.
If you got through this, great job. If you're reading this and you've made it this far, be proud. This was a pretty advanced project. Take a look at my code and compare to what we did differently.
Suggestion or just want to say thanks? You can find me on Twitter at @tylermcginnis33