Hooked with React - Using react router in our react application to route a dynamic page

We have already created the books list page for our book search page. Now lets create another page for each book using react router.

You can check the app in action here,

https://codesandbox.io/s/rl20opm2l4

Creating books detail page

First lets create the routes using react router in App.js and load two pages

  • Index page which is our current search page which shows the books list
  • Books detail page which will be identified through unique ID

Moving all our logic to index page. Create a new folder called pages and create a file called searchPage.js

import React, { useState } from 'react'
import axios from 'axios'

import BookSearchForm from '../components/bookSearchForm'
import Loader from '../components/loader'
import BooksList from '../components/booksList'

const SearchPage = () => {
  const [searchTerm, setSearchTerm] = useState('')
  const [books, setBooks] = useState({ items: [] })
  const [error, setError] = useState(false)
  const [loading, setLoading] = useState(false)

  let API_URL = `https://www.googleapis.com/books/v1/volumes`

  const fetchBooks = async () => {
    setLoading(true)
    setError(false)
    try {
      const result = await axios.get(`${API_URL}?q=${searchTerm}`)
      setBooks(result.data)
    } catch (error) {
      setError(true)
    }
    setLoading(false)
  }

  const onInputChange = (e) => {
    setSearchTerm(e.target.value)
  }

  const onSubmitHandler = (e) => {
    e.preventDefault()
    fetchBooks()
  }

  return (
    <>
      <BookSearchForm
        onSubmitHandler={onSubmitHandler}
        onInputChange={onInputChange}
        searchTerm={searchTerm}
        error={error}
      />
      <Loader searchTerm={searchTerm} loading={loading} />
      <BooksList books={books} />
    </>
  )
}

export default SearchPage

and refactor the App.js file

import React from 'react'

import SearchPage from './pages/searchPage.js'
import './App.css'

const App = () => {
  return (
    <>
      <SearchPage />
    </>
  )
}

export default App

Add react-router-dom package and render the search page through routes

yarn add react-router-dom
import React from 'react'
import { BrowserRouter as Router, Route, Switch } from 'react-router-dom'

import SearchPage from './pages/searchPage.js'
import './App.css'

const NoMatchRoute = () => <div>404 Page</div>

const App = () => {
  return (
    <Router>
      <Switch>
        <Route path="/" exact component={SearchPage} />
        <Route component={NoMatchRoute} />
      </Switch>
    </Router>
  )
}

export default App

Lets create the details page and route through our react router. Create a new file bookDetailPage.js in pages folder

import React from 'react'

const BookDetailPage = () => {
  return <div>Book details page</div>
}

export default BookDetailPage

Add the route to the App.js. Here the path will have the route params bookId to identify the book through its ID.

<Route path="/book/:bookId" exact component={BookDetailPage} />

Now get the book ID in BookDetailPage through props send by react-router. If you want to see details of how it works, refer my blog post about dynamic pages in react router.

Route params are send through a props called match.

import React from 'react'

const BookDetailPage = ({ match }) => {
  const {
    params: { bookId },
  } = match

  return (
    <div>
      Book details page: <strong>{bookId}</strong>
    </div>
  )
}

export default BookDetailPage

Link the details page from books list component

Add a link to go to details page in BooksList component.

import { Link } from "react-router-dom";

...

<Link to={`/book/${book.id}`}>Show details</Link>

Link is used because, it uses client side routing. HTML a tag will refresh the page and hit the server to load the new URL

Alright, we have done linking between the pages. Now we need to show details about the book in the details page. For that,

  • we need to call the API with book ID and fetch the details and render the output. Lets do it.
  • While calling the API, we need to set loading state.
  • If API throws error, we need to set error state.
  • If API returns content, then set the book state.

Here is the code for books detail page,

  • It uses useEffect react hooks to call the book detail API only when the page mounts. To know more about useEffect hook, check this official docs here.
import React, { useState, useEffect } from 'react'
import { Link } from 'react-router-dom'
import axios from 'axios'

import BookDetail from '../components/bookDetail'

const BookDetailPage = ({ match }) => {
  const {
    params: { bookId },
  } = match
  const [book, setBook] = useState(null)
  const [error, setError] = useState(false)
  const [loading, setLoading] = useState(false)

  useEffect(() => {
    const API_BASE_URL = `https://www.googleapis.com/books/v1/volumes`
    const fetchBook = async () => {
      setLoading(true)
      setError(false)
      try {
        const result = await axios.get(`${API_BASE_URL}/${bookId}`)
        setBook(result.data)
      } catch (error) {
        setError(true)
      }
      setLoading(false)
    }
    // Call the API
    fetchBook()
  }, [bookId])

  return (
    <>
      <Link to={`/`}>Go back to search books</Link>
      {loading && (
        <div style={{ color: `green` }}>
          loading book detail for book ID: <strong>{bookId}</strong>
        </div>
      )}
      {error && (
        <div style={{ color: `red` }}>
          some error occurred, while fetching api
        </div>
      )}
      {book && <BookDetail book={book} />}
    </>
  )
}

export default BookDetailPage

It uses a new component BookDetail to render the books detail. bookDetail.js contains

import React from 'react'

import { bookAuthors } from '../utils'

const BookDetail = ({ book }) => {
  const createDescMarkup = (description) => {
    return { __html: description }
  }

  return (
    <section>
      <div>
        <img
          alt={`${book.volumeInfo.title} book`}
          src={`http://books.google.com/books/content?id=${book.id}&printsec=frontcover&img=1&zoom=1&source=gbs_api`}
        />
        <div>
          <h3>
            <strong>Title:</strong> {book.volumeInfo.title}
          </h3>
          <p>
            <strong>Authors:</strong> {bookAuthors(book.volumeInfo.authors)}
          </p>
          <p>
            <strong>Published Date:</strong> {book.volumeInfo.publishedDate}
          </p>
          <p>
            <strong>Publisher:</strong> {book.volumeInfo.publisher}
          </p>
          <p>
            <strong>Page Count:</strong> {book.volumeInfo.pageCount}
          </p>
          <div
            dangerouslySetInnerHTML={createDescMarkup(
              book.volumeInfo.description,
            )}
          />
        </div>
      </div>
    </section>
  )
}

export default BookDetail

Now we have successfully rendered the whole page. You can go back to search page and search for any books and check their details.

Try it out here,

https://codesandbox.io/s/books-detail-page-by2nj

Thats it folks, we can further extend these apps with styles and testing. I would either write it as continuation or as separate blog post soon 😎

Checkout the codebase for this part 4 here and the whole series codebase can be referred here.

Beginners to ProNode Js

Visual Guide to API Design Best Practices

This visual eBook covers essential best practices for designing robust APIs using REST principles.

This book is ideal for beginners and backend developers seeking to enhance their API design skills. However, it is not suited for those seeking an in-depth exploration of API design. This book is a quick read under 40 slides like scrolling through your instagram feed.

Visual Guide to API Design Best Practices