Skip to content

My workshop repo for React From Scratch updated for Webpack 5 and React 17 for the Fullstack React Friday's Meetup.

Notifications You must be signed in to change notification settings

httpJunkie/react-from-scratch

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

7 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

Build a React App From Scratch (Using Webpack, Babel & ES Lint)

This repo contains the finished state of the "React From Scratch" application from our Workshop. Walk through building a basic React app using a custom Webpack 5 and Babel configuration with ESLint and teaches the basics around component creation, composition, layout, styling with SCSS, and utilizing Context API for state management in React.

This document serves as a step-by-step guide for the workshop that can be used to easily follow along or as a reference if you want to walk through at your own pace at a later time.

Getting to "Hello React" with Webpack & Babel

Let's build up our React toolchain.

Create Project Folders and Initialize npm

mkdir space-x-app
cd space-x-app
npm init -y
mkdir app
code .

Go ahead and add your name as the author and set the description in your package.json file with: "SpaceX Launches and Historical Events".

We now have a package.json and app directory, let's start by creating a few files in the root directory:

  • .gitignore
  • webpack.config.js

Then create four more files inside the app directory:

  • index.js
  • index.html
  • App.js
  • App.scss

Dev Dependencies for React

I created three separate commands for installing our developer dependencies by category (Webpack, Babel, loaders). Let's install them!

  • npm i --save-dev webpack webpack-cli webpack-dev-server html-webpack-plugin@5.0.0-alpha.17
  • npm i --save-dev @babel/core @babel/preset-env @babel/preset-react
  • npm i --save-dev babel-loader node-sass file-loader sass-loader style-loader css-loader url-loader

If you would like to learn more about Webpack and Babel, I suggest a podcast episode Syntax.fm #004 - JavaScript Tooling which does a great job of describing why we need Webpack and Babel together for most React projects, as well to visit the links below for instruction on the latest Webpack 5 configuration in JavaScript + React.

Babel and JavaScript + React Resource

  1. How to set up a modern JavaScript project
  2. How to set up Webpack 5
  3. How to set up Webpack 5 with Babel
  4. How to React with Webpack 5
  5. How to Images with Webpack 5

To get more information on the html-webpack-plugin you can visit webpack.js.org and using html-webpack-plugin to generate index.html.

Create Build/Start Scripts to Run Webpack & Dev-server

We need to add a build and start command to our package.json in place of the existing test script:

    "test": "echo \"No test script\"",
    "build": "webpack",
    "start": "webpack serve"

Add Babel & Presets to package.json

What is preset-env? Check out this article:What is babel-preset-env and why do I need it?. Know that each plugin is its own npm library and this one allows transpiling ES6 to legacy JS code.

@babel/preset-env is a smart preset that allows you to use the latest JavaScript without needing to micromanage which syntax transforms (and optionally, browser polyfills) are needed by your target environment(s). This both makes your life easier and JavaScript bundles smaller!

preset-react is a bundle of other npm packages which I sugest getting to know more about each one individually:

  • @babel/plugin-syntax-jsx
  • @babel/plugin-transform-react-jsx
  • @babel/plugin-transform-react-display-name

What is preset-react? It allows us to use JSX in conjunction with Webpack, Babel and React.

Even in our most simple “Hello React” example, we need JSX, so we need to add the following code to our package.json just above our scripts:

"babel": {
  "presets": [
    "@babel/preset-env",
    "@babel/preset-react"
  ]
},

📘 Other Babel Config Options

Using package.json is just one option for configuring Babel and in my opinion, the easiest to use, however; you could also go the .babelrc route. Both are explained on the same documentation link above!

Production Dependencies for “Hello React”

We need to install React 17 and Normalize for this project.

Articles on Normalize('normalize.css'):

Normalize is a modern CSS reset for cross-browser consistency considering the styling of HTML elements.

npm i --save react react-dom react-router-dom normalize.css

Bootstrap Our App and Create Our First Component

Next, we bootstrap our Index.html file and get our React app working with our initial App component.

Index.html

<!DOCTYPE html>
<html lang='en' dir='ltr'>
<head>
  <title>SpaceX Events Home</title>
  <link href='http://fonts.googleapis.com/css?family=Lato:200,300,400' rel='stylesheet'>
  <meta name='viewport' content='width=device-width, initial-scale=1'>
  <meta name='Description' content='Author: Your Name'>
</head>
<body>
  <div id='root'></div>
</body>
</html>

index.js (JS Entrypoint/Bootstrapper)

import React from 'react'
import ReactDOM from 'react-dom'

import App from './App'

ReactDOM.render(<App />, document.getElementById('root'))

App.js (Initial Component)

import React from 'react'
import 'normalize.css'
import './App.scss'

import reactLogo from './assets/images/react-transparent.png'

const App = () => {
  return (
    <>
      <h1>Hello React</h1> 
      <br/>
      <img height="150" src={reactLogo} />
    </>
  )
}
export default App

App.scss (Ensure Transpiling to Css Works)

An Article on Sass: Why Sass

Why Sass(SCSS) over CSS? This is something we utilize in our project and without Sass/Scss we would not be able to easily create our light and dark theme (later on). If you would like to learn more about Sass/SCSS, check out this article.

body {
  background-color: #222;
}
h1 {
  padding: 0 0 0 1em;
  color: #61dafb;
  font-family: Lato;
}
img {
  margin-left: 2em;
}

.gitignore (Assets Not Tracked by Git)

For more information on creating a .gitignore file and why we need them, check out an article on ignoring files for more info.

# dependencies
/node_modules

# testing
/coverage

# production
/build

# distribution
/dist
/public

# misc
.DS_Store
.env.local
.env.development.local
.env.test.local
.env.production.local

npm-debug.log*
yarn-debug.log*
yarn-error.log*

webpack.config.babel.js

let path = require('path')
let HtmlWebpackPlugin = require('html-webpack-plugin')

module.exports = {
  entry: "./app/index.js",
  module: {
    rules: [
      {
        test: /\.(js|jsx)$/,
        exclude: /node_modules/,
        use: ['babel-loader'],
      },
      {
        test: /\.css$/,
        use: ["style-loader", "css-loader"]
      },
      {
        test: /\.scss$/,
        use: [
          "style-loader", // creates style nodes from JS strings
          "css-loader", // translates CSS into CommonJS
          "sass-loader" // compiles Sass to CSS, using Node Sass by default
        ]
      },
      {
        test: /\.(png|jpg|jpe?g|gif|svg)$/, use: { loader: "url-loader?limit=100000" }
              // limit: Number or String specifying the maximum size of a file in bytes. 
              // If the file size is equal or greater than the limit will be used.
      },
      {
        test: /\.woff(2)?(\?v=[0-9]\.[0-9]\.[0-9])?$/, use: { loader: "url-loader?limit=10000&mimetype=application/font-woff" }
      },
      {
        test: /\.(ttf|eot|svg)(\?v=[0-9]\.[0-9]\.[0-9])?$/, use: { loader: "file-loader" }
      }
    ]
  },
  mode: "development",
  resolve: {
    extensions: ['*', '.js', '.jsx'],
  },
  output: {
    path: path.resolve(__dirname, "dist"),
    filename: "index_bundle.js",
    publicPath: "/"
  },
  devtool: "source-map", // https://survivejs.com/webpack/building/source-maps/#inline-source-map-types
  devServer: {
    historyApiFallback: true, // https://jaxenter.com/build-reactjs-history-api-fallback-153122.html
    contentBase: path.resolve(__dirname, './dist'),
  },
  plugins: [
    new HtmlWebpackPlugin({
      template: "app/index.html"
    })
  ]
}

