@elderbyte/ngx-starter 15.14.0 → 15.15.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -114,7 +114,7 @@ export class ElderTableActivationDirective {
114
114
  }
115
115
  hasItemFocus(item) {
116
116
  const row = this.getRowForItem(item);
117
- return row.hasFocus;
117
+ return row?.hasFocus;
118
118
  }
119
119
  nextCloseTo(active) {
120
120
  const next = this.nextRowItem();
@@ -272,4 +272,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
272
272
  }], activeItem: [{
273
273
  type: Input
274
274
  }] } });
275
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-table-activation.directive.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/data-view/table/activation/elder-table-activation.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Directive,\n  HostListener,\n  Input,\n  OnDestroy, Output,\n} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderTableRowDirective} from '../elder-table-row.directive';\nimport {switchMap, takeUntil, tap} from 'rxjs/operators';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {ElderTableComponent} from '../elder-table/elder-table.component';\nimport {ElderTableActivationOptions} from './elder-table-activation-options';\nimport {ElderItemActivator} from './elder-item-activator';\n\n@Directive({\n  selector: 'elder-table[elderTableActivation]',\n  exportAs: 'elderTableActivation'\n})\nexport class ElderTableActivationDirective<T> implements ElderItemActivator<T>, AfterViewInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<void>();\n\n  public readonly activeItem$ = new BehaviorSubject<T>(null);\n\n  @Input('elderTableActivation')\n  public activationOptions: ElderTableActivationOptions = {};\n\n  private _data: T[] = [];\n\n  public readonly rows$ = new BehaviorSubject<ElderTableRowDirective<T>[]>([]);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly elderTable: ElderTableComponent\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterViewInit(): void {\n\n    const rows$ = this.elderTable.rows$;\n    const data$ = this.elderTable.dataContext$.pipe(\n      switchMap(dc => dc.data),\n      tap(data => this._data = data)\n    );\n\n    combineLatest([rows$, data$]).pipe(\n      takeUntil(this.destroy$)\n    ).subscribe(\n      ([rows, data]) => {\n        this.rows$.next(\n          this.rowDataUpdated(rows, data)\n        );\n        this.handleAutoActivations();\n      }\n    );\n\n    combineLatest([this.rows$, this.activeItem$]).pipe(\n      takeUntil(this.destroy$)\n    ).subscribe(\n      ([rows, activeItem]) => {\n        this.updateRowsActivation(rows, activeItem);\n      }\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    // this.elderTable.rows$.getValue().forEach(r => r.bindTableActivation(null));\n  }\n\n  /***************************************************************************\n   *                                                                         *.\n   * Host Listeners                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown', ['$event'])\n  public onKeydown(event: KeyboardEvent) {\n    let newActiveItem: T;\n    if (event.key === 'ArrowDown') {\n      newActiveItem = this.nextRowItem();\n    } else if (event.key === 'ArrowUp') {\n      newActiveItem = this.previousRowItem();\n    }\n\n    if (newActiveItem) {\n      this.focusAndActivate(newActiveItem);\n    }\n\n    const active = this.activeItem;\n    if (active) {\n      if (event.key === 'Enter') {\n        this.elderTable.onItemClick(active);\n        event.stopPropagation();\n      } else if (event.key === 'Spacebar' || event.key === ' ') {\n        this.elderTable.selectionModel.toggle(active);\n        event.stopPropagation();\n      }\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Output()\n  public get activeItemChange(): Observable<T> {\n    return this.activeItem$;\n  }\n\n  public get activeItem(): T {\n    return this.activeItem$.getValue();\n  }\n\n  @Input()\n  public set activeItem(item: T) {\n    if (item !== this.activeItem) {\n      this.activeItem$.next(item);\n      this.logger.debug('Set active item to ', item);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public activate(item: T): void {\n    this.activeItem = item;\n  }\n\n  public focusAndActivate(item: T): void {\n    const row = this.getRowForItem(item);\n    if (row) {\n      row.foucs(); // Focus will auto activate the row\n    }\n  }\n\n  public isItemActive(item: T): boolean {\n    if (this.activeItem) {\n      return this.itemEquals(this.activeItem, item);\n    } else {\n      return false;\n    }\n  }\n\n  public hasItemFocus(item: T): boolean {\n    const row = this.getRowForItem(item);\n    return row.hasFocus;\n  }\n\n  public nextCloseTo(active: T): T {\n    const next = this.nextRowItem();\n    if (next !== active) {\n      return next;\n    }\n    return this.previousRowItem();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateRowsActivation(rows: ElderTableRowDirective<T>[], activeItem: T): void {\n    const activeId = this.getId(activeItem);\n    rows.forEach(row => {\n      this.updateRowActivation(row, activeId);\n    });\n  }\n\n  private updateRowActivation(row: ElderTableRowDirective<T>, activeId: any): void {\n    if (row.model) {\n      if (this.itemEqualsId(row.model, activeId)) {\n        if (!row.activated) {\n          row.activated = true;\n          this.logger.debug('Activated row: ', row);\n          if (!row.hasFocus) {\n            row.bringToView();\n          }\n        }\n      } else {\n        row.activated = false;\n      }\n    } else {\n      row.activated = false;\n    }\n  }\n\n  private rowDataUpdated(\n    rows: ElderTableRowDirective<T>[],\n    data: T[]\n  ): ElderTableRowDirective<T>[] {\n    // this.logger.debug('Found rows:', rows);\n    rows.forEach(row => {\n      row.bindTableActivation(this);\n    });\n    return rows;\n  }\n\n\n  private getRowForItem(item: T): ElderTableRowDirective<T> | null {\n    const rows = this.rows$.getValue();\n    return rows.find(r => r.model === item);\n  }\n\n  private handleAutoActivations(): void {\n    if (!this.hasActiveItem || this.activationOptions.mode === 'always') {\n      switch (this.activationOptions.row) {\n        case 'none':\n          break;\n        case 'first':\n          this.activateFirst();\n          break;\n        case 'last':\n          this.activateLast();\n          break;\n      }\n    }\n  }\n\n  private get hasActiveItem(): boolean {\n    return !!this.activeItem;\n  }\n\n  private activateFirst(): void {\n    const item = this.getItemAtIndex(0);\n    if (item) {\n      this.activate(item);\n    }\n  }\n\n  private activateLast(): void {\n    if (this._data) {\n      const item = this.getItemAtIndex(this._data.length - 1);\n      if (item) {\n        this.activate(item);\n      }\n    }\n  }\n\n  private previousRowItem(): T {\n    const rows = this._data;\n    let currentIndex = this.getCurrentIndex();\n    if (currentIndex > 0) {\n      --currentIndex;\n    } else {\n      if (this.activationOptions.loop) {\n        currentIndex = rows.length - 1;\n      }\n    }\n    return this.getItemAtIndex(currentIndex);\n  }\n\n  private nextRowItem(): T {\n    const rows = this._data;\n    let currentIndex = this.getCurrentIndex();\n\n    if (currentIndex < (rows.length - 1)) {\n      ++currentIndex;\n    } else {\n      if (this.activationOptions.loop) {\n        currentIndex = 0;\n      }\n    }\n    return this.getItemAtIndex(currentIndex);\n  }\n\n  private getCurrentIndex(): number {\n    const activeItem = this.activeItem;\n    if (activeItem) {\n      return this.findItemIndex(activeItem);\n    } else {\n      return -1;\n    }\n  }\n\n  private getItemAtIndex(index: number): T | undefined {\n    return this._data[index];\n  }\n\n  private findItemIndex(item: T): number {\n    const needleId = this.getId(item);\n    return this._data.findIndex(itm => this.itemEqualsId(itm, needleId));\n  }\n\n  private itemEquals(a: T, b: T): boolean {\n    return this.getId(a) === this.getId(b);\n  }\n\n  private itemEqualsId(a: T, needleId: any): boolean {\n    return this.getId(a) === needleId;\n  }\n\n  private getId(item: T): any {\n    if (item) {\n      const dc = this.elderTable.dataContext$.getValue();\n      return dc?.dataSource?.getId(item);\n    } else {\n      return item;\n    }\n  }\n\n}\n"]}
275
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"elder-table-activation.directive.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/data-view/table/activation/elder-table-activation.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  Directive,\n  HostListener,\n  Input,\n  OnDestroy, Output,\n} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderTableRowDirective} from '../elder-table-row.directive';\nimport {switchMap, takeUntil, tap} from 'rxjs/operators';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {ElderTableComponent} from '../elder-table/elder-table.component';\nimport {ElderTableActivationOptions} from './elder-table-activation-options';\nimport {ElderItemActivator} from './elder-item-activator';\n\n@Directive({\n  selector: 'elder-table[elderTableActivation]',\n  exportAs: 'elderTableActivation'\n})\nexport class ElderTableActivationDirective<T> implements ElderItemActivator<T>, AfterViewInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly logger = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly destroy$ = new Subject<void>();\n\n  public readonly activeItem$ = new BehaviorSubject<T>(null);\n\n  @Input('elderTableActivation')\n  public activationOptions: ElderTableActivationOptions = {};\n\n  private _data: T[] = [];\n\n  public readonly rows$ = new BehaviorSubject<ElderTableRowDirective<T>[]>([]);\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly elderTable: ElderTableComponent\n  ) {\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngAfterViewInit(): void {\n\n    const rows$ = this.elderTable.rows$;\n    const data$ = this.elderTable.dataContext$.pipe(\n      switchMap(dc => dc.data),\n      tap(data => this._data = data)\n    );\n\n    combineLatest([rows$, data$]).pipe(\n      takeUntil(this.destroy$)\n    ).subscribe(\n      ([rows, data]) => {\n        this.rows$.next(\n          this.rowDataUpdated(rows, data)\n        );\n        this.handleAutoActivations();\n      }\n    );\n\n    combineLatest([this.rows$, this.activeItem$]).pipe(\n      takeUntil(this.destroy$)\n    ).subscribe(\n      ([rows, activeItem]) => {\n        this.updateRowsActivation(rows, activeItem);\n      }\n    );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    // this.elderTable.rows$.getValue().forEach(r => r.bindTableActivation(null));\n  }\n\n  /***************************************************************************\n   *                                                                         *.\n   * Host Listeners                                                          *\n   *                                                                         *\n   **************************************************************************/\n\n  @HostListener('keydown', ['$event'])\n  public onKeydown(event: KeyboardEvent) {\n    let newActiveItem: T;\n    if (event.key === 'ArrowDown') {\n      newActiveItem = this.nextRowItem();\n    } else if (event.key === 'ArrowUp') {\n      newActiveItem = this.previousRowItem();\n    }\n\n    if (newActiveItem) {\n      this.focusAndActivate(newActiveItem);\n    }\n\n    const active = this.activeItem;\n    if (active) {\n      if (event.key === 'Enter') {\n        this.elderTable.onItemClick(active);\n        event.stopPropagation();\n      } else if (event.key === 'Spacebar' || event.key === ' ') {\n        this.elderTable.selectionModel.toggle(active);\n        event.stopPropagation();\n      }\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Output()\n  public get activeItemChange(): Observable<T> {\n    return this.activeItem$;\n  }\n\n  public get activeItem(): T {\n    return this.activeItem$.getValue();\n  }\n\n  @Input()\n  public set activeItem(item: T) {\n    if (item !== this.activeItem) {\n      this.activeItem$.next(item);\n      this.logger.debug('Set active item to ', item);\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public activate(item: T): void {\n    this.activeItem = item;\n  }\n\n  public focusAndActivate(item: T): void {\n    const row = this.getRowForItem(item);\n    if (row) {\n      row.foucs(); // Focus will auto activate the row\n    }\n  }\n\n  public isItemActive(item: T): boolean {\n    if (this.activeItem) {\n      return this.itemEquals(this.activeItem, item);\n    } else {\n      return false;\n    }\n  }\n\n  public hasItemFocus(item: T): boolean {\n    const row = this.getRowForItem(item);\n    return row?.hasFocus;\n  }\n\n  public nextCloseTo(active: T): T {\n    const next = this.nextRowItem();\n    if (next !== active) {\n      return next;\n    }\n    return this.previousRowItem();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private updateRowsActivation(rows: ElderTableRowDirective<T>[], activeItem: T): void {\n    const activeId = this.getId(activeItem);\n    rows.forEach(row => {\n      this.updateRowActivation(row, activeId);\n    });\n  }\n\n  private updateRowActivation(row: ElderTableRowDirective<T>, activeId: any): void {\n    if (row.model) {\n      if (this.itemEqualsId(row.model, activeId)) {\n        if (!row.activated) {\n          row.activated = true;\n          this.logger.debug('Activated row: ', row);\n          if (!row.hasFocus) {\n            row.bringToView();\n          }\n        }\n      } else {\n        row.activated = false;\n      }\n    } else {\n      row.activated = false;\n    }\n  }\n\n  private rowDataUpdated(\n    rows: ElderTableRowDirective<T>[],\n    data: T[]\n  ): ElderTableRowDirective<T>[] {\n    // this.logger.debug('Found rows:', rows);\n    rows.forEach(row => {\n      row.bindTableActivation(this);\n    });\n    return rows;\n  }\n\n\n  private getRowForItem(item: T): ElderTableRowDirective<T> | null {\n    const rows = this.rows$.getValue();\n    return rows.find(r => r.model === item);\n  }\n\n  private handleAutoActivations(): void {\n    if (!this.hasActiveItem || this.activationOptions.mode === 'always') {\n      switch (this.activationOptions.row) {\n        case 'none':\n          break;\n        case 'first':\n          this.activateFirst();\n          break;\n        case 'last':\n          this.activateLast();\n          break;\n      }\n    }\n  }\n\n  private get hasActiveItem(): boolean {\n    return !!this.activeItem;\n  }\n\n  private activateFirst(): void {\n    const item = this.getItemAtIndex(0);\n    if (item) {\n      this.activate(item);\n    }\n  }\n\n  private activateLast(): void {\n    if (this._data) {\n      const item = this.getItemAtIndex(this._data.length - 1);\n      if (item) {\n        this.activate(item);\n      }\n    }\n  }\n\n  private previousRowItem(): T {\n    const rows = this._data;\n    let currentIndex = this.getCurrentIndex();\n    if (currentIndex > 0) {\n      --currentIndex;\n    } else {\n      if (this.activationOptions.loop) {\n        currentIndex = rows.length - 1;\n      }\n    }\n    return this.getItemAtIndex(currentIndex);\n  }\n\n  private nextRowItem(): T {\n    const rows = this._data;\n    let currentIndex = this.getCurrentIndex();\n\n    if (currentIndex < (rows.length - 1)) {\n      ++currentIndex;\n    } else {\n      if (this.activationOptions.loop) {\n        currentIndex = 0;\n      }\n    }\n    return this.getItemAtIndex(currentIndex);\n  }\n\n  private getCurrentIndex(): number {\n    const activeItem = this.activeItem;\n    if (activeItem) {\n      return this.findItemIndex(activeItem);\n    } else {\n      return -1;\n    }\n  }\n\n  private getItemAtIndex(index: number): T | undefined {\n    return this._data[index];\n  }\n\n  private findItemIndex(item: T): number {\n    const needleId = this.getId(item);\n    return this._data.findIndex(itm => this.itemEqualsId(itm, needleId));\n  }\n\n  private itemEquals(a: T, b: T): boolean {\n    return this.getId(a) === this.getId(b);\n  }\n\n  private itemEqualsId(a: T, needleId: any): boolean {\n    return this.getId(a) === needleId;\n  }\n\n  private getId(item: T): any {\n    if (item) {\n      const dc = this.elderTable.dataContext$.getValue();\n      return dc?.dataSource?.getId(item);\n    } else {\n      return item;\n    }\n  }\n\n}\n"]}
@@ -97,7 +97,10 @@ export class DataViewIframeComponent {
97
97
  combineLatest([
98
98
  this._dataViewOptions$,
99
99
  iframeHost.iframeState$
100
- ]).pipe(takeUntil(this._destroy$), filter(([options, state]) => !!options && state.hasLoaded)).subscribe(([options, state]) => this.sendMessageToChild(this.buildDataViewInitMessage(options, this.initialSelection)));
100
+ ]).pipe(takeUntil(this._destroy$),
101
+ // Send message on load and viewReady state for backwards compatibility between host and child.
102
+ // Actually using only viewReady should be enough
103
+ filter(([options, state]) => !!options && (state.hasLoaded || state.isViewReady))).subscribe(([options, state]) => this.sendMessageToChild(this.buildDataViewInitMessage(options, this.initialSelection)));
101
104
  }
