pagination - NextJS 中的 Apollo 分页
问题描述
我已经在 Nextjs 中实现了分页。但我认为这不是一个好方法。我用 JS 切片方法实现了分页..
像这样实现分页是不是很糟糕?
我正在使用apollo client + apollo-server + mongodb(to save blog posts)+ graphql
堆栈
我知道其他人没有像这样实现分页..
让我知道这样实现是可以的,如果不是这样,请告诉我实现分页的正确方法。
/components/pagination.tsx
import React from 'react';
interface PaginationProps {
postsPerPage: number
totalPosts: number
paginate: (n: number) => void
}
const Pagination : React.FC<any> = ({ postsPerPage, totalPosts, paginate } : PaginationProps) => {
const pageNumbers : number[] = [];
for (let i = 1; i <= Math.ceil(totalPosts/postsPerPage); i++) {
pageNumbers.push(i);
}
return (
<div>
<nav>
<ul className="bg-purple-400 float-left mt-4 mb-4 text-center rounded-md text-white p-px">
{pageNumbers.map(number => (
<li key={number} className="inline-block text-base font-semibold rounded-md p-1 w-6 hover:bg-purple-500">
<span onClick={() => paginate(number)} className="text-white">
{number}
</span>
</li>
))}
</ul>
</nav>
</div>
)
}
export default Pagination
/pages/index.tsx
import type { NextPage } from 'next'
import client from "../apollo-client";
import Link from 'next/link'
import Head from "next/head";
import {GET_ALL_POSTS} from "../lib/query";
import {useState} from "react";
import Pagination from "../components/pagination";
export async function getStaticProps() {
const { data } = await client.query({
query: GET_ALL_POSTS
})
return {
props : {
posts: data.posts
},
revalidate: 10
}
}
const Home: NextPage = ({posts} : any) => {
// Pagination
const [currentPage, setCurrentPage] = useState(1);
const [postsPerPage] = useState(10)
const lastIndex = currentPage * postsPerPage;
const firstIndex = lastIndex - postsPerPage;
const currentPosts = (tmp : object[]) => {
let currentPosts : object[] = [];
currentPosts = tmp.slice(firstIndex, lastIndex);
return currentPosts;
}
return (
<div>
<Head>
<title>...</title>
</Head>
<h1 className="text-gray-550 dark:text-white">전체 글({posts.length})</h1>
<div className="flex items-center flex-col justify-center list-none">
{currentPosts(posts).map((post : any, index : any) => (
<Link href={`/posts/${post._id}`} key={index}>
<a className="group container mt-5 p-4 border-b-2">
<h3 className="text-black text-xl dark:text-white mt-4 group-hover:text-purple-500">{post.title}</h3>
<p className="text-black dark:text-white mb-4">{renderDate(post.createdAt)}</p>
{renderTags(post.tags)}
</a>
</Link>
))}
<Pagination postsPerPage={postsPerPage} totalPosts={posts.length} paginate={setCurrentPage}/>
</div>
</div>
)
}
export default Home
解决方案
基本上现在您正在预先获取所有帖子并动态计算它们所需的页面数量。根据帖子的数量,这可能是用户可能不需要的大量数据。
当用户需要数据时,最好以增量方式获取数据。Apollo有一个Pagination api可以帮助你。从文档:
服务器可以为特定的列表字段使用许多不同的分页策略:基于偏移量、基于光标、基于页码、向前、向后等。每个策略都需要一组略有不同的参数。因为这些策略在不同的情况下都有用,所以 Apollo 和 GraphQL 规范都没有规定规范的分页策略。
我能给你的最好建议是通读文档并找到你认为最适合你情况的一种方法。您将大量使用的是fetchMore
功能,它允许您在用户与您的网站交互时获取更多帖子。这具有积极的副作用,您可能不需要revalidate
每 10 秒调用一次。您可以将查询移动到Pagination
组件并在那里获取您想要在一页上显示的帖子数量。
推荐阅读
- python - Postgres:返回由整数组成的复合对象作为整数数组的函数
- visual-studio - Visual Studio ASP.NET Core 调试 IIS Express - 无法访问站点
- javascript - 输入中的日期格式
- wpf - WPF:样式覆盖属性
- antlr4 - 为什么词法分析器规则名称单词导致解析错误
- azure - 通过 REST API 从 Azure Devops 构建中获取使用的管道工件
- java - BigQuery 出现问题
- flutter - 使用flutter_inappwebview加载网页时的CircularProgressIndicator在ios中不起作用
- r - 在R中的光栅.tif文件中设置最大值
- docker - Dockerfile:使用 ADD 文件作为 dockerfile 的第一行,而不是 FROM