Handling Routing in React Apps with React Router

Avatar

By squashlabs, Last Updated: July 20, 2023

Handling Routing in React Apps with React Router

Table of Contents

Introduction to Routing

Routing is a crucial aspect of building single-page applications (SPAs) in React. It allows us to define different paths or URLs that correspond to different components or views within our application. With routing, we can create a seamless and intuitive user experience by enabling navigation between different pages without the need for page refreshes.

Related Article: How to Build Forms in React

Why Use React Router?

React Router is a popular routing library for React applications. It provides a declarative way to handle routing, allowing us to define our routes as components and render them based on the current URL. React Router also offers additional features such as nested routes, redirects, and prompts, making it a powerful tool for managing complex routing scenarios.

Installing React Router

To get started with React Router, we first need to install it as a dependency in our project. We can do this using npm or yarn. Open your terminal and navigate to your project directory. Then, run the following command:

npm install react-router-dom

or

yarn add react-router-dom

Once the installation is complete, we can import the necessary components from React Router and start building our routes.

Basic Routing Concepts

In React Router, routing is achieved by using the <BrowserRouter> component, which wraps our application and provides the routing functionality. Inside the <BrowserRouter>, we define our routes using the <Route> component.

Related Article: How to Integrate UseHistory from the React Router DOM

Setting up the Router

To set up the router in our application, we need to wrap our root component with the <BrowserRouter> component. This will enable routing for all the components in our application. Here’s an example of how to set up the router:

import React from 'react';
import { BrowserRouter } from 'react-router-dom';

function App() {
  return (
    <BrowserRouter>
      {/* Your application components */}
    </BrowserRouter>
  );
}

export default App;

Defining Routes

Once we have set up the router, we can define our routes using the <Route> component. The <Route> component takes two main props: path and component. The path prop specifies the URL path for the route, and the component prop specifies the component to render when that path is matched. Here’s an example of how to define a basic route:

import React from 'react';
import { Route } from 'react-router-dom';

function Home() {
  return <h1>Welcome to the Home Page!</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/" component={Home} />
    </BrowserRouter>
  );
}

export default App;

In this example, the <Route> component will render the Home component when the URL path matches “/”.

Adding Multiple Routes

We can add multiple routes by nesting <Route> components within the <BrowserRouter>. Each <Route> component can have its own path and component props. Here’s an example of how to define multiple routes:

import React from 'react';
import { Route } from 'react-router-dom';

function Home() {
  return <h1>Welcome to the Home Page!</h1>;
}

function About() {
  return <h1>About Us</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/" component={Home} />
      <Route path="/about" component={About} />
    </BrowserRouter>
  );
}

export default App;

In this example, the Home component will be rendered when the URL path is “/”, and the About component will be rendered when the URL path is “/about”.

Related Article: Exploring Differences in Rendering Components in ReactJS

Detailed Analysis of Router Components

React Router provides several components that we can use to handle different aspects of routing in our application. Let’s take a closer look at some of these components and their usage.

Switch Component

The <Switch> component is used to group multiple <Route> components together and render only the first one that matches the current URL. This is useful when we want to render a specific component for a particular route and ignore the rest. Here’s an example of how to use the <Switch> component:

import React from 'react';
import { BrowserRouter, Switch, Route } from 'react-router-dom';

function Home() {
  return <h1>Welcome to the Home Page!</h1>;
}

function About() {
  return <h1>About Us</h1>;
}

function NotFound() {
  return <h1>404 - Page Not Found</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Switch>
        <Route exact path="/" component={Home} />
        <Route path="/about" component={About} />
        <Route component={NotFound} />
      </Switch>
    </BrowserRouter>
  );
}

export default App;

In this example, if the URL path matches “/”, the Home component will be rendered. If the URL path matches “/about”, the About component will be rendered. If none of the routes match, the NotFound component will be rendered.

The <Link> component is used to create links to different routes within our application. It provides a declarative way to navigate between pages without triggering a full page refresh. Here’s an example of how to use the <Link> component:

import React from 'react';
import { BrowserRouter, Link, Route } from 'react-router-dom';

function Home() {
  return <h1>Welcome to the Home Page!</h1>;
}

function About() {
  return <h1>About Us</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <nav>
        <Link to="/">Home</Link>
        <Link to="/about">About</Link>
      </nav>

      <Route exact path="/" component={Home} />
      <Route path="/about" component={About} />
    </BrowserRouter>
  );
}

export default App;

In this example, the <Link> components will render as anchor tags, and when clicked, they will navigate to the specified routes without triggering a full page refresh.

Related Article: Implementing HTML Templates in ReactJS

Nested Routes Exploration

Nested routes allow us to define routes within routes, creating a hierarchy of components that correspond to different URL paths. This is useful when building complex applications with nested UI structures. Let’s explore how to set up nested routes in React Router.

