TypeScript Usage
TanStack Table itself is written in TypeScript, and Material React Table builds on top of its great type definitions for best in class TypeScript experience.
If, however, you cannot use TypeScript in your project for some reason, checkout down below for how to use JSDoc instead of TypeScript to get the same type hints.
Is TypeScript Required?
No, TypeScript is not required to use Material React Table. You can just use JavaScript and everything will work just fine, but you will be missing out on a lot of great type hints and type safety that can help you build your app faster and with less bugs.
Define You TData Type
Material React Table makes use of generics to make working with your specific row data structures easier. You will see that most of the MRT_*
types that you can use accept a TData
generic.
Let's say that the data in your table is an array of users that looks like this:
const data: User[] = [{ id: 1, name: 'John', age: 23 },{ id: 2, name: 'Alice', age: 17 },{ id: 3, name: 'Bob', age: 32 },];
Then your TData
type can be defined as:
export type User = {id: number;name: string;age: number;};
Define Your Column Definitions with MRT_ColumnDef
Material React Table provides you with a MRT_ColumnDef
type that you can use to define your column definitions. It is a generic type that accepts your TData
type as a generic.
import {MaterialReactTable,useMaterialReactTable,type MRT_ColumnDef, // <--- import MRT_ColumnDef} from '@mui/core';import { type User } from './types'; // <--- import your TData type from wherever you defined it// define your columns, pass User as a generic to MRT_ColumnDefconst columns: MRT_ColumnDef<User>[] = [{accessorKey: 'id', //you should get type hints for all of your keys if you defined your TData type correctlyheader: 'ID',enableSorting: false, //you should get type hints for all possible column options that you can define here},{accessorKey: 'name',header: 'Name',},{accessorFn: (originalRow) => Number(originalRow.age), //you should also get type hints for your accessorFnheader: 'Age',},];
Use JSDoc instead of TypeScript
If you are in a situation where you are not able to install TypeScript in your project, you can technically do the same thing as up above in JavaScript using JSDoc.
import {MaterialReactTable,useMaterialReactTable,} from 'material-react-table';//define TData type with JSDoc/*** @typedef {Object} User* @property {number} id* @property {string} name* @property {number} age*///import MRT_ColumnDef type with JSDoc/*** @type {import('material-react-table').MRT_ColumnDef<User>[]}*/const columns = [{accessorKey: 'id', //you should get type hints for all of your keys if you defined your TData type correctlyheader: 'ID',enableSorting: false, //you should get type hints for all possible column options that you can define here},{accessorKey: 'name',header: 'Name',},{accessorFn: (originalRow) => Number(originalRow.age), //you should also get type hints for your accessorFnheader: 'Age',},];
Re-Usable MRT Components
If you are going to have multiple tables in your app, chances are that you will want to make a re-usable component built on top of Material React Table. This is a good idea and good practice, but here are a few suggestions to maintain type safety with some TypeScript generics.
Re-usable Components or Options?
In my opinion, instead of creating a re-usable component, it is instead actually best to define your default options and share them between all of your tables.
Re-usable Default Options
import { type MRT_TableOptions } from 'material-react-table';//define re-useable default table options for all tables in your appexport const getDefaultMRTOptions = <TData extends Record<string, any> = {},>(): Partial<MRT_TableOptions<TData>> => ({enableGlobalFilter: false,enableRowPinning: true,initialState: { showColumnFilters: true },manualFiltering: true,manualPagination: true,manualSorting: true,muiTableHeadCellProps: {sx: { fontSize: '1.1rem' },},paginationDisplayMode: 'pages',//etc...});
Then you can use these options in every new table that you create:
import {MaterialReactTable,useMaterialReactTable,type MRT_ColumnDef,} from 'material-react-table';import { getDefaultMRTOptions } from './utils'; //your default optionsinterface User {id: number;name: string;age: number;}const defaultMRTOptions = getDefaultMRTOptions<User>(); //get your default optionsexport const OneOfYourTableComponents = () => {const columns: MRT_ColumnDef<User>[] = [//...];const { data } = useQuery({//...});const table = useMaterialReactTable({...defaultMRTOptions,columns,data,enableGlobalFilter: true, //override default optionsinitialState: {...defaultMRTOptions.initialState,showColumnFilters: false,},//...});};
Doing it this way, you maintain 100% control of your table instance and any state that you are managing in each table component.
Re-usable MRT Component
If you still want to just create a re-usable MRT component instead, you can do that too, of course. Here is a type-safe way to do that:
interface Props<TData extends Record<string, any> = {}>extends MRT_TableOptions<TData> {columns: MRT_ColumnDef<TData>[];data: TData[];}export const CustomMRTTable = <TData extends Record<string, any> = {}>({columns,data,...rest}: Props<TData>) => {const table = useMaterialReactTable({columns,data,//your custom table options......rest, //accept props to override default table options});return <MaterialReactTable table={table} />;};
By using the TData
generic correctly, you can maintain type-safety in your re-usable component that will adapt to different types of data you will have throughout your application.