102
105
  buildDataViewInitMessage(options, initialSelection) {
103
106
  return DataViewMessage.initialSelection(new DataViewSelectionInit(options, initialSelection, this.idField));
@@ -124,4 +127,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
124
127
  }], dataViewOptions: [{
125
128
  type: Input
126
129
  }] } });
127
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-view-iframe.component.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.ts","../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional, SkipSelf,\n  ViewChild\n} from '@angular/core';\nimport {IframeHostComponent} from '../../iframe-host/iframe-host.component';\nimport {SelectionModel} from '../../../../common/selection/selection-model';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {filter, takeUntil} from 'rxjs/operators';\nimport {DataViewMessage, DataViewSelection, DataViewSelectionInit, TypedEventMessage} from '../../typed-event-message';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderDataViewOptionsProvider} from '../../../data-view/base/elder-data-view-options-provider';\nimport {ElderDataViewOptions} from '../../../data-view/base/elder-data-view-options';\n\n/**\n * Container for displaying a remote data view in an iframe.\n * Provides capabilities to communicate between remote data view (child)\n * and iframe parent.\n */\n@Component({\n  selector: 'elder-data-view-iframe',\n  templateUrl: './data-view-iframe.component.html',\n  styleUrls: ['./data-view-iframe.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DataViewIframeComponent implements OnInit, OnDestroy, AfterViewInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _dataViewOptions$ = new BehaviorSubject<ElderDataViewOptions>(ElderDataViewOptions.defaultOptions());\n\n  private readonly _destroy$ = new Subject<void>();\n\n  @Input()\n  public srcUrl: string;\n\n  @Input()\n  public idField: string;\n\n  @Input()\n  public initialSelection: any[];\n\n  @ViewChild(IframeHostComponent)\n  public iframeHost: IframeHostComponent;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    @Optional() private dataViewOptionsProvider: ElderDataViewOptionsProvider,\n    @Optional() private selectionModel: SelectionModel<any>\n  ) {\n    if (dataViewOptionsProvider) {\n      dataViewOptionsProvider.options$.pipe(\n        takeUntil(this._destroy$)\n      ).subscribe(options => this.dataViewOptions = options)\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n  }\n\n  public ngAfterViewInit(): void {\n    // iframe is only available after view init\n    this.setupDataViewOptionsBridge(this.iframeHost);\n  }\n\n  public ngOnDestroy() {\n    this._destroy$.next();\n    this._destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get dataViewOptions(): ElderDataViewOptions {\n    return this._dataViewOptions$.getValue();\n  }\n\n  @Input()\n  public set dataViewOptions(options: ElderDataViewOptions) {\n    const newOptions = this.dataViewOptions.withOptions(options);\n    this._dataViewOptions$.next(newOptions);\n  }\n\n  public get dataViewOptions$(): Observable<ElderDataViewOptions> {\n    return this._dataViewOptions$.asObservable();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onMessageFromChild(message: TypedEventMessage<any,any>): void {\n    if (DataViewMessage.isOfType(message)){\n      switch (message.type) {\n        case 'SELECTION_SELECTED':\n          this.handleSelectionMessage(message.data);\n          break;\n      }\n    }\n\n  }\n\n  public sendMessageToChild(message: DataViewMessage<any>) {\n    this.log.debug(\"Sending message to iframe child:\", message);\n    this.iframeHost.sendMessageToChild(message)\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleSelectionMessage(dataViewSelection: DataViewSelection<any>): void {\n    if (this.selectionModel) {\n      this.selectionModel.replaceSelection(dataViewSelection.selection);\n    }\n  }\n\n  private setupDataViewOptionsBridge(\n    iframeHost: IframeHostComponent\n  ) {\n    combineLatest([\n      this._dataViewOptions$,\n      iframeHost.iframeState$\n    ]).pipe(\n      takeUntil(this._destroy$),\n      filter(([options, state]) => !!options && state.hasLoaded)\n    ).subscribe(\n      ([options, state]) => this.sendMessageToChild(\n        this.buildDataViewInitMessage(options, this.initialSelection)\n      )\n    )\n  }\n\n  private buildDataViewInitMessage(\n    options: ElderDataViewOptions,\n    initialSelection: any[]\n  ): DataViewMessage<DataViewSelectionInit<any>> {\n    return DataViewMessage.initialSelection(\n      new DataViewSelectionInit(options, initialSelection, this.idField)\n    );\n  }\n\n\n}\n","<div class=\"layout-col full\">\n  <elder-i-frame-host class=\"flex\"\n                      [iframeParams]=\"{srcUrl: srcUrl, hideActionBar: true}\"\n                      (message)=\"onMessageFromChild($event)\">\n  </elder-i-frame-host>\n</div>\n"]}
130
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-view-iframe.component.js","sourceRoot":"","sources":["../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.ts","../../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe/data-view-iframe.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnDestroy,\n  OnInit,\n  Optional, SkipSelf,\n  ViewChild\n} from '@angular/core';\nimport {IframeHostComponent} from '../../iframe-host/iframe-host.component';\nimport {SelectionModel} from '../../../../common/selection/selection-model';\nimport {BehaviorSubject, combineLatest, Observable, Subject} from 'rxjs';\nimport {filter, takeUntil} from 'rxjs/operators';\nimport {DataViewMessage, DataViewSelection, DataViewSelectionInit, TypedEventMessage} from '../../typed-event-message';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderDataViewOptionsProvider} from '../../../data-view/base/elder-data-view-options-provider';\nimport {ElderDataViewOptions} from '../../../data-view/base/elder-data-view-options';\n\n/**\n * Container for displaying a remote data view in an iframe.\n * Provides capabilities to communicate between remote data view (child)\n * and iframe parent.\n */\n@Component({\n  selector: 'elder-data-view-iframe',\n  templateUrl: './data-view-iframe.component.html',\n  styleUrls: ['./data-view-iframe.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class DataViewIframeComponent implements OnInit, OnDestroy, AfterViewInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _dataViewOptions$ = new BehaviorSubject<ElderDataViewOptions>(ElderDataViewOptions.defaultOptions());\n\n  private readonly _destroy$ = new Subject<void>();\n\n  @Input()\n  public srcUrl: string;\n\n  @Input()\n  public idField: string;\n\n  @Input()\n  public initialSelection: any[];\n\n  @ViewChild(IframeHostComponent)\n  public iframeHost: IframeHostComponent;\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    @Optional() private dataViewOptionsProvider: ElderDataViewOptionsProvider,\n    @Optional() private selectionModel: SelectionModel<any>\n  ) {\n    if (dataViewOptionsProvider) {\n      dataViewOptionsProvider.options$.pipe(\n        takeUntil(this._destroy$)\n      ).subscribe(options => this.dataViewOptions = options)\n    }\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n  }\n\n  public ngAfterViewInit(): void {\n    // iframe is only available after view init\n    this.setupDataViewOptionsBridge(this.iframeHost);\n  }\n\n  public ngOnDestroy() {\n    this._destroy$.next();\n    this._destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public get dataViewOptions(): ElderDataViewOptions {\n    return this._dataViewOptions$.getValue();\n  }\n\n  @Input()\n  public set dataViewOptions(options: ElderDataViewOptions) {\n    const newOptions = this.dataViewOptions.withOptions(options);\n    this._dataViewOptions$.next(newOptions);\n  }\n\n  public get dataViewOptions$(): Observable<ElderDataViewOptions> {\n    return this._dataViewOptions$.asObservable();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onMessageFromChild(message: TypedEventMessage<any,any>): void {\n    if (DataViewMessage.isOfType(message)){\n      switch (message.type) {\n        case 'SELECTION_SELECTED':\n          this.handleSelectionMessage(message.data);\n          break;\n      }\n    }\n\n  }\n\n  public sendMessageToChild(message: DataViewMessage<any>) {\n    this.log.debug(\"Sending message to iframe child:\", message);\n    this.iframeHost.sendMessageToChild(message)\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleSelectionMessage(dataViewSelection: DataViewSelection<any>): void {\n    if (this.selectionModel) {\n      this.selectionModel.replaceSelection(dataViewSelection.selection);\n    }\n  }\n\n  private setupDataViewOptionsBridge(\n    iframeHost: IframeHostComponent\n  ) {\n    combineLatest([\n      this._dataViewOptions$,\n      iframeHost.iframeState$\n    ]).pipe(\n      takeUntil(this._destroy$),\n      // Send message on load and viewReady state for backwards compatibility between host and child.\n      // Actually using only viewReady should be enough\n      filter(([options, state]) => !!options && (state.hasLoaded || state.isViewReady))\n    ).subscribe(\n      ([options, state]) => this.sendMessageToChild(\n        this.buildDataViewInitMessage(options, this.initialSelection)\n      )\n    )\n  }\n\n  private buildDataViewInitMessage(\n    options: ElderDataViewOptions,\n    initialSelection: any[]\n  ): DataViewMessage<DataViewSelectionInit<any>> {\n    return DataViewMessage.initialSelection(\n      new DataViewSelectionInit(options, initialSelection, this.idField)\n    );\n  }\n\n\n}\n","<div class=\"layout-col full\">\n  <elder-i-frame-host class=\"flex\"\n                      [iframeParams]=\"{srcUrl: srcUrl, hideActionBar: true}\"\n                      (message)=\"onMessageFromChild($event)\">\n  </elder-i-frame-host>\n</div>\n"]}
@@ -53,8 +53,8 @@ export class DataViewIframeAdapterDirective {
53
53
  * @param event message from parent window
54
54
  */
55
55
  onWindowsMessage(event) {
56
- this.log.debug('Received message event from parent:', event);
57
- if (event) {
56
+ this.log.trace('Received message event from parent:', event);
57
+ if (event && event.data) {
58
58
  if (DataViewMessage.isOfType(event.data)) {
59
59
  this.log.debug('Received data view message from parent:', event);
60
60
  this.handleMessageFromParent(event.data);
@@ -68,6 +68,8 @@ export class DataViewIframeAdapterDirective {
68
68
  * *
69
69
  **************************************************************************/
70
70
  ngOnInit() {
71
+ // indicating iframe host that we are ready (e.g. ready to receive messages)
72
+ this.iframeService.postViewReadyEventMessage();
71
73
  }
72
74
  ngOnDestroy() {
73
75
  this._destroy$.next();
@@ -133,4 +135,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
133
135
  type: HostListener,
134
136
  args: ['window:message', ['$event']]
135
137
  }] } });
136
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-view-iframe-adapter.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe-adapter.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  HostListener,\n  OnDestroy,\n  OnInit,\n} from '@angular/core';\nimport {SelectionModel} from '../../../common/selection/selection-model';\nimport {Subject} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\nimport {DataViewMessage, DataViewSelection, DataViewSelectionInit} from '../typed-event-message';\nimport {IframeService} from '../iframe.service';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderDataViewOptionsProvider} from '../../data-view/base/elder-data-view-options-provider';\n\nexport function createSelectionModel(): SelectionModel<any> {\n  return new SelectionModel<any>();\n}\n\nexport function createDataOptionsProvider(): ElderDataViewOptionsProvider {\n  return new ElderDataViewOptionsProvider();\n}\n\n/**\n * An adapter bridging communication between\n * a data view displayed in an iframe and its parent.\n *\n * For example propagates selection model changes\n * to the iframe parent.\n */\n@Directive({\n  selector: '[elderDataViewIframeAdapter]',\n  providers: [\n    {\n      provide: SelectionModel,\n      useFactory: createSelectionModel\n    },\n    {\n      provide: ElderDataViewOptionsProvider,\n      useFactory: createDataOptionsProvider\n    }\n  ]\n})\nexport class DataViewIframeAdapterDirective implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _destroy$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly iframeService: IframeService,\n    private readonly selectionModel: SelectionModel<any>,\n    private readonly dataViewOptionsProvider: ElderDataViewOptionsProvider\n  ) {\n    selectionModel.selection.pipe(\n      takeUntil(this._destroy$)\n    ).subscribe(\n      selection => this.propagateSelectionToParent(selection)\n    )\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Listeners                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Listens for messages from parent window.\n   * Makes it possible to react upon messages from iframe parent.\n   *\n   * @param event message from parent window\n   */\n  @HostListener('window:message', ['$event'])\n  public onWindowsMessage(event: MessageEvent): boolean {\n    this.log.debug('Received message event from parent:', event);\n    if (event) {\n      if (DataViewMessage.isOfType(event.data)) {\n        this.log.debug('Received data view message from parent:', event);\n        this.handleMessageFromParent(event.data);\n      }\n    }\n    return true;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n  }\n\n  public ngOnDestroy(): void {\n    this._destroy$.next();\n    this._destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleMessageFromParent(message: DataViewMessage<any, any>): void {\n    switch (message.type) {\n      case 'SELECTION_INIT':\n        this.initializeDataView(message.data)\n        break;\n    }\n  }\n\n  private initializeDataView(init: DataViewSelectionInit<any>): void {\n\n    // updating options of child data view given by iframe parent\n    this.dataViewOptionsProvider.updateOptions(\n      options => options.withOptions(init.options)\n    )\n\n    this.selectionModel.keyGetterFn = (entity) => entity[init.idField];\n    this.selectionModel.isMultipleSelection = init.options.selectionMultiEnabled;\n    if (init.initialSelection) {\n      this.selectionModel.replaceSelection(init.initialSelection);\n    }\n  }\n\n  private propagateSelectionToParent(selection: any[]): void {\n    this.log.debug('Propagating selection to iframe parent:', selection);\n\n    const message = this.buildSelectionMessage(selection);\n    this.iframeService.postDataMessage(message);\n  }\n\n  private buildSelectionMessage(selection: any[]): DataViewMessage<any> {\n    return DataViewMessage.selection(new DataViewSelection(selection));\n  }\n\n}\n"]}
138
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"data-view-iframe-adapter.directive.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/data-view/data-view-iframe-adapter.directive.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Directive,\n  HostListener,\n  OnDestroy,\n  OnInit,\n} from '@angular/core';\nimport {SelectionModel} from '../../../common/selection/selection-model';\nimport {Subject} from 'rxjs';\nimport {takeUntil} from 'rxjs/operators';\nimport {DataViewMessage, DataViewSelection, DataViewSelectionInit} from '../typed-event-message';\nimport {IframeService} from '../iframe.service';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {ElderDataViewOptionsProvider} from '../../data-view/base/elder-data-view-options-provider';\n\nexport function createSelectionModel(): SelectionModel<any> {\n  return new SelectionModel<any>();\n}\n\nexport function createDataOptionsProvider(): ElderDataViewOptionsProvider {\n  return new ElderDataViewOptionsProvider();\n}\n\n/**\n * An adapter bridging communication between\n * a data view displayed in an iframe and its parent.\n *\n * For example propagates selection model changes\n * to the iframe parent.\n */\n@Directive({\n  selector: '[elderDataViewIframeAdapter]',\n  providers: [\n    {\n      provide: SelectionModel,\n      useFactory: createSelectionModel\n    },\n    {\n      provide: ElderDataViewOptionsProvider,\n      useFactory: createDataOptionsProvider\n    }\n  ]\n})\nexport class DataViewIframeAdapterDirective implements OnInit, OnDestroy {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  private readonly _destroy$ = new Subject<void>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor(\n    private readonly iframeService: IframeService,\n    private readonly selectionModel: SelectionModel<any>,\n    private readonly dataViewOptionsProvider: ElderDataViewOptionsProvider\n  ) {\n    selectionModel.selection.pipe(\n      takeUntil(this._destroy$)\n    ).subscribe(\n      selection => this.propagateSelectionToParent(selection)\n    )\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Listeners                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Listens for messages from parent window.\n   * Makes it possible to react upon messages from iframe parent.\n   *\n   * @param event message from parent window\n   */\n  @HostListener('window:message', ['$event'])\n  public onWindowsMessage(event: MessageEvent): boolean {\n    this.log.trace('Received message event from parent:', event);\n    if (event && event.data) {\n      if (DataViewMessage.isOfType(event.data)) {\n        this.log.debug('Received data view message from parent:', event);\n        this.handleMessageFromParent(event.data);\n      }\n    }\n    return true;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Life Cycle                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void {\n    // indicating iframe host that we are ready (e.g. ready to receive messages)\n    this.iframeService.postViewReadyEventMessage();\n  }\n\n  public ngOnDestroy(): void {\n    this._destroy$.next();\n    this._destroy$.complete();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private Methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleMessageFromParent(message: DataViewMessage<any, any>): void {\n    switch (message.type) {\n      case 'SELECTION_INIT':\n        this.initializeDataView(message.data)\n        break;\n    }\n  }\n\n  private initializeDataView(init: DataViewSelectionInit<any>): void {\n\n    // updating options of child data view given by iframe parent\n    this.dataViewOptionsProvider.updateOptions(\n      options => options.withOptions(init.options)\n    )\n\n    this.selectionModel.keyGetterFn = (entity) => entity[init.idField];\n    this.selectionModel.isMultipleSelection = init.options.selectionMultiEnabled;\n    if (init.initialSelection) {\n      this.selectionModel.replaceSelection(init.initialSelection);\n    }\n  }\n\n  private propagateSelectionToParent(selection: any[]): void {\n    this.log.debug('Propagating selection to iframe parent:', selection);\n\n    const message = this.buildSelectionMessage(selection);\n    this.iframeService.postDataMessage(message);\n  }\n\n  private buildSelectionMessage(selection: any[]): DataViewMessage<any> {\n    return DataViewMessage.selection(new DataViewSelection(selection));\n  }\n\n}\n"]}
@@ -2,6 +2,7 @@ import { ChangeDetectionStrategy, Component, EventEmitter, HostListener, Input,
2
2
  import { LoggerFactory } from '@elderbyte/ts-logger';
3
3
  import { BehaviorSubject } from 'rxjs';
4
4
  import { TypedEventMessage } from '../typed-event-message';
5
+ import { IframeMessages } from '../iframe-messages';
5
6
  import * as i0 from "@angular/core";
6
7
  import * as i1 from "@angular/common";
7
8
  import * as i2 from "@angular/material/icon";
@@ -18,7 +19,10 @@ export class IFrameState {
18
19
  return this.type === 'LOADING';
19
20
  }
20
21
  get hasLoaded() {
21
- return this.type == 'LOADED';
22
+ return this.type === 'LOADED';
23
+ }
24
+ get isViewReady() {
25
+ return this.type === 'VIEW_READY';
22
26
  }
23
27
  get hasError() {
24
28
  return this.type === 'ERROR';
@@ -54,10 +58,11 @@ export class IframeHostComponent {
54
58
  * @param event message from child window
55
59
  */
56
60
  onWindowMessage(event) {
57
- this.log.debug('Received message from child:', event);
58
- if (event) {
59
- if (TypedEventMessage.isOfType(event.data))
61
+ if (event && event.data) {
62
+ if (TypedEventMessage.isOfType(event.data)) {
63
+ this.log.debug('Received message from child:', event.data);
60
64
  this.handleMessageFromChild(event.data);
65
+ }
61
66
  }
62
67
  return true;
63
68
  }
@@ -102,9 +107,12 @@ export class IframeHostComponent {
102
107
  **************************************************************************/
103
108
  handleMessageFromChild(message) {
104
109
  this.message.next(message);
105
- if (message.type === 'CLOSE') {
110
+ if (message.type === IframeMessages.CLOSE) {
106
111
  this.close.emit(message.data);
107
112
  }
113
+ if (message.type === IframeMessages.VIEW_READY_EVENT) {
114
+ this.iframeState$.next(new IFrameState('VIEW_READY'));
115
+ }
108
116
  }
109
117
  }
110
118
  IframeHostComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.7", ngImport: i0, type: IframeHostComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
@@ -125,4 +133,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
125
133
  }], iframeParams: [{
126
134
  type: Input
127
135
  }] } });
128
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iframe-host.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/iframe-host/iframe-host.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/iframe-host/iframe-host.component.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectionStrategy, Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, ViewChild} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject} from 'rxjs';\nimport {TypedEventMessage} from '../typed-event-message';\n\nexport type IFrameStateType = 'LOADING' | 'LOADED' | 'ERROR';\n\nexport class IFrameState {\n\n  constructor(\n    public readonly type: IFrameStateType,\n    public readonly message: string | null = null\n  ) {\n  }\n\n  public get isLoading(): boolean {\n    return this.type === 'LOADING'\n  }\n\n  public get hasLoaded(): boolean {\n    return this.type == 'LOADED'\n  }\n\n  public get hasError(): boolean {\n    return this.type === 'ERROR';\n  }\n\n}\n\nexport interface IFrameParams {\n\n  /**\n   * Source URL of iframe.\n   */\n  readonly srcUrl: string;\n\n  /**\n   * Width of the iframe.\n   */\n  readonly width?: string;\n\n  /**\n   * Height of the iframe.\n   */\n  readonly height?: string;\n\n  /**\n   * If true, the bottom action bar\n   * is hidden.\n   */\n  readonly hideActionBar?: boolean;\n\n}\n\n\n@Component({\n  selector: 'elder-i-frame-host',\n  templateUrl: './iframe-host.component.html',\n  styleUrls: ['./iframe-host.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class IframeHostComponent implements OnInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  public readonly iframeParams$ = new BehaviorSubject<IFrameParams>(null);\n  public readonly iframeState$ = new BehaviorSubject<IFrameState>(new IFrameState('LOADING'));\n\n  @ViewChild('iframe')\n  public iframe: ElementRef;\n\n  @Output()\n  public message = new EventEmitter<TypedEventMessage<any, any>>();\n\n  @Output()\n  public close = new EventEmitter<any>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor() {}\n\n  /***************************************************************************\n   *                                                                         *\n   * Listeners                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Listens for messages from window.\n   * Makes it possible to react upon messages from iframe child window.\n   *\n   * @param event message from child window\n   */\n  @HostListener('window:message', ['$event'])\n  public onWindowMessage(event: MessageEvent): boolean {\n    this.log.debug('Received message from child:', event);\n    if (event) {\n      if (TypedEventMessage.isOfType(event.data))\n      this.handleMessageFromChild(event.data);\n    }\n    return true;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Lifecycle                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void { }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set iframeParams(params: IFrameParams) {\n    this.iframeParams$.next(params);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onIframeLoadEvent(event: Event): void {\n    this.log.debug('Finished loading iframe', event);\n    this.iframeState$.next(new IFrameState('LOADED'));\n  }\n\n  public onIframeErrorEvent(err: any): void {\n    this.log.warn('Failed to load iframe', err);\n    this.iframeState$.next(new IFrameState('ERROR', err + ''));\n  }\n\n  public sendMessageToChild(msg: TypedEventMessage<any,any>): void {\n    this.log.debug('Sending message to iframe child:', msg);\n    this.iframe.nativeElement.contentWindow.postMessage(msg, \"*\");\n  }\n\n  public doClose(event?: Event): void {\n    this.close.emit();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleMessageFromChild(message: TypedEventMessage<any, any>): void {\n    this.message.next(message);\n\n    if (message.type === 'CLOSE') {\n      this.close.emit(message.data);\n    }\n  }\n\n}\n","<ng-container *ngIf=\"{state: iframeState$ | async, params: iframeParams$ | async} as ctx\">\n\n    <div class=\"layout-col full\">\n\n        <!-- In Case of Error -->\n        <div *ngIf=\"ctx.state.hasError\"\n             class=\"layout-row place-center-center flex\"\n        >\n            <span>Failed to load Iframe: {{ctx.state.message}}</span>\n        </div>\n\n        <!-- IFrame -->\n        <iframe #iframe\n                *ngIf=\"!ctx.state.hasError\"\n                (load)=\"onIframeLoadEvent($event)\"\n                (error)=\"onIframeErrorEvent($event)\"\n                [src]=\"ctx.params.srcUrl | elderSafeUrl\"\n                class=\"iframe flex\"\n                allow=\"clipboard *;clipboard-read *;clipboard-write *;\"\n        ></iframe>\n\n        <!-- Bottom Toolbar -->\n      <mat-toolbar *ngIf=\"!ctx.params.hideActionBar\" elderTheme class=\"layout-row place-between-center flex-none elder-dense-toolbar\">\n\n        <a mat-icon-button (click)=\"doClose($event)\" class=\"elder-icon-button-sm\">\n          <mat-icon>close</mat-icon>\n        </a>\n\n        <a mat-icon-button class=\"elder-icon-button-sm\"\n           [href]=\"ctx.params.srcUrl | elderSafeUrl\" target=\"_blank\">\n          <mat-icon>open_in_new</mat-icon>\n        </a>\n      </mat-toolbar>\n  </div>\n\n</ng-container>\n\n\n\n"]}
136
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"iframe-host.component.js","sourceRoot":"","sources":["../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/iframe-host/iframe-host.component.ts","../../../../../../../../projects/elderbyte/ngx-starter/src/lib/components/iframes/iframe-host/iframe-host.component.html"],"names":[],"mappings":"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","sourcesContent":["import {ChangeDetectionStrategy, Component, ElementRef, EventEmitter, HostListener, Input, OnInit, Output, ViewChild} from '@angular/core';\nimport {LoggerFactory} from '@elderbyte/ts-logger';\nimport {BehaviorSubject} from 'rxjs';\nimport {TypedEventMessage} from '../typed-event-message';\nimport {IframeMessages} from '../iframe-messages';\n\nexport type IFrameStateType = 'LOADING' | 'LOADED' | 'VIEW_READY' | 'ERROR';\n\nexport class IFrameState {\n\n  constructor(\n    public readonly type: IFrameStateType,\n    public readonly message: string | null = null\n  ) {\n  }\n\n  public get isLoading(): boolean {\n    return this.type === 'LOADING'\n  }\n\n  public get hasLoaded(): boolean {\n    return this.type === 'LOADED'\n  }\n\n  public get isViewReady(): boolean {\n    return this.type === 'VIEW_READY'\n  }\n\n  public get hasError(): boolean {\n    return this.type === 'ERROR';\n  }\n\n}\n\nexport interface IFrameParams {\n\n  /**\n   * Source URL of iframe.\n   */\n  readonly srcUrl: string;\n\n  /**\n   * Width of the iframe.\n   */\n  readonly width?: string;\n\n  /**\n   * Height of the iframe.\n   */\n  readonly height?: string;\n\n  /**\n   * If true, the bottom action bar\n   * is hidden.\n   */\n  readonly hideActionBar?: boolean;\n\n}\n\n\n@Component({\n  selector: 'elder-i-frame-host',\n  templateUrl: './iframe-host.component.html',\n  styleUrls: ['./iframe-host.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush\n})\nexport class IframeHostComponent implements OnInit {\n\n  /***************************************************************************\n   *                                                                         *\n   * Fields                                                                  *\n   *                                                                         *\n   **************************************************************************/\n\n  private readonly log = LoggerFactory.getLogger(this.constructor.name);\n\n  public readonly iframeParams$ = new BehaviorSubject<IFrameParams>(null);\n  public readonly iframeState$ = new BehaviorSubject<IFrameState>(new IFrameState('LOADING'));\n\n  @ViewChild('iframe')\n  public iframe: ElementRef;\n\n  @Output()\n  public message = new EventEmitter<TypedEventMessage<any, any>>();\n\n  @Output()\n  public close = new EventEmitter<any>();\n\n  /***************************************************************************\n   *                                                                         *\n   * Constructor                                                             *\n   *                                                                         *\n   **************************************************************************/\n\n  constructor() {}\n\n  /***************************************************************************\n   *                                                                         *\n   * Listeners                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  /**\n   * Listens for messages from window.\n   * Makes it possible to react upon messages from iframe child window.\n   *\n   * @param event message from child window\n   */\n  @HostListener('window:message', ['$event'])\n  public onWindowMessage(event: MessageEvent): boolean {\n    if (event && event.data) {\n      if (TypedEventMessage.isOfType(event.data)){\n        this.log.debug('Received message from child:', event.data);\n        this.handleMessageFromChild(event.data);\n      }\n\n    }\n    return true;\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Lifecycle                                                               *\n   *                                                                         *\n   **************************************************************************/\n\n  public ngOnInit(): void { }\n\n  /***************************************************************************\n   *                                                                         *\n   * Properties                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  @Input()\n  public set iframeParams(params: IFrameParams) {\n    this.iframeParams$.next(params);\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Public API                                                              *\n   *                                                                         *\n   **************************************************************************/\n\n  public onIframeLoadEvent(event: Event): void {\n    this.log.debug('Finished loading iframe', event);\n    this.iframeState$.next(new IFrameState('LOADED'));\n  }\n\n  public onIframeErrorEvent(err: any): void {\n    this.log.warn('Failed to load iframe', err);\n    this.iframeState$.next(new IFrameState('ERROR', err + ''));\n  }\n\n  public sendMessageToChild(msg: TypedEventMessage<any,any>): void {\n    this.log.debug('Sending message to iframe child:', msg);\n    this.iframe.nativeElement.contentWindow.postMessage(msg, \"*\");\n  }\n\n  public doClose(event?: Event): void {\n    this.close.emit();\n  }\n\n  /***************************************************************************\n   *                                                                         *\n   * Private methods                                                         *\n   *                                                                         *\n   **************************************************************************/\n\n  private handleMessageFromChild(message: TypedEventMessage<any, any>): void {\n    this.message.next(message);\n\n    if (message.type === IframeMessages.CLOSE) {\n      this.close.emit(message.data);\n    }\n\n    if (message.type === IframeMessages.VIEW_READY_EVENT) {\n      this.iframeState$.next(new IFrameState('VIEW_READY'));\n    }\n  }\n\n}\n","<ng-container *ngIf=\"{state: iframeState$ | async, params: iframeParams$ | async} as ctx\">\n\n    <div class=\"layout-col full\">\n\n        <!-- In Case of Error -->\n        <div *ngIf=\"ctx.state.hasError\"\n             class=\"layout-row place-center-center flex\"\n        >\n            <span>Failed to load Iframe: {{ctx.state.message}}</span>\n        </div>\n\n        <!-- IFrame -->\n        <iframe #iframe\n                *ngIf=\"!ctx.state.hasError\"\n                (load)=\"onIframeLoadEvent($event)\"\n                (error)=\"onIframeErrorEvent($event)\"\n                [src]=\"ctx.params.srcUrl | elderSafeUrl\"\n                class=\"iframe flex\"\n                allow=\"clipboard *;clipboard-read *;clipboard-write *;\"\n        ></iframe>\n\n        <!-- Bottom Toolbar -->\n      <mat-toolbar *ngIf=\"!ctx.params.hideActionBar\" elderTheme class=\"layout-row place-between-center flex-none elder-dense-toolbar\">\n\n        <a mat-icon-button (click)=\"doClose($event)\" class=\"elder-icon-button-sm\">\n          <mat-icon>close</mat-icon>\n        </a>\n\n        <a mat-icon-button class=\"elder-icon-button-sm\"\n           [href]=\"ctx.params.srcUrl | elderSafeUrl\" target=\"_blank\">\n          <mat-icon>open_in_new</mat-icon>\n        </a>\n      </mat-toolbar>\n  </div>\n\n</ng-container>\n\n\n\n"]}
@@ -0,0 +1,5 @@
1
+ export class IframeMessages {
2
+ }
3
+ IframeMessages.CLOSE = 'CLOSE';
4
+ IframeMessages.VIEW_READY_EVENT = 'VIEW_READY_EVENT';
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWZyYW1lLW1lc3NhZ2VzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZWxkZXJieXRlL25neC1zdGFydGVyL3NyYy9saWIvY29tcG9uZW50cy9pZnJhbWVzL2lmcmFtZS1tZXNzYWdlcy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFDQSxNQUFNLE9BQU8sY0FBYzs7QUFFRixvQkFBSyxHQUFHLE9BQU8sQ0FBQztBQUVoQiwrQkFBZ0IsR0FBRyxrQkFBa0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIlxuZXhwb3J0IGNsYXNzIElmcmFtZU1lc3NhZ2VzIHtcblxuICBwdWJsaWMgc3RhdGljIHJlYWRvbmx5IENMT1NFID0gJ0NMT1NFJztcblxuICBwdWJsaWMgc3RhdGljIHJlYWRvbmx5IFZJRVdfUkVBRFlfRVZFTlQgPSAnVklFV19SRUFEWV9FVkVOVCc7XG5cblxufVxuIl19
@@ -1,6 +1,7 @@
1
1
  import { Injectable } from '@angular/core';
2
2
  import { IframeDialogComponent } from './iframe-dialog/iframe-dialog.component';
3
3
  import { TypedEventMessage } from './typed-event-message';
4
+ import { IframeMessages } from './iframe-messages';
4
5
  import * as i0 from "@angular/core";
5
6
  import * as i1 from "@angular/material/dialog";
6
7
  export class IframeService {
@@ -31,7 +32,10 @@ export class IframeService {
31
32
  }).afterClosed();
32
33
  }
33
34
  postCloseMessage(data) {
34
- IframeService.postToParent(new TypedEventMessage('CLOSE', data));
35
+ IframeService.postToParent(new TypedEventMessage(IframeMessages.CLOSE, data));
36
+ }
37
+ postViewReadyEventMessage(data) {
38
+ IframeService.postToParent(new TypedEventMessage(IframeMessages.VIEW_READY_EVENT, data));
35
39
  }
36
40
  postDataMessage(message) {
37
41
  IframeService.postToParent(message);
@@ -45,4 +49,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.7", ngImpor
45
49
  providedIn: 'root'
46
50
  }]
47
51
  }], ctorParameters: function () { return [{ type: i1.MatDialog }]; } });
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWZyYW1lLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9jb21wb25lbnRzL2lmcmFtZXMvaWZyYW1lLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFVBQVUsRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUd6QyxPQUFPLEVBQUMscUJBQXFCLEVBQUMsTUFBTSx5Q0FBeUMsQ0FBQztBQUU5RSxPQUFPLEVBQUMsaUJBQWlCLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQzs7O0FBS3hELE1BQU0sT0FBTyxhQUFhO0lBU3RCOzs7O2dGQUk0RTtJQUU1RSxZQUNTLE1BQWlCO1FBQWpCLFdBQU0sR0FBTixNQUFNLENBQVc7SUFDdEIsQ0FBQztJQWZFLE1BQU0sQ0FBQyxZQUFZLENBQU8sT0FBZ0M7UUFDN0QsTUFBTSxDQUFDLFdBQVcsQ0FDaEIsT0FBTyxFQUNQLEdBQUcsQ0FDSixDQUFDLENBQUUscUJBQXFCO0lBQzdCLENBQUM7SUFZRDs7OztnRkFJNEU7SUFFckUsZ0JBQWdCLENBQUksTUFBb0I7UUFDM0MsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBeUMscUJBQXFCLEVBQUU7WUFDbkYsVUFBVSxFQUFFLG1CQUFtQjtZQUMvQixRQUFRLEVBQUUsT0FBTztZQUNqQixTQUFTLEVBQUUsT0FBTztZQUNsQixLQUFLLEVBQUUsTUFBTSxDQUFDLEtBQUssSUFBSSxNQUFNO1lBQzdCLE1BQU0sRUFBRSxNQUFNLENBQUMsTUFBTSxJQUFJLE1BQU07WUFDL0IsSUFBSSxFQUFFLE1BQU07U0FDZixDQUFDLENBQUMsV0FBVyxFQUFFLENBQUM7SUFDckIsQ0FBQztJQUVNLGdCQUFnQixDQUFJLElBQVE7UUFDL0IsYUFBYSxDQUFDLFlBQVksQ0FBQyxJQUFJLGlCQUFpQixDQUM5QyxPQUFPLEVBQ1AsSUFBSSxDQUNMLENBQUMsQ0FBQztJQUNQLENBQUM7SUFFTSxlQUFlLENBQU8sT0FBZ0M7UUFDekQsYUFBYSxDQUFDLFlBQVksQ0FBQyxPQUFPLENBQUMsQ0FBQztJQUN4QyxDQUFDOzswR0E3Q1EsYUFBYTs4R0FBYixhQUFhLGNBRlYsTUFBTTsyRkFFVCxhQUFhO2tCQUh6QixVQUFVO21CQUFDO29CQUNSLFVBQVUsRUFBRSxNQUFNO2lCQUNyQiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7SW5qZWN0YWJsZX0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge09ic2VydmFibGV9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtNYXREaWFsb2d9IGZyb20gJ0Bhbmd1bGFyL21hdGVyaWFsL2RpYWxvZyc7XG5pbXBvcnQge0lmcmFtZURpYWxvZ0NvbXBvbmVudH0gZnJvbSAnLi9pZnJhbWUtZGlhbG9nL2lmcmFtZS1kaWFsb2cuY29tcG9uZW50JztcbmltcG9ydCB7SUZyYW1lUGFyYW1zfSBmcm9tICcuL2lmcmFtZS1ob3N0L2lmcmFtZS1ob3N0LmNvbXBvbmVudCc7XG5pbXBvcnQge1R5cGVkRXZlbnRNZXNzYWdlfSBmcm9tICcuL3R5cGVkLWV2ZW50LW1lc3NhZ2UnO1xuXG5ASW5qZWN0YWJsZSh7XG4gICAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIElmcmFtZVNlcnZpY2Uge1xuXG4gICAgcHVibGljIHN0YXRpYyBwb3N0VG9QYXJlbnQ8VCwgRD4obWVzc2FnZTogVHlwZWRFdmVudE1lc3NhZ2U8VCwgRD4pOiB2b2lkIHtcbiAgICAgICAgcGFyZW50LnBvc3RNZXNzYWdlKFxuICAgICAgICAgIG1lc3NhZ2UsXG4gICAgICAgICAgJyonXG4gICAgICAgICk7ICAvLyAgYCpgIG9uIGFueSBkb21haW5cbiAgICB9XG5cbiAgICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgICAqIEZpZWxkcyAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gICAgY29uc3RydWN0b3IoXG4gICAgICBwdWJsaWMgZGlhbG9nOiBNYXREaWFsb2dcbiAgICApIHsgfVxuXG4gICAgLyoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKlxuICAgICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICAgKiBQdWJsaWMgQVBJICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgICAqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKi9cblxuICAgIHB1YmxpYyBvcGVuSWZyYW1lRGlhbG9nPFI+KHBhcmFtczogSUZyYW1lUGFyYW1zKTogT2JzZXJ2YWJsZTxSPiB7XG4gICAgICAgIHJldHVybiB0aGlzLmRpYWxvZy5vcGVuPElmcmFtZURpYWxvZ0NvbXBvbmVudCwgSUZyYW1lUGFyYW1zLCBSPihJZnJhbWVEaWFsb2dDb21wb25lbnQsIHtcbiAgICAgICAgICAgIHBhbmVsQ2xhc3M6ICduby1wYWRkaW5nLWRpYWxvZycsXG4gICAgICAgICAgICBtYXhXaWR0aDogJzEwMHZ3JyxcbiAgICAgICAgICAgIG1heEhlaWdodDogJzEwMHZoJyxcbiAgICAgICAgICAgIHdpZHRoOiBwYXJhbXMud2lkdGggfHwgJzkwdncnLFxuICAgICAgICAgICAgaGVpZ2h0OiBwYXJhbXMuaGVpZ2h0IHx8ICc5MHZoJyxcbiAgICAgICAgICAgIGRhdGE6IHBhcmFtc1xuICAgICAgICB9KS5hZnRlckNsb3NlZCgpO1xuICAgIH1cblxuICAgIHB1YmxpYyBwb3N0Q2xvc2VNZXNzYWdlPFQ+KGRhdGE/OiBUKTogdm9pZCB7XG4gICAgICAgIElmcmFtZVNlcnZpY2UucG9zdFRvUGFyZW50KG5ldyBUeXBlZEV2ZW50TWVzc2FnZTxzdHJpbmcsIFQ+KFxuICAgICAgICAgICdDTE9TRScsXG4gICAgICAgICAgZGF0YVxuICAgICAgICApKTtcbiAgICB9XG5cbiAgICBwdWJsaWMgcG9zdERhdGFNZXNzYWdlPFQsIEQ+KG1lc3NhZ2U6IFR5cGVkRXZlbnRNZXNzYWdlPFQsIEQ+KTogdm9pZCB7XG4gICAgICAgIElmcmFtZVNlcnZpY2UucG9zdFRvUGFyZW50KG1lc3NhZ2UpO1xuICAgIH1cblxuXG59XG4iXX0=
52
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaWZyYW1lLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9lbGRlcmJ5dGUvbmd4LXN0YXJ0ZXIvc3JjL2xpYi9jb21wb25lbnRzL2lmcmFtZXMvaWZyYW1lLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFDLFVBQVUsRUFBQyxNQUFNLGVBQWUsQ0FBQztBQUd6QyxPQUFPLEVBQUMscUJBQXFCLEVBQUMsTUFBTSx5Q0FBeUMsQ0FBQztBQUU5RSxPQUFPLEVBQUMsaUJBQWlCLEVBQUMsTUFBTSx1QkFBdUIsQ0FBQztBQUN4RCxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sbUJBQW1CLENBQUM7OztBQUtqRCxNQUFNLE9BQU8sYUFBYTtJQVN0Qjs7OztnRkFJNEU7SUFFNUUsWUFDUyxNQUFpQjtRQUFqQixXQUFNLEdBQU4sTUFBTSxDQUFXO0lBQ3RCLENBQUM7SUFmRSxNQUFNLENBQUMsWUFBWSxDQUFPLE9BQWdDO1FBQzdELE1BQU0sQ0FBQyxXQUFXLENBQ2hCLE9BQU8sRUFDUCxHQUFHLENBQ0osQ0FBQyxDQUFFLHFCQUFxQjtJQUM3QixDQUFDO0lBWUQ7Ozs7Z0ZBSTRFO0lBRXJFLGdCQUFnQixDQUFJLE1BQW9CO1FBQzNDLE9BQU8sSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQXlDLHFCQUFxQixFQUFFO1lBQ25GLFVBQVUsRUFBRSxtQkFBbUI7WUFDL0IsUUFBUSxFQUFFLE9BQU87WUFDakIsU0FBUyxFQUFFLE9BQU87WUFDbEIsS0FBSyxFQUFFLE1BQU0sQ0FBQyxLQUFLLElBQUksTUFBTTtZQUM3QixNQUFNLEVBQUUsTUFBTSxDQUFDLE1BQU0sSUFBSSxNQUFNO1lBQy9CLElBQUksRUFBRSxNQUFNO1NBQ2YsQ0FBQyxDQUFDLFdBQVcsRUFBRSxDQUFDO0lBQ3JCLENBQUM7SUFFTSxnQkFBZ0IsQ0FBSSxJQUFRO1FBQy9CLGFBQWEsQ0FBQyxZQUFZLENBQUMsSUFBSSxpQkFBaUIsQ0FDOUMsY0FBYyxDQUFDLEtBQUssRUFDcEIsSUFBSSxDQUNMLENBQUMsQ0FBQztJQUNQLENBQUM7SUFFSSx5QkFBeUIsQ0FBSSxJQUFRO1FBQzFDLGFBQWEsQ0FBQyxZQUFZLENBQUMsSUFBSSxpQkFBaUIsQ0FDOUMsY0FBYyxDQUFDLGdCQUFnQixFQUMvQixJQUFJLENBQ0wsQ0FBQyxDQUFDO0lBQ0wsQ0FBQztJQUVRLGVBQWUsQ0FBTyxPQUFnQztRQUN6RCxhQUFhLENBQUMsWUFBWSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ3hDLENBQUM7OzBHQXBEUSxhQUFhOzhHQUFiLGFBQWEsY0FGVixNQUFNOzJGQUVULGFBQWE7a0JBSHpCLFVBQVU7bUJBQUM7b0JBQ1IsVUFBVSxFQUFFLE1BQU07aUJBQ3JCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHtJbmplY3RhYmxlfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7T2JzZXJ2YWJsZX0gZnJvbSAncnhqcyc7XG5pbXBvcnQge01hdERpYWxvZ30gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvZGlhbG9nJztcbmltcG9ydCB7SWZyYW1lRGlhbG9nQ29tcG9uZW50fSBmcm9tICcuL2lmcmFtZS1kaWFsb2cvaWZyYW1lLWRpYWxvZy5jb21wb25lbnQnO1xuaW1wb3J0IHtJRnJhbWVQYXJhbXN9IGZyb20gJy4vaWZyYW1lLWhvc3QvaWZyYW1lLWhvc3QuY29tcG9uZW50JztcbmltcG9ydCB7VHlwZWRFdmVudE1lc3NhZ2V9IGZyb20gJy4vdHlwZWQtZXZlbnQtbWVzc2FnZSc7XG5pbXBvcnQge0lmcmFtZU1lc3NhZ2VzfSBmcm9tICcuL2lmcmFtZS1tZXNzYWdlcyc7XG5cbkBJbmplY3RhYmxlKHtcbiAgICBwcm92aWRlZEluOiAncm9vdCdcbn0pXG5leHBvcnQgY2xhc3MgSWZyYW1lU2VydmljZSB7XG5cbiAgICBwdWJsaWMgc3RhdGljIHBvc3RUb1BhcmVudDxULCBEPihtZXNzYWdlOiBUeXBlZEV2ZW50TWVzc2FnZTxULCBEPik6IHZvaWQge1xuICAgICAgICBwYXJlbnQucG9zdE1lc3NhZ2UoXG4gICAgICAgICAgbWVzc2FnZSxcbiAgICAgICAgICAnKidcbiAgICAgICAgKTsgIC8vICBgKmAgb24gYW55IGRvbWFpblxuICAgIH1cblxuICAgIC8qKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKipcbiAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAgICogRmllbGRzICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgICAqICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICAgKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKiovXG5cbiAgICBjb25zdHJ1Y3RvcihcbiAgICAgIHB1YmxpYyBkaWFsb2c6IE1hdERpYWxvZ1xuICAgICkgeyB9XG5cbiAgICAvKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqXG4gICAgICogICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgKlxuICAgICAqIFB1YmxpYyBBUEkgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICpcbiAgICAgKiAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAgICAqXG4gICAgICoqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqKioqL1xuXG4gICAgcHVibGljIG9wZW5JZnJhbWVEaWFsb2c8Uj4ocGFyYW1zOiBJRnJhbWVQYXJhbXMpOiBPYnNlcnZhYmxlPFI+IHtcbiAgICAgICAgcmV0dXJuIHRoaXMuZGlhbG9nLm9wZW48SWZyYW1lRGlhbG9nQ29tcG9uZW50LCBJRnJhbWVQYXJhbXMsIFI+KElmcmFtZURpYWxvZ0NvbXBvbmVudCwge1xuICAgICAgICAgICAgcGFuZWxDbGFzczogJ25vLXBhZGRpbmctZGlhbG9nJyxcbiAgICAgICAgICAgIG1heFdpZHRoOiAnMTAwdncnLFxuICAgICAgICAgICAgbWF4SGVpZ2h0OiAnMTAwdmgnLFxuICAgICAgICAgICAgd2lkdGg6IHBhcmFtcy53aWR0aCB8fCAnOTB2dycsXG4gICAgICAgICAgICBoZWlnaHQ6IHBhcmFtcy5oZWlnaHQgfHwgJzkwdmgnLFxuICAgICAgICAgICAgZGF0YTogcGFyYW1zXG4gICAgICAgIH0pLmFmdGVyQ2xvc2VkKCk7XG4gICAgfVxuXG4gICAgcHVibGljIHBvc3RDbG9zZU1lc3NhZ2U8VD4oZGF0YT86IFQpOiB2b2lkIHtcbiAgICAgICAgSWZyYW1lU2VydmljZS5wb3N0VG9QYXJlbnQobmV3IFR5cGVkRXZlbnRNZXNzYWdlPHN0cmluZywgVD4oXG4gICAgICAgICAgSWZyYW1lTWVzc2FnZXMuQ0xPU0UsXG4gICAgICAgICAgZGF0YVxuICAgICAgICApKTtcbiAgICB9XG5cbiAgcHVibGljIHBvc3RWaWV3UmVhZHlFdmVudE1lc3NhZ2U8VD4oZGF0YT86IFQpOiB2b2lkIHtcbiAgICBJZnJhbWVTZXJ2aWNlLnBvc3RUb1BhcmVudChuZXcgVHlwZWRFdmVudE1lc3NhZ2U8c3RyaW5nLCBUPihcbiAgICAgIElmcmFtZU1lc3NhZ2VzLlZJRVdfUkVBRFlfRVZFTlQsXG4gICAgICBkYXRhXG4gICAgKSk7XG4gIH1cblxuICAgIHB1YmxpYyBwb3N0RGF0YU1lc3NhZ2U8VCwgRD4obWVzc2FnZTogVHlwZWRFdmVudE1lc3NhZ2U8VCwgRD4pOiB2b2lkIHtcbiAgICAgICAgSWZyYW1lU2VydmljZS5wb3N0VG9QYXJlbnQobWVzc2FnZSk7XG4gICAgfVxuXG5cbn1cbiJdfQ==
@@ -5,8 +5,7 @@ export class TypedEventMessage {
5
5
  }