Setting up Nested Routes

To set up nested routes, we can simply nest <Route> components within other <Route> components. Each nested <Route> component should have its own path and component props. Here’s an example of how to set up nested routes:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';

function Home() {
  return <h1>Welcome to the Home Page!</h1>;
}

function Products() {
  return <h1>Products Page</h1>;
}

function ProductDetails() {
  return <h1>Product Details Page</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/" component={Home} />
      <Route path="/products" component={Products} />
      <Route path="/products/:id" component={ProductDetails} />
    </BrowserRouter>
  );
}

export default App;

In this example, the Products component will be rendered when the URL path is “/products”, and the ProductDetails component will be rendered when the URL path is “/products/:id”, where “:id” is a dynamic parameter that represents the ID of a specific product.

Accessing Nested Route Parameters

When using nested routes with dynamic parameters, we can access the parameter values using the useParams hook provided by React Router. This hook allows us to retrieve the parameter values from the URL. Here’s an example of how to access nested route parameters:

import React from 'react';
import { BrowserRouter, Route, useParams } from 'react-router-dom';

function ProductDetails() {
  const { id } = useParams();

  return <h1>Product Details - ID: {id}</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/products/:id" component={ProductDetails} />
    </BrowserRouter>
  );
}

export default App;

In this example, the ProductDetails component retrieves the id parameter value from the URL using the useParams hook and displays it in the component.

Related Article: Accessing Array Length in this.state in ReactJS

Redirects and Prompts

React Router provides built-in components for handling redirects and prompts. Redirects allow us to redirect users to a different route, while prompts enable us to prompt users before leaving a page. Let’s explore how to use these components in our React applications.

Redirecting to a Different Route

To redirect users to a different route, we can use the <Redirect> component provided by React Router. This component can be rendered conditionally based on certain conditions, such as user authentication status or form submission. Here’s an example of how to use the <Redirect> component:

import React, { useState } from 'react';
import { BrowserRouter, Route, Redirect } from 'react-router-dom';

function LoginPage() {
  const [isLoggedIn, setIsLoggedIn] = useState(false);

  const handleLogin = () => {
    setIsLoggedIn(true);
  };

  if (isLoggedIn) {
    return <Redirect to="/dashboard" />;
  }

  return (
    <div>
      <h1>Login Page</h1>
      <button onClick={handleLogin}>Log In</button>
    </div>
  );
}

function DashboardPage() {
  return <h1>Dashboard Page</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/login" component={LoginPage} />
      <Route path="/dashboard" component={DashboardPage} />
    </BrowserRouter>
  );
}

export default App;

In this example, when the user clicks the “Log In” button on the LoginPage, the isLoggedIn state is updated to true, and the <Redirect> component redirects the user to the /dashboard route.

Prompting Before Leaving a Page

React Router also provides the <Prompt> component, which allows us to prompt users before they navigate away from a page. This can be useful when we want to prevent users from losing unsaved changes. Here’s an example of how to use the <Prompt> component:

import React, { useState } from 'react';
import { BrowserRouter, Route, Prompt } from 'react-router-dom';

function EditProfilePage() {
  const [hasUnsavedChanges, setHasUnsavedChanges] = useState(false);

  const handleInputChange = () => {
    setHasUnsavedChanges(true);
  };

  return (
    <div>
      <Prompt
        when={hasUnsavedChanges}
        message="Are you sure you want to leave? You have unsaved changes."
      />

      <h1>Edit Profile Page</h1>
      <input type="text" onChange={handleInputChange} />
    </div>
  );
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/edit-profile" component={EditProfilePage} />
    </BrowserRouter>
  );
}

export default App;

In this example, when the user starts typing in the input field on the EditProfilePage, the hasUnsavedChanges state is updated to true, and the <Prompt> component displays a confirmation message when the user tries to navigate away from the page.

Related Article: Inserting Plain Text into an Array Using ReactJS

Use Case: User Authentication

User authentication is a common use case in web applications. React Router provides a way to handle authentication and protect certain routes from unauthorized access. Let’s explore how to implement user authentication in a React app using React Router.

PrivateRoute Component

To protect certain routes and allow only authenticated users to access them, we can create a custom PrivateRoute component. This component uses the <Route> component internally and checks if the user is authenticated before rendering the protected component. If the user is not authenticated, it redirects them to the login page. Here’s an example of how to create a PrivateRoute component:

import React from 'react';
import { Route, Redirect } from 'react-router-dom';

function PrivateRoute({ component: Component, isAuthenticated, ...rest }) {
  return (
    <Route
      {...rest}
      render={(props) =>
        isAuthenticated ? (
          <Component {...props} />
        ) : (
          <Redirect to="/login" />
        )
      }
    />
  );
}

