2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* TableData maintains a single table's data.
|
|
|
|
*/
|
2021-11-01 15:48:08 +00:00
|
|
|
import { ColumnACIndexes } from 'app/client/models/ColumnACIndexes';
|
|
|
|
import { ColumnCache } from 'app/client/models/ColumnCache';
|
|
|
|
import { DocData } from 'app/client/models/DocData';
|
|
|
|
import { DocAction, ReplaceTableData, TableDataAction, UserAction } from 'app/common/DocActions';
|
|
|
|
import { isRaisedException } from 'app/common/gristTypes';
|
|
|
|
import { countIf } from 'app/common/gutil';
|
|
|
|
import { TableData as BaseTableData, ColTypeMap } from 'app/common/TableData';
|
|
|
|
import { Emitter } from 'grainjs';
|
2020-10-02 15:10:00 +00:00
|
|
|
|
2021-11-01 15:48:08 +00:00
|
|
|
export type SearchFunc = (value: any) => boolean;
|
2020-10-02 15:10:00 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* TableData class to maintain a single table's data.
|
|
|
|
*/
|
|
|
|
export class TableData extends BaseTableData {
|
|
|
|
public readonly tableActionEmitter = new Emitter();
|
|
|
|
public readonly dataLoadedEmitter = new Emitter();
|
|
|
|
|
|
|
|
public readonly columnACIndexes = new ColumnACIndexes(this);
|
|
|
|
|
2021-04-20 21:57:45 +00:00
|
|
|
private _columnErrorCounts = new ColumnCache<number|undefined>(this);
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* Constructor for TableData.
|
|
|
|
* @param {DocData} docData: The root DocData object for this document.
|
|
|
|
* @param {String} tableId: The name of this table.
|
|
|
|
* @param {Object} tableData: An object equivalent to BulkAddRecord, i.e.
|
|
|
|
* ["TableData", tableId, rowIds, columnValues].
|
|
|
|
* @param {Object} columnTypes: A map of colId to colType.
|
|
|
|
*/
|
|
|
|
constructor(public readonly docData: DocData,
|
|
|
|
tableId: string, tableData: TableDataAction|null, columnTypes: ColTypeMap) {
|
|
|
|
super(tableId, tableData, columnTypes);
|
|
|
|
}
|
|
|
|
|
|
|
|
public loadData(tableData: TableDataAction|ReplaceTableData): number[] {
|
|
|
|
const oldRowIds = super.loadData(tableData);
|
|
|
|
// If called from base constructor, this.dataLoadedEmitter may be unset; in that case there
|
|
|
|
// are no subscribers anyway.
|
|
|
|
if (this.dataLoadedEmitter) {
|
|
|
|
this.dataLoadedEmitter.emit(oldRowIds, this.getRowIds());
|
|
|
|
}
|
|
|
|
return oldRowIds;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used by QuerySet to load new rows for onDemand tables.
|
|
|
|
public loadPartial(data: TableDataAction): void {
|
|
|
|
super.loadPartial(data);
|
|
|
|
// Emit dataLoaded event, to trigger ('rowChange', 'add') on the TableModel RowSource.
|
|
|
|
this.dataLoadedEmitter.emit([], data[2]);
|
|
|
|
}
|
|
|
|
|
|
|
|
// Used by QuerySet to remove unused rows for onDemand tables when a QuerySet is disposed.
|
|
|
|
public unloadPartial(rowIds: number[]): void {
|
|
|
|
super.unloadPartial(rowIds);
|
|
|
|
// Emit dataLoaded event, to trigger ('rowChange', 'rm') on the TableModel RowSource.
|
|
|
|
this.dataLoadedEmitter.emit(rowIds, []);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2021-11-01 15:48:08 +00:00
|
|
|
* Given a colId and a search function, returns a list of matching row IDs, optionally limiting their number.
|
2020-10-02 15:10:00 +00:00
|
|
|
* @param {String} colId: identifies the column to search.
|
2021-11-01 15:48:08 +00:00
|
|
|
* @param {Function} searchFunc: A function which, given a column value, returns whether to include it.
|
2020-10-02 15:10:00 +00:00
|
|
|
* @param [Number] optMaxResults: if given, limit the number of returned results to this.
|
2021-11-01 15:48:08 +00:00
|
|
|
* @returns Array[Number] array of row IDs.
|
2020-10-02 15:10:00 +00:00
|
|
|
*/
|
2021-11-01 15:48:08 +00:00
|
|
|
public columnSearch(colId: string, searchFunc: SearchFunc, optMaxResults?: number) {
|
2020-10-02 15:10:00 +00:00
|
|
|
const maxResults = optMaxResults || Number.POSITIVE_INFINITY;
|
|
|
|
|
|
|
|
const rowIds = this.getRowIds();
|
|
|
|
const valColumn = this.getColValues(colId);
|
|
|
|
const ret = [];
|
|
|
|
if (!valColumn) {
|
|
|
|
// tslint:disable-next-line:no-console
|
|
|
|
console.warn(`TableData.columnSearch called on invalid column ${this.tableId}.${colId}`);
|
|
|
|
} else {
|
|
|
|
for (let i = 0; i < rowIds.length && ret.length < maxResults; i++) {
|
2021-11-01 15:48:08 +00:00
|
|
|
const value = valColumn[i];
|
2020-10-02 15:10:00 +00:00
|
|
|
if (value && searchFunc(value)) {
|
2021-11-01 15:48:08 +00:00
|
|
|
ret.push(rowIds[i]);
|
2020-10-02 15:10:00 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return ret;
|
|
|
|
}
|
|
|
|
|
2021-04-20 21:57:45 +00:00
|
|
|
/**
|
|
|
|
* Counts and returns the number of error values in the given column. The count is cached to
|
|
|
|
* keep it faster for large tables, and the cache is cleared as needed on changes to the table.
|
|
|
|
*/
|
|
|
|
public countErrors(colId: string): number|undefined {
|
|
|
|
return this._columnErrorCounts.getValue(colId, () => {
|
|
|
|
const values = this.getColValues(colId);
|
|
|
|
return values && countIf(values, isRaisedException);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2020-10-02 15:10:00 +00:00
|
|
|
/**
|
|
|
|
* Sends an array of table-specific action to the server to be applied. The tableId should be
|
|
|
|
* omitted from each `action` parameter and will be inserted automatically.
|
|
|
|
*
|
|
|
|
* @param {Array} actions: Array of user actions of the form [actionType, rowId, etc], which is sent
|
|
|
|
* to the server as [actionType, **tableId**, rowId, etc]
|
|
|
|
* @param {String} optDesc: Optional description of the actions to be shown in the log.
|
|
|
|
* @returns {Array} Array of return values for all the UserActions as produced by the data engine.
|
|
|
|
*/
|
|
|
|
public sendTableActions(actions: UserAction[], optDesc?: string) {
|
|
|
|
actions.forEach((action) => action.splice(1, 0, this.tableId));
|
|
|
|
return this.docData.sendActions(actions as DocAction[], optDesc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Sends a table-specific action to the server. The tableId should be omitted from the action parameter
|
|
|
|
* and will be inserted automatically.
|
|
|
|
*
|
|
|
|
* @param {Array} action: [actionType, rowId...], sent as [actionType, **tableId**, rowId...]
|
|
|
|
* @param {String} optDesc: Optional description of the actions to be shown in the log.
|
|
|
|
* @returns {Object} Return value for the UserAction as produced by the data engine.
|
|
|
|
*/
|
|
|
|
public sendTableAction(action: UserAction, optDesc?: string) {
|
|
|
|
if (!action) { return; }
|
|
|
|
action.splice(1, 0, this.tableId);
|
|
|
|
return this.docData.sendAction(action as DocAction, optDesc);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Emits a table-specific action received from the server as a 'tableAction' event.
|
|
|
|
*/
|
|
|
|
public receiveAction(action: DocAction): boolean {
|
|
|
|
const applied = super.receiveAction(action);
|
|
|
|
if (applied) {
|
|
|
|
this.tableActionEmitter.emit(action);
|
|
|
|
}
|
|
|
|
return applied;
|
|
|
|
}
|
|
|
|
}
|