Skip to content

React-JS with [learnwithfair, Learn with fair, Rahatul Rabbi, Md Rahatul Rabbi ,rahatulrabbi]

Notifications You must be signed in to change notification settings

learnwithfair/react-project-template

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

7 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

REACT-JS TEMPLATE

Thanks for visiting my GitHub account!

React is a free and open-source front-end JavaScript library for building user interfaces based on components. It is maintained by Meta and a community of individual developers and companies. React can be used to develop single-page, mobile, or server-rendered applications with frameworks like Next.js

Source Code (Download)

Click Here

Required Software (Download)

========== Environment Setup ==========

  1. Instal Node.js
  2. To verify installation into command form by node -v or node --version

How to use the template?

  • fork
  • clone to your local machine
  • npm install
  • npm start
  • Or, create and run react app with npx
// create react app command
npx create-react-app appName

// If any error occurs
npm install npm -g

// run react app command
cd appName
npm start

Template Includes

  • react-icons
  • Eslint
  • Prettier

React Documentation

Table of Contents

  1. Basic React.js Topics

    1.1 Introduction to React

    1.2 JSX and JS Expression

    1.3 Component

    1.4 Adding CSS Styling

    1.5 Props and destructuring

    1.6 Mapping components

    1.7 Proptotypes

    1.8 Conditional rendering

    1.9 developer tools, react and font-awesome icons

    1.10 Adding Interactivity - event & event handler

    1.11 useState Hooks

    1.12 Forms and Controlled components

    1.13 data passing: child to parent component, state lifting

    1.14 useRef hook - Uncontrolled component

  2. Intermediate React.js Topics

    2.1 life cycle methods of a class component

    2.2 useEffect Hook

    2.3 useReducer Hook

    2.4 Routing

    2.5 CRUD Operations - http methods - user management app

    2.6 Optimization memo, useCallback and useMemo

    2.7 props drilling, useContext Hook

  3. Advanced React.js Topics

  4. Assignments

    Assignment 1: product listing App

    Assignment 2: Counter App

    Assignment 3: Add New Product

    Assignment 4: fetch products

    Assignment-5

1. Basic React.js Topics

What is React?

React is an open-source JavaScript library for building user interfaces. It was developed in 2013 and is maintained by Facebook. React is known for its performance, flexibility, and component-based architecture, which allows developers to build reusable UI components and efficiently manage the state of an application.

Why React / Features of React.js

React

Here are some key features and concepts associated with React:

  1. Component-Based: React applications are built using components, which are self-contained, reusable building blocks for user interfaces. Components encapsulate both the UI and the logic associated with it. It helps us to create reusable components (small and isloated pieces of code using html, css, js). It helps us to render UI. Think about youtube's website

    • Single Page Application (SPA) allows us to render as much as we need for reducing unnecessary rendering such as loading navbar, footer etc. in all the pages
    • think about html tag and creating your own tag with react
  2. Virtual DOM: React uses a virtual representation of the DOM (Document Object Model) to optimize updates. Instead of directly manipulating the actual DOM, React compares changes in the virtual DOM and efficiently updates only the necessary parts of the real DOM, reducing rendering time and improving performance.

    • Load fast - Why React.js is faster? - virtual DOM compares with previous states
  3. Declarative Syntax: React uses a declarative approach to building UIs. Developers describe what the UI should look like based on the application's state, and React takes care of updating the DOM to match that state.

  4. Unidirectional Data Flow: React enforces a one-way data flow, which means data flows down the component hierarchy from parent components to child components. This helps maintain predictable and debuggable code.

  5. JSX (JavaScript XML): React allows you to write UI components using JSX, which is a syntax extension for JavaScript. JSX allows you to write HTML-like code within your JavaScript files, making it easier to define UI elements.

  6. Component Lifecycle: React components have lifecycle methods that allow developers to hook into specific points in a component's lifecycle, such as when it's mounted or updated. This is useful for performing actions like data fetching, initialization, or cleanup.

  7. State Management: React provides a mechanism for managing component-specific state using the useState hook for functional components and setState for class components. For global state management, libraries like Redux or React Context can be used.

  8. Routing: React can be used in combination with routing libraries like React Router to create single-page applications with client-side routing.

  9. Community and Ecosystem: React has a large and active community, which has led to the development of a rich ecosystem of third-party libraries and tools to enhance development and improve productivity.

  10. Example of React app and competitors - facebook, twitter, airbnb, netflix etc. competitor: Vue.js, Angular (more full-fledged / developed no need 3rd party library just like react-router-dom)

Overall, React is a powerful and popular choice for building modern web applications, and it's widely adopted by developers and organizations for its efficiency and developer-friendly approach to UI development.

  • VSCode (code editor)
  • node.js (Download LTS: Long Term Support one) (npm is included in node.js by default)
  • React Developer tools extention for google, firefox, edge
  • Extension: ES7 react, JS JSX snippets + “editior.snipperSuggestions”: “top”, react developer tools, material theme, setup eslint and prettier

Method 1: add React to a website with CDN

  • Code Example - 1 (create React app)

    <!-- without JSX -->
    <!DOCTYPE html>
    <html lang="en">
      <head>
        <meta charset="UTF-8" />
        <meta http-equiv="X-UA-Compatible" content="IE=edge" />
        <meta name="viewport" content="width=device-width, initial-scale=1.0" />
        <title>Document</title>
        <script
          crossorigin
          src="https://unpkg.com/react@18/umd/react.development.js"
        ></script>
        <script
          crossorigin
          src="https://unpkg.com/react-dom@18/umd/react-dom.development.js"
        ></script>
    
        <style>
          .title {
            background-color: brown;
            color: white;
            text-align: center;
          }
        </style>
      </head>
      <body>
        <div id="root"></div>
    
        <script>
          const headingElement = React.createElement(
            "h1",
            { id: "heading", className: "title" },
            "Welcome to React"
          );
          const root = ReactDOM.createRoot(document.getElementById("root"));
          root.render(headingElement);
        </script>
      </body>
    </html>
    • Code Example - 2 (create React app)

      <!-- With JSX -->
      <!DOCTYPE html>
      <html lang="en">
        <head>
          <meta charset="UTF-8" />
          <meta http-equiv="X-UA-Compatible" content="IE=edge" />
          <meta name="viewport" content="width=device-width, initial-scale=1.0" />
          <title>Document</title>
      
          <!-- adding React  -->
          <script
            crossorigin
            src="https://unpkg.com/react@18/umd/react.development.js"
          ></script>
      
          <!-- adding ReactDOM  -->
          <script
            crossorigin
            src="https://unpkg.com/react-dom@18/umd/react-dom.development.js"
          ></script>
      
          <!-- add the babel and allow jsx  -->
          <script src="https://unpkg.com/babel-standalone@6/babel.min.js"></script>
      
          <style>
            .title {
              background-color: brown;
              color: white;
              text-align: center;
            }
          </style>
        </head>
        <body>
          <div id="root"></div>
      
          <script type="text/babel">
            const root = ReactDOM.createRoot(document.getElementById("root"));
            root.render(
              <h1 id="heading" className="title">
                Welcome to React
              </h1>
            );
          </script>
        </body>
      </html>

Method 2: create and run react app with npx

// create react app command
npx create-react-app appName

// If any error occurs
npm install npm -g

// run react app command
cd appName
npm start

Method 3: create react app with bundler with vite npm create vite@latest swc(speedy web compiler)

Understand File structure

  • discuss about package.json, node_modules, public, src
  • package-lock.json is for version control for packages. it keeps the record of node_modules tree so that when you clone and use npm i it will install exactly same versions for the packages even though their is a new versions. if you do not have package-lock.json then it will install from package.json
  • keep only the index.js in src and then play with React.js
  • change the title of the app inside index.html file

JSX: stands for JavsScript XML which allows us to use write html-like syntax inside javascript and vice versa. react module has babbel inside of it that helps us to run jsx. JSX is similar like HTML but it is more dynamic. JSX and React are independent things and they can be used independently.

Rules for JSX

  1. We can return single element using JSX, for multiple elements we can use a wrapper. We can also use Fragment here.

    • why we can not return multiple elements in JSX needs to be rendered? jsx is javascript object and we can return 2 objetcs from a function so we need to use array syntax and wrap everything inside one array

    • Code Example - 3 (render single element)

      import React from "react";
      import ReactDOM from "react-dom/client";
      
      const root = ReactDOM.createRoot(document.getElementById("root"));
      root.render(<h2>Anis Ecommerce website</h2>);
    • React render function can render only one element

    • Code Example - 4 (Rendering multiple elements)

      import React from "react";
      import ReactDOM from "react-dom/client";
      
      const root = ReactDOM.createRoot(document.getElementById("root"));
      
      root.render(
         <div>
          <header><h2>Anis Express</h2></header>
          <main>
          <aside>this is sidebar</aside>
            <section>
                <article>
                  <img
                    src="https://i.dummyjson.com/data/products/1/thumbnail.jpg"
                    alt="iPhone 9"
                  />
                  <h2>iPhone 9</h2>
                  <p>description: An apple mobile which is nothing like apple</p>
                  <p>Price: $549</p>
                  <p>rating: 4.69</p>
                  <p>brand: Apple</p>
                  <p>category: smartphones</p>
                <article>
      
                <article>
                  <img
                    src="https://i.dummyjson.com/data/products/2/thumbnail.jpg"
                    alt="iPhone X"
                  />
                  <h2>iPhone X</h2>
                  <p>description: SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...</p>
                  <p>Price: $899</p>
                  <p>rating: 4.44</p>
                  <p>brand: Apple</p>
                  <p>category: smartphones</p>
                <article>
            </section>
          </main>
          <footer>this is a footer</footer>
      </div>
    • React.Fragment or <> </> helps us to avoid div soup or unncessary div nesting

    • Code Example - 5 (Fragment)

      <>
        <img
          src="https://i.dummyjson.com/data/products/1/thumbnail.jpg"
          alt="iPhone 9"
        />
        <h2>iPhone 9</h2>
        <p>description: An apple mobile which is nothing like apple",</p>
        <p>Price: $549</p>
        <p>rating: 4.69</p>
        <p>category: smartphones</p>
        );
      <>
  2. Remember to close all tags - <img />

  3. Camelcase - className

  • We can use Javascript expression inside JSX

  • Code Example - 6 (JS Expression in JSX)

    import React from "react";
    import ReactDOM from "react-dom/client";
    
    const imageSource1 = "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
    const title1 = "iPhone 9";
    const description1 = "An apple mobile which is nothing like apple";
    const price1 = 549;
    const rating1 = 4.69;
    const brand1 = "Apple";
    const category1 = "smartphones";
    
    const imageSource2 = "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
    const title2 = "iPhone X";
    const description2 =
      "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
    const price2 = 899;
    const rating2 = 4.44;
    const brand2 = "Apple";
    const category2 = "smartphones";
    
    const root = ReactDOM.createRoot(document.getElementById("root"));
    
    root.render(
      <div>
        <header>
          <h2>Anis Express</h2>
        </header>
    
        <main>
          <aside>this is sidebar</aside>
          <section>
            <article>
              <img src={imageSource1} alt="iPhone 9" />
              <h2>{title1}</h2>
              <p>description: {description1}</p>
              <p>Price: {price1}</p>
              <p>rating: {rating1}</p>
              <p>brand: {brand1}</p>
              <p>category: {category1}</p>
            </article>
            <article>
              <img src={imageSource2} alt="iPhone 9" />
              <h2>{title2}</h2>
              <p>description: {description2}</p>
              <p>Price: {price2}</p>
              <p>rating: {rating2}</p>
              <p>brand: {brand2}</p>
              <p>category: {category2}</p>
            </article>
          </section>
        </main>
        <footer>this is a footer</footer>
      </div>
    );
  • Code Example - 7 (React under the hood)
const Message = () => {
  // return <h3>Message </h3>;
  return React.createElement("h1", {}, "welcome");
};

