UNPKG

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
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"]}