Skip to content

Latest commit

 

History

History
306 lines (230 loc) · 10 KB

kata6.md

File metadata and controls

306 lines (230 loc) · 10 KB

React App Kata 6

Code for Kata 6 is available in the app6 folder.

Learning aims

The idea here is to start using Redux for our app and understand it's main concepts.

Requirements

Get started

You will need 2 terminals

  1. Web API server

    • go to ./app6
    • verify dotnet version dotnet --version is higher than 2.0.0
    • run dotnet restore
    • run dotnet build
    • run dotnet run

    This should build the web api server and serve it at http:\\localhost:5000

  2. Web app

  • in another terminal
  • go to ./app6/app/
  • follow the instructions in the README.
  • your app should be running at port 3000

Concepts

Let's go through some basic concepts before we start. We have added a simple end to end React Redux example in the code of App6 that we will explain first.

  1. Actions

    Actions are sets of information that are sent to your application. An action has a type (mandatory) and a payload (optional).

    Example of an action with no payload fired:

    dispatch({ type: WEB_SERVER_VERSION_REQUESTED })

    Example of an action with payload fired:

    dispatch({
        type: WEB_SERVER_VERSION_COMPLETED,
        payload: { version: json }
    })
  2. Reducers

    Reducers define how the state of our app should change with respect to actions that are fired.

    All of our reducers are combined in /src/modules/index.js

    import { combineReducers } from 'redux'
    import { routerReducer } from 'react-router-redux'
    
    import versions from './versions'
    
    export default combineReducers({
        routing: routerReducer,
        versions
    })

    Our versions reducer shows examples of how the state of our app changed based on the actions that are fired:

    /src/modules/versions.js

    export const WEB_SERVER_VERSION_REQUESTED = 'versions/WEB_SERVER_VERSION_REQUESTED'
    export const WEB_SERVER_VERSION_COMPLETED = 'versions/WEB_SERVER_VERSION_COMPLETED'
    
    const initialState = {
        inProgress: false,
        version: null,
    }
    
    export default (state = initialState, action) => {
        switch (action.type) {
            case WEB_SERVER_VERSION_REQUESTED:
            return {
                ...state,
                inProgress: true
            }
            case WEB_SERVER_VERSION_COMPLETED:
            return {
                ...state,
                inProgress: false,
                version: action.payload.version
            }
            default:
            return state
        }
    }

    Note: Everytime you return an object in the case statement of the reducer you are defining the new state of that section of the application.

    This means that when you update the state of any reducer you need to do it without changing the exisitng state. I.e. use a new object instead of an the existing one.

    That is the reason why we use the spread operator when returning the new state in each case.

    case WEB_SERVER_VERSION_REQUESTED:
      return {
        ...state,
        inProgress: true
      }
  3. Store

    The store contains all the state of your application.

    Ours is defined at /src/store.js

    const initialState = {}
    
    const store = createStore(
        rootReducer,
        initialState
    )
    
    export default store

    It already uses a rootReducer where our app defines the reducers it needs under /src/modules/index.js

    import rootReducer from './modules'
  4. give access to the store to all components

    To do this we inject in our src/index.js the store using the Provider component. For more information see: Passing the Store

    import React from 'react'
    import ReactDOM from 'react-dom'
    import { Provider } from 'react-redux'
    import { ConnectedRouter } from 'react-router-redux'
    import store, { history } from './store'
    
    import './index.css'
    import App from './App'
    
    ReactDOM.render(
        <Provider store={store}>
            <ConnectedRouter history={history}>
                <div>
                    <App />
                </div>
            </ConnectedRouter>
        </Provider>,
        document.getElementById('root')
    )
  5. create a dispatch function for fetching versions

    • we define a fetchWebServerVersion function in /src/modules/versions.js

      export const fetchWebServerVersion = () => {
          return dispatch => {
              dispatch({ type: WEB_SERVER_VERSION_REQUESTED })
              const url = '/api/versions/get'
              fetch(url, {
                  method: 'GET',
                  headers: {
                      'Accept': 'application/json',
                      'Content-Type': 'application/json'
                  },
                  credentials: 'same-origin'
              }).then(response => {
                  return response.json()
              }).then(json => {
                  dispatch({
                      type: WEB_SERVER_VERSION_COMPLETED,
                      payload: { version: json }
                  })
              })
          }
      }

      Things to notice here:

      • fetchWebServerVersion function returns a dispatch function
      • we dispatch WEB_SERVER_VERSION_REQUESTED action before the fetch call
      • we dispatch WEB_SERVER_VERSION_COMPLETED action when the fetch call completes
  6. Connect your dispatch function and the data present in the store to the App component

    • we import fetch function and connect state to props

      import { bindActionCreators } from 'redux'
      import { connect } from 'react-redux'
      import { withRouter } from 'react-router'
      
      import { fetchWebServerVersion } from './modules/versions'
      const mapStateToProps = state => ({
          version: state.versions.version
      })
      
      const mapDispatchToProps = dispatch => bindActionCreators({
          fetchWebServerVersion
      }, dispatch)
      
      export default withRouter(connect(mapStateToProps, mapDispatchToProps)(App))

      Remark 1: Notice the line version: state.versions.version where we connect the data from the store as a property of the component so that it can be used as this.props.version directly.

      Remark 2: Notice the mapDispatchToProps, where we add the fetchWebServerVersion function. So that we can call this.props.fetchWebServerVersion and that all calls with dispatch for example dispatch({ type: WEB_SERVER_VERSION_REQUESTED }) get properly fired.

    • use the version data present in the store in our App component

      <div className='App-header'>
          <h2>Kata 6 - Redux</h2>
          <pre>v{this.props.version}</pre>
      </div>
  7. Check redux actions in the app

    The store we created in src/store.js has a logger as one of our middleware. This will allow us to check the actions that are fired within the app.

    import logger from 'redux-logger'
    
    const middleware = [
        thunk,
        logger,
        routerMiddleware(history)
    ]
    
    const composedEnhancers = compose(
        applyMiddleware(...middleware),
        ...enhancers
    )
    
    const store = createStore(
        rootReducer,
        initialState,
        composedEnhancers
    )

    If you check you browser console you should see the actions and the change of each state in the store.

    Redux actions in the console

  8. The versions example

    Throughout the Redux concepts we showed how the versions request example work in this app. Have a look at the different pieces to understand how they work:

    • src/index.js
    • src/App.js
    • src/store.js
    • src/modules/versions.js
    • src/modules/index.js