const Todo = () => {
  return (
    // <article>
    //   <h2>Check students attendance</h2>
    //   <p>
    //     Lorem ipsum dolor sit amet consectetur adipisicing elit. Ab, molestiae.
    //   </p>
    // </article>

    React.createElement(
      "article",
      {},
      React.createElement("h2", {}, "Check students attendance"),
      React.createElement(
        "p",
        {},
        " Lorem ipsum dolor sit amet consectetur adipisicing elit. Ab, molestiae."
      )
    )
  );
};
  • Component: A reusable, nestable building block constrcut with mainly Javascript function and HTML; CSS can be added

    • Component VS Function: Component should always start with capital letter and return JSX
    • There are 2 main types of components: functional component and class component
    • keep a blank line when importing your components for separting built in modules
  • Code Example - 8 (Create a reusable functional component)

    // reusable component
    import React from "react";
    import ReactDOM from "react-dom/client";
    
    const Products = () => {
      const imageSource1 =
        "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
      const title1 = "iPhone 9";
      const description1 = "An apple mobile which is nothing like apple";
      const price1 = 549;
      const rating1 = 4.6;
      const brand1 = "Apple";
      const category1 = "smartphones";
    
      const imageSource2 =
        "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
      const title2 = "iPhone X";
      const description2 =
        "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
      const price2 = 899;
      const rating2 = 4.44;
      const brand2 = "Apple";
      const category2 = "smartphones";
    
      return (
        <section>
          <article>
            <img src={imageSource1} alt="iPhone 9" />
            <h2>{title1}</h2>
            <p>description: {description1}</p>
            <p>Price: {price1}</p>
            <p>rating: {rating1}</p>
            <p>brand: {brand1}</p>
            <p>category: {category1}</p>
          </article>
          <article>
            <img src={imageSource2} alt="iPhone X" />
            <h2>{title2}</h2>
            <p>description: {description2}</p>
            <p>Price: {price2}</p>
            <p>rating: {rating2}</p>
            <p>brand: {brand2}</p>
            <p>category: {category2}</p>
          </article>
        </section>
      );
    };
    
    const root = ReactDOM.createRoot(document.getElementById("root"));
    root.render(
      <div>
        <header>
          <h2>Anis Express</h2>
        </header>
        <main>
          <aside>this is sidebar</aside>
          <Products />
          <Products />
        </main>
        <footer>this is a footer</footer>
      </div>
    );
  • Code Example - 9 (create Header, Sidebar, Footer component)

    const Header = () => {
      return (
        <header>
          <h2>Anis Express</h2>
        </header>
      );
    };
    const Sidebar = () => {
      return <aside>this is sidebar</aside>;
    };
    const Footer = () => {
      return <footer>this is a footer</footer>;
    };
  • Code Example - 10 (App root component, Nested components)

    import React from "react";
    import ReactDOM from "react-dom/client";
    
    const Products = () => {
      const imageSource1 = "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
      const title1 = "iPhone 9";
      const description1 = "An apple mobile which is nothing like apple";
      const price1 = 549;
      const rating1 = 4.6;
      const brand1 = "Apple";
      const category1 = "smartphones";
    
      const imageSource2 = "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
      const title2 = "iPhone X";
      const description2 =
        "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
      const price2 = 899;
      const rating2 = 4.44;
      const brand2 = "Apple";
      const category2 = "smartphones";
    
      return (
        <section>
          <article>
            <img src={imageSource1} alt="iPhone 9" />
            <h2>{title1}</h2>
            <p>description: {description1}</p>
            <p>Price: {price1}</p>
            <p>rating: {rating1}</p>
            <p>brand: {brand1}</p>
            <p>category: {category1}</p>
          </article>
          <article>
            <img src={imageSource2} alt="iPhone X" />
            <h2>{title2}</h2>
            <p>description: {description2}</p>
            <p>Price: {price2}</p>
            <p>rating: {rating2}</p>
            <p>brand: {brand2}</p>
            <p>category: {category2}</p>
          </article>
        </section>
      );
    };
    
    const Header = () => {
      return (
        <header>
          <h2>Anis Express</h2>
        </header>
      );
    };
    const Sidebar = () => {
      return <aside>this is sidebar</aside>;
    };
    const Footer = () => {
      return <footer>this is a footer</footer>;
    };
    
    const App = () => {
      return (
        <div>
          <Header />
          <main>
            <Sidebar />
            <Products />
          </main>
          <Footer />
        </div>
      );
    };
    
    const root = ReactDOM.createRoot(document.getElementById("root"));
    root.render(<App />);
  • Code Example - 11 (export, import modular component)

    • export default can be used once in a file where multiple exports can be used
    • importing export default does not require {} and you can name anything on the other hand only exports requires {} and naming is strict
    • export App component
    // App.js
    
    import React from "react";
    import ReactDOM from "react-dom/client";
    
    const Products = () => {
      const imageSource1 =
        "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
      const title1 = "iPhone 9";
      const description1 = "An apple mobile which is nothing like apple";
      const price1 = 549;
      const rating1 = 4.6;
      const brand1 = "Apple";
      const category1 = "smartphones";
    
      const imageSource2 =
        "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
      const title2 = "iPhone X";
      const description2 =
        "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
      const price2 = 899;
      const rating2 = 4.44;
      const brand2 = "Apple";
      const category2 = "smartphones";
    
      return (
        <section>
          <article>
            <img src={imageSource1} alt="iPhone 9" />
            <h2>{title1}</h2>
            <p>description: {description1}</p>
            <p>Price: {price1}</p>
            <p>rating: {rating1}</p>
            <p>brand: {brand1}</p>
            <p>category: {category1}</p>
          </article>
          <article>
            <img src={imageSource2} alt="iPhone X" />
            <h2>{title2}</h2>
            <p>description: {description2}</p>
            <p>Price: {price2}</p>
            <p>rating: {rating2}</p>
            <p>brand: {brand2}</p>
            <p>category: {category2}</p>
          </article>
        </section>
      );
    };
    
    const Header = () => {
      return (
        <header>
          <h2>Anis Express</h2>
        </header>
      );
    };
    const Sidebar = () => {
      return <aside>this is sidebar</aside>;
    };
    const Footer = () => {
      return <footer>this is a footer</footer>;
    };
    const App = () => {
      return (
        <div>
          <Header />
          <main>
            <Sidebar />
            <Products />
          </main>
          <Footer />
        </div>
      );
    };
    
    export default App;
    
    // import in index.js
    import React from "react";
    import ReactDOM from "react-dom/client";
    import App from "./App";
    
    const root = ReactDOM.createRoot(document.getElementById("root"));
    root.render(<App />);
  • More practice for export,import

    • Code Example - 13 (More export, import)
    • separate all the components

Inline styling, CSS Stylesheet, CSS module [why you should use css module], third party packages such as Material UI, styled components

  • Inline styling

      const headingStyle = { color: 'red', fontSize: '3rem' };
      <aside style={headingStyle}>
        sidebar goes here
      </aside>
  • CSS Stylesheet

    • Code Example - 14 (Styling component with CSS)

      /*code for the App.css */
      /* reset code and common starts here  */
      :root {
        --primary-color: rgb(11, 181, 48);
        --secondary-color: rgba(36, 122, 55, 0.9);
        --padding: 0.5rem;
        --transition: all 0.3s;
        --border-radius: 0.6rem;
        --box-shadow: 0.1rem 0.2rem 0.8rem rgba(205, 202, 202, 0.5);
      }
      * {
        box-sizing: border-box;
        margin: 0;
        padding: 0;
        text-decoration: none;
        list-style-type: none;
        outline: none;
      }
      html {
        scroll-behavior: smooth;
      }
      .flex-space-around {
        display: flex;
        justify-content: space-around;
        align-items: center;
      }
      .flex-center {
        display: flex;
        justify-content: center;
        align-items: center;
      }
      /* reset code and common ends here  */
      
      /* header starts here  */
      .header {
        height: 10vh;
        background-color: var(--primary-color);
        color: white;
      }
      /* header ends here  */
      
      /* main starts here  */
      main {
        height: 80vh;
      }
      .sidebar {
        flex: 1;
        padding: var(--padding);
        display: flex;
        flex-direction: column;
        justify-content: center;
        gap: 1rem;
        align-self: flex-start;
        background-color: #e6e3e3;
        height: 100%;
        border-right: 1px solid var(--primary-color);
      }
      .main-content {
        flex: 3;
        height: 100%;
        padding: var(--padding);
        overflow: scroll;
      }
      .products {
        display: grid;
        grid-template-columns: repeat(3, minmax(0, 1fr));
        gap: 2rem;
        padding: 2rem 0;
      }
      .product__img {
        width: 100%;
        height: 15rem;
        filter: saturate(0);
        transition: var(--transition);
      }
      /* main ends here  */
      
      /* footer starts here  */
      .footer {
        min-height: 10vh;
        padding: var(--padding);
        background-color: var(--primary-color);
        color: white;
        font-size: 1.1rem;
      }
      /* footer ends here  */
      
      /* responsiveness starts here  */
      @media (max-width: 992px) {
        .flex-space-around,
        .flex-center {
          flex-direction: column;
          gap: 1rem;
          padding: 1rem 0;
        }
        .header {
          min-height: 10vh;
        }
        .sidebar {
          width: 100%;
        }
        .products {
          grid-template-columns: repeat(2, minmax(0, 1fr));
        }
      }
      
      @media (max-width: 768px) {
        .products {
          grid-template-columns: repeat(1, minmax(0, 1fr));
        }
      }
      /* responsiveness ends here  */
      // App.js
      import React from "react";
      
      import Header from "./components/Header";
      import Sidebar from "./components/Sidebar";
      import Footer from "./components/Footer";
      import Products from "./components/Products";
      
      import "./App.css";
      
      const App = () => {
        return (
          <div>
            <Header />
            <main className="flex-center">
              <Sidebar />
              <div className="main-content">
                <Products />
              </div>
            </main>
            <Footer />
          </div>
        );
      };
      
      export default App;
      
      // Header.js
        import React from "react";
      
        const Header = () => {
          return (
            <header className="header flex-center">
              <h2 className="header__title">Anis Express</h2>
            </header>
          );
        };
        export default Header;
      
      // Sidebar.js
      import React from "react";
      
      const Sidebar = () => {
        return <aside className="sidebar">this is sidebar</aside>;
      };
      export default Sidebar;
      
      // Products.js
      import React from "react";
      
      
      const Products = () => {
        const imageSource1 =
          "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
        const title1 = "iPhone 9";
        const description1 = "An apple mobile which is nothing like apple";
        const price1 = 549;
        const rating1 = 4.6;
        const brand1 = "Apple";
        const category1 = "smartphones";
      
        const imageSource2 =
          "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
        const title2 = "iPhone X";
        const description2 =
          "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
        const price2 = 899;
        const rating2 = 4.44;
        const brand2 = "Apple";
        const category2 = "smartphones";
      
        return (
          <section className="products">
            <article className="product">
              <img src={imageSource1} alt="iPhone 9" className="product__img"/>
              <h2>{title1}</h2>
              <p>description: {description1}</p>
              <p>Price: {price1}</p>
              <p>rating: {rating1}</p>
              <p>brand: {brand1}</p>
              <p>category: {category1}</p>
            </article>
            <article className="product">
              <img src={imageSource2} alt="iPhone X" className="product__img"/>
              <h2>{title2}</h2>
              <p>description: {description2}</p>
              <p>Price: {price2}</p>
              <p>rating: {rating2}</p>
              <p>brand: {brand2}</p>
              <p>category: {category2}</p>
            </article>
          </section>
        );
      };
      export default Products;
      
      // Footer.js
      import React from "react";
      
        const Footer = () => {
          return (
            <footer className="footer flex-space-around">
              <div className="footer__left">
                <p className="footer__title">&copy;Copyright by Anisul Islam</p>
              </div>
              <div className="footer__right">
                <p>social media icons</p>
              </div>
            </footer>
          );
        };
      
        export default Footer;
  • CSS module: create a file name such as fileName.module.css as shown below

    footer {
      height: 5vh;
      background-color: bisque;
      display: flex;
      justify-content: center;
      align-items: center;
    }
    
    .copyright {
      font-size: 0.9rem;
    }
    • use it in from another file as shown below

      import React from "react";
      
      import styles from "./footer.module.css";
      
      const Footer = () => {
        return (
          <footer>
            <p className={styles.copyright}>
              All rights reserved by Anisul Islam
            </p>
          </footer>
        );
      };
      
      export default Footer;
  • props object: we can pass information from one component to another using props object. components communicate with each others via props. props is an object. props are like attributes in our HTML tag. props are readonly
    • how to pass, recieve, set default props
    • how to pass JSX to component,
  • Code Example - 15 (Props sending)
import React from "react";
import Product from "./Product";

const Products = () => {
  const imageSource1 = "https://i.dummyjson.com/data/products/1/thumbnail.jpg";
  const title1 = "iPhone 9";
  const description1 = "An apple mobile which is nothing like apple";
  const price1 = 549;
  const rating1 = 4.6;
  const brand1 = "Apple";
  const category1 = "smartphones";

  const imageSource2 = "https://i.dummyjson.com/data/products/2/thumbnail.jpg";
  const title2 = "iPhone X";
  const description2 =
    "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...";
  const price2 = 899;
  const rating2 = 4.44;
  const brand2 = "Apple";
  const category2 = "smartphones";

  return (
    <section className="products">
      <Product
        imageSource={imageSource1}
        title={title1}
        description={description1}
        price={price1}
        rating={rating1}
        brand={brand1}
        category={category1}
      />
      <Product
        imageSource={imageSource2}
        title={title2}
        description={description2}
        price={price2}
        rating={rating2}
        brand={brand2}
        category={category2}
      />
    </section>
  );
};
export default Products;
  • Code Example - 16 (props recieveing & Destructuring)

    import React from "react";
    
    const Product = (props) => {
      const { imageSource, title, description, price, rating, brand, category } =
        props;
      return (
        <article className="product">
          <img src={thumbnail} alt="iPhone 9" className="product__img" />
          <div className="product__body">
            <h2 className="product__title">{title}</h2>
            <p className="product__description">description: {description}</p>
            <p className="product__price">Price: {price}</p>
            <p className="product__rating">rating: {rating}</p>
            <p className="product__brand">brand: {brand}</p>
            <p className="product__category">category: {category}</p>
          </div>
        </article>
      );
    };
    export default Product;
  • Code Example - 17 (props for complex data)

    // Products.js
    import React from "react";
    import Product from "./Product";
    
    const Products = () => {
      const products = [
        {
          id: 1,
          title: "iPhone 9",
          description: "An apple mobile which is nothing like apple",
          price: 549,
          discountPercentage: 12.96,
          rating: 4.69,
          stock: 94,
          brand: "Apple",
          category: "smartphones",
          thumbnail: "https://i.dummyjson.com/data/products/1/thumbnail.jpg",
        },
        {
          id: 2,
          title: "iPhone X",
          description:
            "SIM-Free, Model A19211 6.5-inch Super Retina HD display with OLED technology A12 Bionic chip with ...",
          price: 899,
          discountPercentage: 17.94,
          rating: 4.44,
          stock: 34,
          brand: "Apple",
          category: "smartphones",
          thumbnail: "https://i.dummyjson.com/data/products/2/thumbnail.jpg",
        },
      ];
      return (
        <section className="products">
          <Product
            imageSource={products[0].thumbnail}
            title={products[0].title}
            description={products[0].description}
            price={products[0].price}
            rating={products[0].rating}
            brand={products[0].brand}
            category={products[0].category}
          />
          <Product
            imageSource={products[1].thumbnail}
            title={products[1].title}
            description={products[1].description}
            price={products[1].price}
            rating={products[1].rating}
            brand={products[1].brand}
            category={products[1].category}
          />
        </section>
      );
    };
    export default Products;
  • create data.js in src folder and move all the todos dummy data there and import in App.js for using it.

  • Can you load all the data from App.js to Products.js as props

  • Code Example - 18 (import data from another file and pass as props)

// App.js
import React from "react";

import Header from "./components/Header";
import Sidebar from "./components/Sidebar";
import Footer from "./components/Footer";
import Products from "./components/Products";
import { products } from "./data";

import "./App.css";

const App = () => {
  return (
    <div>
      <Header />
      <main className="flex-center">
        <Sidebar />
        <div className="main-content">
          <Products products={products} />
        </div>
      </main>
      <Footer />
    </div>
  );
};

