From 98f581260dfef4a6f8bc571858413c15c5ca456c Mon Sep 17 00:00:00 2001 From: Luis Gustavo Date: Tue, 4 Jun 2024 07:25:27 -0300 Subject: [PATCH 1/3] Passing generateGroupId props --- src/TreeDataGrid.tsx | 16 ++++++++++--- test/browser/TreeDataGrid.test.tsx | 36 +++++++++++++++++++++++++++--- 2 files changed, 46 insertions(+), 6 deletions(-) diff --git a/src/TreeDataGrid.tsx b/src/TreeDataGrid.tsx index 16e4f87232..6733ce1dbb 100644 --- a/src/TreeDataGrid.tsx +++ b/src/TreeDataGrid.tsx @@ -36,6 +36,7 @@ export interface TreeDataGridProps ) => Record[]>; expandedGroupIds: ReadonlySet; onExpandedGroupIdsChange: (expandedGroupIds: Set) => void; + generateGroupId: ((groupKey: string, parentId?: string) => string) | undefined; } type GroupByDictionary = Record< @@ -62,6 +63,7 @@ function TreeDataGrid( rowGrouper, expandedGroupIds, onExpandedGroupIdsChange, + generateGroupId, ...props }: TreeDataGridProps, ref: React.Ref @@ -144,6 +146,15 @@ function TreeDataGrid( if (!groupedRows) return [rawRows, isGroupRow]; const flattenedRows: Array> = []; + + const groupIdGenerator = (groupKey: string, parentId?: string) => { + if (generateGroupId !== undefined && typeof generateGroupId === 'function') { + return generateGroupId(groupKey, parentId); + } + + return `${parentId}__${groupKey}`; + }; + const expandGroup = ( rows: GroupByDictionary | readonly R[], parentId: string | undefined, @@ -154,8 +165,7 @@ function TreeDataGrid( return; } Object.keys(rows).forEach((groupKey, posInSet, keys) => { - // TODO: should users have control over the generated key? - const id = parentId !== undefined ? `${parentId}__${groupKey}` : groupKey; + const id = groupIdGenerator(groupKey, parentId); const isExpanded = expandedGroupIds.has(id); const { childRows, childGroups, startRowIndex } = rows[groupKey]; @@ -185,7 +195,7 @@ function TreeDataGrid( function isGroupRow(row: R | GroupRow): row is GroupRow { return allGroupRows.has(row); } - }, [expandedGroupIds, groupedRows, rawRows]); + }, [expandedGroupIds, groupedRows, rawRows, generateGroupId]); const rowHeight = useMemo(() => { if (typeof rawRowHeight === 'function') { diff --git a/test/browser/TreeDataGrid.test.tsx b/test/browser/TreeDataGrid.test.tsx index c5073b4617..01b774b556 100644 --- a/test/browser/TreeDataGrid.test.tsx +++ b/test/browser/TreeDataGrid.test.tsx @@ -92,7 +92,12 @@ function rowKeyGetter(row: Row) { return row.id; } -function TestGrid({ groupBy }: { groupBy: string[] }) { +interface TestGridProps { + groupBy: string[]; + generateGroupId: ((groupKey: string, parentId?: string) => string) | undefined; +} + +function TestGrid({ groupBy, generateGroupId }: TestGridProps) { const [rows, setRows] = useState(initialRows); const [selectedRows, setSelectedRows] = useState((): ReadonlySet => new Set()); const [expandedGroupIds, setExpandedGroupIds] = useState( @@ -121,6 +126,7 @@ function TestGrid({ groupBy }: { groupBy: string[] }) { onExpandedGroupIdsChange={setExpandedGroupIds} onRowsChange={setRows} onPaste={onPaste} + generateGroupId={generateGroupId} /> ); } @@ -131,8 +137,11 @@ function rowGrouper(rows: readonly Row[], columnKey: string) { return Object.groupBy(rows, (r) => r[columnKey]) as Record; } -function setup(groupBy: string[]) { - render(); +function setup( + groupBy: string[], + generateGroupId?: (groupKey: string, parentId?: string) => string +) { + render(); } function getHeaderCellsContent() { @@ -166,6 +175,15 @@ test('should group by multiple columns', () => { expect(getRowsOld()).toHaveLength(4); }); +test('should group by multiple columns when passing generateGroupId', () => { + setup(['country', 'year'], (groupKey, parentId) => + parentId !== undefined ? `${groupKey}#${parentId}` : groupKey + ); + expect(getTreeGrid()).toHaveAttribute('aria-rowcount', '13'); + expect(getHeaderCellsContent()).toStrictEqual(['', 'Country', 'Year', 'Sport', 'Id']); + expect(getRows()).toHaveLength(4); +}); + test('should ignore duplicate groupBy columns', () => { setup(['year', 'year', 'year']); expect(getTreeGridOld()).toHaveAttribute('aria-rowcount', '10'); @@ -189,6 +207,18 @@ test('should toggle group when group cell is clicked', async () => { expect(getRowsOld()).toHaveLength(5); }); +test('should toggle group when group cell is clicked and is passing `generateGroupId` props', async () => { + setup(['country', 'year'], (groupKey, parentId) => + parentId !== undefined ? `${groupKey}#${parentId}` : groupKey + ); + expect(getTreeGrid()).toHaveAttribute('aria-rowcount', '13'); + expect(getHeaderCellsContent()).toStrictEqual(['', 'Country', 'Year', 'Sport', 'Id']); + await userEvent.click(screen.getByRole('gridcell', { name: 'USA' })); + expect(getRows()).toHaveLength(6); + await userEvent.click(screen.getByRole('gridcell', { name: 'Canada' })); + expect(getRows()).toHaveLength(8); +}); + test('should toggle group using keyboard', async () => { setup(['year']); expect(getRowsOld()).toHaveLength(5); From a64fc4b73e9c3bbe8ad61ebe2d069d603e29fe56 Mon Sep 17 00:00:00 2001 From: Luis Gustavo Date: Tue, 4 Jun 2024 07:42:54 -0300 Subject: [PATCH 2/3] expand all levels --- test/browser/TreeDataGrid.test.tsx | 2 ++ 1 file changed, 2 insertions(+) diff --git a/test/browser/TreeDataGrid.test.tsx b/test/browser/TreeDataGrid.test.tsx index 01b774b556..979a956f12 100644 --- a/test/browser/TreeDataGrid.test.tsx +++ b/test/browser/TreeDataGrid.test.tsx @@ -217,6 +217,8 @@ test('should toggle group when group cell is clicked and is passing `generateGro expect(getRows()).toHaveLength(6); await userEvent.click(screen.getByRole('gridcell', { name: 'Canada' })); expect(getRows()).toHaveLength(8); + await userEvent.click(screen.getByRole('gridcell', { name: '2020' })); + expect(getRows()).toHaveLength(9); }); test('should toggle group using keyboard', async () => { From 25516150e74be50e14749fdf5233d54aa55cac56 Mon Sep 17 00:00:00 2001 From: Luis Gustavo Date: Tue, 4 Jun 2024 07:44:18 -0300 Subject: [PATCH 3/3] conditional props in test --- src/TreeDataGrid.tsx | 2 +- test/browser/TreeDataGrid.test.tsx | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/src/TreeDataGrid.tsx b/src/TreeDataGrid.tsx index 6733ce1dbb..d2f036e768 100644 --- a/src/TreeDataGrid.tsx +++ b/src/TreeDataGrid.tsx @@ -36,7 +36,7 @@ export interface TreeDataGridProps ) => Record[]>; expandedGroupIds: ReadonlySet; onExpandedGroupIdsChange: (expandedGroupIds: Set) => void; - generateGroupId: ((groupKey: string, parentId?: string) => string) | undefined; + generateGroupId?: (groupKey: string, parentId?: string) => string; } type GroupByDictionary = Record< diff --git a/test/browser/TreeDataGrid.test.tsx b/test/browser/TreeDataGrid.test.tsx index 979a956f12..ab269f3435 100644 --- a/test/browser/TreeDataGrid.test.tsx +++ b/test/browser/TreeDataGrid.test.tsx @@ -126,7 +126,7 @@ function TestGrid({ groupBy, generateGroupId }: TestGridProps) { onExpandedGroupIdsChange={setExpandedGroupIds} onRowsChange={setRows} onPaste={onPaste} - generateGroupId={generateGroupId} + {...(generateGroupId && { generateGroupId })} /> ); }