@dereekb/dbx-web 12.1.6 → 12.1.8

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.
Files changed (43) hide show
  1. package/esm2022/lib/extension/download/text/download.text.component.mjs +18 -14
  2. package/esm2022/table/lib/index.mjs +4 -1
  3. package/esm2022/table/lib/table.cell.action.component.mjs +2 -2
  4. package/esm2022/table/lib/table.component.mjs +99 -8
  5. package/esm2022/table/lib/table.fullsummaryrow.component.mjs +34 -0
  6. package/esm2022/table/lib/table.group.directive.mjs +20 -0
  7. package/esm2022/table/lib/table.group.footer.component.mjs +35 -0
  8. package/esm2022/table/lib/table.group.header.component.mjs +35 -0
  9. package/esm2022/table/lib/table.item.action.component.mjs +4 -4
  10. package/esm2022/table/lib/table.item.cell.component.mjs +4 -4
  11. package/esm2022/table/lib/table.item.directive.mjs +7 -7
  12. package/esm2022/table/lib/table.item.header.component.mjs +4 -4
  13. package/esm2022/table/lib/table.mjs +10 -2
  14. package/esm2022/table/lib/table.module.mjs +4 -16
  15. package/esm2022/table/lib/table.store.mjs +16 -2
  16. package/fesm2022/dereekb-dbx-web-table.mjs +232 -27
  17. package/fesm2022/dereekb-dbx-web-table.mjs.map +1 -1
  18. package/fesm2022/dereekb-dbx-web.mjs +15 -11
  19. package/fesm2022/dereekb-dbx-web.mjs.map +1 -1
  20. package/lib/extension/download/text/download.text.component.d.ts +7 -9
  21. package/lib/extension/table/_table.scss +7 -0
  22. package/lib/layout/style/_style.scss +4 -0
  23. package/package.json +1 -1
  24. package/table/lib/date/daterange.table.cell.input.component.d.ts +1 -1
  25. package/table/lib/index.d.ts +3 -0
  26. package/table/lib/table.cell.action.component.d.ts +1 -1
  27. package/table/lib/table.cell.input.component.d.ts +1 -1
  28. package/table/lib/table.cell.summaryend.component.d.ts +1 -1
  29. package/table/lib/table.cell.summarystart.component.d.ts +1 -1
  30. package/table/lib/table.column.directive.d.ts +1 -1
  31. package/table/lib/table.component.d.ts +38 -15
  32. package/table/lib/table.d.ts +47 -7
  33. package/table/lib/table.directive.d.ts +2 -2
  34. package/table/lib/table.fullsummaryrow.component.d.ts +12 -0
  35. package/table/lib/table.group.directive.d.ts +14 -0
  36. package/table/lib/table.group.footer.component.d.ts +9 -0
  37. package/table/lib/table.group.header.component.d.ts +9 -0
  38. package/table/lib/table.item.action.component.d.ts +2 -2
  39. package/table/lib/table.item.cell.component.d.ts +2 -2
  40. package/table/lib/table.item.directive.d.ts +6 -6
  41. package/table/lib/table.item.header.component.d.ts +2 -2
  42. package/table/lib/table.module.d.ts +1 -13
  43. package/table/lib/table.store.d.ts +11 -9
@@ -1,8 +1,8 @@
1
1
  import { ChangeDetectionStrategy, Component, ElementRef, computed, inject, input, viewChild } from '@angular/core';
2
2
  import { MatSnackBar } from '@angular/material/snack-bar';
3
- import { loadingStateContext, successResult, valueFromFinishedLoadingState } from '@dereekb/rxjs';
3
+ import { loadingStateContext, successResult, valueFromFinishedLoadingState, maybeValueFromObservableOrValue } from '@dereekb/rxjs';
4
4
  import { MS_IN_SECOND } from '@dereekb/util';
5
- import { combineLatest, distinctUntilChanged, first, map, of, shareReplay, switchMap, tap } from 'rxjs';
5
+ import { first, of, shareReplay, switchMap, tap } from 'rxjs';
6
6
  import { Clipboard } from '@angular/cdk/clipboard';
7
7
  import { DomSanitizer } from '@angular/platform-browser';
8
8
  import { AbstractSubscriptionDirective, DbxActionButtonDirective } from '@dereekb/dbx-core';
@@ -28,9 +28,11 @@ export class DbxDownloadTextViewComponent extends AbstractSubscriptionDirective
28
28
  showPreview = input(true);
29
29
  content = input(undefined);
30
30
  contentState = input(undefined);
31
+ contentState$ = toObservable(this.contentState).pipe(maybeValueFromObservableOrValue(), shareReplay(1));
32
+ contentStateSignal = toSignal(this.contentState$);
31
33
  contentLoadingStateSignal = computed(() => {
32
34
  const content = this.content();
33
- const contentState = this.contentState();
35
+ const contentState = this.contentStateSignal();
34
36
  let result;
35
37
  if (contentState) {
36
38
  result = contentState;
@@ -49,22 +51,24 @@ export class DbxDownloadTextViewComponent extends AbstractSubscriptionDirective
49
51
  return of(undefined);
50
52
  }
51
53
  }));
52
- contentData$ = this.content$.pipe(map((x) => x?.content));
53
- fileName$ = this.content$.pipe(map((x) => x?.name ?? 'File'), shareReplay(1));
54
- fileUrl$ = this.content$.pipe(map((content) => {
54
+ contentSignal = toSignal(this.content$);
55
+ contentDataSignal = computed(() => this.contentSignal()?.content);
56
+ fileUrlSignal = computed(() => {
57
+ const content = this.contentSignal();
55
58
  let fileUrl = undefined;
56
59
  if (content) {
57
60
  const blob = new Blob([content.content], { type: content.mimeType ?? 'application/octet-stream' });
58
61
  fileUrl = this._sanitizer.bypassSecurityTrustResourceUrl(window.URL.createObjectURL(blob));
59
62
  }
60
63
  return fileUrl;
61
- }), shareReplay(1));
62
- downloadReady$ = combineLatest([toObservable(this.downloadButton), this.fileName$, this.fileUrl$]).pipe(map(([button, name, url]) => Boolean(button && name && url)), distinctUntilChanged(), shareReplay(1));
63
- contentSignal = toSignal(this.content$);
64
- contentDataSignal = toSignal(this.contentData$);
65
- fileNameSignal = toSignal(this.fileName$);
66
- fileUrlSignal = toSignal(this.fileUrl$);
67
- downloadReadySignal = toSignal(this.downloadReady$);
64
+ });
65
+ fileNameSignal = computed(() => this.contentSignal()?.name ?? 'File');
66
+ downloadReadySignal = computed(() => {
67
+ const downloadButton = this.downloadButton();
68
+ const fileName = this.fileNameSignal();
69
+ const fileUrl = this.fileUrlSignal();
70
+ return Boolean(downloadButton && fileName && fileUrl);
71
+ });
68
72
  context = loadingStateContext({ obs: this.contentLoadingState$ });
69
73
  handleCopyToClipboard = () => {
70
74
  return this.content$.pipe(first(), switchMap((downloadTextContent) => {
@@ -110,4 +114,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
110
114
  type: Component,
111
115
  args: [{ selector: 'dbx-download-text-view', standalone: true, imports: [NgTemplateOutlet, DbxLoadingComponent, DbxActionModule, DbxActionButtonDirective, DbxButtonComponent, DbxButtonSpacerDirective], changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"dbx-download-text-view\">\n <dbx-loading [context]=\"context\" [linear]=\"linear()\" [text]=\"loadingText()\">\n <ng-container *ngTemplateOutlet=\"contentView\"></ng-container>\n </dbx-loading>\n</div>\n\n<!-- Template -->\n<ng-template #contentView>\n <div class=\"dbx-download-text-view-content\">\n @if (showTitle()) {\n <div class=\"dbx-download-text-view-content-title dbx-mb2 mat-subtitle-2\">{{ fileNameSignal() }}</div>\n }\n @if (showPreview()) {\n <div class=\"dbx-download-text-preview dbx-json dbx-content-pit dbx-content-pit-scrollable dbx-mb3\">{{ contentDataSignal() }}</div>\n }\n <div class=\"dbx-download-text-view-actions\">\n <dbx-button dbxAction dbxActionValue [raised]=\"true\" [dbxActionHandler]=\"handleCopyToClipboard\" icon=\"content_copy\" dbxActionButton text=\"Copy To Clipboard\"></dbx-button>\n <dbx-button-spacer></dbx-button-spacer>\n <a #downloadButton [href]=\"fileUrlSignal()\" [attr.download]=\"fileNameSignal()\"><dbx-button icon=\"download\" text=\"Download\" [raised]=\"true\" [working]=\"!downloadReadySignal()\" [disabled]=\"!downloadReadySignal()\"></dbx-button></a>\n </div>\n </div>\n</ng-template>\n" }]
112
116
  }] });