export default App;

// Products.js
import React from "react";
import Product from "./Product";

const Products = ({ products }) => {
  return (
    <section className="products">
      <Product
        imageSource={products[0].thumbnail}
        title={products[0].title}
        description={products[0].description}
        price={products[0].price}
        rating={products[0].rating}
        brand={products[0].brand}
        category={products[0].category}
      />
      <Product
        imageSource={products[1].thumbnail}
        title={products[1].title}
        description={products[1].description}
        price={products[1].price}
        rating={products[1].rating}
        brand={products[1].brand}
        category={products[1].category}
      />
    </section>
  );
};
export default Products;
  • learn how to use map() and filter() from an array of Data.
  • Code Example - 19 (Map component with for loop)
import React from "react";
import Product from "./Product";

const Products = ({ products }) => {
  const productsElement = [];
  for (let index = 0; index < products.length; index++) {
    productsElement.push(
      <Product
        imageSource={products[index].thumbnail}
        title={products[index].title}
        description={products[index].description}
        price={products[index].price}
        rating={products[index].rating}
        brand={products[index].brand}
        category={products[index].category}
      />
    );
  }
  return <section className="products">{productsElement}</section>;
};
export default Products;
  • Code Example - 20 (Map component with forEach higher order Array function)

    import React from "react";
    import Product from "./Product";
    
    const Products = ({ products }) => {
      const productsElement = [];
    
      products.forEach((product) => {
        productsElement.push(
          <Product
            imageSource={product.thumbnail}
            title={product.title}
            description={product.description}
            price={product.price}
            rating={product.rating}
            brand={product.brand}
            category={product.category}
          />
        );
      });
    
      return <section className="products">{productsElement}</section>;
    };
    export default Products;
  • Code Example - 21 (Map component with map higher order Array function)

    import React from "react";
    import Product from "./Product";
    
    const Products = ({ products }) => {
      const productsElement = products.map((product) => {
        return (
          <Product
            imageSource={product.thumbnail}
            title={product.title}
            description={product.description}
            price={product.price}
            rating={product.rating}
            brand={product.brand}
            category={product.category}
          />
        );
      });
    
      return <section className="products">{productsElement}</section>;
    };
    export default Products;

we need to map each children of list uniquly so that react can identify them wor properly.Keys are unique. we can use same keys for JSX nodes in different arrays.

  • Where I can find key?

    • from database

    • generate locally - crypto.randomUUID() , uuid, nanoid

      • we can use index or any external package like uuid

      • How to use uuid:

        step 1: npm install uuid
        step 2: import { v4 as uuidv4 } from "uuid";
        step 3: uuidv4(); // ⇨ '9b1deb4d-3b7d-4bad-9bdd-2b0d7b3dcb6d'
  • Code Example - 22 (Adding unique key)

    // first use index
    // second use the available id
    // thirs use the uuid if id is not available inside the data
    
    import React from "react";
    import Product from "./Product";
    
    const Products = ({ products }) => {
      const productsElement = products.map((product) => {
        return (
          <Product
            key={product.id}
            imageSource={product.thumbnail}
            title={product.title}
            description={product.description}
            price={product.price}
            rating={product.rating}
            brand={product.brand}
            category={product.category}
          />
        );
      });
    
      return <section className="products">{productsElement}</section>;
    };
    export default Products;
    
    // get a uniqueId -> utility/getUniqueId.js
    import { v4 as uuidv4 } from "uuid";
    export const getUniqueId = () => uuidv4();
    
    // Now add the unique Id in App.js

[jsx spread syntax]

  • Code Example - 23 (jsx spread syntax)
import React from "react";
import Product from "./Product";

const Products = ({ products }) => {
  const productsElement = products.map((product) => {
    return <Product key={product.id} {...product} />;
  });

  return <section className="products">{productsElement}</section>;
};
export default Products;
  • Code Example - 24 (passing jsx as children)
import React from "react";

const Card = ({ children }) => {
  return <div className="card">{children}</div>;
};

export default Card;

import React from "react";
import Card from "./Card";

const Product = (props) => {
  const { thumbnail, title, description, price, rating, brand, category } =
    props;
  return (
    <Card>
      <article className="product">
        <img src={thumbnail} alt="iPhone 9" className="product__img" />
        <h2>{title}</h2>
        <p>description: {description}</p>
        <p>Price: {price}</p>
        <p>rating: {rating}</p>
        <p>brand: {brand}</p>
        <p>category: {category}</p>
      </article>
    </Card>
  );
};
export default Product;
  • documentation is here

  • catch bugs with typechecking.

  • We can use Typescript or Flow for checking types in the entire application for sure but PropTypes can be the first guard here for checking types.

  • default props is essential when first time running without any value passing to props

      // lets think value can be primary or secondary one of two valyes
      <List background = 'primary' >
    
      // checking prop types
      background: PropTypes.oneOf([
        'primary',
        'secondary'
      ])
  • Code Example - 25 (PropTypes)

    // Todos.js
    import React from "react";
    import PropTypes from "prop-types";
    
    import Todo from "./Todo";
    
    const Todos = (props) => {
      const { todos } = props;
    
      const renderTodosElement = todos.map((todo) => (
        <Todo key={todo.id} todo={todo} />
      ));
    
      return <section className="todos">{renderTodosElement}</section>;
    };
    Todos.defaultProps = {
      todos: [],
    };
      Todos.propTypes = {
      todos: PropTypes.arrayOf(
        PropTypes.shape({
          id: PropTypes.string,
          title: PropTypes.string,
          desc: PropTypes.string,
        })
      ),
    };
    
    
    export default Todos;
    
    // Todo.js
    import React from "react";
    import PropTypes from "prop-types";
    
    const Todo = (props) => {
      const { todo } = props;
      return (
        <article className="todo" key={todo.id}>
          <h3>{todo.title}</h3>
          <p>{todo.desc} </p>
        </article>
      );
    };
    
    Todo.propTypes = {
      todo: PropTypes.shape({
        title: PropTypes.string,
        desc: PropTypes.string,
      }),
    };
    
    export default Todo;
  • rendering components based on if-else, element variable, ternary, short circuit

  • Code Example - 26 (Conditional rendering: element variable)

    // Inside App.js make adjustments
    let productsElement;
    if (products.length > 0) {
      productsElement = <Products products={products} />;
    } else {
      productsElement = <p>Product list is empty</p>;
    }
    return (
      <div>
        <Header />
        <main className="flex-center">
          <Sidebar />
          <div className="main-content">{productsElement}</div>
        </main>
        <Footer />
      </div>
    );
  • Code Example - 27 (Conditional rendering: iternary)

    // Inside App.js make adjustments
    let productsElement =
      products.length > 0 ? (
        <Products products={products} />
      ) : (
        <p>products list is empty</p>
      );
    
    return (
      <div>
        <Header />
        <main className="flex-center">
          <Sidebar />
          <div className="main-content">{productsElement}</div>
        </main>
        <Footer />
      </div>
    );
    
    // An alternative - we can use ternary inside return () function
    return (
      <div>
        <Header />
        <main className="flex-center">
          <Sidebar />
          <div className="main-content">
            {products.length > 0 ? (
              <Products products={products} />
            ) : (
              <p>products list is empty</p>
            )}
          </div>
        </main>
        <Footer />
      </div>
    );
  • Code Example - 28 (Conditional rendering: short circuit)

    // Inside App.js return() make following adjustments
    return (
      <div>
        <Header />
        <main>{products.length > 0 && <Products products={products} />}</main>
        <Footer />
      </div>
    );

1.9 developer tools, react and font-awesome icons

  • developer tools and extension

  • 1.10 add font awesome / react icons

  • How to use react-icons

  • Code Example - 29 (Adding & styling react icons )

    import React from "react";
    import { FaFacebookF, FaYoutube, FaTwitter } from "react-icons/fa";
    const Footer = () => {
      return (
        <footer className="footer flex-space-around">
          <div className="footer__left">
            <p className="footer__title">&copy;Copyright by Anisul Islam</p>
          </div>
          <div className="footer__right">
            <FaFacebookF />
            <FaYoutube />
            <FaTwitter />
          </div>
        </footer>
      );
    };
    
    export default Footer;
    
    // Product.js
    import React from "react";
    import { FaCartPlus } from "react-icons/fa";
    
    import Card from "./Card";
    
    const Product = (props) => {
      const { thumbnail, title, description, price, rating, brand, category } =
        props;
      return (
        <Card>
          <article className="product">
            <img src={thumbnail} alt="iPhone 9" className="product__img" />
            <div className="product__body">
              <h2 className="product__title">{title}</h2>
              <p className="product__description">description: {description}</p>
              <p className="product__price">Price: {price}</p>
              <p className="product__rating">rating: {rating}</p>
              <p className="product__brand">brand: {brand}</p>
              <p className="product__category">category: {category}</p>
              <button className="btn product__btn icon">
                <FaCartPlus className="icon" /> Add To Cart
              </button>
            </div>
          </article>
              </Card>
            );
          };
          export default Product;
    
    
    // App.css
    /* reset code and common starts here  */
      :root {
        --primary-color: rgb(11, 181, 48);
        --secondary-color: rgba(36, 122, 55, 0.9);
        --padding: 0.5rem;
        --transition: all 0.3s;
        --border-radius: 0.6rem;
        --box-shadow: 0.1rem 0.2rem 0.8rem rgba(205, 202, 202, 0.5);
      }
      * {
        box-sizing: border-box;
        margin: 0;
        padding: 0;
        text-decoration: none;
        list-style-type: none;
        outline: none;
      }
      html {
        scroll-behavior: smooth;
      }
      .flex-space-around {
        display: flex;
        justify-content: space-around;
        align-items: center;
      }
      .flex-center {
        display: flex;
        justify-content: center;
        align-items: center;
      }
      .card {
        box-shadow: var(--box-shadow);
        border-radius: var(--border-radius);
        /* padding: var(--padding); */
        transition: var(--transition);
      }
      img {
        width: 100%;
        height: auto;
      }
      .btn {
        border: none;
        padding: var(--padding);
        border-radius: var(--border-radius);
        transition: all 0.3s;
        color: var(--secondary-color);
      }
      .btn:hover {
        background-color: orange;
        color: black;
      }
      /* reset code and common ends here  */
    
      /* header starts here  */
      .header {
        height: 10vh;
        background-color: var(--primary-color);
        color: white;
      }
      /* header ends here  */
    
      /* main starts here  */
      main {
        height: 80vh;
      }
      .sidebar {
        flex: 1;
        padding: var(--padding);
        display: flex;
        flex-direction: column;
        justify-content: center;
        gap: 1rem;
        align-self: flex-start;
        background-color: #e6e3e3;
        height: 100%;
        border-right: 1px solid var(--primary-color);
      }
      .main-content {
        flex: 3;
        height: 100%;
        padding: var(--padding);
        overflow: scroll;
      }
      .products {
        display: grid;
        grid-template-columns: repeat(3, minmax(0, 1fr));
        gap: 2rem;
      }
      .product {
        display: flex;
        flex-direction: column;
        gap: 0.5rem;
      }
      .product__body {
        padding: var(--padding);
        display: flex;
        flex-direction: column;
        gap: 0.5rem;
      }
      .product__img {
        width: 100%;
        height: 15rem;
        filter: saturate(0);
        transition: var(--transition);
      }
      /* main ends here  */
    
      /* footer starts here  */
      .footer {
        min-height: 10vh;
        padding: var(--padding);
        background-color: var(--primary-color);
        color: white;
        font-size: 1.1rem;
      }
      .footer__right {
        display: flex;
        gap: 1rem;
      }
      /* footer ends here  */
    
      /* responsiveness starts here  */
      @media (max-width: 992px) {
        .flex-space-around,
        .flex-center {
          flex-direction: column;
          gap: 1rem;
          padding: 1rem 0;
        }
        .header {
          min-height: 10vh;
        }
        .sidebar {
          width: 100%;
        }
        .products {
          grid-template-columns: repeat(2, minmax(0, 1fr));
        }
      }
    
      @media (max-width: 768px) {
        .products {
          grid-template-columns: repeat(1, minmax(0, 1fr));
        }
      }
      /* responsiveness ends here  */
  • How to use font awesome icons directly

       // Add SVG Core
      npm i --save @fortawesome/fontawesome-svg-core
    
      // add Free icons styles
      npm i --save @fortawesome/free-solid-svg-icons
      npm i --save @fortawesome/free-regular-svg-icons
      npm i --save @fortawesome/free-brands-svg-icons
    
      // add the react component
      npm i --save @fortawesome/react-fontawesome@latest
    
    
      // usage
      import { FontAwesomeIcon } from '@fortawesome/react-fontawesome';
      import { faPlusCircle, faMinusCircle } from '@fortawesome/free-solid-svg-icons';
      import { faYoutube } from '@fortawesome/free-brands-svg-icons';
    
      <FontAwesomeIcon icon={faPlusCircle}></FontAwesomeIcon>
    
      // use className for styling icons

