igniteui-angular
Version:
Ignite UI for Angular is a dependency-free Angular toolkit for building modern web apps
354 lines • 39.8 kB
JavaScript
/**
* @fileoverview added by tsickle
* @suppress {checkTypes,extraRequire,missingReturn,unusedPrivateMembers,uselessCode} checked by tsc
*/
import { IgxSorting, IgxDataRecordSorting } from './sorting-strategy';
import { IgxGrouping } from './grouping-strategy';
import { PagingError } from './paging-state.interface';
import { TreeGridFilteringStrategy } from '../grids/tree-grid/tree-grid.filtering.pipe';
import { FilteringStrategy } from './filtering-strategy';
import { cloneValue, mergeObjects } from '../core/utils';
import { TransactionType } from '../services/transaction/transaction';
/** @enum {string} */
const DataType = {
String: 'string',
Number: 'number',
Boolean: 'boolean',
Date: 'date',
};
export { DataType };
/**
* @hidden
*/
export class DataUtil {
/**
* @template T
* @param {?} data
* @param {?} expressions
* @param {?=} sorting
* @return {?}
*/
static sort(data, expressions, sorting = new IgxSorting()) {
return sorting.sort(data, expressions);
}
/**
* @param {?} hierarchicalData
* @param {?} expressions
* @param {?=} parent
* @return {?}
*/
static treeGridSort(hierarchicalData, expressions, parent) {
/** @type {?} */
let res = [];
hierarchicalData.forEach((hr) => {
/** @type {?} */
const rec = DataUtil.cloneTreeGridRecord(hr);
rec.parent = parent;
if (rec.children) {
rec.children = DataUtil.treeGridSort(rec.children, expressions, rec);
}
res.push(rec);
});
res = DataUtil.sort(res, expressions, new IgxDataRecordSorting());
return res;
}
/**
* @param {?} hierarchicalRecord
* @return {?}
*/
static cloneTreeGridRecord(hierarchicalRecord) {
/** @type {?} */
const rec = {
rowID: hierarchicalRecord.rowID,
data: hierarchicalRecord.data,
children: hierarchicalRecord.children,
isFilteredOutParent: hierarchicalRecord.isFilteredOutParent,
level: hierarchicalRecord.level,
expanded: hierarchicalRecord.expanded
};
return rec;
}
/**
* @template T
* @param {?} data
* @param {?} state
* @param {?=} groupsRecords
* @return {?}
*/
static group(data, state, groupsRecords = []) {
/** @type {?} */
const grouping = new IgxGrouping();
groupsRecords.splice(0, groupsRecords.length);
return grouping.groupBy(data, state.expressions, groupsRecords);
}
/**
* @param {?} groupData
* @param {?} state
* @return {?}
*/
static restoreGroups(groupData, state) {
if (state.expressions.length === 0) {
return groupData.data;
}
return this.restoreGroupsIterative(groupData, state);
}
/**
* @private
* @param {?} groupData
* @param {?} state
* @return {?}
*/
static restoreGroupsIterative(groupData, state) {
/** @type {?} */
const metadata = groupData.metadata;
/** @type {?} */
const result = [];
/** @type {?} */
const added = [];
/** @type {?} */
let chain;
/** @type {?} */
let i = 0;
/** @type {?} */
let j;
/** @type {?} */
let pointer;
/** @type {?} */
let expanded;
for (i = 0; i < metadata.length;) {
chain = [metadata[i]];
pointer = metadata[i].groupParent;
// break off if the parent is already added
while (pointer && added[0] !== pointer) {
chain.push(pointer);
if (added[0] && added[0].level === pointer.level) {
added.shift();
}
pointer = pointer.groupParent;
}
for (j = chain.length - 1; j >= 0; j--) {
result.push(chain[j]);
added.unshift(chain[j]);
/** @type {?} */
const hierarchy = this.getHierarchy(chain[j]);
/** @type {?} */
const expandState = state.expansion.find((s) => this.isHierarchyMatch(s.hierarchy || [{ fieldName: chain[j].expression.fieldName, value: chain[j].value }], hierarchy));
expanded = expandState ? expandState.expanded : state.defaultExpanded;
if (!expanded) {
break;
}
}
added.shift();
j = Math.max(j, 0);
/** @type {?} */
const start = chain[j].records.findIndex(r => r === groupData.data[i]);
/** @type {?} */
const end = Math.min(metadata.length - i + start, chain[j].records.length);
if (expanded) {
result.push(...chain[j].records.slice(start, end));
}
i += end - start;
}
return result;
}
/**
* @template T
* @param {?} data
* @param {?} state
* @return {?}
*/
static page(data, state) {
if (!state) {
return data;
}
/** @type {?} */
const len = data.length;
/** @type {?} */
const index = state.index;
/** @type {?} */
const res = [];
/** @type {?} */
const recordsPerPage = state.recordsPerPage;
state.metadata = {
countPages: 0,
countRecords: data.length,
error: PagingError.None
};
if (index < 0 || isNaN(index)) {
state.metadata.error = PagingError.IncorrectPageIndex;
return res;
}
if (recordsPerPage <= 0 || isNaN(recordsPerPage)) {
state.metadata.error = PagingError.IncorrectRecordsPerPage;
return res;
}
state.metadata.countPages = Math.ceil(len / recordsPerPage);
if (!len) {
return data;
}
if (index >= state.metadata.countPages) {
state.metadata.error = PagingError.IncorrectPageIndex;
return res;
}
return data.slice(index * recordsPerPage, (index + 1) * recordsPerPage);
}
/**
* @template T
* @param {?} data
* @param {?} state
* @return {?}
*/
static filter(data, state) {
if (!state.strategy) {
state.strategy = new FilteringStrategy();
}
return state.strategy.filter(data, state.expressionsTree);
}
/**
* @param {?} data
* @param {?} state
* @return {?}
*/
static treeGridFilter(data, state) {
if (!state.strategy) {
state.strategy = new TreeGridFilteringStrategy();
}
return state.strategy.filter(data, state.expressionsTree);
}
/**
* @param {?} gRow
* @return {?}
*/
static getHierarchy(gRow) {
/** @type {?} */
const hierarchy = [];
if (gRow !== undefined && gRow.expression) {
hierarchy.push({ fieldName: gRow.expression.fieldName, value: gRow.value });
while (gRow.groupParent) {
gRow = gRow.groupParent;
hierarchy.unshift({ fieldName: gRow.expression.fieldName, value: gRow.value });
}
}
return hierarchy;
}
/**
* @param {?} h1
* @param {?} h2
* @return {?}
*/
static isHierarchyMatch(h1, h2) {
if (h1.length !== h2.length) {
return false;
}
return h1.every((level, index) => {
return level.fieldName === h2[index].fieldName && level.value === h2[index].value;
});
}
/**
* Merges all changes from provided transactions into provided data collection
* @template T
* @param {?} data Collection to merge
* @param {?} transactions Transactions to merge into data
* @param {?=} primaryKey Primary key of the collection, if any
* @param {?=} deleteRows Should delete rows with DELETE transaction type from data
* @return {?} Provided data collections updated with all provided transactions
*/
static mergeTransactions(data, transactions, primaryKey, deleteRows = false) {
data.forEach((item, index) => {
/** @type {?} */
const rowId = primaryKey ? item[primaryKey] : item;
/** @type {?} */
const transaction = transactions.find(t => t.id === rowId);
if (transaction && transaction.type === TransactionType.UPDATE) {
data[index] = transaction.newValue;
}
});
if (deleteRows) {
transactions
.filter(t => t.type === TransactionType.DELETE)
.forEach(t => {
/** @type {?} */
const index = primaryKey ? data.findIndex(d => d[primaryKey] === t.id) : data.findIndex(d => d === t.id);
if (0 <= index && index < data.length) {
data.splice(index, 1);
}
});
}
data.push(...transactions
.filter(t => t.type === TransactionType.ADD)
.map(t => t.newValue));
return data;
}
/**
* Merges all changes from provided transactions into provided hierarchical data collection
* @param {?} data Collection to merge
* @param {?} transactions Transactions to merge into data
* @param {?} childDataKey Data key of child collections
* @param {?=} primaryKey Primary key of the collection, if any
* @param {?=} deleteRows Should delete rows with DELETE transaction type from data
* @return {?} Provided data collections updated with all provided transactions
*/
static mergeHierarchicalTransactions(data, transactions, childDataKey, primaryKey, deleteRows = false) {
for (const transaction of transactions) {
if (transaction.path) {
/** @type {?} */
const parent = this.findParentFromPath(data, primaryKey, childDataKey, transaction.path);
/** @type {?} */
let collection = parent ? parent[childDataKey] : data;
switch (transaction.type) {
case TransactionType.ADD:
// if there is no parent this is ADD row at root level
if (parent && !parent[childDataKey]) {
parent[childDataKey] = collection = [];
}
collection.push(transaction.newValue);
break;
case TransactionType.UPDATE:
/** @type {?} */
const updateIndex = collection.findIndex(x => x[primaryKey] === transaction.id);
if (updateIndex !== -1) {
collection[updateIndex] = mergeObjects(cloneValue(collection[updateIndex]), transaction.newValue);
}
break;
case TransactionType.DELETE:
if (deleteRows) {
/** @type {?} */
const deleteIndex = collection.findIndex(r => r[primaryKey] === transaction.id);
if (deleteIndex !== -1) {
collection.splice(deleteIndex, 1);
}
}
break;
}
}
else {
// if there is no path this is ADD row in root. Push the newValue to data
data.push(transaction.newValue);
}
}
return data;
}
/**
* @private
* @param {?} data
* @param {?} primaryKey
* @param {?} childDataKey
* @param {?} path
* @return {?}
*/
static findParentFromPath(data, primaryKey, childDataKey, path) {
/** @type {?} */
let collection = data;
/** @type {?} */
let result;
for (const id of path) {
result = collection && collection.find(x => x[primaryKey] === id);
if (!result) {
break;
}
collection = result[childDataKey];
}
return result;
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-util.js","sourceRoot":"ng://igniteui-angular/","sources":["lib/data-operations/data-util.ts"],"names":[],"mappings":"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","sourcesContent":["import { IFilteringState } from './filtering-state.interface';\n\nimport { IgxSorting, IgxDataRecordSorting } from './sorting-strategy';\nimport { IGroupByResult, IgxGrouping } from './grouping-strategy';\n\nimport { IPagingState, PagingError } from './paging-state.interface';\n\nimport { IGroupByExpandState, IGroupByKey } from './groupby-expand-state.interface';\nimport { IGroupByRecord } from './groupby-record.interface';\nimport { IGroupingState } from './groupby-state.interface';\nimport { TreeGridFilteringStrategy } from '../grids/tree-grid/tree-grid.filtering.pipe';\nimport { ISortingExpression } from './sorting-expression.interface';\nimport { FilteringStrategy } from './filtering-strategy';\nimport { ITreeGridRecord } from '../grids/tree-grid';\nimport { cloneValue, mergeObjects } from '../core/utils';\nimport { Transaction, TransactionType, HierarchicalTransaction } from '../services/transaction/transaction';\n\n/**\n * @hidden\n */\nexport enum DataType {\n    String = 'string',\n    Number = 'number',\n    Boolean = 'boolean',\n    Date = 'date'\n}\n\n/**\n * @hidden\n */\nexport class DataUtil {\n    public static sort<T>(data: T[], expressions: ISortingExpression[], sorting: IgxSorting = new IgxSorting()): T[] {\n        return sorting.sort(data, expressions);\n    }\n\n    public static treeGridSort(hierarchicalData: ITreeGridRecord[],\n        expressions: ISortingExpression[],\n        parent?: ITreeGridRecord): ITreeGridRecord[] {\n        let res: ITreeGridRecord[] = [];\n        hierarchicalData.forEach((hr: ITreeGridRecord) => {\n            const rec: ITreeGridRecord = DataUtil.cloneTreeGridRecord(hr);\n            rec.parent = parent;\n            if (rec.children) {\n                rec.children = DataUtil.treeGridSort(rec.children, expressions, rec);\n            }\n            res.push(rec);\n        });\n\n        res = DataUtil.sort(res, expressions, new IgxDataRecordSorting());\n\n        return res;\n    }\n\n    public static cloneTreeGridRecord(hierarchicalRecord: ITreeGridRecord) {\n        const rec: ITreeGridRecord = {\n            rowID: hierarchicalRecord.rowID,\n            data: hierarchicalRecord.data,\n            children: hierarchicalRecord.children,\n            isFilteredOutParent: hierarchicalRecord.isFilteredOutParent,\n            level: hierarchicalRecord.level,\n            expanded: hierarchicalRecord.expanded\n        };\n        return rec;\n    }\n\n    public static group<T>(data: T[], state: IGroupingState, groupsRecords: any[] = []): IGroupByResult {\n        const grouping = new IgxGrouping();\n        groupsRecords.splice(0, groupsRecords.length);\n        return grouping.groupBy(data, state.expressions, groupsRecords);\n    }\n    public static restoreGroups(groupData: IGroupByResult, state: IGroupingState): any[] {\n        if (state.expressions.length === 0) {\n            return groupData.data;\n        }\n        return this.restoreGroupsIterative(groupData, state);\n    }\n    private static restoreGroupsIterative(groupData: IGroupByResult, state: IGroupingState): any[] {\n        const metadata = groupData.metadata;\n        const result = [], added = [];\n        let chain: any[];\n        let i = 0, j;\n        let pointer: IGroupByRecord;\n        let expanded: boolean;\n        for (i = 0; i < metadata.length;) {\n            chain = [metadata[i]];\n            pointer = metadata[i].groupParent;\n            // break off if the parent is already added\n            while (pointer && added[0] !== pointer) {\n                chain.push(pointer);\n                if (added[0] && added[0].level === pointer.level) {\n                    added.shift();\n                }\n                pointer = pointer.groupParent;\n            }\n            for (j = chain.length - 1; j >= 0; j--) {\n                result.push(chain[j]);\n                added.unshift(chain[j]);\n                const hierarchy = this.getHierarchy(chain[j]);\n                const expandState: IGroupByExpandState = state.expansion.find((s) =>\n                    this.isHierarchyMatch(s.hierarchy || [{ fieldName: chain[j].expression.fieldName, value: chain[j].value }], hierarchy));\n                expanded = expandState ? expandState.expanded : state.defaultExpanded;\n                if (!expanded) {\n                    break;\n                }\n            }\n            added.shift();\n            j = Math.max(j, 0);\n            const start = chain[j].records.findIndex(r => r === groupData.data[i]);\n            const end = Math.min(metadata.length - i + start, chain[j].records.length);\n            if (expanded) {\n                result.push(...chain[j].records.slice(start, end));\n            }\n            i += end - start;\n        }\n        return result;\n    }\n    public static page<T>(data: T[], state: IPagingState): T[] {\n        if (!state) {\n            return data;\n        }\n        const len = data.length;\n        const index = state.index;\n        const res = [];\n        const recordsPerPage = state.recordsPerPage;\n        state.metadata = {\n            countPages: 0,\n            countRecords: data.length,\n            error: PagingError.None\n        };\n        if (index < 0 || isNaN(index)) {\n            state.metadata.error = PagingError.IncorrectPageIndex;\n            return res;\n        }\n        if (recordsPerPage <= 0 || isNaN(recordsPerPage)) {\n            state.metadata.error = PagingError.IncorrectRecordsPerPage;\n            return res;\n        }\n        state.metadata.countPages = Math.ceil(len / recordsPerPage);\n        if (!len) {\n            return data;\n        }\n        if (index >= state.metadata.countPages) {\n            state.metadata.error = PagingError.IncorrectPageIndex;\n            return res;\n        }\n        return data.slice(index * recordsPerPage, (index + 1) * recordsPerPage);\n    }\n    public static filter<T>(data: T[], state: IFilteringState): T[] {\n        if (!state.strategy) {\n            state.strategy = new FilteringStrategy();\n        }\n        return state.strategy.filter(data, state.expressionsTree);\n    }\n\n    public static treeGridFilter(data: ITreeGridRecord[], state: IFilteringState): ITreeGridRecord[] {\n        if (!state.strategy) {\n            state.strategy = new TreeGridFilteringStrategy();\n        }\n        return state.strategy.filter(data, state.expressionsTree);\n    }\n\n    public static getHierarchy(gRow: IGroupByRecord): Array<IGroupByKey> {\n        const hierarchy: Array<IGroupByKey> = [];\n        if (gRow !== undefined && gRow.expression) {\n            hierarchy.push({ fieldName: gRow.expression.fieldName, value: gRow.value });\n            while (gRow.groupParent) {\n                gRow = gRow.groupParent;\n                hierarchy.unshift({ fieldName: gRow.expression.fieldName, value: gRow.value });\n            }\n        }\n        return hierarchy;\n    }\n\n    public static isHierarchyMatch(h1: Array<IGroupByKey>, h2: Array<IGroupByKey>): boolean {\n        if (h1.length !== h2.length) {\n            return false;\n        }\n        return h1.every((level, index): boolean => {\n            return level.fieldName === h2[index].fieldName && level.value === h2[index].value;\n        });\n    }\n\n    /**\n     * Merges all changes from provided transactions into provided data collection\n     * @param data Collection to merge\n     * @param transactions Transactions to merge into data\n     * @param primaryKey Primary key of the collection, if any\n     * @param deleteRows Should delete rows with DELETE transaction type from data\n     * @returns Provided data collections updated with all provided transactions\n     */\n    public static mergeTransactions<T>(data: T[], transactions: Transaction[], primaryKey?: any, deleteRows: boolean = false): T[] {\n        data.forEach((item: any, index: number) => {\n            const rowId = primaryKey ? item[primaryKey] : item;\n            const transaction = transactions.find(t => t.id === rowId);\n            if (transaction && transaction.type === TransactionType.UPDATE) {\n                data[index] = transaction.newValue;\n            }\n        });\n\n        if (deleteRows) {\n            transactions\n                .filter(t => t.type === TransactionType.DELETE)\n                .forEach(t => {\n                    const index = primaryKey ? data.findIndex(d => d[primaryKey] === t.id) : data.findIndex(d => d === t.id);\n                    if (0 <= index && index < data.length) {\n                        data.splice(index, 1);\n                    }\n                });\n        }\n\n        data.push(...transactions\n            .filter(t => t.type === TransactionType.ADD)\n            .map(t => t.newValue));\n\n        return data;\n    }\n\n    /**\n     * Merges all changes from provided transactions into provided hierarchical data collection\n     * @param data Collection to merge\n     * @param transactions Transactions to merge into data\n     * @param childDataKey Data key of child collections\n     * @param primaryKey Primary key of the collection, if any\n     * @param deleteRows Should delete rows with DELETE transaction type from data\n     * @returns Provided data collections updated with all provided transactions\n     */\n    public static mergeHierarchicalTransactions(\n        data: any[],\n        transactions: HierarchicalTransaction[],\n        childDataKey: any,\n        primaryKey?: any,\n        deleteRows: boolean = false): any[] {\n\n        for (const transaction of transactions) {\n            if (transaction.path) {\n                const parent = this.findParentFromPath(data, primaryKey, childDataKey, transaction.path);\n                let collection: any[] = parent ? parent[childDataKey] : data;\n                switch (transaction.type) {\n                    case TransactionType.ADD:\n                        //  if there is no parent this is ADD row at root level\n                        if (parent && !parent[childDataKey]) {\n                            parent[childDataKey] = collection = [];\n                        }\n                        collection.push(transaction.newValue);\n                        break;\n                    case TransactionType.UPDATE:\n                        const updateIndex = collection.findIndex(x => x[primaryKey] === transaction.id);\n                        if (updateIndex !== -1) {\n                            collection[updateIndex] = mergeObjects(cloneValue(collection[updateIndex]), transaction.newValue);\n                        }\n                        break;\n                    case TransactionType.DELETE:\n                        if (deleteRows) {\n                            const deleteIndex = collection.findIndex(r => r[primaryKey] === transaction.id);\n                            if (deleteIndex !== -1) {\n                                collection.splice(deleteIndex, 1);\n                            }\n                        }\n                        break;\n                }\n            } else {\n                //  if there is no path this is ADD row in root. Push the newValue to data\n                data.push(transaction.newValue);\n            }\n        }\n        return data;\n    }\n\n    private static findParentFromPath(data: any[], primaryKey: any, childDataKey: any, path: any[]): any {\n        let collection: any[] = data;\n        let result: any;\n\n        for (const id of path) {\n            result = collection && collection.find(x => x[primaryKey] === id);\n            if (!result) {\n                break;\n            }\n\n            collection = result[childDataKey];\n        }\n\n        return result;\n    }\n}\n"]}