dynamic-mat-table
Version:
dynamic-mat-table is an Angular component for presenting large and complex data with a lightning fast performance (at least 10x faster) and excellent level of control over the presentation.
134 lines • 20.2 kB
JavaScript
import { combineLatest, merge, Observable, of, ReplaySubject, Subscription } from 'rxjs';
import { map } from 'rxjs/operators';
import { MatTableDataSource } from '@angular/material/table';
import { titleCase } from '../utilizes/utilizes';
export class TableVirtualScrollDataSource extends MatTableDataSource {
constructor() {
super(...arguments);
this.filterMap = {};
this.columns = [];
}
get allData() {
return this.data;
}
toTranslate() {
const tranList = [];
const keys = Object.keys(this.filterMap);
for (const k of keys) {
let fieldTotalTran = '';
for (const f of this.filterMap[k]) {
fieldTotalTran += f.toPrint();
}
if (fieldTotalTran !== '') {
tranList.push({ key: titleCase(k), value: fieldTotalTran });
}
}
return tranList;
}
getFilter(fieldName) {
return this.filterMap[fieldName];
}
setFilter(fieldName, filters) {
this.filterMap[fieldName] = filters;
return new Observable(subscriber => {
setTimeout(() => {
this.refreshFilterPredicate();
subscriber.next();
subscriber.complete();
}, 200); // for show progress
});
}
clearFilter(fieldName = null) {
if (fieldName != null) {
delete this.filterMap[fieldName];
}
else {
this.filterMap = {};
}
this.refreshFilterPredicate();
}
clearData() {
this.data = [];
}
refreshFilterPredicate() {
let conditionsString = '';
Object.keys(this.filterMap).forEach(key => {
let fieldCondition = '';
this.filterMap[key].forEach((fieldFilter, row, array) => {
if (row < array.length - 1) {
fieldCondition += fieldFilter.toString(key) + (fieldFilter.type === 'and' ? ' && ' : ' || ');
}
else {
fieldCondition += fieldFilter.toString(key);
}
});
if (fieldCondition !== '') {
conditionsString += ` ${conditionsString === '' ? '' : ' && '} ( ${fieldCondition} )`;
}
});
if (conditionsString !== '') {
const filterFunction = new Function('_a$', 'return ' + conditionsString);
this.filterPredicate = (data, filter) => filterFunction(data);
}
else {
this.filterPredicate = (data, filter) => true;
}
this.filter = conditionsString;
}
// When client paging active use for retrieve paging data
pagingData(data) {
const p = this._paginator;
if (p && p !== null) {
const end = (p.pageIndex + 1) * p.pageSize;
const start = p.pageIndex * p.pageSize;
return data.slice(start, end);
}
return data;
}
_updateChangeSubscription() {
var _a;
this.initStreams();
const sort = this._sort;
const paginator = this._paginator;
const internalPageChanges = this._internalPageChanges;
const filter = this._filter;
const renderData = this._renderData;
const dataStream = this._data;
const sortChange = sort ?
merge(sort.sortChange, sort.initialized) : of(null);
const pageChange = paginator ?
merge(paginator.page, internalPageChanges, paginator.initialized) : of(null);
// First Filter
const filteredData = combineLatest([dataStream, filter]).pipe(map(([data]) => this._filterData(data)));
// Second Order
const orderedData = combineLatest([filteredData, sortChange]).pipe(map(([data, sortColumn]) => {
const sc = sortColumn;
if (!sc) {
return data;
}
else if (sc.active !== '') {
const column = this.columns.filter(c => c.name == sc.active)[0];
if (column.sort === 'server-side') {
return data;
}
else if (column.sort === 'client-side') {
return this._orderData(data);
}
}
}));
// Last Paging
const paginatedData = combineLatest([orderedData, pageChange]).pipe(map(([data]) => this.pagingData(data)));
(_a = this._renderChangesSubscription) === null || _a === void 0 ? void 0 : _a.unsubscribe();
this._renderChangesSubscription = new Subscription();
this._renderChangesSubscription.add(paginatedData.subscribe(data => this.dataToRender$.next(data)));
this._renderChangesSubscription.add(this.dataOfRange$.subscribe(data => renderData.next(data)));
}
initStreams() {
if (!this.streamsReady) {
this.dataToRender$ = new ReplaySubject(1);
this.dataOfRange$ = new ReplaySubject(1);
this.streamsReady = true;
}
}
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table-data-source.js","sourceRoot":"","sources":["../../../../../projects/dynamic-mat-table/src/lib/cores/table-data-source.ts"],"names":[],"mappings":"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","sourcesContent":["import { BehaviorSubject, combineLatest, merge, Observable, of, ReplaySubject, Subject, Subscription } from 'rxjs';\r\nimport { map } from 'rxjs/operators';\r\nimport { MatTableDataSource } from '@angular/material/table';\r\nimport { MatSort, Sort } from '@angular/material/sort';\r\nimport { MatPaginator, PageEvent } from '@angular/material/paginator';\r\nimport { AbstractFilter } from '../dynamic-mat-table/extensions/filter/compare/abstract-filter';\r\nimport { titleCase } from '../utilizes/utilizes';\r\nimport { HashMap } from './type';\r\nimport { TableField } from '../models/table-field.model';\r\nimport { TableRow } from '../models/table-row.model';\r\n\r\nexport class TableVirtualScrollDataSource<T extends TableRow> extends MatTableDataSource<T> {\r\n  public dataToRender$: Subject<T[]>;\r\n  public dataOfRange$: Subject<T[]>;\r\n  private streamsReady: boolean;\r\n  private filterMap: HashMap<AbstractFilter[]> = {};\r\n  public columns: TableField<T>[] = []\r\n  get allData(): T[]\r\n  {\r\n    return this.data;\r\n  }\r\n\r\n  toTranslate(): any[]\r\n  {\r\n    const tranList = [];\r\n    const keys: string[] = Object.keys(this.filterMap);\r\n    for (const k of keys)\r\n    {\r\n      let fieldTotalTran = '';\r\n      for (const f of this.filterMap[k])\r\n      {\r\n        fieldTotalTran += f.toPrint();\r\n      }\r\n      if (fieldTotalTran !== '')\r\n      {\r\n        tranList.push({ key: titleCase(k), value: fieldTotalTran });\r\n      }\r\n    }\r\n    return tranList;\r\n  }\r\n\r\n  getFilter(fieldName: string): AbstractFilter[]\r\n  {\r\n    return this.filterMap[fieldName];\r\n  }\r\n\r\n  setFilter(fieldName: string, filters: AbstractFilter[]): Observable<null>\r\n  {\r\n    this.filterMap[fieldName] = filters;\r\n    return new Observable(subscriber =>\r\n    {\r\n      setTimeout(() =>\r\n      {\r\n        this.refreshFilterPredicate();\r\n        subscriber.next();\r\n        subscriber.complete();\r\n      }, 200); // for show progress\r\n    });\r\n  }\r\n\r\n  clearFilter(fieldName: string = null)\r\n  {\r\n    if (fieldName != null)\r\n    {\r\n      delete this.filterMap[fieldName];\r\n    } else\r\n    {\r\n      this.filterMap = {};\r\n    }\r\n    this.refreshFilterPredicate();\r\n  }\r\n\r\n  clearData()\r\n  {\r\n    this.data = [];\r\n  }\r\n\r\n  public refreshFilterPredicate()\r\n  {\r\n    let conditionsString = '';\r\n    Object.keys(this.filterMap).forEach(key =>\r\n    {\r\n      let fieldCondition = '';\r\n      this.filterMap[key].forEach((fieldFilter, row, array) =>\r\n      {\r\n        if (row < array.length - 1)\r\n        {\r\n          fieldCondition += fieldFilter.toString(key) + (fieldFilter.type === 'and' ? ' && ' : ' || ');\r\n        } else\r\n        {\r\n          fieldCondition += fieldFilter.toString(key);\r\n        }\r\n      });\r\n      if (fieldCondition !== '')\r\n      {\r\n        conditionsString += ` ${conditionsString === '' ? '' : ' && '} ( ${fieldCondition} )`;\r\n      }\r\n    });\r\n    if (conditionsString !== '')\r\n    {\r\n      const filterFunction = new Function('_a$', 'return ' + conditionsString);\r\n      this.filterPredicate = (data: T, filter: string) => filterFunction(data) as boolean;\r\n    } else\r\n    {\r\n      this.filterPredicate = (data: T, filter: string) => true;\r\n    }\r\n    this.filter = conditionsString;\r\n  }\r\n\r\n  // When client paging active use for retrieve paging data\r\n  pagingData(data)\r\n  {\r\n    const p: MatPaginator = (this as any)._paginator;\r\n    if (p && p !== null)\r\n    {\r\n      const end = (p.pageIndex + 1) * p.pageSize;\r\n      const start = p.pageIndex * p.pageSize;\r\n      return data.slice(start, end);\r\n    }\r\n    return data;\r\n  }\r\n\r\n  _updateChangeSubscription()\r\n  {\r\n    this.initStreams();\r\n    const sort: MatSort | null = (this as any)._sort;\r\n    const paginator: MatPaginator | null = (this as any)._paginator;\r\n    const internalPageChanges: Subject<void> = (this as any)._internalPageChanges;\r\n    const filter: BehaviorSubject<string> = (this as any)._filter;\r\n    const renderData: BehaviorSubject<T[]> = (this as any)._renderData;\r\n    const dataStream: BehaviorSubject<T[]> = (this as any)._data;\r\n\r\n    const sortChange: Observable<Sort | null | void> = sort ?\r\n      merge(sort.sortChange, sort.initialized) as Observable<Sort | void> : of(null);\r\n    const pageChange: Observable<PageEvent | null | void> = paginator ?\r\n      merge(paginator.page, internalPageChanges, paginator.initialized) as Observable<PageEvent | void> : of(null);\r\n\r\n    // First Filter\r\n    const filteredData = combineLatest([dataStream, filter]).pipe(map(([data]) => this._filterData(data)));\r\n    // Second Order\r\n    const orderedData = combineLatest([filteredData, sortChange]).pipe(map(([data, sortColumn]) =>\r\n    {\r\n      const sc: Sort = sortColumn as Sort;\r\n      if (!sc)\r\n      {\r\n        return data;\r\n      } else if (sc.active !== '')\r\n      {\r\n        const column = this.columns.filter(c => c.name == sc.active)[0];\r\n        if (column.sort === 'server-side')\r\n        {\r\n          return data;\r\n        } else if (column.sort === 'client-side')\r\n        {\r\n          return this._orderData(data);\r\n        }\r\n      }\r\n    }));\r\n    // Last Paging\r\n    const paginatedData = combineLatest([orderedData, pageChange]).pipe(map(([data]) => this.pagingData(data)));\r\n\r\n    this._renderChangesSubscription?.unsubscribe();\r\n    this._renderChangesSubscription = new Subscription();\r\n    this._renderChangesSubscription.add(paginatedData.subscribe(data => this.dataToRender$.next(data)));\r\n    this._renderChangesSubscription.add(this.dataOfRange$.subscribe(data => renderData.next(data)));\r\n  }\r\n\r\n  private initStreams()\r\n  {\r\n    if (!this.streamsReady)\r\n    {\r\n      this.dataToRender$ = new ReplaySubject<T[]>(1);\r\n      this.dataOfRange$ = new ReplaySubject<T[]>(1);\r\n      this.streamsReady = true;\r\n    }\r\n  }\r\n}\r\n"]}