@elderbyte/ngx-starter 13.7.7 → 13.8.1

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 (22) hide show
  1. package/esm2020/lib/common/data/data-context/data-context-builder.mjs +4 -3
  2. package/esm2020/lib/common/templates/template-slot-manager.mjs +34 -6
  3. package/esm2020/lib/common/utils/filter-util.mjs +11 -2
  4. package/esm2020/lib/components/navigation/toolbar/elder-toolbar.service.mjs +5 -3
  5. package/esm2020/lib/components/navigation/toolbar/toolbar/elder-toolbar.component.mjs +17 -13
  6. package/esm2020/lib/components/navigation/toolbar/toolbar-column-position.mjs +1 -1
  7. package/esm2020/lib/components/select/elder-select/elder-select.component.mjs +2 -17
  8. package/esm2020/lib/components/select/elder-select-base.mjs +2 -1
  9. package/esm2020/lib/components/select/multi/elder-multi-select-base.mjs +13 -4
  10. package/esm2020/lib/components/select/multi/elder-multi-select-chips/elder-multi-select-chips.component.mjs +3 -3
  11. package/esm2020/lib/components/shell/shell/elder-shell.component.mjs +1 -1
  12. package/fesm2015/elderbyte-ngx-starter.mjs +122 -83
  13. package/fesm2015/elderbyte-ngx-starter.mjs.map +1 -1
  14. package/fesm2020/elderbyte-ngx-starter.mjs +121 -83
  15. package/fesm2020/elderbyte-ngx-starter.mjs.map +1 -1
  16. package/lib/common/templates/template-slot-manager.d.ts +13 -1
  17. package/lib/components/navigation/toolbar/toolbar/elder-toolbar.component.d.ts +13 -10
  18. package/lib/components/navigation/toolbar/toolbar-column-position.d.ts +9 -1
  19. package/lib/components/select/elder-select/elder-select.component.d.ts +0 -11
  20. package/lib/components/select/multi/elder-multi-select-base.d.ts +6 -0
  21. package/lib/components/shell/shell/elder-shell.component.d.ts +3 -2
  22. package/package.json +1 -1
@@ -169,4 +169,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.4", ngImpor
169
169
  type: ViewChild,
170
170
  args: ['rightSideDetail', { static: true }]
171
171
  }] } });
