From 456018fbeaf703c18bce5f50873b529d63a048b0 Mon Sep 17 00:00:00 2001 From: Andrew Cherniavskyi Date: Fri, 3 Nov 2023 12:56:48 +0100 Subject: [PATCH 1/4] fix cell value type in quick filtering v7 --- packages/grid/x-data-grid/src/colDef/gridStringOperators.ts | 3 ++- .../x-data-grid/src/hooks/features/filter/gridFilterUtils.ts | 2 +- 2 files changed, 3 insertions(+), 2 deletions(-) diff --git a/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts b/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts index aa875f803a13e..9a00deb28c33d 100644 --- a/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts +++ b/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts @@ -12,7 +12,8 @@ export const getGridStringQuickFilterFn = tagInternalFilter( return null; } const filterRegex = new RegExp(escapeRegExp(value), 'i'); - return (columnValue): boolean => { + return (_, row, column, apiRef): boolean => { + const columnValue = apiRef.current.getRowFormattedValue(row, column); return columnValue != null ? filterRegex.test(columnValue.toString()) : false; }; }, diff --git a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts index a4418aa3626d6..04653f2c85cac 100644 --- a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts +++ b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts @@ -406,7 +406,7 @@ const buildAggregatedQuickFilterApplier = ( } const applier = appliers[v]; - let value = apiRef.current.getRowFormattedValue(row, column); + let value = apiRef.current.getRowValue(row, column); if (applier.fn === null) { continue; From c89d5252acd3e287bdb7521ca95102d6f2f55750 Mon Sep 17 00:00:00 2001 From: Andrew Cherniavskyi Date: Sat, 4 Nov 2023 11:56:26 +0100 Subject: [PATCH 2/4] make ignoreDiacritics work in quick filtering --- docs/pages/x/api/data-grid/grid-api.md | 1 + docs/pages/x/api/data-grid/grid-filter-api.json | 5 +++++ packages/grid/x-data-grid/src/colDef/gridStringOperators.ts | 6 +++++- .../src/hooks/features/filter/gridFilterUtils.ts | 2 +- .../x-data-grid/src/hooks/features/filter/useGridFilter.tsx | 2 ++ packages/grid/x-data-grid/src/models/api/gridFilterApi.ts | 5 +++++ 6 files changed, 19 insertions(+), 2 deletions(-) diff --git a/docs/pages/x/api/data-grid/grid-api.md b/docs/pages/x/api/data-grid/grid-api.md index 82c74b0cea430..255aa9b096bfc 100644 --- a/docs/pages/x/api/data-grid/grid-api.md +++ b/docs/pages/x/api/data-grid/grid-api.md @@ -74,6 +74,7 @@ import { GridApi } from '@mui/x-data-grid'; | hideFilterPanel | () => void | Hides the filter panel. | | hideHeaderFilterMenu | () => void | Hides the header filter menu. | | hidePreferences | () => void | Hides the preferences panel. | +| ignoreDiacritics | DataGridProcessedProps['ignoreDiacritics'] | Returns the value of the `ignoreDiacritics` prop. | | isCellEditable | (params: GridCellParams) => boolean | Controls if a cell is editable. | | isColumnPinned [](/x/introduction/licensing/#pro-plan) | (field: string) => GridPinnedPosition \| false | Returns which side a column is pinned to. | | isRowSelectable | (id: GridRowId) => boolean | Determines if a row can be selected or not. | diff --git a/docs/pages/x/api/data-grid/grid-filter-api.json b/docs/pages/x/api/data-grid/grid-filter-api.json index c80e144545f89..626ae4fcb29ec 100644 --- a/docs/pages/x/api/data-grid/grid-filter-api.json +++ b/docs/pages/x/api/data-grid/grid-filter-api.json @@ -8,6 +8,11 @@ "type": "(item: GridFilterItem) => void" }, { "name": "hideFilterPanel", "description": "Hides the filter panel.", "type": "() => void" }, + { + "name": "ignoreDiacritics", + "description": "Returns the value of the ignoreDiacritics prop.", + "type": "DataGridProcessedProps['ignoreDiacritics']" + }, { "name": "setFilterLogicOperator", "description": "Changes the GridLogicOperator used to connect the filters.", diff --git a/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts b/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts index 9a00deb28c33d..dca999638aa87 100644 --- a/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts +++ b/packages/grid/x-data-grid/src/colDef/gridStringOperators.ts @@ -5,6 +5,7 @@ import { GridFilterItem } from '../models/gridFilterItem'; import { GridFilterOperator } from '../models/gridFilterOperator'; import { GridFilterInputMultipleValue } from '../components/panel/filterPanel/GridFilterInputMultipleValue'; import { convertLegacyOperators, tagInternalFilter } from './utils'; +import { removeDiacritics } from '../hooks/features/filter/gridFilterUtils'; export const getGridStringQuickFilterFn = tagInternalFilter( (value: any): GridApplyQuickFilterV7 | null => { @@ -13,7 +14,10 @@ export const getGridStringQuickFilterFn = tagInternalFilter( } const filterRegex = new RegExp(escapeRegExp(value), 'i'); return (_, row, column, apiRef): boolean => { - const columnValue = apiRef.current.getRowFormattedValue(row, column); + let columnValue = apiRef.current.getRowFormattedValue(row, column); + if (apiRef.current.ignoreDiacritics) { + columnValue = removeDiacritics(columnValue); + } return columnValue != null ? filterRegex.test(columnValue.toString()) : false; }; }, diff --git a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts index 04653f2c85cac..84ddb35c5d0c8 100644 --- a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts +++ b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts @@ -152,7 +152,7 @@ export const mergeStateWithFilterModel = filterModel: sanitizeFilterModel(filterModel, disableMultipleColumnsFiltering, apiRef), }); -const removeDiacritics = (value: unknown) => { +export const removeDiacritics = (value: unknown) => { if (typeof value === 'string') { return value.normalize('NFD').replace(/[\u0300-\u036f]/g, ''); } diff --git a/packages/grid/x-data-grid/src/hooks/features/filter/useGridFilter.tsx b/packages/grid/x-data-grid/src/hooks/features/filter/useGridFilter.tsx index 6e70483397e03..166e52a5ced0d 100644 --- a/packages/grid/x-data-grid/src/hooks/features/filter/useGridFilter.tsx +++ b/packages/grid/x-data-grid/src/hooks/features/filter/useGridFilter.tsx @@ -90,6 +90,7 @@ export const useGridFilter = ( | 'slotProps' | 'disableColumnFilter' | 'disableEval' + | 'ignoreDiacritics' >, ): void => { const logger = useGridLogger(apiRef, 'useGridFilter'); @@ -329,6 +330,7 @@ export const useGridFilter = ( showFilterPanel, hideFilterPanel, setQuickFilterValues, + ignoreDiacritics: props.ignoreDiacritics, }; useGridApiMethod(apiRef, filterApi, 'public'); diff --git a/packages/grid/x-data-grid/src/models/api/gridFilterApi.ts b/packages/grid/x-data-grid/src/models/api/gridFilterApi.ts index 546c0787bd432..16f3b6194f991 100644 --- a/packages/grid/x-data-grid/src/models/api/gridFilterApi.ts +++ b/packages/grid/x-data-grid/src/models/api/gridFilterApi.ts @@ -1,6 +1,7 @@ import { GridFilterModel } from '../gridFilterModel'; import { GridFilterItem, GridLogicOperator } from '../gridFilterItem'; import { GridControlledStateReasonLookup } from '../events'; +import { DataGridProcessedProps } from '../props/DataGridProps'; /** * The filter API interface that is available in the grid [[apiRef]]. @@ -56,4 +57,8 @@ export interface GridFilterApi { * @param {any[]} values The list of element to quick filter */ setQuickFilterValues: (values: any[]) => void; + /** + * Returns the value of the `ignoreDiacritics` prop. + */ + ignoreDiacritics: DataGridProcessedProps['ignoreDiacritics']; } From a768f3111c1f9fd645aa626afaf3b5b40a909f81 Mon Sep 17 00:00:00 2001 From: Andrew Cherniavskyi Date: Sat, 4 Nov 2023 11:58:14 +0100 Subject: [PATCH 3/4] avoid exposing private apiRef --- .../hooks/features/clipboard/useGridClipboardImport.ts | 3 ++- .../src/hooks/features/filter/gridFilterUtils.ts | 9 +++++---- packages/grid/x-data-grid/src/internals/index.ts | 1 + packages/grid/x-data-grid/src/utils/getPublicApiRef.ts | 9 +++++++++ 4 files changed, 17 insertions(+), 5 deletions(-) create mode 100644 packages/grid/x-data-grid/src/utils/getPublicApiRef.ts diff --git a/packages/grid/x-data-grid-premium/src/hooks/features/clipboard/useGridClipboardImport.ts b/packages/grid/x-data-grid-premium/src/hooks/features/clipboard/useGridClipboardImport.ts index c4dc4ac80dfdf..2c36a29fc294b 100644 --- a/packages/grid/x-data-grid-premium/src/hooks/features/clipboard/useGridClipboardImport.ts +++ b/packages/grid/x-data-grid-premium/src/hooks/features/clipboard/useGridClipboardImport.ts @@ -19,6 +19,7 @@ import { getActiveElement, GridPipeProcessor, useGridRegisterPipeProcessor, + getPublicApiRef, } from '@mui/x-data-grid/internals'; import { GRID_DETAIL_PANEL_TOGGLE_FIELD, GRID_REORDER_COL_DEF } from '@mui/x-data-grid-pro'; import { unstable_debounce as debounce } from '@mui/utils'; @@ -379,7 +380,7 @@ export const useGridClipboardImport = ( defaultPasteResolver({ pastedData, - apiRef: { current: apiRef.current.getPublicApi() }, + apiRef: getPublicApiRef(apiRef), updateCell: (...args) => { cellUpdater.updateCell(...args); }, diff --git a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts index 84ddb35c5d0c8..6c2aff267d43a 100644 --- a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts +++ b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts @@ -18,6 +18,7 @@ import { GridQuickFilterValueResult, } from './gridFilterState'; import { buildWarning } from '../../../utils/warning'; +import { getPublicApiRef } from '../../../utils/getPublicApiRef'; import { gridColumnFieldsSelector, gridColumnLookupSelector, @@ -220,7 +221,7 @@ const getFilterCallbackFromItem = ( if (ignoreDiacritics) { value = removeDiacritics(value); } - return applyFilterOnRow(value, row, column, apiRef); + return applyFilterOnRow(value, row, column, getPublicApiRef(apiRef)); }, }; } @@ -371,7 +372,7 @@ const buildAggregatedQuickFilterApplier = ( const value = ignoreDiacritics ? removeDiacritics(quickFilterValue) : quickFilterValue; return { v7: true, - fn: getApplyQuickFilterFnV7(value, column, apiRef), + fn: getApplyQuickFilterFnV7(value, column, getPublicApiRef(apiRef)), }; }), }); @@ -382,7 +383,7 @@ const buildAggregatedQuickFilterApplier = ( const value = ignoreDiacritics ? removeDiacritics(quickFilterValue) : quickFilterValue; return { v7: false, - fn: getApplyQuickFilterFn(value, column, apiRef), + fn: getApplyQuickFilterFn(value, column, getPublicApiRef(apiRef)), }; }), }); @@ -416,7 +417,7 @@ const buildAggregatedQuickFilterApplier = ( if (ignoreDiacritics) { value = removeDiacritics(value); } - const isMatching = applier.fn(value, row, column, apiRef); + const isMatching = applier.fn(value, row, column, getPublicApiRef(apiRef)); if (isMatching) { result[filterValue] = true; continue outer; diff --git a/packages/grid/x-data-grid/src/internals/index.ts b/packages/grid/x-data-grid/src/internals/index.ts index 282be9d713862..8b5cd1a8c102f 100644 --- a/packages/grid/x-data-grid/src/internals/index.ts +++ b/packages/grid/x-data-grid/src/internals/index.ts @@ -141,6 +141,7 @@ export { isNavigationKey } from '../utils/keyboardUtils'; export { clamp, isDeepEqual, isNumber, isFunction, isObject } from '../utils/utils'; export { buildWarning } from '../utils/warning'; export { exportAs } from '../utils/exportAs'; +export * from '../utils/getPublicApiRef'; export type { GridPrivateOnlyApiCommon } from '../models/api/gridApiCommon'; export { useGridPrivateApiContext } from '../hooks/utils/useGridPrivateApiContext'; export * from '../hooks/utils/useOnMount'; diff --git a/packages/grid/x-data-grid/src/utils/getPublicApiRef.ts b/packages/grid/x-data-grid/src/utils/getPublicApiRef.ts new file mode 100644 index 0000000000000..0f1a980ad93ba --- /dev/null +++ b/packages/grid/x-data-grid/src/utils/getPublicApiRef.ts @@ -0,0 +1,9 @@ +import type { GridPrivateApiCommunity } from '../models/api/gridApiCommunity'; + +export function getPublicApiRef( + apiRef: React.MutableRefObject, +) { + return { current: apiRef.current.getPublicApi() } as React.MutableRefObject< + ReturnType + >; +} From f41c87eb6d31eccf658cdfaae93cd221a4818269 Mon Sep 17 00:00:00 2001 From: Andrew Cherniavskyi Date: Mon, 6 Nov 2023 14:17:58 +0100 Subject: [PATCH 4/4] reduce the number of getPublicApiRef calls --- .../src/hooks/features/filter/gridFilterUtils.ts | 13 ++++++++----- 1 file changed, 8 insertions(+), 5 deletions(-) diff --git a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts index 6c2aff267d43a..e719d42760de2 100644 --- a/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts +++ b/packages/grid/x-data-grid/src/hooks/features/filter/gridFilterUtils.ts @@ -208,6 +208,8 @@ const getFilterCallbackFromItem = ( const hasUserFunctionLegacy = !isInternalFilter(filterOperator.getApplyFilterFn); const hasUserFunctionV7 = !isInternalFilter(filterOperator.getApplyFilterFnV7); + const publicApiRef = getPublicApiRef(apiRef); + if (filterOperator.getApplyFilterFnV7 && !(hasUserFunctionLegacy && !hasUserFunctionV7)) { const applyFilterOnRow = filterOperator.getApplyFilterFnV7(newFilterItem, column)!; if (typeof applyFilterOnRow !== 'function') { @@ -221,7 +223,7 @@ const getFilterCallbackFromItem = ( if (ignoreDiacritics) { value = removeDiacritics(value); } - return applyFilterOnRow(value, row, column, getPublicApiRef(apiRef)); + return applyFilterOnRow(value, row, column, publicApiRef); }, }; } @@ -236,7 +238,7 @@ const getFilterCallbackFromItem = ( item: newFilterItem, fn: (rowId: GridRowId) => { const params = apiRef.current.getCellParams(rowId, newFilterItem.field!); - GLOBAL_API_REF.current = apiRef; + GLOBAL_API_REF.current = publicApiRef; if (ignoreDiacritics) { params.value = removeDiacritics(params.value); } @@ -356,6 +358,7 @@ const buildAggregatedQuickFilterApplier = ( }[]; const { ignoreDiacritics } = apiRef.current.rootProps; + const publicApiRef = getPublicApiRef(apiRef); columnFields.forEach((field) => { const column = apiRef.current.getColumn(field); @@ -372,7 +375,7 @@ const buildAggregatedQuickFilterApplier = ( const value = ignoreDiacritics ? removeDiacritics(quickFilterValue) : quickFilterValue; return { v7: true, - fn: getApplyQuickFilterFnV7(value, column, getPublicApiRef(apiRef)), + fn: getApplyQuickFilterFnV7(value, column, publicApiRef), }; }), }); @@ -383,7 +386,7 @@ const buildAggregatedQuickFilterApplier = ( const value = ignoreDiacritics ? removeDiacritics(quickFilterValue) : quickFilterValue; return { v7: false, - fn: getApplyQuickFilterFn(value, column, getPublicApiRef(apiRef)), + fn: getApplyQuickFilterFn(value, column, publicApiRef), }; }), }); @@ -417,7 +420,7 @@ const buildAggregatedQuickFilterApplier = ( if (ignoreDiacritics) { value = removeDiacritics(value); } - const isMatching = applier.fn(value, row, column, getPublicApiRef(apiRef)); + const isMatching = applier.fn(value, row, column, publicApiRef); if (isMatching) { result[filterValue] = true; continue outer;