[React Docs(https://react.dev/)
-
React 18 Tutorial and Projects Course - 2024 Edition - John smilga
-
MERN 2024 Edition - MongoDB, Express, React and NodeJS - John smilga
-
Mastering Next.js - Build Airbnb Clone from Scratch (2024) - John smilga
-
npx create-react-app .
-
JS Library because more flexible as compared to angular framework.
-
svelte and vue.js are more easier and straight forward compared to react.
- Whenever a state of the component is changed react compares the virtual and actual DOM and if there is a change it re-renders that part.
- Later in v19 compiler will replace the virtual DOM to make things faster.
- Reusable piece of code
- can be thought of as custom html elements
- can be either classes(older) and functional
- can get props passed in and can hold their own state
-
represents the data that a component manages internally.
-
input data, fetched data, UI related data like if a modal is open/close.
-
useState hook can be used to define state in the functional component.
-
there is also global state, as a whole not related to single component. (Context API, state management(redux))
-
External state management library like redux avoids the problem of prop driling, which can also be avoided by using context api for react but not take care of mulitple context or global states. useReducer hook solves this problem.
- allows us to use state and other react features within functional components without writing a class(they used to have lifecycle methods).
- useState, useEffect, useRef, useReducer
- useContext, useMemo, useCallback will be phased out in v19
- output of component.
- An HTML like syntax within JS(Components)
-
SPA - Load a single HTML file and JS loads the entire UI including routes.
-
SSR - Server sends the fully rendered page to client. You can fetch data and load it as well (Nextjs)
-
SSG - framework generates static HTML files at build time. These are very fast(Gatsby, Astro)
-
instead of using create-react-app, we use vite(npm create vite@latest)
npm create vite@latest .
git branch --move master main -
react developer tools in browser
-
difference in create react app and create vite@latest?
-
npm create vite@latest .
-
vite.config - server:{[ port:5000 ]}
-
npm install
-
npm run dev
-
index.html is the single page and everything else load using js.
-
there is a div inside the index.html file with id of root and in main jsx file we bring it in.
-
index.html - main.jsx - App.jsx
-
delete app.css
-
In main.jsx it is including the index.css file
-
use ES7 + react snipet extension in vscode
-
delete content of App.jsx - create a functional component(rafce) react arrow function component export, (rafc, rfc)
-
npm install -D tailwindcss postcss autoprefixer
-
npx tailwindcss init -p
-
In tailwind.config.js
/** @type {import('tailwindcss').Config} */
export default {
content: [
"./index.html",
"./src/**/*.{js,ts,jsx,tsx}",
],
theme: {
extend: {
fontFamily:{
sans: ['Roboto', 'sans-serif']
},
gridTemplateColumns: {
'70/30':'70% 28%'
}
},
},
plugins: [],
}
- In index.css
@tailwind base;
@tailwind components;
@tailwind utilities;
- tailwind plugin (npm i -D prettier prettier-plugin-tailwindcss)
- .prettierrc
{
"plugins":[prettier-plugin-tailwindcss]
}
- every react component is going to return JSX.
- it can only return a single element(one parent element).
return(
<>//Fragment (empty html tag)
<div className="text-5xl">App</div>
<>
)
- if you want to add dynamic variable
- if we want condition inside the jsx we can use ternary
const App = ()=>{
const names = ['John', 'Jane']
const loggedIn = true;
return(
<>//Fragment (empty html tag)
<div className="text-5xl">App</div>
<ul>
{names.map((name, index)=>( //using map we return jsx
// <li>{name}</li> // Whenever we create a list, this element must have unique key prop.
<li key={ index}>{name}</li>
))}
{
if() // we cannot have if statement in here as it has to be a one line expression.
loggedIn ? <h1> Hello Member </h1> : <h1> Hello Guest</h1>
loggedIn ? <h1> Hello Member </h1> : '' // If you dont want to show anything
loggedIn && <h1> Hello Member </h1> // If you dont want to handle the other case , works with js as well
}
</ul>
<>
)
}
export default App
- we can have different returns with conditions
- inline styling
const App = ()=>{
const loggedIn = true;
const styles = {
color:'red',
fontSize:'55px' // not font-size like in css
}
if(loggedIn){
return <h1 style={{ color:'red', fontSize:'24'}}> Hello Member</h1>
}
return(
<h1 style={styles}> Hello Guest</h1>
)
}
export default App
- create file src/components/Navbar.jsx
- import Navbar from "./components/Navbar"
- Visulaize structure in a page as components.
- Navbar
- Hero
- HomeCards // Wrapper component for Card
- Card
- JobListings // Wrapper component for Listing
- JobListing
//app.jsx
<>
<Navbar />
<Hero />
<HomeCards />
<JobListings />
<ViewAllJobs /> // href="/jobs"
<>
//app.jsx
<>
<Navbar />
<Hero title="Test Title" subtitle="This is the subtitlt" />
<>
//Hero.jsx
const Hero = (props)=>{ // destructure it ({title = 'Become a dev', subtitle="Find react job"}) adding default
return(
{ props.title }
{ props.subtitle }
)
}
//app.jsx
import HomeCards from './components/HomeCards'
<HomeCards />
// HomeCards.jsx
import Card from './components/Card'
<Card> // basically its the div with class wrapper
...contains html // this acts as children in card component
</Card>
<Card bg='bg-indigo-100'> // basically its the div with class wrapper
...contains html // this acts as children in card component
</Card>
//Cards.jsx
// children(whatever we wrap) as a prop - refer to the content passed inside the Card component in the HomeCard component -
const Card = ({children, bg = 'bg-gray-100'})=>{
return <div className = {`wrapper ${bg}`}>{children}</div>
}
//Actual api { jobs:[{},{}] }
// jobs json [{},{}]
- Create a wrapper JobListings component and in this import the jobs json
- console ninja extension vscode
return (
<section>
...html
{jobs.map((job)=>(
<JobListing key={job.id} job = {job}/>
))}
</section>
)
const JobListing=({job})=>{
return(
...html // single parent
{job.type}
<a href={`/job/${job.id}`}>
)
}
- 2 types of state - component level, app level(pass down to components)
import {useState} from 'react'
const JobListing = ({job})=>{
const [showFullDescription, setShowFullDescription] = useState(false) //default value
let description = job.description
if(!showFullDescription){
description = description.substring(0, 90) + '...'
}
return(
...html // single parent
{job.type}
<a href={`/job/${job.id}`}>
{description}
// <button onClick={setShowFullDescription(!showFullDescription)}>{showFullDescription ? 'Less' : 'More'}</button> // Cant do this as it will run the function
// <button onClick={()=> setShowFullDescription(!showFullDescription)}>{showFullDescription ? 'Less' : 'More'}</button> // There is a better way than this
<button onClick={()=> setShowFullDescription((prevState)=>!prevState)}>{showFullDescription ? 'Less' : 'More'}</button> // similar behaviour can be used in several places like hamburger menu wherever toggling of UI is required.
)
}
function Parent(){
const [userData, setUserData]= useState('')
const handleData = (data)=>{
setUserData(data)
}
return (
<>
<h1>Data from Child Component: {userData}</h1>
<Child setData={handleData}/>
<>
)
}
function Child({setData}){
const handleChange = (e)=>{
setData(e.target.value)
}
return (
<>
<input onChange={handleChange} />
<>
)
}
- React icons docs
- npm i react-icons
- import {FaMapMarker} from 'react-icons/fa'
- create separate folder layouts
- we can have multiple layout
- MainLayout.jsx
import {Outlet} from 'react-router-dom'
// component you want on all the pages under this Layout
import {Navbar} from '../components/Navbar'
const MainLayout = ()=>{
return (
<>
<Navbar />
<Outlet />
</>
)
}
export default MainLayout
- create separate folder pages or screens
- HomePage.jsx
import Hero from "../components/Hero";
import HomeCards from "../components/HomeCards";
import JobListings from "../components/JobListings";
import ViewAllJobs from "../components/ViewAllJobs";
const HomePage = () => {
return (
<>
<Hero />
<HomeCards />
<JobListings isHome={true} />
<ViewAllJobs />
</>
);
};
export default HomePage;
// Jobs Page
- Remix framework is build on top of react router.
- used to add separate pages and links between them.
- npm i react-router-dom
// App.jsx
import {Route, createBrowserRouter, createRoutesFromElements, RouterProvider} from "react-router-dom"
import MainLayout from "./layouts/MainLayout"
import HomePage from "./pages/HomePage"
const router = createBrowserRouter(
// createRoutesFromElements(<Route index element={<h1>My App</h1>} />)
createRoutesFromElements(
<Route path="/" element={<MainLayout />} />
<Route index element={<HomePage />} />
<Route path='/jobs' element={<JobsPage />} />
<Route path='/jobs/:id' element={<JobPage />} />
<Route path='/*' element={<NotFound />} /> // Any page thats not found
</Route>
)
// createRoutesFromElements(<Route path="/about" element={<h1>My App</h1>} />)
)
const App = () =>{
return <RouterProvider router={router} />
}
// Navbar
import { NavLink } from "react-router-dom"; // use NavLink instead of Link as link doesn't add the active class like NavLink does
// use NavLink instead of a tag as a tag refresh the page everytime we click on the link
const linkClass = ({ isActive }) =>
isActive
? "bg-black text-white hover:bg-gray-900 hover:text-white rounded-md px-3 py-2"
: "text-white hover:bg-gray-900 hover:text-white rounded-md px-3 py-2";
<NavLink className="flex flex-shrink-0 items-center mr-4" to="/"></NavLink>
<NavLink to="/" className={linkClass}>Home</NavLink>
<NavLink to="/jobs" className={linkClass}>Jobs</NavLink>
<NavLink to="/add-job" className={linkClass}>Add Job</NavLink>
// Jobs Listing component
<Link to={`/jobs/${job.id}`} className="h-[36px] bg-indigo-500 hover:bg-indigo-600 text-white px-4 py-2 rounded-lg text-center text-sm">Read More
</Link>
- we create a new page and we add that in app file.
const JobListings = ({isHome = false})=>{ // destructuring
const jobListings = isHome ? jobs.slice(0,3):jobs
}
// HomePage
<JobListings isHome={true} />
- makes our data file as a server
- npm i -D json-server
- "server":"json-server --watch src/jobs.json --port 8000" // on this we are running json server and serving our api
- npm run server
- npm i react-spinners
import ClipLoader from "react-spinners/ClipLoader";
const override = {
display: "block",
margin: "100px auto",
};
const Spinner = ({ loading }) => {
return (
<ClipLoader
color="#4338ca"
loading={loading}
cssOverride={override}
size={150}
/>
);
};
export default Spinner;
- instead using http://localhost:8000
- we can create a proxy to avoid this
- If app is created using npm create-react-app we do it in package.json
- using vite we create it in vite.config
import { defineConfig } from "vite";
import react from "@vitejs/plugin-react";
// https://vitejs.dev/config/
export default defineConfig({
plugins: [react()],
server: {
port: 3000,
proxy:{
'/api':{ //if you see a route prefixed with /api we set the below target
target:'http://localhost:8000',
changeOrigin:true,
rewrite:(path)=>path.replace(/^\/api/,""), // to remove /api from route
}
}
},
});
Fetching data useEffect() -fetch on render, as we are rendering there is a side effect of fetching the data.
-
react suspense can also be used it renders while fetching, skelton can be used.
-
react query, swr also can be used, that makes data fetching a little easier
-
useHook also can be used with react 19.
-
but we are using useEffect
-
component loads it has a side effect where we fetch the data using the id and then we are setting it to the state.
import { useState, useEffect } from "react";
import JobListing from "./JobListing";
import Spinner from "./Spinner";
const JobListings = ({ isHome = false }) => {
const [jobs, setJobs] = useState([]);
const [loading, setLoading] = useState(true);
useEffect(() => { // we cannot create this fn as async so we need to create one inside
const fetchJobs = async () => {
const apiUrl = isHome ? "/api/jobs?_limit=3" : "/api/jobs"; //http://localhost:8000/jobs
try {
const res = await fetch(apiUrl);
const data = await res.json();
setJobs(data);
} catch (error) {
console.log("Error fetching data", error);
} finally {
setLoading(false);
}
};
fetchJobs();
}, []); // dependency array - we can add variable in that, which when changed will trigger this hook
return (
<section className="bg-blue-50 px-4 py-10">
<div className="container-xl lg:container m-auto">
<h2 className="text-3xl font-bold text-indigo-500 mb-6 text-center">
{isHome ? "Recent Jobs" : "Browse Jobs"}
</h2>
{loading ? (
<Spinner loading={loading} />
) : (
<div className="grid grid-cols-1 md:grid-cols-3 gap-6">
{jobs.map((job) => (
<JobListing key={job.id} job={job} />
))}
</div>
)}
</div>
</section>
);
};
export default JobListings;
import {userParams} from 'react-router-dom'
const { id } = useParams()
- component loads it has a side effect where we fetch the data using the id and then we are setting it to the state. - using useEffect but there is another way using data loaders.
- we can also export the data to other components.
// JobPage
import { useParams, useLoaderData, useNavigate } from "react-router-dom";
const JobPage = () => {
const job = useLoaderData(); // it is a hook from react router dom - this is used to access the jobLoader data
return(
)
}
const jobLoader = async ({ params }) => { // used in jobpage to show jobs and edit
const res = await fetch(`/api/jobs/${params.id}`);
const data = await res.json();
return data;
};
export { JobPage as default, jobLoader}
//app.jsx
import JobPage, { jobLoader } from "./pages/JobPage";
// ...inside main layout Route
<Route path="/edit-job/:id" element={<EditJobPage updateJobSubmit={updateJob} />} loader={jobLoader}/>
<Route path="/jobs/:id" element={<JobPage deleteJob={deleteJob} />} loader={jobLoader} />
- you can use formick, ref
- common way is to add a piece of state for every field in your form
- class - className
- for - htmlFor
import { useState } from 'react'
const AddJobPage = ({ addJobSubmit }) => {
// you can have a single object in your state and have all the fields in that object
const [title, setTitle]= useState(''); // state item for each field
const [type, setType] = useState("Full-Time");
const [location, setLocation] = useState("");
const submitForm = (e)=>{
e.preventDefault()
const newJob = {
title,
type,
location,
description,
salary,
company: {
name: companyName,
description: companyDescription,
contactEmail,
contactPhone,
},
};
addJobSubmit(newJob);
toast.success("Job Added Successfully");
return navigate("/jobs");
}
return (
<>
// <form action> - react 19
<form onSubmite={submitForm}> // we can create an arrow fn here or pass the name of the function
// value and onChange are imp
<input type="text" id="title" name="title" className="border rounded w-full py-2 px-3 mb-2" placeholder="eg. Beautiful Apartment In Miami" required value={title} onChange={(e) => setTitle(e.target.value)} />
<select id="type" name="type" className="border rounded w-full py-2 px-3" required value={type} onChange={(e) => setType(e.target.value)} >
<input type="text" id="location" name="location" className="border rounded w-full py-2 px-3 mb-2" placeholder="Company Location" required value={location} onChange={(e) => setLocation(e.target.value)} />
</>
)}
- multiple cursor case preserve vscode extension
- npm i react-toastify
import { Outlet } from "react-router-dom";
import { ToastContainer } from "react-toastify";
import "react-toastify/dist/ReactToastify.css";
import Navbar from "../components/Navbar";
const MainLayout = () => {
return (
<>
<Navbar />
<Outlet />
<ToastContainer />
</>
);
};
export default MainLayout;
- passing fn as a prop
import {
Route,
createBrowserRouter,
createRoutesFromElements,
RouterProvider,
} from "react-router-dom";
import MainLayout from "./layouts/MainLayout";
import HomePage from "./pages/HomePage";
import JobsPage from "./pages/JobsPage";
import NotFoundPage from "./pages/NotFoundPage";
import JobPage, { jobLoader } from "./pages/JobPage";
import AddJobPage from "./pages/AddJobPage";
import EditJobPage from "./pages/EditJobPage";
const App = () => {
// Add New Job
const addJob = async (newJob) => {
const res = await fetch("/api/jobs", {
method: "POST",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(newJob),
});
return;
};
// Delete Job
const deleteJob = async (id) => {
const res = await fetch(`/api/jobs/${id}`, {
method: "DELETE",
});
return;
};
// Update Job
const updateJob = async (job) => {
const res = await fetch(`/api/jobs/${job.id}`, {
method: "PUT",
headers: {
"Content-Type": "application/json",
},
body: JSON.stringify(job),
});
return;
};
const router = createBrowserRouter(
createRoutesFromElements(
<Route path="/" element={<MainLayout />}>
<Route index element={<HomePage />} />
<Route path="/jobs" element={<JobsPage />} />
<Route
path="/add-job"
element={<AddJobPage addJobSubmit={addJob} />} // passing fn as prop
/>
<Route
path="/edit-job/:id"
element={<EditJobPage updateJobSubmit={updateJob} />}
loader={jobLoader}
/>
<Route
path="/jobs/:id"
element={<JobPage deleteJob={deleteJob} />}
loader={jobLoader}
/>
<Route path="*" element={<NotFoundPage />} />
</Route>
)
);
return <RouterProvider router={router} />;
};
export default App;
// AddJobPage
import { useNavigate } from "react-router-dom";
import { toast } from "react-toastify"; // using toastify
const AddJobPage = ({ addJobSubmit }) => {
const navigate = useNavigate(); // iinitializing
const newJob = {
title,
type,
location,
description,
salary,
company: {
name: companyName,
description: companyDescription,
contactEmail,
contactPhone,
},
};
addJobSubmit(newJob);
toast.success("Job Added Successfully"); // toastify toast.error for error
return navigate("/jobs"); // redirecting
}
// EditJobPage
const EditJobPage = ({ updateJobSubmit }) => {
const updatedJob = {
id,
title,
type,
location,
description,
salary,
company: {
name: companyName,
description: companyDescription,
contactEmail,
contactPhone,
},
};
updateJobSubmit(updatedJob);
}
// JobPage
import { useParams, useLoaderData, useNavigate } from "react-router-dom";
import { FaArrowLeft, FaMapMarker } from "react-icons/fa";
import { Link } from "react-router-dom";
import { toast } from "react-toastify";
const JobPage = ({ deleteJob }) => {
const navigate = useNavigate();
const { id } = useParams();
const job = useLoaderData();
const onDeleteClick = (jobId) => {
const confirm = window.confirm(
"Are you sure you want to delete this listing?"
);
if (!confirm) return;
deleteJob(jobId);
toast.success("Job deleted successfully");
navigate("/jobs");
};
return (
<>
<Link to={`/edit-job/${job.id}`} className="bg-indigo-500 hover:bg-indigo-600 text-white text-center font-bold py-2 px-4 rounded-full w-full focus:outline-none focus:shadow-outline mt-4 block">Edit Job</Link>
<button onClick={() => onDeleteClick(job.id)} className="bg-red-500 hover:bg-red-600 text-white font-bold py-2 px-4 rounded-full w-full focus:outline-none focus:shadow-outline mt-4 block">Delete Job</button>
</>
)
}
- npm run build -> creates dist thats production build
- npm run preview -> runs production build