[1.10 Adding Interactivity - event & event handler]

  • Event: any user interaction like clicking button, hovering button, giving values in input field etc.

  • Event handler: your response to user interactions. create a function for handling the interaction.

  • state - value of anything can be changed based on your interaction and you may want to update UI based on state.

  • Code Example - 30 (event - onClick )

    <button
      className="btn product__btn"
      onClick={alert("product is added to the cart")} // not depends on user inetractivity
    >
      <FaCartPlus className="icon" /> Add To Cart
    </button>
  • Code Example - 31 (event Handler)

    const handleClick = () => {
      alert("product is added to the cart");
    };
    
    // method 1
    <button
      className="btn product__btn"
      onClick={() => {
        alert("product is added to the cart");
      }} // depends on user inetractivity as we definied the function and it is not running until we click the button
    >
      <FaCartPlus className="icon" /> Add To Cart
    </button>
    
    // method 2
    <button
      className="btn product__btn"
      onClick={handleClick} // do not call just define the function
    >
      <FaCartPlus className="icon" /> Add To Cart
    </button>
    
    // we can create own custom Button component
    import React from "react";
    
    const Button = (props) => {
      console.log(props);
    
      const handleClick = () => {
        // we can use condition for calling multiple functions
        props.onHandleAddProduct();
      };
    
     const handleAddProduct = (product) => {
      alert(JSON.stringify(product, null, 4));
    };
    
      return (
        <button className={props.className} onClick={props.onClick} type={props.type}>
        {props.children}
      </button>
      );
    };
    
    export default Button;
    
    // more simplified
    import React from "react";
    
    const Button = (props) => {
      return (
        <button className={props.className} onClick={props.onClick}>
          {props.children}
        </button>
      );
    };
    
    export default Button;
  • Code Example - 32 (Pass parameter with event )

    const handleAddProduct = (e, product) => {
      e.stopPropagation();
      console.log("Product Event: ", e);
      alert(JSON.stringify(product, null, 4));
    };
    
    <Button
      className="btn product__btn"
      onClick={(e) => {
        handleAddProduct(e, product);
      }}
    >
      <FaCartPlus className="icon" /> Add To Cart
    </Button>;
  • Code Example - 33 (onChange Event )

    import React from "react";
    
    const NewProduct = () => {
      const handleChange = (event) => {
        console.log(event.target.value);
      };
      return (
        <div className="new-user">
          <h2>Create Product</h2>
    
          <div className="form__control">
            <label htmlFor="title">Title: </label>
            <input
              className="form__input"
              type="text"
              placeholder="Enter product title"
              name="title"
              onChange={handleChange}
              required={true}
            />
          </div>
    
          <div className="form__control">
            <label htmlFor="price">Price: </label>
            <input
              className="form__input"
              type="number"
              placeholder="Enter product price here"
              name="price"
              onChange={handleChange}
              required={true}
            />
          </div>
        </div>
      );
    };
    
    export default NewProduct;
  • Code Example - 34 (More complex example of onChange Event )

    import React from "react";
    
    const NewProduct = () => {
      const inputs = [
        {
          id: 1,
          type: "text",
          name: "title",
          placeholder: "Enter product title",
          required: true,
        },
        {
          id: 2,
          type: "number",
          name: "price",
          placeholder: "Enter product price",
          required: true,
        },
      ];
      const handleChange = (event) => {
        console.log(event.target.value);
      };
    
      return (
        <div className="new-user">
          <h2>Create Product</h2>
    
          <input
            className="form__input"
            type={inputs[0].type}
            placeholder={inputs[0].placeholder}
            name={inputs[0].name}
            onChange={handleChange}
            required={inputs[0].required}
          />
    
          <input
            className="form__input"
            type={inputs[1].type}
            placeholder={inputs[1].placeholder}
            name={inputs[1].name}
            onChange={handleChange}
            required={inputs[1].required}
          />
        </div>
      );
    };
    
    // More simplified
    const NewProduct = () => {
      const inputs = [
        {
          id: 1,
          type: "text",
          name: "title",
          placeholder: "Enter product title",
          required: true,
        },
        {
          id: 2,
          type: "number",
          name: "price",
          placeholder: "Enter product price",
          required: true,
        },
      ];
      const handleChange = (event) => {
        console.log(event.target.value);
      };
    
      const renderInputs = inputs.map((input) => {
        return (
          <Input
            key={input.id}
            className="form__input"
            type={input.type}
            placeholder={input.placeholder}
            name={input.name}
            onChange={handleChange}
            required={input.required}
          />
        );
      });
    
      return (
        <div className="new-user">
          <h2>Create Product</h2>
          {renderInputs}
        </div>
      );
    };
    
    // more simplified
    <Input key={input.id} {...input} onChange={handleChange} />;
    
    // Input.js
    import React from "react";
    const Input = (props) => {
      const { onChange, ...inputProps } = props; // getting onChnage from props and putting rest of the things inside inputProps
    
      console.log([inputProps]);
      return (
        <input className="form__input" {...inputProps} onChange={onChange} />
      );
    };
    
    export default Input;
  • Code Example - 35 (Event Bubbling - child event can effect parent event )

    const handleAddProduct = (e, product) => {
      e.stopPropagation();
      console.log("Product Event: ", e);
      alert(JSON.stringify(product, null, 4));
    };
  • Code Example - 36 (onSubmit Event )

  • without state management
const Counter = () => {
  let count = 0;

  const handleIncrement = () => {
    count++;
    alert(count);
  };
  const handleDecrement = () => {
    count--;
    alert(count);
  };
  const handleReset = () => {
    count = 0;
    alert(count);
  };

  return (
    <div>
      <h2>Count: {count}</h2>
      <button onClick={handleIncrement}>+</button>
      <button onClick={handleReset}>0</button>
      <button onClick={handleDecrement}>-</button>
    </div>
  );
};

export default Counter;
  • state is a memory for component where we can update value and re-render the component

  • state is a global variable thats why when you even re-render it fetch the last updated value

  • hooks are function which we can implement in our component - useState, useEffect

  • useState() hook helps us to track state in a functional component.

  • Code Example - 36 (Counter App )

    // App.js
    import React from "react";
    
    import Counter from "./components/Counter";
    
    const App = () => {
      return (
        <div>
          <Counter />
        </div>
      );
    };
    
    export default App;
    // Counter.js
    import React, { useState } from "react";
    
    function Counter() {
      const [count, setCount] = useState(0);
      const handleIncrement = () => {
        setCount(count + 1); // setCount (count => count + 1)
      };
    
      return (
        <div>
          <h2>Counter: {count}</h2>
          <button onClick={handleIncrement}>+</button>
        </div>
      );
    }
    
    export default Counter;
  • Code Example - 37 (store data in state )

    import React, { useState } from "react";
    import { FaCartPlus } from "react-icons/fa";
    import Button from "./Button";
    
    import Card from "./Card";
    
    const Product = (props) => {
      const { thumbnail, title, description, price, rating, brand, category } =
        props;
      const product = props;
      const [cartProducts, setCartProducts] = useState([]);
      const handleAddProduct = (e, product) => {
        e.stopPropagation();
        setCartProducts((prevState) => {
          return [...prevState, product];
        });
      };
      console.log(cartProducts);
      return (
        <Card>
          <article className="product">
            <img src={thumbnail} alt="iPhone 9" className="product__img" />
            <div className="product__body">
              <h2 className="product__title">{title}</h2>
              <p className="product__description">description: {description}</p>
              <p className="product__price">Price: {price}</p>
              <p className="product__rating">rating: {rating}</p>
              <p className="product__brand">brand: {brand}</p>
              <p className="product__category">category: {category}</p>
              {/* <button className="btn product__btn icon">
                <FaCartPlus className="icon" /> Add To Cart
              </button> */}
              <Button
                className="btn product__btn"
                onClick={(e) => {
                  handleAddProduct(e, product);
                }}
              >
                <FaCartPlus className="icon" /> Add To Cart
              </Button>
            </div>
          </article>
        </Card>
      );
    };
    export default Product;
  • Code Example - 38 (why update data based on prevState?)

    • update state based on prev value - setCount (count => count + 1)
        const handleIncrement = (e) => {
          e.stopPropagation();
    
          setcount(count + 1);
          setcount(count + 1);
          setcount(count + 1);
    
          console.log("increment count: ", count);
        };
    
        const handleIncrement = (e) => {
            e.stopPropagation();
    
            setcount((count) => count + 1);
            setcount((count) => count + 1);
            setcount((count) => count + 1);
    
            console.log("increment count: ", count);
          };
    
        const handleIncrement = (e) => {
           e.stopPropagation();
          setTimeout(() => {
            <!-- setcount(count + 1); -->
            setcount((count) => count + 1);
          }, 3000);
          console.log("increment count: ", count);
        };
  • Code Example - 40 (updating object in state)

    import React, { useState } from "react";
    
    const NewUser = () => {
      const [user, setUser] = useState({
        name: "",
        email: "",
        password: "",
      });
      const handleChange = (event) => {
        event.stopPropagation();
        setUser((prevUser) => {
          return { ...prevUser, [event.target.name]: event.target.value };
        });
      };
    
      const handleSubmit = (event) => {
        event.preventDefault();
        console.log("submitted");
        alert(JSON.stringify(user));
      };
    
      return (
        <div>
          <form action="" className="form" onSubmit={handleSubmit}>
            <div className="form__control">
              <label htmlFor="name">name: </label>
              <input
                type="text"
                placeholder="Enter Name"
                name="name"
                value={user.name}
                onChange={handleChange}
                required={true}
              />
            </div>
            <div className="form__control">
              <label htmlFor="email">email: </label>
              <input
                type="email"
                placeholder="Enter Email"
                name="email"
                value={user.email}
                onChange={handleChange}
                required={true}
              />
            </div>
            <div className="form__control">
              <label htmlFor="password">password: </label>
              <input
                type="password"
                placeholder="Enter password"
                name="password"
                value={user.password}
                onChange={handleChange}
                required={true}
              />
            </div>
            <button type="submit">Add User</button>
          </form>
        </div>
      );
    };
    
    export default NewUser;
  • Code Example - 41 (updating nested object in state)

  • Code Example - 42 (updating array in state)

  • Another practical example: https://youtu.be/h7yq5lfDZc8

  • Freecodecamp doc

  • Code Example - 43 (state lifting)

    // App.js
    step 1: create a callback function that will help us to get the data from child component
     const handleAddNewTodo = (newTodo) => {
      console.log(newTodo);
    };
    
     step 2: pass the function as props to child to component
    <AddTodo onHandleAddNewTodo={handleAddNewTodo} />
    
    step 3: receive the function props and use it for passing the data from child to parent
    // AddTodo.js
    import PropTypes from "prop-types";
    
    const handleSubmit = (event) => {
      event.preventDefault();
      const newTodo = { id: getUniqueId(), title, desc };
      setTitle("");
      setDesc("");
      props.onHandleAddNewTodo(newTodo);
    };
    
    AddTodo.propTypes = {
      onHandleAddNewTodo: PropTypes.func,
    };
    
    
    // Another Example
      function App() {
        const [users, setUsers] = useState([
          { name: 'alex', email: 'alex@gmail.com' },
        ]);
    
        console.log(users);
    
        return (
          <div className="container">
            <Test setUsers={setUsers} />
          </div>
          )
      }
    
      const Test = (props) => {
        const { setUsers } = props;
    
        const [user, setUser] = useState({ name: '', email: '' });
        const handleSubmit = (event) => {
          event.preventDefault();
          setUsers((users) => [...users, user]);
          setUser({
            name: '',
            email: '',
          });
        };
        const handleChange = (event) => {
          const name = event.target.name;
          setUser({ ...user, [name]: event.target.value });
        };
     }
    • Code Example - 44 (state lifting and delete item)

      // App.js
      step 1: create a function that will help us to get the id from child component Todo.js
      const handleDeleteTodo = (id) => {
        console.log(id);
      };
      
      step 2: pass the function as props to App.js -> Todos.js -> Todo.js
      <Todos onHandleDeleteTodo={handleDeleteTodo} />
      
      step 3: After passing the function from App.js to Todo.js now lets handle the button click and get the id so that we pass the id to App.js with the help of the function onHandleDeleteTodo
      // Todo.js
      
      const { todo, onHandleDeleteTodo } = props;
      
      const handleDelete = (id) => {
        onHandleDeleteTodo(id);
      };
      
      <button
              className="btn"
              onClick={() => {
                handleDelete(todo.id);
              }}
            >
              <FaTrash className="icon" />
      </button>
      
      step 4: finally with the help of the id lets delete the item from todos state inside App.js
      
      const handleDeleteTodo = (id) => {
        setTodos(todos.filter((todo) => todo.id !== id));
      };
  • If we look at the AddTodo component then you will see we are not using those title and desc state inside the component that much so we can avoid state and make the component stateless

  • Code Example - 45 (useRef hook for getting form value)

    import React, { useRef } from "react";
    import PropTypes from "prop-types";
    
    import { getUniqueId } from "../utility/getUniqueId";
    
    const AddTodo = (props) => {
      const titleRef = useRef("");
      const descRef = useRef("");
    
      const handleSubmit = (event) => {
        event.preventDefault();
        const title = titleRef.current.value;
        const desc = descRef.current.value;
        const newTodo = { id: getUniqueId(), title, desc };
        props.onHandleAddNewTodo(newTodo);
        titleRef.current.value = "";
        descRef.current.value = "";
      };
    
      return (
        <div className="form-container">
          <form onSubmit={handleSubmit}>
            <h2 className="form-title">Add New Todo</h2>
            <div className="form-input">
              <label htmlFor="title">Todo Title: </label>
              <input
                type="text"
                name="title"
                id="title"
                ref={titleRef}
                required
                autoFocus
              />
            </div>
            <div className="form-input">
              <label htmlFor="desc">Todo description: </label>
              <textarea name="desc" id="desc" ref={descRef} required></textarea>
            </div>
            <div className="form-input">
              <button type="submit" className="btn form-btn">
                Add Todo
              </button>
            </div>
          </form>
        </div>
      );
    };
    
    AddTodo.propTypes = {
      onHandleAddNewTodo: PropTypes.func,
    };
    
    export default AddTodo;
  • Now lets add some coditional styling

  • Code Example - 46 (conditional styling)

    import React, { useState, useEffect } from "react";
    import PropTypes from "prop-types";
    
    import { getUniqueId } from "../utility/getUniqueId";
    
    const AddTodo = (props) => {
      const [title, setTitle] = useState("");
      const [desc, setDesc] = useState("");
    
      const [isTitleValid, setIsTitleValid] = useState(false);
      const [isDescValid, setIsDescValid] = useState(false);
    
      useEffect(() => {
        if (title.trim().length > 3) {
          setIsTitleValid(true);
        }
        if (desc.trim().length > 9) {
          setIsDescValid(true);
        }
      }, [title, desc]);
    
      const handleTitleChange = (event) => {
        setTitle(event.target.value);
      };
    
      const handleDescriptionChange = (event) => {
        setDesc(event.target.value);
      };
    
      const handleSubmit = (event) => {
        event.preventDefault();
    
        const newTodo = { id: getUniqueId(), title, desc };
        props.onHandleAddNewTodo(newTodo);
        setTitle("");
        setDesc("");
      };
    
      return (
        <div className="form-container">
          <pre style={{ backgroundColor: "white" }}>
            {JSON.stringify({ id: getUniqueId(), title, desc }, undefined, 2)}
          </pre>
          <form onSubmit={handleSubmit}>
            <h2 className="form-title">Add New Todo</h2>
            <div className="form-input">
              <label htmlFor="title">Todo Title: </label>
              <input
                type="text"
                name="title"
                id="title"
                value={title}
                onChange={handleTitleChange}
                required
                autoFocus
              />
              {!isTitleValid && <p>Title Should be at least 4 characters</p>}
            </div>
            <div className="form-input">
              <label htmlFor="desc">Todo description: </label>
              <textarea
                name="desc"
                id="desc"
                value={desc}
                onChange={handleDescriptionChange}
                required
              ></textarea>
              {!isDescValid && (
                <p>Description Should be at least 10 characters</p>
              )}
            </div>
            <div className="form-input">
              <button
                type="submit"
                className="btn form-btn"
                disabled={!(isTitleValid && isDescValid)}
              >
                Add Todo
              </button>
            </div>
          </form>
        </div>
      );
    };
    
    AddTodo.propTypes = {
      onHandleAddNewTodo: PropTypes.func,
    };
    
    export default AddTodo;
  • state is a js object for storing current situation of a component

  • Code Example - 47 (Counter App using class component)

    // App.js
    import React from "react";
    
    import Counter from "./components/Counter";
    
    const App = () => {
      return (
        <div>
          <Counter />
        </div>
      );
    };
    
    export default App;
    // Counter.js
    import React, { Component } from "react";
    
    export default class Counter extends Component {
      constructor(props) {
        super(props);
    
        this.state = {
          count: 0,
        };
      }
    
      handleIncrement = () => {
        this.setState({
          count: this.state.count + 1,
        });
      };
    
      render() {
        return (
          <div>
            <h2>Counter: {this.state.count}</h2>
            <button onClick={this.handleIncrement}>+</button>
          </div>
        );
      }
    }

