Blogs

Published At Last Updated At
Nikita_Dabari_profile
Nikita DabariSoftware Engineerauthor linkedin

Detailed Guide On React Tables - Features. Implementation, Types & Challenges

img
Quick Insight
The article emphasises the usage of react tables , providing a definitive guide to get you started with their implementation in your projects. Show More...

What is React Table?

React Table is a versatile and efficient library to build tables in React applications.

The library offers developers a quick way to display, sort, filter, and paginate large data sets in a consistent, organized table design.

One of the best features of React Table is that one can customize the table in multiple ways to suit any need or fit the design of the application.

It is possible to create fast tables with basic data or complex data grids with a variety of additional functions . Such task can be easily implemented using a React Table .

React Table uses hooks to help you build tables, giving you complete control over how you want the table to look and work (this is called Headless UI).

These dedicated hooks are lightweight, fast and flexible but they don't come with any in built markup or styles.

This philosophy aligns closely with the broader use of hooks in the React ecosystem, including React Native hooks leveraged to build interactive components for mobile apps.

Many big tech companies like Google, Apple, and Microsoft use React Table because it's flexible and easy to use.

Features like sorting, pagination, and filtering are all done with hooks, making it simple to add these functions to your tables.

Prerequisites

Before you start using React Table, make sure you have the following:

  • Basic knowledge of React and JavaScript: You should understand how React components and hooks work.
  • Node.js and npm/yarn installed: These tools are needed to install and manage libraries in your project.
  • Familiarity with ES6 syntax and JSX: You should know how to use modern JavaScript features and React's JSX syntax.

React Table Features

React Table comes with a variety of powerful features that make it easy to create flexible and customizable tables.

Customizable Headers, Rows, and Cells

You can customize the headers, rows, and cells to fit your needs. This allows you to control what data is displayed and how it looks.

Opt for the most suitable react based CMS to implement customised components and elements catering to your content management system.

1import { useTable } from 'react-table';
2
3const columns = [
4 {
5 Header: 'Name',
6 accessor: 'name', // accessor is the "key" in the data
7 },
8 {
9 Header: 'Age',
10 accessor: 'age',
11 },
12];
13
14const data = [
15 { name: 'John Doe', age: 28 },
16 { name: 'Jane Smith', age: 34 },
17];
18
19function TableComponent() {
20 const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow } = useTable({ columns, data });
21
22 return (
23 <table {...getTableProps()}>
24 <thead>
25 {headerGroups.map(headerGroup => (
26 <tr {...headerGroup.getHeaderGroupProps()}>
27 {headerGroup.headers.map(column => (
28 <th {...column.getHeaderProps()}>{column.render('Header')}</th>
29 ))}
30 </tr>
31 ))}
32 </thead>
33 <tbody {...getTableBodyProps()}>
34 {rows.map(row => {
35 prepareRow(row);
36 return (
37 <tr {...row.getRowProps()}>
38 {row.cells.map(cell => (
39 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
40 ))}
41 </tr>
42 );
43 })}
44 </tbody>
45 </table>
46 );
47}
48
49
50

output:

output-react tables -1

Multi-Column Sorting

You can sort the table by multiple columns, making it easier to organize and find data.

1import React from 'react';
2import { useTable, useSortBy } from 'react-table';
3
4const TableComponent = ({ columns, data }) => {
5 const {
6 getTableProps,
7 getTableBodyProps,
8 headerGroups,
9 rows,
10 prepareRow
11 } = useTable(
12 {
13 columns,
14 data
15 },
16 useSortBy
17 );
18
19 return (
20 <div className="overflow-x-auto">
21 <table {...getTableProps()} className="min-w-full divide-y divide-gray-200">
22 <thead className="bg-gray-50">
23 {headerGroups.map(headerGroup => (
24 <tr {...headerGroup.getHeaderGroupProps()}>
25 {headerGroup.headers.map(column => (
26 <th
27 {...column.getHeaderProps(column.getSortByToggleProps())}
28 className="px-6 py-3 text-left text-xs font-medium text-gray-500 uppercase tracking-wider"
29 >
30 <div className="flex items-center">
31 {column.render('Header')}
32 <span className="ml-2">
33 {column.isSorted ? (
34 column.isSortedDesc ? (
35 <svg className="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
36 <path fillRule="evenodd" d="M10 12a.5.5 0 01-.4-.2l-5-5a.5.5 0 11.8-.6L10 10.8l4.6-4.6a.5.5 0 11.8.6l-5 5a.5.5 0 01-.4.2z" clipRule="evenodd" />
37 </svg>
38 ) : (
39 <svg className="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
40 <path fillRule="evenodd" d="M10 8a.5.5 0 01.4.2l5 5a.5.5 0 11-.8.6L10 9.2 5.4 14a.5.5 0 11-.8-.6l5-5a.5.5 0 01.4-.2z" clipRule="evenodd" />
41 </svg>
42 )
43 ) : (
44 <svg className="w-4 h-4 text-gray-400" fill="currentColor" viewBox="0 0 20 20">
45 <path fillRule="evenodd" d="M10 8a.5.5 0 01.4.2l5 5a.5.5 0 11-.8.6L10 9.2 5.4 14a.5.5 0 11-.8-.6l5-5a.5.5 0 01.4-.2zM10 12a.5.5 0 01-.4-.2l-5-5a.5.5 0 11.8-.6L10 10.8l4.6-4.6a.5.5 0 11.8.6l-5 5a.5.5 0 01-.4.2z" clipRule="evenodd" />
46 </svg>
47 )}
48 </span>
49 </div>
50 </th>
51 ))}
52 </tr>
53 ))}
54 </thead>
55 <tbody {...getTableBodyProps()} className="bg-white divide-y divide-gray-200">
56 {rows.map(row => {
57 prepareRow(row);
58 return (
59 <tr {...row.getRowProps()} className="hover:bg-gray-100">
60 {row.cells.map(cell => (
61 <td {...cell.getCellProps()} className="px-6 py-4 whitespace-nowrap text-sm text-gray-900">
62 {cell.render('Cell')}
63 </td>
64 ))}
65 </tr>
66 );
67 })}
68 </tbody>
69 </table>
70 </div>
71 );
72};
73
74export default TableComponent;
75
76