Finally, let's add a directory inside app called assets and another directory inside of it named images and you can drop the file: React Logo inside.

With these files in place, run the project for the first time.

npm start

And visit: http://localhost:9000

Init, Stage and Commit Our Files

You should get a page running on http://localhost:9000 with a dark background and “Hello React” heading with no errors. This indicates that our configuration and styles are working and React is rendering an application for us!

Let's stop and save changes.

Initialize Our Git Repository in Place

From the root of our project, initialize git:

git init

We should notice our .gitignore working as it will no longer be tracking our npm_modules directory 😉!

Create an Empty Repository on Github

Go to you're GiHub Repository and create a repo named: space-x-app to push this project to. Let's stage, commit and push to a new repo:

git add .
git commit -m "Hello React (Webpack, Babel, and React)"
git remote add origin https://github.com/[YourGithubAccount]/space-x-app.git
git push -u origin master

That's it for our “Hello React” portion of the workshop

We created our own Webpack, Babel & React build ourselves, without using something like Create React App (CRA) to and because of this we are smarter about our application and have a better understanding of how a minimal React application works under the hood and how to build and run that application from a basic standpoint.

We will tackle linting in the next section.

Getting Started with ESLint

Article on ESLint: Getting Started

We need to install and initialize ESLint.

npm i --save-dev eslint babel-eslint eslint-watch eslint-plugin-react

Add Linting for ES6, JSX and React Hooks

Create an ESLint configuration file named .eslintrc in the root directory of your project:

.eslintrc (ESLint Config)

{
  "plugins": [
    "react"
  ],
  "parser": "@babel/eslint-parser",
  "parserOptions": {
    "ecmaVersion": 6,
    "sourceType": "module",
    "ecmaFeatures": {
      "jsx": true
    }
  },
  "env": {
    "es6": true,
    "browser": true,
    "node": true
  },
  "extends": [],
  "rules": {
    "semi": ["error", "always"],
    "quotes": ["error", "double"]
  },
  "settings": {
    "react": {
      "version": "17.0"
    }
  }
}

Before we can run our linter, let's add a lint and lint:fix script to our package.json just above the test script:

    "lint": "eslint app --ext .js,.jsx",
    "lint:fix": "eslint --fix app --ext .js,.jsx",

Again, they are made to work specifically with React hence the .js and .jsx file extensions. Now let's run our linter.

npm run lint

Right away we get some errors:

✖ 18 problems (18 errors, 0 warnings)
  18 errors and 0 warnings potentially fixable with the `--fix` option.

This is great, it means our linter is working. But as a team, let's assume that we do not want to follow the double quote rule and we only want to follow the semi-colon rule. Let's make the following changes to the .eslintrc file:

    "semi": ["error", "never"],
    "quotes": ["error", "single", { "allowTemplateLiterals": true }]

We can find the rule here: ESLint Quote Rules

Now we should not get any errors.

Adding linting for React Hooks

Some rules that we may want to apply are not built into ESLint because they are framework-specific and we are working in React. If we want to apply the suggested rules for using Hooks in React, we need to add a new package that can be used as a plugin, we already have one plugin added and set up for JSX, but let's show how to add another one.

Let's install the package we need:

npm i --save-dev eslint-plugin-react-hooks

First we will add react-hooks to the plugin section the .eslintrc file:

  "plugins": [
    "react",
    "react-hooks"
  ],

then, we will add the rule to the rules section the .eslintrc file:

    "react-hooks/rules-of-hooks": "error",
    "react-hooks/exhaustive-deps": "warn",
    "semi": ["error", "never"],
    "quotes": ["error", "single", { "allowTemplateLiterals": true }]

To check the plugin is working, replace the first line in App.js with:

import React, { useEffect } from 'react'
useEffect(() => {
  console.log('This should trigger our Hooks rule')
})

Run our linter again and you should get a warning, appended to the end is the rule responsible for this error react-hooks/rules-of-hooks:

C:\dev\workshops\react-from-scratch\app\App.js
  2:1  error  React Hook "useEffect" cannot be called at the top level. React Hooks must be called in a React function component or a custom React Hook function  react-hooks/rules-of-hooks

Discard our changes in App.js Our Hooks linting is working.

Let's use a rule that will work with our react plugin! In the JSX-specific-rules section of their documentation we can see a rule for no duplicate props.

Add the following rule to the top of the .eslintrc file's rules section:

    "react/jsx-no-duplicate-props": "error",

We could then go into our index.js file and add duplicate props:

ReactDOM.render(<App isTrue={true} isTrue={false} />, document.getElementById('root'))

And we get the following error:

C:\dev\workshops\react-from-scratch\app\index.js
  6:36  error  No duplicate props allowed  react/jsx-no-duplicate-props

Again, the rule that triggered this error is listed at the end of the error.

Great, linting is working and I can leave exploring new rules up to you! This concludes the linting section of the workshop. Next up is some prep work needed to create a responsive frame for our application.

Create App Frame with Semantic HTML and Sass

Our app will live in this frame.

We need a component Frame.js. It's going to contain the shell of our application consisting of an outer div tag called app-container and inside of that, a semantic HTML structure:

<div className={`app-container`}>
  <main>
    <header></header>
    <section></section>
    <footer></footer>
  </main>
</div>
  • The app-container <div> contains the main and Sidenav component.
  • <main> contains a header, section and footer.
  • <header> will contain a Logo and Topnav component.
  • <section> will house our React Router.
  • <footer> will contain one component named Footer.

We can see the semantic tags mixed with our React Components below:

<div className={`app-container`}>
  <main>
    <header>
      <Logo />
      <Topnav />
    </header>
    <section>
      {/* Routes Load Here */}
    </section>
    <footer>
      <Foot />
    </footer>
  </main>
  <Sidenav />
</div>

We will use a combination of Flexbox and CSS Grid styles in conjunction with these semantic HTML tags to create a solid layout. Before we add those styles, let's get the files we need set up, and the navigation working and switching routes.

The Frame

Create a new file in the app directory called Frame.js.

Create Sub-Directories For Components