Task

Redux is alredy installed and working in this application. Your goal is to switch the app code into using Redux.

Write the JavaScript/React code to avoid any use of state in the App.js component. Use redux instead

  1. move all api calls into the products reducer:
    1. under src/modules create file called products.js and export an empty reducer. Connect it to the store.
    2. create the action types for products
      1. products/PRODUCTS_REQUESTED
      2. products/PRODUCTS_COMPLETED
      3. products/PRODUCT_ADD_REQUESTED
      4. products/PRODUCT_ADD_COMPLETED
      5. products/PRODUCT_REMOVE_REQUESTED
      6. products/PRODUCT_REMOVE_COMPLETED
    3. create and implement all api functions that currently exist in App.js
      1. fetchProducts
      2. addProduct
      3. removeProduct (passing newProduct as an argument)
    4. be sure each function dispatches a requested action before calling the server, and a completed action if the request succeeds
  2. update the products reducer so that the data of the products list is updated acordingly:
    1. create and implement a case for each action type (requested and completed). Set the products returned from server (if applicable)
  3. import all dispatch product functions into App.js
    1. be sure to map them to props in the mapDispatchToProps section
  4. use the products data from the store
    1. be sure to map the products data in the mapStateToProps section
  5. update your App.js code so that you don't use this.state anymore. You shoul be using this.props instead
  6. check in the browser console that all actions are being fired

Resources