6
6
  // type predicate to check weather an object is a DataViewMessage type
7
7
  static isOfType(obj) {
8
- return obj.type !== undefined
9
- && obj.data !== undefined;
8
+ return obj.type !== undefined;
10
9
  }
11
10
  }
12
11
  export class DataViewSelectionInit {
@@ -21,6 +20,8 @@ export class DataViewSelection {
21
20
  this.selection = selection;
22
21
  }
23
22
  }
23
+ export const DataViewMessageTypeValues = ["SELECTION_INIT", "SELECTION_SELECTED"];
24
+ export const isDataViewMessageType = (x) => DataViewMessageTypeValues.includes(x);
24
25
  export class DataViewMessage extends TypedEventMessage {
25
26
  constructor(type, data) {
26
27
  super(type, data);
@@ -35,8 +36,7 @@ export class DataViewMessage extends TypedEventMessage {
35
36
  }
36
37
  // type predicate to check weather an object is a DataViewMessage type
37
38
  static isOfType(obj) {
38
- return obj.type !== undefined
39
- && obj.data !== undefined;
39
+ return isDataViewMessageType(obj?.type);
40
40
  }
41
41
  }
42
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWQtZXZlbnQtbWVzc2FnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbXBvbmVudHMvaWZyYW1lcy90eXBlZC1ldmVudC1tZXNzYWdlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sT0FBTyxpQkFBaUI7SUFRNUIsWUFDUyxJQUFPLEVBQ1AsSUFBUTtRQURSLFNBQUksR0FBSixJQUFJLENBQUc7UUFDUCxTQUFJLEdBQUosSUFBSSxDQUFJO0lBQ2QsQ0FBQztJQVRKLHNFQUFzRTtJQUMvRCxNQUFNLENBQUMsUUFBUSxDQUFDLEdBQVE7UUFDN0IsT0FBUSxHQUE0QixDQUFDLElBQUksS0FBSyxTQUFTO2VBQ2pELEdBQTRCLENBQUMsSUFBSSxLQUFLLFNBQVMsQ0FBQTtJQUN2RCxDQUFDO0NBT0Y7QUFFRCxNQUFNLE9BQU8scUJBQXFCO0lBRWhDLFlBQ1MsT0FBNkIsRUFDN0IsbUJBQXdCLEVBQUUsRUFDMUIsVUFBa0IsSUFBSTtRQUZ0QixZQUFPLEdBQVAsT0FBTyxDQUFzQjtRQUM3QixxQkFBZ0IsR0FBaEIsZ0JBQWdCLENBQVU7UUFDMUIsWUFBTyxHQUFQLE9BQU8sQ0FBZTtJQUUvQixDQUFDO0NBQ0Y7QUFFRCxNQUFNLE9BQU8saUJBQWlCO0lBQzVCLFlBQ1MsU0FBYztRQUFkLGNBQVMsR0FBVCxTQUFTLENBQUs7SUFFdkIsQ0FBQztDQUNGO0FBSUQsTUFBTSxPQUFPLGVBQXdFLFNBQVEsaUJBQXlDO0lBc0JwSSxZQUNTLElBQXlCLEVBQ3pCLElBQU87UUFFZCxLQUFLLENBQUMsSUFBSSxFQUFFLElBQUksQ0FBQyxDQUFDO1FBSFgsU0FBSSxHQUFKLElBQUksQ0FBcUI7UUFDekIsU0FBSSxHQUFKLElBQUksQ0FBRztJQUdoQixDQUFDO0lBekJNLE1BQU0sQ0FBQyxnQkFBZ0IsQ0FBSSxJQUE4QjtRQUM5RCxPQUFPLElBQUksZUFBZSxDQUN4QixnQkFBZ0IsRUFDaEIsSUFBSSxDQUNMLENBQUE7SUFDSCxDQUFDO0lBRU0sTUFBTSxDQUFDLFNBQVMsQ0FBSSxTQUErQjtRQUN4RCxPQUFPLElBQUksZUFBZSxDQUN4QixvQkFBb0IsRUFDcEIsU0FBUyxDQUNWLENBQUE7SUFDSCxDQUFDO0lBRUQsc0VBQXNFO0lBQy9ELE1BQU0sQ0FBQyxRQUFRLENBQUMsR0FBUTtRQUM3QixPQUFRLEdBQTRCLENBQUMsSUFBSSxLQUFLLFNBQVM7ZUFDakQsR0FBNEIsQ0FBQyxJQUFJLEtBQUssU0FBUyxDQUFBO0lBQ3ZELENBQUM7Q0FTRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7RWxkZXJEYXRhVmlld09wdGlvbnN9IGZyb20gJy4uL2RhdGEtdmlldy9iYXNlL2VsZGVyLWRhdGEtdmlldy1vcHRpb25zJztcblxuZXhwb3J0IGNsYXNzIFR5cGVkRXZlbnRNZXNzYWdlPFQsIEQ+IHtcblxuICAvLyB0eXBlIHByZWRpY2F0ZSB0byBjaGVjayB3ZWF0aGVyIGFuIG9iamVjdCBpcyBhIERhdGFWaWV3TWVzc2FnZSB0eXBlXG4gIHB1YmxpYyBzdGF0aWMgaXNPZlR5cGUob2JqOiBhbnkpOiBvYmogaXMgVHlwZWRFdmVudE1lc3NhZ2U8YW55LGFueT4ge1xuICAgIHJldHVybiAob2JqIGFzIERhdGFWaWV3TWVzc2FnZTxhbnk+KS50eXBlICE9PSB1bmRlZmluZWRcbiAgICAgICYmIChvYmogYXMgRGF0YVZpZXdNZXNzYWdlPGFueT4pLmRhdGEgIT09IHVuZGVmaW5lZFxuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHR5cGU6IFQsXG4gICAgcHVibGljIGRhdGE/OiBEXG4gICkge31cblxufVxuXG5leHBvcnQgY2xhc3MgRGF0YVZpZXdTZWxlY3Rpb25Jbml0PFQ+IHtcblxuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgb3B0aW9uczogRWxkZXJEYXRhVmlld09wdGlvbnMsXG4gICAgcHVibGljIGluaXRpYWxTZWxlY3Rpb246IFRbXSA9IFtdLFxuICAgIHB1YmxpYyBpZEZpZWxkOiBzdHJpbmcgPSAnaWQnXG4gICkge1xuICB9XG59XG5cbmV4cG9ydCBjbGFzcyBEYXRhVmlld1NlbGVjdGlvbjxUPiB7XG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBzZWxlY3Rpb246IFRbXVxuICApIHtcbiAgfVxufVxuXG5leHBvcnQgdHlwZSBEYXRhVmlld01lc3NhZ2VUeXBlID0gJ1NFTEVDVElPTl9JTklUJyB8ICdTRUxFQ1RJT05fU0VMRUNURUQnXG5cbmV4cG9ydCBjbGFzcyBEYXRhVmlld01lc3NhZ2U8VCwgRCA9IERhdGFWaWV3U2VsZWN0aW9uSW5pdDxUPiB8IERhdGFWaWV3U2VsZWN0aW9uPFQ+PiBleHRlbmRzIFR5cGVkRXZlbnRNZXNzYWdlPERhdGFWaWV3TWVzc2FnZVR5cGUsIEQ+IHtcblxuICBwdWJsaWMgc3RhdGljIGluaXRpYWxTZWxlY3Rpb248RD4oaW5pdDogRGF0YVZpZXdTZWxlY3Rpb25Jbml0PEQ+KTogRGF0YVZpZXdNZXNzYWdlPEQ+IHtcbiAgICByZXR1cm4gbmV3IERhdGFWaWV3TWVzc2FnZShcbiAgICAgICdTRUxFQ1RJT05fSU5JVCcsXG4gICAgICBpbml0XG4gICAgKVxuICB9XG5cbiAgcHVibGljIHN0YXRpYyBzZWxlY3Rpb248RD4oc2VsZWN0aW9uOiBEYXRhVmlld1NlbGVjdGlvbjxEPik6IERhdGFWaWV3TWVzc2FnZTxEPiB7XG4gICAgcmV0dXJuIG5ldyBEYXRhVmlld01lc3NhZ2UoXG4gICAgICAnU0VMRUNUSU9OX1NFTEVDVEVEJyxcbiAgICAgIHNlbGVjdGlvblxuICAgIClcbiAgfVxuXG4gIC8vIHR5cGUgcHJlZGljYXRlIHRvIGNoZWNrIHdlYXRoZXIgYW4gb2JqZWN0IGlzIGEgRGF0YVZpZXdNZXNzYWdlIHR5cGVcbiAgcHVibGljIHN0YXRpYyBpc09mVHlwZShvYmo6IGFueSk6IG9iaiBpcyBEYXRhVmlld01lc3NhZ2U8YW55LGFueT4ge1xuICAgIHJldHVybiAob2JqIGFzIERhdGFWaWV3TWVzc2FnZTxhbnk+KS50eXBlICE9PSB1bmRlZmluZWRcbiAgICAgICYmIChvYmogYXMgRGF0YVZpZXdNZXNzYWdlPGFueT4pLmRhdGEgIT09IHVuZGVmaW5lZFxuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHR5cGU6IERhdGFWaWV3TWVzc2FnZVR5cGUsXG4gICAgcHVibGljIGRhdGE6IERcbiAgKSB7XG4gICAgc3VwZXIodHlwZSwgZGF0YSk7XG4gIH1cblxufVxuIl19
42
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZWQtZXZlbnQtbWVzc2FnZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL2VsZGVyYnl0ZS9uZ3gtc3RhcnRlci9zcmMvbGliL2NvbXBvbmVudHMvaWZyYW1lcy90eXBlZC1ldmVudC1tZXNzYWdlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUVBLE1BQU0sT0FBTyxpQkFBaUI7SUFPNUIsWUFDUyxJQUFPLEVBQ1AsSUFBUTtRQURSLFNBQUksR0FBSixJQUFJLENBQUc7UUFDUCxTQUFJLEdBQUosSUFBSSxDQUFJO0lBQ2QsQ0FBQztJQVJKLHNFQUFzRTtJQUMvRCxNQUFNLENBQUMsUUFBUSxDQUFDLEdBQVE7UUFDN0IsT0FBUSxHQUFtQyxDQUFDLElBQUksS0FBSyxTQUFTLENBQUE7SUFDaEUsQ0FBQztDQU9GO0FBRUQsTUFBTSxPQUFPLHFCQUFxQjtJQUVoQyxZQUNTLE9BQTZCLEVBQzdCLG1CQUF3QixFQUFFLEVBQzFCLFVBQWtCLElBQUk7UUFGdEIsWUFBTyxHQUFQLE9BQU8sQ0FBc0I7UUFDN0IscUJBQWdCLEdBQWhCLGdCQUFnQixDQUFVO1FBQzFCLFlBQU8sR0FBUCxPQUFPLENBQWU7SUFFL0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxPQUFPLGlCQUFpQjtJQUM1QixZQUNTLFNBQWM7UUFBZCxjQUFTLEdBQVQsU0FBUyxDQUFLO0lBRXZCLENBQUM7Q0FDRjtBQUVELE1BQU0sQ0FBQyxNQUFNLHlCQUF5QixHQUFHLENBQUMsZ0JBQWdCLEVBQUUsb0JBQW9CLENBQVUsQ0FBQztBQUUzRixNQUFNLENBQUMsTUFBTSxxQkFBcUIsR0FBRyxDQUFDLENBQU0sRUFBNEIsRUFBRSxDQUFDLHlCQUF5QixDQUFDLFFBQVEsQ0FBQyxDQUFDLENBQUMsQ0FBQztBQUVqSCxNQUFNLE9BQU8sZUFBd0UsU0FBUSxpQkFBeUM7SUFxQnBJLFlBQ1MsSUFBeUIsRUFDekIsSUFBTztRQUVkLEtBQUssQ0FBQyxJQUFJLEVBQUUsSUFBSSxDQUFDLENBQUM7UUFIWCxTQUFJLEdBQUosSUFBSSxDQUFxQjtRQUN6QixTQUFJLEdBQUosSUFBSSxDQUFHO0lBR2hCLENBQUM7SUF4Qk0sTUFBTSxDQUFDLGdCQUFnQixDQUFJLElBQThCO1FBQzlELE9BQU8sSUFBSSxlQUFlLENBQ3hCLGdCQUFnQixFQUNoQixJQUFJLENBQ0wsQ0FBQTtJQUNILENBQUM7SUFFTSxNQUFNLENBQUMsU0FBUyxDQUFJLFNBQStCO1FBQ3hELE9BQU8sSUFBSSxlQUFlLENBQ3hCLG9CQUFvQixFQUNwQixTQUFTLENBQ1YsQ0FBQTtJQUNILENBQUM7SUFFRCxzRUFBc0U7SUFDL0QsTUFBTSxDQUFDLFFBQVEsQ0FBQyxHQUFRO1FBQzdCLE9BQU8scUJBQXFCLENBQUMsR0FBRyxFQUFFLElBQUksQ0FBQyxDQUFDO0lBQzFDLENBQUM7Q0FTRiIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7RWxkZXJEYXRhVmlld09wdGlvbnN9IGZyb20gJy4uL2RhdGEtdmlldy9iYXNlL2VsZGVyLWRhdGEtdmlldy1vcHRpb25zJztcblxuZXhwb3J0IGNsYXNzIFR5cGVkRXZlbnRNZXNzYWdlPFQsIEQ+IHtcblxuICAvLyB0eXBlIHByZWRpY2F0ZSB0byBjaGVjayB3ZWF0aGVyIGFuIG9iamVjdCBpcyBhIERhdGFWaWV3TWVzc2FnZSB0eXBlXG4gIHB1YmxpYyBzdGF0aWMgaXNPZlR5cGUob2JqOiBhbnkpOiBvYmogaXMgVHlwZWRFdmVudE1lc3NhZ2U8YW55LGFueT4ge1xuICAgIHJldHVybiAob2JqIGFzIFR5cGVkRXZlbnRNZXNzYWdlPGFueSwgYW55PikudHlwZSAhPT0gdW5kZWZpbmVkXG4gIH1cblxuICBjb25zdHJ1Y3RvcihcbiAgICBwdWJsaWMgdHlwZTogVCxcbiAgICBwdWJsaWMgZGF0YT86IERcbiAgKSB7fVxuXG59XG5cbmV4cG9ydCBjbGFzcyBEYXRhVmlld1NlbGVjdGlvbkluaXQ8VD4ge1xuXG4gIGNvbnN0cnVjdG9yKFxuICAgIHB1YmxpYyBvcHRpb25zOiBFbGRlckRhdGFWaWV3T3B0aW9ucyxcbiAgICBwdWJsaWMgaW5pdGlhbFNlbGVjdGlvbjogVFtdID0gW10sXG4gICAgcHVibGljIGlkRmllbGQ6IHN0cmluZyA9ICdpZCdcbiAgKSB7XG4gIH1cbn1cblxuZXhwb3J0IGNsYXNzIERhdGFWaWV3U2VsZWN0aW9uPFQ+IHtcbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHNlbGVjdGlvbjogVFtdXG4gICkge1xuICB9XG59XG5cbmV4cG9ydCBjb25zdCBEYXRhVmlld01lc3NhZ2VUeXBlVmFsdWVzID0gW1wiU0VMRUNUSU9OX0lOSVRcIiwgXCJTRUxFQ1RJT05fU0VMRUNURURcIl0gYXMgY29uc3Q7XG5leHBvcnQgdHlwZSBEYXRhVmlld01lc3NhZ2VUeXBlID0gKHR5cGVvZiBEYXRhVmlld01lc3NhZ2VUeXBlVmFsdWVzKVtudW1iZXJdO1xuZXhwb3J0IGNvbnN0IGlzRGF0YVZpZXdNZXNzYWdlVHlwZSA9ICh4OiBhbnkpOiB4IGlzIERhdGFWaWV3TWVzc2FnZVR5cGUgPT4gRGF0YVZpZXdNZXNzYWdlVHlwZVZhbHVlcy5pbmNsdWRlcyh4KTtcblxuZXhwb3J0IGNsYXNzIERhdGFWaWV3TWVzc2FnZTxULCBEID0gRGF0YVZpZXdTZWxlY3Rpb25Jbml0PFQ+IHwgRGF0YVZpZXdTZWxlY3Rpb248VD4+IGV4dGVuZHMgVHlwZWRFdmVudE1lc3NhZ2U8RGF0YVZpZXdNZXNzYWdlVHlwZSwgRD4ge1xuXG4gIHB1YmxpYyBzdGF0aWMgaW5pdGlhbFNlbGVjdGlvbjxEPihpbml0OiBEYXRhVmlld1NlbGVjdGlvbkluaXQ8RD4pOiBEYXRhVmlld01lc3NhZ2U8RD4ge1xuICAgIHJldHVybiBuZXcgRGF0YVZpZXdNZXNzYWdlKFxuICAgICAgJ1NFTEVDVElPTl9JTklUJyxcbiAgICAgIGluaXRcbiAgICApXG4gIH1cblxuICBwdWJsaWMgc3RhdGljIHNlbGVjdGlvbjxEPihzZWxlY3Rpb246IERhdGFWaWV3U2VsZWN0aW9uPEQ+KTogRGF0YVZpZXdNZXNzYWdlPEQ+IHtcbiAgICByZXR1cm4gbmV3IERhdGFWaWV3TWVzc2FnZShcbiAgICAgICdTRUxFQ1RJT05fU0VMRUNURUQnLFxuICAgICAgc2VsZWN0aW9uXG4gICAgKVxuICB9XG5cbiAgLy8gdHlwZSBwcmVkaWNhdGUgdG8gY2hlY2sgd2VhdGhlciBhbiBvYmplY3QgaXMgYSBEYXRhVmlld01lc3NhZ2UgdHlwZVxuICBwdWJsaWMgc3RhdGljIGlzT2ZUeXBlKG9iajogYW55KTogb2JqIGlzIERhdGFWaWV3TWVzc2FnZTxhbnksYW55PiB7XG4gICAgcmV0dXJuIGlzRGF0YVZpZXdNZXNzYWdlVHlwZShvYmo/LnR5cGUpO1xuICB9XG5cbiAgY29uc3RydWN0b3IoXG4gICAgcHVibGljIHR5cGU6IERhdGFWaWV3TWVzc2FnZVR5cGUsXG4gICAgcHVibGljIGRhdGE6IERcbiAgKSB7XG4gICAgc3VwZXIodHlwZSwgZGF0YSk7XG4gIH1cblxufVxuIl19