@mui/x-data-grid-premium
Version:
The Premium plan edition of the MUI X Data Grid Components.
177 lines (175 loc) • 6.39 kB
JavaScript
import _extends from "@babel/runtime/helpers/esm/extends";
import { passFilterLogic, GRID_ROW_GROUPING_SINGLE_GROUPING_FIELD, getRowGroupingCriteriaFromGroupingField, isGroupingColumn, GridStrategyGroup, getRowValue, RowGroupingStrategy } from '@mui/x-data-grid-pro/internals';
import { gridRowGroupingSanitizedModelSelector } from "./gridRowGroupingSelector.js";
export { GRID_ROW_GROUPING_SINGLE_GROUPING_FIELD, getRowGroupingCriteriaFromGroupingField, isGroupingColumn };
export const getRowGroupingFieldFromGroupingCriteria = groupingCriteria => {
if (groupingCriteria === null) {
return GRID_ROW_GROUPING_SINGLE_GROUPING_FIELD;
}
return `__row_group_by_columns_group_${groupingCriteria}__`;
};
/**
* When filtering a group, we only want to filter according to the items related to this grouping column.
*/
const shouldApplyFilterItemOnGroup = (columnField, node) => {
if (columnField === GRID_ROW_GROUPING_SINGLE_GROUPING_FIELD) {
return true;
}
const groupingCriteriaField = getRowGroupingCriteriaFromGroupingField(columnField);
return groupingCriteriaField === node.groupingField;
};
/**
* A leaf is visible if it passed the filter
* A group is visible if all the following criteria are met:
* - One of its children is passing the filter
* - It is passing the filter
*/
export const filterRowTreeFromGroupingColumns = params => {
const {
apiRef,
rowTree,
isRowMatchingFilters,
filterModel,
filterValueGetter
} = params;
const filteredRowsLookup = {};
const filteredChildrenCountLookup = {};
const filteredDescendantCountLookup = {};
const filterCache = {};
const filterTreeNode = (node, areAncestorsExpanded, ancestorsResults) => {
const filterResults = {
passingFilterItems: null,
passingQuickFilterValues: null
};
let isPassingFiltering = false;
if (isRowMatchingFilters && node.type !== 'footer') {
const shouldApplyItem = node.type === 'group' && node.isAutoGenerated ? columnField => shouldApplyFilterItemOnGroup(columnField, node) : undefined;
const row = apiRef.current.getRow(node.id);
isRowMatchingFilters(row, shouldApplyItem, filterResults);
} else {
isPassingFiltering = true;
}
let filteredChildrenCount = 0;
let filteredDescendantCount = 0;
if (node.type === 'group') {
node.children.forEach(childId => {
const childNode = rowTree[childId];
const childSubTreeSize = filterTreeNode(childNode, areAncestorsExpanded && !!node.childrenExpanded, [...ancestorsResults, filterResults]);
filteredDescendantCount += childSubTreeSize;
if (childSubTreeSize > 0) {
filteredChildrenCount += 1;
}
});
}
if (isPassingFiltering === false) {
if (node.type === 'group') {
// If node has children - it's passing if at least one child passes filters
isPassingFiltering = filteredDescendantCount > 0;
} else {
const allResults = [...ancestorsResults, filterResults];
isPassingFiltering = passFilterLogic(allResults.map(result => result.passingFilterItems), allResults.map(result => result.passingQuickFilterValues), filterModel, filterValueGetter, params.apiRef, filterCache);
}
}
if (!isPassingFiltering) {
filteredRowsLookup[node.id] = false;
}
if (!isPassingFiltering) {
return 0;
}
filteredChildrenCountLookup[node.id] = filteredChildrenCount;
filteredDescendantCountLookup[node.id] = filteredDescendantCount;
if (node.type !== 'group') {
return filteredDescendantCount + 1;
}
return filteredDescendantCount;
};
const nodes = Object.values(rowTree);
for (let i = 0; i < nodes.length; i += 1) {
const node = nodes[i];
if (node.depth === 0) {
filterTreeNode(node, true, []);
}
}
return {
filteredRowsLookup,
filteredChildrenCountLookup,
filteredDescendantCountLookup
};
};
export const getColDefOverrides = (groupingColDefProp, fields, strategy) => {
if (typeof groupingColDefProp === 'function') {
return groupingColDefProp({
groupingName: strategy ?? RowGroupingStrategy.Default,
fields
});
}
return groupingColDefProp;
};
export const mergeStateWithRowGroupingModel = rowGroupingModel => state => _extends({}, state, {
rowGrouping: _extends({}, state.rowGrouping, {
model: rowGroupingModel
})
});
export const setStrategyAvailability = (privateApiRef, disableRowGrouping, dataSource) => {
const strategy = dataSource ? RowGroupingStrategy.DataSource : RowGroupingStrategy.Default;
if (privateApiRef.current.getActiveStrategy(GridStrategyGroup.RowTree) === strategy) {
// If the strategy is already active, we don't need to set it again
return;
}
let isAvailable;
if (disableRowGrouping) {
isAvailable = () => false;
} else {
isAvailable = () => {
const rowGroupingSanitizedModel = gridRowGroupingSanitizedModelSelector(privateApiRef);
return rowGroupingSanitizedModel.length > 0;
};
}
privateApiRef.current.setStrategyAvailability(GridStrategyGroup.RowTree, strategy, isAvailable);
};
export const getCellGroupingCriteria = ({
row,
colDef,
groupingRule,
apiRef
}) => {
let key;
if (groupingRule.groupingValueGetter) {
key = groupingRule.groupingValueGetter(row[groupingRule.field], row, colDef, apiRef);
} else {
key = getRowValue(row, colDef, apiRef);
}
return {
key,
field: groupingRule.field
};
};
export const getGroupingRules = ({
sanitizedRowGroupingModel,
columnsLookup
}) => sanitizedRowGroupingModel.map(field => ({
field,
groupingValueGetter: columnsLookup[field]?.groupingValueGetter,
groupingValueSetter: columnsLookup[field]?.groupingValueSetter
}));
/**
* Compares two sets of grouping rules to determine if they are equal or not.
*/
export const areGroupingRulesEqual = (newValue, previousValue) => {
if (previousValue.length !== newValue.length) {
return false;
}
return newValue.every((newRule, newRuleIndex) => {
const previousRule = previousValue[newRuleIndex];
if (previousRule.groupingValueGetter !== newRule.groupingValueGetter) {
return false;
}
if (previousRule.groupingValueSetter !== newRule.groupingValueSetter) {
return false;
}
if (previousRule.field !== newRule.field) {
return false;
}
return true;
});
};