/redux-drills

Primary LanguageJavaScript

Redux Drills


You are in charge of building an app that can manage the guest list of DevMountain's next big hackathon. Complete drills 1-4 to build the guest list app.

Drill-1 (Setup)

You will start by installing dependencies and creating a single reducer.

  • npm install
  • Install redux and react-redux.
  • Create a ducks folder in src.
  • Inside of the ducks folder, create a file called guestList.js.
    • Create a reducer. The reducer is just a function that takes in state and an action. For now, have the reducer immediately return state. Export the reducer.

      Solution: src/ducks/guestList.js
      const initialState = {}
      
      export default function reducer(state = initialState, action) {
        return state;
      }
      

Create your store.

  • Create a store.js file in the src folder.
    • In store.js import createStore ( from redux ) and the reducer.

    • Export the invocation of createStore with the reducer as the only argument.

      Solution: src/store.js
      import { createStore } from 'redux';
      import guestlist_reducer from './ducks/guestList';
      export default createStore(guestlist_reducer);
      

In index.js:

  • Import Provider (from react-redux) and the store.

  • In the render method, wrap <App /> with Provider.

  • Pass the store, as a prop, to Provider.

    Solution: src/index.js
    import React from 'react';
    
    import ReactDOM from 'react-dom';
    import { Provider } from 'react-redux';
    import App from './App';
    import store from './store';
    import registerServiceWorker from './registerServiceWorker';
    import './index.css';
    
    ReactDOM.render(
      <Provider store={store}>
        <App />
      </Provider>, document.getElementById('root'));
    registerServiceWorker();
    

Drill-2 (Display guest list)

Building on the work you did in drill-1, you will now connect a component to the store so that you can display the guest list.

  • npm install
  • Install redux and react-redux.
  • In guestList.js:
    • Set initialState to
    const initialState = {
          guests: ['Tony Stark', 'Steve Rodgers', ' Nick Fury', 'Natasha Romanova', 'Clint Barton', 'Bruce Banner', 'Wanda Maximoff']
      }
    

In App.js:

  • Import connect from 'react-redux';
  • Create your mapStateToProps function. Pull the list of guests off of state.
  • mapStateToProps needs to be the first argument when connect is invoked.
  • map over the guest list array that is now on props. The map should return some jsx with the guest's name and a remove button.
    <div key={i} className="list-item">
        <li>{guest}</li>
        <button type="" className="">Remove</button>
    </div>
    
Solution: App.js
import React, { Component } from 'react';
import { connect } from 'react-redux'
import './App.css';

class App extends Component {
  render() {
    return (
      <div className="App">
        <h1>DevMountain Hackathon</h1>
        <h3>Guest List:</h3>
        <ul>
          {this.props.list.map( (guest, i) => {
            return (
              <div key={i} className="list-item">
                <li>{guest}</li>
                <button type="" className="">Remove</button>
              </div>
            )
          })}
        </ul>
        <div className="add-guest">
          Add guest: <input type="" className=""/>
          <button type="" className="">Add</button>
        </div>
      </div>
    );
  }
}

function mapStateToProps(state) {
  return {
    list: state.guests
  };
}

export default connect(mapStateToProps)(App);

Drill-3 (Add/Delete guests)