export default PrivateRoute;

In this example, the PrivateRoute component takes three props: component, isAuthenticated, and ...rest (which represents the remaining props passed to the component). It renders the protected component if the user is authenticated, otherwise it redirects them to the login page.

Implementing User Authentication

To implement user authentication in our app, we need to manage the user’s authentication state and pass it to the PrivateRoute component. Here’s an example of how to implement user authentication using React Router:

import React, { useState } from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import PrivateRoute from './PrivateRoute';
import LoginPage from './LoginPage';
import DashboardPage from './DashboardPage';

function App() {
  const [isAuthenticated, setIsAuthenticated] = useState(false);

  const handleLogin = () => {
    setIsAuthenticated(true);
  };

  const handleLogout = () => {
    setIsAuthenticated(false);
  };

  return (
    <BrowserRouter>
      <Route
        path="/login"
        render={(props) => (
          <LoginPage {...props} onLogin={handleLogin} />
        )}
      />

      <PrivateRoute
        path="/dashboard"
        component={DashboardPage}
        isAuthenticated={isAuthenticated}
      />

      <button onClick={handleLogout}>Log Out</button>
    </BrowserRouter>
  );
}

export default App;

In this example, the App component manages the isAuthenticated state. When the user clicks the “Log In” button on the LoginPage, the handleLogin function is called, updating the isAuthenticated state to true. The PrivateRoute component is used to protect the /dashboard route and only render the DashboardPage component if the user is authenticated. Finally, the “Log Out” button triggers the handleLogout function, updating the isAuthenticated state to false.

Related Article: Displaying Arrays with Onclick Events in ReactJS

Redirecting After Login

After the user successfully logs in, we may want to redirect them to a specific page, such as the dashboard. React Router provides the useHistory hook, which allows us to programmatically navigate to a different route. Here’s an example of how to redirect after login:

import React, { useState } from 'react';
import { BrowserRouter, Route, Redirect, useHistory } from 'react-router-dom';

function LoginPage() {
  const [isLoggedIn, setIsLoggedIn] = useState(false);
  const history = useHistory();

  const handleLogin = () => {
    setIsLoggedIn(true);
    history.push('/dashboard');
  };

  if (isLoggedIn) {
    return <Redirect to="/dashboard" />;
  }

  return (
    <div>
      <h1>Login Page</h1>
      <button onClick={handleLogin}>Log In</button>
    </div>
  );
}

function DashboardPage() {
  return <h1>Dashboard Page</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/login" component={LoginPage} />
      <Route path="/dashboard" component={DashboardPage} />
    </BrowserRouter>
  );
}

export default App;

In this example, when the user clicks the “Log In” button on the LoginPage, the handleLogin function is called. It sets the isLoggedIn state to true and uses the history.push() method to programmatically navigate to the /dashboard route.

Use Case: Page Not Found (404)

Handling a “Page Not Found” (404) error is an important aspect of building robust web applications. React Router provides a way to handle this error by rendering a specific component when none of the defined routes match the current URL. Let’s explore how to handle the 404 error using React Router.

Creating a NotFound Component

To create a “Page Not Found” component, we can define a simple component that displays an appropriate message. This component will be rendered when none of the routes match the current URL. Here’s an example of how to create a NotFound component:

import React from 'react';

function NotFound() {
  return <h1>404 - Page Not Found</h1>;
}

export default NotFound;

In this example, the NotFound component simply displays the “404 – Page Not Found” message.

Related Article: How to to Deploy a ReactJS Application to Production

Defining the NotFound Route

To handle the 404 error, we need to define a route that matches all URLs. This route will be the last route in our application, ensuring that it only renders when none of the other routes match. Here’s an example of how to define the NotFound route:

import React from 'react';
import { BrowserRouter, Route, Switch } from 'react-router-dom';
import Home from './Home';
import About from './About';
import NotFound from './NotFound';

function App() {
  return (
    <BrowserRouter>
      <Switch>
        <Route exact path="/" component={Home} />
        <Route path="/about" component={About} />
        <Route component={NotFound} />
      </Switch>
    </BrowserRouter>
  );
}

export default App;

In this example, the NotFound component will be rendered when none of the routes (/ and /about) match the current URL.

Use Case: Loading Screen

Displaying a loading screen or spinner is a common practice in web applications to provide feedback to users during asynchronous operations. React Router allows us to easily implement a loading screen while data is being fetched or components are being loaded. Let’s explore how to create a loading screen using React Router.

Creating a Loading Component

To create a loading screen, we can define a component that displays a loading indicator or message. This component will be rendered while the data is being fetched or components are being loaded. Here’s an example of how to create a Loading component:

import React from 'react';

function Loading() {
  return <h1>Loading...</h1>;
}

export default Loading;