[1.18 react todo projects]

2. Intermediate React.js Topics

  • Code Example - 48 (life cycle methods of a class component)

    // Counter App
    import React, { Component } from "react";
    
    class Counter extends Component {
      constructor(props) {
        super(props);
    
        this.state = {
          count: 0,
        };
      }
    
      handleIncrement = () => {
        this.setState({
          count: this.state.count + 1,
        });
      };
    
      render() {
        return (
          <div>
            <h2>Count: {this.state.count}</h2>
            <button onClick={this.handleIncrement}>Increment</button>
          </div>
        );
      }
    }
    export default Counter;
    
    // LifeCycle Methods
    import React, { Component } from 'react';
    // mounting - constructor -> render -> componentDidMount() (it will be called once, API Call is recommended here)
    // updating - state/props ->  shouldComponentUpdate()-> render -> componentDidUpdate ->
    // unmounting -> componentDidUnmount()
    class LifeCycle extends Component {
        constructor(props) {
            super(props);
            console.log('constructor');
            this.state = {
                count: 0
            };
        }
    
        handleIncrement = () => {
            this.setState({
                count: this.state.count + 1
            });
        };
        componentDidMount() {
            console.log('componentDidMount');
        }
        componentDidUpdate() {
            console.log('componentDidUpdate');
        }
        shouldComponentUpdate() {
            console.log('shouldComponentUpdate()');
            return true;
        }
    
        render() {
            console.log('render');
            return (
                <div>
                    <h2>Count: {this.state.count}</h2>
                    <button onClick={this.handleIncrement}>Increment</button>
                </div>
            );
        }
    }
    export default LifeCycle;

The useEffect hook in React allows you to perform side effects in your functional components. Side effects can include data fetching, DOM manipulation, setting up subscriptions, and more. It's a crucial hook for managing the lifecycle of your components.

Rule: Don’t call Hooks inside loops, conditions, or nested functions

useEffct = componentDidMount + componentDidUpdate + componentWillUnmount

Here's how to use the useEffect hook in a React component:

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

function ExampleComponent() {
  // State and other code here

  useEffect(() => {
    // This function will run after the component renders

    // Place your side effect code here
    console.log('Component has rendered');

    // If needed, you can return a cleanup function
    return () => {
      // This code will run before the component unmounts
      console.log('Component will unmount');
    };
  }, []); // Dependency array (optional)

  return (
    // JSX to render
  );
}

export default ExampleComponent;

Key points about the useEffect hook:

  1. The useEffect hook takes two arguments:

    • The first argument is a function that contains your side effect code.
    • The second argument is an array of dependencies. If any of these dependencies change, the side effect function will be re-executed. If you want the side effect to run only once (similar to componentDidMount in class components), you can pass an empty array ([]) as the second argument.
  2. The side effect function is run after the component has rendered. It can run multiple times if the dependencies change (if provided in the dependency array).

  3. You can return a cleanup function from the side effect function. This cleanup function will run before the component unmounts. It's useful for unsubscribing from subscriptions, canceling network requests, or any cleanup work.

Here's a breakdown of how you might use the useEffect hook for common scenarios:

  • Data Fetching: Fetch data when the component mounts, and clean up any resources when it unmounts.
useEffect(() => {
  const fetchData = async () => {
    // Fetch data here
  };

  fetchData();

  return () => {
    // Cleanup resources (e.g., cancel network request)
  };
}, []);
  • Updating the Title: Change the document title when the component mounts.
useEffect(() => {
  document.title = 'New Page Title';

  return () => {
    document.title = 'Previous Page Title';
  };
}, []);
  • Listening for State Changes: Execute code when specific state values change.
useEffect(() => {
  // Execute code when someStateValue changes
}, [someStateValue]);
  • Subscription Management: Subscribe to a service and unsubscribe when the component unmounts.
useEffect(() => {
  const subscription = service.subscribe(data => {
    // Handle data
  });

  return () => {
    subscription.unsubscribe();
  };
}, []);

The useEffect hook is a versatile tool for managing side effects in your React components and can help you keep your code clean and efficient.

  • Code Example - 49 (useEffect hook)

    import React, { useEffect, useState } from "react";
    
     const UseEffectHook = () => {
      const [count, setCount] = useState(0);
    
      // useEffect(() => {
      //   //Runs on every render
      // });
    
      // useEffect(() => {
      //   //Runs only on the first render
      // }, []);
    
      // useEffect(() => {
      //   //Runs on the first render
      //   //And any time any dependency value changes
      // }, [prop, state]);
    
      //Runs on every render
      // useEffect(() => {
      //   console.log("useEffect: " + count);
      // });
    
      //Runs only on the first render
      // useEffect(() => {
      //   console.log(count);
      // }, []);
    
      //Runs on the first render and also when the dependecy value changes
      useEffect(() => {
        console.log(count);
      }, [count]);
    
      return (
        <div>
          <h1>Count: {count}</h1>
          <button
            onClick={() => {
              setCount((count) => count + 1);
            }}
          >
            +
          </button>
        </div>
      );
    };
    
    export default UseEffectHook;
    // Another example
    import React, { useState, useEffect } from "react";
    
    const UseEffectExample = () => {
      const [count, setCount] = useState(0);
      const [greeting, setGreeting] = useState("good morning");
      useEffect(() => {
        console.log("hello from useeffect");
      }, [count]);
    
      return (
        <div>
          <h2>Count: {count}</h2>
          <button
            onClick={() => {
              setCount((count) => count + 1);
            }}
          >
            +
          </button>
          <button
            onClick={() => {
              setGreeting(greeting + "2");
            }}
          >
            greeting
          </button>
          <p>{greeting}</p>
        </div>
      );
    };
    
    export default UseEffectExample;
  • Code Example - 50 (fetch data using useEffect hook)

    import React, { useEffect, useState } from "react";
    
    import "./App.css";
    
    const App = () => {
      const [users, setUsers] = useState([]);
      const [isLoading, setIsLoading] = useState(false);
      const [error, setError] = useState(null);
    
      useEffect(() => {
        setIsLoading(true);
        fetch("https://jsonplaceholder.typicode.com/users")
          .then((res) => {
            if (!res.ok) {
              throw new Error("could not fetch the data");
            }
            return res.json();
          })
          .then((result) => {
            setUsers(result);
            setIsLoading(false);
          })
          .catch((error) => {
            setIsLoading(false);
            setError(error.message);
          });
      }, []);
      console.log(users);
      const renderUsers = users.map((user) => {
        const { id, name, username, email, phone, website } = user;
        return (
          <article key={id} style={{ margin: "2rem" }}>
            <h2>{id}</h2>
            <h3>{name}</h3>
            <p>{username}</p>
            <p>{email}</p>
            <p>{phone}</p>
            <p>{website}</p>
          </article>
        );
      });
    
      return (
        <div>
          {isLoading && <p>Loading...</p>}
          {error ? <p>{error}</p> : <section>{renderUsers}</section>}
        </div>
      );
    };
    
    export default App;
  • Code Example - 51 (fetch data using useEffect hook - async await)

    import React, { useEffect, useState } from "react";
    
    import "./App.css";
    
    const App = () => {
      const [users, setUsers] = useState([]);
      const [isLoading, setIsLoading] = useState(false);
      const [error, setError] = useState(null);
    
      const fetchUsers = async () => {
        try {
          const response = await fetch(
            "https://jsonplaceholder.typicode.com/users"
          );
          if (!response.ok) {
            throw new Error(`Request failed with status: ${response.status}`);
          }
          const result = await response.json();
          setUsers(result);
          setIsLoading(false);
        } catch (error) {
          setIsLoading(false);
          setError(error.message);
        }
      };
    
      useEffect(() => {
        setIsLoading(true);
        fetchUsers();
      }, []);
    
      const renderUsers = users.map((user) => {
        const { id, name, username, email, phone, website } = user;
        return (
          <article key={id} style={{ margin: "2rem" }}>
            <h2>{id}</h2>
            <h3>{name}</h3>
            <p>{username}</p>
            <p>{email}</p>
            <p>{phone}</p>
            <p>{website}</p>
          </article>
        );
      });
    
      return (
        <div>
          {isLoading && <p>Loading...</p>}
          {error ? <p>{error}</p> : <section>{renderUsers}</section>}
        </div>
      );
    };
    
    export default App;
  • Code Example - 52 (custom hook)

    import React, { useState, useEffect } from "react";
    
    const useFetch = (URL) => {
      const [data, setData] = useState([]);
      const [isLoading, setIsLoading] = useState(false);
      const [error, setError] = useState(null);
    
      const fetchUsers = async () => {
        try {
          const response = await fetch(URL);
          const result = await response.json();
          setData(result);
          setIsLoading(false);
        } catch (error) {
          setIsLoading(false);
          setError(error.message);
        }
      };
    
      useEffect(() => {
        setIsLoading(true);
        fetchUsers();
      }, []);
    
      return { data, isLoading, error };
    };
    
    export default useFetch;

