ag-grid-enterprise
Version:
ag-Grid Enterprise Features
519 lines (432 loc) • 21.4 kB
text/typescript
import {
_,
Autowired,
Bean,
ChangedPath,
Column,
ColumnController,
Context,
EventService,
GetDataPath,
GridOptionsWrapper,
IRowNodeStage,
NumberSequence,
PostConstruct,
RowNode,
RowNodeTransaction,
SelectableService,
SelectionController,
StageExecuteParams,
ValueService
} from "ag-grid-community";
interface GroupInfo {
key: string; // e.g. 'Ireland'
field: string | null; // e.g. 'country'
rowGroupColumn: Column | null;
}
interface GroupingDetails {
pivotMode: boolean;
includeParents: boolean;
expandByDefault: number;
changedPath: ChangedPath;
rootNode: RowNode;
groupedCols: Column[];
groupedColCount: number;
transaction: RowNodeTransaction;
rowNodeOrder: { [id: string]: number };
}
export class GroupStage implements IRowNodeStage {
private selectionController: SelectionController;
private gridOptionsWrapper: GridOptionsWrapper;
private columnController: ColumnController;
private selectableService: SelectableService;
private valueService: ValueService;
private eventService: EventService;
private context: Context;
// if doing tree data, this is true. we set this at create time - as our code does not
// cater for the scenario where this is switched on / off dynamically
private usingTreeData: boolean;
private getDataPath: GetDataPath | undefined;
// we use a sequence variable so that each time we do a grouping, we don't
// reuse the ids - otherwise the rowRenderer will confuse rowNodes between redraws
// when it tries to animate between rows. we set to -1 as others row id 0 will be shared
// with the other rows.
private groupIdSequence = new NumberSequence(1);
// when grouping, these items are of note:
// rowNode.parent: RowNode: set to the parent
// rowNode.childrenAfterGroup: RowNode[] = the direct children of this group
// rowNode.childrenMapped: string=>RowNode = children mapped by group key (when groups) or an empty map if leaf group (this is then used by pivot)
// for leaf groups, rowNode.childrenAfterGroup = rowNode.allLeafChildren;
private postConstruct(): void {
this.usingTreeData = this.gridOptionsWrapper.isTreeData();
if (this.usingTreeData) {
this.getDataPath = this.gridOptionsWrapper.getDataPathFunc();
}
}
public execute(params: StageExecuteParams): void {
let details = this.createGroupingDetails(params);
if (details.transaction) {
this.handleTransaction(details);
} else {
this.shotgunResetEverything(details);
}
this.sortGroupsWithComparator(details.rootNode);
this.selectableService.updateSelectableAfterGrouping(details.rootNode);
}
private createGroupingDetails(params: StageExecuteParams): GroupingDetails {
let {rowNode, changedPath, rowNodeTransaction, rowNodeOrder} = params;
let groupedCols = this.usingTreeData ? null : this.columnController.getRowGroupColumns();
let isGrouping = this.usingTreeData || (groupedCols && groupedCols.length > 0);
let usingTransaction = isGrouping && _.exists(rowNodeTransaction);
let details = <GroupingDetails> {
// someone complained that the parent attribute was causing some change detection
// to break is some angular add-on - which i never used. taking the parent out breaks
// a cyclic dependency, hence this flag got introduced.
includeParents: !this.gridOptionsWrapper.isSuppressParentsInRowNodes(),
expandByDefault: this.gridOptionsWrapper.isGroupSuppressRow() ?
-1 : this.gridOptionsWrapper.getGroupDefaultExpanded(),
groupedCols: groupedCols,
rootNode: rowNode,
pivotMode: this.columnController.isPivotMode(),
groupedColCount: this.usingTreeData || !groupedCols ? 0 : groupedCols.length,
rowNodeOrder: rowNodeOrder,
// important not to do transaction if we are not grouping, as otherwise the 'insert index' is ignored.
// ie, if not grouping, then we just want to shotgun so the rootNode.allLeafChildren gets copied
// to rootNode.childrenAfterGroup and maintaining order (as delta transaction misses the order).
transaction: usingTransaction ? rowNodeTransaction : null,
// if no transaction, then it's shotgun, changed path would be 'not active' at this point anyway
changedPath: changedPath
};
return details;
}
private handleTransaction(details: GroupingDetails): void {
let tran = details.transaction;
if (tran.add) {
this.insertNodes(tran.add, details);
}
if (tran.update) {
this.moveNodesInWrongPath(tran.update, details);
}
if (tran.remove) {
this.removeNodes(tran.remove, details);
}
if (details.rowNodeOrder) {
this.recursiveSortChildren(details.rootNode, details);
}
}
// this is used when doing delta updates, eg Redux, keeps nodes in right order
private recursiveSortChildren(node: RowNode, details: GroupingDetails): void {
_.sortRowNodesByOrder(node.childrenAfterGroup, details.rowNodeOrder);
node.childrenAfterGroup.forEach(childNode => {
if (childNode.childrenAfterGroup) {
this.recursiveSortChildren(childNode, details);
}
});
}
private sortGroupsWithComparator(rootNode: RowNode): void {
// we don't do group sorting for tree data
if (this.usingTreeData) {
return;
}
let comparator = this.gridOptionsWrapper.getDefaultGroupSortComparator();
if (_.exists(comparator)) {
recursiveSort(rootNode);
}
function recursiveSort(rowNode: RowNode): void {
let doSort = _.exists(rowNode.childrenAfterGroup) &&
// we only want to sort groups, so we do not sort leafs (a leaf group has leafs as children)
!rowNode.leafGroup;
if (doSort) {
rowNode.childrenAfterGroup.sort(comparator);
rowNode.childrenAfterGroup.forEach(childNode => recursiveSort(childNode));
}
}
}
private getExistingPathForNode(node: RowNode, details: GroupingDetails): GroupInfo[] {
let res: GroupInfo[] = [];
// when doing tree data, the node is part of the path,
// but when doing grid grouping, the node is not part of the path so we start with the parent.
let pointer = this.usingTreeData ? node : node.parent;
while (pointer && pointer !== details.rootNode) {
res.push({
key: pointer.key,
rowGroupColumn: pointer.rowGroupColumn,
field: pointer.field
});
pointer = pointer.parent;
}
res.reverse();
return res;
}
private moveNodesInWrongPath(childNodes: RowNode[], details: GroupingDetails): void {
childNodes.forEach(childNode => {
// we add node, even if parent has not changed, as the data could have
// changed, hence aggregations will be wrong
if (details.changedPath.isActive()) {
details.changedPath.addParentNode(childNode.parent);
}
let infoToKeyMapper = (item: GroupInfo) => item.key;
let oldPath: string[] = this.getExistingPathForNode(childNode, details).map(infoToKeyMapper);
let newPath: string[] = this.getGroupInfo(childNode, details).map(infoToKeyMapper);
let nodeInCorrectPath = _.compareArrays(oldPath, newPath);
if (!nodeInCorrectPath) {
this.moveNode(childNode, details);
}
});
}
private moveNode(childNode: RowNode, details: GroupingDetails): void {
this.removeOneNode(childNode, details);
this.insertOneNode(childNode, details);
// hack - if we didn't do this, then renaming a tree item (ie changing rowNode.key) wouldn't get
// refreshed into the gui.
// this is needed to kick off the event that rowComp listens to for refresh. this in turn
// then will get each cell in the row to refresh - which is what we need as we don't know which
// columns will be displaying the rowNode.key info.
childNode.setData(childNode.data);
// we add both old and new parents to changed path, as both will need to be refreshed.
// we already added the old parent (in calling method), so just add the new parent here
if (details.changedPath.isActive()) {
let newParent = childNode.parent;
details.changedPath.addParentNode(newParent);
}
}
private removeNodes(leafRowNodes: RowNode[], details: GroupingDetails): void {
leafRowNodes.forEach(leafToRemove => {
this.removeOneNode(leafToRemove, details);
if (details.changedPath.isActive()) {
details.changedPath.addParentNode(leafToRemove.parent);
}
});
}
private removeOneNode(childNode: RowNode, details: GroupingDetails): void {
// utility func to execute once on each parent node
let forEachParentGroup = (callback: (parent: RowNode) => void) => {
let pointer = childNode.parent;
while (pointer && pointer !== details.rootNode) {
callback(pointer);
pointer = pointer.parent;
}
};
// remove leaf from direct parent
this.removeFromParent(childNode);
// remove from allLeafChildren
forEachParentGroup(parentNode => _.removeFromArray(parentNode.allLeafChildren, childNode));
// if not group, and children are present, need to move children to a group.
// otherwise if no children, we can just remove without replacing.
let replaceWithGroup = childNode.hasChildren();
if (replaceWithGroup) {
let oldPath = this.getExistingPathForNode(childNode, details);
// because we just removed the userGroup, this will always return new support group
let newGroupNode = this.findParentForNode(childNode, oldPath, details);
// these properties are the ones that will be incorrect in the newly created group,
// so copy them form the old childNode
newGroupNode.expanded = childNode.expanded;
newGroupNode.allLeafChildren = childNode.allLeafChildren;
newGroupNode.childrenAfterGroup = childNode.childrenAfterGroup;
newGroupNode.childrenMapped = childNode.childrenMapped;
newGroupNode.childrenAfterGroup.forEach(rowNode => rowNode.parent = newGroupNode);
}
// remove empty groups
forEachParentGroup(node => {
if (node.isEmptyFillerNode()) {
this.removeFromParent(node);
// we remove selection on filler nodes here, as the selection would not be removed
// from the RowNodeManager, as filler nodes don't exist on teh RowNodeManager
node.setSelected(false);
}
});
}
private removeFromParent(child: RowNode) {
if (child.parent) {
_.removeFromArray(child.parent.childrenAfterGroup, child);
}
let mapKey = this.getChildrenMappedKey(child.key, child.rowGroupColumn);
if (child.parent && child.parent.childrenMapped) {
child.parent.childrenMapped[mapKey] = undefined;
}
// this is important for transition, see rowComp removeFirstPassFuncs. when doing animation and
// remove, if rowTop is still present, the rowComp thinks it's just moved position.
child.setRowTop(null);
}
private addToParent(child: RowNode, parent: RowNode | null) {
let mapKey = this.getChildrenMappedKey(child.key, child.rowGroupColumn);
if (parent) {
if (parent.childrenMapped) {
parent.childrenMapped[mapKey] = child;
}
parent.childrenAfterGroup.push(child);
}
}
private shotgunResetEverything(details: GroupingDetails): void {
// because we are not creating the root node each time, we have the logic
// here to change leafGroup once.
// we set .leafGroup to false for tree data, as .leafGroup is only used when pivoting, and pivoting
// isn't allowed with treeData, so the grid never actually use .leafGroup when doing treeData.
details.rootNode.leafGroup = this.usingTreeData ? false : details.groupedCols.length === 0;
// we are going everything from scratch, so reset childrenAfterGroup and childrenMapped from the rootNode
details.rootNode.childrenAfterGroup = [];
details.rootNode.childrenMapped = {};
this.insertNodes(details.rootNode.allLeafChildren, details);
}
private insertNodes(newRowNodes: RowNode[], details: GroupingDetails): void {
newRowNodes.forEach(rowNode => {
this.insertOneNode(rowNode, details);
if (details.changedPath.isActive()) {
details.changedPath.addParentNode(rowNode.parent);
}
});
}
private insertOneNode(childNode: RowNode, details: GroupingDetails): void {
let path: GroupInfo[] = this.getGroupInfo(childNode, details);
let parentGroup = this.findParentForNode(childNode, path, details);
if (!parentGroup.group) {
console.warn(`ag-Grid: duplicate group keys for row data, keys should be unique`,
[parentGroup.data, childNode.data]);
}
if (this.usingTreeData) {
this.swapGroupWithUserNode(parentGroup, childNode);
} else {
childNode.parent = parentGroup;
childNode.level = path.length;
parentGroup.childrenAfterGroup.push(childNode);
}
}
private findParentForNode(childNode: RowNode, path: GroupInfo[], details: GroupingDetails): RowNode {
let nextNode: RowNode = details.rootNode;
path.forEach((groupInfo, level) => {
nextNode = this.getOrCreateNextNode(nextNode, groupInfo, level, details);
// node gets added to all group nodes.
// note: we do not add to rootNode here, as the rootNode is the master list of rowNodes
nextNode.allLeafChildren.push(childNode);
});
return nextNode;
}
private swapGroupWithUserNode(fillerGroup: RowNode, userGroup: RowNode) {
userGroup.parent = fillerGroup.parent;
userGroup.key = fillerGroup.key;
userGroup.field = fillerGroup.field;
userGroup.groupData = fillerGroup.groupData;
userGroup.level = fillerGroup.level;
userGroup.expanded = fillerGroup.expanded;
// we set .leafGroup to false for tree data, as .leafGroup is only used when pivoting, and pivoting
// isn't allowed with treeData, so the grid never actually use .leafGroup when doing treeData.
userGroup.leafGroup = fillerGroup.leafGroup;
// always null for userGroups, as row grouping is not allowed when doing tree data
userGroup.rowGroupIndex = fillerGroup.rowGroupIndex;
userGroup.allLeafChildren = fillerGroup.allLeafChildren;
userGroup.childrenAfterGroup = fillerGroup.childrenAfterGroup;
userGroup.childrenMapped = fillerGroup.childrenMapped;
this.removeFromParent(fillerGroup);
userGroup.childrenAfterGroup.forEach(rowNode => rowNode.parent = userGroup);
this.addToParent(userGroup, fillerGroup.parent);
}
private getOrCreateNextNode(parentGroup: RowNode, groupInfo: GroupInfo, level: number,
details: GroupingDetails): RowNode {
let mapKey = this.getChildrenMappedKey(groupInfo.key, groupInfo.rowGroupColumn);
let nextNode = parentGroup.childrenMapped ? <RowNode> parentGroup.childrenMapped[mapKey] : undefined;
if (!nextNode) {
nextNode = this.createGroup(groupInfo, parentGroup, level, details);
// attach the new group to the parent
this.addToParent(nextNode, parentGroup);
}
return nextNode;
}
private createGroup(groupInfo: GroupInfo, parent: RowNode, level: number, details: GroupingDetails): RowNode {
let groupNode = new RowNode();
this.context.wireBean(groupNode);
groupNode.group = true;
groupNode.field = groupInfo.field;
groupNode.rowGroupColumn = groupInfo.rowGroupColumn;
groupNode.groupData = {};
let groupDisplayCols: Column[] = this.columnController.getGroupDisplayColumns();
groupDisplayCols.forEach(col => {
// newGroup.rowGroupColumn=null when working off GroupInfo, and we always display the group in the group column
// if rowGroupColumn is present, then it's grid row grouping and we only include if configuration says so
let displayGroupForCol = this.usingTreeData || (groupNode.rowGroupColumn ? col.isRowGroupDisplayed(groupNode.rowGroupColumn.getId()) : false);
if (displayGroupForCol) {
groupNode.groupData[col.getColId()] = groupInfo.key;
}
});
// we use negative number for the ids of the groups, this makes sure we don't clash with the
// id's of the leaf nodes.
groupNode.id = (this.groupIdSequence.next() * -1).toString();
groupNode.key = groupInfo.key;
groupNode.level = level;
groupNode.leafGroup = this.usingTreeData ? false : level === (details.groupedColCount - 1);
// if doing pivoting, then the leaf group is never expanded,
// as we do not show leaf rows
if (details.pivotMode && groupNode.leafGroup) {
groupNode.expanded = false;
} else {
groupNode.expanded = this.isExpanded(details.expandByDefault, level);
}
groupNode.allLeafChildren = [];
// why is this done here? we are not updating the children could as we go,
// i suspect this is updated in the filter stage
groupNode.setAllChildrenCount(0);
groupNode.rowGroupIndex = this.usingTreeData ? null : level;
groupNode.childrenAfterGroup = [];
groupNode.childrenMapped = {};
groupNode.parent = details.includeParents ? parent : null;
return groupNode;
}
private getChildrenMappedKey(key: string, rowGroupColumn: Column | null): string {
if (rowGroupColumn) {
// grouping by columns
return rowGroupColumn.getId() + '-' + key;
} else {
// tree data - we don't have rowGroupColumns
return key;
}
}
private isExpanded(expandByDefault: number, level: number) {
if (expandByDefault === -1) {
return true;
} else {
return level < expandByDefault;
}
}
private getGroupInfo(rowNode: RowNode, details: GroupingDetails): GroupInfo[] {
if (this.usingTreeData) {
return this.getGroupInfoFromCallback(rowNode);
} else {
return this.getGroupInfoFromGroupColumns(rowNode, details);
}
}
private getGroupInfoFromCallback(rowNode: RowNode): GroupInfo[] {
let keys: string[] | null = this.getDataPath ? this.getDataPath(rowNode.data) : null;
if (keys === null || keys === undefined || keys.length === 0) {
_.doOnce(
() => console.warn(`getDataPath() should not return an empty path for data`, rowNode.data),
'groupStage.getGroupInfoFromCallback'
);
}
let groupInfoMapper = (key: string) => <GroupInfo> {key: key, field: null, rowGroupColumn: null};
return keys ? keys.map(groupInfoMapper) : [];
}
private getGroupInfoFromGroupColumns(rowNode: RowNode, details: GroupingDetails) {
let res: GroupInfo[] = [];
details.groupedCols.forEach(groupCol => {
let key: string = this.valueService.getKeyForNode(groupCol, rowNode);
let keyExists = key !== null && key !== undefined;
// unbalanced tree and pivot mode don't work together - not because of the grid, it doesn't make
// mathematical sense as you are building up a cube. so if pivot mode, we put in a blank key where missing.
// this keeps the tree balanced and hence can be represented as a group.
if (details.pivotMode && !keyExists) {
key = ' ';
keyExists = true;
}
if (keyExists) {
let item = <GroupInfo> {
key: key,
field: groupCol.getColDef().field,
rowGroupColumn: groupCol
};
res.push(item);
}
});
return res;
}
}