In this example, the Loading component simply displays the “Loading…” message.

Related Article: How to Update a State Property in ReactJS

Using the Suspense Component

React Router provides the <Suspense> component, which allows us to wrap lazy-loaded components and display a fallback (loading) component while the lazy-loaded components are being loaded. This is useful when code-splitting our application to improve performance. Here’s an example of how to use the <Suspense> component:

import React, { lazy, Suspense } from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Loading from './Loading';

const Home = lazy(() => import('./Home'));
const About = lazy(() => import('./About'));

function App() {
  return (
    <BrowserRouter>
      <Suspense fallback={<Loading />}>
        <Route exact path="/" component={Home} />
        <Route path="/about" component={About} />
      </Suspense>
    </BrowserRouter>
  );
}

export default App;

In this example, the Home and About components are lazily loaded using the lazy function from React. The <Suspense> component wraps the <Route> components and displays the <Loading> component while the lazy-loaded components are being loaded.

Best Practice: Consistent Route Definitions

When defining routes in our application, it’s important to maintain consistency in how we structure and organize our routes. Consistent route definitions make it easier to understand and maintain our codebase. Let’s explore some best practices for defining routes in React Router.

Route Structure

When defining routes, it’s recommended to use a consistent structure that reflects the hierarchy and organization of our application. This can be achieved by using nested routes and organizing related routes together. Here’s an example of a consistent route structure:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Home from './Home';
import Products from './Products';
import ProductDetails from './ProductDetails';
import Orders from './Orders';
import OrderDetails from './OrderDetails';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />

      <Route path="/products" component={Products} />
      <Route path="/products/:id" component={ProductDetails} />

      <Route path="/orders" component={Orders} />
      <Route path="/orders/:id" component={OrderDetails} />
    </BrowserRouter>
  );
}

export default App;

In this example, the routes are organized hierarchically and relate to different sections of the application. The Products and Orders components serve as parent components for their respective child components (ProductDetails and OrderDetails).

Related Article: How to Implement Hover State in ReactJS with Inline Styles

Route Ordering

The order in which routes are defined is important, as React Router matches routes sequentially and renders the first match it encounters. It’s important to order routes from most specific to least specific to ensure that the correct component is rendered. Here’s an example of the correct route ordering:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Home from './Home';
import Products from './Products';
import ProductDetails from './ProductDetails';
import NotFound from './NotFound';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />

      <Route exact path="/products" component={Products} />
      <Route path="/products/:id" component={ProductDetails} />

      <Route component={NotFound} />
    </BrowserRouter>
  );
}

export default App;

In this example, the /products/:id route is defined after the /products route to ensure that it takes precedence. If the /products/:id route was defined before the /products route, it would never match because the /products route would always take precedence.

Best Practice: URL Parameter Validation

When working with URL parameters in React Router, it’s important to validate and sanitize the input to prevent security vulnerabilities and unexpected behavior. React Router provides built-in mechanisms to validate URL parameters using regular expressions. Let’s explore how to validate URL parameters in React Router.

Validating URL Parameters

To validate URL parameters, we can use regular expressions in the path prop of the <Route> component. Regular expressions allow us to define patterns that the URL parameters must match. Here’s an example of how to validate a numeric ID parameter:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import ProductDetails from './ProductDetails';

function App() {
  return (
    <BrowserRouter>
      <Route
        path="/products/:id(\d+)"
        component={ProductDetails}
      />
    </BrowserRouter>
  );
}

export default App;

In this example, the (\d+) part of the path prop is a regular expression pattern that matches one or more digits. This ensures that the id parameter in the URL must be a numeric value.

Related Article: Handling State Persistence in ReactJS After Refresh

Accessing Validated URL Parameters

When validating URL parameters, it’s important to handle cases where the parameter does not match the expected pattern. React Router provides the useParams hook, which returns an object containing all the URL parameters. We can then validate and sanitize the parameter values as needed. Here’s an example of how to access and validate URL parameters:

import React from 'react';
import { BrowserRouter, Route, useParams } from 'react-router-dom';

function ProductDetails() {
  const { id } = useParams();

  if (!/^\d+$/.test(id)) {
    return <h1>Invalid Product ID</h1>;
  }

  // Rest of the component logic
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/products/:id" component={ProductDetails} />
    </BrowserRouter>
  );
}

export default App;

In this example, the useParams hook is used to access the id parameter from the URL. We then use a regular expression (/^\d+$/) to check if the id parameter is a valid numeric value. If the parameter does not match the expected pattern, we render an “Invalid Product ID” message.

Real World Example: E-commerce Website Navigation

Let’s explore a real-world example of how to handle navigation in an e-commerce website using React Router. We’ll cover the common navigation patterns such as viewing products, adding them to the cart, and checking out.

Defining Routes