[Create services for making http requests]

  • Code Example - 53 (services for http requests)

    // UserService.js
    import axios from "axios";
    
    const BASE_URL = "https://jsonplaceholder.typicode.com";
    
    // using axios
    export const getAllUsers = async () => {
      const response = await axios.get(`${BASE_URL}/users/202`);
      return response.data;
    };
    
    // using fetch
    export const createUser = async () => {
      const response = await fetch("https://jsonplaceholder.typicode.com/posts", {
        method: "POST",
        body: JSON.stringify({
          title: "foo",
          body: "bar",
          userId: 1,
        }),
        headers: {
          "Content-type": "application/json; charset=UTF-8",
        },
      });
      return response;
    };
    
    // App.js
    import React, { useEffect, useState } from "react";
    import { ToastContainer, toast } from "react-toastify";
    import "react-toastify/dist/ReactToastify.css";
    
    import "./App.css";
    import { getAllUsers } from "./services/UserService";
    
    const App = () => {
      const [users, setUsers] = useState([]);
      const [isLoading, setIsLoading] = useState(false);
      const [error, setError] = useState(null);
    
      const fetchUsers = async () => {
        try {
          const result = await getAllUsers();
          console.log(result);
          setUsers(result);
          setIsLoading(false);
        } catch (err) {
          setIsLoading(false);
          setError(err.message);
          toast(err.message);
        }
      };
    
      useEffect(() => {
        setIsLoading(true);
        fetchUsers();
      }, []);
    
      const renderUsers = users.map((user) => {
        const { id, name, username, email, phone, website } = user;
        return (
          <article key={id} style={{ margin: "2rem" }}>
            <h2>{id}</h2>
            <h3>{name}</h3>
            <p>{username}</p>
            <p>{email}</p>
            <p>{phone}</p>
            <p>{website}</p>
          </article>
        );
      });
    
      return (
        <div>
          <ToastContainer />
          {isLoading && <p>Loading...</p>}
          {error ? <p>{error}</p> : <section>{renderUsers}</section>}
        </div>
      );
    };
    
    export default App;
  • useState and useReducers helps us to manage states

  • useReducer is a good choice if you have multiple & complex states (objects, arrays)

  • useReducer is powerful when managing complex logic for updating the states

  • useState is better if you are using state for simple purpose and handling string, boolean or number type.

  • Code Example - 54 (without useReducer)

    // without useReducer
    import React, { useState } from "react";
    
    // books, modalText, isModalOpen
    // add book - modalText
    // remove book - modalText
    const dummyBooks = [
      { id: 1, title: "book1" },
      { id: 2, title: "book2" },
    ];
    const Modal = ({ modalText }) => {
      return <p>{modalText}</p>;
    };
    const App = () => {
      // every time update 3 relevant states
      const [books, setBooks] = useState(dummyBooks);
      const [modalText, setModalText] = useState("");
      const [isModalOpen, setIsModalOpen] = useState(false);
    
      const [bookTitle, setBookTitle] = useState("");
    
      const handleAddBook = (event) => {
        event.stopPropagation();
        setBookTitle(event.target.value);
      };
      const handleDeleteBook = (event, id) => {
        event.stopPropagation();
        const filterBooks = books.filter((book) => book.id !== id);
        setBooks(filterBooks);
        setIsModalOpen(true);
        setModalText(" book is deleted");
      };
    
      const handleSubmit = (event) => {
        event.preventDefault();
        const newBook = { id: new Date().getTime().toString(), title: bookTitle };
        setBooks((prevBooks) => {
          return [...prevBooks, newBook];
        });
        setIsModalOpen(true);
        setModalText("New book is added");
      };
    
      return (
        <div>
          <form onSubmit={handleSubmit}>
            <input
              type="text"
              placeholder="enter book title"
              value={bookTitle}
              onChange={handleAddBook}
              required
            />
            <button type="submit">Add Book</button>
          </form>
    
          {isModalOpen && <Modal modalText={modalText} />}
    
          <section className="books">
            {books.map((book) => {
              const { id, title } = book;
              return (
                <article key={id} className="book">
                  <h2>id: {id}</h2>
                  <p>title: {title}</p>
                  <button
                    onClick={(event) => {
                      handleDeleteBook(event, id);
                    }}
                  >
                    Delete
                  </button>
                </article>
              );
            })}
          </section>
        </div>
      );
    };
    
    export default App;
  • Code Example - 55 (with useReducer)

    import React, { useState, useReducer } from "react";
    
    // books, modalText, isModalOpen
    // add book - modalText
    // remove book - modalText
    const dummyBooks = [
      { id: 1, title: "book1" },
      { id: 2, title: "book2" },
    ];
    const Modal = ({ modalText }) => {
      return <p>{modalText}</p>;
    };
    
    // based on action type reducer will update the state & return the state
    const reducer = (state, action) => {
      // action object has action.type and action.payload
      switch (action.type) {
        case "ADD":
          const allBooks = [...state.books, action.payload];
          return {
            ...state,
            books: allBooks,
            isModalOpen: true,
            modalText: "new book is added",
          };
        case "DELETE":
          const filteredBooks = state.books.filter(
            (book) => book.id !== action.payload
          );
          return {
            ...state,
            books: filteredBooks,
            isModalOpen: true,
            modalText: "book is deleted",
          };
    
        default:
          return state;
      }
    };
    
    const initialState = {
      books: dummyBooks,
      isModalOpen: false,
      modalText: "",
    };
    
    const App = () => {
      // const [books, setBooks] = useState(dummyBooks);
      // const [modalText, setModalText] = useState('');
      // const [isModalOpen, setIsModalOpen] = useState(false);
    
      const [bookState, dispatch] = useReducer(reducer, initialState);
    
      const [bookTitle, setBookTitle] = useState("");
    
      const handleTitleChange = (event) => {
        event.stopPropagation();
        setBookTitle(event.target.value);
      };
      const handleDeleteBook = (event, id) => {
        event.stopPropagation();
        dispatch({ type: "DELETE", payload: id });
      };
    
      const handleSubmit = (event) => {
        event.preventDefault();
        const newBook = { id: new Date().getTime().toString(), title: bookTitle };
        dispatch({ type: "ADD", payload: newBook });
        setBookTitle("");
      };
    
      return (
        <div>
          <form onSubmit={handleSubmit}>
            <input
              type="text"
              placeholder="enter book title"
              value={bookTitle}
              onChange={handleTitleChange}
              required
            />
            <button type="submit">Add Book</button>
          </form>
    
          {bookState.isModalOpen && <Modal modalText={bookState.modalText} />}
    
          <section className="books">
            {bookState.books.map((book) => {
              const { id, title } = book;
              return (
                <article key={id} className="book">
                  <h2>id: {id}</h2>
                  <p>title: {title}</p>
                  <button
                    onClick={(event) => {
                      handleDeleteBook(event, id);
                    }}
                  >
                    Delete
                  </button>
                </article>
              );
            })}
          </section>
        </div>
      );
    };
    
    export default App;
// create few pages -> Home.js, About.js, Contact.js, Error.js
import React from "react";
import { BrowserRouter, Link, Route, Routes } from "react-router-dom";

import About from "./pages/About";
import Contact from "./pages/Contact";
import Home from "./pages/Home";
import Error from "./pages/Error";
const App = () => {
  return (
    <div>
      <BrowserRouter>
        <Routes>
          <Route path="/" element={<Home />} />
          <Route path="/about" element={<About />} />
          <Route path="/contact" element={<Contact />} />
          <Route path="*" element={<Error />} />
        </Routes>
      </BrowserRouter>
    </div>
  );
};

export default App;
  • Code Example - 57 (basic routing with Navbar)
import React from "react";
import { BrowserRouter, Link, Route, Routes } from "react-router-dom";

import About from "./pages/About";
import Contact from "./pages/Contact";
import Home from "./pages/Home";
import Error from "./pages/Error";

const App = () => {
  return (
    <div>
      <BrowserRouter>
        <nav>
          <Link to="/" className="nav__link">
            Home
          </Link>
          <Link to="/about" className="nav__link">
            About
          </Link>
          <Link to="/contact" className="nav__link">
            Contact
          </Link>
        </nav>
        <Routes>
          <Route path="/" element={<Home />} />
          <Route path="/about" element={<About />} />
          <Route path="/contact" element={<Contact />} />
          <Route path="*" element={<Error />} />
        </Routes>
      </BrowserRouter>
    </div>
  );
};

export default App;

// create a separate Navbar.js component inside layout folder

// Active Link
import React from 'react';
import { NavLink } from 'react-router-dom';

const Navbar = () => {
    return (
        <nav>
            <NavLink to="/" className="nav__link">
                Home
            </NavLink>
            <NavLink to="/about" className="nav__link">
                About
            </NavLink>
            <NavLink to="/contact" className="nav__link">
                Contact
            </NavLink>
        </nav>
    );
};

export default Navbar;
.active {
    color: orange;
}
  • Code Example - 58 (navigate with useNavigate)
import React from "react";
import { useNavigate } from "react-router-dom";

const Error = () => {
  const navigate = useNavigate();

  return (
    <div>
      <h2>Error Page</h2>
      <button
        onClick={() => {
          navigate("/");
        }}
      >
        Go to Home Page
      </button>
    </div>
  );
};

export default Error;
  • Code Example - 59 (dynamic routing)
  • based on the parameters show dynamic data in a page
  • step 1: create some dummy data
export const blogsData = [
  {
    id: 1,
    title: "html",
    body: "Lorem ipsum dolor sit amet consectetur adipisicing elit. Saepe doloremque impedit cumque aperiam dignissimos cum eveniet reiciendis iste asperiores voluptates repudiandae et, deserunt nobis commodi hic! Ratione accusamus assumenda alias rerum, esse et aperiam nihil sunt amet nobis quod libero sapiente! Harum reiciendis quos tempora blanditiis recusandae impedit iusto ipsum?",
  },
  {
    id: 2,
    title: "css",
    body: "Lorem ipsum dolor sit amet consectetur adipisicing elit. Saepe doloremque impedit cumque aperiam dignissimos cum eveniet reiciendis iste asperiores voluptates repudiandae et, deserunt nobis commodi hic! Ratione accusamus assumenda alias rerum, esse et aperiam nihil sunt amet nobis quod libero sapiente! Harum reiciendis quos tempora blanditiis recusandae impedit iusto ipsum?",
  },
  {
    id: 3,
    title: "js",
    body: "Lorem ipsum dolor sit amet consectetur adipisicing elit. Saepe doloremque impedit cumque aperiam dignissimos cum eveniet reiciendis iste asperiores voluptates repudiandae et, deserunt nobis commodi hic! Ratione accusamus assumenda alias rerum, esse et aperiam nihil sunt amet nobis quod libero sapiente! Harum reiciendis quos tempora blanditiis recusandae impedit iusto ipsum?",
  },
  {
    id: 4,
    title: "react.js",
    body: "Lorem ipsum dolor sit amet consectetur adipisicing elit. Saepe doloremque impedit cumque aperiam dignissimos cum eveniet reiciendis iste asperiores voluptates repudiandae et, deserunt nobis commodi hic! Ratione accusamus assumenda alias rerum, esse et aperiam nihil sunt amet nobis quod libero sapiente! Harum reiciendis quos tempora blanditiis recusandae impedit iusto ipsum?",
  },
];
  • step 2: load and map the data in Blogs.js page
import React, { useState } from "react";
import { Link } from "react-router-dom";

import { blogsData } from "../data";
const Blogs = () => {
  const [blogs, setBlogs] = useState(blogsData);

  const truncateString = (str, number) => {
    if (str.length > number) {
      return str.slice(0, number) + "...";
    } else return str;
  };

  const renderBlogsElement = blogs.map((blog) => {
    return (
      <article key={blog.id} className="blog">
        <h3>{blog.title}</h3>
        <p>{truncateString(blog.body, 100)}</p>
        <Link className="link" to={blog.title}>
          Learn More
        </Link>
      </article>
    );
  });

  return (
    <div>
      <h2>Blogs Page</h2>
      <section className="blogs">{renderBlogsElement}</section>
    </div>
  );
};

export default Blogs;
// blogs/html -> HTML.JS -> Blog.JS
// blogs/css -> CSS.JS -> Blog.JS
// blogs/js -> JS.JS -> Blog.JS
// blogs/react -> React.JS -> Blog.JS
  • step 3: set and get parameter
// in route setup
<Route path="/blogs/:title" element={<Blog />} />;

// where we wish to recieve the dynamic params
import React, { useEffect, useState } from "react";
import { useParams } from "react-router-dom";
import { blogsData } from "../data";

const Blog = () => {
  const { title } = useParams();
  const [blogs, setBlogs] = useState(blogsData);
  const [singleBlog, setSingleBlog] = useState();

  useEffect(() => {
    const blogData = blogs.find((blog) => blog.title === title);
    setSingleBlog(blogData);
  }, [blogs, title]);

  console.log(singleBlog);

  return (
    <div>
      <div>
        <h2>{title}</h2>
        {singleBlog && <p>{singleBlog.body}</p>}
      </div>
    </div>
  );
};

export default Blog;
  • Code Example - 60 (pass data using useLocation hook)
// what do we have inside useLocation
// we have state for passing data
// Blog.js
import { useParams, useLocation } from "react-router-dom";
const Blog = () => {
  const location = useLocation();
  console.log(location);
};

//pass data using state in Link
<Link className="link" to={title} state={{ id, title, body }}>
  Learn More
</Link>;

import React from "react";
import { useParams, useLocation } from "react-router-dom";

const Blog = () => {
  const location = useLocation();
  const { title } = useParams();

  return (
    <div>
      <div>
        <h2>{title}</h2>
        {location.state.body && <p>{location.state.body}</p>}
      </div>
    </div>
  );
};

export default Blog;
  • Code Example - 61 (protected routing)
// Protected.js
import React from "react";
import { Navigate } from "react-router-dom";

const Protected = ({ isLoggedIn, children }) => {
  if (!isLoggedIn) {
    return <Navigate to="/" replace />;
  }
  return children;
};

export default Protected;

// index.js - routing
import React, { useState } from 'react';
import { BrowserRouter, Route, Routes } from 'react-router-dom';
import Navbar from '../layout/Navbar';

import About from '../pages/About';
import Blog from '../pages/Blog';
import Blogs from '../pages/Blogs';
import Contact from '../pages/Contact';
import Error from '../pages/Error';
import Home from '../pages/Home';
import Protected from './Protected';

const Index = () => {
    const [isLoggedIn, setIsLoggedIn] = useState(false);
    return (
        <BrowserRouter>
            <Navbar />
            <button
                onClick={() => {
                    setIsLoggedIn(!isLoggedIn);
                }}>
                {isLoggedIn ? 'logout' : 'login'}
            </button>
            <Routes>
                <Route path="/" element={<Home />} />
                <Route
                    path="/blogs"
                    element={
                        <Protected isLoggedIn={isLoggedIn}>
                            <Blogs />
                        </Protected>
                    }
                />
                <Route path="/blogs/:title" element={<Blog />} />
                <Route path="/about" element={<About />} />
                <Route path="/contact" element={<Contact />} />
                <Route path="*" element={<Error />} />
            </Routes>
        </BrowserRouter>
    );
};

export default Index;
  • running a json server in react -> npm i json-server && npx json-server -p 3001 -w database/db.json

[2.6 Optimization: React.memo(), useCallback(), useMemo()]

  • components re-render when state or props changes

  • A good reference: https://dmitripavlutin.com/use-react-memo-wisely/

  • It helps to avoid unnecessary components rendering

         // App.js
         import React, { useState } from 'react';
          import Message from './components/Message';
    
          const App = () => {
            const [count, setCount] = useState(0);
            console.log('app rendering');
            return (
              <div>
                <h1>Welcome to Memo</h1>
                <h2>Count : {count}</h2>
                <button
                  onClick={() => {
                    setCount((count) => count + 1);
                  }}>
                  Increment
                </button>
                <Message numberOfMessages={0} />
              </div>
            );
          };
    
          export default App;
    
      // Message.js
      import React, { memo } from 'react';
      import PropTypes from 'prop-types';
      const Message = (props) => {
        console.log('message rendering');
        return <div>Message {props.numberOfMessages}</div>;
      };
    
      Message.propTypes = {
        numberOfMessages: PropTypes.number
      };
    
      export default memo(Message);
  • It helps to avoid unnecessary components rendering for defining callback methods

  • only component will be rendered when some states or props change

  • callback function definition re-rendering the componenet unnecessarily

         // App.js
         import React, { useState, useCallback, useMemo } from 'react';
          import Message from './components/Message';
    
          const App = () => {
            const [count, setCount] = useState(0);
            console.log('app component rendering');
    
            const handleDataFromMessage = useCallback((data) => {
              console.log(data);
            }, []);
    
            return (
              <div>
                <h2>{numbers}</h2>
                <h2>Count : {count}</h2>
                <button
                  onClick={() => {
                    setCount((count) => count + 1);
                  }}>
                  Increment
                </button>
                <Message onHandleDataFromMessage={handleDataFromMessage} />
              </div>
            );
          };
    
          export default App;
    
    
      // Message.js
      import React from 'react';
      import PropTypes from 'prop-types';
      const Message = ({ onHandleDataFromMessage }) => {
        console.log('message component rendering');
        onHandleDataFromMessage('hello I am from message component');
        return <div>Message</div>;
      };
    
      Message.propTypes = {
        onHandleDataFromMessage: PropTypes.func
      };
    
      export default React.memo(Message);
  • It helps to avoid taking unnecessary time for same kind of complex calculation for each rendering

         // App.js
         import React, { useState, useCallback, useMemo } from 'react';
          import Message from './components/Message';
    
          const App = () => {
            const [count, setCount] = useState(0);
            console.log('app component rendering');
    
            const numbers = useMemo(() => {
              let number = 0;
              for (let index = 0; index < 5000000000; index++) {
                number++;
              }
              return number;
            }, []);
    
            return (
              <div>
                <h2>{numbers}</h2>
                <h2>Count : {count}</h2>
                <button
                  onClick={() => {
                    setCount((count) => count + 1);
                  }}>
                  Increment
                </button>
              </div>
            );
          };
    
          export default App;
    
    
      // Message.js
      import React from 'react';
      import PropTypes from 'prop-types';
      const Message = ({ onHandleDataFromMessage }) => {
        console.log('message component rendering');
        onHandleDataFromMessage('hello I am from message component');
        return <div>Message</div>;
      };
    
      Message.propTypes = {
        onHandleDataFromMessage: PropTypes.func
      };
    
      export default React.memo(Message);