Inside app directory, create two sub-directories: partial-components, ui-components, and view-components.

Create View Component Files

Inside the view-components directory create two files: Home.js and Events.js.

Create Partial-View Component Files

Inside the partial-components directory create seven files: Logo.js, Menu.js, Foot.js, Sidenav.js, Sidenav.scss, Topnav.js, and Topnav.scss.

Create UI Component Files

Inside the ui-components directory create two more directories: hamburger and switch.

Inside the hamburger directory create two files: Hamburger.js, Hamburger.scss.

Inside the switch directory create three files: Switch.js, Switch.scss, switch-variables.scss.

Filling in the Blanks

Below is the code for each file we created:

Hamburger.js (UI Component)

import React from 'react'
import './Hamburger.scss'

const Hamburger = ({ onClick }) => {
  return (
    <div className="hamburger-menu">
      <input type="checkbox" className="input-check2" id="input-check2" 
        hidden onClick={onClick} />
      <label className="bars-container bars-container2" htmlFor="input-check2">
        <span className="bar bar1"></span>
        <span className="bar bar2"></span>
        <span className="bar bar3"></span>
      </label>
    </div>
  )
}

export default Hamburger

Hamburger.scss (Stylesheet)

:root {
  --animationSpeed: 0.25s;
}

.hamburger-menu {
  display: inline-block;
  padding: 3px;
  width: 29px;
  margin-top: -7px;
  margin-right: -10px;
}

.bars-container {
  display: flex;
  flex-direction: column;
  justify-content: space-between;
  height: 20px;
}

.bars-container,
.bar {
  cursor: pointer;
}

.app-container.light .bar {
  width: 100%;
  height: 3px;
  background-color: #222;
  border-radius: 5px;
  display: block;
}
.app-container.dark .bar {
  width: 100%;
  height: 3px;
  background-color: #FFF;
  border-radius: 5px;
  display: block;
}

.bar1 {
  transform-origin: center left;
  position: relative;
}

.bars-container2 {
  position: relative;
}

.bars-container2 .bar {
  position: absolute;
  transform-origin: center;
}

.bars-container2 .bar1 {
  top: 5%;
  animation: makeCross1Backward var(--animationSpeed) ease 0s 1 forwards, moveDownBackward var(--animationSpeed) ease var(--animationSpeed) 1 forwards;
}

.bars-container2 .bar2 {
  top: 50%;
  transition: opacity 0.4s ease;
}

.bars-container2 .bar3 {
  top: 95%;
  animation: makeCross2Backward var(--animationSpeed) ease 0s 1 forwards, moveUpBackward var(--animationSpeed) ease var(--animationSpeed) 1 forwards;
}

.input-check2:checked + label .bar1 {
  animation: moveDown var(--animationSpeed) ease 0s 1 forwards, makeCross1 var(--animationSpeed) ease var(--animationSpeed) 1 forwards;
}

.input-check2:checked + label .bar2 {
  opacity: 0;
}

.input-check2:checked + label .bar3 {
  animation: moveUp var(--animationSpeed) ease 0s 1 forwards,
    makeCross2 var(--animationSpeed) ease var(--animationSpeed) 1 forwards;
}

@keyframes moveDown {
  from { top: 5%; transform: translateY(0); }
  to { top: 50%; transform: translateY(-50%); }
}

@keyframes moveDownBackward {
  from { top: 50%; transform: translateY(-50%); }
  to { top: 5%; transform: translateY(0); }
}

@keyframes moveUp {
  from { top: 95%; transform: translateY(0); }
  to { top: 50%; transform: translateY(-50%); }
}

@keyframes moveUpBackward {
  from { top: 50%; transform: translateY(-50%); }
  to { top: 95%; transform: translateY(0); }
}

@keyframes makeCross1 {
  from { transform: rotate(0deg); }
  to { transform: rotate(40deg); }
}

@keyframes makeCross1Backward {
  from { top: 50%; transform: rotate(40deg) translateY(-50%); }
  to { top: 50%; transform: rotate(0deg) translateY(-50%); }
}

@keyframes makeCross2 {
  from { transform: rotate(0deg); }
  to { transform: rotate(-40deg); }
}

@keyframes makeCross2Backward {
  from { top: 50%; transform: rotate(-40deg) translateY(-50%); }
  to { top: 50%; transform: rotate(0deg) translateY(-50%); }
}

Switch.js (UI Component)

import React from 'react'
import './switch.scss'

const Switch = ({ onChange }) => {
  return (
    <div>
      <label className="ui-switch">
        <input id="check" className="input" type="checkbox" 
          onChange={onChange} />
        <div className="circle"></div>
      </label>
    </div>
  )
}

export default Switch

Switch.scss (Stylesheet)

:root {
  --switch-background: #444;
}

@import 'switch-variables';

label.ui-switch {
  display: inline-block;
  width: $switch-width;
  height: $switch-height;
  background: var(--switch-background);
  position: relative;
  border-radius: 25px;
  cursor: pointer;
}
label.ui-switch:active > .circle {
  width: $circle-size;
}
label.ui-switch:active > .input:checked + .circle {
  left: $circle-size;
}

label.ui-switch > .input {
  width: 100%;
  height: 100%;
  position: absolute;
  left: 0px;
  top: 0px;
  opacity: 0;
  z-index: 100;
  cursor: pointer;
}
label.ui-switch:hover {
  background: #555;
}
label.ui-switch > .input:checked + .circle {
  left: $circle-size;
  top: 0px;
  background: rgba($circle-onn-color, $circle-onn-opacity);
}
label.ui-switch > .circle {
  position: absolute;
  left: 0px;
  top: 0px;
  margin: 4px;
  width: $circle-size;
  height: $circle-size;
  border-radius: 50%;
  background: rgba($circle-off-color, $circle-off-opacity);
  z-index: 100;
  transition: all $transition-time $transition-style;
  box-shadow: 0px 10px 20px 0px rgba(99, 107, 144,.35);
}

switch-variables.scss (Stylesheet)

$circle-onn-color: rgb(255,255,255);
$circle-onn-opacity: 0.99;
$circle-off-color: rgb(155,155,155);
$circle-off-opacity: 0.99;

$switch-width: 48px;
$switch-height: 28px;
$circle-size: 20px;

$transition-time: 0.3s;
$transition-style: ease;

Foot.js (Partial-View Component)

import React from 'react'

import Switch from '../ui-components/switch/Switch'

const Foot = () => {
  return (
    <>
      <div className="left">
        🚀 The Boldly Go Company
      </div>
      <div className="right">
        <Switch />
      </div>
    </>
  )
}

export default Foot

Logo.js (Partial-View Component)

import React from 'react'

const Logo = () => {
  return (
    <div className={`logo`}>
      <span style={
        { marginTop: '-2px', transform:'rotate(-90deg)' }
      }></span> <span>SpaceX</span>
    </div>
  )
}