To handle the navigation in an e-commerce website, we need to define routes for different pages and components. Here’s an example of how to define routes for an e-commerce website:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Home from './Home';
import ProductList from './ProductList';
import ProductDetails from './ProductDetails';
import Cart from './Cart';
import Checkout from './Checkout';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />
      <Route exact path="/products" component={ProductList} />
      <Route path="/products/:id" component={ProductDetails} />
      <Route path="/cart" component={Cart} />
      <Route path="/checkout" component={Checkout} />
    </BrowserRouter>
  );
}

export default App;

In this example, the routes are defined for the home page (/), product list page (/products), product details page (/products/:id), cart page (/cart), and checkout page (/checkout).

Related Article: Adding a Newline in the JSON Property Render with ReactJS

To navigate to the product details page when a product is clicked, we can use the <Link> component provided by React Router. Here’s an example of how to navigate to the product details page:

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

function ProductList() {
  return (
    <div>
      <h1>Product List</h1>
      <ul>
        <li>
          <Link to="/products/1">Product 1</Link>
        </li>
        <li>
          <Link to="/products/2">Product 2</Link>
        </li>
        {/* Add more products */}
      </ul>
    </div>
  );
}

export default ProductList;

In this example, the <Link> components are used to create links to the product details page for each product. When a product is clicked, the user will be navigated to the corresponding product details page.

Adding Products to the Cart

To add products to the cart, we can use a combination of React state and React Router. Here’s an example of how to add products to the cart:

import React, { useState } from 'react';
import { Link } from 'react-router-dom';

function ProductDetails({ match }) {
  const [cart, setCart] = useState([]);

  const handleAddToCart = (productId) => {
    setCart([...cart, productId]);
  };

  return (
    <div>
      <h1>Product Details - ID: {match.params.id}</h1>
      <button onClick={() => handleAddToCart(match.params.id)}>
        Add to Cart
      </button>
      <Link to="/cart">View Cart</Link>
    </div>
  );
}

export default ProductDetails;

In this example, the ProductDetails component uses the useState hook to manage the cart state. When the “Add to Cart” button is clicked, the handleAddToCart function is called, which adds the product ID to the cart state. The user can then click the “View Cart” link to navigate to the cart page.

Real World Example: Multi-Step Form Navigation

Another real-world example of using React Router is handling multi-step forms. Multi-step forms are commonly used in applications that require users to provide input in multiple stages. Let’s explore how to handle multi-step forms using React Router.

Related Article: How Rendering Works in ReactJS

Defining Routes

To handle multi-step forms, we need to define routes for each step of the form. Here’s an example of how to define routes for a multi-step form:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Step1 from './Step1';
import Step2 from './Step2';
import Step3 from './Step3';
import Confirmation from './Confirmation';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Step1} />
      <Route path="/step2" component={Step2} />
      <Route path="/step3" component={Step3} />
      <Route path="/confirmation" component={Confirmation} />
    </BrowserRouter>
  );
}

export default App;

In this example, the routes are defined for the first step of the form (/), the second step of the form (/step2), the third step of the form (/step3), and the confirmation page (/confirmation).

To navigate between steps of the form, we can use the <Link> component provided by React Router. Here’s an example of how to navigate between steps:

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

function Step1() {
  return (
    <div>
      <h1>Step 1</h1>
      <Link to="/step2">Next</Link>
    </div>
  );
}

export default Step1;

In this example, the “Next” link is used to navigate to the next step of the form (/step2). When the user clicks the link, they will be taken to the next step of the form.

Submitting the Form

To submit the form and display a confirmation page, we can use a combination of React state and React Router. Here’s an example of how to submit the form:

import React, { useState } from 'react';
import { Link, useHistory } from 'react-router-dom';

function Step3() {
  const [formData, setFormData] = useState({});
  const history = useHistory();

  const handleSubmit = (e) => {
    e.preventDefault();
    // Process form data
    history.push('/confirmation');
  };

  return (
    <div>
      <h1>Step 3</h1>
      <form onSubmit={handleSubmit}>
        {/* Form fields */}
        <button type="submit">Submit</button>
      </form>
      <Link to="/step2">Back</Link>
    </div>
  );
}

export default Step3;

In this example, the Step3 component uses the useState hook to manage the form data. When the form is submitted, the handleSubmit function is called, which prevents the default form submission behavior, processes the form data, and navigates to the confirmation page (/confirmation) using the history.push() method. The user can also click the “Back” link to navigate to the previous step of the form (/step2).

Related Article: How to Use a For Loop Inside Render in ReactJS

Real World Example: Admin Dashboard Navigation

Admin dashboards often have complex navigation structures with multiple levels of nested menus and pages. React Router provides the flexibility to handle such navigation requirements in an efficient manner. Let’s explore how to handle admin dashboard navigation using React Router.