Your guest list needs to be able to add and remove guests.

  • npm install

  • Install redux and react-redux.

  • Add functionality of adding guest to list. In guestList.js:

    • Export a function called addGuest. This function is an action creator. It should return an object with a type and payload.
    • The addGuest function should have one parameter, which will be a guest name.
    • Set up a switch statement in the reducer function. When adding a guest, we should return a new piece of state that includes the new guest we are adding.
  • In App.js:

    • Import the addGuest function from guestList.js.
    • As the second argument for the connect method, pass in an object with the key and value being addGuest.
    export default connect(mapStateToProps, { addGuest })(App);
    
    • Add the constructor and set up the initial state for the component. You will need to keep track of what is typed into the input box. (Hint: you will need to use the onChange event handler and this.setState())
    • When the add button is clicked, you need to call the addGuest function (on props) and pass in a guest name (the value of the input, which is on App's component state)
  • You should be able to add guests to the list now. Following a similar process as you did add a guest, add the functionality of removing a guest when the Remove button is clicked.

    Solution: App.js
    import React, { Component } from 'react';
    import { addGuest, removeGuest } from './ducks/guestList';
    import { connect } from 'react-redux';
    import './App.css';
    
    class App extends Component {
      constructor() {
        super();
        this.state = {
          text: ''
        }
        this.handleInputChange = this.handleInputChange.bind(this);
        this.handleSubmit = this.handleSubmit.bind(this);
      }
    
      handleInputChange(e) {
        this.setState({
          text: e.target.value
        })
      }
      handleSubmit(e) {
        e.preventDefault();
        this.props.addGuest(this.state.text);
        this.setState({
          text: ''
        })
      }
    
      render() {
        return (
          <div className="App">
            <h1>DevMountain Hackathon</h1>
            <h3>Guest List:</h3>
            <ul>
              {this.props.list.map( (guest, i) => {
                return (
                  <div key={i} className="list-item">
                    <li>{guest}</li>
                    <button onClick={()=> this.props.removeGuest(i)}>Remove</button>
                  </div>
                )
              })}
            </ul>
            <form
              onSubmit={this.handleSubmit}
              className="add-guest">
              Add guest: <input
              value={this.state.text}
              onChange={this.handleInputChange}
              />
              <button>Add</button>
            </form>
          </div>
        );
      }
    }
    
    function mapStateToProps(state) {
      return {
        list: state.guests
      }
    }
    
    export default connect(mapStateToProps,{ addGuest, removeGuest })(App);
    
    Solution: guestList.js
    const ADD_GUEST = 'ADD_GUEST';
    const REMOVE_GUEST = 'REMOVE_GUEST';
    
    const initialState = {
      guests: ['Tony Stark', 'Steve Rodgers', ' Nick Fury', 'Natasha Romanova', 'Clint Barton', 'Bruce Banner', 'Wanda Maximoff']
    };
    
    export function addGuest(guest) {
      return {
        type: ADD_GUEST,
        payload: guest
      }
    }
    
    export function removeGuest(i) {
      return {
        type: REMOVE_GUEST,
        payload: i
      }
    }
    
    export default function reducer(state = initialState, action) {
      switch (action.type) {
        case ADD_GUEST:
          return Object.assign({}, state, {guests: [...state.guests, action.payload]});
        case REMOVE_GUEST:
          return Object.assign({}, state, {guests: state.guests.filter((guest, i) => i !== action.payload)});
        default:
          return state;
        }
    }
    

Drill-4 (Update guest names)

For drill-4, you will have limited help. You have seen the process of building a store and a reducer, connecting a component to the store, dispatching actions, and displaying data from the store. Use this knowledge to complete drill-4. Try to think through what needs to be done and how to do it.

  • npm install

  • Install redux and react-redux.

    1. Create EditGuest.js in src.
    2. The EditGuest component should be a view component (just a function, not a class). Go ahead and set up your component.
    • Import ./EditGuest.css

      EditGuest.js setup
      import React from 'react';
      import './EditGuest.css';
      
      function EditGuest(props) {
          return (
      
          )
      }
      
      export default EditGuest;
      
    1. Paste the following code inside the return:
        <div className="modal-bg">
          <div className="modal">
            <input className="modal-input"/>
            <button className="modal-btn">Update</button>
            <button className="modal-btn">Cancel</button>
          </div>
        </div>
    
    
  1. Add edit: false to the state object in the constructor. Import the EditGuest component to App.js. Use a ternary operator to test whether this.state.edit is true or false. If true, display an instance of the EditGuest component. This code should be in the jsx under the form tags.

    Solution
            {
              this.state.edit ?
                    <EditGuest />
                    : null
            }
    
  2. We now need to add functionality to the edit button. When the edit button is clicked, the modal need to show. Create a method called showModal on the App class. The method should set this.state.edit to true when the edit button is clicked on.

    App.js
    import React, { Component } from 'react';
    import { addGuest, removeGuest } from './ducks/guestList';
    import EditGuest from './EditGuest';
    import { connect } from 'react-redux';
    import './App.css';
    
    class App extends Component {
      constructor() {
        super();
        this.state = {
          text: '',
          edit: false
        }
        this.handleInputChange = this.handleInputChange.bind(this);
        this.handleSubmit = this.handleSubmit.bind(this);
        this.showModal = this.showModal.bind(this);
      }
    
      handleInputChange(e) {
        this.setState({
          text: e.target.value
        })
      }
      handleSubmit(e) {
        e.preventDefault();
        this.props.addGuest(this.state.text);
        this.setState({
          text: ''
        })
      }
    
      showModal() {
        this.setState({
          edit: true
        })
      }
    
      render() {
        return (
          <div className="App">
            <h1>DevMountain Hackathon</h1>
            <h3>Guest List:</h3>
            <ul>
              {
                  this.props.list.map( (guest, i) => {
                    return (
                      <div key={i} className="list-item">
                        <li>{guest}</li>
                        <div>
                          <button
                          onClick={this.showModal}
                          >Edit</button>
                          <button onClick={()=> this.props.removeGuest(i)}>Remove</button>
                        </div> 
                      </div>
                    )
                  })
              }
            </ul>
            <form
              onSubmit={this.handleSubmit}
              className="add-guest">
              Add guest: <input
              value={this.state.text}
              onChange={this.handleInputChange}
              />
              <button>Add</button>
            </form>
            {
              this.state.edit ? 
              <EditGuest />
              : null
            }
            
          </div>
        );
      }
    }
    
    function mapStateToProps(state) {
      return {
        list: state.guests
      }
    }
    
    export default connect(mapStateToProps,{ addGuest, removeGuest })(App);
    
  3. The modal should close if the cancel button is clicked. Create a method called hideModal on the App class that sets this.state.edit to false. Pass this method as a prop to the EditGuest component and use it to add functionality to the cancel button.

    App.js
    import React, { Component } from 'react';
    import { addGuest, removeGuest } from './ducks/guestList';
    import EditGuest from './EditGuest';
    import { connect } from 'react-redux';
    import './App.css';
    
    class App extends Component {
      constructor() {
        super();
        this.state = {
          text: '',
          edit: false
        }
        this.handleInputChange = this.handleInputChange.bind(this);
        this.handleSubmit = this.handleSubmit.bind(this);
        this.showModal = this.showModal.bind(this);
        this.hideModal = this.hideModal.bind(this);
      }
    
      handleInputChange(e) {
        this.setState({
          text: e.target.value
        })
      }
      handleSubmit(e) {
        e.preventDefault();
        this.props.addGuest(this.state.text);
        this.setState({
          text: ''
        })
      }
    
      showModal() {
        this.setState({
          edit: true
        })
      }
    
      hideModal() {
        this.setState({
          edit: false
        })
      }
    
      render() {
        return (
          <div className="App">
            <h1>DevMountain Hackathon</h1>
            <h3>Guest List:</h3>
            <ul>
              {
                  this.props.list.map( (guest, i) => {
                    return (
                      <div key={i} className="list-item">
                        <li>{guest}</li>
                        <div>
                          <button
                          onClick={this.showModal}
                          >Edit</button>
                          <button onClick={()=> this.props.removeGuest(i)}>Remove</button>
                        </div> 
                      </div>
                    )
                  })
              }
            </ul>
            <form
              onSubmit={this.handleSubmit}
              className="add-guest">
              Add guest: <input
              value={this.state.text}
              onChange={this.handleInputChange}
              />
              <button>Add</button>
            </form>
            {
              this.state.edit ? 
              <EditGuest
                hide={this.hideModal} />
              : null
            }
            
          </div>
        );
      }
    }
    
    function mapStateToProps(state) {
      return {
        list: state.guests
      }
    }
    
    export default connect(mapStateToProps,{ addGuest, removeGuest })(App);
    
    EditGuest.js
    import React from 'react';
    import './EditGuest.css';
    
    function EditGuest(props) {
        return (
            <div className="modal-bg">
                <div className="modal">
                    <input className="modal-input"/>
                    <button className="modal-btn">Update</button>
                    <button 
                    onClick={props.hide}
                    className="modal-btn">Cancel</button>
                </div>
            </div>
      )
    }
    
    export default EditGuest;
    
  4. Your modal should now show when you click edit, and hide when you click cancel. We now need to populate the input box on the modal with the name that we want to edit.

    • We will keep track of the name and index of the guest we are editing in App's component state.
    this.state = {
       text: '',
       edit: false,
       guestToEdit: '',
       index: 0
     }
    
    • We need to pass the guest name and index to our showModal method, and we have access to both while we map over this.props.list. When then Edit button is clicked, is should invoke this.showModal and pass in guest and i as arguments.

      Solution
      
      this.props.list.map( (guest, i) => {
                    return (
                      <div key={i} className="list-item">
                        <li>{guest}</li>
                        <div>
                          <button
                          onClick={()=> this.showModal(guest, i)}
                          >Edit</button>
                          <button onClick={()=> this.props.removeGuest(i)}>Remove</button>
                        </div> 
                      </div>
                    )
                  })
      
      
    • Update the showModal method so that it updates guestToEdit and index on state.

      Solution
      
      <input
                value={props.guest} 
                onChange={props.edit}
                className="modal-input"/>
      
      
  5. Pass the guest name (on App state) to the EditGuest component as a prop. Display the guest's name in the modal's input (as value).

  6. When you click the edit button, the modal should appear with the correct guest name displayed in the input. We now need a way to keep track of the changes that we make to the name.

    • Create a method on the App component called editName. This method should update this.state.guestToEdit with the value typed in to the EditGuest component's input.

      • HINT: Don't forget to bind!
      editName
      
      editName(e) {
        this.setState({
          guestToEdit: e.target.value
        })
      }
      
  • Pass the editName method as a prop to the EditGuest component. In EditGuest.js, use the onChange event with the editName method as the event handler.

    Solution
    
    
    {
          this.state.edit ? 
          <EditGuest
            hide={this.hideModal}
            guest={this.state.guestToEdit}
            edit={this.editName} />
          : null
        }  
    
input (EditGuest.js)

 <input
   value={props.guest} 
   onChange={props.edit}
   className="modal-input"/>
  1. We now need to make the update button work. If we are going to update information in our redux store then we need to head over to our guestList.js file.

    • At the top of the file, create a new constant:
    const UPDATE_GUEST = 'UPDATE_GUEST';
    
    • Export a function called updateName with two parameters, name and index.

    • The updateName function should return an object with type and payload properties. The value of type should be UPDATE_GUEST. The value of payload should be an object that contains the values of the name and index parameters.

      guestList.js
      
      export function updateName(name, index) {
        return {
          type: UPDATE_GUEST,
          payload: {
            name: name, 
            index: index
          }
        }
      }
      
    • Update the reducer to handle an action with the type of UPDATE_GUEST. Use the information in action.payload to return a new piece of state with the updated user name.

      guestList.js
      case UPDATE_GUEST:
        return {
          guests: state.guests.map((name, i) => {
            if (action.payload.index === i) return action.payload.name;
            return name;
          })
        } 
      
    • In App.js, import the updateName function. Add it to the object that is passed as the second argument in the connect method.

      Solution
      export default connect(mapStateToProps,{ addGuest, removeGuest, updateName })(App);
      
    • Create a method on the App component called updateGuestName. This method will invoke updateName (action creator) and pass in guestToEdit and index from App's state. This method will also invoke the hideModal method.

      • HINT: Don't forget to bind!

        updateGuestName
        updateGuestName() {
            this.props.updateName(this.state.guestToEdit, this.state.index);
            this.hideModal();
          }
        
    • Pass the updateGuestName method as a prop to the EditGuest component. In EditGuest.js, use the method as an event handler for when the update button gets clicked.

      App.js
        <EditGuest
            hide={this.hideModal}
            guest={this.state.guestToEdit}
            edit={this.editName}
            update={this.updateGuestName} />
      
      EditGuest.js
        <button 
          onClick={props.update}
          className="modal-btn">Update</button>
      

Congrats! You should now have a fully working guest list that can add, remove, and edit guest names.

Final Solution:

App.js
import React, { Component } from 'react';
import { addGuest, removeGuest, updateName } from './ducks/guestList';
import EditGuest from './EditGuest';
import { connect } from 'react-redux';
import './App.css';

class App extends Component {
  constructor() {
    super();
    this.state = {
      text: '',
      edit: false,
      guestToEdit: '',
      index: 0
    }
    this.handleInputChange = this.handleInputChange.bind(this);
    this.handleSubmit = this.handleSubmit.bind(this);
    this.showModal = this.showModal.bind(this);
    this.hideModal = this.hideModal.bind(this);
    this.editName = this.editName.bind(this);
    this.updateGuestName = this.updateGuestName.bind(this)
  }

  handleInputChange(e) {
    this.setState({
      text: e.target.value
    })
  }
  handleSubmit(e) {
    e.preventDefault();
    this.props.addGuest(this.state.text);
    this.setState({
      text: ''
    })
  }

  showModal(guest, index) {
    this.setState({
      edit: true,
      guestToEdit: guest,
      index: index
    })
  }

  hideModal() {
    this.setState({
      edit: false
    })
  }

  editName(e) {
    this.setState({
      guestToEdit: e.target.value
    })
  }

  updateGuestName() {
    this.props.updateName(this.state.guestToEdit, this.state.index)
    this.hideModal()
  }

  render() {
    return (
      <div className="App">
        <h1>DevMountain Hackathon</h1>
        <h3>Guest List:</h3>
        <ul>
          {
              this.props.list.map( (guest, i) => {
                return (
                  <div key={i} className="list-item">
                    <li>{guest}</li>
                    <div>
                      <button
                      onClick={()=>this.showModal( guest, i)}
                      >Edit</button>
                      <button onClick={()=> this.props.removeGuest(i)}>Remove</button>
                    </div> 
                  </div>
                )
              })
          }
        </ul>
        <form
          onSubmit={this.handleSubmit}
          className="add-guest">
          Add guest: <input
          value={this.state.text}
          onChange={this.handleInputChange}
          />
          <button>Add</button>
        </form>
        {
          this.state.edit ? 
          <EditGuest
            hide={this.hideModal}
            guest={this.state.guestToEdit}
            edit={this.editName}
            update={this.updateGuestName} />
          : null
        }
        
      </div>
    );
  }
}

function mapStateToProps(state) {
  return {
    list: state.guests
  }
}

export default connect(mapStateToProps,{ addGuest, removeGuest, updateName })(App);


EditGuest.js
import React from 'react';
import './EditGuest.css';

function EditGuest(props) {
    return (
        <div className="modal-bg">
            <div className="modal">
                <input
                value={props.guest} 
                onChange={props.edit}
                className="modal-input"/>
                <button 
                onClick={props.update}
                className="modal-btn">Update</button>
                <button 
                onClick={props.hide}
                className="modal-btn">Cancel</button>
            </div>
        </div>
  )
}

export default EditGuest;

guestList.js
const ADD_GUEST = 'ADD_GUEST';
const REMOVE_GUEST = 'REMOVE_GUEST';
const UPDATE_GUEST = 'UPDATE_GUEST';

const initialState = {
  guests: ['Tony Stark', 'Steve Rodgers', ' Nick Fury', 'Natasha Romanova', 'Clint Barton', 'Bruce Banner', 'Wanda Maximoff']
};

export function addGuest(guest) {
  return {
    type: ADD_GUEST,
    payload: guest
  }
}

export function removeGuest(i) {
  return {
    type: REMOVE_GUEST,
    payload: i
  }
}

export function updateName(name, index) {
  return {
    type: UPDATE_GUEST,
    payload: {
      name: name, 
      index: index
    }
  }
}

export default function reducer(state = initialState, action) {
  switch (action.type) {
    case ADD_GUEST:
      return Object.assign({}, state, {guests: [...state.guests, action.payload]});
    case REMOVE_GUEST:
      return Object.assign({}, state, {guests: state.guests.filter((guest, i) => i !== action.payload)});
    case UPDATE_GUEST:
      return {
        guests: state.guests.map((name, i) => {
          if (action.payload.index === i) return action.payload.name;
          return name;
        })
      } 
    default:
      return state;
    }
}