Output:

react tables - pagination

Built-in and Custom Filtering

You can filter the data in the table using built-in or custom filters.

1import { useTable, useFilters } from 'react-table';
2
3// Custom filter UI for a column
4function DefaultColumnFilter({
5 column: { filterValue, preFilteredRows, setFilter },
6}) {
7 const count = preFilteredRows.length;
8
9 return (
10 <input
11 value={filterValue || ''}
12 onChange={e => {
13 setFilter(e.target.value || undefined); // Set undefined to remove the filter entirely
14 }}
15 placeholder={`Search ${count} records...`}
16 />
17 );
18}
19
20function TableComponent() {
21 const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow } = useTable(
22 { columns, data, defaultColumn: { Filter: DefaultColumnFilter } },
23 useFilters
24 );
25
26 return (
27 <table {...getTableProps()}>
28 <thead>
29 {headerGroups.map(headerGroup => (
30 <tr {...headerGroup.getHeaderGroupProps()}>
31 {headerGroup.headers.map(column => (
32 <th {...column.getHeaderProps()}>
33 {column.render('Header')}
34 <div>{column.canFilter ? column.render('Filter') : null}</div>
35 </th>
36 ))}
37 </tr>
38 ))}
39 </thead>
40 <tbody {...getTableBodyProps()}>
41 {rows.map(row => {
42 prepareRow(row);
43 return (
44 <tr {...row.getRowProps()}>
45 {row.cells.map(cell => (
46 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
47 ))}
48 </tr>
49 );
50 })}
51 </tbody>
52 </table>
53 );
54}
55
56
57

Pagination for Large Datasets

Pagination helps manage large datasets by splitting them into smaller, more manageable pages.

1import { useTable, usePagination } from 'react-table';
2
3function TableComponent() {
4 const { getTableProps, getTableBodyProps, headerGroups, page, prepareRow, canPreviousPage, canNextPage, pageOptions, nextPage, previousPage, state: { pageIndex } } = useTable(
5 { columns, data, initialState: { pageIndex: 0 } },
6 usePagination
7 );
8
9 return (
10 <>
11 <table {...getTableProps()}>
12 <thead>
13 {headerGroups.map(headerGroup => (
14 <tr {...headerGroup.getHeaderGroupProps()}>
15 {headerGroup.headers.map(column => (
16 <th {...column.getHeaderProps()}>{column.render('Header')}</th>
17 ))}
18 </tr>
19 ))}
20 </thead>
21 <tbody {...getTableBodyProps()}>
22 {page.map(row => {
23 prepareRow(row);
24 return (
25 <tr {...row.getRowProps()}>
26 {row.cells.map(cell => (
27 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
28 ))}
29 </tr>
30 );
31 })}
32 </tbody>
33 </table>
34 <div>
35 <button onClick={() => previousPage()} disabled={!canPreviousPage}>
36 Previous
37 </button>
38 <span>
39 Page{' '}
40 <strong>
41 {pageIndex + 1} of {pageOptions.length}
42 </strong>
43 </span>
44 <button onClick={() => nextPage()} disabled={!canNextPage}>
45 Next
46 </button>
47 </div>
48 </>
49 );
50}
51
52

Inline Editing of Table Data

Inline editing allows users to edit table data directly within the table.

1import { useState } from 'react';
2import { useTable } from 'react-table';
3
4function EditableCell({ value: initialValue, row: { index }, column: { id }, updateMyData }) {
5 const [value, setValue] = useState(initialValue);
6
7 const onChange = e => {
8 setValue(e.target.value);
9 };
10
11 const onBlur = () => {
12 updateMyData(index, id, value);
13 };
14
15 return <input value={value} onChange={onChange} onBlur={onBlur} />;
16}
17
18const defaultColumn = {
19 Cell: EditableCell,
20};
21
22function TableComponent({ columns, data, updateMyData }) {
23 const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow } = useTable(
24 { columns, data, defaultColumn, updateMyData }
25 );
26
27 return (
28 <table {...getTableProps()}>
29 <thead>
30 {headerGroups.map(headerGroup => (
31 <tr {...headerGroup.getHeaderGroupProps()}>
32 {headerGroup.headers.map(column => (
33 <th {...column.getHeaderProps()}>{column.render('Header')}</th>
34 ))}
35 </tr>
36 ))}
37 </thead>
38 <tbody {...getTableBodyProps()}>
39 {rows.map(row => {
40 prepareRow(row);
41 return (
42 <tr {...row.getRowProps()}>
43 {row.cells.map(cell => (
44 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
45 ))}
46 </tr>
47 );
48 })}
49 </tbody>
50 </table>
51 );
52}
53
54