172
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-shell.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/shell/shell/elder-shell.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/shell/shell/elder-shell.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy, ChangeDetectorRef,\n  Component,\n  ContentChild,\n  Directive,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  ViewChild,\n  ViewContainerRef\n} from '@angular/core';\nimport {ElderShellService} from '../elder-shell.service';\nimport { MatSidenav } from '@angular/material/sidenav';\nimport {ElderRouteOutletDrawerService} from '../drawers/elder-route-outlet-drawer.service';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {Observable, Subscription} from 'rxjs';\nimport {ShellContentSlot} from '../shell-content-slot';\nimport {tap} from 'rxjs/operators';\n\n\n@Directive({selector: '[elderShellSideLeft], [ebsShellSideLeft]'})\nexport class ElderShellSideLeftDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Directive({selector: '[elderShellSideRight], [ebsShellSideRight]'})\nexport class ElderShellSideRightDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Directive({selector: '[elderShellCenter], [ebsShellCenter]'})\nexport class ElderShellCenterDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Component({\n  selector: 'elder-shell, ebs-shell', // ebs-* prefix is deprecated!\n  templateUrl: './elder-shell.component.html',\n  styleUrls: ['./elder-shell.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ElderShellComponent implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger('ElderShellComponent');\n\n  /** Controls if the SideNav toggle should be displayed. Default: true */\n  @Input()\n  public sideNavToggleEnabled = true;\n\n  @Input()\n  public color = 'primary';\n\n  @Input()\n  public menuColor = 'accent';\n\n  @Input()\n  public leftSideAutoFocus = false;\n\n  @Input()\n  public rightSideAutoFocus = false;\n\n  @ContentChild(ElderShellSideLeftDirective, { read: TemplateRef, static: true })\n  public sideContentLeft: TemplateRef<any>;\n\n  @ContentChild(ElderShellSideRightDirective, { read: TemplateRef, static: true })\n  public sideContentRight: TemplateRef<any>;\n\n  @ContentChild(ElderShellCenterDirective, { read: TemplateRef, static: true })\n  public centerContent: TemplateRef<any>;\n\n  @ViewChild('rightSideDetail', { static: true })\n  public rightSideDrawer: MatSidenav;\n\n  public rightSideOutletName = 'side';\n\n  public leftSideContentOpen$: Observable<boolean>;\n\n\n  public readonly headerTemplate$: Observable<TemplateRef<any>>;\n  public readonly centerTemplate$: Observable<TemplateRef<any>>;\n  public readonly footerTemplate$: Observable<TemplateRef<any>>;\n\n  private _sub: Subscription;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private shellService: ElderShellService,\n    private outletDrawerService: ElderRouteOutletDrawerService,\n    private changeDetectorRef: ChangeDetectorRef\n  ) {\n    this.headerTemplate$ = shellService.activeSlotTemplate('header').pipe(\n      tap(() => this.checkSoon())\n    );\n    this.centerTemplate$ = shellService.activeSlotTemplate('center').pipe(\n      tap(() => this.checkSoon())\n    );\n    this.footerTemplate$ = shellService.activeSlotTemplate('footer').pipe(\n      tap(() => this.checkSoon())\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this.outletDrawerService.registerOutletDrawer(\n      this.rightSideOutletName,\n      this.rightSideDrawer\n    );\n\n    this._sub = this.outletDrawerService.drawerVisibilityChange.subscribe(\n      drawer =>  this.changeDetectorRef.markForCheck()\n    );\n\n    this.leftSideContentOpen$ =  this.shellService.navigationOpenChange;\n  }\n\n  public ngOnDestroy(): void {\n\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public closeLeftSideContent() {\n    this.shellService.closeSideNav();\n  }\n\n  public toggleSideNav(): void {\n    this.shellService.toggleSidenav();\n  }\n\n  public onBackdropClick(event: any): void {\n    if (this.shellService.isSideContentActive()) {\n      this.shellService.closeSideContent();\n    }\n  }\n\n  public onEscapeRightSide(event: Event): void {\n    const target = event.target as HTMLElement;\n    if (target && this.isContained(target, 'elder-right-side-detail')) {\n      // Ensure we only react to Esc Events targeted at our side nav!\n      if (this.shellService.isSideContentActive()) {\n        this.shellService.closeSideContent();\n      }\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private isContained(child: HTMLElement, id: string): boolean {\n\n    if (child.id === id) {\n      return true;\n    }\n    const maxDepth = 1;\n    let depth = 0;\n    let node = child.parentNode as HTMLElement;\n    while (node) {\n      if (node.id === id) {\n        return true;\n      }\n      node = node.parentNode as HTMLElement;\n      if (maxDepth >= ++depth) { break; }\n    }\n    return false;\n  }\n\n  private checkSoon(): void {\n    setTimeout(() => {\n      this.changeDetectorRef.markForCheck();\n    }, 5);\n  }\n}\n","\n<mat-sidenav-container style=\"height:100%; width: 100%\" (backdropClick)=\"onBackdropClick($event)\">\n\n  <!-- Left Side Nav -->\n  <mat-sidenav position=\"start\"\n               mode=\"over\"\n               [fixedInViewport]=\"true\"\n               [autoFocus]=\"leftSideAutoFocus\"\n               [opened]=\"leftSideContentOpen$ | async\"\n               (closed)=\"closeLeftSideContent()\">\n    <div fxFill fxLayout=\"column\">\n      <ng-container *ngTemplateOutlet=\"sideContentLeft || fallbackSideContentLeft;\"></ng-container>\n    </div>\n  </mat-sidenav>\n\n  <!-- Main Content -->\n  <mat-sidenav-content>\n    <div fxLayout=\"column\" fxFill>\n      <ng-container *ngTemplateOutlet=\"centerContent || fallbackCenterContent;\"></ng-container>\n    </div>\n  </mat-sidenav-content>\n\n  <!-- Right Side Detail -->\n  <mat-sidenav mode=\"over\" #rightSideDetail id=\"elder-right-side-detail\"\n               position=\"end\"\n               [fixedInViewport]=\"true\"\n               [autoFocus]=\"rightSideAutoFocus\"\n               [disableClose]=\"true\" (keydown.escape)=\"onEscapeRightSide($event)\">\n    <div fxFill fxLayout=\"column\">\n      <ng-container *ngTemplateOutlet=\"sideContentRight || fallbackSideContentRight;\"></ng-container>\n    </div>\n  </mat-sidenav>\n\n</mat-sidenav-container>\n\n<ng-template #fallbackSideContentLeft>\n  <div fxLayout=\"column\" fxFlex>\n    <p class=\"noselect\">No Left Side Content Defined!</p>\n  </div>\n</ng-template>\n\n<ng-template #fallbackSideContentRight>\n  <router-outlet name=\"side\" class=\"router-flex\"></router-outlet>\n</ng-template>\n\n<ng-template #fallbackCenterContent>\n  <div fxLayout=\"column\" fxFill>\n\n    <!-- Header -->\n    <ng-container *ngTemplateOutlet=\"(headerTemplate$ | async) || defaultHeaderTemplate\"></ng-container>\n\n    <!-- Center -->\n    <ng-container *ngTemplateOutlet=\"(centerTemplate$ | async) || defaultCenterTemplate\"></ng-container>\n\n    <!-- Footer -->\n    <ng-container *ngTemplateOutlet=\"(footerTemplate$ | async) || defaultFooterTemplate\"></ng-container>\n\n  </div>\n</ng-template>\n\n\n<ng-template #defaultHeaderTemplate>\n\n  <elder-toolbar [color]=\"color\">\n\n    <!-- Toolbar Prefix: Sidenav Toggle -->\n    <mat-toolbar *ngIf=\"sideNavToggleEnabled\" fxFlex=\"0\"\n                 fxLayout=\"row\" fxLayoutAlign=\"center center\"\n                 [color]=\"menuColor\">\n      <button mat-icon-button type=\"button\" (click)=\"toggleSideNav()\">\n        <mat-icon>menu</mat-icon>\n      </button>\n    </mat-toolbar>\n  </elder-toolbar>\n\n</ng-template>\n\n<ng-template #defaultCenterTemplate>\n\n  <!-- Primary Router Outlet -->\n  <router-outlet class=\"router-flex\"></router-outlet>\n\n</ng-template>\n\n<ng-template #defaultFooterTemplate>\n  <!-- Default Footer is empty -->\n</ng-template>\n"]}
172
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-shell.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/shell/shell/elder-shell.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/shell/shell/elder-shell.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  ChangeDetectionStrategy, ChangeDetectorRef,\n  Component,\n  ContentChild,\n  Directive,\n  Input,\n  OnDestroy,\n  OnInit,\n  TemplateRef,\n  ViewChild,\n  ViewContainerRef\n} from '@angular/core';\nimport {ElderShellService} from '../elder-shell.service';\nimport { MatSidenav } from '@angular/material/sidenav';\nimport {ElderRouteOutletDrawerService} from '../drawers/elder-route-outlet-drawer.service';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {Observable, Subscription} from 'rxjs';\nimport {ShellContentSlot} from '../shell-content-slot';\nimport {tap} from 'rxjs/operators';\nimport {ThemePalette} from '@angular/material/core/common-behaviors/color';\n\n\n@Directive({selector: '[elderShellSideLeft], [ebsShellSideLeft]'})\nexport class ElderShellSideLeftDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Directive({selector: '[elderShellSideRight], [ebsShellSideRight]'})\nexport class ElderShellSideRightDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Directive({selector: '[elderShellCenter], [ebsShellCenter]'})\nexport class ElderShellCenterDirective {\n  constructor(\n    public templateRef: TemplateRef<any>,\n    public viewContainer: ViewContainerRef) { }\n}\n\n@Component({\n  selector: 'elder-shell, ebs-shell', // ebs-* prefix is deprecated!\n  templateUrl: './elder-shell.component.html',\n  styleUrls: ['./elder-shell.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class ElderShellComponent implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger('ElderShellComponent');\n\n  /** Controls if the SideNav toggle should be displayed. Default: true */\n  @Input()\n  public sideNavToggleEnabled = true;\n\n  @Input()\n  public color: ThemePalette = 'primary';\n\n  @Input()\n  public menuColor: ThemePalette = 'accent';\n\n  @Input()\n  public leftSideAutoFocus = false;\n\n  @Input()\n  public rightSideAutoFocus = false;\n\n  @ContentChild(ElderShellSideLeftDirective, { read: TemplateRef, static: true })\n  public sideContentLeft: TemplateRef<any>;\n\n  @ContentChild(ElderShellSideRightDirective, { read: TemplateRef, static: true })\n  public sideContentRight: TemplateRef<any>;\n\n  @ContentChild(ElderShellCenterDirective, { read: TemplateRef, static: true })\n  public centerContent: TemplateRef<any>;\n\n  @ViewChild('rightSideDetail', { static: true })\n  public rightSideDrawer: MatSidenav;\n\n  public rightSideOutletName = 'side';\n\n  public leftSideContentOpen$: Observable<boolean>;\n\n\n  public readonly headerTemplate$: Observable<TemplateRef<any>>;\n  public readonly centerTemplate$: Observable<TemplateRef<any>>;\n  public readonly footerTemplate$: Observable<TemplateRef<any>>;\n\n  private _sub: Subscription;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private shellService: ElderShellService,\n    private outletDrawerService: ElderRouteOutletDrawerService,\n    private changeDetectorRef: ChangeDetectorRef\n  ) {\n    this.headerTemplate$ = shellService.activeSlotTemplate('header').pipe(\n      tap(() => this.checkSoon())\n    );\n    this.centerTemplate$ = shellService.activeSlotTemplate('center').pipe(\n      tap(() => this.checkSoon())\n    );\n    this.footerTemplate$ = shellService.activeSlotTemplate('footer').pipe(\n      tap(() => this.checkSoon())\n    );\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    this.outletDrawerService.registerOutletDrawer(\n      this.rightSideOutletName,\n      this.rightSideDrawer\n    );\n\n    this._sub = this.outletDrawerService.drawerVisibilityChange.subscribe(\n      drawer =>  this.changeDetectorRef.markForCheck()\n    );\n\n    this.leftSideContentOpen$ =  this.shellService.navigationOpenChange;\n  }\n\n  public ngOnDestroy(): void {\n\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public closeLeftSideContent() {\n    this.shellService.closeSideNav();\n  }\n\n  public toggleSideNav(): void {\n    this.shellService.toggleSidenav();\n  }\n\n  public onBackdropClick(event: any): void {\n    if (this.shellService.isSideContentActive()) {\n      this.shellService.closeSideContent();\n    }\n  }\n\n  public onEscapeRightSide(event: Event): void {\n    const target = event.target as HTMLElement;\n    if (target && this.isContained(target, 'elder-right-side-detail')) {\n      // Ensure we only react to Esc Events targeted at our side nav!\n      if (this.shellService.isSideContentActive()) {\n        this.shellService.closeSideContent();\n      }\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private isContained(child: HTMLElement, id: string): boolean {\n\n    if (child.id === id) {\n      return true;\n    }\n    const maxDepth = 1;\n    let depth = 0;\n    let node = child.parentNode as HTMLElement;\n    while (node) {\n      if (node.id === id) {\n        return true;\n      }\n      node = node.parentNode as HTMLElement;\n      if (maxDepth >= ++depth) { break; }\n    }\n    return false;\n  }\n\n  private checkSoon(): void {\n    setTimeout(() => {\n      this.changeDetectorRef.markForCheck();\n    }, 5);\n  }\n}\n","\n<mat-sidenav-container style=\"height:100%; width: 100%\" (backdropClick)=\"onBackdropClick($event)\">\n\n  <!-- Left Side Nav -->\n  <mat-sidenav position=\"start\"\n               mode=\"over\"\n               [fixedInViewport]=\"true\"\n               [autoFocus]=\"leftSideAutoFocus\"\n               [opened]=\"leftSideContentOpen$ | async\"\n               (closed)=\"closeLeftSideContent()\">\n    <div fxFill fxLayout=\"column\">\n      <ng-container *ngTemplateOutlet=\"sideContentLeft || fallbackSideContentLeft;\"></ng-container>\n    </div>\n  </mat-sidenav>\n\n  <!-- Main Content -->\n  <mat-sidenav-content>\n    <div fxLayout=\"column\" fxFill>\n      <ng-container *ngTemplateOutlet=\"centerContent || fallbackCenterContent;\"></ng-container>\n    </div>\n  </mat-sidenav-content>\n\n  <!-- Right Side Detail -->\n  <mat-sidenav mode=\"over\" #rightSideDetail id=\"elder-right-side-detail\"\n               position=\"end\"\n               [fixedInViewport]=\"true\"\n               [autoFocus]=\"rightSideAutoFocus\"\n               [disableClose]=\"true\" (keydown.escape)=\"onEscapeRightSide($event)\">\n    <div fxFill fxLayout=\"column\">\n      <ng-container *ngTemplateOutlet=\"sideContentRight || fallbackSideContentRight;\"></ng-container>\n    </div>\n  </mat-sidenav>\n\n</mat-sidenav-container>\n\n<ng-template #fallbackSideContentLeft>\n  <div fxLayout=\"column\" fxFlex>\n    <p class=\"noselect\">No Left Side Content Defined!</p>\n  </div>\n</ng-template>\n\n<ng-template #fallbackSideContentRight>\n  <router-outlet name=\"side\" class=\"router-flex\"></router-outlet>\n</ng-template>\n\n<ng-template #fallbackCenterContent>\n  <div fxLayout=\"column\" fxFill>\n\n    <!-- Header -->\n    <ng-container *ngTemplateOutlet=\"(headerTemplate$ | async) || defaultHeaderTemplate\"></ng-container>\n\n    <!-- Center -->\n    <ng-container *ngTemplateOutlet=\"(centerTemplate$ | async) || defaultCenterTemplate\"></ng-container>\n\n    <!-- Footer -->\n    <ng-container *ngTemplateOutlet=\"(footerTemplate$ | async) || defaultFooterTemplate\"></ng-container>\n\n  </div>\n</ng-template>\n\n\n<ng-template #defaultHeaderTemplate>\n\n  <elder-toolbar [color]=\"color\">\n\n    <!-- Toolbar Prefix: Sidenav Toggle -->\n    <mat-toolbar *ngIf=\"sideNavToggleEnabled\" fxFlex=\"0\"\n                 fxLayout=\"row\" fxLayoutAlign=\"center center\"\n                 [color]=\"menuColor\">\n      <button mat-icon-button type=\"button\" (click)=\"toggleSideNav()\">\n        <mat-icon>menu</mat-icon>\n      </button>\n    </mat-toolbar>\n  </elder-toolbar>\n\n</ng-template>\n\n<ng-template #defaultCenterTemplate>\n\n  <!-- Primary Router Outlet -->\n  <router-outlet class=\"router-flex\"></router-outlet>\n\n</ng-template>\n\n<ng-template #defaultFooterTemplate>\n  <!-- Default Footer is empty -->\n</ng-template>\n"]}
@@ -4417,6 +4417,57 @@ class DelegateContinuableDataSource extends DelegateDataSource {
4417
4417
  }
4418
4418
  }
4419
4419
 
4420
+ class FilterUtil {
4421
+ static filterData(data, filters) {
4422
+ FilterUtil.logger.debug('Filtering data with ' + data.length + ' items.', filters);
4423
+ if (filters && filters.length > 0) {
4424
+ for (const filter of filters) {
4425
+ data = data.filter(e => FilterUtil.matches(e, filter));
4426
+ }
4427
+ }
4428
+ return data;
4429
+ }
4430
+ static matches(entity, filter) {
4431
+ if (filter.value === null || filter.value === undefined) {
4432
+ return true;
4433
+ }
4434
+ let value;
4435
+ if (entity === null || entity === undefined) {
4436
+ value = null;
4437
+ }
4438
+ else if (typeof entity === 'object') {
4439
+ value = entity[filter.key];
4440
+ }
4441
+ else {
4442
+ value = String(entity); // Support filtering primitive values
4443
+ }
4444
+ if (Array.isArray(filter.value)) {
4445
+ for (const val of filter.value) {
4446
+ if (FilterUtil.matchesValue(value, val)) {
4447
+ return true;
4448
+ }
4449
+ }
4450
+ return false;
4451
+ }
4452
+ else {
4453
+ return FilterUtil.matchesValue(value, filter.value);
4454
+ }
4455
+ }
4456
+ static matchesValue(haystack, needle) {
4457
+ if (haystack === needle) {
4458
+ return true;
4459
+ }
4460
+ if (haystack === null || haystack === undefined) {
4461
+ return false;
4462
+ }
4463
+ const str = String(haystack);
4464
+ const match = str.toLowerCase().startsWith(needle.toLowerCase());
4465
+ // FilterUtil.logger.debug('Haystack: ' + str + ' starts with needle: ' + needle + ' --> ' + match);
4466
+ return match;
4467
+ }
4468
+ }
4469
+ FilterUtil.logger = LoggerFactory.getLogger(FilterUtil.constructor.name);
4470
+
4420
4471
  /**
4421
4472
  * Provides the ability to build a IDataContext<T>.
4422
4473
  */
@@ -4542,12 +4593,12 @@ class DataContextBuilder {
4542
4593
  * *
4543
4594
  **************************************************************************/
4544
4595
  buildLocal(items, idProperty) {
4545
- return this.build(LocalListDataSource.from(items, idProperty));
4596
+ return this.build(LocalListDataSource.from(items, idProperty, this._localSort, FilterUtil.filterData));
4546
4597
  }
4547
4598
  buildLocalActivePaged(data, idProperty) {
4548
4599
  this._skipLocalSort = true;
4549
4600
  this.activePaged();
4550
- return this.build(LocalPagedDataSource.from(data, idProperty, this._localSort));
4601
+ return this.build(LocalPagedDataSource.from(data, idProperty, this._localSort, FilterUtil.filterData));
4551
4602
  }
4552
4603
  /***************************************************************************
4553
4604
  * *
@@ -6107,48 +6158,6 @@ class PropertyPathUtil {
6107
6158
  }
6108
6159
  }
6109
6160
 
6110
- class FilterUtil {
6111
- static filterData(data, filters) {
6112
- FilterUtil.logger.debug('Filtering data with ' + data.length + ' items.', filters);
6113
- if (filters && filters.length > 0) {
6114
- for (const filter of filters) {
6115
- data = data.filter(e => FilterUtil.matches(e, filter));
6116
- }
6117
- }
6118
- return data;
6119
- }
6120
- static matches(entity, filter) {
6121
- if (filter.value === null || filter.value === undefined) {
6122
- return true;
6123
- }
6124
- const value = entity[filter.key];
6125
- if (Array.isArray(filter.value)) {
6126
- for (const val of filter.value) {
6127
- if (FilterUtil.matchesValue(value, val)) {
6128
- return true;
6129
- }
6130
- }
6131
- return false;
6132
- }
6133
- else {
6134
- return FilterUtil.matchesValue(value, filter.value);
6135
- }
6136
- }
6137
- static matchesValue(haystack, needle) {
6138
- if (haystack === needle) {
6139
- return true;
6140
- }
6141
- if (haystack === null || haystack === undefined) {
6142
- return false;
6143
- }
6144
- const str = String(haystack);
6145
- const match = str.toLowerCase().startsWith(needle.toLowerCase());
6146
- // FilterUtil.logger.debug('Haystack: ' + str + ' starts with needle: ' + needle + ' --> ' + match);
6147
- return match;
6148
- }
6149
- }
6150
- FilterUtil.logger = LoggerFactory.getLogger(FilterUtil.constructor.name);
6151
-
6152
6161
  class NextNumberUtil {
6153
6162
  /***************************************************************************
6154
6163
  * *
@@ -14677,7 +14686,11 @@ class TemplateSlotManager {
14677
14686
  * Constructors *
14678
14687
  * *
14679
14688
  **************************************************************************/
14680
- constructor(availableSlots) {
14689
+ /**
14690
+ * Create a new Template Slot Manager.
14691
+ * @param slotDefinitions Slot definitions, supporting synonyms
14692
+ */
14693
+ constructor(slotDefinitions) {
14681
14694
  /***************************************************************************
14682
14695
  * *
14683
14696
  * Fields *
@@ -14687,10 +14700,15 @@ class TemplateSlotManager {
14687
14700
  /** Default templates (App Level) */
14688
14701
  this._defaultSlotTemplates = new Map();
14689
14702
  /** Custom templates (Component Level) */
14690
- this._slotTemplates = new Map(); // TODO Still required??
14703
+ this._slotTemplates = new Map();
14691
14704
  this._activeSlotTemplates = new Map();
14692
- for (const slot of availableSlots) {
14693
- this._activeSlotTemplates.set(slot, new BehaviorSubject(null));
14705
+ /**
14706
+ * Key: A synonym slot name
14707
+ * Value: The primary slot the synonym is resolving to
14708
+ */
14709
+ this._slotSynonyms = new Map();
14710
+ for (const slotDefinition of slotDefinitions) {
14711
+ this.initSlot(slotDefinition);
14694
14712
  }
14695
14713
  }
14696
14714
  /***************************************************************************
@@ -14703,6 +14721,7 @@ class TemplateSlotManager {
14703
14721
  * over time.
14704
14722
  */
14705
14723
  activeSlotTemplate(slot) {
14724
+ slot = this.resolveSlot(slot);
14706
14725
  const activeColumn = this._activeSlotTemplates.get(slot);
14707
14726
  if (activeColumn) {
14708
14727
  return activeColumn.asObservable();
@@ -14719,7 +14738,7 @@ class TemplateSlotManager {
14719
14738
  * @param isDefault if the given template should be used as global default
14720
14739
  */
14721
14740
  registerSlotTemplate(slot, template, isDefault = false) {
14722
- // this.logger.trace('Registering toolbar column at slot ' + slot, template);
14741
+ slot = this.resolveSlot(slot);
14723
14742
  if (isDefault) {
14724
14743
  this._defaultSlotTemplates.set(slot, template);
14725
14744
  }
@@ -14736,6 +14755,7 @@ class TemplateSlotManager {
14736
14755
  */
14737
14756
  deregisterSlotTemplate(template, slot) {
14738
14757
  if (slot) {
14758
+ slot = this.resolveSlot(slot);
14739
14759
  if (this._slotTemplates.get(slot) === template) {
14740
14760
  this.removeSlotTemplate(slot);
14741
14761
  }
@@ -14753,6 +14773,24 @@ class TemplateSlotManager {
14753
14773
  * Private methods *
14754
14774
  * *
14755
14775
  **************************************************************************/
14776
+ initSlot(slotDefinition) {
14777
+ let slot;
14778
+ if (slotDefinition instanceof Array) {
14779
+ slot = slotDefinition.shift();
14780
+ this.registerSlotSynonyms(slot, slotDefinition);
14781
+ }
14782
+ else {
14783
+ slot = slotDefinition;
14784
+ }
14785
+ this._activeSlotTemplates.set(slot, new BehaviorSubject(null));
14786
+ }
14787
+ registerSlotSynonyms(primarySlot, synonyms) {
14788
+ synonyms.forEach(synonym => this._slotSynonyms.set(synonym, primarySlot));
14789
+ }
14790
+ resolveSlot(slot) {
14791
+ var _a;
14792
+ return (_a = this._slotSynonyms.get(slot)) !== null && _a !== void 0 ? _a : slot;
14793
+ }
14756
14794
  removeSlotTemplate(slot) {
14757
14795
  this.logger.trace('Deregistering template slot at ', slot);
14758
14796
  this._slotTemplates.delete(slot);
@@ -14787,10 +14825,12 @@ class ElderToolbarService {
14787
14825
  this.logger = LoggerFactory.getLogger(this.constructor.name);
14788
14826
  this.slotManager = new TemplateSlotManager([
14789
14827
  'left',
14828
+ 'left.begin',
14829
+ ['left.end', 'left.actions'],
14790
14830
  'center',
14791
14831
  'right',
14792
- 'left.actions',
14793
- 'right.actions',
14832
+ ['right.begin', 'right.actions'],
14833
+ 'right.end',
14794
14834
  ]);
14795
14835
  }
14796
14836
  /***************************************************************************
@@ -15113,10 +15153,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.4", ngImpor
15113
15153
 
15114
15154
  class ElderToolbarComponent {
15115
15155
  /***************************************************************************
15116
- * *
15117
- * Constructor *
15118
- * *
15119
- **************************************************************************/
15156
+ * *
15157
+ * Constructor *
15158
+ * *
15159
+ **************************************************************************/
15120
15160
  constructor(toolbarService) {
15121
15161
  this.toolbarService = toolbarService;
15122
15162
  /***************************************************************************
@@ -15125,24 +15165,28 @@ class ElderToolbarComponent {
15125
15165
  * *
15126
15166
  **************************************************************************/
15127
15167
  this.logger = LoggerFactory.getLogger(this.constructor.name);
15128
- this.leftColumnTemplate$ = toolbarService.activeColumnTemplate('left');
15129
- this.centerColumnTemplate$ = toolbarService.activeColumnTemplate('center');
15130
- this.rightColumnTemplate$ = toolbarService.activeColumnTemplate('right');
15131
- this.leftActionsColumnTemplate$ = toolbarService.activeColumnTemplate('left.actions');
15132
- this.rightActionsColumnTemplate$ = toolbarService.activeColumnTemplate('right.actions');
15133
15168
  }
15134
15169
  /***************************************************************************
15135
15170
  * *
15136
15171
  * Life Cycle *
15137
15172
  * *
15138
15173
  **************************************************************************/
15139
- ngOnInit() { }
15174
+ ngOnInit() {
15175
+ }
15176
+ /***************************************************************************
15177
+ * *
15178
+ * Public API *
15179
+ * *
15180
+ **************************************************************************/
15181
+ slotTemplate$(slot) {
15182
+ return this.toolbarService.activeColumnTemplate(slot);
15183
+ }
15140
15184
  }
15141
15185
  ElderToolbarComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.4", ngImport: i0, type: ElderToolbarComponent, deps: [{ token: ElderToolbarService }], target: i0.ɵɵFactoryTarget.Component });
15142
- ElderToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.4", type: ElderToolbarComponent, selector: "elder-toolbar, ebs-toolbar", inputs: { color: "color" }, ngImport: i0, template: "\n\n<div fxLayout=\"row\">\n\n <ng-content></ng-content>\n\n <mat-toolbar fxLayout=\"row\" fxLayoutAlign=\"start center\"\n [color]=\"color ? color : 'primary'\">\n\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\">\n <ng-container *ngTemplateOutlet=\"(leftColumnTemplate$ | async) || defaultLeftColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(leftActionsColumnTemplate$ | async) || defaultEmptyTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"center center\" fxFlex>\n <ng-container *ngTemplateOutlet=\"(centerColumnTemplate$ | async) || defaultCenterColumnTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"end center\">\n <ng-container *ngTemplateOutlet=\"(rightActionsColumnTemplate$ | async) || defaultEmptyTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(rightColumnTemplate$ | async) || defaultRightColumnTemplate\"></ng-container>\n </div>\n\n\n <!-- Default Templates -->\n\n <ng-template #defaultLeftColumnTemplate>\n\n <elder-toolbar-title></elder-toolbar-title>\n\n </ng-template>\n\n\n <ng-template #defaultCenterColumnTemplate></ng-template>\n\n <ng-template #defaultEmptyTemplate></ng-template>\n\n\n <ng-template #defaultRightColumnTemplate>\n\n <elder-language-switcher [slimMode]=\"true\"></elder-language-switcher>\n\n </ng-template>\n\n </mat-toolbar>\n\n</div>\n\n", styles: [""], components: [{ type: i2$3.MatToolbar, selector: "mat-toolbar", inputs: ["color"], exportAs: ["matToolbar"] }, { type: ElderToolbarTitleComponent, selector: "elder-toolbar-title, ebs-toolbar-title" }, { type: ElderLanguageSwitcherComponent, selector: "elder-language-switcher, ebs-language-switcher", inputs: ["slimMode"] }], directives: [{ type: i5.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { type: i5.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { type: i3$1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i5.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }], pipes: { "async": i3$1.AsyncPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
15186
+ ElderToolbarComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.4", type: ElderToolbarComponent, selector: "elder-toolbar, ebs-toolbar", inputs: { color: "color" }, ngImport: i0, template: "\n\n<div fxLayout=\"row\">\n\n <ng-content></ng-content>\n\n <mat-toolbar fxLayout=\"row\" fxLayoutAlign=\"start center\"\n [color]=\"color ? color : 'primary'\">\n\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\">\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left.begin') | async)\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left') | async) || defaultLeftColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left.end') | async) || defaultEmptyTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"center center\" fxFlex>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('center') | async) || defaultCenterColumnTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"end center\">\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right.begin') | async) || defaultEmptyTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right') | async) || defaultRightColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right.end') | async)\"></ng-container>\n </div>\n\n\n <!-- Default Templates -->\n\n <ng-template #defaultLeftColumnTemplate>\n\n <elder-toolbar-title></elder-toolbar-title>\n\n </ng-template>\n\n\n <ng-template #defaultCenterColumnTemplate></ng-template>\n\n <ng-template #defaultEmptyTemplate></ng-template>\n\n\n <ng-template #defaultRightColumnTemplate>\n\n <elder-language-switcher [slimMode]=\"true\"></elder-language-switcher>\n\n </ng-template>\n\n </mat-toolbar>\n\n</div>\n\n", styles: [""], components: [{ type: i2$3.MatToolbar, selector: "mat-toolbar", inputs: ["color"], exportAs: ["matToolbar"] }, { type: ElderToolbarTitleComponent, selector: "elder-toolbar-title, ebs-toolbar-title" }, { type: ElderLanguageSwitcherComponent, selector: "elder-language-switcher, ebs-language-switcher", inputs: ["slimMode"] }], directives: [{ type: i5.DefaultLayoutDirective, selector: " [fxLayout], [fxLayout.xs], [fxLayout.sm], [fxLayout.md], [fxLayout.lg], [fxLayout.xl], [fxLayout.lt-sm], [fxLayout.lt-md], [fxLayout.lt-lg], [fxLayout.lt-xl], [fxLayout.gt-xs], [fxLayout.gt-sm], [fxLayout.gt-md], [fxLayout.gt-lg]", inputs: ["fxLayout", "fxLayout.xs", "fxLayout.sm", "fxLayout.md", "fxLayout.lg", "fxLayout.xl", "fxLayout.lt-sm", "fxLayout.lt-md", "fxLayout.lt-lg", "fxLayout.lt-xl", "fxLayout.gt-xs", "fxLayout.gt-sm", "fxLayout.gt-md", "fxLayout.gt-lg"] }, { type: i5.DefaultLayoutAlignDirective, selector: " [fxLayoutAlign], [fxLayoutAlign.xs], [fxLayoutAlign.sm], [fxLayoutAlign.md], [fxLayoutAlign.lg], [fxLayoutAlign.xl], [fxLayoutAlign.lt-sm], [fxLayoutAlign.lt-md], [fxLayoutAlign.lt-lg], [fxLayoutAlign.lt-xl], [fxLayoutAlign.gt-xs], [fxLayoutAlign.gt-sm], [fxLayoutAlign.gt-md], [fxLayoutAlign.gt-lg]", inputs: ["fxLayoutAlign", "fxLayoutAlign.xs", "fxLayoutAlign.sm", "fxLayoutAlign.md", "fxLayoutAlign.lg", "fxLayoutAlign.xl", "fxLayoutAlign.lt-sm", "fxLayoutAlign.lt-md", "fxLayoutAlign.lt-lg", "fxLayoutAlign.lt-xl", "fxLayoutAlign.gt-xs", "fxLayoutAlign.gt-sm", "fxLayoutAlign.gt-md", "fxLayoutAlign.gt-lg"] }, { type: i3$1.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i5.DefaultFlexDirective, selector: " [fxFlex], [fxFlex.xs], [fxFlex.sm], [fxFlex.md], [fxFlex.lg], [fxFlex.xl], [fxFlex.lt-sm], [fxFlex.lt-md], [fxFlex.lt-lg], [fxFlex.lt-xl], [fxFlex.gt-xs], [fxFlex.gt-sm], [fxFlex.gt-md], [fxFlex.gt-lg]", inputs: ["fxFlex", "fxFlex.xs", "fxFlex.sm", "fxFlex.md", "fxFlex.lg", "fxFlex.xl", "fxFlex.lt-sm", "fxFlex.lt-md", "fxFlex.lt-lg", "fxFlex.lt-xl", "fxFlex.gt-xs", "fxFlex.gt-sm", "fxFlex.gt-md", "fxFlex.gt-lg"] }], pipes: { "async": i3$1.AsyncPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
15143
15187
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "13.3.4", ngImport: i0, type: ElderToolbarComponent, decorators: [{
15144
15188
  type: Component,
15145
- args: [{ selector: 'elder-toolbar, ebs-toolbar', changeDetection: ChangeDetectionStrategy.OnPush, template: "\n\n<div fxLayout=\"row\">\n\n <ng-content></ng-content>\n\n <mat-toolbar fxLayout=\"row\" fxLayoutAlign=\"start center\"\n [color]=\"color ? color : 'primary'\">\n\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\">\n <ng-container *ngTemplateOutlet=\"(leftColumnTemplate$ | async) || defaultLeftColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(leftActionsColumnTemplate$ | async) || defaultEmptyTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"center center\" fxFlex>\n <ng-container *ngTemplateOutlet=\"(centerColumnTemplate$ | async) || defaultCenterColumnTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"end center\">\n <ng-container *ngTemplateOutlet=\"(rightActionsColumnTemplate$ | async) || defaultEmptyTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(rightColumnTemplate$ | async) || defaultRightColumnTemplate\"></ng-container>\n </div>\n\n\n <!-- Default Templates -->\n\n <ng-template #defaultLeftColumnTemplate>\n\n <elder-toolbar-title></elder-toolbar-title>\n\n </ng-template>\n\n\n <ng-template #defaultCenterColumnTemplate></ng-template>\n\n <ng-template #defaultEmptyTemplate></ng-template>\n\n\n <ng-template #defaultRightColumnTemplate>\n\n <elder-language-switcher [slimMode]=\"true\"></elder-language-switcher>\n\n </ng-template>\n\n </mat-toolbar>\n\n</div>\n\n", styles: [""] }]
15189
+ args: [{ selector: 'elder-toolbar, ebs-toolbar', changeDetection: ChangeDetectionStrategy.OnPush, template: "\n\n<div fxLayout=\"row\">\n\n <ng-content></ng-content>\n\n <mat-toolbar fxLayout=\"row\" fxLayoutAlign=\"start center\"\n [color]=\"color ? color : 'primary'\">\n\n <div fxLayout=\"row\" fxLayoutAlign=\"start center\">\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left.begin') | async)\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left') | async) || defaultLeftColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('left.end') | async) || defaultEmptyTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"center center\" fxFlex>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('center') | async) || defaultCenterColumnTemplate\"></ng-container>\n </div>\n\n <div fxLayout=\"row\" fxLayoutAlign=\"end center\">\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right.begin') | async) || defaultEmptyTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right') | async) || defaultRightColumnTemplate\"></ng-container>\n <ng-container *ngTemplateOutlet=\"(slotTemplate$('right.end') | async)\"></ng-container>\n </div>\n\n\n <!-- Default Templates -->\n\n <ng-template #defaultLeftColumnTemplate>\n\n <elder-toolbar-title></elder-toolbar-title>\n\n </ng-template>\n\n\n <ng-template #defaultCenterColumnTemplate></ng-template>\n\n <ng-template #defaultEmptyTemplate></ng-template>\n\n\n <ng-template #defaultRightColumnTemplate>\n\n <elder-language-switcher [slimMode]=\"true\"></elder-language-switcher>\n\n </ng-template>\n\n </mat-toolbar>\n\n</div>\n\n", styles: [""] }]
15146
15190
  }], ctorParameters: function () { return [{ type: ElderToolbarService }]; }, propDecorators: { color: [{
15147
15191
  type: Input
15148
15192
  }] } });
@@ -20379,6 +20423,7 @@ class ElderSelectBase extends FormFieldBaseComponent {
20379
20423
  this.autoCleanUp();
20380
20424
  if (data instanceof Array) {
20381
20425
  this.suggestionsDc = DataContextBuilder.start()
20426
+ .localSort()
20382
20427
  .buildLocal(data); // Memory leak
20383
20428
  this.suggestionsDc.start();
20384
20429
  }
@@ -21146,6 +21191,7 @@ class ElderSelectComponent extends ElderSelectBase {
21146
21191
  ElderSelectFirstUtil.trySelectFirst(this);
21147
21192
  }
21148
21193
  updateValueByEntity(entity) {
21194
+ this._entity$.next(entity);
21149
21195
  this.updateValue(this.entityToValue(entity));
21150
21196
  }
21151
21197
  /***************************************************************************
@@ -21257,22 +21303,6 @@ class ElderSelectComponent extends ElderSelectBase {
21257
21303
  }
21258
21304
  super.writeToControl(value);
21259
21305
  }
21260
- /**
21261
- * @description
21262
- * Writes a new value to the element.
21263
- *
21264
- * This method is called by the forms API to write to the view when programmatic
21265
- * changes from model to view are requested.
21266
- *
21267
- * @param value The new value for the element
21268
- */
21269
- writeValue(value) {
21270
- // this.logger.debug('ElderSelectBase.writeValue(): Current value-id ' + this._valueId + ', new value: ', value);
21271
- super.writeValue(value);
21272
- }
21273
- isNullOrEmpty(value) {
21274
- return (value == null || value === '');
21275
- }
21276
21306
  }
21277
21307
  ElderSelectComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "13.3.4", ngImport: i0, type: ElderSelectComponent, deps: [{ token: i0.NgZone }], target: i0.ɵɵFactoryTarget.Component });
21278
21308
  ElderSelectComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "13.3.4", type: ElderSelectComponent, selector: "elder-select", inputs: { nullDisplay: "nullDisplay", autocomplete: "autocomplete", allowNull: "allowNull", entity: "entity", entityId: "entityId", hintProperty: "hintProperty", hintPropertyResolver: "hintPropertyResolver" }, outputs: { entityIdChange: "entityIdChange", entityIdUpdated: "entityIdUpdated", entityChange: "entityChange", entityUpdated: "entityUpdated" }, providers: [
@@ -21383,13 +21413,22 @@ class ElderMultiSelectBase extends ElderSelectBase {
21383
21413
  this.entities$ = new BehaviorSubject([]);
21384
21414
  this.entityIdsChange = this.valueChange.pipe(map(values => this.entityIdsFromValues(values)));
21385
21415
  this.entityIdsUpdated = this.valueUpdated.pipe(map(values => this.entityIdsFromValues(values)));
21386
- this.entitiesUpdated = this.entityIdsUpdated.pipe(switchMap(entityId => this.awaitEntitiesWithId(entityId, this.timeoutAfterMs)), catchError(err => {
21416
+ this.entitiesUpdated = this.entityIdsUpdated.pipe(tap(entityIds => this.logger.debug('Awaiting entities with ids', entityIds)), switchMap(entityIds => this.awaitEntitiesWithId(entityIds, this.timeoutAfterMs)), catchError(err => {
21387
21417
  this.logger.warn(`awaitEntitiesWithId -> timed out after: ${this.timeoutAfterMs}`, err);
21388
21418
  return EMPTY;
21389
21419
  }));
21390
21420
  }
21391
21421
  awaitEntitiesWithId(entityIds, timeoutMs) {
21392
- return this.entities$.pipe(filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)), take(1), timeout(timeoutMs));
21422
+ return this.entities$.pipe(tap(entities => this.logger.debug('Got entities for ids' + entityIds, entities)), filter(entities => this.equalIds(this.getEntityIds(entities), entityIds)), take(1), timeout(timeoutMs));
21423
+ }
21424
+ /***************************************************************************
21425
+ * *
21426
+ * Properties *
21427
+ * *
21428
+ **************************************************************************/
21429
+ updateValueByEntities(entities) {
21430
+ this.entities$.next(entities);
21431
+ this.updateValue(this.entitiesToValues(entities));
21393
21432
  }
21394
21433
  /***************************************************************************
21395
21434
  * *
@@ -21681,7 +21720,7 @@ class ElderMultiSelectChipsComponent extends ElderMultiSelectBase {
21681
21720
  }
21682
21721
  addLabels(labels) {
21683
21722
  const current = this.entities ? this.entities : [];
21684
- this.entities = [...current, ...labels];
21723
+ this.updateValueByEntities([...current, ...labels]);
21685
21724
  }
21686
21725
  addLabel(label) {
21687
21726
  this.addLabels([label]);
@@ -21689,7 +21728,7 @@ class ElderMultiSelectChipsComponent extends ElderMultiSelectBase {
21689
21728
  removeLabel(toRemove) {
21690
21729
  const remaining = this.entities
21691
21730
  .filter(l => !this.isEqual(l, toRemove));
21692
- this.entities = remaining;
21731
+ this.updateValueByEntities(remaining);
21693
21732
  }
21694
21733
  /***************************************************************************
21695
21734
  * *