Defining Routes

To handle the navigation in an admin dashboard, we need to define routes for different pages and components. Here’s an example of how to define routes for an admin dashboard:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Dashboard from './Dashboard';
import Users from './Users';
import Products from './Products';
import Orders from './Orders';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Dashboard} />
      <Route path="/users" component={Users} />
      <Route path="/products" component={Products} />
      <Route path="/orders" component={Orders} />
    </BrowserRouter>
  );
}

export default App;

In this example, the routes are defined for the dashboard page (/), users page (/users), products page (/products), and orders page (/orders).

Creating Navigation Menus

To create navigation menus in the admin dashboard, we can use the <Link> component provided by React Router. Here’s an example of how to create navigation menus:

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

function Sidebar() {
  return (
    <div>
      <ul>
        <li>
          <Link to="/">Dashboard</Link>
        </li>
        <li>
          <Link to="/users">Users</Link>
        </li>
        <li>
          <Link to="/products">Products</Link>
        </li>
        <li>
          <Link to="/orders">Orders</Link>
        </li>
      </ul>
    </div>
  );
}

export default Sidebar;

In this example, the <Link> components are used to create links to different pages of the admin dashboard. When a link is clicked, the corresponding page will be rendered.

Related Article: Executing Multiple Fetch Calls Simultaneously in ReactJS

Nested Routes and Components

To handle nested routes and components in the admin dashboard, we can use the <Switch> and <Route> components provided by React Router. Here’s an example of how to handle nested routes and components:

import React from 'react';
import { BrowserRouter, Route, Switch } from 'react-router-dom';
import Dashboard from './Dashboard';
import Users from './Users';
import Products from './Products';
import Orders from './Orders';
import UserDetail from './UserDetail';
import ProductDetail from './ProductDetail';
import NotFound from './NotFound';

function App() {
  return (
    <BrowserRouter>
      <Switch>
        <Route exact path="/" component={Dashboard} />
        <Route exact path="/users" component={Users} />
        <Route exact path="/users/:id" component={UserDetail} />
        <Route exact path="/products" component={Products} />
        <Route exact path="/products/:id" component={ProductDetail} />
        <Route exact path="/orders" component={Orders} />
        <Route component={NotFound} />
      </Switch>
    </BrowserRouter>
  );
}

export default App;

In this example, the <Switch> component is used to group the <Route> components together. The nested routes (/users/:id and /products/:id) are defined after their parent routes to ensure that they take precedence. The <NotFound> component is rendered when none of the routes match the current URL.

Performance Consideration: Route-based Code Splitting

Code splitting is a technique used to split a large JavaScript bundle into smaller chunks that can be loaded on-demand. React Router provides a way to implement code splitting on a per-route basis, improving the performance of our application. Let’s explore how to implement route-based code splitting using React Router.

Lazy Loading Components

To implement code splitting, we can use the lazy function provided by React. The lazy function allows us to lazily load components, which means they will be loaded only when they are needed. Here’s an example of how to lazily load components:

import React, { lazy, Suspense } from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Loading from './Loading';

const Home = lazy(() => import('./Home'));
const About = lazy(() => import('./About'));

function App() {
  return (
    <BrowserRouter>
      <Suspense fallback={<Loading />}>
        <Route exact path="/" component={Home} />
        <Route path="/about" component={About} />
      </Suspense>
    </BrowserRouter>
  );
}

export default App;

In this example, the Home and About components are lazily loaded using the lazy function. The <Suspense> component wraps the <Route> components and displays the <Loading> component while the lazy-loaded components are being loaded.

Related Article: Comparing Reactivity in ReactJS and VueJS Variables

Route-level Code Splitting

To apply code splitting on a per-route basis, we can use the fallback prop of the <Suspense> component. This prop specifies the component to render while the lazy-loaded component is being loaded. Here’s an example of how to implement route-level code splitting:

import React, { lazy, Suspense } from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Loading from './Loading';

const Home = lazy(() => import('./Home'));
const About = lazy(() => import('./About'));

function App() {
  return (
    <BrowserRouter>
      <Suspense fallback={<Loading />}>
        <Route
          exact
          path="/"
          render={() => <Home />}
        />
        <Route
          path="/about"
          render={() => <About />}
        />
      </Suspense>
    </BrowserRouter>
  );
}

export default App;

In this example, the <Suspense> component wraps the <Route> components and specifies the <Loading> component as the fallback while the lazy-loaded components (Home and About) are being loaded. This ensures that the loading screen is displayed only for the specific route being loaded.

Performance Consideration: Precaching Route Components

Precaching route components is a technique used to preload and cache the resources (JavaScript, CSS, etc.) required for a specific route before the user navigates to that route. React Router provides a way to precache route components, improving the performance of our application. Let’s explore how to precache route components using React Router.

Precaching with react-router-config

To precache route components, we can use the react-router-config package, which provides a matchRoutes function for matching routes and their components. We can use this function to prefetch the components for specific routes. Here’s an example of how to precache route components:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import { matchRoutes } from 'react-router-config';
import Home from './Home';
import About from './About';

const routes = [
  { path: '/', component: Home },
  { path: '/about', component: About },
  // Add more routes
];

function App() {
  const prefetchRoutes = () => {
    const matchedRoutes = matchRoutes(routes, window.location.pathname);
    const prefetchPromises = matchedRoutes.map(({ route }) => {
      const Component = route.component;
      return Component.preload ? Component.preload() : Promise.resolve();
    });
    return Promise.all(prefetchPromises);
  };

  prefetchRoutes();

  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />
      <Route path="/about" component={About} />
      {/* Add more routes */}
    </BrowserRouter>
  );
}

export default App;

In this example, the routes array contains the routes and their corresponding components. The prefetchRoutes function uses the matchRoutes function to match the current route and its component. It then checks if the component has a preload method (which should return a promise) and calls it to prefetch the resources for that component. Finally, the prefetchRoutes function is called when the app loads to precache the route components.

Related Article: How to Implement Custom CSS in ReactJS

React Router provides the <Link> component for creating links to different routes. However, we may need more control over the behavior and styling of our links. In such cases, we can create custom link components that extend the functionality of the <Link> component. Let’s explore how to create custom link components using React Router.

To create a custom link component, we can use the <Link> component provided by React Router and add additional functionality and styling as needed. Here’s an example of how to create a CustomLink component:

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

function CustomLink({ to, children }) {
  const isActive = window.location.pathname === to;

  return (
    <Link to={to} className={isActive ? 'active' : ''}>
      {children}
    </Link>
  );
}

export default CustomLink;

In this example, the CustomLink component takes the to prop and children prop. It uses the <Link> component to create the link, and adds the active class to the link if the current URL matches the to prop.

To use the CustomLink component, we can replace the <Link> component in our application with the CustomLink component. Here’s an example of how to use the CustomLink component:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import CustomLink from './CustomLink';
import Home from './Home';
import About from './About';

function App() {
  return (
    <BrowserRouter>
      <nav>
        <CustomLink to="/">Home</CustomLink>
        <CustomLink to="/about">About</CustomLink>
      </nav>

      <Route exact path="/" component={Home} />
      <Route path="/about" component={About} />
    </BrowserRouter>
  );
}

export default App;

In this example, the <CustomLink> components are used instead of the <Link> components to create customized links. The CustomLink component will handle the routing and styling based on the current URL.

Related Article: How to Solve "_ is not defined" Errors in ReactJS

Advanced Technique: Animated Route Transitions

Animating route transitions can improve the user experience by providing visual feedback during navigation. React Router allows us to animate route transitions by using CSS transitions and React state. Let’s explore how to animate route transitions using React Router.

Setting Up the Transition Animation

To set up the transition animation, we can use CSS transitions to animate the entering and leaving components. Here’s an example of how to set up the transition animation:

import React from 'react';
import { BrowserRouter, Route, Switch } from 'react-router-dom';
import { CSSTransition, TransitionGroup } from 'react-transition-group';
import Home from './Home';
import About from './About';
import './App.css';

function App() {
  return (
    <BrowserRouter>
      <Route
        render={({ location }) => (
          <TransitionGroup>
            <CSSTransition
              key={location.key}
              timeout={300}
              classNames="fade"
            >
              <Switch location={location}>
                <Route exact path="/" component={Home} />
                <Route path="/about" component={About} />
              </Switch>
            </CSSTransition>
          </TransitionGroup>
        )}
      />
    </BrowserRouter>
  );
}

export default App;

In this example, the <TransitionGroup> component from react-transition-group wraps the <CSSTransition> component. The key prop of the <CSSTransition> component is set to location.key to ensure that the transition is applied when the location changes. The timeout prop specifies the duration of the animation in milliseconds. The classNames prop specifies the CSS class names to apply during the transition.

Defining the CSS Animation

To define the CSS animation, we can create a CSS file and define the animation styles. Here’s an example of how to define the CSS animation:

.fade-enter {
  opacity: 0.01;
}

.fade-enter.fade-enter-active {
  opacity: 1;
  transition: opacity 300ms ease-in;
}

.fade-exit {
  opacity: 1;
}

.fade-exit.fade-exit-active {
  opacity: 0.01;
  transition: opacity 300ms ease-out;
}

In this example, the fade-enter class defines the initial state of the entering component, and the fade-enter-active class defines the transition state. The fade-exit class defines the initial state of the leaving component, and the fade-exit-active class defines the transition state.

