DEV Community

Cover image for Let's create Data Table. Part 3: Virtualization
Dima Vyshniakov
Dima Vyshniakov

Posted on • Edited on

Let's create Data Table. Part 3: Virtualization

This is an article from the series about creating of an advanced Data table component using React, TanStack Table 8, Tailwind CSS and Headless UI.

In the previous article, we implemented a Data table capable to display various data formats using TanStack table.

We tested our table with a dataset of 100 rows. But what if we increase this number to 1,000 or even 10,000? Eventually, we will hit the browser's rendering limitations. The ability to display and operate on large datasets is crucial for the Data Table. That's why we will implement render virtualization.

What is virtualization?

Virtualization is a technique used to improve the performance of rendering large datasets by only rendering the visible rows and a small buffer around them. This reduces the number of DOM elements and improves the overall performance and responsiveness of the table.

Demo of virtualized scrolling

Here is the demo of Data table with more 30 thousands rows scrolling. We are going to use @tanstack/react-virtual library to implement this.

Implement virtual table

In order to keep code organized, we will create folder src/DataTable/features. Here will be the place for the logic responsible for Data table features. We will implement this as React hooks.

Virtualizer React hook

Here is a custom hook using TanStack virtualizer. We also create a workaround for sticky header compatibility.

import { notUndefined, useVirtualizer } from '@tanstack/react-virtual';
import type { MutableRefObject } from 'react';

// Cell height needs to be consistent for each row
const CELL_HEIGHT = 31;

// Number of rows to render before and after viewport.
const OVERSCAN = 6;

export type Props = {
  /** Total number of rows */
  rowsCount: number;
  /** Reference to the table container element, which has scroll */
  scrollRef: MutableRefObject<HTMLElement | null>;
};

export const useVirtualRows = ({ rowsCount, scrollRef }: Props) => {
  const virtualizer = useVirtualizer({
    count: rowsCount,
    getScrollElement: () => scrollRef.current,
    estimateSize: () => CELL_HEIGHT,
    overscan: OVERSCAN,
  });

  // This will replace "real" rows for rendering
  const virtualRows = virtualizer.getVirtualItems();

  /**
   * This is required to fix the issue with sticky table header rendering.
   * @see DataTable
   */
  const [before, after] =
    virtualRows.length > 0
      ? [
          notUndefined(virtualRows[0]).start - virtualizer.options.scrollMargin,
          virtualizer.getTotalSize() -
            notUndefined(virtualRows[virtualRows.length - 1]).end,
        ]
      : [0, 0];

  return { virtualRows, before, after };
};


Enter fullscreen mode Exit fullscreen mode

DataTable component adjustments

Now we change DataTable component to use virtual rows.

const DataTable: FC<Props> = ({ tableData }) => {
  //...

  /* Virtualizer logic start */
  const scrollRef = useRef<HTMLDivElement>(null);
  const { rows } = table.getRowModel();
  const { before, after, virtualRows } = useVirtualRows({
    scrollRef,
    rowsCount: rows.length,
  });
  /* Virtualizer logic end */

  return (
    //...
    <tbody>
      <Fragment>
        {/* Fix the issue with a sticky table header and infinite scroll */}
        {before > 0 && (
          <tr>
            <td colSpan={columns.length} style={{height: before}} />
          </tr>
        )}
        {virtualRows.map((virtualRow) => {
          // this is the "real" current row
          const row = rows[virtualRow.index];
          return (
            <tr key={row.id}>
              {row.getVisibleCells().map((cell) => /*...*/)}
            </tr>
          );
        })}
        {/* Fix the issue with a sticky table header and infinite scroll */}
        {after > 0 && (
          <tr>
            <td colSpan={columns.length} style={{height: after}} />
          </tr>
        )}
      </Fragment>
    </tbody>
  );
};

Enter fullscreen mode Exit fullscreen mode

Working demo

So, now we can set rows' amount to 33333 and see what happens.

Next: Column pinning

Top comments (0)