113
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"download.text.component.js","sourceRoot":"","sources":["../../../../../../../../packages/dbx-web/src/lib/extension/download/text/download.text.component.ts","../../../../../../../../packages/dbx-web/src/lib/extension/download/text/download.text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, computed, inject, input, viewChild } from '@angular/core';\nimport { MatSnackBar } from '@angular/material/snack-bar';\nimport { WorkUsingObservable, LoadingState, loadingStateContext, successResult, valueFromFinishedLoadingState } from '@dereekb/rxjs';\nimport { MS_IN_SECOND, type Maybe } from '@dereekb/util';\nimport { Observable, combineLatest, distinctUntilChanged, first, map, of, shareReplay, switchMap, tap } from 'rxjs';\nimport { Clipboard } from '@angular/cdk/clipboard';\nimport { DownloadTextContent } from './download.text';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { AbstractSubscriptionDirective, DbxActionButtonDirective } from '@dereekb/dbx-core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { DbxLoadingComponent } from '../../../loading/loading.component';\nimport { NgTemplateOutlet } from '@angular/common';\nimport { DbxButtonComponent } from '../../../button/button.component';\nimport { DbxActionModule } from '../../../action/action.module';\nimport { DbxButtonSpacerDirective } from '../../../button/button.spacer.directive';\n\n/**\n * DbxStructureDirective used specifically on the body of the app.\n */\n@Component({\n  templateUrl: './download.text.component.html',\n  selector: 'dbx-download-text-view',\n  standalone: true,\n  imports: [NgTemplateOutlet, DbxLoadingComponent, DbxActionModule, DbxActionButtonDirective, DbxButtonComponent, DbxButtonSpacerDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DbxDownloadTextViewComponent extends AbstractSubscriptionDirective {\n  private readonly _clipboard = inject(Clipboard);\n  private readonly _matSnackbar = inject(MatSnackBar);\n  private readonly _sanitizer = inject(DomSanitizer);\n\n  readonly downloadButton = viewChild<string, Maybe<ElementRef>>('downloadButton', { read: ElementRef });\n\n  readonly loadingText = input<Maybe<string>>(undefined);\n  readonly linear = input<Maybe<boolean>>(undefined);\n  readonly showTitle = input<boolean>(true);\n  readonly showPreview = input<boolean>(true);\n\n  readonly content = input<Maybe<DownloadTextContent>>(undefined);\n  readonly contentState = input<Maybe<LoadingState<DownloadTextContent>>>(undefined);\n\n  readonly contentLoadingStateSignal = computed(() => {\n    const content = this.content();\n    const contentState = this.contentState();\n\n    let result: Maybe<LoadingState<DownloadTextContent>>;\n\n    if (contentState) {\n      result = contentState;\n    } else if (content) {\n      result = successResult(content);\n    }\n\n    return result;\n  });\n\n  readonly contentLoadingState$ = toObservable(this.contentLoadingStateSignal);\n  readonly content$: Observable<Maybe<DownloadTextContent>> = this.contentLoadingState$.pipe(\n    switchMap((x) => {\n      if (x) {\n        return of(x).pipe(valueFromFinishedLoadingState());\n      } else {\n        return of(undefined);\n      }\n    })\n  );\n\n  readonly contentData$ = this.content$.pipe(map((x) => x?.content));\n\n  readonly fileName$ = this.content$.pipe(\n    map((x) => x?.name ?? 'File'),\n    shareReplay(1)\n  );\n\n  readonly fileUrl$: Observable<Maybe<SafeResourceUrl>> = this.content$.pipe(\n    map((content) => {\n      let fileUrl: Maybe<SafeResourceUrl> = undefined;\n\n      if (content) {\n        const blob = new Blob([content.content], { type: content.mimeType ?? 'application/octet-stream' });\n        fileUrl = this._sanitizer.bypassSecurityTrustResourceUrl(window.URL.createObjectURL(blob));\n      }\n\n      return fileUrl;\n    }),\n    shareReplay(1)\n  );\n\n  readonly downloadReady$ = combineLatest([toObservable(this.downloadButton), this.fileName$, this.fileUrl$]).pipe(\n    map(([button, name, url]) => Boolean(button && name && url)),\n    distinctUntilChanged(),\n    shareReplay(1)\n  );\n\n  readonly contentSignal = toSignal(this.content$);\n  readonly contentDataSignal = toSignal(this.contentData$);\n\n  readonly fileNameSignal = toSignal(this.fileName$);\n  readonly fileUrlSignal = toSignal(this.fileUrl$);\n  readonly downloadReadySignal = toSignal(this.downloadReady$);\n\n  readonly context = loadingStateContext({ obs: this.contentLoadingState$ });\n\n  readonly handleCopyToClipboard: WorkUsingObservable = () => {\n    return this.content$.pipe(\n      first(),\n      switchMap((downloadTextContent: Maybe<DownloadTextContent>) => {\n        if (downloadTextContent) {\n          return new Promise<boolean>((resolve, reject) => {\n            const pending = this._clipboard.beginCopy(downloadTextContent.content);\n\n            let secondsRemainingForCopy = 20; // attempt to copy for up to 20 seconds\n\n            const attempt = () => {\n              const copyIsFinished = pending.copy();\n\n              if (!copyIsFinished && --secondsRemainingForCopy) {\n                setTimeout(attempt, MS_IN_SECOND);\n              } else {\n                // Remember to destroy when you're done!\n                pending.destroy();\n\n                if (copyIsFinished) {\n                  resolve(true);\n                } else {\n                  reject(false);\n                }\n              }\n            };\n\n            attempt();\n          });\n        } else {\n          return of(false);\n        }\n      }),\n      tap((success) => {\n        if (success) {\n          this._matSnackbar.open('Copied to clipboard', undefined, { duration: 3 * MS_IN_SECOND });\n        } else {\n          this._matSnackbar.open('Content failed to copy...', undefined, { duration: 3 * MS_IN_SECOND });\n        }\n      })\n    );\n  };\n}\n","<div class=\"dbx-download-text-view\">\n  <dbx-loading [context]=\"context\" [linear]=\"linear()\" [text]=\"loadingText()\">\n    <ng-container *ngTemplateOutlet=\"contentView\"></ng-container>\n  </dbx-loading>\n</div>\n\n<!-- Template -->\n<ng-template #contentView>\n  <div class=\"dbx-download-text-view-content\">\n    @if (showTitle()) {\n      <div class=\"dbx-download-text-view-content-title dbx-mb2 mat-subtitle-2\">{{ fileNameSignal() }}</div>\n    }\n    @if (showPreview()) {\n      <div class=\"dbx-download-text-preview dbx-json dbx-content-pit dbx-content-pit-scrollable dbx-mb3\">{{ contentDataSignal() }}</div>\n    }\n    <div class=\"dbx-download-text-view-actions\">\n      <dbx-button dbxAction dbxActionValue [raised]=\"true\" [dbxActionHandler]=\"handleCopyToClipboard\" icon=\"content_copy\" dbxActionButton text=\"Copy To Clipboard\"></dbx-button>\n      <dbx-button-spacer></dbx-button-spacer>\n      <a #downloadButton [href]=\"fileUrlSignal()\" [attr.download]=\"fileNameSignal()\"><dbx-button icon=\"download\" text=\"Download\" [raised]=\"true\" [working]=\"!downloadReadySignal()\" [disabled]=\"!downloadReadySignal()\"></dbx-button></a>\n    </div>\n  </div>\n</ng-template>\n"]}
117
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"download.text.component.js","sourceRoot":"","sources":["../../../../../../../../packages/dbx-web/src/lib/extension/download/text/download.text.component.ts","../../../../../../../../packages/dbx-web/src/lib/extension/download/text/download.text.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, ElementRef, computed, inject, input, viewChild } from '@angular/core';\nimport { MatSnackBar } from '@angular/material/snack-bar';\nimport { WorkUsingObservable, LoadingState, loadingStateContext, successResult, valueFromFinishedLoadingState, MaybeObservableOrValue, maybeValueFromObservableOrValue } from '@dereekb/rxjs';\nimport { MS_IN_SECOND, type Maybe } from '@dereekb/util';\nimport { Observable, first, of, shareReplay, switchMap, tap } from 'rxjs';\nimport { Clipboard } from '@angular/cdk/clipboard';\nimport { DownloadTextContent } from './download.text';\nimport { DomSanitizer, SafeResourceUrl } from '@angular/platform-browser';\nimport { AbstractSubscriptionDirective, DbxActionButtonDirective } from '@dereekb/dbx-core';\nimport { toObservable, toSignal } from '@angular/core/rxjs-interop';\nimport { DbxLoadingComponent } from '../../../loading/loading.component';\nimport { NgTemplateOutlet } from '@angular/common';\nimport { DbxButtonComponent } from '../../../button/button.component';\nimport { DbxActionModule } from '../../../action/action.module';\nimport { DbxButtonSpacerDirective } from '../../../button/button.spacer.directive';\n\n/**\n * DbxStructureDirective used specifically on the body of the app.\n */\n@Component({\n  templateUrl: './download.text.component.html',\n  selector: 'dbx-download-text-view',\n  standalone: true,\n  imports: [NgTemplateOutlet, DbxLoadingComponent, DbxActionModule, DbxActionButtonDirective, DbxButtonComponent, DbxButtonSpacerDirective],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DbxDownloadTextViewComponent extends AbstractSubscriptionDirective {\n  private readonly _clipboard = inject(Clipboard);\n  private readonly _matSnackbar = inject(MatSnackBar);\n  private readonly _sanitizer = inject(DomSanitizer);\n\n  readonly downloadButton = viewChild<string, Maybe<ElementRef>>('downloadButton', { read: ElementRef });\n\n  readonly loadingText = input<Maybe<string>>(undefined);\n  readonly linear = input<Maybe<boolean>>(undefined);\n  readonly showTitle = input<boolean>(true);\n  readonly showPreview = input<boolean>(true);\n\n  readonly content = input<Maybe<DownloadTextContent>>(undefined);\n  readonly contentState = input<MaybeObservableOrValue<LoadingState<DownloadTextContent>>>(undefined);\n\n  readonly contentState$ = toObservable(this.contentState).pipe(maybeValueFromObservableOrValue(), shareReplay(1));\n\n  readonly contentStateSignal = toSignal(this.contentState$);\n\n  readonly contentLoadingStateSignal = computed(() => {\n    const content = this.content();\n    const contentState = this.contentStateSignal();\n\n    let result: Maybe<LoadingState<DownloadTextContent>>;\n\n    if (contentState) {\n      result = contentState;\n    } else if (content) {\n      result = successResult(content);\n    }\n\n    return result;\n  });\n\n  readonly contentLoadingState$ = toObservable(this.contentLoadingStateSignal);\n  readonly content$: Observable<Maybe<DownloadTextContent>> = this.contentLoadingState$.pipe(\n    switchMap((x) => {\n      if (x) {\n        return of(x).pipe(valueFromFinishedLoadingState());\n      } else {\n        return of(undefined);\n      }\n    })\n  );\n\n  readonly contentSignal = toSignal(this.content$);\n\n  readonly contentDataSignal = computed(() => this.contentSignal()?.content);\n\n  readonly fileUrlSignal = computed(() => {\n    const content = this.contentSignal();\n\n    let fileUrl: Maybe<SafeResourceUrl> = undefined;\n\n    if (content) {\n      const blob = new Blob([content.content], { type: content.mimeType ?? 'application/octet-stream' });\n      fileUrl = this._sanitizer.bypassSecurityTrustResourceUrl(window.URL.createObjectURL(blob));\n    }\n\n    return fileUrl;\n  });\n\n  readonly fileNameSignal = computed(() => this.contentSignal()?.name ?? 'File');\n\n  readonly downloadReadySignal = computed(() => {\n    const downloadButton = this.downloadButton();\n    const fileName = this.fileNameSignal();\n    const fileUrl = this.fileUrlSignal();\n    return Boolean(downloadButton && fileName && fileUrl);\n  });\n\n  readonly context = loadingStateContext({ obs: this.contentLoadingState$ });\n\n  readonly handleCopyToClipboard: WorkUsingObservable = () => {\n    return this.content$.pipe(\n      first(),\n      switchMap((downloadTextContent: Maybe<DownloadTextContent>) => {\n        if (downloadTextContent) {\n          return new Promise<boolean>((resolve, reject) => {\n            const pending = this._clipboard.beginCopy(downloadTextContent.content);\n\n            let secondsRemainingForCopy = 20; // attempt to copy for up to 20 seconds\n\n            const attempt = () => {\n              const copyIsFinished = pending.copy();\n\n              if (!copyIsFinished && --secondsRemainingForCopy) {\n                setTimeout(attempt, MS_IN_SECOND);\n              } else {\n                // Remember to destroy when you're done!\n                pending.destroy();\n\n                if (copyIsFinished) {\n                  resolve(true);\n                } else {\n                  reject(false);\n                }\n              }\n            };\n\n            attempt();\n          });\n        } else {\n          return of(false);\n        }\n      }),\n      tap((success) => {\n        if (success) {\n          this._matSnackbar.open('Copied to clipboard', undefined, { duration: 3 * MS_IN_SECOND });\n        } else {\n          this._matSnackbar.open('Content failed to copy...', undefined, { duration: 3 * MS_IN_SECOND });\n        }\n      })\n    );\n  };\n}\n","<div class=\"dbx-download-text-view\">\n  <dbx-loading [context]=\"context\" [linear]=\"linear()\" [text]=\"loadingText()\">\n    <ng-container *ngTemplateOutlet=\"contentView\"></ng-container>\n  </dbx-loading>\n</div>\n\n<!-- Template -->\n<ng-template #contentView>\n  <div class=\"dbx-download-text-view-content\">\n    @if (showTitle()) {\n      <div class=\"dbx-download-text-view-content-title dbx-mb2 mat-subtitle-2\">{{ fileNameSignal() }}</div>\n    }\n    @if (showPreview()) {\n      <div class=\"dbx-download-text-preview dbx-json dbx-content-pit dbx-content-pit-scrollable dbx-mb3\">{{ contentDataSignal() }}</div>\n    }\n    <div class=\"dbx-download-text-view-actions\">\n      <dbx-button dbxAction dbxActionValue [raised]=\"true\" [dbxActionHandler]=\"handleCopyToClipboard\" icon=\"content_copy\" dbxActionButton text=\"Copy To Clipboard\"></dbx-button>\n      <dbx-button-spacer></dbx-button-spacer>\n      <a #downloadButton [href]=\"fileUrlSignal()\" [attr.download]=\"fileNameSignal()\"><dbx-button icon=\"download\" text=\"Download\" [raised]=\"true\" [working]=\"!downloadReadySignal()\" [disabled]=\"!downloadReadySignal()\"></dbx-button></a>\n    </div>\n  </div>\n</ng-template>\n"]}
@@ -8,6 +8,9 @@ export * from './table.column.header.component';
8
8
  export * from './table.column.footer.component';
