From d4983fd4e16a18ea5a10d4247ade04e329022f5d Mon Sep 17 00:00:00 2001 From: Federico Ercoles Date: Thu, 14 Sep 2023 16:22:29 +0200 Subject: [PATCH 1/6] Add support for header groups --- .../src/Table/Table.css.ts | 14 +- .../bento-design-system/src/Table/Table.tsx | 167 ++++++++++++------ .../bento-design-system/src/Table/types.ts | 8 + 3 files changed, 132 insertions(+), 57 deletions(-) diff --git a/packages/bento-design-system/src/Table/Table.css.ts b/packages/bento-design-system/src/Table/Table.css.ts index 73eb34bed..c8027e72c 100644 --- a/packages/bento-design-system/src/Table/Table.css.ts +++ b/packages/bento-design-system/src/Table/Table.css.ts @@ -25,10 +25,16 @@ export const sortIconContainer = style({ filter: "opacity(80%)", }); -export const stickyColumnHeader = bentoSprinkles({ - position: "sticky", - top: 0, -}); +export const stickyTopHeight = createVar(); + +export const stickyColumnHeader = style([ + { + top: stickyTopHeight, + }, + bentoSprinkles({ + position: "sticky", + }), +]); export const rowContainer = style({ // NOTE(gabro): this allows us to use the entire row as a parent selector, diff --git a/packages/bento-design-system/src/Table/Table.tsx b/packages/bento-design-system/src/Table/Table.tsx index fbfc5e623..c367ffb96 100644 --- a/packages/bento-design-system/src/Table/Table.tsx +++ b/packages/bento-design-system/src/Table/Table.tsx @@ -9,6 +9,7 @@ import { useGroupBy, Cell, SortingRule, + HeaderGroup, } from "react-table"; import { IconProps } from "../Icons/IconProps"; import { useDefaultMessages } from "../util/useDefaultMessages"; @@ -38,21 +39,27 @@ import { selectedRowBackgroundColor, sortIconContainer, stickyColumnHeader, + stickyTopHeight, table, } from "./Table.css"; -import { Column as ColumnType, GridWidth, Row as RowType } from "./types"; +import { + Column as SimpleColumnType, + GroupedColumn as GroupedColumnType, + GridWidth, + Row as RowType, +} from "./types"; import { useLayoutEffect, useMemo, useState, CSSProperties, useEffect } from "react"; import { IconQuestionSolid, IconInfo } from "../Icons"; import { match, __ } from "ts-pattern"; import { useBentoConfig } from "../BentoConfigContext"; import { assignInlineVars } from "@vanilla-extract/dynamic"; -type SortFn>> = ( +type SortFn>> = ( a: Row>, b: Row> ) => number; -type SortingProps>> = +type SortingProps>> = | { /** * `customSorting` can be used to customize the sorting logic of the table. It supports cross-columns comparison. @@ -74,8 +81,11 @@ type SortingProps>> = onSort?: never; }; -type Props>> = { - columns: C; +type Props< + C extends ReadonlyArray>, + D extends ReadonlyArray> +> = { + columns: C | D; data: ReadonlyArray>; groupBy?: C[number]["accessor"]; noResultsTitle?: LocalizedString; @@ -106,7 +116,10 @@ type Props>> = { * * ``` */ -export function Table>>({ +export function Table< + C extends ReadonlyArray>, + D extends ReadonlyArray> +>({ columns, data, groupBy, @@ -119,7 +132,7 @@ export function Table>>({ stickyHeaders, height, onRowPress, -}: Props) { +}: Props) { const config = useBentoConfig().table; const customOrderByFn = useMemo( () => @@ -146,6 +159,11 @@ export function Table>>({ [customSorting] ); + const flatColumns = useMemo( + () => columns.flatMap((c) => ("columns" in c ? c.columns : [c])), + [columns] + ); + const { getTableProps, headerGroups, @@ -179,15 +197,19 @@ export function Table>>({ // Determine the ids of the sticky columns to the left const stickyLeftColumnsIds = useMemo( () => - columns - .filter((c) => c.sticky === "left") - .map((c) => headerGroups[0].headers.find((h) => h.id === c.accessor)?.id) - .filter((id): id is string => id !== undefined), - [columns, headerGroups] + headerGroups[0].headers + .filter((h) => h.sticky) + .flatMap((h) => h.columns || [h]) + .map((h) => h.id), + [headerGroups] + ); + const stickyLeftColumnGroupsIds = useMemo( + () => headerGroups[0].headers.filter((h) => h.sticky).map((h) => h.id), + [headerGroups] ); // Determine the id of the last left sticky column (used to draw a visual separator in the UI) - const lastStickyColumnIndex = columns + const lastStickyColumnIndex = flatColumns .map((c) => c.accessor) .indexOf(stickyLeftColumnsIds[stickyLeftColumnsIds.length - 1]); @@ -196,6 +218,9 @@ export function Table>>({ {} as Record ); + // Keep a state for the height of the first row of headers, which will be updated by the useLayoutEffect below + const [stickyHeaderHeight, setStickyHeaderHeight] = useState(0); + /** Get the width of each sticky column (using the header width as reference) and use it to set the `left` of each sticky column. * Each sticky column must have as `left` the total width of the previous sticky columns. */ @@ -206,35 +231,55 @@ export function Table>>({ const columnWidths = stickyLeftColumnsIds.map( (id) => document.getElementById(`header-cell-${id}`)!.clientWidth ); + const columnGroupWidths = stickyLeftColumnGroupsIds.map( + (id) => document.getElementById(`header-cell-${id}`)!.clientWidth + ); + const columnGroupHeight = Math.max( + ...headerGroups[0].headers.map((h) => + h.columns ? document.getElementById(`header-cell-${h.id}`)!.clientHeight : 0 + ) + ); - const columnStyles = stickyLeftColumnsIds.reduce((styles, id, index) => { - if (index > 0) { - const totalLeftWidth = columnWidths - .filter((_w, i) => i < index) - .reduce((acc, w) => acc + w, 0); - return { - ...styles, - [id]: { - left: totalLeftWidth, - zIndex: zIndexes.leftStickyCell, - position: "sticky", - } as CSSProperties, - }; - } else { - return { - ...styles, - [id]: { - left: 0, - zIndex: zIndexes.leftStickyCell, - position: "sticky", - } as CSSProperties, - }; - } - }, {} as Record); + const styleReducer = + (widths: number[]) => + (styles: Record, id: string, index: number) => { + if (index > 0) { + const totalLeftWidth = widths + .filter((_w, i) => i < index) + .reduce((acc, w) => acc + w, 0); + return { + ...styles, + [id]: { + left: totalLeftWidth, + zIndex: zIndexes.leftStickyCell, + position: "sticky", + } as CSSProperties, + }; + } else { + return { + ...styles, + [id]: { + left: 0, + zIndex: zIndexes.leftStickyCell, + position: "sticky", + } as CSSProperties, + }; + } + }; + + const columnStyles = stickyLeftColumnsIds.reduce( + styleReducer(columnWidths), + {} as Record + ); + const columnGroupStyles = stickyLeftColumnGroupsIds.reduce( + styleReducer(columnGroupWidths), + {} as Record + ); - setStickyLeftColumnStyle(columnStyles); + setStickyLeftColumnStyle({ ...columnStyles, ...columnGroupStyles }); + setStickyHeaderHeight(columnGroupHeight); } - }, [data.length, stickyLeftColumnsIds]); + }, [data.length, headerGroups, stickyLeftColumnsIds, stickyLeftColumnGroupsIds]); if (data.length === 0) { return ( @@ -265,7 +310,7 @@ export function Table>>({ .exhaustive(); } - function tableHeight(height: Props["height"]): string | undefined { + function tableHeight(height: Props["height"]): string | undefined { return match(height) .with({ custom: __.string }, ({ custom: width }) => width) .with({ custom: __.number }, ({ custom: width }) => `${width}px`) @@ -273,7 +318,7 @@ export function Table>>({ .exhaustive(); } - const gridTemplateColumns = columns + const gridTemplateColumns = flatColumns .filter(({ accessor }) => accessor !== groupBy) .map(({ gridWidth = "fit-content" }) => gridWidthStyle(gridWidth)) .join(" "); @@ -290,7 +335,7 @@ export function Table>>({ lastLeftSticky={index === lastStickyColumnIndex} style={stickyLeftColumnStyle[cell.column.id]} first={index === 0} - last={(index + 1) % columns.length === 0} + last={(index + 1) % flatColumns.length === 0} interactiveRow={interactiveRow} > {cell.render("Cell")} @@ -307,16 +352,28 @@ export function Table>>({ style={{ ...getTableProps().style, gridTemplateColumns, height: tableHeight(height) }} > {headerGroups.map((headerGroup) => - headerGroup.headers.map((column, index) => ( + headerGroup.headers.map((header, index) => ( )) )} @@ -347,7 +404,7 @@ export function Table>>({ ); } -function RowContainer>>({ +function RowContainer>>({ row, children, onPress, @@ -379,7 +436,7 @@ function ColumnHeader>({ first, last, }: { - column: ColumnInstance; + column: ColumnInstance | HeaderGroup; style: CSSProperties; lastLeftSticky: boolean; stickyHeaders?: boolean; @@ -432,7 +489,11 @@ function ColumnHeader>({ return ( = { Cell: (props: CellProps) => ReturnType; } & Omit, "accessor" | "Cell" | "width">; +export type GroupedColumn = { + Header: string; + columns: Array>; + align?: "left" | "right" | "center"; + sticky?: "left"; + hint?: LocalizedString | { onPress: () => void }; +}; + type ColumnValueByAccessor = C extends Column ? V : never; From 4c41f944ed011391101068d782ce9af31c1e1d07 Mon Sep 17 00:00:00 2001 From: Federico Ercoles Date: Thu, 14 Sep 2023 16:59:26 +0200 Subject: [PATCH 2/6] Add story for Table with grouped headers --- .../stories/Components/Table.stories.tsx | 88 ++++++++++++++++++- 1 file changed, 87 insertions(+), 1 deletion(-) diff --git a/packages/bento-design-system/stories/Components/Table.stories.tsx b/packages/bento-design-system/stories/Components/Table.stories.tsx index 692a94ed0..5eafc02ec 100644 --- a/packages/bento-design-system/stories/Components/Table.stories.tsx +++ b/packages/bento-design-system/stories/Components/Table.stories.tsx @@ -78,6 +78,84 @@ const exampleColumns = [ }), ] as const; +const exampleGroupedColumns = [ + { + Header: "Group 1", + sticky: "left" as const, + hint: "This is a hint", + columns: [ + tableColumn.button({ + headerLabel: "Button", + accessor: "button", + sticky: "left", + disableSortBy: true, + align: "center", + }), + tableColumn.text({ + headerLabel: "Name", + accessor: "name", + }), + tableColumn.text({ + headerLabel: "Extended address", + accessor: "address", + width: { custom: 200 }, + }), + tableColumn.textWithIcon({ + headerLabel: "Country", + accessor: "country", + iconPosition: "right", + hint: "This is a hint", + }), + ], + }, + { + Header: "Group 2", + columns: [ + tableColumn.number({ + headerLabel: "Applications", + accessor: "applications", + valueFormatter: (value) => Intl.NumberFormat("en").format(value), + align: "right", + hint: { onPress: action("hint") }, + }), + tableColumn.numberWithIcon({ + headerLabel: "Value", + accessor: "value", + valueFormatter: (value) => Intl.NumberFormat("en").format(value), + align: "right", + }), + tableColumn.label({ + headerLabel: "Type", + accessor: "type", + }), + tableColumn.link({ + headerLabel: "Website", + accessor: "website", + }), + ], + }, + { + Header: "Group 3", + columns: [ + tableColumn.icon({ + headerLabel: "Alerts", + accessor: "alerts", + }), + tableColumn.chip({ + headerLabel: "Status", + accessor: "status", + align: "center", + }), + tableColumn.iconButton({ + headerLabel: "Actions", + accessor: "deleteAction", + align: "center", + disableSortBy: true, + }), + ], + }, +]; + const customizedColumns = [ tableColumn.button({ headerLabel: "Button", @@ -290,7 +368,7 @@ const meta = { data: exampleData, }, parameters: { actions: { argTypesRegex: "" } }, -} satisfies Meta>; +} satisfies Meta>; export default meta; @@ -473,3 +551,11 @@ export const InteractiveRow = { onRowPress: action("onRowPress"), }, } satisfies Story; + +export const GroupedHeaders = { + args: { + columns: exampleGroupedColumns, + stickyHeaders: true, + height: { custom: 320 }, + }, +} satisfies Story; From 1ff9df89dd824a111ce29c8ec6e22130b18adb7b Mon Sep 17 00:00:00 2001 From: Federico Ercoles Date: Thu, 21 Sep 2023 10:53:46 +0200 Subject: [PATCH 3/6] Apply code improvements --- packages/bento-design-system/src/Table/Table.tsx | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/packages/bento-design-system/src/Table/Table.tsx b/packages/bento-design-system/src/Table/Table.tsx index c367ffb96..70a577e83 100644 --- a/packages/bento-design-system/src/Table/Table.tsx +++ b/packages/bento-design-system/src/Table/Table.tsx @@ -199,7 +199,7 @@ export function Table< () => headerGroups[0].headers .filter((h) => h.sticky) - .flatMap((h) => h.columns || [h]) + .flatMap((h) => h.columns ?? [h]) .map((h) => h.id), [headerGroups] ); @@ -364,7 +364,7 @@ export function Table< }} lastLeftSticky={ header.columns - ? header.id === stickyLeftColumnGroupsIds[stickyLeftColumnGroupsIds.length - 1] + ? header.id === stickyLeftColumnGroupsIds.at(-1) : index === lastStickyColumnIndex } stickyHeaders={stickyHeaders} From f9a8b2c8c5dea1d4fbccbe6eeca459692b6372bf Mon Sep 17 00:00:00 2001 From: Gabriele Petronella Date: Thu, 21 Sep 2023 15:24:12 +0200 Subject: [PATCH 4/6] Improve GroupedColumn typings --- .../bento-design-system/src/Table/Table.tsx | 45 ++++++++++++------- .../src/Table/tableColumn.tsx | 4 +- .../bento-design-system/src/Table/types.ts | 19 ++++++-- packages/bento-design-system/test/Table.tsx | 26 +++++++++++ 4 files changed, 74 insertions(+), 20 deletions(-) diff --git a/packages/bento-design-system/src/Table/Table.tsx b/packages/bento-design-system/src/Table/Table.tsx index 70a577e83..bf0e45453 100644 --- a/packages/bento-design-system/src/Table/Table.tsx +++ b/packages/bento-design-system/src/Table/Table.tsx @@ -54,12 +54,17 @@ import { match, __ } from "ts-pattern"; import { useBentoConfig } from "../BentoConfigContext"; import { assignInlineVars } from "@vanilla-extract/dynamic"; -type SortFn>> = ( - a: Row>, - b: Row> -) => number; - -type SortingProps>> = +type SortFn< + C extends + | ReadonlyArray> + | ReadonlyArray> +> = (a: Row>, b: Row>) => number; + +type SortingProps< + C extends + | ReadonlyArray> + | ReadonlyArray> +> = | { /** * `customSorting` can be used to customize the sorting logic of the table. It supports cross-columns comparison. @@ -82,12 +87,17 @@ type SortingProps>> = }; type Props< - C extends ReadonlyArray>, - D extends ReadonlyArray> + C extends + | ReadonlyArray> + | ReadonlyArray> > = { - columns: C | D; + columns: C; data: ReadonlyArray>; - groupBy?: C[number]["accessor"]; + groupBy?: C extends ReadonlyArray> + ? C[number]["accessor"] + : C extends ReadonlyArray> + ? C[number]["columns"][number]["accessor"] + : never; noResultsTitle?: LocalizedString; noResultsDescription?: LocalizedString; noResultsFeedbackSize?: FeedbackProps["size"]; @@ -117,8 +127,9 @@ type Props< * ``` */ export function Table< - C extends ReadonlyArray>, - D extends ReadonlyArray> + C extends + | ReadonlyArray> + | ReadonlyArray> >({ columns, data, @@ -132,7 +143,7 @@ export function Table< stickyHeaders, height, onRowPress, -}: Props) { +}: Props) { const config = useBentoConfig().table; const customOrderByFn = useMemo( () => @@ -310,7 +321,7 @@ export function Table< .exhaustive(); } - function tableHeight(height: Props["height"]): string | undefined { + function tableHeight(height: Props["height"]): string | undefined { return match(height) .with({ custom: __.string }, ({ custom: width }) => width) .with({ custom: __.number }, ({ custom: width }) => `${width}px`) @@ -404,7 +415,11 @@ export function Table< ); } -function RowContainer>>({ +function RowContainer< + C extends + | ReadonlyArray> + | ReadonlyArray> +>({ row, children, onPress, diff --git a/packages/bento-design-system/src/Table/tableColumn.tsx b/packages/bento-design-system/src/Table/tableColumn.tsx index 748f36b04..dcce71376 100644 --- a/packages/bento-design-system/src/Table/tableColumn.tsx +++ b/packages/bento-design-system/src/Table/tableColumn.tsx @@ -31,8 +31,8 @@ export type ColumnOptionsBase = { accessor: A; headerLabel?: LocalizedString; missingValue?: LocalizedString; - width?: Column_<{}>["gridWidth"]; -} & Omit, "accessor" | "Header" | "Cell" | "sortType" | "width" | "gridWidth">; + width?: Column_["gridWidth"]; +} & Omit, "accessor" | "Header" | "Cell" | "sortType" | "width" | "gridWidth">; export function custom>({ headerLabel, diff --git a/packages/bento-design-system/src/Table/types.ts b/packages/bento-design-system/src/Table/types.ts index e7351b8a0..ce6409c8e 100644 --- a/packages/bento-design-system/src/Table/types.ts +++ b/packages/bento-design-system/src/Table/types.ts @@ -41,8 +41,21 @@ type ColumnValueByAccessor = C extends Column>> = { - [k in Columns[number]["accessor"]]?: ColumnValueByAccessor; -}; +export type Row< + Columns extends + | ReadonlyArray> + | ReadonlyArray> +> = Columns[number] extends GroupedColumn + ? { + [k in Columns[number]["columns"][number]["accessor"]]?: ColumnValueByAccessor< + Columns[number]["columns"][number], + k + >; + } + : Columns[number] extends Column + ? { + [k in Columns[number]["accessor"]]?: ColumnValueByAccessor; + } + : never; export type GridWidth = "fit-content" | "fill-available" | { custom: string | number }; diff --git a/packages/bento-design-system/test/Table.tsx b/packages/bento-design-system/test/Table.tsx index f06fbeb94..35d2da88c 100644 --- a/packages/bento-design-system/test/Table.tsx +++ b/packages/bento-design-system/test/Table.tsx @@ -52,3 +52,29 @@ const columns = [ // passing columns inline groupBy="error" />; + +
; From 267064849c8b60e014831e9f7442042887772b77 Mon Sep 17 00:00:00 2001 From: Gabriele Petronella Date: Thu, 21 Sep 2023 15:37:51 +0200 Subject: [PATCH 5/6] Add type --- packages/bento-design-system/src/Table/tableColumn.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/bento-design-system/src/Table/tableColumn.tsx b/packages/bento-design-system/src/Table/tableColumn.tsx index dcce71376..2ae0b3cc2 100644 --- a/packages/bento-design-system/src/Table/tableColumn.tsx +++ b/packages/bento-design-system/src/Table/tableColumn.tsx @@ -47,7 +47,7 @@ export function custom>({ const column = { ...options, gridWidth: width, - Cell: (props) => { + Cell: (props: CellProps) => { const { defaultMessages } = useDefaultMessages(); const config = useBentoConfig().table; if (props.value == null) { From 1e98a251590cdbb252b1ff7506e4ad963aa7a811 Mon Sep 17 00:00:00 2001 From: Gabriele Petronella Date: Thu, 21 Sep 2023 15:44:27 +0200 Subject: [PATCH 6/6] Whatever --- packages/bento-design-system/src/useComponentsShowcase.tsx | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/packages/bento-design-system/src/useComponentsShowcase.tsx b/packages/bento-design-system/src/useComponentsShowcase.tsx index 1dd64edd7..c574af335 100644 --- a/packages/bento-design-system/src/useComponentsShowcase.tsx +++ b/packages/bento-design-system/src/useComponentsShowcase.tsx @@ -556,7 +556,7 @@ export function useComponentsShowcase({ action }: { action: (s: string) => () => experiments: 0, status: { label: "pending", color: "yellow" }, }, - ], + ] as any, }, ], ],