export default Logo

Menu.js (Partial-View Component)

import React from 'react'

import { NavLink } from 'react-router-dom'
import Hamburger from '../ui-components/hamburger/Hamburger'

const Menu = () => {
  return (
    <ul style={{userSelect: 'none'}}>
      <li className='link'>
        <NavLink className='text_link' tabIndex='2' activeclassname='active' to='/'>Home</NavLink>
      </li>
      <li className='link'>
        <NavLink className='text_link' tabIndex='3' activeclassname='active' to='/events'>Events</NavLink>
      </li>
      <li className='link'>
        <a className='text_link' tabIndex='4' href='https://github.com/httpJunkie/react-from-scratch'>
          Source Code
        </a>
      </li>
      <li className='menu'>
        <Hamburger />
      </li>
    </ul>
  )
}

export default Menu

Sidenav.js (Partial-View Component)

import React from 'react'
import './Sidenav.scss'

import Menu from './Menu'

const Sidenav = () => {
  return (
    <div className={'sidenav'}>
      <Menu />
    </div>
  )
}

export default Sidenav

Sidenav.scss (Stylesheet)

.sidenav ul {
  display: flex;
  flex-direction: column;
  margin-block-start: 0;
  margin-block-end: 0;
  margin-inline-start: 0;
  margin-inline-end: 0;
  padding-inline-start: 0;
}
.sidenav li {
  display: flex;
  justify-content: flex-start;
  align-items: center;
  min-height: 60px;
}
.sidenav li > a {
  margin-left: 1em;
}
.sidenav li:last-child {
  display: none;
}
.active {
  font-weight: bold;
}
.sidenav {
  background-color: #EFEFEF;
  color: #222222;
}
.sidenav li {
  border-bottom: 1px solid #555555;
}

Topnav.js (Partial-View Component)

import React from 'react'
import './Topnav.scss'

import Menu from './Menu'

const Topnav = () => {
  return (
    <div className={'topnav'}>
      <Menu />
    </div>
  )
}

export default Topnav

Topnav.scss (Stylesheet)

.topnav {
  width: 70%;
  display: flex;
  justify-content: flex-end;
}
  
