Remote Data Fetching Example
You will most likely be using a remote data source for your table, which is fully supported. Here is an example of data being fetched from a remote server but also filtered, paginated, and sorted on the server.
Also, be sure to check out the React Query Example, which is very similar to this one, except it uses react-query to simplify much of the state management needed for fetching data, so that you don't need to fetch from a useEffect
hook.
First Name | Last Name | Address | State | Phone Number | |
---|---|---|---|---|---|
No records to display |
10
0-0 of 0
1import { useEffect, useMemo, useState } from 'react';2import {3 MaterialReactTable,4 type MRT_ColumnDef,5 type MRT_ColumnFiltersState,6 type MRT_PaginationState,7 type MRT_SortingState,8} from 'material-react-table';910type UserApiResponse = {11 data: Array<User>;12 meta: {13 totalRowCount: number;14 };15};1617type User = {18 firstName: string;19 lastName: string;20 address: string;21 state: string;22 phoneNumber: string;23};2425const Example = () => {26 //data and fetching state27 const [data, setData] = useState<User[]>([]);28 const [isError, setIsError] = useState(false);29 const [isLoading, setIsLoading] = useState(false);30 const [isRefetching, setIsRefetching] = useState(false);31 const [rowCount, setRowCount] = useState(0);3233 //table state34 const [columnFilters, setColumnFilters] = useState<MRT_ColumnFiltersState>(35 [],36 );37 const [globalFilter, setGlobalFilter] = useState('');38 const [sorting, setSorting] = useState<MRT_SortingState>([]);39 const [pagination, setPagination] = useState<MRT_PaginationState>({40 pageIndex: 0,41 pageSize: 10,42 });4344 //if you want to avoid useEffect, look at the React Query example instead45 useEffect(() => {46 const fetchData = async () => {47 if (!data.length) {48 setIsLoading(true);49 } else {50 setIsRefetching(true);51 }5253 const url = new URL(54 '/api/data',55 process.env.NODE_ENV === 'production'56 ? 'https://www.material-react-table.com'57 : 'http://localhost:3000',58 );59 url.searchParams.set(60 'start',61 `${pagination.pageIndex * pagination.pageSize}`,62 );63 url.searchParams.set('size', `${pagination.pageSize}`);64 url.searchParams.set('filters', JSON.stringify(columnFilters ?? []));65 url.searchParams.set('globalFilter', globalFilter ?? '');66 url.searchParams.set('sorting', JSON.stringify(sorting ?? []));6768 try {69 const response = await fetch(url.href);70 const json = (await response.json()) as UserApiResponse;71 setData(json.data);72 setRowCount(json.meta.totalRowCount);73 } catch (error) {74 setIsError(true);75 console.error(error);76 return;77 }78 setIsError(false);79 setIsLoading(false);80 setIsRefetching(false);81 };82 fetchData();83 // eslint-disable-next-line react-hooks/exhaustive-deps84 }, [85 columnFilters,86 globalFilter,87 pagination.pageIndex,88 pagination.pageSize,89 sorting,90 ]);9192 const columns = useMemo<MRT_ColumnDef<User>[]>(93 () => [94 {95 accessorKey: 'firstName',96 header: 'First Name',97 },98 //column definitions...116 ],117 [],118 );119120 return (121 <MaterialReactTable122 columns={columns}123 data={data}124 enableRowSelection125 getRowId={(row) => row.phoneNumber}126 initialState={{ showColumnFilters: true }}127 manualFiltering128 manualPagination129 manualSorting130 muiToolbarAlertBannerProps={131 isError132 ? {133 color: 'error',134 children: 'Error loading data',135 }136 : undefined137 }138 onColumnFiltersChange={setColumnFilters}139 onGlobalFilterChange={setGlobalFilter}140 onPaginationChange={setPagination}141 onSortingChange={setSorting}142 rowCount={rowCount}143 state={{144 columnFilters,145 globalFilter,146 isLoading,147 pagination,148 showAlertBanner: isError,149 showProgressBars: isRefetching,150 sorting,151 }}152 />153 );154};155156export default Example;157
View Extra Storybook Examples