Row Selection with Checkboxes

You can add checkboxes for row selection, making it easier to manage selected rows.

1import { useTable, useRowSelect } from 'react-table';
2
3// Add a checkbox to each row
4const IndeterminateCheckbox = React.forwardRef(({ indeterminate, ...rest }, ref) => {
5 const defaultRef = React.useRef();
6 const resolvedRef = ref || defaultRef;
7
8 React.useEffect(() => {
9 resolvedRef.current.indeterminate = indeterminate;
10 }, [resolvedRef, indeterminate]);
11
12 return <input type="checkbox" ref={resolvedRef} {...rest} />;
13});
14
15function TableComponent() {
16 const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow, selectedFlatRows } = useTable(
17 { columns, data },
18 useRowSelect,
19 hooks => {
20 hooks.visibleColumns.push(columns => [
21 {
22 id: 'selection',
23 Header: ({ getToggleAllRowsSelectedProps }) => (
24 <div>
25 <IndeterminateCheckbox {...getToggleAllRowsSelectedProps()} />
26 </div>
27 ),
28 Cell: ({ row }) => (
29 <div>
30 <IndeterminateCheckbox {...row.getToggleRowSelectedProps()} />
31 </div>
32 ),
33 },
34 ...columns,
35 ]);
36 }
37 );
38
39 return (
40 <table {...getTableProps()}>
41 <thead>
42 {headerGroups.map(headerGroup => (
43 <tr {...headerGroup.getHeaderGroupProps()}>
44 {headerGroup.headers.map(column => (
45 <th {...column.getHeaderProps()}>{column.render('Header')}</th>
46 ))}
47 </tr>
48 ))}
49 </thead>
50 <tbody {...getTableBodyProps()}>
51 {rows.map(row => {
52 prepareRow(row);
53 return (
54 <tr {...row.getRowProps()}>
55 {row.cells.map(cell => (
56 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
57 ))}
58 </tr>
59 );
60 })}
61 </tbody>
62 </table>
63 );
64}
65
66

Column Resizing and Reordering

You can resize and reorder columns to suit your needs.

1import { useTable, useResizeColumns, useFlexLayout } from 'react-table';
2
3function TableComponent() {
4 const { getTableProps, getTableBodyProps, headerGroups, rows, prepareRow } = useTable(
5 { columns, data },
6 useResizeColumns,
7 useFlexLayout
8 );
9
10 return (
11 <table {...getTableProps()}>
12 <thead>
13 {headerGroups.map(headerGroup => (
14 <tr {...headerGroup.getHeaderGroupProps()}>
15 {headerGroup.headers.map(column => (
16 <th {...column.getHeaderProps()} style={{ minWidth: column.minWidth, width: column.width }}>
17 {column.render('Header')}
18 <div {...column.getResizerProps()} className="resizer" />
19 </th>
20 ))}
21 </tr>
22 ))}
23 </thead>
24 <tbody {...getTableBodyProps()}>
25 {rows.map(row => {
26 prepareRow(row);
27 return (
28 <tr {...row.getRowProps()}>
29 {row.cells.map(cell => (
30 <td {...cell.getCellProps()}>{cell.render('Cell')}</td>
31 ))}
32 </tr>
33 );
34 })}
35 </tbody>
36 </table>
37 );
38}

Optimized Performance for Large Datasets

React Table is optimized for handling large datasets efficiently, making it suitable for applications with extensive data.

Its virtualized rendering, pagination, and sorting features ensure smooth performance, even with complex data sets.

This is particularly crucial for applications that rely on real-time data processing, analytics dashboards, or backend systems that interact heavily with RESTful web services.

There are resources (platforms)to asess RESTful web services and their working capabilities

Easy Styling with CSS-in-JS or Other Methods

You can easily style React Table using CSS-in-JS libraries like styled-components or any other CSS method.


2import styled from 'styled-components';
3
4const Styles = styled.div` table { width: 100%; border-spacing: 0; border: 1px solid black; th, td { margin: 0; padding: 0.5rem; border-bottom: 1px solid black; border-right: 1px solid black; :last-child { border-right: 0; } } th { background: lightgray; } } `;
5
6function TableComponent() {
7 return (
8 <Styles>
9 <table>
10 {/* Table content */}
11 </table>
12 </Styles>
13 );
14}
15
16

Conclusion

In this article, we learned how to build a table interface using React. Creating a basic table is not too hard, but try to avoid redoing things that have already been done whenever possible. I hope you found learning about table interfaces enjoyable.