- prerequisities: HTML, CSS, Javascript
- React.js official Site: https://reactjs.org/
- React is a flexible, efficent Open Source UI Based JavaScript Library.
- It is developed by Facebook in 2013 for building user interface.
- 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
- Load fast - Why React.js is faster? - virtual DOM compares with previous states
- Allows us to use external plugin
- Example of React app - facebook, twitter, airbnb, netflix
- competitor - Vue.js, Angular (more full-fledged / developed no need 3rd party library just like react-router-dom)
- VSCode
- 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
-
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>
// create react app command
npx create-react-app appName
// run react app command
cd appName
npm start
- show how to run and stop the react app
- 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
Part-2 (JSX, Component, react component under the hood, Styling, Props, PropTypes, Conditional rendering, Fragment, react dev-tools, icons)
-
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
-
- 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> ); <>
-
-
- Remember to close all tags -
<img />
- Remember to close all tags -
-
- 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
-
Code Example - 14 (Styling component with CSS)
-
create and import App.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">©Copyright by Anisul Islam</p> </div> <div className="footer__right"> <p>social media icons</p> </div> </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
- default props
- 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;
-
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> );
-
Inline styling
<div style={{ width: "300px", backgroundColor: "pink" }}> Inline styling </div>
-
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;
-
-
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">©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
Part-3 (event handler, state,useState hook, controlled component, state lifting, more on css, class component)
-
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 )
-
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;
- we can also use name attribute for identifying element and use 1 function instead of many. from ebent handler we can use event.target.name and then decide what to do or not? - [1 handler for multiple elements] (https://github.com/anisul-Islam/react-counter-app-1-function/blob/master/src/components/Counter.js)
-
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); };
- update state based on prev value -
-
create a new resouce such as NewProduct, NewUser, NewTodo etc.
-
Code Example - 39 (get data from a form)
-
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
-
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: '[email protected]' }, ]); 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> ); } }
-
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;
-
Code Example - 49 (useEffect hook)
import React, { useEffect, useState } from "react"; // Rule: Don’t call Hooks inside loops, conditions, or nested functions // The useEffect Hook allows to perform side effects (fetching data, timers, directly manullay updating the DOM) in components. // useEffct = componentDidMount + componentDidUpdate + componentWillUnmount 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" ); 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;
-
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;
- react-routing-project
- react-router
- Code Example - 56 (basic routing)
// 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
- 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);
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;
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;
}
// 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;
- 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;
- 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;