[2.7 props drilling, useContext Hook]

import React, { useState } from "react";

import Component1 from "./components/Component1";

const App = () => {
  const [user, setUser] = useState({ id: 1, name: "anisul islam" });
  return (
    <div>
      <Component1 user={user} />
    </div>
  );
};

export default App;

import React from 'react';
import Component2 from './Component2';

const Component1 = ({ user }) => {
    return (
        <div>
            <Component2 user={user} />
        </div>
    );
};

export default Component1;

import React from 'react';
import Component3 from './Component3';

const Component2 = ({ user }) => {
    return (
        <div>
            <Component3 user={user} />
        </div>
    );
};

export default Component2;

import React from 'react';
import Component4 from './Component4';

const Component3 = ({ user }) => {
    return (
        <div>
            <Component4 user={user} />
        </div>
    );
};

export default Component3;

import React from 'react';

const Component4 = ({ user }) => {
    console.log(user);
    return (
        <div>
            <h2>Component 4</h2>
            <h3>{user.id}</h3>
            <p>{user.name}</p>
        </div>
    );
};

export default Component4;
  • it helps us to create component level, app level or global states. It also helps us to avoid state lifting.
  • Context lets the parent component make some information available to any component in the tree below it—no matter how deep—without passing it explicitly through props. (collected from react documentation)
  • use cases of context: theming, routing, managing states
  • step 1: create a context
import React from "react";

// we can pass any value or object in createContext()
export const UserContext = React.createContext(); // it allows us to use Provider and consumer (we will use useContext instead of consumer)
  • step 2: provide the context - wrap the parent with the context
import React, { useState } from "react";

import Component1 from "./components/Component1";
import { UserContext } from "./UserContext";

const App = () => {
  const [user, setUser] = useState({ id: 1, name: "anisul islam" });
  const text = "hello everyone!";
  return (
    <div>
      <UserContext.Provider value={{ user, text }}>
        <Component1 />
      </UserContext.Provider>
    </div>
  );
};

export default App;
  • step 3: access the context with useContext
import React, { useContext } from "react";
import { UserContext } from "../UserContext";

const Component4 = () => {
  const { user } = useContext(UserContext);

  return (
    <div>
      <h2>Component 4</h2>
      <h3>{user.id}</h3>
      <p>{user.name}</p>
    </div>
  );
};

export default Component4;
  • Another example
// context/UserContext.js
import { createContext } from 'react';
export const UserContext = createContext({});

// App.js
import React, { useState } from "react";
import NewUser from "./components/NewUser";

import Users from "./components/Users";
import { UserContext } from "./context/UserContext";

const App = () => {
  const [users, setUsers] = useState([{ id: 1, name: "anisul islam" }]);

  return (
    <div>
      <UserContext.Provider value={{ users, setUsers }}>
        <NewUser />
        <Users />
      </UserContext.Provider>
    </div>
  );
};

export default App;

// adding useMemo and modify
import React, { useState, useMemo } from 'react';
import NewUser from './components/NewUser';

import Users from './components/Users';
import { UserContext } from './context/UserContext';

const App = () => {
    const [users, setUsers] = useState([{ id: 1, name: 'anisul islam' }]);

    const providerValue = useMemo(() => ({ users, setUsers }), [users, setUsers]);

    return (
        <div>
            <UserContext.Provider value={providerValue}>
                <NewUser />
                <Users />
            </UserContext.Provider>
        </div>
    );
};

export default App;


// Users.js
import React, { useContext } from 'react';
import { UserContext } from '../context/UserContext';

import User from './User';

const Users = () => {
    const { users } = useContext(UserContext);
    return (
        <section>
            <h2>Users Management</h2>
            {users.map((user) => (
                <User key={user.id} {...user} />
            ))}
        </section>
    );
};

export default Users;

// User.js
import React from 'react';

const User = ({ ...user }) => {
    const { id, name } = user;

    return (
        <article>
            <h3>ID: {id}</h3>
            <p>Name: {name}</p>
        </article>
    );
};

export default User;

// NewUser.js
import React, { useState, useContext } from 'react';
import { UserContext } from '../context/UserContext';

const NewUser = () => {
    const [name, setName] = useState('');
    const { setUsers } = useContext(UserContext);

    const handleSubmit = (event) => {
        event.preventDefault();
        const newUser = {
            id: new Date().getTime().toString(),
            name: name
        };
        setUsers((prevUsers) => {
            return [...prevUsers, newUser];
        });
        // alert(JSON.stringify(newUser, null, 4));
        setName('');
    };

    const handleNameChange = (event) => {
        event.stopPropagation();
        setName(event.target.value);
    };

    return (
        <div>
            <h2>Create New User</h2>
            <form onSubmit={handleSubmit}>
                <input type="text" placeholder="Enter name here" onChange={handleNameChange} value={name} />
                <button type="submit">Add New User</button>
            </form>
        </div>
    );
};

export default NewUser;

[Theme change project using useContext]

import React, { useState, useContext } from 'react';
import { ThemeContext } from './context/ThemeContext';

const App = () => {
    const [theme, setTheme] = useState('dark');
    // const themeContext = useContext(ThemeContext);
    // console.log({ themeContext });

    const toggleTheme = () => {
        setTheme(theme === 'dark' ? 'light' : 'dark');
    };
    return (
        <ThemeContext.Provider value={theme}>
            <div className={`App ${theme}`}>
                <h1>welcome</h1>
                <button onClick={toggleTheme}>Change Theme</button>
            </div>
        </ThemeContext.Provider>
    );
};

export default App;


.dark {
    background-color: #000;
    color: #fff;
}
.light {
    background-color: #fff;
    color: #000;
}

[useReducer + useContext]

// reducer/bookReducer.js
export const initialState = {
  books: [
    {
      id: 1,
      title: "book1",
    },
    {
      id: 2,
      title: "book2",
    },
  ],
};

export const reducer = (state, action) => {
  switch (action.type) {
    case "ADD":
      return state;
    case "DELETE":
      const filteredBooks = state.books.filter(
        (book) => book.id !== action.payload
      );
      return {
        ...state,
        books: filteredBooks,
      };

    default:
      return state;
  }
};

// context/BookContext.js
import { createContext } from "react";
const BookContext = createContext(null);

export default BookContext;

// hooks/useBookContext.js
// custom hook for using context
import { useContext } from "react";
import BookContext from "../context/BookContext";

export const useBookContext = () => {
  return useContext(BookContext);
};


// App.js
import React, { useReducer, useState } from "react";

import "./App.css";
import Books from "./components/Books";
import BookContext from "./context/BookContext";
import { initialState, reducer } from "./reducer/booksReducer";

function App() {
  const [state, dispatch] = useReducer(reducer, initialState);
  return (
    <div className="App">
        <BookContext.Provider value={{ state, dispatch }}>
          <Books />
        </BookContext.Provider>
    </div>
  );
}

export default App;

// Books.js
import React, { useContext } from "react";
import BookContext from "../context/BookContext";
import { useThemeContext } from "../context/ThemeContext";
import { useBookContext } from "../hooks/useBookContext";

const Books = () => {
  const context = useThemeContext();
  const { state, dispatch } = useBookContext();
  // const handleDeleteBook = (id) => {
  //   const filteredBooks = books.filter((book) => book.id !== id);
  //   setBooks(filteredBooks);
  // };
  return (
    <div className={context}>
      {state.books &&
        state.books.map((book) => {
          const { id, title } = book;
          return (
            <article key={id}>
              <h2>{id}</h2>
              <p>{title}</p>
              <button>edit book</button>
              <button
                onClick={() => {
                  dispatch({ type: "DELETE", payload: id });
                }}
              >
                delete book
              </button>
            </article>
          );
        })}
    </div>
  );
};

export default Books;

Part-10 (useState, useEffect, useReducer, useContext, custom hook practice)

[57. user mgt crud app using useState]

  • version 1: only useState, useEffect
//App.js
import React, { useState } from "react";

import Users from "./components/Users";
import "./App.css";
import NewUser from "./components/NewUser";

const App = () => {
  const [users, setUsers] = useState([
    {
      id: 1,
      username: "anisul",
    },
    {
      id: 2,
      username: "sakib",
    },
  ]);
  const [editableUser, setEditableUser] = useState(null);

  const addNewUser = (user) => {
    setUsers((prevUsers) => [...prevUsers, user]);
  };

  const editUser = (user) => {
    setEditableUser(user);
  };

  const deleteUser = (id) => {
    const filteredUsers = users.filter((user) => user.id !== id);
    setUsers(filteredUsers);
  };

  const updateUser = (updatedUser) => {
    console.log(updatedUser);
    const userIndex = users.findIndex((user) => user.id === updatedUser.id);
    const copiedUsers = [...users];
    copiedUsers.splice(userIndex, 1, updatedUser);
    setUsers(copiedUsers);
  };

  return (
    <div>
      <NewUser
        onAddNewUser={addNewUser}
        editableUser={editableUser}
        onUpdateUser={updateUser}
      />
      {users && (
        <Users users={users} onDeleteUser={deleteUser} onEditUser={editUser} />
      )}
    </div>
  );
};

export default App;

// components/NewUser.js
import React, { useState, useEffect } from "react";
import { v4 as uuidv4 } from "uuid";

const NewUser = ({ onAddNewUser, editableUser, onUpdateUser }) => {
  const [username, setUsername] = useState("");

  const handleSubmit = (event) => {
    event.preventDefault();

    if (editableUser) {
      const updatedUser = { id: editableUser.id, username };
      onUpdateUser(updatedUser);
    } else {
      const newUser = { id: uuidv4(), username };
      onAddNewUser(newUser);
      setUsername("");
    }
  };

  useEffect(() => {
    editableUser && setUsername(editableUser.username);
  }, [editableUser]);

  return (
    <div className="new-user">
      <h2>Register</h2>
      <form onSubmit={handleSubmit}>
        <input
          type="text"
          name="username"
          value={username}
          placeholder="Enter username"
          onChange={(e) => {
            setUsername(e.target.value);
          }}
          required
        />
        <button type="submit">{editableUser ? "Edit User" : "Add User"}</button>
      </form>
    </div>
  );
};

export default NewUser;

// components/Users.js
import React from "react";
import User from "./User";

const Users = ({ users, onDeleteUser, onEditUser }) => {
  return (
    <section className="users">
      {users.map((user) => (
        <User
          key={user.id}
          user={user}
          onDeleteUser={onDeleteUser}
          onEditUser={onEditUser}
        />
      ))}
    </section>
  );
};

export default Users;


// components/User.js
import React from "react";

const User = ({ user, onDeleteUser, onEditUser }) => {
  const { id, username } = user;

  const handleDelete = (id) => {
    onDeleteUser(id);
  };

  const handleEdit = (user) => {
    onEditUser(user);
  };

  return (
    <article className="user">
      <h2>{id}</h2>
      <p>{username}</p>
      <button
        onClick={() => {
          handleEdit(user);
        }}
      >
        Edit
      </button>
      <button
        onClick={() => {
          handleDelete(id);
        }}
      >
        Delete
      </button>
    </article>
  );
};

export default User;
  • version 2 (useReducer added)
import React, { useState, useReducer } from "react";

import Users from "./components/Users";
import "./App.css";
import NewUser from "./components/NewUser";

const initialState = {
  users: [
    {
      id: 1,
      username: "anisul",
    },
    {
      id: 2,
      username: "sakib",
    },
  ],
};

const reducer = (state, action) => {
  switch (action.type) {
    case "ADD_USER":
      return {
        ...state,
        users: [...state.users, action.payload],
      };
    case "DELETE_USER":
      const filteredUsers = state.users.filter(
        (user) => user.id !== action.payload
      );
      return {
        ...state,
        users: filteredUsers,
      };
    case "UPDATE_USER":
      const userIndex = state.users.findIndex(
        (user) => user.id === action.payload.id
      );
      const copiedUsers = [...state.users];
      copiedUsers.splice(userIndex, 1, action.payload);
      return {
        ...state,
        users: copiedUsers,
      };

    default:
      return state;
  }
};

const App = () => {
  const [state, dispatch] = useReducer(reducer, initialState);
  const [editableUser, setEditableUser] = useState(null);

  const addNewUser = (user) => {
    dispatch({ type: "ADD_USER", payload: user });
  };

  const editUser = (user) => {
    setEditableUser(user);
  };

  const deleteUser = (id) => {
    dispatch({ type: "DELETE_USER", payload: id });
  };

  const updateUser = (updatedUser) => {
    dispatch({ type: "UPDATE_USER", payload: updatedUser });
  };

  return (
    <div>
      <NewUser
        onAddNewUser={addNewUser}
        editableUser={editableUser}
        onUpdateUser={updateUser}
      />
      {state.users && (
        <Users
          users={state.users}
          onDeleteUser={deleteUser}
          onEditUser={editUser}
        />
      )}
    </div>
  );
};

