跟着官方示例学习 @tanStack-table --- Pagination

🌲系列一:跟着官方示例学习 @tanStack-table --- Basic

🌲系列二:跟着官方示例学习 @tanStack-table --- Header Groups

🌲系列三:跟着官方示例学习 @tanStack-table --- Column Filters

🌲系列四:跟着官方示例学习 @tanStack-table --- Column Ordering

🌲系列五:跟着官方示例学习 @tanStack-table --- Sticky Column Pinning

🌲系列六:跟着官方示例学习 @tanStack-table --- Column Sizing

🌲系列七:跟着官方示例学习 @tanStack-table --- Expanding


今天我们来聊聊一个经典而又不可忽视的功能:分页(Pagination)!没错,当你的数据量一多起来,分页就是你的好朋友,能帮你分门别类、有条不紊地展示数据。

🌬️ 前端分页

🛠 基础准备:Pagination 的状态管理

TanStack Table 没有强制你用它内置的分页逻辑,但如果你想用起来也非常方便。我们一般用 ReactuseState 来管理分页状态,比如这样:

tsx 复制代码
const [pagination, setPagination] = React.useState<PaginationState>({
  pageIndex: 0,
  pageSize: 10,
});
  • pageIndex: 当前页(从 0 开始)

  • pageSize: 每页多少条数据

然后把它传给 useReactTablestate.paginationonPaginationChange 👇

tsx 复制代码
const table = useReactTable({
  data,
  columns,
  getCoreRowModel: getCoreRowModel(),
  getPaginationRowModel: getPaginationRowModel(), // 🚨 别忘了这个
  state: {
    pagination,
  },
  onPaginationChange: setPagination,
})

📌 getPaginationRowModel() 是分页的关键插件,没有它分页就不工作哟~

📋 表格渲染 & 分页操作面板

我们再来看看分页操作的 UI 控制,像这样:

tsx 复制代码
<button onClick={() => table.firstPage()} disabled={!table.getCanPreviousPage()}>
  {'<<'}
</button>
<button onClick={() => table.previousPage()} disabled={!table.getCanPreviousPage()}>
  {'<'}
</button>
<button onClick={() => table.nextPage()} disabled={!table.getCanNextPage()}>
  {'>'}
</button>
<button onClick={() => table.lastPage()} disabled={!table.getCanNextPage()}>
  {'>>'}
</button>

是不是很直观?TanStack Table 提供了这些非常人性化的方法:

  • table.firstPage() → 跳转到第一页

  • table.previousPage() → 上一页

  • table.nextPage() → 下一页

  • table.lastPage() → 跳转到最后一页

  • table.getCanPreviousPage() / table.getCanNextPage() → 判断是否还能翻页

而我们也可以加一个 跳页输入框 和 页大小切换器,让交互更灵活:

tsx 复制代码
<input
  type="number"
  onChange={e => table.setPageIndex(Number(e.target.value) - 1)}
  defaultValue={table.getState().pagination.pageIndex + 1}
/>

<select
  value={table.getState().pagination.pageSize}
  onChange={e => table.setPageSize(Number(e.target.value))}
/>

🔗 Pagination API

🤔 分页和服务端结合?

一般来说,在一个实际项目中,数据是从服务器拉取的,那分页的逻辑可不能全靠前端来搞定了。此时就轮到我们的"服务端分页"大显身手啦!

我们需要做两件事:

1️⃣ 使用 manualPagination: true 通知表格:分页由我接管

tsx 复制代码
manualPagination: true,

这个配置告诉 TanStack Table:"嘿,我要自己处理分页逻辑啦,别给我管页数和数据啦!"💪

同时注意:既然分页逻辑是手动的,就不需要使用 getPaginationRowModel() 了。

2️⃣ 使用 @tanstack/react-query 拉数据,监听分页变化自动刷新

我们使用 useQuery 来获取数据,并把 pagination 状态作为 queryKey 的一部分。

tsx 复制代码
const dataQuery = useQuery({
  queryKey: ['data', pagination],
  queryFn: () => fetchData(pagination), // 👈 你根据分页参数请求数据
  placeholderData: keepPreviousData, // 🤩 保持上次的数据,避免翻页时空白闪屏
})

是不是很丝滑?还可以用 isFetching 显示加载动画,用户体验 +1!

tsx 复制代码
{dataQuery.isFetching ? 'Loading...' : null}

3️⃣ 配置表格的核心参数

表格的 data 直接来自 dataQuery.data?.rows,并且还要告诉它:总共有多少行数据(rowCount):

tsx 复制代码
const table = useReactTable({
  data: dataQuery.data?.rows ?? [],
  columns,
  rowCount: dataQuery.data?.rowCount, // 👈 告诉表格总共有多少行
  state: {
    pagination,
  },
  onPaginationChange: setPagination,
  manualPagination: true,
  getCoreRowModel: getCoreRowModel(),
})

🔥 小贴士:从 v8.13.0 开始,你也可以用 rowCount 来自动计算 pageCount,不再非得手动传 pageCount,是不是方便很多!

相关推荐
im_AMBER7 分钟前
weather-app开发手记 04 AntDesign组件库使用解析 | 项目设计困惑
开发语言·前端·javascript·笔记·学习·react.js
用泥种荷花8 分钟前
VueCropper加载OBS图片跨域问题
前端
董世昌4115 分钟前
什么是事件冒泡?如何阻止事件冒泡和浏览器默认事件?
java·前端
Bigger17 分钟前
在 React 里优雅地 “隐藏 iframe 滚动条”
前端·css·react.js
小沐°18 分钟前
vue3-ElementPlus出现Uncaught (in promise) cancel 报错
前端·javascript·vue.js
四瓣纸鹤23 分钟前
F2图表在Vue3中的使用方法
前端·javascript·vue.js·antv/f2
web前端12330 分钟前
# @shopify/react-native-skia 完整指南
前端·css
shanLion30 分钟前
从 iframe 到 Shadow DOM:一次关于「隔离」的前端边界思考
前端·javascript
精神状态良好33 分钟前
RAG 是什么?如何让大模型基于文档作答
前端
CRAB33 分钟前
解锁移动端H5调试:Eruda & VConsole 实战指南
前端·debug·webview