From d42fae1def4cd17217f0e959df75a3ea15fc9a45 Mon Sep 17 00:00:00 2001 From: Bilal Shafi Date: Mon, 15 Apr 2024 23:50:21 +0500 Subject: [PATCH] [DataGrid] Support state export and restore on grid density (#12671) --- .../statePersistence.DataGridPro.test.tsx | 5 ++ .../hooks/features/density/useGridDensity.tsx | 56 ++++++++++++++++--- 2 files changed, 54 insertions(+), 7 deletions(-) diff --git a/packages/x-data-grid-pro/src/tests/statePersistence.DataGridPro.test.tsx b/packages/x-data-grid-pro/src/tests/statePersistence.DataGridPro.test.tsx index 091f51a8697c4..85880a295f635 100644 --- a/packages/x-data-grid-pro/src/tests/statePersistence.DataGridPro.test.tsx +++ b/packages/x-data-grid-pro/src/tests/statePersistence.DataGridPro.test.tsx @@ -78,6 +78,7 @@ const FULL_INITIAL_STATE: GridInitialState = { sorting: { sortModel: [{ field: 'id', sort: 'desc' }], }, + density: 'compact', }; describe(' - State persistence', () => { @@ -135,6 +136,7 @@ describe(' - State persistence', () => { sorting: { sortModel: [], }, + density: 'standard', }); }); @@ -165,6 +167,7 @@ describe(' - State persistence', () => { paginationMode="server" rowCount={FULL_INITIAL_STATE.pagination?.rowCount} pinnedColumns={FULL_INITIAL_STATE.pinnedColumns} + density={FULL_INITIAL_STATE.density} // Some portable states don't have a controllable model initialState={{ columns: { @@ -191,6 +194,7 @@ describe(' - State persistence', () => { paginationMode="server" rowCount={FULL_INITIAL_STATE.pagination?.rowCount} pinnedColumns={FULL_INITIAL_STATE.pinnedColumns} + density={FULL_INITIAL_STATE.density} // Some portable states don't have a controllable model initialState={{ columns: { @@ -226,6 +230,7 @@ describe(' - State persistence', () => { apiRef.current.setColumnIndex('category', 1); apiRef.current.setColumnWidth('category', 75); apiRef.current.setColumnVisibilityModel({ idBis: false }); + apiRef.current.setDensity('compact'); }); expect(apiRef.current.exportState()).to.deep.equal(FULL_INITIAL_STATE); }); diff --git a/packages/x-data-grid/src/hooks/features/density/useGridDensity.tsx b/packages/x-data-grid/src/hooks/features/density/useGridDensity.tsx index 172b161c2cdbf..0d21ec3a071c4 100644 --- a/packages/x-data-grid/src/hooks/features/density/useGridDensity.tsx +++ b/packages/x-data-grid/src/hooks/features/density/useGridDensity.tsx @@ -7,6 +7,7 @@ import { GridDensityApi } from '../../../models/api/gridDensityApi'; import { DataGridProcessedProps } from '../../../models/props/DataGridProps'; import { gridDensitySelector } from './densitySelector'; import { GridStateInitializer } from '../../utils/useGridInitializeState'; +import { GridPipeProcessor, useGridRegisterPipeProcessor } from '../../core/pipeProcessing'; export const densityStateInitializer: GridStateInitializer< Pick @@ -17,7 +18,7 @@ export const densityStateInitializer: GridStateInitializer< export const useGridDensity = ( apiRef: React.MutableRefObject, - props: Pick, + props: Pick, ): void => { const logger = useGridLogger(apiRef, 'useDensity'); @@ -43,15 +44,56 @@ export const useGridDensity = ( })); }); - React.useEffect(() => { - if (props.density) { - apiRef.current.setDensity(props.density); - } - }, [apiRef, props.density]); - const densityApi: GridDensityApi = { setDensity, }; useGridApiMethod(apiRef, densityApi, 'public'); + + const stateExportPreProcessing = React.useCallback>( + (prevState, context) => { + const exportedDensity = gridDensitySelector(apiRef.current.state); + + const shouldExportRowCount = + // Always export if the `exportOnlyDirtyModels` property is not activated + !context.exportOnlyDirtyModels || + // Always export if the `density` is controlled + props.density != null || + // Always export if the `density` has been initialized + props.initialState?.density != null; + + if (!shouldExportRowCount) { + return prevState; + } + + return { + ...prevState, + density: exportedDensity, + }; + }, + [apiRef, props.density, props.initialState?.density], + ); + + const stateRestorePreProcessing = React.useCallback>( + (params, context) => { + const restoredDensity = context.stateToRestore?.density + ? context.stateToRestore.density + : gridDensitySelector(apiRef.current.state); + apiRef.current.setState((state) => ({ + ...state, + density: restoredDensity, + })); + return params; + }, + [apiRef], + ); + + useGridRegisterPipeProcessor(apiRef, 'exportState', stateExportPreProcessing); + useGridRegisterPipeProcessor(apiRef, 'restoreState', stateRestorePreProcessing); + + React.useEffect(() => { + if (props.density) { + apiRef.current.setDensity(props.density); + } + }, [apiRef, props.density]); };