@elderbyte/ngx-starter 15.3.4 → 15.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/_index.scss +6 -3
- package/esm2020/lib/common/data/data-context/data-context-active-page.mjs +2 -2
- package/esm2020/lib/common/data/data-context/data-context-base.mjs +46 -27
- package/esm2020/lib/common/data/data-context/data-context-builder.mjs +9 -8
- package/esm2020/lib/common/data/data-context/data-context-continuable-paged.mjs +7 -8
- package/esm2020/lib/common/data/data-context/data-context-continuable-token.mjs +2 -2
- package/esm2020/lib/common/data/data-context/data-context-life-cycle-binding.mjs +4 -2
- package/esm2020/lib/common/data/data-context/data-context-source-event-binding.mjs +46 -0
- package/esm2020/lib/common/data/data-context/data-context.mjs +1 -1
- package/esm2020/lib/common/data/data-context/indexed-entities.mjs +143 -0
- package/esm2020/lib/common/data/data-context/public_api.mjs +3 -2
- package/esm2020/lib/common/data/datasource/data-source-base.mjs +38 -0
- package/esm2020/lib/common/data/datasource/data-source-change-event.mjs +24 -0
- package/esm2020/lib/common/data/datasource/data-source.mjs +1 -1
- package/esm2020/lib/common/data/datasource/local/local-list-data-source.mjs +11 -19
- package/esm2020/lib/common/data/datasource/local/local-paged-data-source.mjs +8 -8
- package/esm2020/lib/common/data/datasource/public_api.mjs +3 -1
- package/esm2020/lib/common/data/datasource/rest/rest-client.mjs +23 -38
- package/fesm2015/elderbyte-ngx-starter.mjs +408 -178
- package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
- package/fesm2020/elderbyte-ngx-starter.mjs +407 -178
- package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
- package/lib/common/data/data-context/data-context-base.d.ts +12 -3
- package/lib/common/data/data-context/data-context-builder.d.ts +2 -1
- package/lib/common/data/data-context/data-context-source-event-binding.d.ts +31 -0
- package/lib/common/data/data-context/data-context.d.ts +14 -2
- package/lib/common/data/data-context/indexed-entities.d.ts +72 -0
- package/lib/common/data/data-context/public_api.d.ts +2 -1
- package/lib/common/data/datasource/data-source-base.d.ts +38 -0
- package/lib/common/data/datasource/data-source-change-event.d.ts +13 -0
- package/lib/common/data/datasource/data-source.d.ts +2 -1
- package/lib/common/data/datasource/local/local-list-data-source.d.ts +3 -7
- package/lib/common/data/datasource/local/local-paged-data-source.d.ts +5 -3
- package/lib/common/data/datasource/public_api.d.ts +2 -0
- package/lib/common/data/datasource/rest/rest-client.d.ts +10 -18
- package/package.json +1 -1
- package/theming/_elder-common.scss +14 -8
- package/theming/_elder-config.scss +37 -0
- package/theming/_elder-flex-layout.scss +231 -8
- package/esm2020/lib/common/data/data-context/data-context-source-auto-reloader.mjs +0 -12
- package/lib/common/data/data-context/data-context-source-auto-reloader.d.ts +0 -8
|
@@ -111,7 +111,7 @@ export class DataContextContinuablePaged extends DataContextContinuableBase {
|
|
|
111
111
|
this.onIdle();
|
|
112
112
|
}, err => {
|
|
113
113
|
this.onError(err);
|
|
114
|
-
this.
|
|
114
|
+
this.clearData();
|
|
115
115
|
this.setTotal(0);
|
|
116
116
|
this.logger.error('Failed to query data', err);
|
|
117
117
|
subject.error(err);
|
|
@@ -126,13 +126,12 @@ export class DataContextContinuablePaged extends DataContextContinuableBase {
|
|
|
126
126
|
try {
|
|
127
127
|
this.setTotal(page.totalElements);
|
|
128
128
|
const start = page.number * page.size;
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
this.
|
|
129
|
+
if (clear) {
|
|
130
|
+
this.setData(page.content);
|
|
131
|
+
}
|
|
132
|
+
else {
|
|
133
|
+
this.insertData(page.content, start);
|
|
134
134
|
}
|
|
135
|
-
this.setData(newData, true);
|
|
136
135
|
}
|
|
137
136
|
catch (err) {
|
|
138
137
|
this.onError(err);
|
|
@@ -140,4 +139,4 @@ export class DataContextContinuablePaged extends DataContextContinuableBase {
|
|
|
140
139
|
}
|
|
141
140
|
}
|
|
142
141
|
}
|
|
143
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-context-continuable-paged.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/data-context-continuable-paged.ts"],"names":[],"mappings":"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","sourcesContent":["\nimport {combineLatest, EMPTY, Observable, Subject} from 'rxjs';\nimport {Page, Pageable} from '../page';\nimport {Logger, LoggerFactory} from '@elderbyte/ts-logger';\nimport {DataContextContinuableBase} from './data-context-continuable-base';\nimport {first, map, takeUntil} from 'rxjs/operators';\nimport {Sort} from '../sort';\nimport {IPagedDataSource} from '../datasource/data-source';\n\n/**\n * Extends a simple flat list data-context with infinite-scroll pagination support.\n *\n */\nexport class DataContextContinuablePaged<T> extends DataContextContinuableBase<T> {\n\n    /***************************************************************************\n     *                                                                         *\n     * Fields                                                                  *\n     *                                                                         *\n     **************************************************************************/\n\n    private readonly logger: Logger = LoggerFactory.getLogger(this.constructor.name);\n\n    private _pageCache: Map<number, Observable<Page<T>>> = new Map();\n    private _latestPage = 0;\n\n    private readonly _hasMoreData: Observable<boolean>;\n\n    /***************************************************************************\n     *                                                                         *\n     * Constructors                                                            *\n     *                                                                         *\n     **************************************************************************/\n\n    constructor(\n        dataSource: IPagedDataSource<T>,\n        pageSize: number,\n        indexFn: ((item: T) => any),\n        localApply: ((data: T[]) => T[]),\n        localSort: ((data: T[], sorts: Sort[]) => T[])\n    ) {\n        super(dataSource, pageSize, indexFn, localApply, localSort);\n\n        this._hasMoreData = combineLatest([this.total, this.data]).pipe(\n          map(([total, data]) => this.checkHasMoreData(total, data)),\n          takeUntil(this.unsubscribe$)\n        );\n    }\n\n    /***************************************************************************\n     *                                                                         *\n     * Properties                                                              *\n     *                                                                         *\n     **************************************************************************/\n\n    public get dataSource(): IPagedDataSource<T> {\n      return <IPagedDataSource<T>>super.dataSource;\n    }\n\n    /***************************************************************************\n     *                                                                         *\n     * Public API                                                              *\n     *                                                                         *\n     **************************************************************************/\n\n    /**\n     * Load the next chunk of data.\n     * Useful for infinite scroll like data flows.\n     *\n     */\n    public loadMore(): Observable<any> {\n        if (this.hasMoreDataSnapshot) {\n            this.logger.info('Loading more...' + this._latestPage);\n\n            if (this.loadingSnapshot) { return EMPTY; }\n            const nextPage = this._latestPage + 1;\n            return this.fetchPage(nextPage, this.chunkSize);\n        } else {\n            this.logger.debug('Cannot load more data, since no more data available.');\n            return EMPTY;\n        }\n    }\n\n    public get hasMoreData(): Observable<boolean> {\n      return this._hasMoreData;\n    }\n\n    public get hasMoreDataSnapshot(): boolean {\n        return this.checkHasMoreData(this.totalSnapshot, this.dataSnapshot);\n    }\n\n    /***************************************************************************\n     *                                                                         *\n     * Private Methods                                                         *\n     *                                                                         *\n     **************************************************************************/\n\n    private checkHasMoreData(total: number | undefined, data: T[]): boolean {\n      if (total) {\n        return total > data.length;\n      }\n      return false;\n    }\n\n    protected onChunkSizeChanged(newSize: number): void {\n        this.reload();\n    }\n\n    protected reloadInternal(): Observable<any> {\n\n      // Since continuable data-contexts are appending data,\n      // we need to clear it for a reload.\n      this.clearAll(true);\n\n      return this.fetchPage(0, this.chunkSize, true);\n    }\n\n    protected clearAll(silent = false): void {\n        super.clearAll(silent);\n        this._pageCache = new Map();\n        this._latestPage = 0;\n    }\n\n    private fetchPage(pageIndex: number, pageSize: number, clear = false): Observable<any> {\n\n        const subject = new Subject<void>();\n\n        const pageRequest = new Pageable(pageIndex, pageSize, this.sort.sortsSnapshot);\n\n        if (this._pageCache.has(pageIndex)) {\n            // Page already loaded - skipping request!\n            this.logger.trace('Skipping fetching page since its already in page observable cache.');\n            subject.next();\n        } else {\n\n            this.onLoading();\n\n            this.logger.debug(`Loading page ${pageIndex} using pageable:`, pageRequest);\n\n            const pageObs = this.dataSource.findAllPaged(pageRequest, this.filter.filtersSnapshot);\n\n            this._pageCache.set(pageIndex, pageObs);\n\n            pageObs.subscribe((page: Page<T>) => {\n\n\n                this.logger.debug('Got page data:', page);\n\n                this.populatePageData(page, clear);\n\n                if (this._latestPage < page.number) {\n                    this._latestPage = page.number; // TODO This might cause that pages are skipped\n                }\n\n                subject.next();\n                this.onIdle();\n            }, err => {\n                this.onError(err);\n                this.setData([]);\n                this.setTotal(0);\n                this.logger.error('Failed to query data', err);\n                subject.error(err);\n            });\n        }\n\n        return subject.pipe(first());\n    }\n\n    /**\n     * Load the data from the given page into the current data context\n     */\n    private populatePageData(page: Page<T>, clear: boolean): void {\n        try {\n            this.setTotal(page.totalElements);\n            const start = page.number * page.size;\n\n            const newData = clear ? [] : [...this.dataSnapshot];\n            for (let i = 0; i < page.content.length; i++) {\n                const item = page.content[i];\n                newData[i + start] = item;\n                this.indexItem(item);\n            }\n            this.setData(newData, true);\n        } catch (err) {\n            this.onError(err);\n            this.logger.error('Failed to populate data with page', page, err);\n        }\n    }\n\n\n}\n"]}
|
|
142
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-context-continuable-paged.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/data-context-continuable-paged.ts"],"names":[],"mappings":"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","sourcesContent":["import {combineLatest, EMPTY, Observable, Subject} from 'rxjs';\nimport {Page, Pageable} from '../page';\nimport {Logger, LoggerFactory} from '@elderbyte/ts-logger';\nimport {DataContextContinuableBase} from './data-context-continuable-base';\nimport {first, map, takeUntil} from 'rxjs/operators';\nimport {Sort} from '../sort';\nimport {IPagedDataSource} from '../datasource/data-source';\n\n/**\n * Extends a simple flat list data-context with infinite-scroll pagination support.\n *\n */\nexport class DataContextContinuablePaged<T> extends DataContextContinuableBase<T> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger: Logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private _pageCache: Map<number, Observable<Page<T>>> = new Map();\n  private _latestPage = 0;\n\n  private readonly _hasMoreData: Observable<boolean>;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructors                                                            *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    dataSource: IPagedDataSource<T>,\n    pageSize: number,\n    indexFn: ((item: T) => any),\n    localApply: ((data: T[]) => T[]),\n    localSort: ((data: T[], sorts: Sort[]) => T[])\n  ) {\n    super(dataSource, pageSize, indexFn, localApply, localSort);\n\n    this._hasMoreData = combineLatest([this.total, this.data]).pipe(\n      map(([total, data]) => this.checkHasMoreData(total, data)),\n      takeUntil(this.unsubscribe$)\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get dataSource(): IPagedDataSource<T> {\n    return <IPagedDataSource<T>>super.dataSource;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Load the next chunk of data.\n   * Useful for infinite scroll like data flows.\n   *\n   */\n  public loadMore(): Observable<any> {\n    if (this.hasMoreDataSnapshot) {\n      this.logger.info('Loading more...' + this._latestPage);\n\n      if (this.loadingSnapshot) {\n        return EMPTY;\n      }\n      const nextPage = this._latestPage + 1;\n      return this.fetchPage(nextPage, this.chunkSize);\n    } else {\n      this.logger.debug('Cannot load more data, since no more data available.');\n      return EMPTY;\n    }\n  }\n\n  public get hasMoreData(): Observable<boolean> {\n    return this._hasMoreData;\n  }\n\n  public get hasMoreDataSnapshot(): boolean {\n    return this.checkHasMoreData(this.totalSnapshot, this.dataSnapshot);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private checkHasMoreData(total: number | undefined, data: T[]): boolean {\n    if (total) {\n      return total > data.length;\n    }\n    return false;\n  }\n\n  protected onChunkSizeChanged(newSize: number): void {\n    this.reload();\n  }\n\n  protected reloadInternal(): Observable<any> {\n    // Since continuable data-contexts are appending data,\n    // we need to clear it for a reload.\n    this.clearAll(true);\n    return this.fetchPage(0, this.chunkSize, true);\n  }\n\n  protected clearAll(silent = false): void {\n    super.clearAll(silent);\n    this._pageCache = new Map();\n    this._latestPage = 0;\n  }\n\n  private fetchPage(pageIndex: number, pageSize: number, clear = false): Observable<any> {\n\n    const subject = new Subject<void>();\n\n    const pageRequest = new Pageable(pageIndex, pageSize, this.sort.sortsSnapshot);\n\n    if (this._pageCache.has(pageIndex)) {\n      // Page already loaded - skipping request!\n      this.logger.trace('Skipping fetching page since its already in page observable cache.');\n      subject.next();\n    } else {\n\n      this.onLoading();\n\n      this.logger.debug(`Loading page ${pageIndex} using pageable:`, pageRequest);\n\n      const pageObs = this.dataSource.findAllPaged(pageRequest, this.filter.filtersSnapshot);\n\n      this._pageCache.set(pageIndex, pageObs);\n\n      pageObs.subscribe((page: Page<T>) => {\n\n\n        this.logger.debug('Got page data:', page);\n\n        this.populatePageData(page, clear);\n\n        if (this._latestPage < page.number) {\n          this._latestPage = page.number; // TODO This might cause that pages are skipped\n        }\n\n        subject.next();\n        this.onIdle();\n      }, err => {\n        this.onError(err);\n        this.clearData();\n        this.setTotal(0);\n        this.logger.error('Failed to query data', err);\n        subject.error(err);\n      });\n    }\n\n    return subject.pipe(first());\n  }\n\n  /**\n   * Load the data from the given page into the current data context\n   */\n  private populatePageData(page: Page<T>, clear: boolean): void {\n    try {\n      this.setTotal(page.totalElements);\n      const start = page.number * page.size;\n      if (clear) {\n        this.setData(page.content);\n      } else {\n        this.insertData(page.content, start);\n      }\n    } catch (err) {\n      this.onError(err);\n      this.logger.error('Failed to populate data with page', page, err);\n    }\n  }\n}\n"]}
|
|
@@ -93,7 +93,7 @@ export class DataContextContinuableToken extends DataContextContinuableBase {
|
|
|
93
93
|
}, err => {
|
|
94
94
|
this.onError(err);
|
|
95
95
|
this.logger.error('Failed to query data', err);
|
|
96
|
-
this.
|
|
96
|
+
this.clearData();
|
|
97
97
|
this.setTotal(0);
|
|
98
98
|
subject.error(err);
|
|
99
99
|
});
|
|
@@ -133,4 +133,4 @@ export class DataContextContinuableToken extends DataContextContinuableBase {
|
|
|
133
133
|
return token1 === token2;
|
|
134
134
|
}
|
|
135
135
|
}
|
|
136
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-context-continuable-token.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/data-context-continuable-token.ts"],"names":[],"mappings":"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","sourcesContent":["import {Logger, LoggerFactory} from '@elderbyte/ts-logger';\nimport {ContinuableListing} from '../continuable-listing';\nimport {DataContextContinuableBase} from './data-context-continuable-base';\nimport {BehaviorSubject, EMPTY, Observable, Subject} from 'rxjs';\nimport {first, take} from 'rxjs/operators';\nimport {TokenChunkRequest} from '../token-chunk-request';\nimport {Sort} from '../sort';\nimport {IContinuableDataSource} from '../datasource/data-source';\n\nexport class DataContextContinuableToken<T> extends DataContextContinuableBase<T> {\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger: Logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _hasMoreData = new BehaviorSubject<boolean>(false);\n  private readonly _chunkCache: Set<string|undefined> = new Set();\n\n  private _expectedChunkToken?: string;\n\n  // protected nextContinuationToken?: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    dataSource: IContinuableDataSource<T>,\n    chunkSize: number,\n    indexFn: ((item: T) => any),\n    localApply: ((data: T[]) => T[]),\n    localSort: ((data: T[], sorts: Sort[]) => T[])\n  ) {\n    super(dataSource, chunkSize, indexFn, localApply, localSort);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get dataSource(): IContinuableDataSource<T> {\n    return <IContinuableDataSource<T>>super.dataSource;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get hasMoreDataSnapshot(): boolean {\n    return this._hasMoreData.getValue();\n  }\n\n  public get hasMoreData(): Observable<boolean> {\n    return this._hasMoreData.asObservable();\n  }\n\n  public loadMore(): Observable<any> {\n\n    if (this.loadingSnapshot) {\n      this.logger.debug('Skipping load-more since already loading a chunk!');\n      return EMPTY;\n    }\n\n    const token = this._expectedChunkToken;\n\n    if (token && token.length > 0) {\n      return this.fetchNextChunk(token);\n    } else {\n      this.logger.debug('Cannot load more data, since no more data available.');\n      return EMPTY;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Protect API                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  protected clearAll(silent = false): void {\n    super.clearAll(silent);\n    this._chunkCache.clear();\n    this._hasMoreData.next(true);\n    this._expectedChunkToken = undefined;\n  }\n\n  protected reloadInternal(): Observable<any> {\n\n    // Since continuable data-contexts are appending data,\n    // we need to clear it for a reload.\n    this.clearAll(true);\n\n    return this.fetchNextChunk(undefined);\n  }\n\n  private fetchNextChunk(nextToken?: string): Observable<ContinuableListing<T>> {\n\n\n    const subject = new Subject<ContinuableListing<T>>();\n\n    nextToken = nextToken ? nextToken : undefined;\n\n    if (this._chunkCache.has(nextToken)) {\n      this.logger.debug('Skipping fetching chunk for token \"' + nextToken + '\" since its already in observable cache.');\n      subject.complete();\n    } else {\n\n      this.onLoading();\n\n      this._chunkCache.add(nextToken);\n\n      this.dataSource.findAllContinuable(new TokenChunkRequest(nextToken, this.filter.filtersSnapshot, this.sort.sortsSnapshot))\n        .pipe(first())\n        .subscribe(\n          chunk => {\n            this.logger.debug('Got next chunk data:', chunk);\n            this._hasMoreData.next(chunk.hasMore);\n            this.chunkSize = chunk.chunkSize;\n            this.populateChunkData(chunk);\n            subject.next(chunk);\n            this.onIdle();\n          }, err => {\n            this.onError(err);\n            this.logger.error('Failed to query data', err);\n            this.setData([]);\n            this.setTotal(0);\n            subject.error(err);\n          }\n        );\n    }\n    return subject.pipe(take(1));\n  }\n\n  /**\n   * Load the data from the given page into the current data context\n   */\n  private populateChunkData(chunk: ContinuableListing<T>): void {\n    if (this.areTokenEqual(chunk.continuationToken, this._expectedChunkToken)) {\n      try {\n        this.setTotal(chunk.total);\n\n        if (chunk.continuationToken) {\n          // We had previous chunks so append to current data.\n          this.appendData(chunk.content);\n        } else {\n          this.setData(chunk.content);\n        }\n\n      } catch (err) {\n        this.onError(err);\n        this.logger.error('Failed to populate data with chunk', chunk, err);\n      }\n      this._expectedChunkToken = chunk.nextContinuationToken;\n    } else {\n      this.logger.warn('Discarding continuable chunk (items: ' + chunk.content.length + ', token: ' + chunk.continuationToken + ' )' +\n        ' as it does not match the expected contiunation-token: ' + this._expectedChunkToken);\n    }\n  }\n\n  private areTokenEqual(token1?: string, token2?: string): boolean {\n    if (!token1 && !token2) { return true; }\n    return token1 === token2;\n  }\n\n\n}\n"]}
|
|
136
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-context-continuable-token.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/data-context-continuable-token.ts"],"names":[],"mappings":"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","sourcesContent":["import {Logger, LoggerFactory} from '@elderbyte/ts-logger';\nimport {ContinuableListing} from '../continuable-listing';\nimport {DataContextContinuableBase} from './data-context-continuable-base';\nimport {BehaviorSubject, EMPTY, Observable, Subject} from 'rxjs';\nimport {first, take} from 'rxjs/operators';\nimport {TokenChunkRequest} from '../token-chunk-request';\nimport {Sort} from '../sort';\nimport {IContinuableDataSource} from '../datasource/data-source';\n\nexport class DataContextContinuableToken<T> extends DataContextContinuableBase<T> {\n\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger: Logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _hasMoreData = new BehaviorSubject<boolean>(false);\n  private readonly _chunkCache: Set<string|undefined> = new Set();\n\n  private _expectedChunkToken?: string;\n\n  // protected nextContinuationToken?: string;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    dataSource: IContinuableDataSource<T>,\n    chunkSize: number,\n    indexFn: ((item: T) => any),\n    localApply: ((data: T[]) => T[]),\n    localSort: ((data: T[], sorts: Sort[]) => T[])\n  ) {\n    super(dataSource, chunkSize, indexFn, localApply, localSort);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get dataSource(): IContinuableDataSource<T> {\n    return <IContinuableDataSource<T>>super.dataSource;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get hasMoreDataSnapshot(): boolean {\n    return this._hasMoreData.getValue();\n  }\n\n  public get hasMoreData(): Observable<boolean> {\n    return this._hasMoreData.asObservable();\n  }\n\n  public loadMore(): Observable<any> {\n\n    if (this.loadingSnapshot) {\n      this.logger.debug('Skipping load-more since already loading a chunk!');\n      return EMPTY;\n    }\n\n    const token = this._expectedChunkToken;\n\n    if (token && token.length > 0) {\n      return this.fetchNextChunk(token);\n    } else {\n      this.logger.debug('Cannot load more data, since no more data available.');\n      return EMPTY;\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Protect API                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  protected clearAll(silent = false): void {\n    super.clearAll(silent);\n    this._chunkCache.clear();\n    this._hasMoreData.next(true);\n    this._expectedChunkToken = undefined;\n  }\n\n  protected reloadInternal(): Observable<any> {\n\n    // Since continuable data-contexts are appending data,\n    // we need to clear it for a reload.\n    this.clearAll(true);\n\n    return this.fetchNextChunk(undefined);\n  }\n\n  private fetchNextChunk(nextToken?: string): Observable<ContinuableListing<T>> {\n\n    const subject = new Subject<ContinuableListing<T>>();\n\n    nextToken = nextToken ? nextToken : undefined;\n\n    if (this._chunkCache.has(nextToken)) {\n      this.logger.debug('Skipping fetching chunk for token \"' + nextToken + '\" since its already in observable cache.');\n      subject.complete();\n    } else {\n\n      this.onLoading();\n\n      this._chunkCache.add(nextToken);\n\n      this.dataSource.findAllContinuable(new TokenChunkRequest(nextToken, this.filter.filtersSnapshot, this.sort.sortsSnapshot))\n        .pipe(first())\n        .subscribe(\n          chunk => {\n            this.logger.debug('Got next chunk data:', chunk);\n            this._hasMoreData.next(chunk.hasMore);\n            this.chunkSize = chunk.chunkSize;\n            this.populateChunkData(chunk);\n            subject.next(chunk);\n            this.onIdle();\n          }, err => {\n            this.onError(err);\n            this.logger.error('Failed to query data', err);\n            this.clearData();\n            this.setTotal(0);\n            subject.error(err);\n          }\n        );\n    }\n    return subject.pipe(take(1));\n  }\n\n  /**\n   * Load the data from the given page into the current data context\n   */\n  private populateChunkData(chunk: ContinuableListing<T>): void {\n    if (this.areTokenEqual(chunk.continuationToken, this._expectedChunkToken)) {\n      try {\n        this.setTotal(chunk.total);\n\n        if (chunk.continuationToken) {\n          // We had previous chunks so append to current data.\n          this.appendData(chunk.content);\n        } else {\n          this.setData(chunk.content);\n        }\n\n      } catch (err) {\n        this.onError(err);\n        this.logger.error('Failed to populate data with chunk', chunk, err);\n      }\n      this._expectedChunkToken = chunk.nextContinuationToken;\n    } else {\n      this.logger.warn('Discarding continuable chunk (items: ' + chunk.content.length + ', token: ' + chunk.continuationToken + ' )' +\n        ' as it does not match the expected contiunation-token: ' + this._expectedChunkToken);\n    }\n  }\n\n  private areTokenEqual(token1?: string, token2?: string): boolean {\n    if (!token1 && !token2) { return true; }\n    return token1 === token2;\n  }\n\n\n}\n"]}
|
|
@@ -13,7 +13,9 @@ export class DataContextLifeCycleBinding {
|
|
|
13
13
|
if (_dataContext == null) {
|
|
14
14
|
throw new Error('dataContext must not be null!');
|
|
15
15
|
}
|
|
16
|
-
this._dataContext.data.subscribe(
|
|
16
|
+
this._dataContext.data.subscribe({
|
|
17
|
+
complete: () => this.unsubscribe()
|
|
18
|
+
});
|
|
17
19
|
}
|
|
18
20
|
/***************************************************************************
|
|
19
21
|
* *
|
|
@@ -27,4 +29,4 @@ export class DataContextLifeCycleBinding {
|
|
|
27
29
|
}
|
|
28
30
|
}
|
|
29
31
|
}
|
|
30
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
32
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YS1jb250ZXh0LWxpZmUtY3ljbGUtYmluZGluZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbW1vbi9kYXRhL2RhdGEtY29udGV4dC9kYXRhLWNvbnRleHQtbGlmZS1jeWNsZS1iaW5kaW5nLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUdBOzs7R0FHRztBQUNILE1BQU0sT0FBZ0IsMkJBQTJCO0lBVS9DOzs7O2dGQUk0RTtJQUU1RSxZQUNxQixZQUErQjtRQUEvQixpQkFBWSxHQUFaLFlBQVksQ0FBbUI7UUFFbEQsSUFBSSxZQUFZLElBQUksSUFBSSxFQUFFO1lBQ3hCLE1BQU0sSUFBSSxLQUFLLENBQUMsK0JBQStCLENBQUMsQ0FBQztTQUNsRDtRQUVELElBQUksQ0FBQyxZQUFZLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FDOUI7WUFDRSxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRTtTQUNuQyxDQUNGLENBQUM7SUFDSixDQUFDO0lBRUQ7Ozs7Z0ZBSTRFO0lBRXJFLFdBQVc7UUFDaEIsSUFBSSxJQUFJLENBQUMsYUFBYSxFQUFFO1lBQ3RCLElBQUksQ0FBQyxhQUFhLENBQUMsV0FBVyxFQUFFLENBQUM7WUFDakMsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUM7U0FDM0I7SUFDSCxDQUFDO0NBVUYiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge1N1YnNjcmlwdGlvbiwgVW5zdWJzY3JpYmFibGV9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtJRGF0YUNvbnRleHR9IGZyb20gJy4vZGF0YS1jb250ZXh0JztcblxuLyoqXG4gKiBBbGxvd3MgbWFraW5nIGFueSBraW5kIG9mIHN1YnNjcmlwdGlvbiB3aGljaCB3aWxsIGJlIGF1dG9tYXRpY2FsbHkgdW5zdWJzY3JpYmVkXG4gKiB1cG9uIGRhdGEgY29udGV4dCBjbG9zaW5nL2NsZWFudXAuXG4gKi9cbmV4cG9ydCBhYnN0cmFjdCBjbGFzcyBEYXRhQ29udGV4dExpZmVDeWNsZUJpbmRpbmcgaW1wbGVtZW50cyBVbnN1YnNjcmliYWJsZSB7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIEZpZWxkcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgcHJvdGVjdGVkIF9zdWJzY3JpcHRpb246IFN1YnNjcmlwdGlvbjtcblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogQ29uc3RydWN0b3IgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IF9kYXRhQ29udGV4dDogSURhdGFDb250ZXh0PGFueT5cbiAgKSB7XG4gICAgaWYgKF9kYXRhQ29udGV4dCA9PSBudWxsKSB7XG4gICAgICB0aHJvdyBuZXcgRXJyb3IoJ2RhdGFDb250ZXh0IG11c3Qgbm90IGJlIG51bGwhJyk7XG4gICAgfVxuXG4gICAgdGhpcy5fZGF0YUNvbnRleHQuZGF0YS5zdWJzY3JpYmUoXG4gICAgICB7XG4gICAgICAgIGNvbXBsZXRlOiAoKSA9PiB0aGlzLnVuc3Vic2NyaWJlKClcbiAgICAgIH1cbiAgICApO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIFB1YmxpYyBBUEkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgcHVibGljIHVuc3Vic2NyaWJlKCk6IHZvaWQge1xuICAgIGlmICh0aGlzLl9zdWJzY3JpcHRpb24pIHtcbiAgICAgIHRoaXMuX3N1YnNjcmlwdGlvbi51bnN1YnNjcmliZSgpO1xuICAgICAgdGhpcy5fc3Vic2NyaXB0aW9uID0gbnVsbDtcbiAgICB9XG4gIH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHJvdGVjdGVkICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwcm90ZWN0ZWQgYWJzdHJhY3Qgc3Vic2NyaWJlKCk6IHZvaWQ7XG5cbn1cbiJdfQ==
|
|
@@ -0,0 +1,46 @@
|
|
|
1
|
+
import { DataContextLifeCycleBinding } from './data-context-life-cycle-binding';
|
|
2
|
+
export class DataContextSourceEventBinding extends DataContextLifeCycleBinding {
|
|
3
|
+
/***************************************************************************
|
|
4
|
+
* *
|
|
5
|
+
* Fields *
|
|
6
|
+
* *
|
|
7
|
+
**************************************************************************/
|
|
8
|
+
/***************************************************************************
|
|
9
|
+
* *
|
|
10
|
+
* Constructor *
|
|
11
|
+
* *
|
|
12
|
+
**************************************************************************/
|
|
13
|
+
constructor(dataContext, dataApi, reloadOnChanges) {
|
|
14
|
+
super(dataContext);
|
|
15
|
+
this.dataApi = dataApi;
|
|
16
|
+
this.reloadOnChanges = reloadOnChanges;
|
|
17
|
+
this.subscribe();
|
|
18
|
+
}
|
|
19
|
+
/***************************************************************************
|
|
20
|
+
* *
|
|
21
|
+
* Internal Impl *
|
|
22
|
+
* *
|
|
23
|
+
**************************************************************************/
|
|
24
|
+
subscribe() {
|
|
25
|
+
this._subscription = this.dataApi.dataChanged.subscribe(event => this.handleDataChangeEvent(event));
|
|
26
|
+
}
|
|
27
|
+
/***************************************************************************
|
|
28
|
+
* *
|
|
29
|
+
* Private methods *
|
|
30
|
+
* *
|
|
31
|
+
**************************************************************************/
|
|
32
|
+
handleDataChangeEvent(event) {
|
|
33
|
+
if (this.reloadOnChanges && this.isReloadDesirable(event)) {
|
|
34
|
+
this._dataContext.reload();
|
|
35
|
+
return;
|
|
36
|
+
}
|
|
37
|
+
// We might also be able to perform deletions locally, and avoid reload data
|
|
38
|
+
if (event.modified) {
|
|
39
|
+
this._dataContext.update(event.modified);
|
|
40
|
+
}
|
|
41
|
+
}
|
|
42
|
+
isReloadDesirable(event) {
|
|
43
|
+
return event.unknownChanges || event.created || event.deletedIds;
|
|
44
|
+
}
|
|
45
|
+
}
|
|
46
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YS1jb250ZXh0LXNvdXJjZS1ldmVudC1iaW5kaW5nLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tbW9uL2RhdGEvZGF0YS1jb250ZXh0L2RhdGEtY29udGV4dC1zb3VyY2UtZXZlbnQtYmluZGluZy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxPQUFPLEVBQUMsMkJBQTJCLEVBQUMsTUFBTSxtQ0FBbUMsQ0FBQztBQUc5RSxNQUFNLE9BQU8sNkJBQXVDLFNBQVEsMkJBQTJCO0lBRXJGOzs7O2dGQUk0RTtJQUU1RTs7OztnRkFJNEU7SUFFNUUsWUFDRSxXQUE0QixFQUNYLE9BQXVCLEVBQ3ZCLGVBQXdCO1FBRXpDLEtBQUssQ0FBQyxXQUFXLENBQUMsQ0FBQztRQUhGLFlBQU8sR0FBUCxPQUFPLENBQWdCO1FBQ3ZCLG9CQUFlLEdBQWYsZUFBZSxDQUFTO1FBR3pDLElBQUksQ0FBQyxTQUFTLEVBQUUsQ0FBQztJQUNuQixDQUFDO0lBRUQ7Ozs7Z0ZBSTRFO0lBRWxFLFNBQVM7UUFDakIsSUFBSSxDQUFDLGFBQWEsR0FBRyxJQUFJLENBQUMsT0FBTyxDQUFDLFdBQVcsQ0FBQyxTQUFTLENBQ3JELEtBQUssQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLHFCQUFxQixDQUFDLEtBQUssQ0FBQyxDQUMzQyxDQUFDO0lBQ0osQ0FBQztJQUVEOzs7O2dGQUk0RTtJQUVwRSxxQkFBcUIsQ0FBQyxLQUErQjtRQUUzRCxJQUFJLElBQUksQ0FBQyxlQUFlLElBQUksSUFBSSxDQUFDLGlCQUFpQixDQUFDLEtBQUssQ0FBQyxFQUFFO1lBQ3pELElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxFQUFFLENBQUM7WUFDM0IsT0FBTztTQUNSO1FBRUQsNEVBQTRFO1FBRTVFLElBQUksS0FBSyxDQUFDLFFBQVEsRUFBRTtZQUNsQixJQUFJLENBQUMsWUFBWSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsUUFBUSxDQUFDLENBQUM7U0FDMUM7SUFDSCxDQUFDO0lBRU8saUJBQWlCLENBQUMsS0FBK0I7UUFDdkQsT0FBTyxLQUFLLENBQUMsY0FBYyxJQUFJLEtBQUssQ0FBQyxPQUFPLElBQUksS0FBSyxDQUFDLFVBQVUsQ0FBQztJQUNuRSxDQUFDO0NBRUYiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQge0lEYXRhQ29udGV4dH0gZnJvbSAnLi9kYXRhLWNvbnRleHQnO1xuaW1wb3J0IHtJRGF0YVNvdXJjZX0gZnJvbSAnLi4vZGF0YXNvdXJjZS9kYXRhLXNvdXJjZSc7XG5pbXBvcnQge0RhdGFDb250ZXh0TGlmZUN5Y2xlQmluZGluZ30gZnJvbSAnLi9kYXRhLWNvbnRleHQtbGlmZS1jeWNsZS1iaW5kaW5nJztcbmltcG9ydCB7RGF0YVNvdXJjZUNoYW5nZUV2ZW50fSBmcm9tICcuLi9kYXRhc291cmNlL2RhdGEtc291cmNlLWNoYW5nZS1ldmVudCc7XG5cbmV4cG9ydCBjbGFzcyBEYXRhQ29udGV4dFNvdXJjZUV2ZW50QmluZGluZzxUID0gYW55PiBleHRlbmRzIERhdGFDb250ZXh0TGlmZUN5Y2xlQmluZGluZyB7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIEZpZWxkcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIENvbnN0cnVjdG9yICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgY29uc3RydWN0b3IoXG4gICAgZGF0YUNvbnRleHQ6IElEYXRhQ29udGV4dDxUPixcbiAgICBwcml2YXRlIHJlYWRvbmx5IGRhdGFBcGk6IElEYXRhU291cmNlPFQ+LFxuICAgIHByaXZhdGUgcmVhZG9ubHkgcmVsb2FkT25DaGFuZ2VzOiBib29sZWFuXG4gICkge1xuICAgIHN1cGVyKGRhdGFDb250ZXh0KTtcbiAgICB0aGlzLnN1YnNjcmliZSgpO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIEludGVybmFsIEltcGwgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgcHJvdGVjdGVkIHN1YnNjcmliZSgpOiB2b2lkIHtcbiAgICB0aGlzLl9zdWJzY3JpcHRpb24gPSB0aGlzLmRhdGFBcGkuZGF0YUNoYW5nZWQuc3Vic2NyaWJlKFxuICAgICAgZXZlbnQgPT4gdGhpcy5oYW5kbGVEYXRhQ2hhbmdlRXZlbnQoZXZlbnQpXG4gICAgKTtcbiAgfVxuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBQcml2YXRlIG1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgaGFuZGxlRGF0YUNoYW5nZUV2ZW50KGV2ZW50OiBEYXRhU291cmNlQ2hhbmdlRXZlbnQ8VD4pOiB2b2lkIHtcblxuICAgIGlmICh0aGlzLnJlbG9hZE9uQ2hhbmdlcyAmJiB0aGlzLmlzUmVsb2FkRGVzaXJhYmxlKGV2ZW50KSkge1xuICAgICAgdGhpcy5fZGF0YUNvbnRleHQucmVsb2FkKCk7XG4gICAgICByZXR1cm47XG4gICAgfVxuXG4gICAgLy8gV2UgbWlnaHQgYWxzbyBiZSBhYmxlIHRvIHBlcmZvcm0gZGVsZXRpb25zIGxvY2FsbHksIGFuZCBhdm9pZCByZWxvYWQgZGF0YVxuXG4gICAgaWYgKGV2ZW50Lm1vZGlmaWVkKSB7XG4gICAgICB0aGlzLl9kYXRhQ29udGV4dC51cGRhdGUoZXZlbnQubW9kaWZpZWQpO1xuICAgIH1cbiAgfVxuXG4gIHByaXZhdGUgaXNSZWxvYWREZXNpcmFibGUoZXZlbnQ6IERhdGFTb3VyY2VDaGFuZ2VFdmVudDxUPikge1xuICAgIHJldHVybiBldmVudC51bmtub3duQ2hhbmdlcyB8fCBldmVudC5jcmVhdGVkIHx8IGV2ZW50LmRlbGV0ZWRJZHM7XG4gIH1cblxufVxuIl19
|
|
@@ -27,4 +27,4 @@ export function isContinuableDataContext(object) {
|
|
|
27
27
|
export function isActivePagedDataContext(object) {
|
|
28
28
|
return object.setActivePage !== undefined;
|
|
29
29
|
}
|
|
30
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
30
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-context.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/data-context.ts"],"names":[],"mappings":"AAUA,MAAM,OAAO,mBAAmB;IAC9B,YACkB,IAAS,EACT,OAAgB,EAChB,SAAkB,EAClB,KAAyB,EACzB,MAAyB;QAJzB,SAAI,GAAJ,IAAI,CAAK;QACT,YAAO,GAAP,OAAO,CAAS;QAChB,cAAS,GAAT,SAAS,CAAS;QAClB,UAAK,GAAL,KAAK,CAAoB;QACzB,WAAM,GAAN,MAAM,CAAmB;IACvC,CAAC;CACN;AAiLD;;;;4EAI4E;AAE5E,MAAM,UAAU,aAAa,CAAC,MAAW;IAEvC,IAAI,CAAC,MAAM,EAAE;QAAE,OAAO,KAAK,CAAC;KAAE;IAE9B,OAAQ,MAA4B,CAAC,IAAI,KAAK,SAAS;WACjD,MAA4B,CAAC,KAAK,KAAK,SAAS;WAChD,MAA4B,CAAC,MAAM,KAAK,SAAS;WACjD,MAA4B,CAAC,OAAO,KAAK,SAAS,CAAC;AAC3D,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,MAAyB;IAChE,OAAQ,MAAuC,CAAC,QAAQ,KAAK,SAAS,CAAC;AACzE,CAAC;AAED,MAAM,UAAU,wBAAwB,CAAC,MAAyB;IAChE,OAAQ,MAAsC,CAAC,aAAa,KAAK,SAAS,CAAC;AAC7E,CAAC","sourcesContent":["import {DataContextStatus} from './data-context-status';\nimport {Observable} from 'rxjs';\nimport {Sort} from '../sort';\nimport {Filter} from '../filters/filter';\nimport {FilterContext} from '../filters/filter-context';\nimport {PageRequest} from '../page';\nimport {SortContext} from '../sort-context';\nimport {IDataSource} from '../datasource/data-source';\n\n\nexport class DataContextSnapshot<T> {\n  constructor(\n    public readonly data: T[],\n    public readonly isEmpty: boolean,\n    public readonly isLoading: boolean,\n    public readonly total: number | undefined,\n    public readonly status: DataContextStatus\n  ) { }\n}\n\n/**\n * Manages a set of data (rows) which are to be displayed in a UI Component.\n *\n */\nexport interface IDataContext<T> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Gets the underlying DataSource of this DataContext.\n   */\n  readonly dataSource: IDataSource<T>;\n\n  /**\n   * Gets a snapshot of the current state in this data-context\n   */\n  readonly snapshot: DataContextSnapshot<T>;\n\n  /**\n   * Observable which emits the current data over time.\n   */\n  readonly data: Observable<T[]>;\n\n  /**\n   * Indicates if the context currently holds no data\n   */\n  readonly isEmpty: boolean;\n\n  /**\n   * Indicates if the context has been started.\n   */\n  readonly isStarted: boolean;\n\n  /**\n   * Indicates if the context has been started.\n   */\n  readonly isStarted$: Observable<boolean>;\n\n  /**\n   * Indicates if the context has been closed.\n   * Depending on the implementation, restarting might not be possible.\n   */\n  readonly isClosed: boolean;\n\n  /**\n   * Indicates if data is loading over time\n   */\n  readonly loading: Observable<boolean>;\n\n  /**\n   * The total count of all elements over time\n   * (I.e. the expected count when the data contex is completed)\n   */\n  readonly total: Observable<number | undefined>;\n\n  /**\n   * Observable which emits when the status changes (i.e. error)\n   */\n  readonly status: Observable<DataContextStatus>;\n\n  /**\n   * Gets the sort context. Changes in this context are reflected by the data-context.\n   */\n  readonly sort: SortContext;\n\n  /**\n   * Gets the filter context. Changes in this context are reflected by the data-context.\n   */\n  readonly filter: FilterContext;\n\n  /***************************************************************************\n   *                                                                         *\n   * API                                                                     *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Starts populating data context by loading first\n   * batch of data.\n   */\n  start(sorts?: Sort[], filters?: Filter[]): Observable<any>;\n\n  /**\n   * Requests a reload of the current data / chunk.\n   * Attempts to keep the current position.\n   */\n  reload(): void;\n\n  /**\n   * Refresh the data view from the local data model. (Does not request new data from the server)\n   * Useful if the data model has changes that need to be reflected in the UI.\n   */\n  refresh(): void;\n\n  /**\n   * Closes the data-context and cleans up resources\n   */\n  close(): void;\n\n  /**\n   * Returns the item by a custom index key.\n   * Requires that custom index functionality is enabled.\n   *\n   * For id based lookup, refer to findById().\n   *\n   * @param key\n   */\n  findByIndex(key: any): T | undefined;\n\n  /**\n   * Lookup an entity by its id. This method is guaranteed to be backed by a fast index.\n   * @param id The Entity Id.\n   */\n  findById(id: any): T | undefined;\n\n  /**\n   * Update the given entities in this data-context\n   */\n  update(entities: T[]): void;\n\n}\n\n/**\n * Represents a data-context which backs a infinite like data stream\n * where more data will be loaded and added to the backing rows.\n */\nexport interface IDataContextContinuable<T> extends IDataContext<T> {\n\n  /**\n   * Returns true if this data-context can load more data\n   */\n  readonly hasMoreDataSnapshot: boolean;\n\n  /**\n   * Returns true if this data-context can load more data over time\n   */\n  readonly hasMoreData: Observable<boolean>;\n\n  /**\n   * Loads more data if any available.\n   * E.g. next page.\n   *\n   */\n  loadMore(): Observable<any>;\n\n  /**\n   * Loads all available data. In case of\n   * paged context loads page by page until finished.\n   */\n  loadAll(sorts?: Sort[], filters?: Filter[]): void;\n\n}\n\nexport interface IDataContextActivePage<T> extends IDataContext<T> {\n\n  /**\n   * Gets the active page over time\n   */\n  readonly page: Observable<PageRequest>;\n\n  /**\n   * Gets the current active page\n   */\n  readonly pageSnapshot: PageRequest;\n\n  /**\n   * Sets the active page\n   */\n  setActivePage(page: PageRequest): void;\n}\n\n/***************************************************************************\n *                                                                         *\n * Type Predicates                                                         *\n *                                                                         *\n **************************************************************************/\n\nexport function isDataContext(object: any): object is IDataContext<any> {\n\n  if (!object) { return false; }\n\n  return (object as IDataContext<any>).data !== undefined\n    && (object as IDataContext<any>).start !== undefined\n    && (object as IDataContext<any>).reload !== undefined\n    && (object as IDataContext<any>).refresh !== undefined;\n}\n\nexport function isContinuableDataContext(object: IDataContext<any>): object is IDataContextContinuable<any> {\n  return (object as IDataContextContinuable<any>).loadMore !== undefined;\n}\n\nexport function isActivePagedDataContext(object: IDataContext<any>): object is IDataContextActivePage<any> {\n  return (object as IDataContextActivePage<any>).setActivePage !== undefined;\n}\n"]}
|
|
@@ -0,0 +1,143 @@
|
|
|
1
|
+
import { BehaviorSubject } from 'rxjs';
|
|
2
|
+
import { SortContext } from '../sort-context';
|
|
3
|
+
import { LoggerFactory } from '@elderbyte/ts-logger';
|
|
4
|
+
export class IndexedEntities {
|
|
5
|
+
/***************************************************************************
|
|
6
|
+
* *
|
|
7
|
+
* Constructor *
|
|
8
|
+
* *
|
|
9
|
+
**************************************************************************/
|
|
10
|
+
constructor(idFn, localSortFn, sortContext = new SortContext()) {
|
|
11
|
+
this.idFn = idFn;
|
|
12
|
+
this.localSortFn = localSortFn;
|
|
13
|
+
this.sortContext = sortContext;
|
|
14
|
+
/***************************************************************************
|
|
15
|
+
* *
|
|
16
|
+
* Fields *
|
|
17
|
+
* *
|
|
18
|
+
**************************************************************************/
|
|
19
|
+
this.log = LoggerFactory.getLogger('IndexedEntities');
|
|
20
|
+
this._entities$ = new BehaviorSubject([]);
|
|
21
|
+
this._idToArrayIndex = new Map();
|
|
22
|
+
}
|
|
23
|
+
/***************************************************************************
|
|
24
|
+
* *
|
|
25
|
+
* Properties *
|
|
26
|
+
* *
|
|
27
|
+
**************************************************************************/
|
|
28
|
+
get entities$() {
|
|
29
|
+
return this._entities$.asObservable();
|
|
30
|
+
}
|
|
31
|
+
get entitiesSnapshot() {
|
|
32
|
+
return this._entities$.getValue();
|
|
33
|
+
}
|
|
34
|
+
/***************************************************************************
|
|
35
|
+
* *
|
|
36
|
+
* Public API *
|
|
37
|
+
* *
|
|
38
|
+
**************************************************************************/
|
|
39
|
+
/**
|
|
40
|
+
* Replaces all existing entities with the new ones.
|
|
41
|
+
* The index is rebuilt. O(n)
|
|
42
|
+
*/
|
|
43
|
+
replaceAll(newEntities = []) {
|
|
44
|
+
if (this.localSortFn) {
|
|
45
|
+
newEntities = this.localSortFn(newEntities, this.sortContext.sortsSnapshot);
|
|
46
|
+
}
|
|
47
|
+
this.rebuildIndex(newEntities);
|
|
48
|
+
this.setEntities(newEntities);
|
|
49
|
+
}
|
|
50
|
+
/**
|
|
51
|
+
* Keeps all existing entities, but replaces those which are updated with the new data.
|
|
52
|
+
* Provided updated entities which are currently not in this index are ignored.
|
|
53
|
+
* @param updated
|
|
54
|
+
*/
|
|
55
|
+
updateSome(updated) {
|
|
56
|
+
const newData = [...this.entitiesSnapshot];
|
|
57
|
+
updated.forEach(update => {
|
|
58
|
+
const id = this.getItemId(update);
|
|
59
|
+
const index = this.indexById(id);
|
|
60
|
+
if (index != -1) {
|
|
61
|
+
newData[index] = update;
|
|
62
|
+
}
|
|
63
|
+
});
|
|
64
|
+
this.setEntities(newData);
|
|
65
|
+
}
|
|
66
|
+
append(append) {
|
|
67
|
+
this.insert(append, this.entitiesSnapshot.length);
|
|
68
|
+
}
|
|
69
|
+
/**
|
|
70
|
+
* Appends the given entities to this index.
|
|
71
|
+
* The new appended entities are included in the index.
|
|
72
|
+
* O(n) where n = number of appended entities
|
|
73
|
+
* @param inserted
|
|
74
|
+
* @param offset
|
|
75
|
+
*/
|
|
76
|
+
insert(inserted, offset) {
|
|
77
|
+
if (inserted && inserted.length > 0) {
|
|
78
|
+
const entities = this.entitiesSnapshot;
|
|
79
|
+
const newData = [...entities];
|
|
80
|
+
for (let i = 0; i < inserted.length; i++) {
|
|
81
|
+
newData[i + offset] = inserted[i];
|
|
82
|
+
}
|
|
83
|
+
if (this.localSortFn) {
|
|
84
|
+
// local sort can re-order all items
|
|
85
|
+
this.replaceAll(newData);
|
|
86
|
+
}
|
|
87
|
+
else {
|
|
88
|
+
// Without a sort, we can optimize indexing
|
|
89
|
+
this.reIndexFrom(newData, offset);
|
|
90
|
+
this.setEntities(newData);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Lookup an entity by its id.
|
|
96
|
+
* This will use an index O(1)
|
|
97
|
+
* @param id
|
|
98
|
+
*/
|
|
99
|
+
getById(id) {
|
|
100
|
+
const entities = this.entitiesSnapshot;
|
|
101
|
+
const index = this.indexById(id);
|
|
102
|
+
if (index != -1) {
|
|
103
|
+
return entities[index];
|
|
104
|
+
}
|
|
105
|
+
}
|
|
106
|
+
notify() {
|
|
107
|
+
this._entities$.next(this._entities$.getValue());
|
|
108
|
+
}
|
|
109
|
+
destroy() {
|
|
110
|
+
this._entities$.complete();
|
|
111
|
+
this._idToArrayIndex.clear();
|
|
112
|
+
}
|
|
113
|
+
/***************************************************************************
|
|
114
|
+
* *
|
|
115
|
+
* Private methods *
|
|
116
|
+
* *
|
|
117
|
+
**************************************************************************/
|
|
118
|
+
rebuildIndex(entities) {
|
|
119
|
+
this._idToArrayIndex.clear();
|
|
120
|
+
this.reIndexFrom(entities, 0);
|
|
121
|
+
}
|
|
122
|
+
reIndexFrom(entities, startIndex) {
|
|
123
|
+
for (let i = startIndex; i < entities.length; i++) {
|
|
124
|
+
const entity = entities[i];
|
|
125
|
+
if (entity) {
|
|
126
|
+
this._idToArrayIndex.set(this.getItemId(entity), i);
|
|
127
|
+
}
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
indexById(id) {
|
|
131
|
+
return this._idToArrayIndex.get(id) ?? -1;
|
|
132
|
+
}
|
|
133
|
+
getItemId(item) {
|
|
134
|
+
return this.idFn(item);
|
|
135
|
+
}
|
|
136
|
+
setEntities(entities) {
|
|
137
|
+
if (!(this.entitiesSnapshot.length === 0 && entities.length === 0)) {
|
|
138
|
+
this.log.debug('Entities changed: ' + entities.length, entities);
|
|
139
|
+
this._entities$.next(entities);
|
|
140
|
+
}
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"indexed-entities.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/common/data/data-context/indexed-entities.ts"],"names":[],"mappings":"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","sourcesContent":["import {BehaviorSubject, Observable} from 'rxjs';\nimport {Sort} from '../sort';\nimport {SortContext} from '../sort-context';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\n\nexport class IndexedEntities<T, TId=any> {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger('IndexedEntities');\n\n  private readonly _entities$ = new BehaviorSubject<T[]>([]);\n  private readonly _idToArrayIndex = new Map<TId, number>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly idFn: (e: T) => TId,\n    private readonly localSortFn: ((data: T[], sorts: Sort[]) => T[]) | undefined,\n    private readonly sortContext = new SortContext()\n  ) { }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get entities$(): Observable<T[]> {\n    return this._entities$.asObservable();\n  }\n\n  public get entitiesSnapshot(): T[] {\n    return this._entities$.getValue();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Replaces all existing entities with the new ones.\n   * The index is rebuilt. O(n)\n   */\n  public replaceAll(newEntities: T[] = []) {\n    if (this.localSortFn) {\n      newEntities = this.localSortFn(newEntities, this.sortContext.sortsSnapshot);\n    }\n    this.rebuildIndex(newEntities);\n    this.setEntities(newEntities);\n  }\n\n  /**\n   * Keeps all existing entities, but replaces those which are updated with the new data.\n   * Provided updated entities which are currently not in this index are ignored.\n   * @param updated\n   */\n  public updateSome(updated: T[]) {\n    const newData = [...this.entitiesSnapshot];\n    updated.forEach(\n      update => {\n        const id = this.getItemId(update);\n        const index = this.indexById(id);\n        if (index != -1) {\n          newData[index] = update;\n        }\n      }\n    );\n    this.setEntities(newData);\n  }\n\n\n  public append(append: T[]): void {\n    this.insert(append, this.entitiesSnapshot.length);\n  }\n\n  /**\n   * Appends the given entities to this index.\n   * The new appended entities are included in the index.\n   * O(n) where n = number of appended entities\n   * @param inserted\n   * @param offset\n   */\n  public insert(inserted: T[], offset: number): void {\n    if (inserted && inserted.length > 0) {\n      const entities = this.entitiesSnapshot;\n      const newData = [...entities];\n\n      for (let i = 0; i < inserted.length; i++) {\n        newData[i + offset] = inserted[i];\n      }\n\n      if (this.localSortFn) {\n        // local sort can re-order all items\n        this.replaceAll(newData);\n      } else {\n        // Without a sort, we can optimize indexing\n        this.reIndexFrom(newData, offset);\n        this.setEntities(newData);\n      }\n    }\n  }\n\n  /**\n   * Lookup an entity by its id.\n   * This will use an index O(1)\n   * @param id\n   */\n  public getById(id: TId): T | undefined {\n    const entities = this.entitiesSnapshot;\n    const index = this.indexById(id);\n    if (index != -1) {\n      return entities[index];\n    }\n  }\n\n  public notify(): void {\n    this._entities$.next(\n      this._entities$.getValue()\n    );\n  }\n\n  public destroy() {\n    this._entities$.complete();\n    this._idToArrayIndex.clear();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private rebuildIndex(entities: T[]): void {\n    this._idToArrayIndex.clear();\n    this.reIndexFrom(entities, 0);\n  }\n\n  private reIndexFrom(entities: T[], startIndex: number) {\n    for (let i = startIndex; i < entities.length; i++) {\n      const entity = entities[i];\n      if (entity) {\n        this._idToArrayIndex.set(this.getItemId(entity), i);\n      }\n    }\n  }\n\n  private indexById(id: TId): number | undefined {\n    return this._idToArrayIndex.get(id) ?? -1;\n  }\n\n  private getItemId(item: T): TId {\n    return this.idFn(item);\n  }\n\n  private setEntities(entities: T[]): void {\n    if (!(this.entitiesSnapshot.length === 0 && entities.length === 0)) {\n      this.log.debug('Entities changed: ' + entities.length, entities);\n      this._entities$.next(entities);\n    }\n  }\n\n}\n"]}
|
|
@@ -1,3 +1,4 @@
|
|
|
1
|
+
export * from './indexed-entities';
|
|
1
2
|
export * from './data-context-status';
|
|
2
3
|
export * from './data-context';
|
|
3
4
|
export * from './data-context-base';
|
|
@@ -9,6 +10,6 @@ export * from './data-context-active-page';
|
|
|
9
10
|
export * from './mat-table-data-context-binding';
|
|
10
11
|
export * from './data-context-life-cycle-binding';
|
|
11
12
|
export * from './data-context-auto-starter';
|
|
12
|
-
export * from './data-context-source-
|
|
13
|
+
export * from './data-context-source-event-binding';
|
|
13
14
|
export * from './data-context-builder';
|
|
14
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
15
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicHVibGljX2FwaS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbW1vbi9kYXRhL2RhdGEtY29udGV4dC9wdWJsaWNfYXBpLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsb0JBQW9CLENBQUM7QUFFbkMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMscUJBQXFCLENBQUM7QUFDcEMsY0FBYyx1QkFBdUIsQ0FBQztBQUN0QyxjQUFjLGlDQUFpQyxDQUFDO0FBQ2hELGNBQWMsa0NBQWtDLENBQUM7QUFDakQsY0FBYyxrQ0FBa0MsQ0FBQztBQUNqRCxjQUFjLDRCQUE0QixDQUFDO0FBRTNDLGNBQWMsa0NBQWtDLENBQUM7QUFDakQsY0FBYyxtQ0FBbUMsQ0FBQztBQUNsRCxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMscUNBQXFDLENBQUM7QUFFcEQsY0FBYyx3QkFBd0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCAqIGZyb20gJy4vaW5kZXhlZC1lbnRpdGllcyc7XG5cbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LXN0YXR1cyc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGEtY29udGV4dCc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGEtY29udGV4dC1iYXNlJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LXNpbXBsZSc7XG5leHBvcnQgKiBmcm9tICcuL2RhdGEtY29udGV4dC1jb250aW51YWJsZS1iYXNlJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LWNvbnRpbnVhYmxlLXBhZ2VkJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LWNvbnRpbnVhYmxlLXRva2VuJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LWFjdGl2ZS1wYWdlJztcblxuZXhwb3J0ICogZnJvbSAnLi9tYXQtdGFibGUtZGF0YS1jb250ZXh0LWJpbmRpbmcnO1xuZXhwb3J0ICogZnJvbSAnLi9kYXRhLWNvbnRleHQtbGlmZS1jeWNsZS1iaW5kaW5nJztcbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LWF1dG8tc3RhcnRlcic7XG5leHBvcnQgKiBmcm9tICcuL2RhdGEtY29udGV4dC1zb3VyY2UtZXZlbnQtYmluZGluZyc7XG5cbmV4cG9ydCAqIGZyb20gJy4vZGF0YS1jb250ZXh0LWJ1aWxkZXInO1xuXG5cblxuIl19
|
|
@@ -0,0 +1,38 @@
|
|
|
1
|
+
import { Subject } from 'rxjs';
|
|
2
|
+
import { EntityIdUtil } from './entity-id-util';
|
|
3
|
+
export class DataSourceBase {
|
|
4
|
+
/***************************************************************************
|
|
5
|
+
* *
|
|
6
|
+
* Constructor *
|
|
7
|
+
* *
|
|
8
|
+
**************************************************************************/
|
|
9
|
+
constructor(idProperty) {
|
|
10
|
+
this.idProperty = idProperty;
|
|
11
|
+
/***************************************************************************
|
|
12
|
+
* *
|
|
13
|
+
* Fields *
|
|
14
|
+
* *
|
|
15
|
+
**************************************************************************/
|
|
16
|
+
this.dataChangeEvents$ = new Subject();
|
|
17
|
+
}
|
|
18
|
+
/***************************************************************************
|
|
19
|
+
* *
|
|
20
|
+
* Properties *
|
|
21
|
+
* *
|
|
22
|
+
**************************************************************************/
|
|
23
|
+
get dataChanged() {
|
|
24
|
+
return this.dataChangeEvents$.asObservable();
|
|
25
|
+
}
|
|
26
|
+
/***************************************************************************
|
|
27
|
+
* *
|
|
28
|
+
* Public API *
|
|
29
|
+
* *
|
|
30
|
+
**************************************************************************/
|
|
31
|
+
publishChangeEvent(e) {
|
|
32
|
+
this.dataChangeEvents$.next(e);
|
|
33
|
+
}
|
|
34
|
+
getId(entity) {
|
|
35
|
+
return EntityIdUtil.getId(entity, this.idProperty);
|
|
36
|
+
}
|
|
37
|
+
}
|
|
38
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YS1zb3VyY2UtYmFzZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbW1vbi9kYXRhL2RhdGFzb3VyY2UvZGF0YS1zb3VyY2UtYmFzZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFFQSxPQUFPLEVBQWEsT0FBTyxFQUFDLE1BQU0sTUFBTSxDQUFDO0FBQ3pDLE9BQU8sRUFBQyxZQUFZLEVBQUMsTUFBTSxrQkFBa0IsQ0FBQztBQUU5QyxNQUFNLE9BQWdCLGNBQWM7SUFVbEM7Ozs7Z0ZBSTRFO0lBRTVFLFlBQ3FCLFVBQXlCO1FBQXpCLGVBQVUsR0FBVixVQUFVLENBQWU7UUFmOUM7Ozs7b0ZBSTRFO1FBRTNELHNCQUFpQixHQUFHLElBQUksT0FBTyxFQUE0QixDQUFDO0lBVzdFLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFNUUsSUFBVyxXQUFXO1FBQ3BCLE9BQU8sSUFBSSxDQUFDLGlCQUFpQixDQUFDLFlBQVksRUFBRSxDQUFDO0lBQy9DLENBQUM7SUFFRDs7OztnRkFJNEU7SUFFckUsa0JBQWtCLENBQUMsQ0FBMkI7UUFDbkQsSUFBSSxDQUFDLGlCQUFpQixDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNqQyxDQUFDO0lBRU0sS0FBSyxDQUFDLE1BQVM7UUFDcEIsT0FBTyxZQUFZLENBQUMsS0FBSyxDQUFDLE1BQU0sRUFBRSxJQUFJLENBQUMsVUFBVSxDQUFDLENBQUM7SUFDckQsQ0FBQztDQVlGIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtJRGF0YVNvdXJjZX0gZnJvbSAnLi9kYXRhLXNvdXJjZSc7XG5pbXBvcnQge0RhdGFTb3VyY2VDaGFuZ2VFdmVudH0gZnJvbSAnLi9kYXRhLXNvdXJjZS1jaGFuZ2UtZXZlbnQnO1xuaW1wb3J0IHtPYnNlcnZhYmxlLCBTdWJqZWN0fSBmcm9tICdyeGpzJztcbmltcG9ydCB7RW50aXR5SWRVdGlsfSBmcm9tICcuL2VudGl0eS1pZC11dGlsJztcblxuZXhwb3J0IGFic3RyYWN0IGNsYXNzIERhdGFTb3VyY2VCYXNlPFQ+IGltcGxlbWVudHMgSURhdGFTb3VyY2U8VD4ge1xuXG4gIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiBGaWVsZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gIHByaXZhdGUgcmVhZG9ubHkgZGF0YUNoYW5nZUV2ZW50cyQgPSBuZXcgU3ViamVjdDxEYXRhU291cmNlQ2hhbmdlRXZlbnQ8VD4+KCk7XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIENvbnN0cnVjdG9yICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHJvdGVjdGVkIHJlYWRvbmx5IGlkUHJvcGVydHk6IHN0cmluZyB8IG51bGxcbiAgKSB7XG4gIH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHJvcGVydGllcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwdWJsaWMgZ2V0IGRhdGFDaGFuZ2VkKCk6IE9ic2VydmFibGU8RGF0YVNvdXJjZUNoYW5nZUV2ZW50PFQ+PiB7XG4gICAgcmV0dXJuIHRoaXMuZGF0YUNoYW5nZUV2ZW50cyQuYXNPYnNlcnZhYmxlKCk7XG4gIH1cblxuICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogUHVibGljIEFQSSAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICBwdWJsaWMgcHVibGlzaENoYW5nZUV2ZW50KGU6IERhdGFTb3VyY2VDaGFuZ2VFdmVudDxUPik6IHZvaWQge1xuICAgIHRoaXMuZGF0YUNoYW5nZUV2ZW50cyQubmV4dChlKTtcbiAgfVxuXG4gIHB1YmxpYyBnZXRJZChlbnRpdHk6IFQpOiBhbnkge1xuICAgIHJldHVybiBFbnRpdHlJZFV0aWwuZ2V0SWQoZW50aXR5LCB0aGlzLmlkUHJvcGVydHkpO1xuICB9XG5cbiAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAqIEFic3RyYWN0IG1ldGhvZHMgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgYWJzdHJhY3QgZmluZEJ5SWQoaWQ6IGFueSk6IE9ic2VydmFibGU8VD47XG5cbiAgYWJzdHJhY3QgZmluZEJ5SWRzKGlkczogYW55W10pOiBPYnNlcnZhYmxlPFRbXT47XG5cbn1cbiJdfQ==
|
|
@@ -0,0 +1,24 @@
|
|
|
1
|
+
export var DataSourceChangeType;
|
|
2
|
+
(function (DataSourceChangeType) {
|
|
3
|
+
})(DataSourceChangeType || (DataSourceChangeType = {}));
|
|
4
|
+
export class DataSourceChangeEvent {
|
|
5
|
+
constructor(deletedIds, modified, created) {
|
|
6
|
+
this.deletedIds = deletedIds;
|
|
7
|
+
this.modified = modified;
|
|
8
|
+
this.created = created;
|
|
9
|
+
this.unknownChanges = !deletedIds && !modified && !created;
|
|
10
|
+
}
|
|
11
|
+
static unknownChanges() {
|
|
12
|
+
return new DataSourceChangeEvent(null, null, null);
|
|
13
|
+
}
|
|
14
|
+
static deleted(ids) {
|
|
15
|
+
return new DataSourceChangeEvent(ids, null, null);
|
|
16
|
+
}
|
|
17
|
+
static modified(modified) {
|
|
18
|
+
return new DataSourceChangeEvent(null, modified, null);
|
|
19
|
+
}
|
|
20
|
+
static created(created) {
|
|
21
|
+
return new DataSourceChangeEvent(null, null, created);
|
|
22
|
+
}
|
|
23
|
+
}
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZGF0YS1zb3VyY2UtY2hhbmdlLWV2ZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tbW9uL2RhdGEvZGF0YXNvdXJjZS9kYXRhLXNvdXJjZS1jaGFuZ2UtZXZlbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsTUFBTSxDQUFOLElBQVksb0JBRVg7QUFGRCxXQUFZLG9CQUFvQjtBQUVoQyxDQUFDLEVBRlcsb0JBQW9CLEtBQXBCLG9CQUFvQixRQUUvQjtBQUdELE1BQU0sT0FBTyxxQkFBcUI7SUE0QmhDLFlBQ2tCLFVBQWlCLEVBQ2pCLFFBQWEsRUFDYixPQUFZO1FBRlosZUFBVSxHQUFWLFVBQVUsQ0FBTztRQUNqQixhQUFRLEdBQVIsUUFBUSxDQUFLO1FBQ2IsWUFBTyxHQUFQLE9BQU8sQ0FBSztRQUU1QixJQUFJLENBQUMsY0FBYyxHQUFHLENBQUMsVUFBVSxJQUFJLENBQUMsUUFBUSxJQUFJLENBQUMsT0FBTyxDQUFDO0lBQzdELENBQUM7SUFoQ00sTUFBTSxDQUFFLGNBQWM7UUFDM0IsT0FBTyxJQUFJLHFCQUFxQixDQUFJLElBQUksRUFBRSxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUM7SUFDeEQsQ0FBQztJQUVNLE1BQU0sQ0FBRSxPQUFPLENBQU0sR0FBVTtRQUNwQyxPQUFPLElBQUkscUJBQXFCLENBQVcsR0FBRyxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsQ0FBQztJQUM5RCxDQUFDO0lBRU0sTUFBTSxDQUFFLFFBQVEsQ0FBSSxRQUFhO1FBQ3RDLE9BQU8sSUFBSSxxQkFBcUIsQ0FBSSxJQUFJLEVBQUUsUUFBUSxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQzVELENBQUM7SUFFTSxNQUFNLENBQUUsT0FBTyxDQUFJLE9BQVk7UUFDcEMsT0FBTyxJQUFJLHFCQUFxQixDQUFJLElBQUksRUFBRSxJQUFJLEVBQUUsT0FBTyxDQUFDLENBQUM7SUFDM0QsQ0FBQztDQXNCRiIsInNvdXJjZXNDb250ZW50IjpbIlxuZXhwb3J0IGVudW0gRGF0YVNvdXJjZUNoYW5nZVR5cGUge1xuXG59XG5cblxuZXhwb3J0IGNsYXNzIERhdGFTb3VyY2VDaGFuZ2VFdmVudDxULCBUSWQgPSBhbnk+IHtcblxuICBwdWJsaWMgc3RhdGljICB1bmtub3duQ2hhbmdlczxUPigpOiBEYXRhU291cmNlQ2hhbmdlRXZlbnQ8VD4ge1xuICAgIHJldHVybiBuZXcgRGF0YVNvdXJjZUNoYW5nZUV2ZW50PFQ+KG51bGwsIG51bGwsIG51bGwpO1xuICB9XG5cbiAgcHVibGljIHN0YXRpYyAgZGVsZXRlZDxUSWQ+KGlkczogVElkW10pOiBEYXRhU291cmNlQ2hhbmdlRXZlbnQ8YW55LCBUSWQ+IHtcbiAgICByZXR1cm4gbmV3IERhdGFTb3VyY2VDaGFuZ2VFdmVudDxhbnksIFRJZD4oaWRzLCBudWxsLCBudWxsKTtcbiAgfVxuXG4gIHB1YmxpYyBzdGF0aWMgIG1vZGlmaWVkPFQ+KG1vZGlmaWVkOiBUW10pOiBEYXRhU291cmNlQ2hhbmdlRXZlbnQ8VD4ge1xuICAgIHJldHVybiBuZXcgRGF0YVNvdXJjZUNoYW5nZUV2ZW50PFQ+KG51bGwsIG1vZGlmaWVkLCBudWxsKTtcbiAgfVxuXG4gIHB1YmxpYyBzdGF0aWMgIGNyZWF0ZWQ8VD4oY3JlYXRlZDogVFtdKTogRGF0YVNvdXJjZUNoYW5nZUV2ZW50PFQ+IHtcbiAgICByZXR1cm4gbmV3IERhdGFTb3VyY2VDaGFuZ2VFdmVudDxUPihudWxsLCBudWxsLCBjcmVhdGVkKTtcbiAgfVxuXG4gIC8vIGVudGl0aWVzIC0+IGRhdGEgbXVzdCBiZSB1cGRhdGVkIHdpdGggbmV3IHZlcnNpb25zXG4gIC8vIGVudGl0eS1pZHMgLT4gbXVzdCBiZSBkb3dubG9hZGVkIGZpbmRCeUlkcygpIGFuZCBkYXRhIHVwZGF0ZWRcbiAgLy8gZW50aXR5LXBhdGNoZXMgLT4gZW50aXR5IG11c3QgYmUgbG9va2VkIHVwICYgYXBwbGllZCBsb2NhbGx5ICYgcmVwbGFjZWRcblxuICAvLyBjb25zdCBlbnRpdHkgPSBkYy5maW5kQnlJZChlbnRpdHktaWQpO1xuICAvLyBkYy5yZXBsYWNlKGVudGl0eSk7XG4gIC8vIGRjLnJlcGxhY2VBbGwoZW50aXRpZXMpO1xuXG4gIHB1YmxpYyByZWFkb25seSB1bmtub3duQ2hhbmdlczogYm9vbGVhbjtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgcmVhZG9ubHkgZGVsZXRlZElkczogVElkW10sXG4gICAgcHVibGljIHJlYWRvbmx5IG1vZGlmaWVkOiBUW10sXG4gICAgcHVibGljIHJlYWRvbmx5IGNyZWF0ZWQ6IFRbXVxuICApIHtcbiAgICB0aGlzLnVua25vd25DaGFuZ2VzID0gIWRlbGV0ZWRJZHMgJiYgIW1vZGlmaWVkICYmICFjcmVhdGVkO1xuICB9XG5cblxuXG59XG4iXX0=
|