export default App;

// separte the reducer with initial states and exports them to use from the App.js
  • version 3 (with useContext)
// UsersContext.js
import { createContext } from "react";
export const UsersContext = createContext(null);

// App.js
import React, { useState, useReducer } from "react";

import Users from "./components/Users";
import "./App.css";
import NewUser from "./components/NewUser";
import { initialState, reducer } from "./reducer/userReducer";
import { UsersContext } from "./context/UsersContext";

const App = () => {
  const [state, dispatch] = useReducer(reducer, initialState);
  const [editableUser, setEditableUser] = useState(null);

  return (
    <div>
      <UsersContext.Provider
        value={{ state, dispatch, editableUser, setEditableUser }}
      >
        <NewUser />
        {state.users && <Users />}
      </UsersContext.Provider>
    </div>
  );
};

export default App;

// Users.js
import React, { useContext } from "react";
import { UsersContext } from "../context/UsersContext";

const User = ({ user }) => {
  const { id, username } = user;
  const { dispatch, setEditableUser } = useContext(UsersContext);

  const handleDelete = (id) => {
    dispatch({ type: "DELETE_USER", payload: id });
  };

  const handleEdit = (user) => {
    setEditableUser(user);
  };

  return (
    <article className="user">
      <h2>{id}</h2>
      <p>{username}</p>
      <button
        onClick={() => {
          handleEdit(user);
        }}
      >
        Edit
      </button>
      <button
        onClick={() => {
          handleDelete(id);
        }}
      >
        Delete
      </button>
    </article>
  );
};

export default User;

// User.js
import React, { useContext } from "react";
import { UsersContext } from "../context/UsersContext";

const User = ({ user }) => {
  const { id, username } = user;
  const { dispatch, setEditableUser } = useContext(UsersContext);

  const handleDelete = (id) => {
    dispatch({ type: "DELETE_USER", payload: id });
  };

  const handleEdit = (user) => {
    setEditableUser(user);
  };

  return (
    <article className="user">
      <h2>{id}</h2>
      <p>{username}</p>
      <button
        onClick={() => {
          handleEdit(user);
        }}
      >
        Edit
      </button>
      <button
        onClick={() => {
          handleDelete(id);
        }}
      >
        Delete
      </button>
    </article>
  );
};

export default User;

// NewUser.js
import React, { useState, useEffect, useContext } from "react";
import { v4 as uuidv4 } from "uuid";
import { UsersContext } from "../context/UsersContext";

const NewUser = () => {
  const { dispatch, editableUser } = useContext(UsersContext);
  const [username, setUsername] = useState("");

  const handleSubmit = (event) => {
    event.preventDefault();

    if (editableUser) {
      const updatedUser = { id: editableUser.id, username };
      dispatch({ type: "UPDATE_USER", payload: updatedUser });
    } else {
      const newUser = { id: uuidv4(), username };
      dispatch({ type: "ADD_USER", payload: newUser });
      setUsername("");
    }
  };

  useEffect(() => {
    editableUser && setUsername(editableUser.username);
  }, [editableUser]);

  return (
    <div className="new-user">
      <h2>Register</h2>
      <form onSubmit={handleSubmit}>
        <input
          type="text"
          name="username"
          value={username}
          placeholder="Enter username"
          onChange={(e) => {
            setUsername(e.target.value);
          }}
          required
        />
        <button type="submit">{editableUser ? "Edit User" : "Add User"}</button>
      </form>
    </div>
  );
};

export default NewUser;
  • version 4 (custom hook for useContext)
// hooks/useUsersContext.js
import { useContext } from "react";

import { UsersContext } from "../context/UsersContext";

export const useUserContext = () => {
  return useContext(UsersContext);
};
  • version 5 (creating Store.js)
import React, { useState, useReducer } from "react";
import { UsersContext } from "../context/UsersContext";
import { initialState, reducer } from "../reducer/userReducer";

const Store = ({ children }) => {
  const [state, dispatch] = useReducer(reducer, initialState);
  const [editableUser, setEditableUser] = useState(null);

  const value = {
    users: state.users,
    addUser: (newUser) => {
      dispatch({ type: "ADD_USER", payload: newUser });
    },
    deleteUser: (id) => {
      dispatch({ type: "DELETE_USER", payload: id });
    },
    updateUser: (updatedUser) => {
      dispatch({ type: "UPDATE_USER", payload: updatedUser });
    },
    editableUser,
    setEditableUser,
  };

  return (
    <UsersContext.Provider value={value}>{children}</UsersContext.Provider>
  );
};

export default Store;

// provide the Store.js
import React from "react";

import Users from "./components/Users";
import "./App.css";
import NewUser from "./components/NewUser";
import Store from "./store/Store";

const App = () => {
  return (
    <div>
      <Store>
        <NewUser />
        <Users />
      </Store>
    </div>
  );
};

export default App;

// use the Store.js
// Users.js
import React from "react";
import { useUserContext } from "../hooks/useUsersContext";
import User from "./User";

const Users = () => {
  const { users } = useUserContext();

  return (
    <section className="users">
      {users.map((user) => (
        <User key={user.id} user={user} />
      ))}
    </section>
  );
};

export default Users;

// User.js
import React from "react";

import { useUserContext } from "../hooks/useUsersContext";

const User = ({ user }) => {
  const { id, username } = user;
  const { deleteUser, setEditableUser } = useUserContext();

  const handleDelete = (id) => {
    deleteUser(id);
  };

  const handleEdit = (user) => {
    setEditableUser(user);
  };

  return (
    <article className="user">
      <h2>{id}</h2>
      <p>{username}</p>
      <button
        onClick={() => {
          handleEdit(user);
        }}
      >
        Edit
      </button>
      <button
        onClick={() => {
          handleDelete(id);
        }}
      >
        Delete
      </button>
    </article>
  );
};

export default User;

//NewUser.js
import React, { useState, useEffect } from "react";
import { v4 as uuidv4 } from "uuid";

import { useUserContext } from "../hooks/useUsersContext";

const NewUser = () => {
  const { addUser, updateUser, editableUser } = useUserContext();
  const [username, setUsername] = useState("");

  const handleSubmit = (event) => {
    event.preventDefault();

    if (editableUser) {
      const updatedUser = { id: editableUser.id, username };
      updateUser(updatedUser);
    } else {
      const newUser = { id: uuidv4(), username };
      addUser(newUser);
      setUsername("");
    }
  };

  useEffect(() => {
    editableUser && setUsername(editableUser.username);
  }, [editableUser]);

  return (
    <div className="new-user">
      <h2>Register</h2>
      <form onSubmit={handleSubmit}>
        <input
          type="text"
          name="username"
          value={username}
          placeholder="Enter username"
          onChange={(e) => {
            setUsername(e.target.value);
          }}
          required
        />
        <button type="submit">{editableUser ? "Edit User" : "Add User"}</button>
      </form>
    </div>
  );
};

export default NewUser;
// context/MoviesContext.js
import { createContext } from "react";

export const MoviesContext = createContext([]);

// store/Store.js
import React, { useEffect, useState } from "react";
import axios from "axios";
import { MoviesContext } from "./MoviesContext";

const URL = "https://my.api.mockaroo.com/movies.json?key=c5bae7e0";
const Store = ({ children }) => {
  const [movies, setMovies] = useState([]);

  const fetchMovies = async () => {
    const result = await axios.get(URL);
    setMovies(result.data);
  };
  useEffect(() => {
    fetchMovies();
  }, []);

  return (
    <MoviesContext.Provider value={{ movies, setMovies }}>
      {children}
    </MoviesContext.Provider>
  );
};

export default Store;


// custom hook for useContext
import { useContext } from "react";

import { MoviesContext } from "../context/MoviesContext";

export const useMovieContext = () => {
  return useContext(MoviesContext);
};

// App.js
import React from "react";
import { useMovieContext } from "./hooks/useMoviesContext";

const App = () => {
  const { movies } = useMovieContext();
  return (
    <section className="movies">
      {movies &&
        movies.map((movie) => {
          return (
            <article key={movie.id} className="movie">
              <h3>{movie.id}</h3>
              <p>{movie.title}</p>
            </article>
          );
        })}
    </section>
  );
};

export default App;

Part-11 (redux, redex toolkit)

  • redux documentation
  • redux = useContext + useReducer
  • check redux videos and then redux-toolkit
  • how to use redux devtools
//features/users/usersSlice.js
import { createAsyncThunk, createSlice } from "@reduxjs/toolkit";
import axios from "axios";

// action creator for fetching data
export const fetchUsers = createAsyncThunk(
  "users/fetchUsers",
  async (_, thunkAPI) => {
    const { rejectWithValue } = thunkAPI;
    try {
      const response = await axios.get("http://localhost:3001/users");
      return response.data;
    } catch (error) {
      return rejectWithValue(error.message);
    }
  }
);

// action creator for deleteing data
export const deleteUser = createAsyncThunk(
  "users/deleteUser",
  async (data, thunkAPI) => {
    const { rejectWithValue } = thunkAPI;
    try {
      const response = await axios.delete(
        `http://localhost:3001/users/${data}`
      );
      return response.data;
    } catch (error) {
      return rejectWithValue(error.message);
    }
  }
);

// action creator for deleteing data
export const createUser = createAsyncThunk(
  "users/createUser",
  async (data, thunkAPI) => {
    const { rejectWithValue } = thunkAPI;
    try {
      const response = await axios.post(`http://localhost:3001/users}`, data);
      return response.data;
    } catch (error) {
      return rejectWithValue(error.message);
    }
  }
);

const usersSlice = createSlice({
  name: "users",
  initialState: {
    isLoading: false,
    error: null,
    users: [],
  },
  reducers: {},

  // extraReducers will handle the asynchronous promise states: pending, fulfilled or reject
  extraReducers: (builder) => {
    // fetchUsers
    builder.addCase(fetchUsers.pending, (state) => {
      state.isLoading = true;
      state.error = null;
    });
    builder.addCase(fetchUsers.fulfilled, (state, action) => {
      state.users = action.payload;
      state.isLoading = false;
    });
    builder.addCase(fetchUsers.rejected, (state, action) => {
      state.error = action.payload;
      state.isLoading = false;
    });

    // deleteUser
    builder.addCase(deleteUser.pending, (state, action) => {
      state.isLoading = true;
      state.error = null;
    });
    builder.addCase(deleteUser.fulfilled, (state, action) => {
      state.users = state.users.filter((user) => user.id !== action.payload.id);
      state.isLoading = false;
    });
    builder.addCase(deleteUser.rejected, (state, action) => {
      state.error = action.payload;
      state.isLoading = false;
    });

    // addUser
    builder.addCase(createUser.pending, (state, action) => {
      state.isLoading = true;
      state.error = null;
    });
    builder.addCase(createUser.fulfilled, (state, action) => {
      state.users.push(action.payload);
      state.isLoading = false;
    });
    builder.addCase(createUser.rejected, (state, action) => {
      state.error = action.payload;
      state.isLoading = false;
    });
  },
});

export default usersSlice.reducer;



// app/store.js
import { configureStore } from "@reduxjs/toolkit";

import counterReducer from "../features/counter/counterSlice";
import usersReducer from "../features/users/usersSlice";

export const store = configureStore({
  reducer: {
    counterR: counterReducer,
    usersR: usersReducer,
  },
});


// features/Users.js
import React, { useEffect, useCallback } from "react";
import { useDispatch, useSelector } from "react-redux";
import { deleteUser, fetchUsers } from "./usersSlice";

const Users = () => {
  const { isLoading, users, error } = useSelector((state) => state.usersR);
  const dispatch = useDispatch();

  const fetchAllUsers = useCallback(() => {
    dispatch(fetchUsers());
  }, [dispatch]);

  useEffect(() => {
    fetchAllUsers();
  }, [fetchAllUsers]);

  const deleteUserById = (id) => {
    dispatch(deleteUser(id)).then((res) => console.log(res));
    fetchAllUsers();
  };

  if (isLoading) {
    return <p>Loading Users...</p>;
  }
  if (error) {
    return <p>{error}</p>;
  }

  return (
    <div>
      {users &&
        users.map((user) => {
          return (
            <article key={user.id}>
              <h2>{user.id}</h2>
              <h2>{user.name}</h2>
              <button
                onClick={() => {
                  deleteUserById(user.id);
                }}
              >
                delete
              </button>
            </article>
          );
        })}
    </div>
  );
};

export default Users;
  • Example 1
// features/apiSlice
import { createApi, fetchBaseQuery } from "@reduxjs/toolkit/query/react";

export const productsApi = createApi({
  reducerPath: "productsApi",
  baseQuery: fetchBaseQuery({ baseUrl: "https://dummyjson.com/" }),
  endpoints: (builder) => ({
    getAllProducts: builder.query({
      query: () => "products",
    }),
    getProduct: builder.query({
      query: (product) => `products/search?q=${product}`,
    }),
  }),
});

// created for us
export const { useGetAllProductsQuery, useGetProductQuery } = productsApi;

// App.js
import React from "react";
import { Provider } from "react-redux";
import { store } from "./app/store";
import Data from "./components/Data";
import { ApiProvider } from "@reduxjs/toolkit/query/react";
import { productsApi } from "./features/apiSlice";
const App = () => {
  return (
    <Provider store={store}>
      <ApiProvider api={productsApi}>
        <div>
          <Data />
        </div>
      </ApiProvider>
    </Provider>
  );
};

export default App;

// Data.js
import React from "react";
import {
  useGetAllProductsQuery,
  useGetProductQuery,
} from "../features/apiSlice";

const Data = () => {
  const {
    data: allProducts,
    error,
    isError,
    isLoading,
  } = useGetAllProductsQuery();
  const { data: product } = useGetProductQuery("iphone");

  if (isLoading) return <h1>Loading...</h1>;
  return <div>Data</div>;
};

export default Data;

#learnwithfair #rahtulrabbi #rahatul-rabbi #learn-with-fair

About

React-JS with [learnwithfair, Learn with fair, Rahatul Rabbi, Md Rahatul Rabbi ,rahatulrabbi]

Topics

Resources

Stars

Watchers

Forks

Releases

No releases published

Packages

No packages published