Data Columns
Data Columns are used to display data and are the default columns that are created when you create a column with an accessorKey
or accessorFn
.
The table can perform processing on the data of a Data Column, such as sorting, filtering, grouping, etc.
The other type of column that you can make is a "Display Column", which you can learn more about in the next section.
Accessors (Connect a column to data)
Each column definition must have at least an accessorKey
(or a combination of an id
and accessorFn
) and a header
property. The accessorKey
/accessorFn
property is the key that will be used to join the data from the data
keys. The header
property is used to display the column header, but is also used in other places in the table.
Method 1 - Using an accessorKey (Recommended)
The simplest and most common way to define a column is to use the accessorKey
property. The accessorKey
property is the key that will be used to join the data from the data
keys.
The accessorKey
must match one of the keys in your data, or else no data will show up in the column. The accessorKey
also supports dot notation, so you can access nested data.
By default, the accessorKey
will double as the id
for the column, but if you need the id of the column to be different than the accessorKey, you can use the id
property in addition.
const columns = useMemo<MRT_ColumnDef<Customer>[]>( //TS helps with the autocomplete while writing columns() => [{accessorKey: 'username', //normal recommended usage of an accessorKeyheader: 'Username',},{accessorKey: 'name.firstName', //example of dot notation used to access nested dataheader: 'First Name',},{accessorKey: 'name.lastName', //example of dot notation used to access nested dataheader: 'Last Name',},{accessorKey: 'customerAge',id: 'age' //id overridden, usually not necessary to do this, but can be helpfulheader: 'Age',},],[],);
Method 2 - Using an accessorFn and id
You can alternatively use the accessorFn
property. Here are at least three ways you can use it.
In each case, the id
property is now required since there is no accessorKey
for MRT to derive it from.
const columns = useMemo(() => [{//simple accessorFn that works the same way as an `accessorKey`accessorFn: (row) => row.username,id: 'username',header: 'Username',},{//accessorFn function that combines multiple data togetheraccessorFn: (row) => `${row.firstName} ${row.lastName}`,id: 'name',header: 'Name',},{//accessorFn used to access nested data, though you could just use dot notation in an accessorKeyaccessorFn: (row) => row.personalInfo.age,id: 'age',header: 'Age',},],[],);
Custom Header Render
If you want to pass in custom JSX to render the header, you can pass in a Header
option in addition to the header
string property.
The
header
(lowercase) property is still required and still must only be a string because it is used within multiple components in the table and has string manipulation methods performed on it.
const columns = useMemo(() => [{accessorKey: 'name',header: 'Name',Header: ({ column }) => (<i style={{ color: 'red' }}>{column.columnDef.header}</i>), //arrow function},{accessorKey: 'age',header: 'Age',Header: <i style={{ color: 'red' }}>Age</i>, //plain jsx with no function},],[],);
Custom Cell Render
Similarly, the data cells in a column can have a custom JSX render with the Cell
option.
const columns = useMemo(() => [{accessorKey: 'salary',header: 'Salary',Cell: ({ cell }) => (<span>${cell.getValue<number>().toLocaleString()}</span>),},{accessorKey: 'profileImage',header: 'Profile Image',Cell: ({ cell }) => <img src={cell.getValue<string>()} />,},],[],);
Custom Footer Render
If you want to pass in custom JSX to render the footer, you can pass in a Footer
option. If no custom markup is needed, you can just use the footer
string property.
The footer cells can be a good place to put totals or other summary information.
const columns = useMemo(() => [{accessorKey: 'name',header: 'Name',footer: 'Name', //simple string header},{accessorKey: 'age',header: 'Age',//Custom footer markup for a aggregation calculationFooter: () => (<Stack>Max Age:<Box color="warning.main">{Math.round(maxAge)}</Box></Stack>),},],[],);
See the Customize Components Guide for more ways to style and customize header and cell components.
Enable or Disable Features Per Column
In the same way that you can pass props to the main <MaterialReactTable />
component to enable or disable features, you can also specify options on the column definitions to enable or disable features on a per-column basis.
const columns = useMemo(() => [{accessorKey: 'salary',header: 'Salary',enableClickToCopy: true, //enable click to copy on this column},{accessorKey: 'profileImage',header: 'Profile Image',enableSorting: false, //disable sorting on this column},],[],);
See all the column options you can use in the Column Options API Reference.
Set Column Widths
This topic is covered in detail in the Column Resizing Guide, but here is a brief overview.
Setting a CSS (sx or style) width prop will NOT work. Material React Table (or, more accurately, TanStack Table) will keep track and set the widths of each column internally.
You CAN, however, change the default width of any column by setting its size
option on the column definition. minSize
and maxSize
are also available to set the minimum and maximum width of the column during resizing.
const columns = [{accessorKey: 'id',header: 'ID',size: 50, //small column},{accessorKey: 'username',header: 'Username',minSize: 100, //min size enforced during resizingmaxSize: 200, //max size enforced during resizingsize: 180, //medium column},{accessorKey: 'email',header: 'Email',size: 300, //large column},];
However, the column sizes might not change as much as you would expect. This is because the browser treats <th>
and <td>
elements differently with in a <table>
element by default.
You can improve this slightly by changing the table layout to fixed
instead of the default auto
in the muiTableProps
.
<MaterialReactTablemuiTableProps={{sx: {tableLayout: 'fixed',},}}/>
The columns will still try to increase their width to take up the full width of the table, but the columns that do have a defined size will have their width respected more.
To learn more about how table-layout fixed
vs auto
works, we recommend reading this blog post by CSS-Tricks.
Set Column Alignment
By default, all columns are left-aligned. You can change the alignment of a column by setting the align
option to either "center"
, "right"
, or "justify"
in the muiTableHeadCellProps
and muiTableBodyCellProps
props/column options.
const columns = [{accessorKey: 'id',header: 'ID',//right align the header and body cellsmuiTableHeadCellProps: {align: 'right',},muiTableBodyCellProps: {align: 'right',},},{accessorKey: 'username',header: 'Username',//center align the header and body cellsmuiTableHeadCellProps: {align: 'center',},muiTableBodyCellProps: {align: 'center',},},];
First Name | Last Name | Age | Salary |
---|---|---|---|
Homer | Simpson | 39 | $53,000.00 |
Marge | Simpson | 38 | $60,000.00 |
Bart | Simpson | 10 | $46,000.00 |
Lisa | Simpson | 8 | $120,883.00 |
Maggie | Simpson | 1 | $22.00 |
1import React, { FC, useMemo } from 'react';2import MaterialReactTable, { MRT_ColumnDef } from 'material-react-table';3import { data, Person } from './makeData';45const Example: FC = () => {6 const columns = useMemo<MRT_ColumnDef<Person>[]>(7 () => [8 {9 accessorKey: 'firstName',10 header: 'First Name',11 size: 100,12 muiTableHeadCellProps: {13 align: 'center',14 },15 muiTableBodyCellProps: {16 align: 'center',17 },18 },19 {20 accessorKey: 'lastName',21 header: 'Last Name',22 size: 100,23 muiTableHeadCellProps: {24 align: 'center',25 },26 muiTableBodyCellProps: {27 align: 'center',28 },29 },30 {31 accessorKey: 'age',32 header: 'Age',33 muiTableHeadCellProps: {34 align: 'right',35 },36 muiTableBodyCellProps: {37 align: 'right',38 },39 },40 {41 accessorKey: 'salary',42 header: 'Salary',43 muiTableHeadCellProps: {44 align: 'right',45 },46 muiTableBodyCellProps: {47 align: 'right',48 },49 Cell: ({ cell }) =>50 cell51 .getValue<number>()52 .toLocaleString('en-US', { style: 'currency', currency: 'USD' }),53 },54 ],55 [],56 );5758 return <MaterialReactTable columns={columns} data={data} />;59};6061export default Example;62