Next.js is one of the easiest and most popular ways to build production-ready React applications. Over recent years, Next.js has experienced exponential growth, and many companies have adopted it to build their applications. In this article, we will learn how to add pagination to a Next.js application. We’ll talk about how to set up a Next.js project, data fetching from an API endpoint with getStaticProps
method, and implementing the pagination itself.
Jump ahead:
getStaticProps
Pagination
componentpagination helper
functionFirst, we will create a new Next.js app with the command npx create-next-app next-pagination
. Then, we can go into the project directory with the command cd next-pagination
and run yarn dev or npm run dev
to start the project on a development server. For this project, we will use vanilla CSS for styling as it focuses more on functionality.
getStaticProps
The data that will be used to implement our pagination will come from the {JSON} Placeholder API
. The data will be fetched using the getStaticProps
function.
The getStaticProps
function always runs on the server, and Next.js will pre-render the page at build-time using the props returned by getStaticProps
. The Fetch API will be used to get data from the previously mentioned API endpoint:
import Head from "next/head"; import Image from "next/image"; import styles from "../styles/Home.module.css"; import Pagination from "../src/components/Pagination"; import { useState, useEffect } from "react"; import { paginate } from "../src/helpers/paginate"; export default function Home({ data }) { console.log(data) return ( <div className={styles.container}> <Head> <title>Create Next App</title> <meta name="description" content="Generated by create next app" /> <link rel="icon" href="/favicon.ico" /> </Head> <p>NextJS X Pagination</p> export const getStaticProps = async () => { const res = await fetch("https://jsonplaceholder.typicode.com/todos"); const data = await res.json(); return { props: { data }, }; };
The data returned from the props will be destructured in the home
component so it can be logged in the console. This will confirm that the data has been fetched. So, in the console, we should have a post with an array of 100
objects as it was specified in the JSON Typicode website:
Now, we can render this data on the webpage and see how they look on the UI:
import Head from "next/head"; import Image from "next/image"; import styles from "../styles/Home.module.css"; export default function Home({ data }) { return ( <div className={styles.container}> <Head> <title>Create Next App</title> <meta name="description" content="Generated by create next app" /> <link rel="icon" href="/favicon.ico" /> </Head> <p> <strong>NextJS x Pagination</strong> </p> {data.map((item) => { return <p key={item.id}>{item.title}</p>; })} </div> ); } export const getStaticProps = async () => { const res = await fetch("https://jsonplaceholder.typicode.com/todos"); const data = await res.json(); return { props: { data }, }; };
Pagination
componentFor the Pagination
component implementation, we will create an src
and component
folder. Inside component
, we will create a Pagination.js
file, which will be rendered in the index.js
file:
const Pagination = () => { return ( <div>Pagination</div> ) } export default Pagination
The Pagination
rendered in index.js
will have four props: items
, currentPage
, pageSize
, and onPageChange
. The items
will be the length of the data we are getting from the API, which in this case, is 100
:
The pageSize
will be set to 10
since we want to have pagination from 1-10
, and the currentPage
will be stored in a state with a default value of 1
since the page will start from page one.
The onPageChange
will be a function to set the current page we are in, for example, moving from page one to two:
import Head from "next/head"; import Image from "next/image"; import styles from "../styles/Home.module.css"; import Pagination from "../src/components/Pagination"; import { useState } from "react"; export default function Home({ data }) { const [currentPage, setCurrentPage] = useState(1); const pageSize = 10; const onPageChange = (page) => { setCurrentPage(page); }; return ( <div className={styles.container}> <Head> <title>Create Next App</title> <meta name="description" content="Generated by create next app" /> <link rel="icon" href="/favicon.ico" /> </Head> <p> <strong>NextJS x Pagination</strong> </p> {data.map((item) => { return <p key={item.id}>{item.title}</p>; })} <Pagination items={data.length} // 100 currentPage={currentPage} // 1 pageSize={pageSize} // 10 onPageChange={onPageChange} /> </div> ); } export const getStaticProps = async () => { const res = await fetch("https://jsonplaceholder.typicode.com/todos"); const data = await res.json(); return { props: { data }, }; };
Then, we will destructure these four props in Pagination
and use them for the pagination implementation:
import styles from "../../styles/Home.module.css"; const Pagination = ({ items, pageSize, currentPage, onPageChange }) => { const pagesCount = Math.ceil(items / pageSize); // 100/10 if (pagesCount === 1) return null; const pages = Array.from({ length: pagesCount }, (_, i) => i + 1); console.log(pages) return ( <div> <div>Pagination</div> </div> ); }; export default Pagination;
The items
will be divided by the pageSize
and stored in a pagesCount
variable. The Array.from()
static method will be used to create a new Array
instance from the pagesCount
length, which is 10
.
Let’s log the pages
in the console and see what we have:
Now, we can map over the pages array
and render the value (1-10)
inside it. There will be an anchor
element for each of the values where the onClick
function will be implemented:
const Pagination = ({ items, pageSize, currentPage, onPageChange }) => { const pagesCount = Math.ceil(items / pageSize); // 100/10 if (pagesCount === 1) return null; const pages = Array.from({ length: pagesCount }, (_, i) => i + 1); return ( <div> <ul className={styles.pagination}> {pages.map((page) => ( <li key={page} className={ page === currentPage ? styles.pageItemActive : styles.pageItem } > <a className={styles.pageLink} onClick={() => onPageChange(page)}> {page} </a> </li> ))} </ul> </div> ); }; export default Pagination;
The styling of the pagination
boxes will be done in the Home.module.css
:
.container { padding: 0 2rem; } .pagination { display: flex; justify-content: space-between; align-items: center; list-style: none; } .pageLink { cursor: pointer; } /* pagination pageItem */ .pageItem { display: flex; justify-content: center; align-items: center; width: 2rem; height: 2rem; border: 1px solid #eaeaea; border-radius: 0.5rem; cursor: pointer; } /* pagination page item when active */ .pageItemActive { display: flex; justify-content: center; align-items: center; width: 2rem; height: 2rem; border: 1px solid #eaeaea; border-radius: 0.5rem; cursor: pointer; background-color: red; }
Once we scroll down to the end of the posts
data, we should see boxes from 1-10
being rendered. The onPageChange
function is passed to the anchor
element with the argument of page(1-10)
, so when any part of the box is clicked, it will set the current page to the page number clicked.
Remember, the onPageChange
function in the index.js
file:
const onPageChange = (page) => { setCurrentPage(page); };
Now, let’s see what we have:
pagination helper
functionNow, we have been able to implement the pagination in our app, but we still have 100 posts being rendered for the first page instead of 10. To implement this, we will create a helper
function in the paginate.js
file:
export const paginate = (items, pageNumber, pageSize) => { const startIndex = (pageNumber - 1) * pageSize; return items.slice(startIndex, startIndex + pageSize); };
The paginate
file will have three arguments: items
, pageNumber
, and pageSize
. This function will be called in the index.js
file and the argument will be passed as follows:
const paginatedPosts = paginate(data, currentPage, pageSize);
Here, the data
is passed as the items
, the array of data coming back from the API endpoint. The currentPage
represents the pageNumber
, and the pageSize
is passed in as the pageSize
.
In the pagination
function, the startIndex
is first extracted by subtracting the currentPage
number from 1
and multiplying it by the pageSize(10)
. Then, we slice the items
. This is the array of data from the startIndex
that we got initially until we got the startIndex
+ pageSize
value.
For example, on the first page, the pageNumber
will be 1
, so startIndex
will be: 1-1 * 10 gives 0
.
Then, startIndex (0) + pageSize
(declared in index.js
as 10
) will result in 10
. So, the slicing starts from the Array length
of 0
till 9
.
Let’s log the paginatedPosts
to the console to see what we have:
We can check what we now have on the webpage:
And, we can paginate through all the pages:
There we have it!
Thank you for reading! I hope this tutorial gives you the knowledge needed to add pagination to your Next.js app. You can customize the styling to your use case, as this tutorial majorly focuses on the logic of implementing the functionality. You can find this project’s complete code in my repository and the deployed project on Vercel here. Happy coding!
Debugging Next applications can be difficult, especially when users experience issues that are difficult to reproduce. If you’re interested in monitoring and tracking state, automatically surfacing JavaScript errors, and tracking slow network requests and component load time, try LogRocket.
LogRocket is like a DVR for web and mobile apps, recording literally everything that happens on your Next.js app. Instead of guessing why problems happen, you can aggregate and report on what state your application was in when an issue occurred. LogRocket also monitors your app's performance, reporting with metrics like client CPU load, client memory usage, and more.
The LogRocket Redux middleware package adds an extra layer of visibility into your user sessions. LogRocket logs all actions and state from your Redux stores.
Modernize how you debug your Next.js apps — start monitoring for free.
Would you be interested in joining LogRocket's developer community?
Join LogRocket’s Content Advisory Board. You’ll help inform the type of content we create and get access to exclusive meetups, social accreditation, and swag.
Sign up nowJavaScript generators offer a powerful and often overlooked way to handle asynchronous operations, manage state, and process data streams.
webpack’s Module Federation allows you to easily share code and dependencies between applications, helpful in micro-frontend architecture.
Whether you’re part of the typed club or not, one function within TypeScript that can make life a lot easier is object destructuring.
useState
useState
can effectively replace ref
in many scenarios and prevent Nuxt hydration mismatches that can lead to unexpected behavior and errors.
7 Replies to "How to add pagination to your Next.js app"
I am sorry but I fail to see the point of pagination implemented this way.
The code still fetches all 100 items from the backend. The idea of pagination is not only to split the long array of items for better viewing (arguably, a long scroll is better than a bunch of pages) but to prevent fetching too much from the API server or database.
This example fails to deliver.
Agreed, this will keep the same 100 items fetched at build time and nothing else. I am searching for a way to apply server side pagination.
This blog post should be removed, this is not how pagination should work. As @Suslik says, there is no point to pagination this way if we just fetch all of the items.This does not solve the application of server side pagination.
I am on it too, I have a way to do it, can we connect??
I have solved it, exam we connect
Regardless of what Suslik, Alberto, Jade, or anyone who previously left a reply might have said,
I have successfully implemented the logic and the final export ‘paginate’ as you instructed.
When I searched the internet for pagination-related examples, I could only find frustrating ones that work with React but not with Next.js due to hydration issues.
I attempted to implement libraries like Redux, Redux Toolkit, React Query, and others, but they didn’t apply straightforwardly.
@Taofiq Aiyelabegan, I don’t know where on Earth you posted this, but you’ve greatly assisted a working professional living in a land of Park Ji-sung, Kim Yuna, Son Heung-min, BTS, New Jin’s, and the K-pop phenomenon.
From a distant place, I wish you all the best. God bless you.
Really helpful, thanks