From 44d13578d6dcdceea314eea93cdae86e946a7c50 Mon Sep 17 00:00:00 2001 From: Armin Mehinovic <4390250+arminmeh@users.noreply.github.com> Date: Thu, 31 Oct 2024 19:42:26 +0100 Subject: [PATCH] [DataGridPremium] Fix incorrect rows selection count when selection propagation is enabled with row grouping (#15216) --- .../rowSelection.DataGridPremium.test.tsx | 20 +++++++++++++- .../rowSelection/useGridRowSelection.ts | 26 ++++++++----------- 2 files changed, 30 insertions(+), 16 deletions(-) diff --git a/packages/x-data-grid-premium/src/tests/rowSelection.DataGridPremium.test.tsx b/packages/x-data-grid-premium/src/tests/rowSelection.DataGridPremium.test.tsx index d330b84ebd8c7..88bb7b1dd69e0 100644 --- a/packages/x-data-grid-premium/src/tests/rowSelection.DataGridPremium.test.tsx +++ b/packages/x-data-grid-premium/src/tests/rowSelection.DataGridPremium.test.tsx @@ -1,5 +1,5 @@ import * as React from 'react'; -import { createRenderer, fireEvent } from '@mui/internal-test-utils'; +import { act, createRenderer, fireEvent } from '@mui/internal-test-utils'; import { getCell } from 'test/utils/helperFn'; import { expect } from 'chai'; import { @@ -122,6 +122,24 @@ describe(' - Row selection', () => { expect(apiRef.current.getSelectedRows()).to.have.keys([0, 2]); }); + // Context: https://github.com/mui/mui-x/issues/15206 + it('should keep the correct selection items and the selection count when rows are updated', () => { + render(); + + const expectedKeys = ['auto-generated-row-category1/Cat B', 3, 4]; + const expectedCount = 3; + + fireEvent.click(getCell(1, 0).querySelector('input')!); + expect(apiRef.current.getSelectedRows()).to.have.keys(expectedKeys); + expect(apiRef.current.state.rowSelection.length).to.equal(expectedCount); + + act(() => { + apiRef.current.updateRows([...rows]); + }); + expect(apiRef.current.getSelectedRows()).to.have.keys(expectedKeys); + expect(apiRef.current.state.rowSelection.length).to.equal(expectedCount); + }); + describe("prop: indeterminateCheckboxAction = 'select'", () => { it('should select all the children when selecting an indeterminate parent', () => { render( diff --git a/packages/x-data-grid/src/hooks/features/rowSelection/useGridRowSelection.ts b/packages/x-data-grid/src/hooks/features/rowSelection/useGridRowSelection.ts index 2c8515776a13a..fb8b76b078a40 100644 --- a/packages/x-data-grid/src/hooks/features/rowSelection/useGridRowSelection.ts +++ b/packages/x-data-grid/src/hooks/features/rowSelection/useGridRowSelection.ts @@ -263,6 +263,7 @@ export const useGridRowSelection = ( const removeRow = (rowId: GridRowId) => { newSelection.delete(rowId); }; + if (isSelected) { addRow(id); if (applyAutoSelection) { @@ -309,13 +310,13 @@ export const useGridRowSelection = ( const selectableIds = ids.filter((id) => apiRef.current.isRowSelectable(id)); - let newSelection: GridRowId[]; + let newSelection: Set; if (resetSelection) { if (isSelected) { - newSelection = selectableIds; + newSelection = new Set(selectableIds); if (applyAutoSelection) { const addRow = (rowId: GridRowId) => { - newSelection.push(rowId); + newSelection.add(rowId); }; selectableIds.forEach((id) => { findRowsToSelect( @@ -329,23 +330,20 @@ export const useGridRowSelection = ( }); } } else { - newSelection = []; + newSelection = new Set(); } } else { - // We clone the existing object to avoid mutating the same object returned by the selector to others part of the project - const selectionLookup = { - ...selectedIdsLookupSelector(apiRef), - }; + newSelection = new Set(Object.values(selectedIdsLookupSelector(apiRef))); const addRow = (rowId: GridRowId) => { - selectionLookup[rowId] = rowId; + newSelection.add(rowId); }; const removeRow = (rowId: GridRowId) => { - delete selectionLookup[rowId]; + newSelection.delete(rowId); }; selectableIds.forEach((id) => { if (isSelected) { - selectionLookup[id] = id; + newSelection.add(id); if (applyAutoSelection) { findRowsToSelect( apiRef, @@ -370,13 +368,11 @@ export const useGridRowSelection = ( } } }); - - newSelection = Object.values(selectionLookup); } - const isSelectionValid = newSelection.length < 2 || canHaveMultipleSelection; + const isSelectionValid = newSelection.size < 2 || canHaveMultipleSelection; if (isSelectionValid) { - apiRef.current.setRowSelectionModel(newSelection); + apiRef.current.setRowSelectionModel(Array.from(newSelection)); } }, [