9
9
  export * from './table.component';
10
10
  export * from './table.directive';
11
+ export * from './table.group.directive';
12
+ export * from './table.group.header.component';
13
+ export * from './table.group.footer.component';
11
14
  export * from './table.item.action.component';
12
15
  export * from './table.item.cell.component';
13
16
  export * from './table.item.directive';
@@ -15,4 +18,4 @@ export * from './table.item.header.component';
15
18
  export * from './table.module';
16
19
  export * from './table.store';
17
20
  export * from './table';
18
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYngtd2ViL3RhYmxlL3NyYy9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kYXRlJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5hY3Rpb24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5pbnB1dC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5jZWxsLnN1bW1hcnllbmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5zdW1tYXJ5c3RhcnQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29sdW1uLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLmNvbHVtbi5oZWFkZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29sdW1uLmZvb3Rlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5pdGVtLmFjdGlvbi5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5pdGVtLmNlbGwuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuaXRlbS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5pdGVtLmhlYWRlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5tb2R1bGUnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5zdG9yZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlJztcbiJdfQ==
21
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wYWNrYWdlcy9kYngtd2ViL3RhYmxlL3NyYy9saWIvaW5kZXgudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsY0FBYyxRQUFRLENBQUM7QUFDdkIsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLDhCQUE4QixDQUFDO0FBQzdDLGNBQWMsbUNBQW1DLENBQUM7QUFDbEQsY0FBYyxxQ0FBcUMsQ0FBQztBQUNwRCxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsaUNBQWlDLENBQUM7QUFDaEQsY0FBYyxpQ0FBaUMsQ0FBQztBQUNoRCxjQUFjLG1CQUFtQixDQUFDO0FBQ2xDLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyx5QkFBeUIsQ0FBQztBQUN4QyxjQUFjLGdDQUFnQyxDQUFDO0FBQy9DLGNBQWMsZ0NBQWdDLENBQUM7QUFDL0MsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLDZCQUE2QixDQUFDO0FBQzVDLGNBQWMsd0JBQXdCLENBQUM7QUFDdkMsY0FBYywrQkFBK0IsQ0FBQztBQUM5QyxjQUFjLGdCQUFnQixDQUFDO0FBQy9CLGNBQWMsZUFBZSxDQUFDO0FBQzlCLGNBQWMsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9kYXRlJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5hY3Rpb24uY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5pbnB1dC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5jZWxsLnN1bW1hcnllbmQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY2VsbC5zdW1tYXJ5c3RhcnQuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29sdW1uLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLmNvbHVtbi5oZWFkZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuY29sdW1uLmZvb3Rlci5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5ncm91cC5kaXJlY3RpdmUnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5ncm91cC5oZWFkZXIuY29tcG9uZW50JztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUuZ3JvdXAuZm9vdGVyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLml0ZW0uYWN0aW9uLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLml0ZW0uY2VsbC5jb21wb25lbnQnO1xuZXhwb3J0ICogZnJvbSAnLi90YWJsZS5pdGVtLmRpcmVjdGl2ZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLml0ZW0uaGVhZGVyLmNvbXBvbmVudCc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLm1vZHVsZSc7XG5leHBvcnQgKiBmcm9tICcuL3RhYmxlLnN0b3JlJztcbmV4cG9ydCAqIGZyb20gJy4vdGFibGUnO1xuIl19
@@ -6,7 +6,7 @@ import { DbxInjectionComponent } from '@dereekb/dbx-core';
6
6
  import * as i0 from "@angular/core";
