Introduction to SPAs

Creating a Single Page Application (SPA) with fixed navbar and footer components.

Learning Goals

At the end of this Tutorial, you will be able to:

  • Create a Single Page Application (SPA) with fixed navbar and footer components.

About Single Page (SPA) Applications

In a traditional multi-page website, each page is a different HTML file. Navigating to different pages causes a full page reload, fetching a new HTML document from the server. This is called Server-Side Rendering (SSR). Content is typically rendered on the server, and the complete HTML is sent to the client.

In contrast, a React app dynamically updates the current page rather than loading entire new pages from the server. Content is loaded once, and new data is retrieved as needed without page reloads.

This is called Client-Side Rendering (CSR). The server sends a single HTML page initially, and the content is then rendered on the client side using JavaScript. Such apps are known as Single Page Applications or SPAs.

Starting with create-react-app

Follow these steps to create your basic app structure with the create-react-app script.

  1. Open a new Command prompt window or, in VS Code, a new Terminal.
  2. Navigate to where you want ReactJS to create a folder for your app. For example:
    C:\> users\JohnSmith
  3. Type the following command that includes the name you want to call your new app. For example, spa-blog.
    npx create-react-app spa

After you have created your app, you launch it by running the following command from inside your new app folder.

npm start

A new browser window should display with your new app running on the ReactJS local development server. If not, open a new browser tab and enter http://localhost:3000.

You will see a screen similar to the following.

ReactJS sample screen

Customising your CRA-created content

Now you will customise the 'boilerplate' content provided by the create-react-app (CRA) script.

The /public folder

  1. In your app’s /public folder, open the index.html file and replace all its content with the following.
    <!DOCTYPE html>
       <html lang="en">
       <head>
          <meta charset="utf-8" />
          <meta name="viewport" content="width=device-width, initial-scale=1" />
          <title>Single Page Application Blog App</title>
          <meta name="description" content="An SPA blog app created with ReactJS" />   
          <link rel="manifest" href="%PUBLIC_URL%/manifest.json" />
       </head>
    	 
       <body>
    	 
          <div id="root"></div>
    		   
       </body>
     </html>

The /src folder

  1. In the /src folder, open the index.js file and replace all its content with the following.
    import React from 'react';
    import ReactDOM from 'react-dom/client';
    import App from './App';
    	
    const root = ReactDOM.createRoot(document.getElementById('root'));
    
    root.render(
       <React.StrictMode>
          <App />
       </React.StrictMode>
    );
  2. And again in the /src folder, open the App.js file and replace its content with the following.
    function App() {
       return (
          <>
          <h1>Hello, World!</h1>
          <p>Paragraph of text.</p>
          </>
       );
    }	
    export default App;

Your web page should now look as shown below.

Sample React screen

Working with react-router-dom

ReactJS does not natively support page routing. So you need to use NPM (Node Package Manager) to install a new package named react-router-dom for this purpose. You can then import and use that package in your app.

Installing the react-router-dom package

  1. Open a new Command prompt window or, in VS Code, a new Terminal.
  2. Navigate to the folder that contains your app. For example:
  3. Run the following command inside your app folder:
    npm install react-router-dom
    For example:

Importing react-router-dom into your App component

  1. At the top of your App.js component file, add the following import statement:
    import { BrowserRouter, Routes, Route } from "react-router-dom";
  2. Developers often shorten the name of the BrowserRouter function with the Router alias. See below:
    import { BrowserRouter as Router, Routes, Route } from "react-router-dom";

Here is a summary of what these three imported named functions do:

BrowserRouter

Provides the routing functionality for your app.

Routes

Defines the routes for your app.

Route

Defines a single route.

Adding the ‘pages’ to your app

Adding ‘page’ (really components) to an app is a three-step process:

  • Create the component files that will be served as ‘page’.
  • Import these component files into your main or ‘parent’ App component.
  • Include page routes in your App component.

Creating your page components

Each ‘page’ in your SPA will be a component file, stored in a /pages sub-folder of your /src folder. Follow these steps:

  1. In your app’s /src folder, create a sub-folder named /pages.
  2. Download the files below to your new /src/pages sub-folder.
    Home.js
    Products.js
    Contact.js
    Navbar.js
    Footer.js
    PageNotFound.js
  3. Save the following file in your /src folder. It will overwrite the stylesheet file of the same name created by the create-react-app script.
    App.css

Importing the page components to your App component

At the top of the App.js file, after the react-router-dom import, import your three page components:

import Home from "./pages/Home";
import Products from "./pages/Products";
import Contact from "./pages/Contact";

Adding the page routes in your App component

Now you will add routes to the main App() component to enable users to navigate between the pages of your SPA.

Update the App() function as shown below. Each Route now has a path and an element prop:

function App() {
   return (
    <Router>
        <Routes>
            <Route path="/" element = {<Home />} />
            <Route path="/products" element = {<Products />} />
            <Route path="/contact" element = {<Contact />} />
        </Routes>
    </Router>
   );
}

Hosting your SPA on GitHub Pages

if you plan to host your SPA on Github Pages, update the <Router> as follows

<Router basename = {process.env.PUBLIC_URL} >

Verifying your app’s URL paths

In your web browser, verify that your three web pages display correctly by entering the following web addresses.

  • http://localhost:3000
  • http://localhost:3000/products
  • http://localhost:3000/contact

Adding an error route

It is good practice to create a page route to handle situations where the user enters the web address of a page that does not exist in your SPA. For example:

  • http://localhost:3000/abc123

You can do this with the wildcard asterisk * character as follows.

<Route path="*" element = "Page does not exist." />

A better option would be to use a special web page for this purpose and add a route to it as follows:


import PageNotFound from "./pages/PageNotFound";

<Route path="*" element = { <PageNotFound /> } />

Note that this will work work if you host your SPA on GitHub Pages.

Adding a CSS component to your app

Add the import statement below to your App.js file.

import "./App.css";