If I had a million unique gemstones to sell on my e-commerce web app, then I wouldn’t have much space on my home page to display them all. Pagination separates those million little gemstones and displays only a few at a time. With React and Ruby on Rails, pagination requires getting a count of all the items in inventory, dynamically routing & creating pages, and displaying the correct items on each page.
How many pages will I need to evenly split up all the items in inventory? Count the total number of items and divide the count by however many items each page will display. On the backend, to get the count, add the following line to routes.rb, which directs a Get request from the end point /items_count to a function called count.
get "/items_count", to: "items#count"
In the file items_controller.rb the function count uses a method, count, on the model Item and returns the total count of all the items in inventory.
def count
@count = Item.count
render json: @count
end
On the frontend, create a component, Pagination, and make a Get request using fetch to the end point the backend is now equipped to handle. The backend sends a response with the count of all of the items in inventory, which I store in state.
const getCount = async () => {
const response = await fetch(`${url}/items_count`);
const data = await response.json();
await setItemCount(data);
};
React.useEffect(() => {
getCount();
}, []);
Now that I have a count, I can determine the number of pages to create and create routes to each one. I don’t want to hard code each page or routes for each page. Instead I will dynamically create and route pages, telling React how to figure what page it’s on. React Router, the service I use to navigate between pages, comes with a component, Switch, that has child components, Route, which render components that correspond to specific url endpoints. Since I want to add pagination to my home page, I tell Route to render the component ItemsPage any time the url ends with “/”, “/items”, or “/items/:id”. The last route, “/items/:id”, is how React will know what page to render.
<Route
path={["/items/:id", "/", "/items"]}
component={ItemsPage}
/>
Render <ItemsPage> component for routes "/", "/items", & "/items/:id"
Create <Pagination> component
const { globalState } = React.useContext(GlobalContext);
const { url } = globalState;
const [itemCount, setItemCount] = React.useState();
const perPage = 2;
const pageCount = Math.ceil(itemCount / perPage);
<Link to={page > 1 && `/items/${page - 1}`}>
<Link to={page < pageCount && `/items/${page + 1}`}>
Add <Pagination> component to <ItemsPage>
const page = parseInt(props.match.params.id);
return (
<div>
<Pagination page={page || 1} />
<Items page={page || 1} />
</div>
);
Create <Items> component
const { globalState } = React.useContext(GlobalContext);
const { url } = globalState;
const [items, setItems] = React.useState([]);
make get request to API
const getItems = async () => {
const repsonse = await fetch(`${url}/items?page=${page}`);
const data = await repsonse.json();
setItems(data);
};
React.useEffect(() => {
getItems();
}, [page]);
{items.map((item) => (
<Item item={item} key={item.id} />
))}
Add 'will_paginate' to Gemfile
gem 'will_paginate', '~> 3.1.0'
Use 'will_paginate' in items index route to render a specific number of items per page
def index
@items = Item.paginate(page: params[:page], per_page: 2)
render json: @items
end