Sorting Feature Guide
Material React Table supports almost any sorting scenario you may have. Client-side sorting is enabled by default, but you can opt to implement your own server-side sorting logic, or even replace the default client-side sorting with your own implementation.
Relevant Props
# | Prop Name | Type | Default Value | More Info Links | |
---|---|---|---|---|---|
1 |
|
| |||
No Description Provided... Yet... | |||||
2 |
| ||||
No Description Provided... Yet... | |||||
3 |
|
| |||
No Description Provided... Yet... | |||||
4 |
| TanStack Table Sorting Docs | |||
This function is used to retrieve the sorted row model. If using server-side sorting, this function is not required. To use client-side sorting, pass the exported getSortedRowModel() from your adapter to your table or implement your own. | |||||
5 |
| TanStack Table Sorting Docs | |||
Pass a custom function that will be used to determine if a multi-sort event should be triggered. It is passed the event from the sort toggle handler and should return true if the event should trigger a multi-sort. | |||||
6 |
| TanStack Table Sorting Docs | |||
Enables manual sorting for the table. If this is true, you will be expected to sort your data before it is passed to the table. This is useful if you are doing server-side sorting. | |||||
7 |
| TanStack Table Sorting Docs | |||
Set a maximum number of columns that can be multi-sorted. | |||||
8 |
| TanStack Table Sorting Docs | |||
If provided, this function will be called with an updaterFn when state.sorting changes. This overrides the default internal state management, so you will need to persist the state change either fully or partially outside of the table. | |||||
9 |
| TanStack Table Sorting Docs | |||
Set to true for sorting toggles on this column to start in the descending direction. | |||||
10 |
| TanStack Table Sorting Docs | |||
This option allows you to define custom sorting functions that can be referenced in a column's sortingFn option by their key. Example: |
Relevant Column Options
Relevant State Options
# | State Option | Type | Default Value | More Info Links | |
---|---|---|---|---|---|
1 |
|
| TanStack Table Sorting Docs | ||
No Description Provided... Yet... |
Disable Sorting
Sorting can be disabled globally by setting the enableSorting
prop to false
. This will disable sorting for all columns. You can also disable sorting for individual columns by setting the enableSorting
column option to false
.
const columns = [{accessorKey: 'name',header: 'Name',enableSorting: false, // disable sorting for this column},];return (<MaterialReactTablecolumns={columns}data={data}enableSorting={false} //disable sorting for all columns/>);
Default Sorting Features
Client-side sorting is enabled by default. When sorting is toggled on for a column, the table will be sorted by an alphanumeric
sorting algorithm by default.
Multi-Sorting
Multi-sorting is also enabled by default, which means a user can sort by multiple columns at once. This can be accomplished by clicking on a column header while holding down the shift
key. The table will then be sorted by the previously sorted column, and then by the newly clicked column. You can limit the number of columns that can be sorted at once by setting the maxMultiSortColCount
prop, or you can disable multi-sorting entirely by setting the enableMultiSort
prop to false
.
Sort Direction
By default, columns with string
datatypes will sort alphabetically in ascending order, but columns with number
datatypes will sort numerically in descending order. You can change the default sort direction per column by specifying the sortDescFirst
column option to either true
or false
. You can also change the default sort direction globally by setting the sortDescFirst
prop to either true
or false
.
Sorting Functions
By default, Material React Table will use an alphanumeric
sorting function for all columns.
There are 6 built-in sorting functions that you can choose from: alphanumeric
, alphanumericCaseSensitive
, text
, textCaseSensitive
, datetime
, and basic
. You can learn more about these built-in sorting function in the TanStack Table Sorting API docs.
Add Custom Sorting Functions
If none of these sorting functions meet your needs, you can add your own custom sorting functions by specifying more sorting functions in the sortingFns
prop.
<MaterialReactTablecolumns={columns}data={data}sortingFns={{//will add a new sorting function to the list of other sorting functions already availablemyCustomSortingFn: (rowA, rowB, columnId) => // your custom sorting logic}}/>
Change Sorting Function Per Column
You can now choose a sorting function for each column by either passing a string value of the built-in sorting function names to the sortingFn
column option, or by passing a custom sorting function to the sortingFn
column option.
const columns = [{accessorKey: 'name',header: 'Name',sortingFn: 'textCaseSensitive', //use the built-in textCaseSensitive sorting function instead of the default alphanumeric sorting function},{accessorKey: 'age',header: 'Age',//use your own custom sorting function instead of any of the built-in sorting functionssortingFn: (rowA, rowB, columnId) => // your custom sorting logic},];
Manual Server-Side Sorting
If you are working with large data sets, you may want to let your back-end apis handle all of the sorting and pagination processing instead of doing it client-side. You can do this by setting the manualSorting
prop to true
. This will disable the default client-side sorting and pagination features, and will allow you to implement your own sorting and pagination logic.
When
manualSorting
is set totrue
, Material React Table assumes that yourdata
is already sorted by the time you are passing it to the table.
If you need to sort your data in a back-end api, then you will also probably need access to the internal sorting
state from the table. You can do this by managing the sorting
state yourself, and then passing it to the table via the state
prop. You can also pass a callback function to the onSortingChange
prop, which will be called whenever the sorting
state changes internally in the table
const [sorting, setSorting] = useState([]);useEffect(() => {//do something with the sorting state when it changes}, [sorting]);return (<MaterialReactTablecolumns={columns}data={data}manualSortingstate={{ sorting }}onSortingChange={setSorting}/>);
Remote Sorting Example
Here is the full Remote Data example showing how to implement server-side sorting, filtering, and pagination with Material React Table.
First Name | Last Name | Address | State | Phone Number | |
---|---|---|---|---|---|
No records to display |
1import React, { FC, useEffect, useMemo, useState } from 'react';2import MaterialReactTable, { MRT_ColumnDef } from 'material-react-table';3import type {4 ColumnFiltersState,5 PaginationState,6 SortingState,7} from '@tanstack/react-table';89type UserApiResponse = {10 data: Array<User>;11 meta: {12 totalRowCount: number;13 };14};1516type User = {17 firstName: string;18 lastName: string;19 address: string;20 state: string;21 phoneNumber: string;22};2324const Example: FC = () => {25 const [data, setData] = useState<User[]>([]);26 const [isError, setIsError] = useState(false);27 const [isLoading, setIsLoading] = useState(false);28 const [isRefetching, setIsRefetching] = useState(false);29 const [columnFilters, setColumnFilters] = useState<ColumnFiltersState>([]);30 const [globalFilter, setGlobalFilter] = useState('');31 const [sorting, setSorting] = useState<SortingState>([]);32 const [pagination, setPagination] = useState<PaginationState>({33 pageIndex: 0,34 pageSize: 10,35 });36 const [rowCount, setRowCount] = useState(0);3738 //if you want to avoid useEffect, look at the React Query example instead39 useEffect(() => {40 const fetchData = async () => {41 if (!data.length) {42 setIsLoading(true);43 } else {44 setIsRefetching(true);45 }4647 const url = new URL(48 '/api/data',49 process.env.NODE_ENV === 'production'50 ? 'https://www.material-react-table.com'51 : 'http://localhost:3000',52 );53 url.searchParams.set(54 'start',55 `${pagination.pageIndex * pagination.pageSize}`,56 );57 url.searchParams.set('size', `${pagination.pageSize}`);58 url.searchParams.set('filters', JSON.stringify(columnFilters ?? []));59 url.searchParams.set('globalFilter', globalFilter ?? '');60 url.searchParams.set('sorting', JSON.stringify(sorting ?? []));6162 try {63 const response = await fetch(url.href);64 const json = (await response.json()) as UserApiResponse;65 setData(json.data);66 setRowCount(json.meta.totalRowCount);67 } catch (error) {68 setIsError(true);69 console.error(error);70 return;71 }72 setIsError(false);73 setIsLoading(false);74 setIsRefetching(false);75 };76 fetchData();77 // eslint-disable-next-line react-hooks/exhaustive-deps78 }, [79 columnFilters,80 globalFilter,81 pagination.pageIndex,82 pagination.pageSize,83 sorting,84 ]);8586 const columns = useMemo<MRT_ColumnDef<User>[]>(87 () => [88 {89 accessorKey: 'firstName',90 header: 'First Name',91 },92 //column definitions...110 ],111 [],112 );113114 return (115 <MaterialReactTable116 columns={columns}117 data={data}118 enableRowSelection119 getRowId={(row) => row.phoneNumber}120 initialState={{ showColumnFilters: true }}121 manualFiltering122 manualPagination123 manualSorting124 muiToolbarAlertBannerProps={125 isError126 ? {127 color: 'error',128 children: 'Error loading data',129 }130 : undefined131 }132 onColumnFiltersChange={setColumnFilters}133 onGlobalFilterChange={setGlobalFilter}134 onPaginationChange={setPagination}135 onSortingChange={setSorting}136 rowCount={rowCount}137 state={{138 columnFilters,139 globalFilter,140 isLoading,141 pagination,142 showAlertBanner: isError,143 showProgressBars: isRefetching,144 sorting,145 }}146 />147 );148};149150export default Example;151
View Extra Storybook Examples