.topnav ul { padding: 0; }
.topnav ul { display: flex; margin: 0; }
.topnav ul li { list-style-type: none; }
.topnav ul > li:not(:last-child) { margin-right: 15px; }
.topnav ul > li a { color: #212529; }
.app-container .topnav ul > li.menu { cursor: pointer; cursor: hand; }

Notice how the Topnav.js and Sidenav.js files below are nearly identical. They both simply import the menu (component reuse). Their styles (SCSS) will make them different laying out horizontally (Topnav) vs vertically (Sidenav).

Events.js (View Component)

import React, { useEffect } from 'react'

const Events = () => {
  useEffect(() => { document.title = `SpaceX Historical Events` })

  return (
    <div className='view-events'>
      <h3>Historical Events</h3>
    </div>
  )
}

export default Events

Home.js (View Component)

import React, { useEffect } from 'react'

const Home = () => {
  useEffect(() => {document.title = `Home Page`},[])

  return (
    <div className='view-home'>
      <h3>Next Launch</h3>
      {/* Launch Component */}
    </div>
  )
}

export default Home

Now that we have each of those files created, let's put it all together in our Frame.js and include our routing and everything we need to lazy load our view-components, etc...

Video: Code Splitting with React and React Router v4

At its most basic, Lazily loading your Events component means that the data that is a part of that page does not get fetched and loaded if you visit that route.

Frame.js

import React, { lazy, Suspense } from 'react'
import { BrowserRouter, Route, Switch } from 'react-router-dom'

const Home = lazy(() => import('./view-components/Home'))
const Events = lazy(() => import('./view-components/Events'))
const LoadingMessage = () => `loading...`

import Logo from './partial-components/Logo'
import Sidenav from './partial-components/Sidenav'
import Topnav from './partial-components/Topnav'
import Foot from './partial-components/Foot'

const Frame = () => {
  return (
    <BrowserRouter>
      <div className={`app-container light`}>
        <main>
          <header>
            <Logo />
            <Topnav />
          </header>
          <section>
            <Switch>
              <Suspense fallback={<LoadingMessage />}>
                <Route exact path='/' component={Home} />
                <Route exact path='/events' component={Events} />
              </Suspense>
              <Route render={() => <h2>404 Page Not Found</h2>} />
            </Switch>
          </section>
          <footer>
            <Foot />
          </footer>
        </main>
        <Sidenav />
      </div>
    </BrowserRouter>
  )
}

export default Frame

We will now need to add the Frame component to our App component:

App.js (Load Our Frame)

import React from 'react'
import 'normalize.css'
import './App.scss'

import Frame from './Frame'

const App = () => <Frame />

export default App

At this point our app is working, we have the classes and styles for our imported components, but we need to add styles to our App.scss page that use flexbox and CSS Grid to position our header, section, footer and logo, navigation, and some basic text and link styles. Replace what is in the App.scss page with the following:

App.scss (Stylesheet)

.app-container {
  display: flex;
  height: 100vh;
  font-family: 'Lato', sans-serif;
  text-rendering: optimizeLegibility !important;
  -webkit-font-smoothing: antialiased !important;
}
p {
  font-size: 1em;
  max-width: 80%;
}
a.text_link {
  color: inherit;
  text-decoration: none;
  margin-bottom: 0.25em;
}
a.text_link {
  background-image: linear-gradient(currentColor, currentColor);
  background-position: 0% 100%;
  background-repeat: no-repeat;
  background-size: 0% 4px;
  background-position-y: 1.1em;
  transition: background-size cubic-bezier(0, 0.5, 0, 1) 0.3s;
}
a.text_link:hover,
a.text_link:visited:hover,
a.text_link:active:hover {
  text-decoration: none;
  background-size: 100% 4px;
}
a.text_link:focus,
a.text_link:visited {
  text-decoration: none;
  background-size: 0% 4px;
}
a:active {
  background-color: 555;
}
main {
  width: 100%;
  display: grid;
  grid-template-rows: 60px auto 40px;
}
/* Side Navigation */
.sidenav {
  min-width: 150px;
  height: 100vh;
}
/* Top Navigation */
header {
  display: flex;
  height: 60px;
  font-size: 18px;
}
.logo {
  width: 50%;
  margin: 1em;
  font-size: 20px;
}
.topnav {
  width: 50%;
  margin: 1.2em;
}
/* Content Section */
section {
  height: calc(100% - 2em);
  padding: 1em;
}
/* Footer */
footer {
  height: 40px;
  display: flex;
}
footer .left {
  width: 80%;
  margin: 0 0.5em 10px 1em;
  font-size: 20px;
}
footer .right {
  width: 20%;
  margin: 0 1em 10px 0.5em;
  div {
    margin-left: auto;
    width: 50px;
  }
}

A Walkthrough for Each File and Styles

I know that we have added a lot of styles blindly, so we'll take time in our workshop to go over the structure and what exactly is going on.

This concludes building our App Frame section.

Run npm start and ensure that we can navigate to each route, there are some styles missing that we will add later.

In our next section, we will make the frame responsive and get a crash course in React Hooks and Context API.

Making Our App Frame Responsive

Let's install and use our first React Hook in this project called react-media-hook:

npm i react-media-hook

It has two APIs we can try: useMedia('(min-width: 600px)') will recieve an object similar to using matchMedia() in JavaScript:

import { useMedia } from 'react-media-hook'
const matchMediaObject = useMedia('(min-width: 600px)')

In this case, matchMediaObject could equal something like:

{
  matches: true
  media: "(min-width: 600px)"
}

This is a bit overkill and we just need a boolean. We can instead try useMediaPredicate('(min-width: 600px)') which will return a boolean based on our media query that we passed in matching. So we can do something this:

import { useMediaPredicate } from 'react-media-hook'
const isMedium = useMediaPredicate('(min-width: 600px)')

Now, isMedium will be true when our browser is at least 600 pixels or greater and false when less than 600 pixels.

We'll use that true or false value to add a 'small' or 'medium' class to our app-container div inside Frame.js.

In Frame.js: let's import useMediaPredicate just below our react-router import:

import { useMediaPredicate } from 'react-media-hook'

Then, above our return statement in the same file, add:

const isMedium = useMediaPredicate('(min-width: 600px)')
const breakpoint = isMedium ? 'medium' : 'small'

Next, add a class to the app-container div using ${breakpoint} inside our string interpolation:

<div className={`app-container ${breakpoint} light`}>

To see this working, run the project and inspect the app-container div with F12 (dev tools). Watch the class name change as we resize the browser encountering that 600-pixel threshold.

site preview

With that in place, let's hide the top navigation links and show the menu button on small and visa versa on medium.

In Topnav.scss: add with the following code:

.app-container.small .topnav ul > li.link { display: none; }
.app-container.medium .topnav ul > li.link { display: auto; }
.app-container .topnav ul > li.menu { cursor: pointer; cursor: hand; }
.app-container.small .topnav ul > li.menu { display: auto; }
.app-container.medium .topnav ul > li.menu { display: none; }

In Sidenav.scss: add with the following code:

.app-container.small .sidenav { display: auto; }
.app-container.medium .sidenav { display: none; }

We will also enable a default ensuring the Sidenav is closed initially (even on small). To do that we create a global state object with Context API and Hooks, specifically useState(a built-in React Hook).

Additional Instruction on React State

Although it's beyond the scope of this course to cover React Hooks, which we will be using here, you can learn everything a beginner would need to know about modern React state and Hooks in my React Hooks Guide.

Adding Context to Our App

Create a file that will house our Context Provider.

Create a folder named context inside app directory. Then, inside app, create a file: AppContext.js.

In AppContext.js: add the following code:

import React, { useState, createContext } from 'react'

const AppContext = createContext()
const AppProvider = props => {
  const [appData, setApp] = useState({
    navOpen: false,
    toggleSidenav: value => setApp(data => (
      { ...data, navOpen: value }
    )),
  })
  
  return <AppContext.Provider value={appData}>{props.children}</AppContext.Provider>
}

export { AppContext, AppProvider }

Here we import useState and createContext. Next, we create a context instance named AppContext. We then set up an AppProvider using the render props pattern. We declare a Hook with useState, pass in an object with properties, defaults and methods. In this case, the object has a property named navOpen, and a method called toggleSidenav() which can be used to update the value of the navOpen property.

Our AppProvider works by returning an <AppContext.Provider> component with access to our appData giving all children of our <AppProvider> that we will place on the App.js page access to our new global state. <AppContext.Provider> receives state as a prop, and gives its children access to it.

Provide Context to Our Entire App

Next, we need to have access to this state in many places in our app. We want any component to be able to have access to its data, so we need to place the <AppProvider> component at the highest level in our component tree. To wrap all components, we must place it in the App.js file.

In App.js: add the following import:

import { AppProvider } from './context/AppContext'

And replace:

const App = () => <Frame />

With:

const App = () => {
  return(
    <AppProvider>
      <Frame />
    </AppProvider>
  )
}

This will give any component in our application access to our context by wrapping the <Frame> component with the <AppProvider> component by way of render props.

Consuming Context in Our App

We want to import our AppContext into our Menu.js file and our Sidenav.js file and consume it with useContext.

Menu.js

In Menu.js: import useContext and AppContext:

import React, { useContext } from 'react'
import { AppContext } from '../context/AppContext'

Just above the return statement, we will add our context with a call to useContext:

const context = useContext(AppContext)
const handleClick = () => {
  context.toggleSidenav(!context.navOpen)
}

We tap into that context and change the state of navOpen when clicking the menu icon or pressing Enter upon focus.

Replace the last <li></li> with:

      <li className='menu'>
        <Hamburger onClick={handleClick} />
      </li>

Sidenav.js

In Sidenav.js: import useContext and the AppContext from the provider:

import React, { useContext } from 'react'
import { AppContext } from '../context/AppContext'

Above the return statement, add our context with a call to useContext:

const context = useContext(AppContext)

Alter the className of the parent div with 'show' or 'hide', based on the value of the context.navOpen:

    <div className={`sidenav ${context.navOpen ? 'show' : 'hide'}`}>
      <Menu />
    </div>

App.scss

In App.scss: look for a comment: /* Side navigation */ and replace it with:

/* Side Navigation */
.sidenav,
.sidenav.show {
  min-width: 150px;
  height: 100vh;
}
.sidenav.hide {
  display: none;
}

Now when we click on the menu icon, it will set the state to its opposite value, and our Sidenav component will re-render the new class and the CSS will hide or show the side nav panel.

This concludes our "Creating a Semantic and Responsive Frame (Part 2)" section. Next, we will round out our frame by adding a theme toggle using a custom CSS toggle slider. We will create some global state to track 'light' and 'dark' theme controlled by a Switch component in the footer to toggle between light and dark mode.

Alternate theme with Context API and Hooks

We need to think about our custom styles like background and text color, how they change for each theme. Any CSS property that controls background, color, borders and other attributes that will need to possibly change with the toggle of the theme, it all needs to be scoped to a light and dark theme selector.

Creating Our Custom Styles

We already have a light theme going on, so we'll start there.

Eventually, this class will be bound to our themeMode property in our global state.

<div className={`app-container ${breakpoint} light`}>

With this class, we can now create a nested SCSS selector to encapsulate the light vs dark theme through namespacing.

In App.scss: add the following code:

/* Light Theme */
.app-container.light {
  header {
    background-color: #FFFFFF;
    color: #222222;
  }
  section {
    border-top: 1px solid #010101;
    background-color: #FFFFFF;
    color: #222222;
  }
  section a {
    background-color: #EEEEEE;
  }
  footer {
    background-color: #FFFFFF;
    color: #222222;
  }
  input {
    color: #222222;
  }
}

Understand how this works, and how namespacing is being applied. We moved theme related code from section a selector into a selector of the same name nested (Sass Style) with a app-container.light to scope it to the light theme.

The only modification I did to the originall was that I moved the following properties:

  section {
    border-top: 1px solid #010101;
    background-color: #FFFFFF;
    color: #222222;
  }

Into a new selector giving it more specificity.

.app-container.light {
  section {
    border-top: 1px solid #010101;
    background-color: #FFFFFF;
    color: #222222;
  }
}

Now styles like size and position that need to be consistent over both themes are at the root level of the document. Colors and text and things that are specific to each theme are extracted into the namespace .app-container.light.

For changing the theme from light to dark in the UI we will use a Switch component.

Updating the Foot Component

In Foot.js: import our AppContext, useContext and handleSwitch:

import React, { useContext } from 'react'
import { AppContext } from '../context/AppContext'

import Switch from '../ui-components/switch/Switch'

const Foot = () => {
  const context = useContext(AppContext)
  const isLight = context.themeMode === 'light'

  const handleSwitch = () => {
    context.changeTheme(isLight ? 'dark' : 'light')
  }
  
  return (
    <>
      <div className="left">
        🚀 The Boldly Go Company
      </div>
      <div className="right">
        <Switch onChange={handleSwitch} />
      </div>
    </>
  )
}

export default Foot

This handleSwitch() function will update the global state of themeMode using a ternary that flips either light to dark or visa versa.

We have wired up our switch to toggle the theme with an onChange().

Wiring up Our Switch to Work with Our Context

The Switch component is setup, and we will need to create another property in our global state that we can affect when this Switch is turned on and off, but first I want to show you how we can possibly tell the users preferred theme.

Checking the User's Preferred Color Scheme

We will take advantage of a media query named refers-color-scheme. Normally you could call this in CSS and do something like:

@media (prefers-color-scheme: dark) {
  body {
    background: #111;
    color: #eee;
  }
}

We just used react-media-hook to check a minimum width on the browser, and everytime it changes, we can react to those changes in our component and display the correct breakpoint:

We can do the same thing for prefers-color-scheme

In AppContext.js: import useMediaPredicate:

import React, { useState, useEffect, createContext } from 'react'
import { useMediaPredicate } from 'react-media-hook'

Then we can add a constant named preferredTheme to the AppProvider component inside the context/AppContext.js file. Drop this line of code, just above the appData state:

  const preferredTheme = useMediaPredicate('(prefers-color-scheme: dark)') ? 'dark' : 'light'

This gives me an easy way to know if the user prefers 'dark' vs 'light' on their device. It returns true if it matches.

Next, we will add the following code to our AppProvider's appData default object, put this just after toggleSidenav() method:

    themeMode: localStorage.getItem('space-x_theme') || preferredTheme,
    changeTheme: mode => setApp(data => (
      {...data, themeMode: mode }
    ))

This checks localStorage first because if the user has a saved theme preference with us, we prioritize that, otherwise we will fall back to preferredTheme. The values I can expect to get back are: ['dark','light','no-preference'], as found in the docs on mozilla.org: Mozilla page for prefers-color-scheme.

We check for 'dark' first because if they don't have that as prefers-color-scheme, we know that we want to fall back to 'light' because they either prefer 'light' or they have 'no-preference' meaning we still default to light mode.

Side Effect Needed When themeMode Changes

We should know a little about useEffect(), enough to understand that it is the mechanism used in conjunction with hooks and state to set our localStorage each time we change it with the Switch. So we will add a useEffect right here inside the AppProvider, and it will run anytime that appData.themeMode get's changed.

We have already added an import for useEffect:

import React, { useState, useEffect, createContext } from 'react'

We can add useEffect just below our AppProvider's useState hook and call setItem on our localStorage. This fires on the first render and depends on the [app.Data.themeMode] as seen in our dependency array. Having that value in the dependency array ensures that our component only rerenders when themeMode changes.

  useEffect(() => {
      localStorage.setItem('space-x_theme', appData.themeMode)
    }, [appData.themeMode]
  );

Adding a Theme CSS Class to app-container

Now that the Switch works and changes the context.themeMode when we click it, we need our app-container div inside of Frame.js to reflect this class.

In Frame.js: import useContext and AppContext:

import React, { useContext, lazy, Suspense } from 'react'
import { AppContext } from './context/AppContext'

Create a const named context at the top level of our Frame component, just above const isMedium = useMediaPredicate(...):

const context = useContext(AppContext)

Now we can bind one of our classes on the app-container div to this context.themeMode:

<div className={`app-container ${breakpoint} ${context.themeMode}`}>

We should be able to see the class changing if we run the project and inspect the app-container div:

Although our Switch is functional, we need to add the dark theme and have the CSS respond and switch to a dark theme, we will need to add code to the App.scss, Sidenav.scss and Topnav.scss to achieve this.

Going Dark Mode

We will import these files into our App.scss file and nest them inside of an SCSS selector and this will scope certain styles to the CSS namespace of app-container.light.

We need to create another nested SCSS set of styles for the dark theme. It will have a comment above it, just as the light mode version.

In App.scss: add the following code:

/* Dark Theme */
.app-container.dark {
  header {
    background-color: #010101;
    color: #FFFFFF;
  }
  section {
    border-top: 1px solid #EFEFEF;
    background-color: #010101;
    color: #FFFFFF;
  }
  section a {
    background-color: #333333;
  }
  footer {
    background-color: #010101;
    color: #FFFFFF;
  }
  input {
    color: #FFFFFF;
  }
}

Similar to the light theme code in .app-container.light the dark theme swaps the light backgorund with a dark and the dark text with light.

Sidenav and Topnav Theming

We have some CSS in two other files will also need some light and dark treatment:

Add Dark Theme to Sidenav

In Sidenav.scss: replace:

.sidenav {
  background-color: #EFEFEF;
  color: #222222;
}
.sidenav li {
  border-bottom: 1px solid #555555;
}

with:

/* Light Theme */
.app-container.light {
  .sidenav {
    background-color: #EFEFEF;
    color: #222222;
  }
  .sidenav li {
    border-bottom: 1px solid #555555;
  }
}

/* Dark Theme */
.app-container.dark {
  .sidenav {
    background-color: #1e1e1e;
    color: #FFFFFF;
  }
  .sidenav li {
    border-bottom: 1px solid #EFEFEF;
  }
}

Add Dark Theme to Topnav

In Topnav.scss: add the following code:

/* Light Theme */
.app-container.light {
  .topnav ul > li a {
    color: rgb(33, 37, 41);
  }
}

/* Dark Theme */
.app-container.dark {
  .topnav ul > li a {
    color: rgb(240, 240, 240);
  }
}

If you have not stopped your project, you will need to now and run npm start again. With the project restarted you should get the full theme switching experience:

This concludes our section on creating a theme and making it toggle from light to dark on the fly.

Add a Launch Component With Data from SpaceX API

SpaceX API: r/SpaceX API Docs

With all of the app specific code out of the way we can finally get to the purpose of our app and start working on the Space-X events related code.

We are going to use the SpaceX public API. My idea for the home page is to show the next SpaceX launch. We have a route specifically for this. I'd like to simply render a component on the page that I can pass an individual flight_number={7} to and get any of the SpaceX Launches or to be able to pass a prop like next={true}.

<Launch flight={74} />

or

<Launch next={true} />

To get started we will need a Launch partial component and we should keep it in a separate folder from the site related partials. Before we create this new component, we need to install prop-types to specify the props coming into our components.

npm i prop-types

Create a new directory inside paritial-components called space-x and inside of it we will create a Launch.js file:

In Launch.js: add the following code:

import React, { useState, useEffect} from 'react'

const getSpaceXLaunchNext = () => {Launch.js
  return fetch('https://api.spacexdata.com/v3/launches/next')
    .then((res) => res.json())
}

const Launch = () => {
  const [launch, setLaunch] = useState(null)
  
  useEffect(() => {
    getSpaceXLaunchNext()
      .then(setLaunch)
  }, [])
  
  if (launch === null) {
    return <p>Loading launch...</p>
  }

  return (
    <div className='view-checklist'>
      <h2 className={'launch-name'}>{launch.mission_name}</h2>
      <p><strong>Flight Number</strong> {launch.flight_number}</p>
      <p><strong>Launch Site</strong> {launch.launch_site.site_name_long}</p>
    </div>
  )
}

export default Launch

In Home.js: import Launch.js:

import Launch from '../partial-components/space-x/Launch'

And we can use it in our JSX now in place of the comment {/* Launch Component */}:

  return (
    <div className='view-home'>
      <h3>Next Launch</h3>
      <Launch />
    </div>
  )

We have the basics working and we are fetching the next launch, but as we said above, we want to also have the ability to pass a flight prop with a specific flight and have it show.

First let's set some types for our two props that we want to use.

In Launch.js: import PropTypes:

import PropTypes from 'prop-types'

Add props to the component using destructuring.

In Launch.js: replace the following code:

const Launch = () => {

with:

const Launch = ({next = true, flight}) => {

This shows you how you can default a prop to a certain value if it is not passed in by the component using <Launch />, and is just a contrived example to show you how it can be done. In reality I would opt for not having a next prop and simply having flight prop and if not supplied we would just fetch the next launch. But I digress, and challenge you to make the code in this file better.

As well, we need to add our propTypes at the end of the file:

export default Launch

Launch.propTypes = {
  flight: PropTypes.number,
  next: PropTypes.bool
}

Now we can pass data from the Home.js into our <Launch /> Component as props.

We are going to add an alternate fetch function called getSpaceXLaunch which will receive the flight prop:

In Launch.js: add the following code right after the existing getSpaceXLaunchNext function:

// Get Next Launch
const getSpaceXLaunchNext = () => {
  return fetch('https://api.spacexdata.com/v3/launches/next')
    .then((res) => res.json())
}

// Get Launch by Flight
const getSpaceXLaunch = (flight) => {
  return fetch(`https://api.spacexdata.com/v3/launches/${flight}`)
    .then((res) => res.json())
}

And we need to update our useEffect() hook:

  useEffect(() => {
    if (next) {
      getSpaceXLaunchNext()
        .then(setLaunch)
    } else {
      getSpaceXLaunch(flight)
        .then(setLaunch)
    }
  }, [flight, next])

Although we don't have to change our component on the Home.js page, let's be very explicit and do so:

  return (
    <div className='view-home'>
      <h3>Next Launch</h3>
      <Launch next={true} />
    </div>
  )

and remember that we could also use this component somewhere else in our site if we eanted to by passing in a flight number:

  return (
    <div className='view-home'>
      <h3>Next Launch</h3>
      <Launch flight={76} />
    </div>
  )

This concludes the section on adding a Launch component with the Space-X API.

Implement SpaceX Events as Master / Detail Page

One of the things everyone needs to be able to do in a web application is have a predictable URL scheme to access your various pages and their details. In the traditional web application we would post back to the server and/or reload the browser to go from a list of something to a more detailed version of it. But we have more capabilities in a React single page application.

We'll split up the Events.js page to have a list on the left (<EventList />) and detail on the right (<EventDetails />). Once a selection is made from the <EventList /> component, our Events.js component will descern the event that was selected and populate the <EventDetails /> component. The same data used when getting the event list from the API, can be used for our details page ensuring that we only have to make one request for all events in our application. In a real prodcution app it won't be this simple, but this is a great example to help you get started working with fetching data from and API.

When the Events page loads the following things should happen:

  • Request data from the Space-X History endpoint
  • Populate the <EventList /> component with that data
  • The component on the right is aware that the URL lacks an event_id
  • The <EventDetails /> component displays a message: "select an event!"
  • The Events.js component serves as a container for the <EventList /> and <EventDetails /> components
  • The Events.js component supplies data through props to the <EventList /> and <EventDetails /> components
  • If we have an event_id as part of the URL: /event/6, we need to match that to an event and single the data out.
  • This specific event data can then be passed to the <EventDetails /> component
  • Props change when the user clicks an event and an update to the <EventDetails /> component happens
  • If the URL is loaded cold everything should still work (happens for FREE with React Router)

Add the Files For Events

We will add all of our files in the space-x directory, create the following files:

  • EventList.js
  • EventList.scss
  • EventDetails.js
  • EventDetails.scss
  • events-data.js

From The Left-to-Right

We will start by simply getting the EventList and EventDetails pages to load side by side by using an npm package that will give us components for laying out pages using flexbox.

Install simple-flexbox:

npm i simple-flexbox

In Events.js: import simple-flexbox components and EventList and EventDetails components:

import React, { useEffect } from 'react'
import { Column, Row } from 'simple-flexbox'

import EventList from '../partial-components/space-x/EventList'
import EventDetails from '../partial-components/space-x/EventDetails'

We can add the following code that will replace all of the JSX and split the page 50/50. :

  return (
    <div className='view-events'>
      <h3>Historical Events</h3>
      <Row horizontal='spaced'>
        <Column flexGrow={1} style={{ width: '45%', padding: 5 }}>
          <EventList eventData={[{ title: 'Item One', url: '#' }, { title: 'Item Two', url: '#' }]} />
        </Column>
        <Column flexGrow={1} style={{ width: '55%', padding: 5 }}>
          <EventDetails eventData={{ title: 'Awesome Launch' }} />
        </Column>
      </Row>
    </div>
  )

Adding the EventList and EventDetails Components

First we will add code to get our newly added components displaying some dummy data:

EventList.js

import React from 'react'
import PropTypes from 'prop-types'
import { Link } from 'react-router-dom'

import './EventList.scss'

const EventList = (props) => {
  const listItems = props.eventData.map(event => {
    return <li className='event-post' key={event.id + event.title}>
      <Link className='text_link' to={`/events/${event.id}`}>{event.title}</Link>
    </li>
  })

  return (
    <div className={`view-events`}>
      <h3>Historical Events</h3>
      <ul className='event-list'>
        {listItems}
      </ul>
    </div>
  )
}

export default EventList

EventList.propTypes = {
  eventData: PropTypes.array
}

EventList.scss

ul.event-list {
  padding: 0;
  > li {
    list-style: none;
    line-height: 2em;
    font-size: 17px;
  }
}

ul.event-list > li {
  list-style: none;
  line-height: 2em;
  font-size: 17px;
}

EventDetails.js

import React from 'react'
import PropTypes from 'prop-types'

import './EventDetails.scss'

const EventDetails = (props) => {
  const event = props.eventData
  return (event
    ? <>
      <h3>{event.title}</h3>
    </>
    : <h3>.. select and event!</h3>
  )
}

export default EventDetails

EventDetails.propTypes = {
  eventData: PropTypes.object
}

EventDetails.scss

.flight-number {
  font-size: 2.5em;
}
.flight-number.none {
  text-decoration: line-through;
}
.event-details {
  font-size: 18px;
}
ul.event-links {
  padding: 0;
  > li {
    padding-left: 1em;
    list-style: none;
    line-height: 1.75em;
    font-size: 16px;
    border-left: 4px solid #999;
  }
}

Let's run our project and make sure everything is working, the links will throw an error, but we will be replacing the dummy data mementarily:

npm start

We now have enough to get started with fetching the data and each component is already receiving props even though we are feeding it some fake data. Let's now fetch all of the data and try to get the master-detail working with the routing params.

Fetching the SpaceX History Data on the Events Page

We are going to be fetching the data on the Events page and giving each of our components some data to work with. Let's make the calls and fetch the data on the Events.js page:

In Events.js: replace the following code:

const Events = () => {
  useEffect(() => {document.title = 'SpaceX Historical Events'})

With:

const Events = ({match}) => {
  const event_id = match.params.event_id
  const [events, setEvents] = useState([])
  const [isloading, setIsLoading] = useState(false)
  const [error, setError] = useState(null)

  useEffect(() => {
    setIsLoading(true)
    fetch('https://api.spacexdata.com/v3/history')
      .then(res => {
        if(res.ok) {
          return res.json()
        } else {
          throw Error('Fetching Events')
        }
      })
      .then(events => {
        setEvents(events)
        setIsLoading(false)
      })
      .catch(error => setError(error))
  },[])

  useEffect(() => {
    document.title = 'SpaceX Historical Events'
  })

  if (!events && isLoading) {
    return <p>Loading Events Posts...</p>
  } else if (error) {
    return <p>{error.message}</p>
  }

  const event = events.find(event => {
    return event.id === Number(event_id)
  })

Ensure useState has been imported:

import React, { useEffect, useState } from 'react'

OK, let's talk about what is happening here, we are checking for a URL parameter, fetching data and if we do have a URL param, it's for event_id and we are going to single out a specific event and get that ready to pass to the EventDetails component.

With this in place we can pass the two variables events and event to each respective component:

      <Row horizontal='spaced'>
        <Column flexGrow={1} style={{width:'45%', padding: 5}}>
          <EventList eventData={events}/>
        </Column>
        <Column flexGrow={1} style={{width:'55%', padding: 5}}>
          <EventDetails eventData={event} />
        </Column>
      </Row>

We also need to update our Router in the Frame.js page:

<Route exact path={['/events/:event_id', '/events/']} component={Events} />

If we look at our app, we should now have our title populating on the right-hand side when we click on the link from the left.

Extracting our Data Related State and Effects

Now that we are fetching data and doing the basic mapping and filtering passing it to the components, we would like to add some more features and information displayed as part of the master and detail page.

Let's take all of the events, isLoading, error and the useEffect() that fetches our data and create our own hook and in turn extracting this logic out to the events-data.js file that we created:

In events-data.js: add the following code:

import { useEffect, useState } from 'react'

export function useEventData() {

  /* code extractd from `Events.js` goes here! */

  return { events, isloading, error }
}

In Events.js: extract the follwoing code and move i tto our events-data.js:

  const [events, setEvents] = useState([])
  const [isloading, setIsLoading] = useState(false)
  const [error, setError] = useState(null)

  useEffect(() => {
    setIsLoading(true)
    fetch('https://api.spacexdata.com/v3/history')
      .then(res => {
        if(res.ok) {
          return res.json()
        } else {
          throw Error('Fetching Events')
        }
      })
      .then(events => {
        setEvents(events)
        setIsLoading(false)
      })
      .catch(error => setError(error))
  },[])

Replace the code we removed with:

const { events, isLoading, error } = useEventData()

and import the hook:

import { useEventData } from '../partial-components/space-x/events-data'

Let's briefly talk about what we did here. We made the page more readable and extracted all of that logic that dealt with obtaining and exposing the data through React state into one hook that we simply use in our Events.js page now.

Adding More Information to the EventDetails Page

In EventDetails.js: update the entire return statement:

  return (event
    ? <>
      <h3>{event.title}</h3>
      <div className={`flight-number ${!event.flight_number && 'none'}`}>
        {event.flight_number 
          ? <>Flight Number: {event.flight_number}</>
          : <>No Flight Number</>
        }
      </div>
      <p className={'event-details'}>{event.details}</p>
      <ul className={'event-links'}>
        <li>
          <a className={'text_link'} href={event.links.article}>SpaceX Article</a>
        </li>
        <li>
          <a className={'text_link'} href={event.links.wikipedia}>Wiki Article</a>
        </li>
      </ul>
    </>
    : <h3>.. select and event!</h3>
  )

That helps to fill out the page and we even have a class that is being conditionally rendered based on a flight number being available (or not) and our styled links, borders and font sizes that we pasted in as part of the styles for these pages should be working too!

That concludes the section dealing with taking the Event master details page to the next level!

More Resources

These are additional resources not mentioned above, but I think are all pertinent to React today to help you get started and understand state in React!

Great Advice and Insights from React Podcasts

These are my two favorite React Podcasts, some of the episodes are old, but still super relevant. These are some of the episodes that I think will fast track you in ReactJS!

React, State Management, and Hooks

Recoil JS

Learning React JS

About

My workshop repo for React From Scratch updated for Webpack 5 and React 17 for the Fullstack React Friday's Meetup.

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published