Related Article: How to Set Up Your First ReactJS Project

Applying the CSS Animation

To apply the CSS animation, we can import the CSS file in our application. Here’s an example of how to apply the CSS animation:

import React from 'react';
import { BrowserRouter, Route, Switch } from 'react-router-dom';
import { CSSTransition, TransitionGroup } from 'react-transition-group';
import Home from './Home';
import About from './About';
import './App.css';

function App() {
  return (
    <BrowserRouter>
      <Route
        render={({ location }) => (
          <TransitionGroup>
            <CSSTransition
              key={location.key}
              timeout={300}
              classNames="fade"
            >
              <Switch location={location}>
                <Route exact path="/" component={Home} />
                <Route path="/about" component={About} />
              </Switch>
            </CSSTransition>
          </TransitionGroup>
        )}
      />
    </BrowserRouter>
  );
}

export default App;

In this example, the classNames prop of the <CSSTransition> component is set to "fade", which corresponds to the CSS animation class names defined in the CSS file. The animation will be applied when the location changes, creating a smooth transition between the entering and leaving components.

Code Snippet: Basic Route Setup

Here’s a code snippet that demonstrates the basic setup for defining routes in a React application using React Router:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Home from './Home';
import About from './About';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />
      <Route path="/about" component={About} />
    </BrowserRouter>
  );
}

export default App;

This code snippet sets up the basic routes for a React application using React Router. The BrowserRouter component is used to provide the routing functionality, and the <Route> components are used to define the routes and their corresponding components. In this example, the Home component will be rendered when the URL path is “/”, and the About component will be rendered when the URL path is “/about”.

Code Snippet: Nested Routes Setup

Here’s a code snippet that demonstrates how to set up nested routes in a React application using React Router:

import React from 'react';
import { BrowserRouter, Route } from 'react-router-dom';
import Home from './Home';
import Products from './Products';
import ProductDetails from './ProductDetails';

function App() {
  return (
    <BrowserRouter>
      <Route exact path="/" component={Home} />
      <Route path="/products" component={Products} />
      <Route path="/products/:id" component={ProductDetails} />
    </BrowserRouter>
  );
}

export default App;

This code snippet sets up nested routes in a React application using React Router. The Home component will be rendered when the URL path is “/”, the Products component will be rendered when the URL path is “/products”, and the ProductDetails component will be rendered when the URL path is “/products/:id”, where “:id” is a dynamic parameter representing the ID of a specific product.

Related Article: How to Render ReactJS Code with NPM

Code Snippet: Redirect After Login

Here’s a code snippet that demonstrates how to redirect the user to a specific page after they successfully log in using React Router:

import React, { useState } from 'react';
import { BrowserRouter, Route, Redirect } from 'react-router-dom';

function LoginPage() {
  const [isLoggedIn, setIsLoggedIn] = useState(false);

  const handleLogin = () => {
    setIsLoggedIn(true);
  };

  if (isLoggedIn) {
    return <Redirect to="/dashboard" />;
  }

  return (
    <div>
      <h1>Login Page</h1>
      <button onClick={handleLogin}>Log In</button>
    </div>
  );
}

function DashboardPage() {
  return <h1>Dashboard Page</h1>;
}

function App() {
  return (
    <BrowserRouter>
      <Route path="/login" component={LoginPage} />
      <Route path="/dashboard" component={DashboardPage} />
    </BrowserRouter>
  );
}

export default App;

This code snippet demonstrates a login page (LoginPage) and a dashboard page (DashboardPage). When the user clicks the “Log In” button on the login page, the handleLogin function is called, which updates the isLoggedIn state to true. As a result, the user is redirected to the dashboard page using the <Redirect> component.

You May Also Like

What Are The Benefits of Using ReactJS in Web Development?

ReactJS has revolutionized web development with its numerous benefits. From the efficiency of its Virtual DOM to the reusability achieved through component-based... read more

The Mechanisms Behind ReactJS’s High-Speed Performance

ReactJS has gained popularity in web development due to its high-speed performance. This article explores the mechanisms behind ReactJS's superior speed, including its... read more

Inserting Plain Text into an Array Using ReactJS

Learn how to add plain text to an array in ReactJS. Understand the syntax and ReactJS method for pushing plain text into an array. Discover the steps to store plain text... read more

Implementing HTML Templates in ReactJS

Using HTML templates in ReactJS can greatly enhance the development process. This tutorial provides a practical approach to implementing HTML templates in ReactJS,... read more

How to to Deploy a ReactJS Application to Production

Deploying ReactJS applications is an essential step in the development process. In this comprehensive walkthrough, you will learn about the importance of deploying... read more

How to Style Components in ReactJS

Styled component buttons in ReactJS can be created without an onClick event. However, understanding the necessity of an onClick event for these buttons is crucial for... read more