7
7
  export class DbxTableActionCellComponent {
8
8
  tableStore = inject(DbxTableStore);
9
- config$ = this.tableStore.viewDelegate$.pipe(map((x) => x.summaryRowEnd), distinctUntilChanged());
9
+ config$ = this.tableStore.viewDelegate$.pipe(map((x) => x.actionHeader), distinctUntilChanged());
10
10
  configSignal = toSignal(this.config$);
11
11
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableActionCellComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
12
12
  static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxTableActionCellComponent, isStandalone: true, selector: "dbx-table-action-cell", ngImport: i0, template: `
@@ -25,4 +25,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
25
25
  imports: [DbxInjectionComponent]
26
26
  }]
27
27
  }] });
28
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuY2VsbC5hY3Rpb24uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LXdlYi90YWJsZS9zcmMvbGliL3RhYmxlLmNlbGwuYWN0aW9uLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDdEQsT0FBTyxFQUFFLEdBQUcsRUFBRSxvQkFBb0IsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUNqRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlDLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLG1CQUFtQixDQUFDOztBQVcxRCxNQUFNLE9BQU8sMkJBQTJCO0lBQzdCLFVBQVUsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7SUFFbkMsT0FBTyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLElBQUksQ0FDbkQsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsYUFBYSxDQUFDLEVBQzNCLG9CQUFvQixFQUFFLENBQ3ZCLENBQUM7SUFFTyxZQUFZLEdBQUcsUUFBUSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQzt3R0FScEMsMkJBQTJCOzRGQUEzQiwyQkFBMkIsaUZBUDVCOztHQUVULDREQUdTLHFCQUFxQjs7NEZBRXBCLDJCQUEyQjtrQkFUdkMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUU7O0dBRVQ7b0JBQ0QsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07b0JBQy9DLFVBQVUsRUFBRSxJQUFJO29CQUNoQixPQUFPLEVBQUUsQ0FBQyxxQkFBcUIsQ0FBQztpQkFDakMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHRvU2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgbWFwLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgRGJ4VGFibGVTdG9yZSB9IGZyb20gJy4vdGFibGUuc3RvcmUnO1xuaW1wb3J0IHsgRGJ4SW5qZWN0aW9uQ29tcG9uZW50IH0gZnJvbSAnQGRlcmVla2IvZGJ4LWNvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdkYngtdGFibGUtYWN0aW9uLWNlbGwnLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkYngtaW5qZWN0aW9uIFtjb25maWddPVwiY29uZmlnU2lnbmFsKClcIj48L2RieC1pbmplY3Rpb24+XG4gIGAsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbRGJ4SW5qZWN0aW9uQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBEYnhUYWJsZUFjdGlvbkNlbGxDb21wb25lbnQge1xuICByZWFkb25seSB0YWJsZVN0b3JlID0gaW5qZWN0KERieFRhYmxlU3RvcmUpO1xuXG4gIHJlYWRvbmx5IGNvbmZpZyQgPSB0aGlzLnRhYmxlU3RvcmUudmlld0RlbGVnYXRlJC5waXBlKFxuICAgIG1hcCgoeCkgPT4geC5zdW1tYXJ5Um93RW5kKSxcbiAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpXG4gICk7XG5cbiAgcmVhZG9ubHkgY29uZmlnU2lnbmFsID0gdG9TaWduYWwodGhpcy5jb25maWckKTtcbn1cbiJdfQ==
28
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuY2VsbC5hY3Rpb24uY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LXdlYi90YWJsZS9zcmMvbGliL3RhYmxlLmNlbGwuYWN0aW9uLmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDdEQsT0FBTyxFQUFFLEdBQUcsRUFBRSxvQkFBb0IsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUNqRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlDLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLG1CQUFtQixDQUFDOztBQVcxRCxNQUFNLE9BQU8sMkJBQTJCO0lBQzdCLFVBQVUsR0FBRyxNQUFNLENBQUMsYUFBYSxDQUFDLENBQUM7SUFFbkMsT0FBTyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLElBQUksQ0FDbkQsR0FBRyxDQUFDLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxDQUFDLENBQUMsWUFBWSxDQUFDLEVBQzFCLG9CQUFvQixFQUFFLENBQ3ZCLENBQUM7SUFFTyxZQUFZLEdBQUcsUUFBUSxDQUFDLElBQUksQ0FBQyxPQUFPLENBQUMsQ0FBQzt3R0FScEMsMkJBQTJCOzRGQUEzQiwyQkFBMkIsaUZBUDVCOztHQUVULDREQUdTLHFCQUFxQjs7NEZBRXBCLDJCQUEyQjtrQkFUdkMsU0FBUzttQkFBQztvQkFDVCxRQUFRLEVBQUUsdUJBQXVCO29CQUNqQyxRQUFRLEVBQUU7O0dBRVQ7b0JBQ0QsZUFBZSxFQUFFLHVCQUF1QixDQUFDLE1BQU07b0JBQy9DLFVBQVUsRUFBRSxJQUFJO29CQUNoQixPQUFPLEVBQUUsQ0FBQyxxQkFBcUIsQ0FBQztpQkFDakMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHRvU2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgbWFwLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgRGJ4VGFibGVTdG9yZSB9IGZyb20gJy4vdGFibGUuc3RvcmUnO1xuaW1wb3J0IHsgRGJ4SW5qZWN0aW9uQ29tcG9uZW50IH0gZnJvbSAnQGRlcmVla2IvZGJ4LWNvcmUnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdkYngtdGFibGUtYWN0aW9uLWNlbGwnLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkYngtaW5qZWN0aW9uIFtjb25maWddPVwiY29uZmlnU2lnbmFsKClcIj48L2RieC1pbmplY3Rpb24+XG4gIGAsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBzdGFuZGFsb25lOiB0cnVlLFxuICBpbXBvcnRzOiBbRGJ4SW5qZWN0aW9uQ29tcG9uZW50XVxufSlcbmV4cG9ydCBjbGFzcyBEYnhUYWJsZUFjdGlvbkNlbGxDb21wb25lbnQge1xuICByZWFkb25seSB0YWJsZVN0b3JlID0gaW5qZWN0KERieFRhYmxlU3RvcmUpO1xuXG4gIHJlYWRvbmx5IGNvbmZpZyQgPSB0aGlzLnRhYmxlU3RvcmUudmlld0RlbGVnYXRlJC5waXBlKFxuICAgIG1hcCgoeCkgPT4geC5hY3Rpb25IZWFkZXIpLFxuICAgIGRpc3RpbmN0VW50aWxDaGFuZ2VkKClcbiAgKTtcblxuICByZWFkb25seSBjb25maWdTaWduYWwgPSB0b1NpZ25hbCh0aGlzLmNvbmZpZyQpO1xufVxuIl19
@@ -1,8 +1,8 @@
1
1
  import { ChangeDetectionStrategy, Component, inject, computed, input } from '@angular/core';
2
2
  import { toSignal } from '@angular/core/rxjs-interop';
3
3
  import { DbxTableStore } from './table.store';
4
- import { loadingStateContext } from '@dereekb/rxjs';
5
- import { shareReplay, map } from 'rxjs';
4
+ import { loadingStateContext, mapLoadingStateValueWithOperator, valueFromFinishedLoadingState } from '@dereekb/rxjs';
5
+ import { shareReplay, map, switchMap } from 'rxjs';
6
6
  import { DbxLoadingComponent } from '@dereekb/dbx-web';
7
7
  import { InfiniteScrollDirective } from 'ngx-infinite-scroll';
8
8
  import { MatTableModule } from '@angular/material/table';
@@ -15,10 +15,21 @@ import { DbxTableItemHeaderComponent } from './table.item.header.component';
15
15
  import { DbxTableItemActionComponent } from './table.item.action.component';
16
16
  import { DbxTableActionCellComponent } from './table.cell.action.component';
17
17
  import { DbxTableColumnHeaderComponent } from './table.column.header.component';
18
+ import { DbxTableGroupHeaderComponent } from './table.group.header.component';
19
+ import { DbxTableGroupFooterComponent } from './table.group.footer.component';
20
+ import { pushArrayItemsIntoArray } from '@dereekb/util';
21
+ import { NgClass } from '@angular/common';
22
+ import { DbxTableFullSummaryRowComponent } from './table.fullsummaryrow.component';
18
23
  import * as i0 from "@angular/core";
19
24
  import * as i1 from "@angular/material/table";
20
25
  export const DBX_TABLE_ITEMS_COLUMN_NAME = '_items';
21
26
  export const DBX_TABLE_ACTIONS_COLUMN_NAME = '_actions';
27
+ export function isDbxTableViewGroupElement(element) {
28
+ return element.type === 'group';
29
+ }
30
+ export function isDbxTableViewItemElement(element) {
31
+ return element.type === 'item';
32
+ }
22
33
  /**
23
34
  * A table with fixed content
24
35
  */
@@ -35,7 +46,48 @@ export class DbxTableViewComponent {
35
46
  innerColumnNames$ = this.innerColumns$.pipe(map((x) => x.map((y) => y.columnName)), shareReplay(1));
36
47
  innerColumnsSignal = toSignal(this.innerColumns$);
37
48
  innerColumnNamesSignal = toSignal(this.innerColumnNames$);
38
- elements$ = this.tableStore.items$;
49
+ elementsState$ = this.tableStore.groupsState$.pipe(mapLoadingStateValueWithOperator(switchMap((groups) => {
50
+ return this.tableStore.viewDelegate$.pipe(map((viewDelegate) => {
51
+ const { groupHeader: inputGroupHeader, groupFooter: inputGroupFooter } = viewDelegate;
52
+ const hasGroupHeader = inputGroupHeader != null ? (group) => inputGroupHeader(group) != null : () => false;
53
+ const hasGroupFooter = inputGroupFooter != null ? (group) => inputGroupFooter(group) != null : () => false;
54
+ return groups
55
+ .map((group) => {
56
+ const { items } = group;
57
+ const itemElements = items.map((item) => ({
58
+ type: 'item',
59
+ item
60
+ }));
61
+ let elements;
62
+ if (group.default) {
63
+ elements = itemElements;
64
+ }
65
+ else {
66
+ const header = hasGroupHeader(group);
67
+ const footer = hasGroupFooter(group);
68
+ elements = [];
69
+ if (header) {
70
+ elements.push({
71
+ type: 'group',
72
+ location: 'header',
73
+ group
74
+ });
75
+ }
76
+ pushArrayItemsIntoArray(elements, itemElements);
77
+ if (footer) {
78
+ elements.push({
79
+ type: 'group',
80
+ location: 'footer',
81
+ group
82
+ });
83
+ }
84
+ }
85
+ return elements;
86
+ })
87
+ .flat();
88
+ }));
89
+ })), shareReplay(1));
90
+ elements$ = this.elementsState$.pipe(valueFromFinishedLoadingState([]), shareReplay(1));
39
91
  displayedColumns$ = this.innerColumnNames$.pipe(map((columnNames) => {
40
92
  return [this.itemsColumnName, ...columnNames, this.actionsColumnName];
41
93
  }), shareReplay(1));
@@ -43,20 +95,59 @@ export class DbxTableViewComponent {
43
95
  const columnNames = this.innerColumnNamesSignal() || [];
44
96
  return [this.itemsColumnName, ...columnNames, this.actionsColumnName];
45
97
  });
98
+ visibleColumnsSignal = computed(() => {
99
+ const displayedColumns = this.displayedColumnsSignal();
100
+ // TODO: Every time the table is rendered/size changes/etc we should recompute width of the first n columns that span the viewport
101
+ // in order to calculate a colspan for the group header/footer that is not greater than span of the table for a given view size
102
+ // I.E. if the screen shrinks to only show the first 4 columns, then the colspan should be 4.
103
+ // as a temporary measure, we just show half the columns
104
+ return displayedColumns.length / 2;
105
+ });
46
106
  trackByFunction$ = this.tableStore.viewDelegate$.pipe(map((x) => x.trackBy ?? this.DEFAULT_TRACK_BY_FUNCTION), shareReplay(1));
47
- trackByFunctionSignal = toSignal(this.trackByFunction$, { initialValue: this.DEFAULT_TRACK_BY_FUNCTION });
107
+ inputTrackByFunctionSignal = toSignal(this.trackByFunction$, { initialValue: this.DEFAULT_TRACK_BY_FUNCTION });
108
+ trackElementByFunctionSignal = computed(() => {
109
+ const trackByFunction = this.inputTrackByFunctionSignal();
110
+ const fn = (index, element) => {
111
+ if (element.type === 'item') {
112
+ return `i_${trackByFunction(index, element.item)}`;
113
+ }
114
+ else {
115
+ return `g_${element.group.groupId}`;
116
+ }
117
+ };
118
+ return fn;
119
+ });
48
120
  context = loadingStateContext({ obs: this.tableStore.dataState$ });
49
- dataLoadingContext = loadingStateContext({ obs: this.tableStore.itemsState$ });
121
+ dataLoadingContext = loadingStateContext({ obs: this.elementsState$ });
50
122
  contextSignal = toSignal(this.context.state$);
51
123
  dataLoadingContextSignal = toSignal(this.dataLoadingContext.state$);
124
+ viewDelegateSignal = toSignal(this.tableStore.viewDelegate$);
52
125
  onScrollDown() {
53
126
  this.tableStore.loadMore();
54
127
  }
128
+ showItemRow(_, row) {
129
+ return row.type === 'item';
130
+ }
131
+ showGroupHeaderRow(_, row) {
132
+ return row.type === 'group' && row.location === 'header';
133
+ }
134
+ showGroupFooterRow(_, row) {
135
+ return row.type === 'group' && row.location === 'footer';
136
+ }
137
+ showFooterRowSignal = computed(() => {
138
+ const viewDelegate = this.viewDelegateSignal();
139
+ const showFooterRow = viewDelegate && (viewDelegate.summaryRowHeader != null || viewDelegate.columnFooter != null || viewDelegate.summaryRowEnd != null);
140
+ return showFooterRow;
141
+ });
142
+ showFullSummaryRowSignal = computed(() => {
143
+ const viewDelegate = this.viewDelegateSignal();
144
+ return viewDelegate?.fullSummaryRow != null;
145
+ });
55
146
  static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableViewComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
56
- static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: DbxTableViewComponent, isStandalone: true, selector: "dbx-table-view", inputs: { scrollDistance: { classPropertyName: "scrollDistance", publicName: "scrollDistance", isSignal: true, isRequired: false, transformFunction: null }, throttleScroll: { classPropertyName: "throttleScroll", publicName: "throttleScroll", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<dbx-loading [context]=\"context\">\n <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackByFunctionSignal() || DEFAULT_TRACK_BY_FUNCTION\">\n <!-- Header/Item column -->\n <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-input-cell></dbx-table-input-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-header [element]=\"element\"></dbx-table-item-header>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n </td>\n </ng-container>\n\n <!-- Column Definitions -->\n @for (column of innerColumnsSignal(); track column.columnName) {\n <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-cell [element]=\"element\" [column]=\"column\"></dbx-table-item-cell>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n </td>\n </ng-container>\n }\n\n <!-- Tail/Action column -->\n <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-action-cell></dbx-table-action-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-action [element]=\"element\"></dbx-table-item-action>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n </td>\n </ng-container>\n\n <!-- Table View -->\n <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal()\"></tr>\n\n <tr mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n </table>\n </section>\n</dbx-loading>\n", dependencies: [{ kind: "component", type: DbxLoadingComponent, selector: "dbx-loading", inputs: ["padding", "show", "text", "mode", "color", "diameter", "linear", "loading", "error", "context"] }, { kind: "directive", type: InfiniteScrollDirective, selector: "[infiniteScroll], [infinite-scroll], [data-infinite-scroll]", inputs: ["infiniteScrollDistance", "infiniteScrollUpDistance", "infiniteScrollThrottle", "infiniteScrollDisabled", "infiniteScrollContainer", "scrollWindow", "immediateCheck", "horizontal", "alwaysCallback", "fromRoot"], outputs: ["scrolled", "scrolledUp"] }, { kind: "ngmodule", type: MatTableModule }, { kind: "component", type: i1.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i1.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i1.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i1.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i1.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i1.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i1.MatFooterCellDef, selector: "[matFooterCellDef]" }, { kind: "directive", type: i1.MatFooterRowDef, selector: "[matFooterRowDef]", inputs: ["matFooterRowDef", "matFooterRowDefSticky"] }, { kind: "directive", type: i1.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i1.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "directive", type: i1.MatFooterCell, selector: "mat-footer-cell, td[mat-footer-cell]" }, { kind: "component", type: i1.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i1.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "component", type: i1.MatFooterRow, selector: "mat-footer-row, tr[mat-footer-row]", exportAs: ["matFooterRow"] }, { kind: "component", type: DbxTableInputCellComponent, selector: "dbx-table-input-cell" }, { kind: "component", type: DbxTableItemHeaderComponent, selector: "dbx-table-item-header" }, { kind: "component", type: DbxTableItemCellComponent, selector: "dbx-table-item-cell", inputs: ["column"] }, { kind: "component", type: DbxTableItemActionComponent, selector: "dbx-table-item-action" }, { kind: "component", type: DbxTableActionCellComponent, selector: "dbx-table-action-cell" }, { kind: "component", type: DbxTableColumnHeaderComponent, selector: "dbx-table-column-header" }, { kind: "component", type: DbxTableColumnFooterComponent, selector: "dbx-table-column-footer" }, { kind: "component", type: DbxTableSummaryStartCellComponent, selector: "dbx-table-summary-start-cell" }, { kind: "component", type: DbxTableSummaryEndCellComponent, selector: "dbx-table-summary-end-cell" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
147
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.13", type: DbxTableViewComponent, isStandalone: true, selector: "dbx-table-view", inputs: { scrollDistance: { classPropertyName: "scrollDistance", publicName: "scrollDistance", isSignal: true, isRequired: false, transformFunction: null }, throttleScroll: { classPropertyName: "throttleScroll", publicName: "throttleScroll", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0, template: "<dbx-loading [context]=\"context\">\n <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackElementByFunctionSignal()\" multiTemplateDataRows>\n <!-- Groups -->\n <!-- Group Header column -->\n <ng-container matColumnDef=\"groupHeaderColumn\" [sticky]=\"true\">\n <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n <dbx-table-group-header [group]=\"element.group\"></dbx-table-group-header>\n </td>\n </ng-container>\n\n <!-- Group Footer column -->\n <ng-container matColumnDef=\"groupFooterColumn\" [sticky]=\"true\">\n <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n <dbx-table-group-footer [group]=\"element.group\"></dbx-table-group-footer>\n </td>\n </ng-container>\n\n <!-- Items -->\n <!-- Header/Item column -->\n <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-input-cell></dbx-table-input-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-header [item]=\"element.item\"></dbx-table-item-header>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n </td>\n </ng-container>\n\n <!-- Column Definitions -->\n @for (column of innerColumnsSignal(); track column.columnName) {\n <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-cell [item]=\"element.item\" [column]=\"column\"></dbx-table-item-cell>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n </td>\n </ng-container>\n }\n\n <!-- Tail/Action column -->\n <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-action-cell></dbx-table-action-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-action [item]=\"element.item\"></dbx-table-item-action>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n </td>\n </ng-container>\n\n <!-- Table View -->\n <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n <tr mat-row *matRowDef=\"let row; columns: ['groupHeaderColumn']; when: showGroupHeaderRow\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal(); when: showItemRow\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['groupFooterColumn']; when: showGroupFooterRow\"></tr>\n\n <tr [ngClass]=\"{ 'dbx-hide': !showFooterRowSignal() }\" mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n </table>\n </section>\n <!-- Full summary row -->\n @if (showFullSummaryRowSignal()) {\n <div class=\"mat-mdc-table dbx-table-view-full-summary-row-container\">\n <div class=\"mdc-data-table__cell mat-mdc-row\">\n <dbx-table-full-summary-row></dbx-table-full-summary-row>\n </div>\n </div>\n }\n</dbx-loading>\n", dependencies: [{ kind: "component", type: DbxLoadingComponent, selector: "dbx-loading", inputs: ["padding", "show", "text", "mode", "color", "diameter", "linear", "loading", "error", "context"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: InfiniteScrollDirective, selector: "[infiniteScroll], [infinite-scroll], [data-infinite-scroll]", inputs: ["infiniteScrollDistance", "infiniteScrollUpDistance", "infiniteScrollThrottle", "infiniteScrollDisabled", "infiniteScrollContainer", "scrollWindow", "immediateCheck", "horizontal", "alwaysCallback", "fromRoot"], outputs: ["scrolled", "scrolledUp"] }, { kind: "ngmodule", type: MatTableModule }, { kind: "component", type: i1.MatTable, selector: "mat-table, table[mat-table]", exportAs: ["matTable"] }, { kind: "directive", type: i1.MatHeaderCellDef, selector: "[matHeaderCellDef]" }, { kind: "directive", type: i1.MatHeaderRowDef, selector: "[matHeaderRowDef]", inputs: ["matHeaderRowDef", "matHeaderRowDefSticky"] }, { kind: "directive", type: i1.MatColumnDef, selector: "[matColumnDef]", inputs: ["matColumnDef"] }, { kind: "directive", type: i1.MatCellDef, selector: "[matCellDef]" }, { kind: "directive", type: i1.MatRowDef, selector: "[matRowDef]", inputs: ["matRowDefColumns", "matRowDefWhen"] }, { kind: "directive", type: i1.MatFooterCellDef, selector: "[matFooterCellDef]" }, { kind: "directive", type: i1.MatFooterRowDef, selector: "[matFooterRowDef]", inputs: ["matFooterRowDef", "matFooterRowDefSticky"] }, { kind: "directive", type: i1.MatHeaderCell, selector: "mat-header-cell, th[mat-header-cell]" }, { kind: "directive", type: i1.MatCell, selector: "mat-cell, td[mat-cell]" }, { kind: "directive", type: i1.MatFooterCell, selector: "mat-footer-cell, td[mat-footer-cell]" }, { kind: "component", type: i1.MatHeaderRow, selector: "mat-header-row, tr[mat-header-row]", exportAs: ["matHeaderRow"] }, { kind: "component", type: i1.MatRow, selector: "mat-row, tr[mat-row]", exportAs: ["matRow"] }, { kind: "component", type: i1.MatFooterRow, selector: "mat-footer-row, tr[mat-footer-row]", exportAs: ["matFooterRow"] }, { kind: "component", type: DbxTableInputCellComponent, selector: "dbx-table-input-cell" }, { kind: "component", type: DbxTableItemHeaderComponent, selector: "dbx-table-item-header" }, { kind: "component", type: DbxTableItemCellComponent, selector: "dbx-table-item-cell", inputs: ["column"] }, { kind: "component", type: DbxTableItemActionComponent, selector: "dbx-table-item-action" }, { kind: "component", type: DbxTableActionCellComponent, selector: "dbx-table-action-cell" }, { kind: "component", type: DbxTableColumnHeaderComponent, selector: "dbx-table-column-header" }, { kind: "component", type: DbxTableColumnFooterComponent, selector: "dbx-table-column-footer" }, { kind: "component", type: DbxTableSummaryStartCellComponent, selector: "dbx-table-summary-start-cell" }, { kind: "component", type: DbxTableSummaryEndCellComponent, selector: "dbx-table-summary-end-cell" }, { kind: "component", type: DbxTableGroupHeaderComponent, selector: "dbx-table-group-header" }, { kind: "component", type: DbxTableGroupFooterComponent, selector: "dbx-table-group-footer" }, { kind: "component", type: DbxTableFullSummaryRowComponent, selector: "dbx-table-full-summary-row" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
57
148
  }
58
149
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableViewComponent, decorators: [{
59
150
  type: Component,
60
- args: [{ selector: 'dbx-table-view', imports: [DbxLoadingComponent, InfiniteScrollDirective, MatTableModule, DbxTableInputCellComponent, DbxTableItemHeaderComponent, DbxTableItemCellComponent, DbxTableItemActionComponent, DbxTableActionCellComponent, DbxTableColumnHeaderComponent, DbxTableColumnFooterComponent, DbxTableSummaryStartCellComponent, DbxTableSummaryEndCellComponent], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, template: "<dbx-loading [context]=\"context\">\n <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackByFunctionSignal() || DEFAULT_TRACK_BY_FUNCTION\">\n <!-- Header/Item column -->\n <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-input-cell></dbx-table-input-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-header [element]=\"element\"></dbx-table-item-header>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n </td>\n </ng-container>\n\n <!-- Column Definitions -->\n @for (column of innerColumnsSignal(); track column.columnName) {\n <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-cell [element]=\"element\" [column]=\"column\"></dbx-table-item-cell>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n </td>\n </ng-container>\n }\n\n <!-- Tail/Action column -->\n <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-action-cell></dbx-table-action-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n <dbx-table-item-action [element]=\"element\"></dbx-table-item-action>\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n </td>\n </ng-container>\n\n <!-- Table View -->\n <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal()\"></tr>\n\n <tr mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n </table>\n </section>\n</dbx-loading>\n" }]
151
+ args: [{ selector: 'dbx-table-view', imports: [DbxLoadingComponent, NgClass, InfiniteScrollDirective, MatTableModule, DbxTableInputCellComponent, DbxTableItemHeaderComponent, DbxTableItemCellComponent, DbxTableItemActionComponent, DbxTableActionCellComponent, DbxTableColumnHeaderComponent, DbxTableColumnFooterComponent, DbxTableSummaryStartCellComponent, DbxTableSummaryEndCellComponent, DbxTableGroupHeaderComponent, DbxTableGroupFooterComponent, DbxTableFullSummaryRowComponent], changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, template: "<dbx-loading [context]=\"context\">\n <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackElementByFunctionSignal()\" multiTemplateDataRows>\n <!-- Groups -->\n <!-- Group Header column -->\n <ng-container matColumnDef=\"groupHeaderColumn\" [sticky]=\"true\">\n <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n <dbx-table-group-header [group]=\"element.group\"></dbx-table-group-header>\n </td>\n </ng-container>\n\n <!-- Group Footer column -->\n <ng-container matColumnDef=\"groupFooterColumn\" [sticky]=\"true\">\n <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n <dbx-table-group-footer [group]=\"element.group\"></dbx-table-group-footer>\n </td>\n </ng-container>\n\n <!-- Items -->\n <!-- Header/Item column -->\n <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-input-cell></dbx-table-input-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-header [item]=\"element.item\"></dbx-table-item-header>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n </td>\n </ng-container>\n\n <!-- Column Definitions -->\n @for (column of innerColumnsSignal(); track column.columnName) {\n <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-cell [item]=\"element.item\" [column]=\"column\"></dbx-table-item-cell>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n </td>\n </ng-container>\n }\n\n <!-- Tail/Action column -->\n <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n <th mat-header-cell *matHeaderCellDef>\n <dbx-table-action-cell></dbx-table-action-cell>\n </th>\n <td mat-cell *matCellDef=\"let element\">\n @if (element.item) {\n <dbx-table-item-action [item]=\"element.item\"></dbx-table-item-action>\n }\n </td>\n <td mat-footer-cell *matFooterCellDef>\n <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n </td>\n </ng-container>\n\n <!-- Table View -->\n <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n <tr mat-row *matRowDef=\"let row; columns: ['groupHeaderColumn']; when: showGroupHeaderRow\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal(); when: showItemRow\"></tr>\n <tr mat-row *matRowDef=\"let row; columns: ['groupFooterColumn']; when: showGroupFooterRow\"></tr>\n\n <tr [ngClass]=\"{ 'dbx-hide': !showFooterRowSignal() }\" mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n </table>\n </section>\n <!-- Full summary row -->\n @if (showFullSummaryRowSignal()) {\n <div class=\"mat-mdc-table dbx-table-view-full-summary-row-container\">\n <div class=\"mdc-data-table__cell mat-mdc-row\">\n <dbx-table-full-summary-row></dbx-table-full-summary-row>\n </div>\n </div>\n }\n</dbx-loading>\n" }]
61
152
  }] });
62
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../packages/dbx-web/table/src/lib/table.component.ts","../../../../../../packages/dbx-web/table/src/lib/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, TrackByFunction, inject, computed, input } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DbxTableStore } from './table.store';\nimport { loadingStateContext } from '@dereekb/rxjs';\nimport { shareReplay, map, Observable } from 'rxjs';\nimport { DbxLoadingComponent } from '@dereekb/dbx-web';\nimport { InfiniteScrollDirective } from 'ngx-infinite-scroll';\nimport { MatTableModule } from '@angular/material/table';\nimport { DbxTableInputCellComponent } from './table.cell.input.component';\nimport { DbxTableSummaryEndCellComponent } from './table.cell.summaryend.component';\nimport { DbxTableSummaryStartCellComponent } from './table.cell.summarystart.component';\nimport { DbxTableColumnFooterComponent } from './table.column.footer.component';\nimport { DbxTableItemCellComponent } from './table.item.cell.component';\nimport { DbxTableItemHeaderComponent } from './table.item.header.component';\nimport { DbxTableItemActionComponent } from './table.item.action.component';\nimport { DbxTableActionCellComponent } from './table.cell.action.component';\nimport { DbxTableColumnHeaderComponent } from './table.column.header.component';\n\nexport const DBX_TABLE_ITEMS_COLUMN_NAME = '_items';\nexport const DBX_TABLE_ACTIONS_COLUMN_NAME = '_actions';\n\n/**\n * A table with fixed content\n */\n@Component({\n  selector: 'dbx-table-view',\n  templateUrl: './table.component.html',\n  imports: [DbxLoadingComponent, InfiniteScrollDirective, MatTableModule, DbxTableInputCellComponent, DbxTableItemHeaderComponent, DbxTableItemCellComponent, DbxTableItemActionComponent, DbxTableActionCellComponent, DbxTableColumnHeaderComponent, DbxTableColumnFooterComponent, DbxTableSummaryStartCellComponent, DbxTableSummaryEndCellComponent],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class DbxTableViewComponent<I, C, T> {\n  readonly tableStore = inject(DbxTableStore<I, C, T>);\n\n  readonly DEFAULT_TRACK_BY_FUNCTION: TrackByFunction<any> = (index) => {\n    return index;\n  };\n\n  readonly scrollDistance = input<number>(0.5);\n  readonly throttleScroll = input<number>(50);\n\n  readonly itemsColumnName = DBX_TABLE_ITEMS_COLUMN_NAME;\n  readonly actionsColumnName = DBX_TABLE_ACTIONS_COLUMN_NAME;\n\n  readonly innerColumns$ = this.tableStore.columns$;\n  readonly innerColumnNames$ = this.innerColumns$.pipe(\n    map((x) => x.map((y) => y.columnName)),\n    shareReplay(1)\n  );\n\n  readonly innerColumnsSignal = toSignal(this.innerColumns$);\n  readonly innerColumnNamesSignal = toSignal(this.innerColumnNames$);\n\n  readonly elements$ = this.tableStore.items$;\n\n  readonly displayedColumns$ = this.innerColumnNames$.pipe(\n    map((columnNames) => {\n      return [this.itemsColumnName, ...columnNames, this.actionsColumnName];\n    }),\n    shareReplay(1)\n  );\n\n  readonly displayedColumnsSignal = computed(() => {\n    const columnNames = this.innerColumnNamesSignal() || [];\n    return [this.itemsColumnName, ...columnNames, this.actionsColumnName];\n  });\n\n  readonly trackByFunction$: Observable<TrackByFunction<T>> = this.tableStore.viewDelegate$.pipe(\n    map((x) => x.trackBy ?? this.DEFAULT_TRACK_BY_FUNCTION),\n    shareReplay(1)\n  );\n\n  readonly trackByFunctionSignal = toSignal(this.trackByFunction$, { initialValue: this.DEFAULT_TRACK_BY_FUNCTION });\n\n  readonly context = loadingStateContext({ obs: this.tableStore.dataState$ });\n  readonly dataLoadingContext = loadingStateContext({ obs: this.tableStore.itemsState$ });\n\n  readonly contextSignal = toSignal(this.context.state$);\n  readonly dataLoadingContextSignal = toSignal(this.dataLoadingContext.state$);\n\n  onScrollDown(): void {\n    this.tableStore.loadMore();\n  }\n}\n","<dbx-loading [context]=\"context\">\n  <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n  <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n    <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackByFunctionSignal() || DEFAULT_TRACK_BY_FUNCTION\">\n      <!-- Header/Item column -->\n      <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n        <th mat-header-cell *matHeaderCellDef>\n          <dbx-table-input-cell></dbx-table-input-cell>\n        </th>\n        <td mat-cell *matCellDef=\"let element\">\n          <dbx-table-item-header [element]=\"element\"></dbx-table-item-header>\n        </td>\n        <td mat-footer-cell *matFooterCellDef>\n          <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n        </td>\n      </ng-container>\n\n      <!-- Column Definitions -->\n      @for (column of innerColumnsSignal(); track column.columnName) {\n        <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n          <th mat-header-cell *matHeaderCellDef>\n            <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n          </th>\n          <td mat-cell *matCellDef=\"let element\">\n            <dbx-table-item-cell [element]=\"element\" [column]=\"column\"></dbx-table-item-cell>\n          </td>\n          <td mat-footer-cell *matFooterCellDef>\n            <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n          </td>\n        </ng-container>\n      }\n\n      <!-- Tail/Action column -->\n      <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n        <th mat-header-cell *matHeaderCellDef>\n          <dbx-table-action-cell></dbx-table-action-cell>\n        </th>\n        <td mat-cell *matCellDef=\"let element\">\n          <dbx-table-item-action [element]=\"element\"></dbx-table-item-action>\n        </td>\n        <td mat-footer-cell *matFooterCellDef>\n          <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n        </td>\n      </ng-container>\n\n      <!-- Table View -->\n      <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n      <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal()\"></tr>\n\n      <tr mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n    </table>\n  </section>\n</dbx-loading>\n"]}
153
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"table.component.js","sourceRoot":"","sources":["../../../../../../packages/dbx-web/table/src/lib/table.component.ts","../../../../../../packages/dbx-web/table/src/lib/table.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, TrackByFunction, inject, computed, input, Signal } from '@angular/core';\nimport { toSignal } from '@angular/core/rxjs-interop';\nimport { DbxTableStore } from './table.store';\nimport { LoadingState, loadingStateContext, mapLoadingStateValueWithOperator, valueFromFinishedLoadingState } from '@dereekb/rxjs';\nimport { shareReplay, map, Observable, switchMap } from 'rxjs';\nimport { DbxLoadingComponent } from '@dereekb/dbx-web';\nimport { InfiniteScrollDirective } from 'ngx-infinite-scroll';\nimport { MatTableModule } from '@angular/material/table';\nimport { DbxTableInputCellComponent } from './table.cell.input.component';\nimport { DbxTableSummaryEndCellComponent } from './table.cell.summaryend.component';\nimport { DbxTableSummaryStartCellComponent } from './table.cell.summarystart.component';\nimport { DbxTableColumnFooterComponent } from './table.column.footer.component';\nimport { DbxTableItemCellComponent } from './table.item.cell.component';\nimport { DbxTableItemHeaderComponent } from './table.item.header.component';\nimport { DbxTableItemActionComponent } from './table.item.action.component';\nimport { DbxTableActionCellComponent } from './table.cell.action.component';\nimport { DbxTableColumnHeaderComponent } from './table.column.header.component';\nimport { DbxTableItemGroup, DefaultDbxTableItemGroup } from './table';\nimport { DbxTableGroupHeaderComponent } from './table.group.header.component';\nimport { DbxTableGroupFooterComponent } from './table.group.footer.component';\nimport { pushArrayItemsIntoArray } from '@dereekb/util';\nimport { NgClass } from '@angular/common';\nimport { DbxTableFullSummaryRowComponent } from './table.fullsummaryrow.component';\n\nexport const DBX_TABLE_ITEMS_COLUMN_NAME = '_items';\nexport const DBX_TABLE_ACTIONS_COLUMN_NAME = '_actions';\n\nexport interface DbxTableViewGroupElement<T, G> {\n  readonly type: 'group';\n  readonly location: 'header' | 'footer';\n  readonly group: DbxTableItemGroup<T, G>;\n}\n\nexport interface DbxTableViewItemElement<T, G> {\n  readonly type: 'item';\n  readonly item: T;\n}\n\nexport type DbxTableViewElement<T, G> = DbxTableViewGroupElement<T, G> | DbxTableViewItemElement<T, G>;\n\nexport function isDbxTableViewGroupElement<T, G>(element: DbxTableViewElement<T, G>): element is DbxTableViewGroupElement<T, G> {\n  return element.type === 'group';\n}\n\nexport function isDbxTableViewItemElement<T, G>(element: DbxTableViewElement<T, G>): element is DbxTableViewItemElement<T, G> {\n  return element.type === 'item';\n}\n\n/**\n * A table with fixed content\n */\n@Component({\n  selector: 'dbx-table-view',\n  templateUrl: './table.component.html',\n  imports: [DbxLoadingComponent, NgClass, InfiniteScrollDirective, MatTableModule, DbxTableInputCellComponent, DbxTableItemHeaderComponent, DbxTableItemCellComponent, DbxTableItemActionComponent, DbxTableActionCellComponent, DbxTableColumnHeaderComponent, DbxTableColumnFooterComponent, DbxTableSummaryStartCellComponent, DbxTableSummaryEndCellComponent, DbxTableGroupHeaderComponent, DbxTableGroupFooterComponent, DbxTableFullSummaryRowComponent],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n  standalone: true\n})\nexport class DbxTableViewComponent<I, C, T, G = unknown> {\n  readonly tableStore = inject(DbxTableStore<I, C, T, G>);\n\n  readonly DEFAULT_TRACK_BY_FUNCTION: TrackByFunction<any> = (index) => {\n    return index;\n  };\n\n  readonly scrollDistance = input<number>(0.5);\n  readonly throttleScroll = input<number>(50);\n\n  readonly itemsColumnName = DBX_TABLE_ITEMS_COLUMN_NAME;\n  readonly actionsColumnName = DBX_TABLE_ACTIONS_COLUMN_NAME;\n\n  readonly innerColumns$ = this.tableStore.columns$;\n  readonly innerColumnNames$ = this.innerColumns$.pipe(\n    map((x) => x.map((y) => y.columnName)),\n    shareReplay(1)\n  );\n\n  readonly innerColumnsSignal = toSignal(this.innerColumns$);\n  readonly innerColumnNamesSignal = toSignal(this.innerColumnNames$);\n\n  readonly elementsState$: Observable<LoadingState<DbxTableViewElement<T, G>[]>> = this.tableStore.groupsState$.pipe(\n    mapLoadingStateValueWithOperator(\n      switchMap((groups) => {\n        return this.tableStore.viewDelegate$.pipe(\n          map((viewDelegate) => {\n            const { groupHeader: inputGroupHeader, groupFooter: inputGroupFooter } = viewDelegate;\n            const hasGroupHeader = inputGroupHeader != null ? (group: DbxTableItemGroup<T, G>) => inputGroupHeader(group) != null : () => false;\n            const hasGroupFooter = inputGroupFooter != null ? (group: DbxTableItemGroup<T, G>) => inputGroupFooter(group) != null : () => false;\n\n            return groups\n              .map((group) => {\n                const { items } = group;\n\n                const itemElements: DbxTableViewItemElement<T, G>[] = items.map((item) => ({\n                  type: 'item',\n                  item\n                }));\n\n                let elements: DbxTableViewElement<T, G>[];\n\n                if ((group as DefaultDbxTableItemGroup<T, G>).default) {\n                  elements = itemElements;\n                } else {\n                  const header = hasGroupHeader(group);\n                  const footer = hasGroupFooter(group);\n\n                  elements = [];\n\n                  if (header) {\n                    elements.push({\n                      type: 'group',\n                      location: 'header',\n                      group\n                    });\n                  }\n\n                  pushArrayItemsIntoArray(elements, itemElements);\n\n                  if (footer) {\n                    elements.push({\n                      type: 'group',\n                      location: 'footer',\n                      group\n                    });\n                  }\n                }\n\n                return elements;\n              })\n              .flat();\n          })\n        );\n      })\n    ),\n    shareReplay(1)\n  );\n\n  readonly elements$ = this.elementsState$.pipe(valueFromFinishedLoadingState([]), shareReplay(1));\n\n  readonly displayedColumns$ = this.innerColumnNames$.pipe(\n    map((columnNames) => {\n      return [this.itemsColumnName, ...columnNames, this.actionsColumnName];\n    }),\n    shareReplay(1)\n  );\n\n  readonly displayedColumnsSignal = computed(() => {\n    const columnNames = this.innerColumnNamesSignal() || [];\n    return [this.itemsColumnName, ...columnNames, this.actionsColumnName];\n  });\n\n  readonly visibleColumnsSignal = computed(() => {\n    const displayedColumns = this.displayedColumnsSignal();\n\n    // TODO: Every time the table is rendered/size changes/etc we should recompute width of the first n columns that span the viewport\n    // in order to calculate a colspan for the group header/footer that is not greater than span of the table for a given view size\n    // I.E. if the screen shrinks to only show the first 4 columns, then the colspan should be 4.\n\n    // as a temporary measure, we just show half the columns\n\n    return displayedColumns.length / 2;\n  });\n\n  readonly trackByFunction$: Observable<TrackByFunction<T>> = this.tableStore.viewDelegate$.pipe(\n    map((x) => x.trackBy ?? this.DEFAULT_TRACK_BY_FUNCTION),\n    shareReplay(1)\n  );\n\n  readonly inputTrackByFunctionSignal = toSignal(this.trackByFunction$, { initialValue: this.DEFAULT_TRACK_BY_FUNCTION });\n  readonly trackElementByFunctionSignal: Signal<TrackByFunction<DbxTableViewElement<T, G>>> = computed(() => {\n    const trackByFunction = this.inputTrackByFunctionSignal() as TrackByFunction<T>;\n\n    const fn: TrackByFunction<DbxTableViewElement<T, G>> = (index: number, element: DbxTableViewElement<T, G>) => {\n      if (element.type === 'item') {\n        return `i_${trackByFunction(index, element.item as T)}`;\n      } else {\n        return `g_${element.group.groupId}`;\n      }\n    };\n\n    return fn;\n  });\n\n  readonly context = loadingStateContext({ obs: this.tableStore.dataState$ });\n  readonly dataLoadingContext = loadingStateContext({ obs: this.elementsState$ });\n\n  readonly contextSignal = toSignal(this.context.state$);\n  readonly dataLoadingContextSignal = toSignal(this.dataLoadingContext.state$);\n\n  readonly viewDelegateSignal = toSignal(this.tableStore.viewDelegate$);\n\n  onScrollDown(): void {\n    this.tableStore.loadMore();\n  }\n\n  showItemRow(_: number, row: DbxTableViewElement<T, G>): boolean {\n    return row.type === 'item';\n  }\n\n  showGroupHeaderRow(_: number, row: DbxTableViewElement<T, G>): boolean {\n    return row.type === 'group' && row.location === 'header';\n  }\n\n  showGroupFooterRow(_: number, row: DbxTableViewElement<T, G>): boolean {\n    return row.type === 'group' && row.location === 'footer';\n  }\n\n  readonly showFooterRowSignal = computed(() => {\n    const viewDelegate = this.viewDelegateSignal();\n    const showFooterRow = viewDelegate && (viewDelegate.summaryRowHeader != null || viewDelegate.columnFooter != null || viewDelegate.summaryRowEnd != null);\n    return showFooterRow;\n  });\n\n  readonly showFullSummaryRowSignal = computed(() => {\n    const viewDelegate = this.viewDelegateSignal();\n    return viewDelegate?.fullSummaryRow != null;\n  });\n}\n","<dbx-loading [context]=\"context\">\n  <dbx-loading class=\"dbx-table-view-loading\" [linear]=\"true\" [context]=\"dataLoadingContext\"></dbx-loading>\n  <section class=\"dbx-table-view\" infinite-scroll [infiniteScrollDistance]=\"scrollDistance()\" [infiniteScrollThrottle]=\"throttleScroll()\" [scrollWindow]=\"false\" (scrolled)=\"onScrollDown()\">\n    <table class=\"dbx-table-view-table\" mat-table [dataSource]=\"elements$\" [trackBy]=\"trackElementByFunctionSignal()\" multiTemplateDataRows>\n      <!-- Groups -->\n      <!-- Group Header column -->\n      <ng-container matColumnDef=\"groupHeaderColumn\" [sticky]=\"true\">\n        <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n          <dbx-table-group-header [group]=\"element.group\"></dbx-table-group-header>\n        </td>\n      </ng-container>\n\n      <!-- Group Footer column -->\n      <ng-container matColumnDef=\"groupFooterColumn\" [sticky]=\"true\">\n        <td mat-cell [attr.colspan]=\"visibleColumnsSignal()\" *matCellDef=\"let element\">\n          <dbx-table-group-footer [group]=\"element.group\"></dbx-table-group-footer>\n        </td>\n      </ng-container>\n\n      <!-- Items -->\n      <!-- Header/Item column -->\n      <ng-container [matColumnDef]=\"itemsColumnName\" [sticky]=\"true\">\n        <th mat-header-cell *matHeaderCellDef>\n          <dbx-table-input-cell></dbx-table-input-cell>\n        </th>\n        <td mat-cell *matCellDef=\"let element\">\n          @if (element.item) {\n            <dbx-table-item-header [item]=\"element.item\"></dbx-table-item-header>\n          }\n        </td>\n        <td mat-footer-cell *matFooterCellDef>\n          <dbx-table-summary-start-cell></dbx-table-summary-start-cell>\n        </td>\n      </ng-container>\n\n      <!-- Column Definitions -->\n      @for (column of innerColumnsSignal(); track column.columnName) {\n        <ng-container [matColumnDef]=\"column.columnName\" [sticky]=\"false\">\n          <th mat-header-cell *matHeaderCellDef>\n            <dbx-table-column-header [column]=\"column\"></dbx-table-column-header>\n          </th>\n          <td mat-cell *matCellDef=\"let element\">\n            @if (element.item) {\n              <dbx-table-item-cell [item]=\"element.item\" [column]=\"column\"></dbx-table-item-cell>\n            }\n          </td>\n          <td mat-footer-cell *matFooterCellDef>\n            <dbx-table-column-footer [column]=\"column\"></dbx-table-column-footer>\n          </td>\n        </ng-container>\n      }\n\n      <!-- Tail/Action column -->\n      <ng-container [matColumnDef]=\"actionsColumnName\" [stickyEnd]=\"true\">\n        <th mat-header-cell *matHeaderCellDef>\n          <dbx-table-action-cell></dbx-table-action-cell>\n        </th>\n        <td mat-cell *matCellDef=\"let element\">\n          @if (element.item) {\n            <dbx-table-item-action [item]=\"element.item\"></dbx-table-item-action>\n          }\n        </td>\n        <td mat-footer-cell *matFooterCellDef>\n          <dbx-table-summary-end-cell></dbx-table-summary-end-cell>\n        </td>\n      </ng-container>\n\n      <!-- Table View -->\n      <tr mat-header-row *matHeaderRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n\n      <tr mat-row *matRowDef=\"let row; columns: ['groupHeaderColumn']; when: showGroupHeaderRow\"></tr>\n      <tr mat-row *matRowDef=\"let row; columns: displayedColumnsSignal(); when: showItemRow\"></tr>\n      <tr mat-row *matRowDef=\"let row; columns: ['groupFooterColumn']; when: showGroupFooterRow\"></tr>\n\n      <tr [ngClass]=\"{ 'dbx-hide': !showFooterRowSignal() }\" mat-footer-row *matFooterRowDef=\"displayedColumnsSignal(); sticky: true\"></tr>\n    </table>\n  </section>\n  <!-- Full summary row -->\n  @if (showFullSummaryRowSignal()) {\n    <div class=\"mat-mdc-table dbx-table-view-full-summary-row-container\">\n      <div class=\"mdc-data-table__cell mat-mdc-row\">\n        <dbx-table-full-summary-row></dbx-table-full-summary-row>\n      </div>\n    </div>\n  }\n</dbx-loading>\n"]}
@@ -0,0 +1,34 @@
1
+ import { ChangeDetectionStrategy, Component, inject } from '@angular/core';
2
+ import { toSignal } from '@angular/core/rxjs-interop';
3
+ import { map, distinctUntilChanged } from 'rxjs';
4
+ import { DbxTableStore } from './table.store';
5
+ import { DbxInjectionComponent } from '@dereekb/dbx-core';
6
+ import * as i0 from "@angular/core";
7
+ /**
8
+ * A table header component used for injecting the input picker view.
9
+ */
10
+ export class DbxTableFullSummaryRowComponent {
11
+ tableStore = inject(DbxTableStore);
12
+ config$ = this.tableStore.viewDelegate$.pipe(map((x) => x.fullSummaryRow), distinctUntilChanged());
13
+ configSignal = toSignal(this.config$);
14
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableFullSummaryRowComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
15
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxTableFullSummaryRowComponent, isStandalone: true, selector: "dbx-table-full-summary-row", host: { classAttribute: "dbx-h100 dbx-flex-bar" }, ngImport: i0, template: `
16
+ <dbx-injection [config]="configSignal()"></dbx-injection>
17
+ `, isInline: true, dependencies: [{ kind: "component", type: DbxInjectionComponent, selector: "dbx-injection, [dbxInjection], [dbx-injection]", inputs: ["config", "template"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
18
+ }
19
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableFullSummaryRowComponent, decorators: [{
20
+ type: Component,
21
+ args: [{
22
+ selector: 'dbx-table-full-summary-row',
23
+ template: `
24
+ <dbx-injection [config]="configSignal()"></dbx-injection>
25
+ `,
26
+ changeDetection: ChangeDetectionStrategy.OnPush,
27
+ standalone: true,
28
+ imports: [DbxInjectionComponent],
29
+ host: {
30
+ class: 'dbx-h100 dbx-flex-bar'
31
+ }
32
+ }]
33
+ }] });
34
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuZnVsbHN1bW1hcnlyb3cuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LXdlYi90YWJsZS9zcmMvbGliL3RhYmxlLmZ1bGxzdW1tYXJ5cm93LmNvbXBvbmVudC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUMzRSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDdEQsT0FBTyxFQUFFLEdBQUcsRUFBRSxvQkFBb0IsRUFBRSxNQUFNLE1BQU0sQ0FBQztBQUNqRCxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzlDLE9BQU8sRUFBRSxxQkFBcUIsRUFBRSxNQUFNLG1CQUFtQixDQUFDOztBQUUxRDs7R0FFRztBQWFILE1BQU0sT0FBTywrQkFBK0I7SUFDakMsVUFBVSxHQUFHLE1BQU0sQ0FBQyxhQUFhLENBQUMsQ0FBQztJQUVuQyxPQUFPLEdBQUcsSUFBSSxDQUFDLFVBQVUsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUNuRCxHQUFHLENBQUMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLENBQUMsQ0FBQyxjQUFjLENBQUMsRUFDNUIsb0JBQW9CLEVBQUUsQ0FDdkIsQ0FBQztJQUVPLFlBQVksR0FBRyxRQUFRLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO3dHQVJwQywrQkFBK0I7NEZBQS9CLCtCQUErQix5SUFWaEM7O0dBRVQsNERBR1MscUJBQXFCOzs0RkFLcEIsK0JBQStCO2tCQVozQyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSw0QkFBNEI7b0JBQ3RDLFFBQVEsRUFBRTs7R0FFVDtvQkFDRCxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsVUFBVSxFQUFFLElBQUk7b0JBQ2hCLE9BQU8sRUFBRSxDQUFDLHFCQUFxQixDQUFDO29CQUNoQyxJQUFJLEVBQUU7d0JBQ0osS0FBSyxFQUFFLHVCQUF1QjtxQkFDL0I7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50LCBpbmplY3QgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHRvU2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgbWFwLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgRGJ4VGFibGVTdG9yZSB9IGZyb20gJy4vdGFibGUuc3RvcmUnO1xuaW1wb3J0IHsgRGJ4SW5qZWN0aW9uQ29tcG9uZW50IH0gZnJvbSAnQGRlcmVla2IvZGJ4LWNvcmUnO1xuXG4vKipcbiAqIEEgdGFibGUgaGVhZGVyIGNvbXBvbmVudCB1c2VkIGZvciBpbmplY3RpbmcgdGhlIGlucHV0IHBpY2tlciB2aWV3LlxuICovXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICdkYngtdGFibGUtZnVsbC1zdW1tYXJ5LXJvdycsXG4gIHRlbXBsYXRlOiBgXG4gICAgPGRieC1pbmplY3Rpb24gW2NvbmZpZ109XCJjb25maWdTaWduYWwoKVwiPjwvZGJ4LWluamVjdGlvbj5cbiAgYCxcbiAgY2hhbmdlRGV0ZWN0aW9uOiBDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneS5PblB1c2gsXG4gIHN0YW5kYWxvbmU6IHRydWUsXG4gIGltcG9ydHM6IFtEYnhJbmplY3Rpb25Db21wb25lbnRdLFxuICBob3N0OiB7XG4gICAgY2xhc3M6ICdkYngtaDEwMCBkYngtZmxleC1iYXInXG4gIH1cbn0pXG5leHBvcnQgY2xhc3MgRGJ4VGFibGVGdWxsU3VtbWFyeVJvd0NvbXBvbmVudCB7XG4gIHJlYWRvbmx5IHRhYmxlU3RvcmUgPSBpbmplY3QoRGJ4VGFibGVTdG9yZSk7XG5cbiAgcmVhZG9ubHkgY29uZmlnJCA9IHRoaXMudGFibGVTdG9yZS52aWV3RGVsZWdhdGUkLnBpcGUoXG4gICAgbWFwKCh4KSA9PiB4LmZ1bGxTdW1tYXJ5Um93KSxcbiAgICBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpXG4gICk7XG5cbiAgcmVhZG9ubHkgY29uZmlnU2lnbmFsID0gdG9TaWduYWwodGhpcy5jb25maWckKTtcbn1cbiJdfQ==
@@ -0,0 +1,20 @@
1
+ import { toObservable } from '@angular/core/rxjs-interop';
2
+ import { filterMaybe } from '@dereekb/rxjs';
3
+ import { Directive, inject, input } from '@angular/core';
4
+ import { distinctUntilChanged } from 'rxjs';
5
+ import { DbxTableStore } from './table.store';
6
+ import * as i0 from "@angular/core";
7
+ /**
8
+ * Abstract directive that has an element input.
9
+ */
10
+ export class AbstractDbxTableGroupDirective {
11
+ tableStore = inject((DbxTableStore));
12
+ group = input();
13
+ group$ = toObservable(this.group).pipe(filterMaybe(), distinctUntilChanged());
14
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AbstractDbxTableGroupDirective, deps: [], target: i0.ɵɵFactoryTarget.Directive });
15
+ static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "17.1.0", version: "18.2.13", type: AbstractDbxTableGroupDirective, inputs: { group: { classPropertyName: "group", publicName: "group", isSignal: true, isRequired: false, transformFunction: null } }, ngImport: i0 });
16
+ }
17
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: AbstractDbxTableGroupDirective, decorators: [{
18
+ type: Directive
19
+ }] });
20
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuZ3JvdXAuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcGFja2FnZXMvZGJ4LXdlYi90YWJsZS9zcmMvbGliL3RhYmxlLmdyb3VwLmRpcmVjdGl2ZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0sNEJBQTRCLENBQUM7QUFDMUQsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUM1QyxPQUFPLEVBQUUsU0FBUyxFQUFFLE1BQU0sRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekQsT0FBTyxFQUFFLG9CQUFvQixFQUFFLE1BQU0sTUFBTSxDQUFDO0FBQzVDLE9BQU8sRUFBRSxhQUFhLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBSTlDOztHQUVHO0FBRUgsTUFBTSxPQUFnQiw4QkFBOEI7SUFDekMsVUFBVSxHQUFHLE1BQU0sQ0FBQyxDQUFBLGFBQStCLENBQUEsQ0FBQyxDQUFDO0lBRXJELEtBQUssR0FBRyxLQUFLLEVBQWtDLENBQUM7SUFDaEQsTUFBTSxHQUFHLFlBQVksQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUMsSUFBSSxDQUFDLFdBQVcsRUFBRSxFQUFFLG9CQUFvQixFQUFFLENBQUMsQ0FBQzt3R0FKbkUsOEJBQThCOzRGQUE5Qiw4QkFBOEI7OzRGQUE5Qiw4QkFBOEI7a0JBRG5ELFNBQVMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyB0b09ic2VydmFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlL3J4anMtaW50ZXJvcCc7XG5pbXBvcnQgeyBmaWx0ZXJNYXliZSB9IGZyb20gJ0BkZXJlZWtiL3J4anMnO1xuaW1wb3J0IHsgRGlyZWN0aXZlLCBpbmplY3QsIGlucHV0IH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQgeyBkaXN0aW5jdFVudGlsQ2hhbmdlZCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgRGJ4VGFibGVTdG9yZSB9IGZyb20gJy4vdGFibGUuc3RvcmUnO1xuaW1wb3J0IHsgdHlwZSBNYXliZSB9IGZyb20gJ0BkZXJlZWtiL3V0aWwnO1xuaW1wb3J0IHsgRGJ4VGFibGVJdGVtR3JvdXAgfSBmcm9tICcuL3RhYmxlJztcblxuLyoqXG4gKiBBYnN0cmFjdCBkaXJlY3RpdmUgdGhhdCBoYXMgYW4gZWxlbWVudCBpbnB1dC5cbiAqL1xuQERpcmVjdGl2ZSgpXG5leHBvcnQgYWJzdHJhY3QgY2xhc3MgQWJzdHJhY3REYnhUYWJsZUdyb3VwRGlyZWN0aXZlPFQsIEMgPSB1bmtub3duLCBHID0gdW5rbm93bj4ge1xuICByZWFkb25seSB0YWJsZVN0b3JlID0gaW5qZWN0KERieFRhYmxlU3RvcmU8dW5rbm93biwgQywgVCwgRz4pO1xuXG4gIHJlYWRvbmx5IGdyb3VwID0gaW5wdXQ8TWF5YmU8RGJ4VGFibGVJdGVtR3JvdXA8VCwgRz4+PigpO1xuICByZWFkb25seSBncm91cCQgPSB0b09ic2VydmFibGUodGhpcy5ncm91cCkucGlwZShmaWx0ZXJNYXliZSgpLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCgpKTtcbn1cbiJdfQ==
@@ -0,0 +1,35 @@
1
+ import { ChangeDetectionStrategy, Component } from '@angular/core';
2
+ import { toSignal } from '@angular/core/rxjs-interop';
3
+ import { map, distinctUntilChanged } from 'rxjs';
4
+ import { maybeValueFromObservableOrValue } from '@dereekb/rxjs';
5
+ import { DbxInjectionComponent } from '@dereekb/dbx-core';
6
+ import { AbstractDbxTableGroupDirective } from './table.group.directive';
7
+ import * as i0 from "@angular/core";
8
+ export class DbxTableGroupFooterComponent extends AbstractDbxTableGroupDirective {
9
+ config$ = this.tableStore.viewDelegate$.pipe(map((viewDelegate) => {
10
+ const groupFooter = viewDelegate.groupFooter;
11
+ let obs = undefined;
12
+ if (groupFooter) {
13
+ obs = this.group$.pipe(map((x) => groupFooter(x)));
14
+ }
15
+ return obs;
16
+ }), maybeValueFromObservableOrValue(), distinctUntilChanged());
17
+ configSignal = toSignal(this.config$);
18
+ static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableGroupFooterComponent, deps: null, target: i0.ɵɵFactoryTarget.Component });
19
+ static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: DbxTableGroupFooterComponent, isStandalone: true, selector: "dbx-table-group-footer", usesInheritance: true, ngImport: i0, template: `
20
+ <dbx-injection [config]="configSignal()"></dbx-injection>
21
+ `, isInline: true, dependencies: [{ kind: "component", type: DbxInjectionComponent, selector: "dbx-injection, [dbxInjection], [dbx-injection]", inputs: ["config", "template"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
22
+ }
23
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: DbxTableGroupFooterComponent, decorators: [{
24
+ type: Component,
25
+ args: [{
26
+ selector: 'dbx-table-group-footer',
27
+ template: `
28
+ <dbx-injection [config]="configSignal()"></dbx-injection>
29
+ `,
30
+ changeDetection: ChangeDetectionStrategy.OnPush,
31
+ imports: [DbxInjectionComponent],
32
+ standalone: true
33
+ }]
34
+ }] });
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGFibGUuZ3JvdXAuZm9vdGVyLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3BhY2thZ2VzL2RieC13ZWIvdGFibGUvc3JjL2xpYi90YWJsZS5ncm91cC5mb290ZXIuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSx1QkFBdUIsRUFBRSxTQUFTLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDbkUsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLDRCQUE0QixDQUFDO0FBQ3RELE9BQU8sRUFBRSxHQUFHLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSxNQUFNLENBQUM7QUFDakQsT0FBTyxFQUEwQiwrQkFBK0IsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN4RixPQUFPLEVBQUUscUJBQXFCLEVBQStCLE1BQU0sbUJBQW1CLENBQUM7QUFDdkYsT0FBTyxFQUFFLDhCQUE4QixFQUFFLE1BQU0seUJBQXlCLENBQUM7O0FBV3pFLE1BQU0sT0FBTyw0QkFBZ0MsU0FBUSw4QkFBaUM7SUFDM0UsT0FBTyxHQUFHLElBQUksQ0FBQyxVQUFVLENBQUMsYUFBYSxDQUFDLElBQUksQ0FDbkQsR0FBRyxDQUFDLENBQUMsWUFBWSxFQUFFLEVBQUU7UUFDbkIsTUFBTSxXQUFXLEdBQUcsWUFBWSxDQUFDLFdBQVcsQ0FBQztRQUM3QyxJQUFJLEdBQUcsR0FBd0QsU0FBUyxDQUFDO1FBRXpFLElBQUksV0FBVyxFQUFFLENBQUM7WUFDaEIsR0FBRyxHQUFHLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEdBQUcsQ0FBQyxDQUFDLENBQUMsRUFBRSxFQUFFLENBQUMsV0FBVyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQztRQUNyRCxDQUFDO1FBRUQsT0FBTyxHQUFHLENBQUM7SUFDYixDQUFDLENBQUMsRUFDRiwrQkFBK0IsRUFBRSxFQUNqQyxvQkFBb0IsRUFBRSxDQUN2QixDQUFDO0lBRU8sWUFBWSxHQUFHLFFBQVEsQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDLENBQUM7d0dBaEJwQyw0QkFBNEI7NEZBQTVCLDRCQUE0Qix5R0FQN0I7O0dBRVQsNERBRVMscUJBQXFCOzs0RkFHcEIsNEJBQTRCO2tCQVR4QyxTQUFTO21CQUFDO29CQUNULFFBQVEsRUFBRSx3QkFBd0I7b0JBQ2xDLFFBQVEsRUFBRTs7R0FFVDtvQkFDRCxlQUFlLEVBQUUsdUJBQXVCLENBQUMsTUFBTTtvQkFDL0MsT0FBTyxFQUFFLENBQUMscUJBQXFCLENBQUM7b0JBQ2hDLFVBQVUsRUFBRSxJQUFJO2lCQUNqQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENoYW5nZURldGVjdGlvblN0cmF0ZWd5LCBDb21wb25lbnQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IHRvU2lnbmFsIH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuaW1wb3J0IHsgbWFwLCBkaXN0aW5jdFVudGlsQ2hhbmdlZCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgTWF5YmVPYnNlcnZhYmxlT3JWYWx1ZSwgbWF5YmVWYWx1ZUZyb21PYnNlcnZhYmxlT3JWYWx1ZSB9IGZyb20gJ0BkZXJlZWtiL3J4anMnO1xuaW1wb3J0IHsgRGJ4SW5qZWN0aW9uQ29tcG9uZW50LCBEYnhJbmplY3Rpb25Db21wb25lbnRDb25maWcgfSBmcm9tICdAZGVyZWVrYi9kYngtY29yZSc7XG5pbXBvcnQgeyBBYnN0cmFjdERieFRhYmxlR3JvdXBEaXJlY3RpdmUgfSBmcm9tICcuL3RhYmxlLmdyb3VwLmRpcmVjdGl2ZSc7XG5cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ2RieC10YWJsZS1ncm91cC1mb290ZXInLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkYngtaW5qZWN0aW9uIFtjb25maWddPVwiY29uZmlnU2lnbmFsKClcIj48L2RieC1pbmplY3Rpb24+XG4gIGAsXG4gIGNoYW5nZURldGVjdGlvbjogQ2hhbmdlRGV0ZWN0aW9uU3RyYXRlZ3kuT25QdXNoLFxuICBpbXBvcnRzOiBbRGJ4SW5qZWN0aW9uQ29tcG9uZW50XSxcbiAgc3RhbmRhbG9uZTogdHJ1ZVxufSlcbmV4cG9ydCBjbGFzcyBEYnhUYWJsZUdyb3VwRm9vdGVyQ29tcG9uZW50PFQ+IGV4dGVuZHMgQWJzdHJhY3REYnhUYWJsZUdyb3VwRGlyZWN0aXZlPFQ+IHtcbiAgcmVhZG9ubHkgY29uZmlnJCA9IHRoaXMudGFibGVTdG9yZS52aWV3RGVsZWdhdGUkLnBpcGUoXG4gICAgbWFwKCh2aWV3RGVsZWdhdGUpID0+IHtcbiAgICAgIGNvbnN0IGdyb3VwRm9vdGVyID0gdmlld0RlbGVnYXRlLmdyb3VwRm9vdGVyO1xuICAgICAgbGV0IG9iczogTWF5YmVPYnNlcnZhYmxlT3JWYWx1ZTxEYnhJbmplY3Rpb25Db21wb25lbnRDb25maWc+ID0gdW5kZWZpbmVkO1xuXG4gICAgICBpZiAoZ3JvdXBGb290ZXIpIHtcbiAgICAgICAgb2JzID0gdGhpcy5ncm91cCQucGlwZShtYXAoKHgpID0+IGdyb3VwRm9vdGVyKHgpKSk7XG4gICAgICB9XG5cbiAgICAgIHJldHVybiBvYnM7XG4gICAgfSksXG4gICAgbWF5YmVWYWx1ZUZyb21PYnNlcnZhYmxlT3JWYWx1ZSgpLFxuICAgIGRpc3RpbmN0VW50aWxDaGFuZ2VkKClcbiAgKTtcblxuICByZWFkb25seSBjb25maWdTaWduYWwgPSB0b1NpZ25hbCh0aGlzLmNvbmZpZyQpO1xufVxuIl19