DEV Community

Madhav Pandey
Madhav Pandey

Posted on • Edited on

React-Redux Essentials: Navigating the Basics of State Management

blog-image

Introduction

As per official redux website, redux is a pattern and library for managing and updating application state, using events called actions. Redux serves as a centralized store for state that needs to be used across entire application.
In this blog post we will learn redux in react by building a Counter App.

Installation

To install redux in your react app run following command.

npm install @reduxjs/toolkit react-redux

Create redux store

Inside your src folder create a folder named app and inside this folder create a file named store.js. The redux store holds all state of the app in immutable object tree. Now in store.js file write a following code

import {configureStore} from "@reduxjs/toolkit";

 const store = configureStore({
reducer:{

}

 })

export default store;

Enter fullscreen mode Exit fullscreen mode

Here, we have created a store which is holding empty reducer for now and exported it.
Go to our index.js file setup our redux store

import App from './App.jsx'
import store from './app/store.jsx';
import { Provider } from 'react-redux';
ReactDOM.createRoot(document.getElementById('root')).render(
  <React.StrictMode>

<Provider store={store}>

    <App />

</Provider>
  </React.StrictMode>,
)

Enter fullscreen mode Exit fullscreen mode

We have imported store and Provider and we wrapped our <App/> component inside the provider which holds store. By doing this our store will become global state.

Creating Slice Reducers and Actions

Let's create a folder features/counter inside app where we will have counter slice. A slice is a collection of Redux reducer login and actions for a single feature in your app.
Now, create a file counterSlice.js inside counter folder.

import { createSlice } from "@reduxjs/toolkit";


const initialState = {
    count :0
}


export const counterSlice = createSlice({
name:"counter",
initialState,
reducers:{
    increment :(state)=>{
        state.count+=1;
    },
    decrement :(state)=>{
        state.count-=1;
    },
    incrementByAmount: (state, action)=>{
        state.count+=action.payload;
    }, 
    decrementByAmount:(state, action)=>{
        state.count-= action.payload
    }
}


})

export const {increment, decrement, incrementByAmount, decrementByAmount}= counterSlice.actions;
export default counterSlice.reducer;
Enter fullscreen mode Exit fullscreen mode

createSlice is a function that accept slice name, an initial state, object of reducers function. We have created and exported counterSlice plus we have exported increment, decrement, incrementByAmount and decrementByAmount actions from counterSlice.actions and all reducer from counterSlice.reducer.

We have created a counter slice, it's time to import counterReducer inside our store.js file where we had empty reducer.

import { configureStore } from "@reduxjs/toolkit";
import counterReducer from "./features/counter/counterSlice"
const store = configureStore({
reducer:{
counter:counterReducer
}


});
export default store;
Enter fullscreen mode Exit fullscreen mode

The React Counter Component

Inside features/counter folder create a file name Counter.jsx. The Counter.jsx file will have following code.

import { Container, Button, Card, InputGroup, Form } from "react-bootstrap";
import { useState } from "react";
import {useSelector, useDispatch} from "react-redux";
import { increment, incrementByAmount, decrement, decrementByAmount } from "./counterSlice";

const Counter = ()=>{
const [addAmount, setAddAmount] = useState("");

const count = useSelector((state)=> state.counter.count);

const dispatch = useDispatch();


const amountHandeler = (e)=>{
    if(!Number(e.target.value)){
        return setAddAmount("");
    }
    setAddAmount(e.target.value);
}

    return<>
    <Container className="d-flex justify-content-center m-2 ">
    <Card className="text-center w-50">
      <Card.Header>Counter App</Card.Header>
      <Card.Body>
        <Card.Title>{count}</Card.Title>
       <div style={{display:"flex", gap:"1rem", justifyContent:"center", alignItems:"center"}}>
        <Button variant="primary" onClick={()=>dispatch(increment())}>+</Button>
        <Button variant="secondary" onClick={()=>dispatch(decrement())}>-</Button>
       </div>

       <InputGroup className="m-2">
        <Form.Control
        onChange={amountHandeler}
         value={addAmount}
        />
        <Button variant="primary" id="button-addon2" onClick={()=>dispatch(incrementByAmount(Number(addAmount)||0))}>
        Add Amount
        </Button>
        <Button variant="secondary" id="button-addon2" onClick={()=> dispatch(decrementByAmount(Number(addAmount)||0))}>
        Subtract Amount
        </Button>
      </InputGroup>



      </Card.Body>

    </Card>

    </Container>
    </>
}

export default Counter;
Enter fullscreen mode Exit fullscreen mode

preview

Here, we have imported useSelector and useDispatch from react-redux plus we have imported all actions from counterSlice.js file. We accessed the current count value with the help of useSelector. We have dispatched action with the help of useDispatch hook.

const count = useSelector((state)=> state.counter.count);
const dispatch = useDispatch();

<Button variant="primary" onClick={()=>dispatch(increment())}>+</Button>
        <Button variant="secondary" onClick={()=>dispatch(decrement())}>-</Button>

Enter fullscreen mode Exit fullscreen mode

Conclusion

We have learned a basic about react-redux and redux toolkit. I would like to mentioned some key notes from official redux website at the end.

  • configureStore accepts a reducer function as a named argument
  • configureStore automatically sets up the store with good default settings
  • A "slice" contains the reducer logic and actions related to a specific feature / section of the Redux state
  • Redux Toolkit's createSlice API generates action creators and action types for each individual reducer function you provide
  • Call dispatch(someActionCreator()) in a component to dispatch an action
  • Wrapping the app with <Provider store={store}> enables all components to use the store -Global state should go in the Redux store, local state should stay in React components

Top comments (0)