import React, { useState } from "react";
import styled from "styled-components";
import { useTable, usePagination, useRowSelect } from "react-table";
import { useQuery } from "react-query";
import { fetchPokemonData } from "./fetchData";
const TableContainer = styled.div`
padding: 1rem;
table {
border-spacing: 0;
border: 1px solid black;
tr {
:last-child {
td {
border-bottom: 0;
}
}
}
th,
td {
margin: 0;
padding: 0.5rem;
border-bottom: 1px solid black;
border-right: 1px solid black;
:last-child {
border-right: 0;
}
}
}
.pagination {
padding: 0.5rem;
}
`;
const columns = [
{
Header: "Name",
accessor: "name"
},
{
Header: "Url",
accessor: "url"
}
];
const trimData = (data = []) =>
data.map(({ name, url }) => ({
name,
url
}));
const initialState = {
queryPageIndex: 0,
queryPageSize: 100,
totalCount: null
};
const PAGE_CHANGED = "PAGE_CHANGED";
const PAGE_SIZE_CHANGED = "PAGE_SIZE_CHANGED";
const TOTAL_COUNT_CHANGED = "TOTAL_COUNT_CHANGED";
const reducer = (state, { type, payload }) => {
switch (type) {
case PAGE_CHANGED:
return {
...state,
queryPageIndex: payload
};
case PAGE_SIZE_CHANGED:
return {
...state,
queryPageSize: payload
};
case TOTAL_COUNT_CHANGED:
return {
...state,
totalCount: payload
};
default:
throw new Error(`Unhandled action type: ${type}`);
}
};
const IndeterminateCheckbox = React.forwardRef(
({ indeterminate, ...rest }, ref) => {
const defaultRef = React.useRef();
const resolvedRef = ref || defaultRef;
React.useEffect(() => {
resolvedRef.current.indeterminate = indeterminate;
}, [resolvedRef, indeterminate]);
return (
<>
<input type="checkbox" ref={resolvedRef} {...rest} />
</>
);
}
);
function PokemonTable() {
const [
{ queryPageIndex, queryPageSize, totalCount },
dispatch
] = React.useReducer(reducer, initialState);
const { isLoading, error, data, isSuccess } = useQuery(
[queryPageIndex, queryPageSize],
() => fetchPokemonData(queryPageIndex, queryPageSize),
{
keepPreviousData: false,
staleTime: Infinity
}
);
const [search, setSearch] = useState("");
const {
getTableProps,
getTableBodyProps,
headerGroups,
prepareRow,
selectedFlatRows,
toggleAllRowsSelected,
page,
canPreviousPage,
canNextPage,
pageOptions,
pageCount,
gotoPage,
nextPage,
previousPage,
setPageSize,
// Get the state from the instance
state: { pageIndex, pageSize, selectedRowIds }
} = useTable(
{
columns,
data: isSuccess ? trimData(data.results) : [],
initialState: {
pageIndex: queryPageIndex,
pageSize: queryPageSize
},
manualPagination: true, // Tell the usePagination
// hook that we'll handle our own data fetching
// This means we'll also have to provide our own
// pageCount.
pageCount: isSuccess ? Math.ceil(totalCount / queryPageSize) : null
},
usePagination
// useRowSelect,
// (hooks) => {
// hooks.visibleColumns.push((columns) => [
// {
// id: "selection",
// Header: ({ getToggleAllPageRowsSelectedProps }) => (
// <div>
// <IndeterminateCheckbox {...getToggleAllPageRowsSelectedProps()} />
// </div>
// ),
// Cell: ({ row }) => (
// <div>
// <IndeterminateCheckbox {...row.getToggleRowSelectedProps()} />
// </div>
// )
// },
// ...columns
// ]);
// }
);
React.useEffect(() => {
dispatch({ type: PAGE_CHANGED, payload: pageIndex });
}, [pageIndex]);
React.useEffect(() => {
dispatch({ type: PAGE_SIZE_CHANGED, payload: pageSize });
gotoPage(0);
}, [pageSize, gotoPage]);
React.useEffect(() => {
if (data?.count) {
dispatch({
type: TOTAL_COUNT_CHANGED,
payload: data.count
});
}
}, [data?.count]);
if (error) {
return <p>Error</p>;
}
if (isLoading) {
return <p>Loading...</p>;
}
// This method is used for searching.
const searchFilter = (search) => {};
return (
<TableContainer>
<input
aria-describedby="basic-addon1"
type="search"
id="search-input"
placeholder="search"
value={search}
onChange={(e) => searchFilter(e.target.value)}
/>
{isSuccess ? (
<>
<table {...getTableProps()}>
<thead>
{headerGroups.map((headerGroup) => (
<tr {...headerGroup.getHeaderGroupProps()}>
{headerGroup.headers.map((column) => (
<th {...column.getHeaderProps()}>
{column.render("Header")}
</th>
))}
</tr>
))}
</thead>
<tbody {...getTableBodyProps()}>
{page.map((row) => {
prepareRow(row);
return (
<tr {...row.getRowProps()}>
{row.cells.map((cell) => (
<td {...cell.getCellProps()}>{cell.render("Cell")}</td> // fix return statement
))}
</tr>
);
})}
</tbody>
</table>
<div className="pagination">
<button onClick={() => gotoPage(0)} disabled={!canPreviousPage}>
{"<<"}
</button>{" "}
<button onClick={() => previousPage()} disabled={!canPreviousPage}>
{"<"}
</button>{" "}
<button onClick={() => nextPage()} disabled={!canNextPage}>
{">"}
</button>{" "}
<button
onClick={() => gotoPage(pageCount - 1)}
disabled={!canNextPage}
>
{">>"}
</button>{" "}
<span>
Page{" "}
<strong>
{pageIndex + 1} of {pageOptions.length}
</strong>{" "}
</span>
<span>
| Go to page:{" "}
<input
type="number"
value={pageIndex + 1}
onChange={(e) => {
const page = e.target.value ? Number(e.target.value) - 1 : 0;
gotoPage(page);
}}
style={{ width: "100px" }}
/>
</span>{" "}
<select
value={pageSize}
onChange={(e) => {
setPageSize(Number(e.target.value));
}}
>
{[100, 20, 30, 40, 50].map((pageSize) => (
<option key={pageSize} value={pageSize}>
Show {pageSize}
</option>
))}
</select>
</div>
</>
) : null}
</TableContainer>
);
}
export default PokemonTable;
. I have implemented server-side pagination. where I wanted to use the checkbox. I have tried to use useRowSelect but it's giving maximum depth reached an error.
please find the code link below.
https://codesandbox.io/s/determined-voice-b0n991?file=/src/PokemonTable.js
please uncomment the following code in the link so that the error is reproduced.
// useRowSelect,
// (hooks) => {
// hooks.visibleColumns.push((columns) => [
// {
// id: "selection",
// Header: ({ getToggleAllPageRowsSelectedProps }) => (
// <div>
// <IndeterminateCheckbox {...getToggleAllPageRowsSelectedProps()} />
// </div>
// ),
// Cell: ({ row }) => (
// <div>
// <IndeterminateCheckbox {...row.getToggleRowSelectedProps()} />
// </div>
// )
// },
// ...columns
// ]);
// }
Here, you have to use a memoized data. Let's say we define it as pokemonData
then use it as follow:
Here is the complete code (forked from yours):