dev-resources.site
for different kinds of informations.
How to add dynamic routing to your react single page application
This article aims to explain how to create routes dynamically and make dynamic Api calls, depending on the selected route.
For this demonstration, we will use the example of an online-shop web prototype.
First, create your react app and install react-router-dom
Next, in our
</App>
component (or wherever you decideto locate your routes ) create the route path as a dynamic route. A dynamic route has this shape ::routeName
.
function App() {
return (
<>
<Navigation />
<Routes>
<Route path="category/:categoryName" element={<Categories />} />
</Routes>
</>
);
}
- Create the navigation links in component
<Navigation/>
, each of them will have a unique path (different product categories)
The API used for this demonstration is fakestore.api. FakeStore allows us to fetch by category. When the data is ready, we store the catogories in our state.
const [categories, setCategories] = useState<string[]>([]);
useEffect(() => {
const fetchProductCategories = () => {
fetch("https://fakestoreapi.com/products/categories")
.then((res) => res.json())
.then((json) => setCategories(json));
};
fetchProductCategories();
}, []);
To display the categories in our JSX template we can either display them statically or map through our categories
array. If so, for each category, we will add a link with our category name.
const Navigation = () => (
<Nav>
{categories.map((cat) => (
<Nav.Link as={Link} to={`category/${encodeURIComponent(cat)}`}> // route with encoded URI string
{cat.toUpperCase()}
</Nav.Link>
))}
</Nav>
);
Keep in mind that if your string has spaces or special characters, you will need to use the method encodeURIComponent()
, which will turn a string such as "men's clothing"
into a URI:
https://localhost:3000/category/men's%20clothing
- After we're done with defining the paths of each of our links, the component that will render the content (
Category.tsx
) has to receive the parameter from the router (in this example:categoryName
). This parameter will be different every time, depending on which path we click on our navigation.
export const Category = () => {
const { categoryName } = useParams(); // we receive the dynamic route value from routes
useEffect(() => {
const fetchproducts = () => {
setIsLoading(true);
fetch(`https://fakestoreapi.com/products/category/${categoryName}`) // we use the parameter to fetch the products of the chosen category in <Navigation/>
.then((res) => res.json())
.then((json) => {
setProducts(json);
setIsLoading(false);
});
};
fetchproducts();
}, [categoryName]);
return ... // return component jsx
};
We will use this parameter in the API call url to dinamically fetch our categories. If the categoryName
exists, the products of that category will be rendered; else, nothing will be returned.
In a nutshell:
A location/route is defined
.../category/electronics
.The router receives the element
/category/:categoryName
where:categoryName
is, for example, "electronics".Through
useParams()
we get a parameter with that reference and use it in our<Category/>
component to fetch the specific products for that route.
Hopefully you can appreciate this better with the assistance of the diagram below:
Featured ones: