@scion/workbench 17.0.0-beta.8 → 17.0.0-beta.9

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 (61) hide show
  1. package/esm2022/lib/dialog/dialog-footer/workbench-dialog-action.directive.mjs +6 -3
  2. package/esm2022/lib/dialog/dialog-footer/workbench-dialog-footer.directive.mjs +6 -3
  3. package/esm2022/lib/dialog/dialog-header/workbench-dialog-header.directive.mjs +6 -3
  4. package/esm2022/lib/dialog/workbench-dialog.component.mjs +50 -21
  5. package/esm2022/lib/executor/single-task-executor.mjs +26 -7
  6. package/esm2022/lib/glass-pane/glass-pane.directive.mjs +12 -4
  7. package/esm2022/lib/message-box/workbench-message-box.component.mjs +13 -4
  8. package/esm2022/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.mjs +24 -11
  9. package/esm2022/lib/microfrontend-platform/initialization/workbench-host-manifest-interceptor.service.mjs +22 -10
  10. package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog-intent-handler.interceptor.mjs +8 -8
  11. package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog.component.mjs +3 -3
  12. package/esm2022/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.mjs +7 -5
  13. package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.mjs +112 -0
  14. package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.mjs +43 -0
  15. package/esm2022/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.mjs +9 -7
  16. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-capability-validator.interceptor.mjs +34 -0
  17. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-intent-handler.interceptor.mjs +88 -0
  18. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-legacy-intent-translator.interceptor.mjs +32 -0
  19. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box.component.mjs +121 -0
  20. package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-intent-handler.interceptor.mjs +6 -3
  21. package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.mjs +8 -14
  22. package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.mjs +13 -6
  23. package/esm2022/lib/microfrontend-platform/workbench-microfrontend-support.mjs +40 -11
  24. package/esm2022/lib/popup/popup.component.mjs +20 -8
  25. package/esm2022/lib/popup/popup.config.mjs +3 -1
  26. package/esm2022/lib/popup/popup.service.mjs +1 -1
  27. package/esm2022/lib/routing/router.util.mjs +8 -2
  28. package/esm2022/lib/routing/routing.model.mjs +1 -1
  29. package/esm2022/lib/routing/workbench-dialog-differ.mjs +2 -2
  30. package/esm2022/lib/routing/workbench-message-box-differ.mjs +51 -0
  31. package/esm2022/lib/routing/workbench-url-observer.service.mjs +49 -66
  32. package/esm2022/lib/routing//311/265workbench-router.service.mjs +6 -9
  33. package/esm2022/lib/view/view.component.mjs +14 -7
  34. package/esm2022/lib/workbench.component.mjs +8 -4
  35. package/esm2022/lib/workbench.constants.mjs +5 -1
  36. package/fesm2022/scion-workbench.mjs +792 -269
  37. package/fesm2022/scion-workbench.mjs.map +1 -1
  38. package/lib/dialog/workbench-dialog.component.d.ts +16 -7
  39. package/lib/executor/single-task-executor.d.ts +6 -1
  40. package/lib/glass-pane/glass-pane.directive.d.ts +21 -0
  41. package/lib/message-box/workbench-message-box.component.d.ts +3 -0
  42. package/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.d.ts +7 -1
  43. package/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.d.ts +1 -0
  44. package/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.d.ts +39 -0
  45. package/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.d.ts +25 -0
  46. package/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.d.ts +3 -2
  47. package/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-capability-validator.interceptor.d.ts +10 -0
  48. package/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-intent-handler.interceptor.d.ts +28 -0
  49. package/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-legacy-intent-translator.interceptor.d.ts +11 -0
  50. package/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box.component.d.ts +48 -0
  51. package/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.d.ts +3 -7
  52. package/lib/microfrontend-platform/workbench-microfrontend-support.d.ts +1 -1
  53. package/lib/popup/popup.component.d.ts +1 -0
  54. package/lib/routing/router.util.d.ts +4 -0
  55. package/lib/routing/workbench-message-box-differ.d.ts +25 -0
  56. package/lib/routing/workbench-url-observer.service.d.ts +7 -25
  57. package/lib/routing//311/265workbench-router.service.d.ts +3 -4
  58. package/lib/workbench.constants.d.ts +4 -0
  59. package/package.json +2 -2
  60. package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-intent-handler.service.mjs +0 -46
  61. package/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-intent-handler.service.d.ts +0 -17
@@ -8,6 +8,7 @@
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
10
  import { Directive, Input } from '@angular/core';
11
+ import { asapScheduler } from 'rxjs';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "../\u0275workbench-dialog";
13
14
  /**
@@ -31,10 +32,12 @@ export class WorkbenchDialogActionDirective {
31
32
  * Specifies where to place this action in the dialog footer. Default is `end`.
32
33
  */
33
34
  this.align = 'end';
34
- this._action = dialog.registerAction(this);
35
+ // Defer registering action to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
36
+ asapScheduler.schedule(() => this._action = dialog.registerAction(this));
35
37
  }
36
38
  ngOnDestroy() {
37
- this._action.dispose();
39
+ // Defer disposing action to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
40
+ asapScheduler.schedule(() => this._action?.dispose());
38
41
  }
39
42
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.6", ngImport: i0, type: WorkbenchDialogActionDirective, deps: [{ token: i0.TemplateRef }, { token: i1.ɵWorkbenchDialog }], target: i0.ɵɵFactoryTarget.Directive }); }
40
43
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "17.0.6", type: WorkbenchDialogActionDirective, isStandalone: true, selector: "ng-template[wbDialogAction]", inputs: { align: "align" }, ngImport: i0 }); }
@@ -45,4 +48,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.6", ngImpor
45
48
  }], ctorParameters: () => [{ type: i0.TemplateRef }, { type: i1.ɵWorkbenchDialog }], propDecorators: { align: [{
46
49
  type: Input
47
50
  }] } });
48
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1hY3Rpb24uZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1mb290ZXIvd29ya2JlbmNoLWRpYWxvZy1hY3Rpb24uZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDOzs7QUFJdkU7Ozs7Ozs7Ozs7Ozs7R0FhRztBQUVILE1BQU0sT0FBTyw4QkFBOEI7SUFVekMsWUFBNEIsUUFBMkIsRUFBRSxNQUF3QjtRQUFyRCxhQUFRLEdBQVIsUUFBUSxDQUFtQjtRQU52RDs7V0FFRztRQUVJLFVBQUssR0FBb0IsS0FBSyxDQUFDO1FBR3BDLElBQUksQ0FBQyxPQUFPLEdBQUcsTUFBTSxDQUFDLGNBQWMsQ0FBQyxJQUFJLENBQUMsQ0FBQztJQUM3QyxDQUFDO0lBRU0sV0FBVztRQUNoQixJQUFJLENBQUMsT0FBTyxDQUFDLE9BQU8sRUFBRSxDQUFDO0lBQ3pCLENBQUM7OEdBaEJVLDhCQUE4QjtrR0FBOUIsOEJBQThCOzsyRkFBOUIsOEJBQThCO2tCQUQxQyxTQUFTO21CQUFDLEVBQUMsUUFBUSxFQUFFLDZCQUE2QixFQUFFLFVBQVUsRUFBRSxJQUFJLEVBQUM7K0dBUzdELEtBQUs7c0JBRFgsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtMjAyMyBTd2lzcyBGZWRlcmFsIFJhaWx3YXlzXG4gKlxuICogVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscyBhcmUgbWFkZVxuICogYXZhaWxhYmxlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgRWNsaXBzZSBQdWJsaWMgTGljZW5zZSAyLjBcbiAqIHdoaWNoIGlzIGF2YWlsYWJsZSBhdCBodHRwczovL3d3dy5lY2xpcHNlLm9yZy9sZWdhbC9lcGwtMi4wL1xuICpcbiAqIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBFUEwtMi4wXG4gKi9cblxuaW1wb3J0IHtEaXJlY3RpdmUsIElucHV0LCBPbkRlc3Ryb3ksIFRlbXBsYXRlUmVmfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7ybVXb3JrYmVuY2hEaWFsb2d9IGZyb20gJy4uL8m1d29ya2JlbmNoLWRpYWxvZyc7XG5pbXBvcnQge0Rpc3Bvc2FibGV9IGZyb20gJy4uLy4uL2NvbW1vbi9kaXNwb3NhYmxlJztcblxuLyoqXG4gKiBVc2UgdGhpcyBkaXJlY3RpdmUgdG8gY29udHJpYnV0ZSBhbiBhY3Rpb24gdG8gdGhlIGRpYWxvZyBmb290ZXIgKG9ubHkgYXBwbGljYWJsZSBpZiBub3QgdXNpbmcgYSBjdXN0b20gZGlhbG9nIGZvb3RlcikuXG4gKlxuICogQWN0aW9ucyBhcmUgZGlzcGxheWVkIGluIHRoZSBvcmRlciBhcyBtb2RlbGVkIGluIHRoZSB0ZW1wbGF0ZSBhbmQgY2FuIGJlIHBsYWNlZCBlaXRoZXIgb24gdGhlIGxlZnQgb3IgcmlnaHQuXG4gKlxuICogVGhlIGhvc3QgZWxlbWVudCBvZiB0aGlzIG1vZGVsaW5nIGRpcmVjdGl2ZSBtdXN0IGJlIGEgPG5nLXRlbXBsYXRlPi4gVGhlIGFjdGlvbiBzaGFyZXMgdGhlIGxpZmVjeWNsZSBvZiB0aGUgaG9zdCBlbGVtZW50LlxuICpcbiAqICoqRXhhbXBsZToqKlxuICogYGBgaHRtbFxuICogPG5nLXRlbXBsYXRlIHdiRGlhbG9nQWN0aW9uPlxuICogICA8YnV0dG9uIChjbGljayk9XCJkaWFsb2cuY2xvc2UoKVwiPkNsb3NlPC9idXR0b24+XG4gKiA8L25nLXRlbXBsYXRlPlxuICogYGBgXG4gKi9cbkBEaXJlY3RpdmUoe3NlbGVjdG9yOiAnbmctdGVtcGxhdGVbd2JEaWFsb2dBY3Rpb25dJywgc3RhbmRhbG9uZTogdHJ1ZX0pXG5leHBvcnQgY2xhc3MgV29ya2JlbmNoRGlhbG9nQWN0aW9uRGlyZWN0aXZlIGltcGxlbWVudHMgT25EZXN0cm95IHtcblxuICBwcml2YXRlIF9hY3Rpb246IERpc3Bvc2FibGU7XG5cbiAgLyoqXG4gICAqIFNwZWNpZmllcyB3aGVyZSB0byBwbGFjZSB0aGlzIGFjdGlvbiBpbiB0aGUgZGlhbG9nIGZvb3Rlci4gRGVmYXVsdCBpcyBgZW5kYC5cbiAgICovXG4gIEBJbnB1dCgpXG4gIHB1YmxpYyBhbGlnbjogJ3N0YXJ0JyB8ICdlbmQnID0gJ2VuZCc7XG5cbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHRlbXBsYXRlOiBUZW1wbGF0ZVJlZjx2b2lkPiwgZGlhbG9nOiDJtVdvcmtiZW5jaERpYWxvZykge1xuICAgIHRoaXMuX2FjdGlvbiA9IGRpYWxvZy5yZWdpc3RlckFjdGlvbih0aGlzKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLl9hY3Rpb24uZGlzcG9zZSgpO1xuICB9XG59XG4iXX0=
51
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1hY3Rpb24uZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1mb290ZXIvd29ya2JlbmNoLWRpYWxvZy1hY3Rpb24uZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDO0FBR3ZFLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxNQUFNLENBQUM7OztBQUVuQzs7Ozs7Ozs7Ozs7OztHQWFHO0FBRUgsTUFBTSxPQUFPLDhCQUE4QjtJQVV6QyxZQUE0QixRQUEyQixFQUFFLE1BQXdCO1FBQXJELGFBQVEsR0FBUixRQUFRLENBQW1CO1FBTnZEOztXQUVHO1FBRUksVUFBSyxHQUFvQixLQUFLLENBQUM7UUFHcEMsbUZBQW1GO1FBQ25GLGFBQWEsQ0FBQyxRQUFRLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sR0FBRyxNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7SUFDM0UsQ0FBQztJQUVNLFdBQVc7UUFDaEIsaUZBQWlGO1FBQ2pGLGFBQWEsQ0FBQyxRQUFRLENBQUMsR0FBRyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxPQUFPLEVBQUUsQ0FBQyxDQUFDO0lBQ3hELENBQUM7OEdBbEJVLDhCQUE4QjtrR0FBOUIsOEJBQThCOzsyRkFBOUIsOEJBQThCO2tCQUQxQyxTQUFTO21CQUFDLEVBQUMsUUFBUSxFQUFFLDZCQUE2QixFQUFFLFVBQVUsRUFBRSxJQUFJLEVBQUM7K0dBUzdELEtBQUs7c0JBRFgsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtMjAyMyBTd2lzcyBGZWRlcmFsIFJhaWx3YXlzXG4gKlxuICogVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscyBhcmUgbWFkZVxuICogYXZhaWxhYmxlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgRWNsaXBzZSBQdWJsaWMgTGljZW5zZSAyLjBcbiAqIHdoaWNoIGlzIGF2YWlsYWJsZSBhdCBodHRwczovL3d3dy5lY2xpcHNlLm9yZy9sZWdhbC9lcGwtMi4wL1xuICpcbiAqIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBFUEwtMi4wXG4gKi9cblxuaW1wb3J0IHtEaXJlY3RpdmUsIElucHV0LCBPbkRlc3Ryb3ksIFRlbXBsYXRlUmVmfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7ybVXb3JrYmVuY2hEaWFsb2d9IGZyb20gJy4uL8m1d29ya2JlbmNoLWRpYWxvZyc7XG5pbXBvcnQge0Rpc3Bvc2FibGV9IGZyb20gJy4uLy4uL2NvbW1vbi9kaXNwb3NhYmxlJztcbmltcG9ydCB7YXNhcFNjaGVkdWxlcn0gZnJvbSAncnhqcyc7XG5cbi8qKlxuICogVXNlIHRoaXMgZGlyZWN0aXZlIHRvIGNvbnRyaWJ1dGUgYW4gYWN0aW9uIHRvIHRoZSBkaWFsb2cgZm9vdGVyIChvbmx5IGFwcGxpY2FibGUgaWYgbm90IHVzaW5nIGEgY3VzdG9tIGRpYWxvZyBmb290ZXIpLlxuICpcbiAqIEFjdGlvbnMgYXJlIGRpc3BsYXllZCBpbiB0aGUgb3JkZXIgYXMgbW9kZWxlZCBpbiB0aGUgdGVtcGxhdGUgYW5kIGNhbiBiZSBwbGFjZWQgZWl0aGVyIG9uIHRoZSBsZWZ0IG9yIHJpZ2h0LlxuICpcbiAqIFRoZSBob3N0IGVsZW1lbnQgb2YgdGhpcyBtb2RlbGluZyBkaXJlY3RpdmUgbXVzdCBiZSBhIDxuZy10ZW1wbGF0ZT4uIFRoZSBhY3Rpb24gc2hhcmVzIHRoZSBsaWZlY3ljbGUgb2YgdGhlIGhvc3QgZWxlbWVudC5cbiAqXG4gKiAqKkV4YW1wbGU6KipcbiAqIGBgYGh0bWxcbiAqIDxuZy10ZW1wbGF0ZSB3YkRpYWxvZ0FjdGlvbj5cbiAqICAgPGJ1dHRvbiAoY2xpY2spPVwiZGlhbG9nLmNsb3NlKClcIj5DbG9zZTwvYnV0dG9uPlxuICogPC9uZy10ZW1wbGF0ZT5cbiAqIGBgYFxuICovXG5ARGlyZWN0aXZlKHtzZWxlY3RvcjogJ25nLXRlbXBsYXRlW3diRGlhbG9nQWN0aW9uXScsIHN0YW5kYWxvbmU6IHRydWV9KVxuZXhwb3J0IGNsYXNzIFdvcmtiZW5jaERpYWxvZ0FjdGlvbkRpcmVjdGl2ZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG5cbiAgcHJpdmF0ZSBfYWN0aW9uOiBEaXNwb3NhYmxlIHwgdW5kZWZpbmVkO1xuXG4gIC8qKlxuICAgKiBTcGVjaWZpZXMgd2hlcmUgdG8gcGxhY2UgdGhpcyBhY3Rpb24gaW4gdGhlIGRpYWxvZyBmb290ZXIuIERlZmF1bHQgaXMgYGVuZGAuXG4gICAqL1xuICBASW5wdXQoKVxuICBwdWJsaWMgYWxpZ246ICdzdGFydCcgfCAnZW5kJyA9ICdlbmQnO1xuXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSB0ZW1wbGF0ZTogVGVtcGxhdGVSZWY8dm9pZD4sIGRpYWxvZzogybVXb3JrYmVuY2hEaWFsb2cpIHtcbiAgICAvLyBEZWZlciByZWdpc3RlcmluZyBhY3Rpb24gdG8gYXZvaWQgYEV4cHJlc3Npb25DaGFuZ2VkQWZ0ZXJJdEhhc0JlZW5DaGVja2VkRXJyb3JgLlxuICAgIGFzYXBTY2hlZHVsZXIuc2NoZWR1bGUoKCkgPT4gdGhpcy5fYWN0aW9uID0gZGlhbG9nLnJlZ2lzdGVyQWN0aW9uKHRoaXMpKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICAvLyBEZWZlciBkaXNwb3NpbmcgYWN0aW9uIHRvIGF2b2lkIGBFeHByZXNzaW9uQ2hhbmdlZEFmdGVySXRIYXNCZWVuQ2hlY2tlZEVycm9yYC5cbiAgICBhc2FwU2NoZWR1bGVyLnNjaGVkdWxlKCgpID0+IHRoaXMuX2FjdGlvbj8uZGlzcG9zZSgpKTtcbiAgfVxufVxuIl19
@@ -8,6 +8,7 @@
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
10
  import { booleanAttribute, Directive, Input } from '@angular/core';
11
+ import { asapScheduler } from 'rxjs';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "../\u0275workbench-dialog";
13
14
  /**
@@ -25,10 +26,12 @@ import * as i1 from "../\u0275workbench-dialog";
25
26
  export class WorkbenchDialogFooterDirective {
26
27
  constructor(template, dialog) {
27
28
  this.template = template;
28
- this._footer = dialog.registerFooter(this);
29
+ // Defer registering footer to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
30
+ asapScheduler.schedule(() => this._footer = dialog.registerFooter(this));
29
31
  }
30
32
  ngOnDestroy() {
31
- this._footer.dispose();
33
+ // Defer disposing footer to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
34
+ asapScheduler.schedule(() => this._footer?.dispose());
32
35
  }
33
36
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.6", ngImport: i0, type: WorkbenchDialogFooterDirective, deps: [{ token: i0.TemplateRef }, { token: i1.ɵWorkbenchDialog }], target: i0.ɵɵFactoryTarget.Directive }); }
34
37
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "17.0.6", type: WorkbenchDialogFooterDirective, isStandalone: true, selector: "ng-template[wbDialogFooter]", inputs: { divider: ["divider", "divider", booleanAttribute] }, ngImport: i0 }); }
@@ -40,4 +43,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.6", ngImpor
40
43
  type: Input,
41
44
  args: [{ transform: booleanAttribute }]
42
45
  }] } });
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1mb290ZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1mb290ZXIvd29ya2JlbmNoLWRpYWxvZy1mb290ZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLGdCQUFnQixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDOzs7QUFJekY7Ozs7Ozs7Ozs7O0dBV0c7QUFFSCxNQUFNLE9BQU8sOEJBQThCO0lBV3pDLFlBQTRCLFFBQTJCLEVBQUUsTUFBd0I7UUFBckQsYUFBUSxHQUFSLFFBQVEsQ0FBbUI7UUFDckQsSUFBSSxDQUFDLE9BQU8sR0FBRyxNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFFTSxXQUFXO1FBQ2hCLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDekIsQ0FBQzs4R0FqQlUsOEJBQThCO2tHQUE5Qiw4QkFBOEIseUdBUXRCLGdCQUFnQjs7MkZBUnhCLDhCQUE4QjtrQkFEMUMsU0FBUzttQkFBQyxFQUFDLFFBQVEsRUFBRSw2QkFBNkIsRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFDOytHQVU3RCxPQUFPO3NCQURiLEtBQUs7dUJBQUMsRUFBQyxTQUFTLEVBQUUsZ0JBQWdCLEVBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjMgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7Ym9vbGVhbkF0dHJpYnV0ZSwgRGlyZWN0aXZlLCBJbnB1dCwgT25EZXN0cm95LCBUZW1wbGF0ZVJlZn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge8m1V29ya2JlbmNoRGlhbG9nfSBmcm9tICcuLi/JtXdvcmtiZW5jaC1kaWFsb2cnO1xuaW1wb3J0IHtEaXNwb3NhYmxlfSBmcm9tICcuLi8uLi9jb21tb24vZGlzcG9zYWJsZSc7XG5cbi8qKlxuICogVXNlIHRoaXMgZGlyZWN0aXZlIHRvIHJlcGxhY2UgdGhlIGRlZmF1bHQgZGlhbG9nIGZvb3RlciB0aGF0IHJlbmRlcnMgYWN0aW9ucyBjb250cmlidXRlZCB2aWEgdGhlIHtAbGluayBXb3JrYmVuY2hEaWFsb2dBY3Rpb25EaXJlY3RpdmV9IGRpcmVjdGl2ZS5cbiAqXG4gKiBUaGUgaG9zdCBlbGVtZW50IG9mIHRoaXMgbW9kZWxpbmcgZGlyZWN0aXZlIG11c3QgYmUgYSA8bmctdGVtcGxhdGU+LiBUaGUgZm9vdGVyIHNoYXJlcyB0aGUgbGlmZWN5Y2xlIG9mIHRoZSBob3N0IGVsZW1lbnQuXG4gKlxuICogKipFeGFtcGxlOioqXG4gKiBgYGBodG1sXG4gKiA8bmctdGVtcGxhdGUgd2JEaWFsb2dGb290ZXI+XG4gKiAgIDxhcHAtZGlhbG9nLWZvb3Rlci8+XG4gKiA8L25nLXRlbXBsYXRlPlxuICogYGBgXG4gKi9cbkBEaXJlY3RpdmUoe3NlbGVjdG9yOiAnbmctdGVtcGxhdGVbd2JEaWFsb2dGb290ZXJdJywgc3RhbmRhbG9uZTogdHJ1ZX0pXG5leHBvcnQgY2xhc3MgV29ya2JlbmNoRGlhbG9nRm9vdGVyRGlyZWN0aXZlIGltcGxlbWVudHMgT25EZXN0cm95IHtcblxuICBwcml2YXRlIF9mb290ZXI6IERpc3Bvc2FibGU7XG5cbiAgLyoqXG4gICAqIFNwZWNpZmllcyBpZiB0byBkaXNwbGF5IGEgdmlzdWFsIHNlcGFyYXRvciBiZXR3ZWVuIHRoZSBkaWFsb2cgY29udGVudCBhbmQgdGhpcyBmb290ZXIuXG4gICAqIERlZmF1bHQgaXMgYHRydWVgLlxuICAgKi9cbiAgQElucHV0KHt0cmFuc2Zvcm06IGJvb2xlYW5BdHRyaWJ1dGV9KVxuICBwdWJsaWMgZGl2aWRlcj86IGJvb2xlYW47XG5cbiAgY29uc3RydWN0b3IocHVibGljIHJlYWRvbmx5IHRlbXBsYXRlOiBUZW1wbGF0ZVJlZjx2b2lkPiwgZGlhbG9nOiDJtVdvcmtiZW5jaERpYWxvZykge1xuICAgIHRoaXMuX2Zvb3RlciA9IGRpYWxvZy5yZWdpc3RlckZvb3Rlcih0aGlzKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICB0aGlzLl9mb290ZXIuZGlzcG9zZSgpO1xuICB9XG59XG4iXX0=
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1mb290ZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1mb290ZXIvd29ya2JlbmNoLWRpYWxvZy1mb290ZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLGdCQUFnQixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDO0FBR3pGLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxNQUFNLENBQUM7OztBQUVuQzs7Ozs7Ozs7Ozs7R0FXRztBQUVILE1BQU0sT0FBTyw4QkFBOEI7SUFXekMsWUFBNEIsUUFBMkIsRUFBRSxNQUF3QjtRQUFyRCxhQUFRLEdBQVIsUUFBUSxDQUFtQjtRQUNyRCxtRkFBbUY7UUFDbkYsYUFBYSxDQUFDLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxHQUFHLE1BQU0sQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUMzRSxDQUFDO0lBRU0sV0FBVztRQUNoQixpRkFBaUY7UUFDakYsYUFBYSxDQUFDLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7SUFDeEQsQ0FBQzs4R0FuQlUsOEJBQThCO2tHQUE5Qiw4QkFBOEIseUdBUXRCLGdCQUFnQjs7MkZBUnhCLDhCQUE4QjtrQkFEMUMsU0FBUzttQkFBQyxFQUFDLFFBQVEsRUFBRSw2QkFBNkIsRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFDOytHQVU3RCxPQUFPO3NCQURiLEtBQUs7dUJBQUMsRUFBQyxTQUFTLEVBQUUsZ0JBQWdCLEVBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjMgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7Ym9vbGVhbkF0dHJpYnV0ZSwgRGlyZWN0aXZlLCBJbnB1dCwgT25EZXN0cm95LCBUZW1wbGF0ZVJlZn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge8m1V29ya2JlbmNoRGlhbG9nfSBmcm9tICcuLi/JtXdvcmtiZW5jaC1kaWFsb2cnO1xuaW1wb3J0IHtEaXNwb3NhYmxlfSBmcm9tICcuLi8uLi9jb21tb24vZGlzcG9zYWJsZSc7XG5pbXBvcnQge2FzYXBTY2hlZHVsZXJ9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqIFVzZSB0aGlzIGRpcmVjdGl2ZSB0byByZXBsYWNlIHRoZSBkZWZhdWx0IGRpYWxvZyBmb290ZXIgdGhhdCByZW5kZXJzIGFjdGlvbnMgY29udHJpYnV0ZWQgdmlhIHRoZSB7QGxpbmsgV29ya2JlbmNoRGlhbG9nQWN0aW9uRGlyZWN0aXZlfSBkaXJlY3RpdmUuXG4gKlxuICogVGhlIGhvc3QgZWxlbWVudCBvZiB0aGlzIG1vZGVsaW5nIGRpcmVjdGl2ZSBtdXN0IGJlIGEgPG5nLXRlbXBsYXRlPi4gVGhlIGZvb3RlciBzaGFyZXMgdGhlIGxpZmVjeWNsZSBvZiB0aGUgaG9zdCBlbGVtZW50LlxuICpcbiAqICoqRXhhbXBsZToqKlxuICogYGBgaHRtbFxuICogPG5nLXRlbXBsYXRlIHdiRGlhbG9nRm9vdGVyPlxuICogICA8YXBwLWRpYWxvZy1mb290ZXIvPlxuICogPC9uZy10ZW1wbGF0ZT5cbiAqIGBgYFxuICovXG5ARGlyZWN0aXZlKHtzZWxlY3RvcjogJ25nLXRlbXBsYXRlW3diRGlhbG9nRm9vdGVyXScsIHN0YW5kYWxvbmU6IHRydWV9KVxuZXhwb3J0IGNsYXNzIFdvcmtiZW5jaERpYWxvZ0Zvb3RlckRpcmVjdGl2ZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG5cbiAgcHJpdmF0ZSBfZm9vdGVyOiBEaXNwb3NhYmxlIHwgdW5kZWZpbmVkO1xuXG4gIC8qKlxuICAgKiBTcGVjaWZpZXMgaWYgdG8gZGlzcGxheSBhIHZpc3VhbCBzZXBhcmF0b3IgYmV0d2VlbiB0aGUgZGlhbG9nIGNvbnRlbnQgYW5kIHRoaXMgZm9vdGVyLlxuICAgKiBEZWZhdWx0IGlzIGB0cnVlYC5cbiAgICovXG4gIEBJbnB1dCh7dHJhbnNmb3JtOiBib29sZWFuQXR0cmlidXRlfSlcbiAgcHVibGljIGRpdmlkZXI/OiBib29sZWFuO1xuXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSB0ZW1wbGF0ZTogVGVtcGxhdGVSZWY8dm9pZD4sIGRpYWxvZzogybVXb3JrYmVuY2hEaWFsb2cpIHtcbiAgICAvLyBEZWZlciByZWdpc3RlcmluZyBmb290ZXIgdG8gYXZvaWQgYEV4cHJlc3Npb25DaGFuZ2VkQWZ0ZXJJdEhhc0JlZW5DaGVja2VkRXJyb3JgLlxuICAgIGFzYXBTY2hlZHVsZXIuc2NoZWR1bGUoKCkgPT4gdGhpcy5fZm9vdGVyID0gZGlhbG9nLnJlZ2lzdGVyRm9vdGVyKHRoaXMpKTtcbiAgfVxuXG4gIHB1YmxpYyBuZ09uRGVzdHJveSgpOiB2b2lkIHtcbiAgICAvLyBEZWZlciBkaXNwb3NpbmcgZm9vdGVyIHRvIGF2b2lkIGBFeHByZXNzaW9uQ2hhbmdlZEFmdGVySXRIYXNCZWVuQ2hlY2tlZEVycm9yYC5cbiAgICBhc2FwU2NoZWR1bGVyLnNjaGVkdWxlKCgpID0+IHRoaXMuX2Zvb3Rlcj8uZGlzcG9zZSgpKTtcbiAgfVxufVxuIl19
@@ -8,6 +8,7 @@
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
10
  import { booleanAttribute, Directive, Input } from '@angular/core';
11
+ import { asapScheduler } from 'rxjs';
11
12
  import * as i0 from "@angular/core";
12
13
  import * as i1 from "../\u0275workbench-dialog";
13
14
  /**
@@ -25,10 +26,12 @@ import * as i1 from "../\u0275workbench-dialog";
25
26
  export class WorkbenchDialogHeaderDirective {
26
27
  constructor(template, dialog) {
27
28
  this.template = template;
28
- this._header = dialog.registerHeader(this);
29
+ // Defer registering header to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
30
+ asapScheduler.schedule(() => this._header = dialog.registerHeader(this));
29
31
  }
30
32
  ngOnDestroy() {
31
- this._header.dispose();
33
+ // Defer disposing header to avoid `ExpressionChangedAfterItHasBeenCheckedError`.
34
+ asapScheduler.schedule(() => this._header?.dispose());
32
35
  }
33
36
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.0.6", ngImport: i0, type: WorkbenchDialogHeaderDirective, deps: [{ token: i0.TemplateRef }, { token: i1.ɵWorkbenchDialog }], target: i0.ɵɵFactoryTarget.Directive }); }
34
37
  static { this.ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "16.1.0", version: "17.0.6", type: WorkbenchDialogHeaderDirective, isStandalone: true, selector: "ng-template[wbDialogHeader]", inputs: { divider: ["divider", "divider", booleanAttribute] }, ngImport: i0 }); }
@@ -40,4 +43,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.6", ngImpor
40
43
  type: Input,
41
44
  args: [{ transform: booleanAttribute }]
42
45
  }] } });
43
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1oZWFkZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1oZWFkZXIvd29ya2JlbmNoLWRpYWxvZy1oZWFkZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLGdCQUFnQixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDOzs7QUFJekY7Ozs7Ozs7Ozs7O0dBV0c7QUFFSCxNQUFNLE9BQU8sOEJBQThCO0lBV3pDLFlBQTRCLFFBQTJCLEVBQUUsTUFBd0I7UUFBckQsYUFBUSxHQUFSLFFBQVEsQ0FBbUI7UUFDckQsSUFBSSxDQUFDLE9BQU8sR0FBRyxNQUFNLENBQUMsY0FBYyxDQUFDLElBQUksQ0FBQyxDQUFDO0lBQzdDLENBQUM7SUFFTSxXQUFXO1FBQ2hCLElBQUksQ0FBQyxPQUFPLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDekIsQ0FBQzs4R0FqQlUsOEJBQThCO2tHQUE5Qiw4QkFBOEIseUdBUXRCLGdCQUFnQjs7MkZBUnhCLDhCQUE4QjtrQkFEMUMsU0FBUzttQkFBQyxFQUFDLFFBQVEsRUFBRSw2QkFBNkIsRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFDOytHQVU3RCxPQUFPO3NCQURiLEtBQUs7dUJBQUMsRUFBQyxTQUFTLEVBQUUsZ0JBQWdCLEVBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjMgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7Ym9vbGVhbkF0dHJpYnV0ZSwgRGlyZWN0aXZlLCBJbnB1dCwgT25EZXN0cm95LCBUZW1wbGF0ZVJlZn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge8m1V29ya2JlbmNoRGlhbG9nfSBmcm9tICcuLi/JtXdvcmtiZW5jaC1kaWFsb2cnO1xuaW1wb3J0IHtEaXNwb3NhYmxlfSBmcm9tICcuLi8uLi9jb21tb24vZGlzcG9zYWJsZSc7XG5cbi8qKlxuICogVXNlIHRoaXMgZGlyZWN0aXZlIHRvIHJlcGxhY2UgdGhlIGRlZmF1bHQgZGlhbG9nIGhlYWRlciB0aGF0IGRpc3BsYXlzIHRoZSB0aXRsZSBhbmQgYSBjbG9zZSBidXR0b24uXG4gKlxuICogVGhlIGhvc3QgZWxlbWVudCBvZiB0aGlzIG1vZGVsaW5nIGRpcmVjdGl2ZSBtdXN0IGJlIGEgPG5nLXRlbXBsYXRlPi4gVGhlIGhlYWRlciBzaGFyZXMgdGhlIGxpZmVjeWNsZSBvZiB0aGUgaG9zdCBlbGVtZW50LlxuICpcbiAqICoqRXhhbXBsZToqKlxuICogYGBgaHRtbFxuICogPG5nLXRlbXBsYXRlIHdiRGlhbG9nSGVhZGVyPlxuICogICA8YXBwLWRpYWxvZy1oZWFkZXIvPlxuICogPC9uZy10ZW1wbGF0ZT5cbiAqIGBgYFxuICovXG5ARGlyZWN0aXZlKHtzZWxlY3RvcjogJ25nLXRlbXBsYXRlW3diRGlhbG9nSGVhZGVyXScsIHN0YW5kYWxvbmU6IHRydWV9KVxuZXhwb3J0IGNsYXNzIFdvcmtiZW5jaERpYWxvZ0hlYWRlckRpcmVjdGl2ZSBpbXBsZW1lbnRzIE9uRGVzdHJveSB7XG5cbiAgcHJpdmF0ZSBfaGVhZGVyOiBEaXNwb3NhYmxlO1xuXG4gIC8qKlxuICAgKiBTcGVjaWZpZXMgaWYgdG8gZGlzcGxheSBhIHZpc3VhbCBzZXBhcmF0b3IgYmV0d2VlbiB0aGlzIGhlYWRlciBhbmQgdGhlIGRpYWxvZyBjb250ZW50LlxuICAgKiBEZWZhdWx0IGlzIGB0cnVlYC5cbiAgICovXG4gIEBJbnB1dCh7dHJhbnNmb3JtOiBib29sZWFuQXR0cmlidXRlfSlcbiAgcHVibGljIGRpdmlkZXI/OiBib29sZWFuO1xuXG4gIGNvbnN0cnVjdG9yKHB1YmxpYyByZWFkb25seSB0ZW1wbGF0ZTogVGVtcGxhdGVSZWY8dm9pZD4sIGRpYWxvZzogybVXb3JrYmVuY2hEaWFsb2cpIHtcbiAgICB0aGlzLl9oZWFkZXIgPSBkaWFsb2cucmVnaXN0ZXJIZWFkZXIodGhpcyk7XG4gIH1cblxuICBwdWJsaWMgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5faGVhZGVyLmRpc3Bvc2UoKTtcbiAgfVxufVxuIl19
46
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLWRpYWxvZy1oZWFkZXIuZGlyZWN0aXZlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvZGlhbG9nL2RpYWxvZy1oZWFkZXIvd29ya2JlbmNoLWRpYWxvZy1oZWFkZXIuZGlyZWN0aXZlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLGdCQUFnQixFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQXlCLE1BQU0sZUFBZSxDQUFDO0FBR3pGLE9BQU8sRUFBQyxhQUFhLEVBQUMsTUFBTSxNQUFNLENBQUM7OztBQUVuQzs7Ozs7Ozs7Ozs7R0FXRztBQUVILE1BQU0sT0FBTyw4QkFBOEI7SUFXekMsWUFBNEIsUUFBMkIsRUFBRSxNQUF3QjtRQUFyRCxhQUFRLEdBQVIsUUFBUSxDQUFtQjtRQUNyRCxtRkFBbUY7UUFDbkYsYUFBYSxDQUFDLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxHQUFHLE1BQU0sQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztJQUMzRSxDQUFDO0lBRU0sV0FBVztRQUNoQixpRkFBaUY7UUFDakYsYUFBYSxDQUFDLFFBQVEsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLE9BQU8sRUFBRSxDQUFDLENBQUM7SUFDeEQsQ0FBQzs4R0FuQlUsOEJBQThCO2tHQUE5Qiw4QkFBOEIseUdBUXRCLGdCQUFnQjs7MkZBUnhCLDhCQUE4QjtrQkFEMUMsU0FBUzttQkFBQyxFQUFDLFFBQVEsRUFBRSw2QkFBNkIsRUFBRSxVQUFVLEVBQUUsSUFBSSxFQUFDOytHQVU3RCxPQUFPO3NCQURiLEtBQUs7dUJBQUMsRUFBQyxTQUFTLEVBQUUsZ0JBQWdCLEVBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjMgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7Ym9vbGVhbkF0dHJpYnV0ZSwgRGlyZWN0aXZlLCBJbnB1dCwgT25EZXN0cm95LCBUZW1wbGF0ZVJlZn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge8m1V29ya2JlbmNoRGlhbG9nfSBmcm9tICcuLi/JtXdvcmtiZW5jaC1kaWFsb2cnO1xuaW1wb3J0IHtEaXNwb3NhYmxlfSBmcm9tICcuLi8uLi9jb21tb24vZGlzcG9zYWJsZSc7XG5pbXBvcnQge2FzYXBTY2hlZHVsZXJ9IGZyb20gJ3J4anMnO1xuXG4vKipcbiAqIFVzZSB0aGlzIGRpcmVjdGl2ZSB0byByZXBsYWNlIHRoZSBkZWZhdWx0IGRpYWxvZyBoZWFkZXIgdGhhdCBkaXNwbGF5cyB0aGUgdGl0bGUgYW5kIGEgY2xvc2UgYnV0dG9uLlxuICpcbiAqIFRoZSBob3N0IGVsZW1lbnQgb2YgdGhpcyBtb2RlbGluZyBkaXJlY3RpdmUgbXVzdCBiZSBhIDxuZy10ZW1wbGF0ZT4uIFRoZSBoZWFkZXIgc2hhcmVzIHRoZSBsaWZlY3ljbGUgb2YgdGhlIGhvc3QgZWxlbWVudC5cbiAqXG4gKiAqKkV4YW1wbGU6KipcbiAqIGBgYGh0bWxcbiAqIDxuZy10ZW1wbGF0ZSB3YkRpYWxvZ0hlYWRlcj5cbiAqICAgPGFwcC1kaWFsb2ctaGVhZGVyLz5cbiAqIDwvbmctdGVtcGxhdGU+XG4gKiBgYGBcbiAqL1xuQERpcmVjdGl2ZSh7c2VsZWN0b3I6ICduZy10ZW1wbGF0ZVt3YkRpYWxvZ0hlYWRlcl0nLCBzdGFuZGFsb25lOiB0cnVlfSlcbmV4cG9ydCBjbGFzcyBXb3JrYmVuY2hEaWFsb2dIZWFkZXJEaXJlY3RpdmUgaW1wbGVtZW50cyBPbkRlc3Ryb3kge1xuXG4gIHByaXZhdGUgX2hlYWRlcjogRGlzcG9zYWJsZSB8IHVuZGVmaW5lZDtcblxuICAvKipcbiAgICogU3BlY2lmaWVzIGlmIHRvIGRpc3BsYXkgYSB2aXN1YWwgc2VwYXJhdG9yIGJldHdlZW4gdGhpcyBoZWFkZXIgYW5kIHRoZSBkaWFsb2cgY29udGVudC5cbiAgICogRGVmYXVsdCBpcyBgdHJ1ZWAuXG4gICAqL1xuICBASW5wdXQoe3RyYW5zZm9ybTogYm9vbGVhbkF0dHJpYnV0ZX0pXG4gIHB1YmxpYyBkaXZpZGVyPzogYm9vbGVhbjtcblxuICBjb25zdHJ1Y3RvcihwdWJsaWMgcmVhZG9ubHkgdGVtcGxhdGU6IFRlbXBsYXRlUmVmPHZvaWQ+LCBkaWFsb2c6IMm1V29ya2JlbmNoRGlhbG9nKSB7XG4gICAgLy8gRGVmZXIgcmVnaXN0ZXJpbmcgaGVhZGVyIHRvIGF2b2lkIGBFeHByZXNzaW9uQ2hhbmdlZEFmdGVySXRIYXNCZWVuQ2hlY2tlZEVycm9yYC5cbiAgICBhc2FwU2NoZWR1bGVyLnNjaGVkdWxlKCgpID0+IHRoaXMuX2hlYWRlciA9IGRpYWxvZy5yZWdpc3RlckhlYWRlcih0aGlzKSk7XG4gIH1cblxuICBwdWJsaWMgbmdPbkRlc3Ryb3koKTogdm9pZCB7XG4gICAgLy8gRGVmZXIgZGlzcG9zaW5nIGhlYWRlciB0byBhdm9pZCBgRXhwcmVzc2lvbkNoYW5nZWRBZnRlckl0SGFzQmVlbkNoZWNrZWRFcnJvcmAuXG4gICAgYXNhcFNjaGVkdWxlci5zY2hlZHVsZSgoKSA9PiB0aGlzLl9oZWFkZXI/LmRpc3Bvc2UoKSk7XG4gIH1cbn1cbiJdfQ==
@@ -8,9 +8,9 @@
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
10
  import { Component, forwardRef, HostBinding, HostListener, inject, ViewChild } from '@angular/core';
11
- import { fromEvent } from 'rxjs';
11
+ import { BehaviorSubject, fromEvent } from 'rxjs';
12
12
  import { A11yModule, CdkTrapFocus } from '@angular/cdk/a11y';
13
- import { AsyncPipe, DOCUMENT, NgComponentOutlet, NgTemplateOutlet } from '@angular/common';
13
+ import { AsyncPipe, NgComponentOutlet, NgTemplateOutlet } from '@angular/common';
14
14
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
15
15
  import { ɵWorkbenchDialog } from './ɵworkbench-dialog';
16
16
  import { SciViewportComponent } from '@scion/components/viewport';
@@ -21,7 +21,9 @@ import { ResizableDirective } from './resizable.directive';
21
21
  import { SciDimensionDirective } from '@scion/components/dimension';
22
22
  import { DialogHeaderComponent } from './dialog-header/dialog-header.component';
23
23
  import { DialogFooterComponent } from './dialog-footer/dialog-footer.component';
24
- import { GLASS_PANE_BLOCKABLE, GlassPaneDirective } from '../glass-pane/glass-pane.directive';
24
+ import { GLASS_PANE_BLOCKABLE, GLASS_PANE_OPTIONS, GlassPaneDirective } from '../glass-pane/glass-pane.directive';
25
+ import { filter, map, startWith, takeUntil } from 'rxjs/operators';
26
+ import { fromMutation$ } from '@scion/toolkit/observable';
25
27
  import * as i0 from "@angular/core";
26
28
  import * as i1 from "./\u0275workbench-dialog";
27
29
  import * as i2 from "@angular/cdk/a11y";
@@ -63,41 +65,62 @@ export class WorkbenchDialogComponent {
63
65
  this.dialog = dialog;
64
66
  this._zone = _zone;
65
67
  this._destroyRef = _destroyRef;
66
- this._document = inject(DOCUMENT);
68
+ /**
69
+ * Element of the dialog that has or last had focus.
70
+ */
71
+ this._activeElement$ = new BehaviorSubject(undefined);
67
72
  this.transformTranslateX = 0;
68
73
  this.transformTranslateY = 0;
69
74
  this.setDialogOffset();
70
75
  }
71
76
  ngOnInit() {
72
77
  this.trackFocus();
78
+ this.autoFocus();
73
79
  }
74
- ngAfterViewInit() {
75
- this.focus();
80
+ setDialogOffset() {
81
+ const stackPosition = this.dialog.getPositionInDialogStack();
82
+ this.transformTranslateX = stackPosition * 10;
83
+ this.transformTranslateY = stackPosition * 10;
76
84
  }
77
85
  /**
78
- * Focuses the last focused element, if any, or the first focusable element otherwise.
86
+ * Focuses this dialog, restoring the focus to the last element that had the focus,
87
+ * or otherwise focuses the first focusable element.
79
88
  */
80
89
  focus() {
81
- if (this._activeElement) {
82
- this._activeElement.focus();
90
+ const activeElement = this._activeElement$.getValue();
91
+ if (activeElement) {
92
+ activeElement.focus();
83
93
  }
84
94
  else if (!this._cdkTrapFocus.focusTrap.focusFirstTabbableElement()) {
95
+ // Focus dialog element so that it can be closed via Escape keystroke.
85
96
  this._dialogElement.nativeElement.focus();
86
97
  }
87
98
  }
88
- setDialogOffset() {
89
- const stackPosition = this.dialog.getPositionInDialogStack();
90
- this.transformTranslateX = stackPosition * 10;
91
- this.transformTranslateY = stackPosition * 10;
92
- }
93
99
  /**
94
100
  * Tracks the focus of the dialog.
95
101
  */
96
102
  trackFocus() {
97
103
  fromEvent(this._dialogElement.nativeElement, 'focusin')
98
- .pipe(subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)), takeUntilDestroyed(this._destroyRef))
104
+ .pipe(map(event => event.target instanceof HTMLElement ? event.target : undefined),
105
+ // The dialog is focused if it has no focusable element, so the dialog can be closed via Escape.
106
+ // However, in order not to cancel the autofocus, the dialog element must not be memoized as the
107
+ // active element. Otherwise, delayed content would not be focused.
108
+ filter(element => element !== this._dialogElement.nativeElement), subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)), takeUntilDestroyed(this._destroyRef))
109
+ .subscribe(activeElement => {
110
+ this._activeElement$.next(activeElement);
111
+ });
112
+ }
113
+ /**
114
+ * Focuses the first focusable element in the dialog. Has no effect if an element in the dialog already has the focus.
115
+ *
116
+ * If no focusable element can be found, the focusing will be repeated on the next DOM change until an element has the
117
+ * focus, allowing delayed content to get focus.
118
+ */
119
+ autoFocus() {
120
+ fromMutation$(this._dialogElement.nativeElement, { subtree: true, childList: true })
121
+ .pipe(startWith(undefined), takeUntil(this._activeElement$.pipe(filter(Boolean))), takeUntilDestroyed(this._destroyRef))
99
122
  .subscribe(() => {
100
- this._activeElement = this._document.activeElement instanceof HTMLElement ? this._document.activeElement : undefined;
123
+ this.focus();
101
124
  });
102
125
  }
103
126
  onEscape(event) {
@@ -211,9 +234,15 @@ function provideEnterAnimation() {
211
234
  * Blocks this dialog when other dialog(s) overlay it.
212
235
  */
213
236
  function configureDialogGlassPane() {
214
- return {
215
- provide: GLASS_PANE_BLOCKABLE,
216
- useExisting: forwardRef(() => ɵWorkbenchDialog), // resolve {@link ɵWorkbenchDialog} via forwardRef because not defined yet, i.e., {@link ɵWorkbenchDialog} constructs {@link WorkbenchDialogComponent} in its constructor.
217
- };
237
+ return [
238
+ {
239
+ provide: GLASS_PANE_BLOCKABLE,
240
+ useExisting: forwardRef(() => ɵWorkbenchDialog), // resolve {@link ɵWorkbenchDialog} via forwardRef because not defined yet, i.e., {@link ɵWorkbenchDialog} constructs {@link WorkbenchDialogComponent} in its constructor.
241
+ },
242
+ {
243
+ provide: GLASS_PANE_OPTIONS,
244
+ useFactory: () => ({ attributes: { 'data-dialogid': inject(ɵWorkbenchDialog).id } }),
245
+ },
246
+ ];
218
247
  }
219
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workbench-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/dialog/workbench-dialog.component.ts","../../../../../../projects/scion/workbench/src/lib/dialog/workbench-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {AfterViewInit, Component, DestroyRef, ElementRef, forwardRef, HostBinding, HostListener, inject, NgZone, OnInit, Provider, ViewChild} from '@angular/core';\n\nimport {fromEvent} from 'rxjs';\nimport {A11yModule, CdkTrapFocus} from '@angular/cdk/a11y';\nimport {AsyncPipe, DOCUMENT, NgComponentOutlet, NgTemplateOutlet} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {ɵWorkbenchDialog} from './ɵworkbench-dialog';\nimport {SciViewportComponent} from '@scion/components/viewport';\nimport {animate, AnimationMetadata, style, transition, trigger} from '@angular/animations';\nimport {subscribeInside} from '@scion/toolkit/operators';\nimport {MovableDirective, WbMoveEvent} from './movable.directive';\nimport {ResizableDirective, WbResizeEvent} from './resizable.directive';\nimport {SciDimension, SciDimensionDirective} from '@scion/components/dimension';\nimport {DialogHeaderComponent} from './dialog-header/dialog-header.component';\nimport {DialogFooterComponent} from './dialog-footer/dialog-footer.component';\nimport {GLASS_PANE_BLOCKABLE, GlassPaneDirective} from '../glass-pane/glass-pane.directive';\n\n/**\n * Renders the workbench dialog.\n *\n * This component is added to a CDK overlay aligned to the modality context defined by the dialog.\n * The dialog itself is rendered in the center of its modality context.\n */\n@Component({\n  selector: 'wb-dialog',\n  templateUrl: './workbench-dialog.component.html',\n  styleUrls: ['./workbench-dialog.component.scss'],\n  standalone: true,\n  imports: [\n    NgComponentOutlet,\n    NgTemplateOutlet,\n    A11yModule,\n    AsyncPipe,\n    MovableDirective,\n    ResizableDirective,\n    SciViewportComponent,\n    SciDimensionDirective,\n    DialogHeaderComponent,\n    DialogFooterComponent,\n    GlassPaneDirective,\n  ],\n  animations: [\n    trigger('enter', provideEnterAnimation()),\n  ],\n  viewProviders: [\n    configureDialogGlassPane(),\n  ],\n})\nexport class WorkbenchDialogComponent implements OnInit, AfterViewInit {\n\n  private readonly _document = inject<Document>(DOCUMENT);\n  private _activeElement: HTMLElement | undefined;\n  private _headerHeight: string | undefined;\n\n  @ViewChild(CdkTrapFocus, {static: true})\n  private _cdkTrapFocus!: CdkTrapFocus;\n\n  @ViewChild('dialog_element', {static: true})\n  private _dialogElement!: ElementRef<HTMLElement>;\n\n  @HostBinding('style.--ɵdialog-transform-translate-x')\n  protected transformTranslateX = 0;\n\n  @HostBinding('style.--ɵdialog-transform-translate-y')\n  protected transformTranslateY = 0;\n\n  @HostBinding('style.--ɵdialog-min-height')\n  protected get minHeight(): string | undefined {\n    return this.dialog.size.minHeight || this._headerHeight;\n  }\n\n  @HostBinding('style.--ɵdialog-height')\n  protected get height(): string | undefined {\n    return this.dialog.size.height;\n  }\n\n  @HostBinding('style.--ɵdialog-max-height')\n  protected get maxHeight(): string | undefined {\n    return this.dialog.size.maxHeight;\n  }\n\n  @HostBinding('style.--ɵdialog-min-width')\n  protected get minWidth(): string | undefined {\n    return this.dialog.size.minWidth || '100px';\n  }\n\n  @HostBinding('style.--ɵdialog-width')\n  protected get width(): string | undefined {\n    return this.dialog.size.width;\n  }\n\n  @HostBinding('style.--ɵdialog-max-width')\n  protected get maxWidth(): string | undefined {\n    return this.dialog.size.maxWidth;\n  }\n\n  @HostBinding('class.justified')\n  protected get justified(): boolean {\n    return !this.dialog.padding;\n  }\n\n  @HostBinding('attr.class')\n  public get cssClasses(): string {\n    return this.dialog.cssClass;\n  }\n\n  @HostBinding('attr.data-dialogid')\n  public get id(): string {\n    return this.dialog.id;\n  }\n\n  constructor(public dialog: ɵWorkbenchDialog,\n              private _zone: NgZone,\n              private _destroyRef: DestroyRef) {\n    this.setDialogOffset();\n  }\n\n  public ngOnInit(): void {\n    this.trackFocus();\n  }\n\n  public ngAfterViewInit(): void {\n    this.focus();\n  }\n\n  /**\n   * Focuses the last focused element, if any, or the first focusable element otherwise.\n   */\n  public focus(): void {\n    if (this._activeElement) {\n      this._activeElement.focus();\n    }\n    else if (!this._cdkTrapFocus.focusTrap.focusFirstTabbableElement()) {\n      this._dialogElement.nativeElement.focus();\n    }\n  }\n\n  private setDialogOffset(): void {\n    const stackPosition = this.dialog.getPositionInDialogStack();\n    this.transformTranslateX = stackPosition * 10;\n    this.transformTranslateY = stackPosition * 10;\n  }\n\n  /**\n   * Tracks the focus of the dialog.\n   */\n  private trackFocus(): void {\n    fromEvent(this._dialogElement.nativeElement, 'focusin')\n      .pipe(\n        subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(() => {\n        this._activeElement = this._document.activeElement instanceof HTMLElement ? this._document.activeElement : undefined;\n      });\n  }\n\n  @HostListener('keydown.escape', ['$event'])\n  protected onEscape(event: Event): void {\n    if (this.dialog.closable) {\n      this.dialog.close();\n      event.stopPropagation();\n    }\n  }\n\n  protected onMove(event: WbMoveEvent): void {\n    this.transformTranslateX = event.translateX;\n    this.transformTranslateY = event.translateY;\n  }\n\n  public onResize(event: WbResizeEvent): void {\n    if (event.height !== undefined) {\n      this.dialog.size.height = `${event.height}px`;\n    }\n    if (event.width !== undefined) {\n      this.dialog.size.width = `${event.width}px`;\n    }\n    if (event.translateX !== undefined) {\n      this.transformTranslateX = event.translateX;\n    }\n    if (event.translateY !== undefined) {\n      this.transformTranslateY = event.translateY;\n    }\n  }\n\n  protected onHeaderDimensionChange(dimension: SciDimension): void {\n    this._headerHeight = `${dimension.offsetHeight}px`;\n  }\n}\n\n/**\n * Returns animation metadata to slide-in a new dialog.\n */\nfunction provideEnterAnimation(): AnimationMetadata[] {\n  return [\n    transition(':enter', [\n      style({opacity: 0, bottom: '100%', top: 'unset'}),\n      animate('.1s ease-out', style({opacity: 1, bottom: '*'})),\n    ]),\n  ];\n}\n\n/**\n * Blocks this dialog when other dialog(s) overlay it.\n */\nfunction configureDialogGlassPane(): Provider {\n  return {\n    provide: GLASS_PANE_BLOCKABLE,\n    useExisting: forwardRef(() => ɵWorkbenchDialog), // resolve {@link ɵWorkbenchDialog} via forwardRef because not defined yet, i.e., {@link ɵWorkbenchDialog} constructs {@link WorkbenchDialogComponent} in its constructor.\n  };\n}\n\n","<div class=\"dialog e2e-dialog\"\n     [class.blinking]=\"dialog.blinking$ | async\"\n     [tabindex]=\"-1\"\n     wbMovable [wbHandle]=\"header\" (wbMovableMove)=\"onMove($event)\"\n     wbResizable [wbResizableEnabled]=\"dialog.resizable\" (wbResizableResize)=\"onResize($event)\"\n     @enter\n     [@.disabled]=\"!dialog.animate\"\n     wbGlassPane\n     #dialog_element>\n  <div class=\"dialog-box e2e-dialog-box\" cdkTrapFocus>\n    <header #header\n            class=\"e2e-dialog-header\"\n            [class.divider]=\"dialog.header?.divider ?? true\"\n            sciDimension (sciDimensionChange)=\"onHeaderDimensionChange($event)\">\n      <ng-container *ngTemplateOutlet=\"dialog.header?.template ?? default_dialog_header\"/>\n    </header>\n\n    <sci-viewport class=\"content e2e-dialog-content\">\n      <ng-container *ngComponentOutlet=\"dialog.component; inputs: dialog.inputs\"/>\n    </sci-viewport>\n\n    @if (dialog.footer || dialog.actions.length) {\n      <footer class=\"e2e-dialog-footer\" [class.divider]=\"dialog.footer?.divider ?? true\">\n        <ng-container *ngTemplateOutlet=\"dialog.footer?.template ?? default_dialog_footer\"/>\n      </footer>\n    }\n  </div>\n</div>\n\n<ng-template #default_dialog_header>\n  <wb-dialog-header/>\n</ng-template>\n\n<ng-template #default_dialog_footer>\n  <wb-dialog-footer/>\n</ng-template>\n"]}
248
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workbench-dialog.component.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/dialog/workbench-dialog.component.ts","../../../../../../projects/scion/workbench/src/lib/dialog/workbench-dialog.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Component, DestroyRef, ElementRef, forwardRef, HostBinding, HostListener, inject, NgZone, OnInit, Provider, ViewChild} from '@angular/core';\nimport {BehaviorSubject, fromEvent} from 'rxjs';\nimport {A11yModule, CdkTrapFocus} from '@angular/cdk/a11y';\nimport {AsyncPipe, NgComponentOutlet, NgTemplateOutlet} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {ɵWorkbenchDialog} from './ɵworkbench-dialog';\nimport {SciViewportComponent} from '@scion/components/viewport';\nimport {animate, AnimationMetadata, style, transition, trigger} from '@angular/animations';\nimport {subscribeInside} from '@scion/toolkit/operators';\nimport {MovableDirective, WbMoveEvent} from './movable.directive';\nimport {ResizableDirective, WbResizeEvent} from './resizable.directive';\nimport {SciDimension, SciDimensionDirective} from '@scion/components/dimension';\nimport {DialogHeaderComponent} from './dialog-header/dialog-header.component';\nimport {DialogFooterComponent} from './dialog-footer/dialog-footer.component';\nimport {GLASS_PANE_BLOCKABLE, GLASS_PANE_OPTIONS, GlassPaneDirective, GlassPaneOptions} from '../glass-pane/glass-pane.directive';\nimport {filter, map, startWith, takeUntil} from 'rxjs/operators';\nimport {fromMutation$} from '@scion/toolkit/observable';\n\n/**\n * Renders the workbench dialog.\n *\n * This component is added to a CDK overlay aligned to the modality context defined by the dialog.\n * The dialog itself is rendered in the center of its modality context.\n */\n@Component({\n  selector: 'wb-dialog',\n  templateUrl: './workbench-dialog.component.html',\n  styleUrls: ['./workbench-dialog.component.scss'],\n  standalone: true,\n  imports: [\n    NgComponentOutlet,\n    NgTemplateOutlet,\n    A11yModule,\n    AsyncPipe,\n    MovableDirective,\n    ResizableDirective,\n    SciViewportComponent,\n    SciDimensionDirective,\n    DialogHeaderComponent,\n    DialogFooterComponent,\n    GlassPaneDirective,\n  ],\n  animations: [\n    trigger('enter', provideEnterAnimation()),\n  ],\n  viewProviders: [\n    configureDialogGlassPane(),\n  ],\n})\nexport class WorkbenchDialogComponent implements OnInit {\n\n  /**\n   * Element of the dialog that has or last had focus.\n   */\n  private _activeElement$ = new BehaviorSubject<HTMLElement | undefined>(undefined);\n  private _headerHeight: string | undefined;\n\n  @ViewChild(CdkTrapFocus, {static: true})\n  private _cdkTrapFocus!: CdkTrapFocus;\n\n  @ViewChild('dialog_element', {static: true})\n  private _dialogElement!: ElementRef<HTMLElement>;\n\n  @HostBinding('style.--ɵdialog-transform-translate-x')\n  protected transformTranslateX = 0;\n\n  @HostBinding('style.--ɵdialog-transform-translate-y')\n  protected transformTranslateY = 0;\n\n  @HostBinding('style.--ɵdialog-min-height')\n  protected get minHeight(): string | undefined {\n    return this.dialog.size.minHeight || this._headerHeight;\n  }\n\n  @HostBinding('style.--ɵdialog-height')\n  protected get height(): string | undefined {\n    return this.dialog.size.height;\n  }\n\n  @HostBinding('style.--ɵdialog-max-height')\n  protected get maxHeight(): string | undefined {\n    return this.dialog.size.maxHeight;\n  }\n\n  @HostBinding('style.--ɵdialog-min-width')\n  protected get minWidth(): string | undefined {\n    return this.dialog.size.minWidth || '100px';\n  }\n\n  @HostBinding('style.--ɵdialog-width')\n  protected get width(): string | undefined {\n    return this.dialog.size.width;\n  }\n\n  @HostBinding('style.--ɵdialog-max-width')\n  protected get maxWidth(): string | undefined {\n    return this.dialog.size.maxWidth;\n  }\n\n  @HostBinding('class.justified')\n  protected get justified(): boolean {\n    return !this.dialog.padding;\n  }\n\n  @HostBinding('attr.class')\n  public get cssClasses(): string {\n    return this.dialog.cssClass;\n  }\n\n  @HostBinding('attr.data-dialogid')\n  public get id(): string {\n    return this.dialog.id;\n  }\n\n  constructor(public dialog: ɵWorkbenchDialog,\n              private _zone: NgZone,\n              private _destroyRef: DestroyRef) {\n    this.setDialogOffset();\n  }\n\n  public ngOnInit(): void {\n    this.trackFocus();\n    this.autoFocus();\n  }\n\n  private setDialogOffset(): void {\n    const stackPosition = this.dialog.getPositionInDialogStack();\n    this.transformTranslateX = stackPosition * 10;\n    this.transformTranslateY = stackPosition * 10;\n  }\n\n  /**\n   * Focuses this dialog, restoring the focus to the last element that had the focus,\n   * or otherwise focuses the first focusable element.\n   */\n  public focus(): void {\n    const activeElement = this._activeElement$.getValue();\n    if (activeElement) {\n      activeElement.focus();\n    }\n    else if (!this._cdkTrapFocus.focusTrap.focusFirstTabbableElement()) {\n      // Focus dialog element so that it can be closed via Escape keystroke.\n      this._dialogElement.nativeElement.focus();\n    }\n  }\n\n  /**\n   * Tracks the focus of the dialog.\n   */\n  private trackFocus(): void {\n    fromEvent<FocusEvent>(this._dialogElement.nativeElement, 'focusin')\n      .pipe(\n        map(event => event.target instanceof HTMLElement ? event.target : undefined),\n        // The dialog is focused if it has no focusable element, so the dialog can be closed via Escape.\n        // However, in order not to cancel the autofocus, the dialog element must not be memoized as the\n        // active element. Otherwise, delayed content would not be focused.\n        filter(element => element !== this._dialogElement.nativeElement),\n        subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(activeElement => {\n        this._activeElement$.next(activeElement);\n      });\n  }\n\n  /**\n   * Focuses the first focusable element in the dialog. Has no effect if an element in the dialog already has the focus.\n   *\n   * If no focusable element can be found, the focusing will be repeated on the next DOM change until an element has the\n   * focus, allowing delayed content to get focus.\n   */\n  private autoFocus(): void {\n    fromMutation$(this._dialogElement.nativeElement, {subtree: true, childList: true})\n      .pipe(\n        startWith(undefined as void),\n        takeUntil(this._activeElement$.pipe(filter(Boolean))),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(() => {\n        this.focus();\n      });\n  }\n\n  @HostListener('keydown.escape', ['$event'])\n  protected onEscape(event: Event): void {\n    if (this.dialog.closable) {\n      this.dialog.close();\n      event.stopPropagation();\n    }\n  }\n\n  protected onMove(event: WbMoveEvent): void {\n    this.transformTranslateX = event.translateX;\n    this.transformTranslateY = event.translateY;\n  }\n\n  public onResize(event: WbResizeEvent): void {\n    if (event.height !== undefined) {\n      this.dialog.size.height = `${event.height}px`;\n    }\n    if (event.width !== undefined) {\n      this.dialog.size.width = `${event.width}px`;\n    }\n    if (event.translateX !== undefined) {\n      this.transformTranslateX = event.translateX;\n    }\n    if (event.translateY !== undefined) {\n      this.transformTranslateY = event.translateY;\n    }\n  }\n\n  protected onHeaderDimensionChange(dimension: SciDimension): void {\n    this._headerHeight = `${dimension.offsetHeight}px`;\n  }\n}\n\n/**\n * Returns animation metadata to slide-in a new dialog.\n */\nfunction provideEnterAnimation(): AnimationMetadata[] {\n  return [\n    transition(':enter', [\n      style({opacity: 0, bottom: '100%', top: 'unset'}),\n      animate('.1s ease-out', style({opacity: 1, bottom: '*'})),\n    ]),\n  ];\n}\n\n/**\n * Blocks this dialog when other dialog(s) overlay it.\n */\nfunction configureDialogGlassPane(): Provider[] {\n  return [\n    {\n      provide: GLASS_PANE_BLOCKABLE,\n      useExisting: forwardRef(() => ɵWorkbenchDialog), // resolve {@link ɵWorkbenchDialog} via forwardRef because not defined yet, i.e., {@link ɵWorkbenchDialog} constructs {@link WorkbenchDialogComponent} in its constructor.\n    },\n    {\n      provide: GLASS_PANE_OPTIONS,\n      useFactory: (): GlassPaneOptions => ({attributes: {'data-dialogid': inject(ɵWorkbenchDialog).id}}),\n    },\n  ];\n}\n","<div class=\"dialog e2e-dialog\"\n     [class.blinking]=\"dialog.blinking$ | async\"\n     [tabindex]=\"-1\"\n     wbMovable [wbHandle]=\"header\" (wbMovableMove)=\"onMove($event)\"\n     wbResizable [wbResizableEnabled]=\"dialog.resizable\" (wbResizableResize)=\"onResize($event)\"\n     @enter\n     [@.disabled]=\"!dialog.animate\"\n     wbGlassPane\n     #dialog_element>\n  <div class=\"dialog-box e2e-dialog-box\" cdkTrapFocus>\n    <header #header\n            class=\"e2e-dialog-header\"\n            [class.divider]=\"dialog.header?.divider ?? true\"\n            sciDimension (sciDimensionChange)=\"onHeaderDimensionChange($event)\">\n      <ng-container *ngTemplateOutlet=\"dialog.header?.template ?? default_dialog_header\"/>\n    </header>\n\n    <sci-viewport class=\"content e2e-dialog-content\">\n      <ng-container *ngComponentOutlet=\"dialog.component; inputs: dialog.inputs\"/>\n    </sci-viewport>\n\n    @if (dialog.footer || dialog.actions.length) {\n      <footer class=\"e2e-dialog-footer\" [class.divider]=\"dialog.footer?.divider ?? true\">\n        <ng-container *ngTemplateOutlet=\"dialog.footer?.template ?? default_dialog_footer\"/>\n      </footer>\n    }\n  </div>\n</div>\n\n<ng-template #default_dialog_header>\n  <wb-dialog-header/>\n</ng-template>\n\n<ng-template #default_dialog_footer>\n  <wb-dialog-footer/>\n</ng-template>\n"]}
@@ -1,5 +1,5 @@
1
1
  /*
2
- * Copyright (c) 2018-2022 Swiss Federal Railways
2
+ * Copyright (c) 2018-2024 Swiss Federal Railways
3
3
  *
4
4
  * This program and the accompanying materials are made
5
5
  * available under the terms of the Eclipse Public License 2.0
@@ -7,9 +7,23 @@
7
7
  *
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
- import { AsyncSubject, lastValueFrom, Subject } from 'rxjs';
10
+ import { asapScheduler, AsyncSubject, lastValueFrom, Subject } from 'rxjs';
11
11
  import { serializeExecution } from '../common/operators';
12
- import { catchError, takeUntil } from 'rxjs/operators';
12
+ import { catchError, observeOn } from 'rxjs/operators';
13
+ import { DestroyRef, inject, InjectionToken } from '@angular/core';
14
+ import { ɵDestroyRef } from '../common/ɵdestroy-ref';
15
+ import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
16
+ /**
17
+ * Serializes navigation requests to the Angular Router to prevent cancelling of parallel navigations or race conditions when modifying the currently active workbench layout.
18
+ */
19
+ export const SINGLE_NAVIGATION_EXECUTOR = new InjectionToken('SINGLE_NAVIGATION_EXECUTOR', {
20
+ providedIn: 'root',
21
+ factory: () => {
22
+ const executor = new SingleTaskExecutor();
23
+ inject(DestroyRef).onDestroy(() => executor.destroy());
24
+ return executor;
25
+ },
26
+ });
13
27
  /**
14
28
  * Allows the serial execution of tasks.
15
29
  *
@@ -19,9 +33,14 @@ import { catchError, takeUntil } from 'rxjs/operators';
19
33
  export class SingleTaskExecutor {
20
34
  constructor() {
21
35
  this._task$ = new Subject();
22
- this._destroy$ = new Subject();
36
+ this._destroyRef = new ɵDestroyRef();
23
37
  this._task$
24
- .pipe(serializeExecution(task => task.execute()), catchError((error, caught) => caught), takeUntil(this._destroy$))
38
+ .pipe(
39
+ // Schedule the task asynchronously so that it is not executed if the executor is destroyed in the same "call stack",
40
+ // happening, for example, when destroying the Testbed in unit tests. Angular destroys contexts from the bottom up,
41
+ // i.e., child contexts are destroyed before parent contexts. If a task is scheduled in a destroy lifecycle hook of
42
+ // a child context, the task would still be executed because the executor is not destroyed yet.
43
+ observeOn(asapScheduler), serializeExecution(task => task.execute()), catchError((error, caught) => caught), takeUntilDestroyed(this._destroyRef))
25
44
  .subscribe();
26
45
  }
27
46
  /**
@@ -38,7 +57,7 @@ export class SingleTaskExecutor {
38
57
  * Destroys this executor.
39
58
  */
40
59
  destroy() {
41
- this._destroy$.next();
60
+ this._destroyRef.destroy();
42
61
  }
43
62
  }
44
63
  class Task {
@@ -60,4 +79,4 @@ class Task {
60
79
  return lastValueFrom(this._done$);
61
80
  }
62
81
  }
63
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRhc2stZXhlY3V0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9leGVjdXRvci9zaW5nbGUtdGFzay1leGVjdXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxZQUFZLEVBQUUsYUFBYSxFQUFFLE9BQU8sRUFBQyxNQUFNLE1BQU0sQ0FBQztBQUMxRCxPQUFPLEVBQUMsa0JBQWtCLEVBQUMsTUFBTSxxQkFBcUIsQ0FBQztBQUN2RCxPQUFPLEVBQUMsVUFBVSxFQUFFLFNBQVMsRUFBQyxNQUFNLGdCQUFnQixDQUFDO0FBRXJEOzs7OztHQUtHO0FBQ0gsTUFBTSxPQUFPLGtCQUFrQjtJQUs3QjtRQUhRLFdBQU0sR0FBRyxJQUFJLE9BQU8sRUFBUSxDQUFDO1FBQzdCLGNBQVMsR0FBRyxJQUFJLE9BQU8sRUFBUSxDQUFDO1FBR3RDLElBQUksQ0FBQyxNQUFNO2FBQ1IsSUFBSSxDQUNILGtCQUFrQixDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDLEVBQzFDLFVBQVUsQ0FBQyxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxFQUNyQyxTQUFTLENBQUMsSUFBSSxDQUFDLFNBQVMsQ0FBQyxDQUMxQjthQUNBLFNBQVMsRUFBRSxDQUFDO0lBQ2pCLENBQUM7SUFFRDs7OztPQUlHO0lBQ0ksTUFBTSxDQUFJLElBQXNCO1FBQ3JDLE1BQU0sS0FBSyxHQUFHLElBQUksSUFBSSxDQUFJLElBQUksQ0FBQyxDQUFDO1FBQ2hDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxDQUFDO1FBQ3hCLE9BQU8sS0FBSyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQ3ZCLENBQUM7SUFFRDs7T0FFRztJQUNJLE9BQU87UUFDWixJQUFJLENBQUMsU0FBUyxDQUFDLElBQUksRUFBRSxDQUFDO0lBQ3hCLENBQUM7Q0FDRjtBQUVELE1BQU0sSUFBSTtJQUlSLFlBQW9CLEtBQXVCO1FBQXZCLFVBQUssR0FBTCxLQUFLLENBQWtCO1FBRm5DLFdBQU0sR0FBRyxJQUFJLFlBQVksRUFBSyxDQUFDO0lBR3ZDLENBQUM7SUFFTSxLQUFLLENBQUMsT0FBTztRQUNsQixJQUFJO1lBQ0YsTUFBTSxNQUFNLEdBQUcsTUFBTSxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDbEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7WUFDekIsSUFBSSxDQUFDLE1BQU0sQ0FBQyxRQUFRLEVBQUUsQ0FBQztTQUN4QjtRQUNELE9BQU8sS0FBSyxFQUFFO1lBQ1osSUFBSSxDQUFDLE1BQU0sQ0FBQyxLQUFLLENBQUMsS0FBSyxDQUFDLENBQUM7U0FDMUI7SUFDSCxDQUFDO0lBRU0sS0FBSztRQUNWLE9BQU8sYUFBYSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNwQyxDQUFDO0NBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjIgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7QXN5bmNTdWJqZWN0LCBsYXN0VmFsdWVGcm9tLCBTdWJqZWN0fSBmcm9tICdyeGpzJztcbmltcG9ydCB7c2VyaWFsaXplRXhlY3V0aW9ufSBmcm9tICcuLi9jb21tb24vb3BlcmF0b3JzJztcbmltcG9ydCB7Y2F0Y2hFcnJvciwgdGFrZVVudGlsfSBmcm9tICdyeGpzL29wZXJhdG9ycyc7XG5cbi8qKlxuICogQWxsb3dzIHRoZSBzZXJpYWwgZXhlY3V0aW9uIG9mIHRhc2tzLlxuICpcbiAqIEF0IGFueSBnaXZlbiB0aW1lLCB0aGVyZSBpcyBvbmx5IG9uZSB0YXNrIGV4ZWN1dGluZy4gV2hlbiBzdWJtaXR0aW5nIGEgdGFzayBhbmQgaWYgdGhlcmUgaXMgYSB0YXNrIGFscmVhZHkgZXhlY3V0aW5nLFxuICogdGhlIHN1Ym1pdHRlZCB0YXNrIHdpbGwgYmUgcXVldWVkIGZvciBsYXRlciBleGVjdXRpb24uXG4gKi9cbmV4cG9ydCBjbGFzcyBTaW5nbGVUYXNrRXhlY3V0b3Ige1xuXG4gIHByaXZhdGUgX3Rhc2skID0gbmV3IFN1YmplY3Q8VGFzaz4oKTtcbiAgcHJpdmF0ZSBfZGVzdHJveSQgPSBuZXcgU3ViamVjdDx2b2lkPigpO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMuX3Rhc2skXG4gICAgICAucGlwZShcbiAgICAgICAgc2VyaWFsaXplRXhlY3V0aW9uKHRhc2sgPT4gdGFzay5leGVjdXRlKCkpLFxuICAgICAgICBjYXRjaEVycm9yKChlcnJvciwgY2F1Z2h0KSA9PiBjYXVnaHQpLFxuICAgICAgICB0YWtlVW50aWwodGhpcy5fZGVzdHJveSQpLFxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIFN1Ym1pdHMgYSB0YXNrIGZvciBzZXJpYWwgZXhlY3V0aW9uLlxuICAgKlxuICAgKiBSZXR1cm5zIGEgUHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSByZXN1bHQgb2YgdGhlIHBhc3NlZCB0YXNrLlxuICAgKi9cbiAgcHVibGljIHN1Ym1pdDxUPih0YXNrOiAoKSA9PiBQcm9taXNlPFQ+KTogUHJvbWlzZTxUPiB7XG4gICAgY29uc3QgybV0YXNrID0gbmV3IFRhc2s8VD4odGFzayk7XG4gICAgdGhpcy5fdGFzayQubmV4dCjJtXRhc2spO1xuICAgIHJldHVybiDJtXRhc2suYXdhaXQoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXN0cm95cyB0aGlzIGV4ZWN1dG9yLlxuICAgKi9cbiAgcHVibGljIGRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5fZGVzdHJveSQubmV4dCgpO1xuICB9XG59XG5cbmNsYXNzIFRhc2s8VCA9IGFueT4ge1xuXG4gIHByaXZhdGUgX2RvbmUkID0gbmV3IEFzeW5jU3ViamVjdDxUPigpO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgX3dvcms6ICgpID0+IFByb21pc2U8VD4pIHtcbiAgfVxuXG4gIHB1YmxpYyBhc3luYyBleGVjdXRlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLl93b3JrKCk7XG4gICAgICB0aGlzLl9kb25lJC5uZXh0KHJlc3VsdCk7XG4gICAgICB0aGlzLl9kb25lJC5jb21wbGV0ZSgpO1xuICAgIH1cbiAgICBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIHRoaXMuX2RvbmUkLmVycm9yKGVycm9yKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgYXdhaXQoKTogUHJvbWlzZTxUPiB7XG4gICAgcmV0dXJuIGxhc3RWYWx1ZUZyb20odGhpcy5fZG9uZSQpO1xuICB9XG59XG4iXX0=
82
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRhc2stZXhlY3V0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9leGVjdXRvci9zaW5nbGUtdGFzay1leGVjdXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxhQUFhLEVBQUUsWUFBWSxFQUFFLGFBQWEsRUFBRSxPQUFPLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDekUsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFDdkQsT0FBTyxFQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQUMsVUFBVSxFQUFFLE1BQU0sRUFBRSxjQUFjLEVBQUMsTUFBTSxlQUFlLENBQUM7QUFDakUsT0FBTyxFQUFDLFdBQVcsRUFBQyxNQUFNLHdCQUF3QixDQUFDO0FBQ25ELE9BQU8sRUFBQyxrQkFBa0IsRUFBQyxNQUFNLDRCQUE0QixDQUFDO0FBRTlEOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sMEJBQTBCLEdBQUcsSUFBSSxjQUFjLENBQXFCLDRCQUE0QixFQUFFO0lBQzdHLFVBQVUsRUFBRSxNQUFNO0lBQ2xCLE9BQU8sRUFBRSxHQUFHLEVBQUU7UUFDWixNQUFNLFFBQVEsR0FBRyxJQUFJLGtCQUFrQixFQUFFLENBQUM7UUFDMUMsTUFBTSxDQUFDLFVBQVUsQ0FBQyxDQUFDLFNBQVMsQ0FBQyxHQUFHLEVBQUUsQ0FBQyxRQUFRLENBQUMsT0FBTyxFQUFFLENBQUMsQ0FBQztRQUN2RCxPQUFPLFFBQVEsQ0FBQztJQUNsQixDQUFDO0NBQ0YsQ0FBQyxDQUFDO0FBRUg7Ozs7O0dBS0c7QUFDSCxNQUFNLE9BQU8sa0JBQWtCO0lBSzdCO1FBSFEsV0FBTSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFDN0IsZ0JBQVcsR0FBRyxJQUFJLFdBQVcsRUFBRSxDQUFDO1FBR3RDLElBQUksQ0FBQyxNQUFNO2FBQ1IsSUFBSTtRQUNILHFIQUFxSDtRQUNySCxtSEFBbUg7UUFDbkgsbUhBQW1IO1FBQ25ILCtGQUErRjtRQUMvRixTQUFTLENBQUMsYUFBYSxDQUFDLEVBQ3hCLGtCQUFrQixDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsSUFBSSxDQUFDLE9BQU8sRUFBRSxDQUFDLEVBQzFDLFVBQVUsQ0FBQyxDQUFDLEtBQUssRUFBRSxNQUFNLEVBQUUsRUFBRSxDQUFDLE1BQU0sQ0FBQyxFQUNyQyxrQkFBa0IsQ0FBQyxJQUFJLENBQUMsV0FBVyxDQUFDLENBQ3JDO2FBQ0EsU0FBUyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxNQUFNLENBQUksSUFBc0I7UUFDckMsTUFBTSxLQUFLLEdBQUcsSUFBSSxJQUFJLENBQUksSUFBSSxDQUFDLENBQUM7UUFDaEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDeEIsT0FBTyxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDdkIsQ0FBQztJQUVEOztPQUVHO0lBQ0ksT0FBTztRQUNaLElBQUksQ0FBQyxXQUFXLENBQUMsT0FBTyxFQUFFLENBQUM7SUFDN0IsQ0FBQztDQUNGO0FBRUQsTUFBTSxJQUFJO0lBSVIsWUFBb0IsS0FBdUI7UUFBdkIsVUFBSyxHQUFMLEtBQUssQ0FBa0I7UUFGbkMsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFLLENBQUM7SUFHdkMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFPO1FBQ2xCLElBQUk7WUFDRixNQUFNLE1BQU0sR0FBRyxNQUFNLElBQUksQ0FBQyxLQUFLLEVBQUUsQ0FBQztZQUNsQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxNQUFNLENBQUMsQ0FBQztZQUN6QixJQUFJLENBQUMsTUFBTSxDQUFDLFFBQVEsRUFBRSxDQUFDO1NBQ3hCO1FBQ0QsT0FBTyxLQUFLLEVBQUU7WUFDWixJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQztTQUMxQjtJQUNILENBQUM7SUFFTSxLQUFLO1FBQ1YsT0FBTyxhQUFhLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3BDLENBQUM7Q0FDRiIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtMjAyNCBTd2lzcyBGZWRlcmFsIFJhaWx3YXlzXG4gKlxuICogVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscyBhcmUgbWFkZVxuICogYXZhaWxhYmxlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgRWNsaXBzZSBQdWJsaWMgTGljZW5zZSAyLjBcbiAqIHdoaWNoIGlzIGF2YWlsYWJsZSBhdCBodHRwczovL3d3dy5lY2xpcHNlLm9yZy9sZWdhbC9lcGwtMi4wL1xuICpcbiAqIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBFUEwtMi4wXG4gKi9cblxuaW1wb3J0IHthc2FwU2NoZWR1bGVyLCBBc3luY1N1YmplY3QsIGxhc3RWYWx1ZUZyb20sIFN1YmplY3R9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHtzZXJpYWxpemVFeGVjdXRpb259IGZyb20gJy4uL2NvbW1vbi9vcGVyYXRvcnMnO1xuaW1wb3J0IHtjYXRjaEVycm9yLCBvYnNlcnZlT259IGZyb20gJ3J4anMvb3BlcmF0b3JzJztcbmltcG9ydCB7RGVzdHJveVJlZiwgaW5qZWN0LCBJbmplY3Rpb25Ub2tlbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge8m1RGVzdHJveVJlZn0gZnJvbSAnLi4vY29tbW9uL8m1ZGVzdHJveS1yZWYnO1xuaW1wb3J0IHt0YWtlVW50aWxEZXN0cm95ZWR9IGZyb20gJ0Bhbmd1bGFyL2NvcmUvcnhqcy1pbnRlcm9wJztcblxuLyoqXG4gKiBTZXJpYWxpemVzIG5hdmlnYXRpb24gcmVxdWVzdHMgdG8gdGhlIEFuZ3VsYXIgUm91dGVyIHRvIHByZXZlbnQgY2FuY2VsbGluZyBvZiBwYXJhbGxlbCBuYXZpZ2F0aW9ucyBvciByYWNlIGNvbmRpdGlvbnMgd2hlbiBtb2RpZnlpbmcgdGhlIGN1cnJlbnRseSBhY3RpdmUgd29ya2JlbmNoIGxheW91dC5cbiAqL1xuZXhwb3J0IGNvbnN0IFNJTkdMRV9OQVZJR0FUSU9OX0VYRUNVVE9SID0gbmV3IEluamVjdGlvblRva2VuPFNpbmdsZVRhc2tFeGVjdXRvcj4oJ1NJTkdMRV9OQVZJR0FUSU9OX0VYRUNVVE9SJywge1xuICBwcm92aWRlZEluOiAncm9vdCcsXG4gIGZhY3Rvcnk6ICgpID0+IHtcbiAgICBjb25zdCBleGVjdXRvciA9IG5ldyBTaW5nbGVUYXNrRXhlY3V0b3IoKTtcbiAgICBpbmplY3QoRGVzdHJveVJlZikub25EZXN0cm95KCgpID0+IGV4ZWN1dG9yLmRlc3Ryb3koKSk7XG4gICAgcmV0dXJuIGV4ZWN1dG9yO1xuICB9LFxufSk7XG5cbi8qKlxuICogQWxsb3dzIHRoZSBzZXJpYWwgZXhlY3V0aW9uIG9mIHRhc2tzLlxuICpcbiAqIEF0IGFueSBnaXZlbiB0aW1lLCB0aGVyZSBpcyBvbmx5IG9uZSB0YXNrIGV4ZWN1dGluZy4gV2hlbiBzdWJtaXR0aW5nIGEgdGFzayBhbmQgaWYgdGhlcmUgaXMgYSB0YXNrIGFscmVhZHkgZXhlY3V0aW5nLFxuICogdGhlIHN1Ym1pdHRlZCB0YXNrIHdpbGwgYmUgcXVldWVkIGZvciBsYXRlciBleGVjdXRpb24uXG4gKi9cbmV4cG9ydCBjbGFzcyBTaW5nbGVUYXNrRXhlY3V0b3Ige1xuXG4gIHByaXZhdGUgX3Rhc2skID0gbmV3IFN1YmplY3Q8VGFzaz4oKTtcbiAgcHJpdmF0ZSBfZGVzdHJveVJlZiA9IG5ldyDJtURlc3Ryb3lSZWYoKTtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLl90YXNrJFxuICAgICAgLnBpcGUoXG4gICAgICAgIC8vIFNjaGVkdWxlIHRoZSB0YXNrIGFzeW5jaHJvbm91c2x5IHNvIHRoYXQgaXQgaXMgbm90IGV4ZWN1dGVkIGlmIHRoZSBleGVjdXRvciBpcyBkZXN0cm95ZWQgaW4gdGhlIHNhbWUgXCJjYWxsIHN0YWNrXCIsXG4gICAgICAgIC8vIGhhcHBlbmluZywgZm9yIGV4YW1wbGUsIHdoZW4gZGVzdHJveWluZyB0aGUgVGVzdGJlZCBpbiB1bml0IHRlc3RzLiBBbmd1bGFyIGRlc3Ryb3lzIGNvbnRleHRzIGZyb20gdGhlIGJvdHRvbSB1cCxcbiAgICAgICAgLy8gaS5lLiwgY2hpbGQgY29udGV4dHMgYXJlIGRlc3Ryb3llZCBiZWZvcmUgcGFyZW50IGNvbnRleHRzLiBJZiBhIHRhc2sgaXMgc2NoZWR1bGVkIGluIGEgZGVzdHJveSBsaWZlY3ljbGUgaG9vayBvZlxuICAgICAgICAvLyBhIGNoaWxkIGNvbnRleHQsIHRoZSB0YXNrIHdvdWxkIHN0aWxsIGJlIGV4ZWN1dGVkIGJlY2F1c2UgdGhlIGV4ZWN1dG9yIGlzIG5vdCBkZXN0cm95ZWQgeWV0LlxuICAgICAgICBvYnNlcnZlT24oYXNhcFNjaGVkdWxlciksXG4gICAgICAgIHNlcmlhbGl6ZUV4ZWN1dGlvbih0YXNrID0+IHRhc2suZXhlY3V0ZSgpKSxcbiAgICAgICAgY2F0Y2hFcnJvcigoZXJyb3IsIGNhdWdodCkgPT4gY2F1Z2h0KSxcbiAgICAgICAgdGFrZVVudGlsRGVzdHJveWVkKHRoaXMuX2Rlc3Ryb3lSZWYpLFxuICAgICAgKVxuICAgICAgLnN1YnNjcmliZSgpO1xuICB9XG5cbiAgLyoqXG4gICAqIFN1Ym1pdHMgYSB0YXNrIGZvciBzZXJpYWwgZXhlY3V0aW9uLlxuICAgKlxuICAgKiBSZXR1cm5zIGEgUHJvbWlzZSB0aGF0IHJlc29sdmVzIHRvIHRoZSByZXN1bHQgb2YgdGhlIHBhc3NlZCB0YXNrLlxuICAgKi9cbiAgcHVibGljIHN1Ym1pdDxUPih0YXNrOiAoKSA9PiBQcm9taXNlPFQ+KTogUHJvbWlzZTxUPiB7XG4gICAgY29uc3QgybV0YXNrID0gbmV3IFRhc2s8VD4odGFzayk7XG4gICAgdGhpcy5fdGFzayQubmV4dCjJtXRhc2spO1xuICAgIHJldHVybiDJtXRhc2suYXdhaXQoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBEZXN0cm95cyB0aGlzIGV4ZWN1dG9yLlxuICAgKi9cbiAgcHVibGljIGRlc3Ryb3koKTogdm9pZCB7XG4gICAgdGhpcy5fZGVzdHJveVJlZi5kZXN0cm95KCk7XG4gIH1cbn1cblxuY2xhc3MgVGFzazxUID0gYW55PiB7XG5cbiAgcHJpdmF0ZSBfZG9uZSQgPSBuZXcgQXN5bmNTdWJqZWN0PFQ+KCk7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSBfd29yazogKCkgPT4gUHJvbWlzZTxUPikge1xuICB9XG5cbiAgcHVibGljIGFzeW5jIGV4ZWN1dGUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHRoaXMuX3dvcmsoKTtcbiAgICAgIHRoaXMuX2RvbmUkLm5leHQocmVzdWx0KTtcbiAgICAgIHRoaXMuX2RvbmUkLmNvbXBsZXRlKCk7XG4gICAgfVxuICAgIGNhdGNoIChlcnJvcikge1xuICAgICAgdGhpcy5fZG9uZSQuZXJyb3IoZXJyb3IpO1xuICAgIH1cbiAgfVxuXG4gIHB1YmxpYyBhd2FpdCgpOiBQcm9taXNlPFQ+IHtcbiAgICByZXR1cm4gbGFzdFZhbHVlRnJvbSh0aGlzLl9kb25lJCk7XG4gIH1cbn1cbiJdfQ==
@@ -13,6 +13,7 @@ import { fromEvent } from 'rxjs';
13
13
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
14
14
  import { ɵDestroyRef } from '../common/ɵdestroy-ref';
15
15
  import { coerceElement } from '@angular/cdk/coercion';
16
+ import { Arrays } from '@scion/toolkit/util';
16
17
  import * as i0 from "@angular/core";
17
18
  /**
18
19
  * Prevents user interaction of the host when blocked by placing a glass pane over the host element.
@@ -20,11 +21,13 @@ import * as i0 from "@angular/core";
20
21
  * Configure this direktive via the following DI tokens. These tokens must be provided at the component level.
21
22
  * - {@link GLASS_PANE_BLOCKABLE}: Represents the object to be blocked.
22
23
  * - {@link GLASS_PANE_TARGET_ELEMENT}: Controls the HTML element to block. Defaults to the directive's host element if not set.
24
+ * - {@link GLASS_PANE_OPTIONS}: Configures the glass pane.
23
25
  */
24
26
  export class GlassPaneDirective {
25
27
  constructor(_injector) {
26
28
  this._injector = _injector;
27
29
  this._glassPane = null;
30
+ this._options = inject(GLASS_PANE_OPTIONS, { optional: true, host: true }) ?? undefined;
28
31
  this._targetElement = coerceElement(inject(GLASS_PANE_TARGET_ELEMENT, { optional: true, host: true }) ?? inject(ElementRef));
29
32
  this.ensureHostElementPositioned();
30
33
  }
@@ -39,7 +42,7 @@ export class GlassPaneDirective {
39
42
  .pipe(takeUntilDestroyed())
40
43
  .subscribe((blockedBy) => {
41
44
  this._glassPane?.dispose();
42
- this._glassPane = blockedBy ? new GlassPane(this._targetElement, blockedBy) : null;
45
+ this._glassPane = blockedBy ? new GlassPane(this._targetElement, blockedBy, this._options) : null;
43
46
  });
44
47
  }
45
48
  ensureHostElementPositioned() {
@@ -61,13 +64,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.0.6", ngImpor
61
64
  * Represents the glass pane added to the target element.
62
65
  */
63
66
  class GlassPane {
64
- constructor(targetElement, blockedBy) {
67
+ constructor(targetElement, blockedBy, options) {
65
68
  this._destroyRef = new ɵDestroyRef();
66
69
  const glassPaneElement = createElement('div', {
67
70
  parent: targetElement,
68
- cssClass: ['glasspane', 'e2e-glasspane'],
71
+ cssClass: ['glasspane', 'e2e-glasspane', ...Arrays.coerce(options?.cssClass)],
69
72
  attributes: {
70
73
  'data-owner': blockedBy.id,
74
+ ...options?.attributes,
71
75
  },
72
76
  style: {
73
77
  position: 'absolute',
@@ -109,4 +113,8 @@ export const GLASS_PANE_BLOCKABLE = new InjectionToken('GLASS_PANE_BLOCKABLE');
109
113
  * Defaults to the host element of the {@link GlassPaneDirective} if not specified.
110
114
  */
111
115
  export const GLASS_PANE_TARGET_ELEMENT = new InjectionToken('GLASS_PANE_TARGET_ELEMENT');
112
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"glass-pane.directive.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/glass-pane/glass-pane.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2024 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {AfterViewInit, Directive, ElementRef, inject, InjectionToken, Injector, OnDestroy, runInInjectionContext} from '@angular/core';\nimport {createElement, setStyle} from '../common/dom.util';\nimport {fromEvent} from 'rxjs';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {ɵDestroyRef} from '../common/ɵdestroy-ref';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {Blocking} from './blocking';\nimport {Blockable} from './blockable';\n\n/**\n * Prevents user interaction of the host when blocked by placing a glass pane over the host element.\n *\n * Configure this direktive via the following DI tokens. These tokens must be provided at the component level.\n * - {@link GLASS_PANE_BLOCKABLE}: Represents the object to be blocked.\n * - {@link GLASS_PANE_TARGET_ELEMENT}: Controls the HTML element to block. Defaults to the directive's host element if not set.\n */\n@Directive({selector: '[wbGlassPane]', standalone: true})\nexport class GlassPaneDirective implements OnDestroy, AfterViewInit {\n\n  private readonly _targetElement: HTMLElement;\n  private _glassPane: GlassPane | null = null;\n\n  constructor(private _injector: Injector) {\n    this._targetElement = coerceElement(inject(GLASS_PANE_TARGET_ELEMENT, {optional: true, host: true}) ?? inject(ElementRef));\n    this.ensureHostElementPositioned();\n  }\n\n  public ngAfterViewInit(): void {\n    // Install the glass pane after Angular has finished initializing the component's view, critical\n    // if the component is already blocked at construction time, so that the glass pane is added after\n    // the view children.\n    runInInjectionContext(this._injector, () => this.installGlassPane());\n  }\n\n  private installGlassPane(): void {\n    inject(GLASS_PANE_BLOCKABLE, {host: true}).blockedBy$\n      .pipe(takeUntilDestroyed())\n      .subscribe((blockedBy: Blocking | null) => {\n        this._glassPane?.dispose();\n        this._glassPane = blockedBy ? new GlassPane(this._targetElement, blockedBy) : null;\n      });\n  }\n\n  private ensureHostElementPositioned(): void {\n    if (getComputedStyle(this._targetElement).position === 'static') {\n      setStyle(this._targetElement, {'position': 'relative'});\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this._glassPane?.dispose();\n  }\n}\n\n/**\n * Represents the glass pane added to the target element.\n */\nclass GlassPane {\n\n  private readonly _destroyRef = new ɵDestroyRef();\n\n  constructor(targetElement: HTMLElement, blockedBy: Blocking) {\n    const glassPaneElement = createElement('div', {\n      parent: targetElement,\n      cssClass: ['glasspane', 'e2e-glasspane'],\n      attributes: {\n        'data-owner': blockedBy.id,\n      },\n      style: {\n        position: 'absolute',\n        inset: 0,\n        'pointer-events': 'auto',\n      },\n    });\n\n    // Indicate the user that this element is blocked when clicking it.\n    fromEvent<MouseEvent>(glassPaneElement, 'mousedown')\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe((event: MouseEvent) => {\n        event.preventDefault(); // to not lose focus\n        blockedBy.blink();\n      });\n\n    // Prevent the target from gaining focus via sequential keyboard navigation.\n    fromEvent(targetElement, 'focusin')\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe(() => {\n        blockedBy.focus();\n      });\n\n    // Ignore pointer events to block elements overlapping the glass pane, e.g., resize handles of a dialog.\n    const currentPointerEvents = targetElement.style.pointerEvents || null;\n    setStyle(targetElement, {'pointer-events': 'none'});\n\n    this._destroyRef.onDestroy(() => {\n      glassPaneElement.remove();\n      setStyle(targetElement, {'pointer-events': currentPointerEvents});\n    });\n  }\n\n  public dispose(): void {\n    this._destroyRef.destroy();\n  }\n}\n\n/**\n * DI token to configure the {@link GlassPaneDirective} with the [object]{@link Blockable} to block.\n */\nexport const GLASS_PANE_BLOCKABLE = new InjectionToken<Blockable>('GLASS_PANE_BLOCKABLE');\n\n/**\n * DI token to configure the {@link GlassPaneDirective} with the HTML element to block.\n * Defaults to the host element of the {@link GlassPaneDirective} if not specified.\n */\nexport const GLASS_PANE_TARGET_ELEMENT = new InjectionToken<HTMLElement | ElementRef<HTMLElement>>('GLASS_PANE_TARGET_ELEMENT');\n"]}
116
+ /**
117
+ * DI token to configure the {@link GlassPaneDirective}.
118
+ */
119
+ export const GLASS_PANE_OPTIONS = new InjectionToken('GLASS_PANE_OPTIONS');
120
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"glass-pane.directive.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/glass-pane/glass-pane.directive.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2024 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {AfterViewInit, Directive, ElementRef, inject, InjectionToken, Injector, OnDestroy, runInInjectionContext} from '@angular/core';\nimport {createElement, setStyle} from '../common/dom.util';\nimport {fromEvent} from 'rxjs';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {ɵDestroyRef} from '../common/ɵdestroy-ref';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {Blocking} from './blocking';\nimport {Blockable} from './blockable';\nimport {Arrays} from '@scion/toolkit/util';\n\n/**\n * Prevents user interaction of the host when blocked by placing a glass pane over the host element.\n *\n * Configure this direktive via the following DI tokens. These tokens must be provided at the component level.\n * - {@link GLASS_PANE_BLOCKABLE}: Represents the object to be blocked.\n * - {@link GLASS_PANE_TARGET_ELEMENT}: Controls the HTML element to block. Defaults to the directive's host element if not set.\n * - {@link GLASS_PANE_OPTIONS}: Configures the glass pane.\n */\n@Directive({selector: '[wbGlassPane]', standalone: true})\nexport class GlassPaneDirective implements OnDestroy, AfterViewInit {\n\n  private readonly _targetElement: HTMLElement;\n  private _glassPane: GlassPane | null = null;\n  private _options = inject(GLASS_PANE_OPTIONS, {optional: true, host: true}) ?? undefined;\n\n  constructor(private _injector: Injector) {\n    this._targetElement = coerceElement(inject(GLASS_PANE_TARGET_ELEMENT, {optional: true, host: true}) ?? inject(ElementRef));\n    this.ensureHostElementPositioned();\n  }\n\n  public ngAfterViewInit(): void {\n    // Install the glass pane after Angular has finished initializing the component's view, critical\n    // if the component is already blocked at construction time, so that the glass pane is added after\n    // the view children.\n    runInInjectionContext(this._injector, () => this.installGlassPane());\n  }\n\n  private installGlassPane(): void {\n    inject(GLASS_PANE_BLOCKABLE, {host: true}).blockedBy$\n      .pipe(takeUntilDestroyed())\n      .subscribe((blockedBy: Blocking | null) => {\n        this._glassPane?.dispose();\n        this._glassPane = blockedBy ? new GlassPane(this._targetElement, blockedBy, this._options) : null;\n      });\n  }\n\n  private ensureHostElementPositioned(): void {\n    if (getComputedStyle(this._targetElement).position === 'static') {\n      setStyle(this._targetElement, {'position': 'relative'});\n    }\n  }\n\n  public ngOnDestroy(): void {\n    this._glassPane?.dispose();\n  }\n}\n\n/**\n * Represents the glass pane added to the target element.\n */\nclass GlassPane {\n\n  private readonly _destroyRef = new ɵDestroyRef();\n\n  constructor(targetElement: HTMLElement, blockedBy: Blocking, options?: GlassPaneOptions) {\n    const glassPaneElement = createElement('div', {\n      parent: targetElement,\n      cssClass: ['glasspane', 'e2e-glasspane', ...Arrays.coerce(options?.cssClass)],\n      attributes: {\n        'data-owner': blockedBy.id,\n        ...options?.attributes,\n      },\n      style: {\n        position: 'absolute',\n        inset: 0,\n        'pointer-events': 'auto',\n      },\n    });\n\n    // Indicate the user that this element is blocked when clicking it.\n    fromEvent<MouseEvent>(glassPaneElement, 'mousedown')\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe((event: MouseEvent) => {\n        event.preventDefault(); // to not lose focus\n        blockedBy.blink();\n      });\n\n    // Prevent the target from gaining focus via sequential keyboard navigation.\n    fromEvent(targetElement, 'focusin')\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe(() => {\n        blockedBy.focus();\n      });\n\n    // Ignore pointer events to block elements overlapping the glass pane, e.g., resize handles of a dialog.\n    const currentPointerEvents = targetElement.style.pointerEvents || null;\n    setStyle(targetElement, {'pointer-events': 'none'});\n\n    this._destroyRef.onDestroy(() => {\n      glassPaneElement.remove();\n      setStyle(targetElement, {'pointer-events': currentPointerEvents});\n    });\n  }\n\n  public dispose(): void {\n    this._destroyRef.destroy();\n  }\n}\n\n/**\n * Configures the glass pane.\n */\nexport interface GlassPaneOptions {\n  /**\n   * Specifies CSS class(es) to add to the glass pane HTML element.\n   */\n  cssClass?: string | string[];\n  /**\n   * Specifies attributes to add to the glass pane HTML element.\n   */\n  attributes?: {[key: string]: string};\n}\n\n/**\n * DI token to configure the {@link GlassPaneDirective} with the [object]{@link Blockable} to block.\n */\nexport const GLASS_PANE_BLOCKABLE = new InjectionToken<Blockable>('GLASS_PANE_BLOCKABLE');\n\n/**\n * DI token to configure the {@link GlassPaneDirective} with the HTML element to block.\n * Defaults to the host element of the {@link GlassPaneDirective} if not specified.\n */\nexport const GLASS_PANE_TARGET_ELEMENT = new InjectionToken<HTMLElement | ElementRef<HTMLElement>>('GLASS_PANE_TARGET_ELEMENT');\n\n/**\n * DI token to configure the {@link GlassPaneDirective}.\n */\nexport const GLASS_PANE_OPTIONS = new InjectionToken<GlassPaneOptions>('GLASS_PANE_OPTIONS');\n"]}