Skip to main content

Table


A component that renders a Table, with sorting capabilities

Columns can be built using the tableColumn helper, which provides a type-safe API as well as default implementations for commonly used columns.

Also, when building the data to pass to the table, it's recommended to define columns as a const, as opposed to inline, in order to get better type inference, e.g.:

// Best
const columns = [tableColumn(...), tableColumn(...)] as const

<Table columns={columns} data={data} />

// Ok, but worse type inference
<Table columns={[tableColumn(...), tableColumn(...)]} data={data} />

Example

Button
Name
Country
Applications
Value
Website
Alerts
Status
Actions
Amazon
US
-
100
Active
Google
US
10,000
150
-
Paused
<Table
data={[
{
name: "Amazon",
country: {
icon: IconInfo,
text: "US",
},
button: {
label: "Row 1",
kind: "transparent",
hierarchy: "primary",
onPress: () => {},
} as const,
status: {
label: "Active",
color: "green",
} as const,
value: {
numericValue: 100,
icon: IconInfo,
},
website: {
href: "http://www.amazon.com",
label: "Link",
},
alerts: {
icon: IconWarningSolid,
label: "Warning",
},
group: "Group 1",
deleteAction: {
label: "Delete",
icon: IconX,
onPress: () => {},
},
},
{
name: "Google",
country: {
icon: IconInfoSolid,
text: "US",
},
button: {
label: "Row 2",
kind: "transparent",
hierarchy: "primary",
onPress: () => {},
} as const,
applications: 10_000,
status: {
label: "Paused",
color: "blue",
} as const,
value: {
numericValue: 150,
icon: IconInfoSolid,
},
website: {
href: "http://www.google.com",
label: "Link",
},
group: "Group 2",
deleteAction: {
label: "Delete",
icon: IconX,
onPress: () => {},
},
},
]}
columns={
[
tableColumn.button({
headerLabel: "Button",
accessor: "button",
sticky: "left",
disableSortBy: true,
align: "center",
}),
tableColumn.text({
headerLabel: "Name",
accessor: "name",
}),
tableColumn.textWithIcon({
headerLabel: "Country",
accessor: "country",
iconPosition: "right",
}),
tableColumn.number({
headerLabel: "Applications",
accessor: "applications",
valueFormatter: (value) => Intl.NumberFormat("en").format(value),
align: "right",
}),
tableColumn.numberWithIcon({
headerLabel: "Value",
accessor: "value",
valueFormatter: (value) => Intl.NumberFormat("en").format(value),
align: "right",
}),
tableColumn.link({
headerLabel: "Website",
accessor: "website",
}),
tableColumn.icon({
headerLabel: "Alerts",
accessor: "alerts",
}),
tableColumn.chip({
headerLabel: "Status",
accessor: "status",
align: "center",
}),
tableColumn.iconButton({
headerLabel: "Actions",
accessor: "deleteAction",
align: "center",
disableSortBy: true,
}),
] as const
}
/>

Props

NameTypeDefault ValueRequiredDescription
columnsreadonly Column<string, {}, any>[]Yes
datareadonly Row<C>[]Yes
groupBystringNo
noResultsTitlestringNo
noResultsDescriptionstringNo
noResultsFeedbackSize"large" | "medium"largeNo
initialSortingSortingRule<C>[]No
stickyHeadersbooleanNo
height{ custom: string | number; }No
onRowPress(row: Row<Row<C>>) => voidNo
customSorting(rows: Row<Row<C>>[], sortFns: SortFn<C>[]) => Row<Row<C>>[]No

customSorting can be used to customize the sorting logic of the table. It supports cross-columns comparison.

The sorting is still performed by the table, in an uncontrolled fashion.

This function must be memoized to avoid infinite re-renderings

onSort(sortBy: SortingRule<C>[]) => voidNo

onSort can be used to implement sorting in a controlled fashion. It's typical use case is when sorting and/or filtering is performed externally, for example by a backend service.

If onSort is provided the table won't perform any sorting on the rows given in input via the data property, as the caller is now in charge of performing the sorting.

This function must be memoized to avoid infinite re-renderings