@scion/workbench 18.0.0-beta.5 → 18.0.0-beta.6

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 (34) hide show
  1. package/esm2022/lib/executor/single-task-executor.mjs +3 -3
  2. package/esm2022/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.mjs +5 -4
  3. package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.mjs +5 -4
  4. package/esm2022/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.mjs +9 -8
  5. package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.mjs +7 -8
  6. package/esm2022/lib/part/view-context-menu/text.component.mjs +16 -12
  7. package/esm2022/lib/part/view-context-menu/view-menu.service.mjs +19 -19
  8. package/esm2022/lib/popup/popup.config.mjs +4 -13
  9. package/esm2022/lib/popup/popup.service.mjs +4 -9
  10. package/esm2022/lib/portal/wb-component-portal.mjs +4 -2
  11. package/esm2022/lib/routing//311/265workbench-router.service.mjs +11 -7
  12. package/esm2022/lib/view/view.component.mjs +22 -11
  13. package/esm2022/lib/view//311/265workbench-view.model.mjs +3 -5
  14. package/esm2022/lib/workbench-config.mjs +2 -2
  15. package/esm2022/lib/workbench.component.mjs +22 -2
  16. package/esm2022/lib/workbench.model.mjs +1 -1
  17. package/fesm2022/scion-workbench.mjs +114 -89
  18. package/fesm2022/scion-workbench.mjs.map +1 -1
  19. package/lib/executor/single-task-executor.d.ts +2 -2
  20. package/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.d.ts +2 -1
  21. package/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.d.ts +2 -1
  22. package/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.d.ts +2 -1
  23. package/lib/part/view-context-menu/text.component.d.ts +3 -4
  24. package/lib/part/view-context-menu/view-menu.service.d.ts +5 -5
  25. package/lib/popup/popup.config.d.ts +9 -10
  26. package/lib/popup/popup.service.d.ts +1 -1
  27. package/lib/portal/wb-component-portal.d.ts +18 -0
  28. package/lib/routing//311/265workbench-router.service.d.ts +4 -1
  29. package/lib/view/view.component.d.ts +13 -3
  30. package/lib/view//311/265workbench-view.model.d.ts +0 -2
  31. package/lib/workbench-config.d.ts +15 -3
  32. package/lib/workbench.component.d.ts +5 -0
  33. package/lib/workbench.model.d.ts +7 -3
  34. package/package.json +2 -2
@@ -13,9 +13,9 @@ import { catchError, observeOn } from 'rxjs/operators';
13
13
  import { InjectionToken } from '@angular/core';
14
14
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
15
15
  /**
16
- * Serializes navigation requests to the Angular Router to prevent cancelling of parallel navigations or race conditions when modifying the currently active workbench layout.
16
+ * Serializes navigation requests to the Angular Router to prevent the cancellation of previously initiated asynchronous navigations.
17
17
  */
18
- export const SINGLE_NAVIGATION_EXECUTOR = new InjectionToken('SINGLE_NAVIGATION_EXECUTOR', {
18
+ export const ANGULAR_ROUTER_MUTEX = new InjectionToken('ANGULAR_ROUTER_MUTEX', {
19
19
  providedIn: 'root',
20
20
  factory: () => new SingleTaskExecutor(),
21
21
  });
@@ -69,4 +69,4 @@ class Task {
69
69
  return lastValueFrom(this._done$);
70
70
  }
71
71
  }
72
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRhc2stZXhlY3V0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9leGVjdXRvci9zaW5nbGUtdGFzay1leGVjdXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxhQUFhLEVBQUUsWUFBWSxFQUFFLGFBQWEsRUFBRSxPQUFPLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDekUsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFDdkQsT0FBTyxFQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQzdDLE9BQU8sRUFBQyxrQkFBa0IsRUFBQyxNQUFNLDRCQUE0QixDQUFDO0FBRTlEOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sMEJBQTBCLEdBQUcsSUFBSSxjQUFjLENBQXFCLDRCQUE0QixFQUFFO0lBQzdHLFVBQVUsRUFBRSxNQUFNO0lBQ2xCLE9BQU8sRUFBRSxHQUFHLEVBQUUsQ0FBQyxJQUFJLGtCQUFrQixFQUFFO0NBQ3hDLENBQUMsQ0FBQztBQUVIOzs7Ozs7O0dBT0c7QUFDSCxNQUFNLE9BQU8sa0JBQWtCO0lBSTdCO1FBRlEsV0FBTSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFHbkMsSUFBSSxDQUFDLE1BQU07YUFDUixJQUFJO1FBQ0gscUhBQXFIO1FBQ3JILG1IQUFtSDtRQUNuSCxtSEFBbUg7UUFDbkgsK0ZBQStGO1FBQy9GLFNBQVMsQ0FBQyxhQUFhLENBQUMsRUFDeEIsa0JBQWtCLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUMsRUFDMUMsVUFBVSxDQUFDLENBQUMsS0FBSyxFQUFFLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLEVBQ3JDLGtCQUFrQixFQUFFLENBQ3JCO2FBQ0EsU0FBUyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxNQUFNLENBQUksSUFBc0I7UUFDckMsTUFBTSxLQUFLLEdBQUcsSUFBSSxJQUFJLENBQUksSUFBSSxDQUFDLENBQUM7UUFDaEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDeEIsT0FBTyxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDdkIsQ0FBQztDQUNGO0FBRUQsTUFBTSxJQUFJO0lBSVIsWUFBb0IsS0FBdUI7UUFBdkIsVUFBSyxHQUFMLEtBQUssQ0FBa0I7UUFGbkMsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFLLENBQUM7SUFHdkMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFPO1FBQ2xCLElBQUksQ0FBQztZQUNILE1BQU0sTUFBTSxHQUFHLE1BQU0sSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ2xDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1lBQ3pCLElBQUksQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDekIsQ0FBQztRQUNELE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDYixJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMzQixDQUFDO0lBQ0gsQ0FBQztJQUVNLEtBQUs7UUFDVixPQUFPLGFBQWEsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDcEMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAxOC0yMDI0IFN3aXNzIEZlZGVyYWwgUmFpbHdheXNcbiAqXG4gKiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzIGFyZSBtYWRlXG4gKiBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBFY2xpcHNlIFB1YmxpYyBMaWNlbnNlIDIuMFxuICogd2hpY2ggaXMgYXZhaWxhYmxlIGF0IGh0dHBzOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC0yLjAvXG4gKlxuICogU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEVQTC0yLjBcbiAqL1xuXG5pbXBvcnQge2FzYXBTY2hlZHVsZXIsIEFzeW5jU3ViamVjdCwgbGFzdFZhbHVlRnJvbSwgU3ViamVjdH0gZnJvbSAncnhqcyc7XG5pbXBvcnQge3NlcmlhbGl6ZUV4ZWN1dGlvbn0gZnJvbSAnLi4vY29tbW9uL29wZXJhdG9ycyc7XG5pbXBvcnQge2NhdGNoRXJyb3IsIG9ic2VydmVPbn0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHtJbmplY3Rpb25Ub2tlbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge3Rha2VVbnRpbERlc3Ryb3llZH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuXG4vKipcbiAqIFNlcmlhbGl6ZXMgbmF2aWdhdGlvbiByZXF1ZXN0cyB0byB0aGUgQW5ndWxhciBSb3V0ZXIgdG8gcHJldmVudCBjYW5jZWxsaW5nIG9mIHBhcmFsbGVsIG5hdmlnYXRpb25zIG9yIHJhY2UgY29uZGl0aW9ucyB3aGVuIG1vZGlmeWluZyB0aGUgY3VycmVudGx5IGFjdGl2ZSB3b3JrYmVuY2ggbGF5b3V0LlxuICovXG5leHBvcnQgY29uc3QgU0lOR0xFX05BVklHQVRJT05fRVhFQ1VUT1IgPSBuZXcgSW5qZWN0aW9uVG9rZW48U2luZ2xlVGFza0V4ZWN1dG9yPignU0lOR0xFX05BVklHQVRJT05fRVhFQ1VUT1InLCB7XG4gIHByb3ZpZGVkSW46ICdyb290JyxcbiAgZmFjdG9yeTogKCkgPT4gbmV3IFNpbmdsZVRhc2tFeGVjdXRvcigpLFxufSk7XG5cbi8qKlxuICogQWxsb3dzIHRoZSBzZXJpYWwgZXhlY3V0aW9uIG9mIHRhc2tzLlxuICpcbiAqIEF0IGFueSBvbmUgdGltZSwgdGhlcmUgaXMgb25seSBvbmUgdGFzayBleGVjdXRpbmcuIFdoZW4gc3VibWl0dGluZyBhIHRhc2sgYW5kIGlmIHRoZXJlIGlzIGEgdGFzayBhbHJlYWR5IGV4ZWN1dGluZyxcbiAqIHRoZSBzdWJtaXR0ZWQgdGFzayB3aWxsIGJlIHF1ZXVlZCBmb3IgbGF0ZXIgZXhlY3V0aW9uLlxuICpcbiAqIFRoaXMgZXhlY3V0b3IgbXVzdCBiZSBjb25zdHJ1Y3RlZCB3aXRoaW4gYW4gaW5qZWN0aW9uIGNvbnRleHQuIERlc3Ryb3lpbmcgdGhlIGluamVjdGlvbiBjb250ZXh0IHdpbGwgYWxzbyBkZXN0cm95IHRoZSBleGVjdXRvci5cbiAqL1xuZXhwb3J0IGNsYXNzIFNpbmdsZVRhc2tFeGVjdXRvciB7XG5cbiAgcHJpdmF0ZSBfdGFzayQgPSBuZXcgU3ViamVjdDxUYXNrPigpO1xuXG4gIGNvbnN0cnVjdG9yKCkge1xuICAgIHRoaXMuX3Rhc2skXG4gICAgICAucGlwZShcbiAgICAgICAgLy8gU2NoZWR1bGUgdGhlIHRhc2sgYXN5bmNocm9ub3VzbHkgc28gdGhhdCBpdCBpcyBub3QgZXhlY3V0ZWQgaWYgdGhlIGV4ZWN1dG9yIGlzIGRlc3Ryb3llZCBpbiB0aGUgc2FtZSBcImNhbGwgc3RhY2tcIixcbiAgICAgICAgLy8gaGFwcGVuaW5nLCBmb3IgZXhhbXBsZSwgd2hlbiBkZXN0cm95aW5nIHRoZSBUZXN0YmVkIGluIHVuaXQgdGVzdHMuIEFuZ3VsYXIgZGVzdHJveXMgY29udGV4dHMgZnJvbSB0aGUgYm90dG9tIHVwLFxuICAgICAgICAvLyBpLmUuLCBjaGlsZCBjb250ZXh0cyBhcmUgZGVzdHJveWVkIGJlZm9yZSBwYXJlbnQgY29udGV4dHMuIElmIGEgdGFzayBpcyBzY2hlZHVsZWQgaW4gYSBkZXN0cm95IGxpZmVjeWNsZSBob29rIG9mXG4gICAgICAgIC8vIGEgY2hpbGQgY29udGV4dCwgdGhlIHRhc2sgd291bGQgc3RpbGwgYmUgZXhlY3V0ZWQgYmVjYXVzZSB0aGUgZXhlY3V0b3IgaXMgbm90IGRlc3Ryb3llZCB5ZXQuXG4gICAgICAgIG9ic2VydmVPbihhc2FwU2NoZWR1bGVyKSxcbiAgICAgICAgc2VyaWFsaXplRXhlY3V0aW9uKHRhc2sgPT4gdGFzay5leGVjdXRlKCkpLFxuICAgICAgICBjYXRjaEVycm9yKChlcnJvciwgY2F1Z2h0KSA9PiBjYXVnaHQpLFxuICAgICAgICB0YWtlVW50aWxEZXN0cm95ZWQoKSxcbiAgICAgIClcbiAgICAgIC5zdWJzY3JpYmUoKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBTdWJtaXRzIGEgdGFzayBmb3Igc2VyaWFsIGV4ZWN1dGlvbi5cbiAgICpcbiAgICogUmV0dXJucyBhIFByb21pc2UgdGhhdCByZXNvbHZlcyB0byB0aGUgcmVzdWx0IG9mIHRoZSBwYXNzZWQgdGFzay5cbiAgICovXG4gIHB1YmxpYyBzdWJtaXQ8VD4odGFzazogKCkgPT4gUHJvbWlzZTxUPik6IFByb21pc2U8VD4ge1xuICAgIGNvbnN0IMm1dGFzayA9IG5ldyBUYXNrPFQ+KHRhc2spO1xuICAgIHRoaXMuX3Rhc2skLm5leHQoybV0YXNrKTtcbiAgICByZXR1cm4gybV0YXNrLmF3YWl0KCk7XG4gIH1cbn1cblxuY2xhc3MgVGFzazxUID0gYW55PiB7XG5cbiAgcHJpdmF0ZSBfZG9uZSQgPSBuZXcgQXN5bmNTdWJqZWN0PFQ+KCk7XG5cbiAgY29uc3RydWN0b3IocHJpdmF0ZSBfd29yazogKCkgPT4gUHJvbWlzZTxUPikge1xuICB9XG5cbiAgcHVibGljIGFzeW5jIGV4ZWN1dGUoKTogUHJvbWlzZTx2b2lkPiB7XG4gICAgdHJ5IHtcbiAgICAgIGNvbnN0IHJlc3VsdCA9IGF3YWl0IHRoaXMuX3dvcmsoKTtcbiAgICAgIHRoaXMuX2RvbmUkLm5leHQocmVzdWx0KTtcbiAgICAgIHRoaXMuX2RvbmUkLmNvbXBsZXRlKCk7XG4gICAgfVxuICAgIGNhdGNoIChlcnJvcikge1xuICAgICAgdGhpcy5fZG9uZSQuZXJyb3IoZXJyb3IpO1xuICAgIH1cbiAgfVxuXG4gIHB1YmxpYyBhd2FpdCgpOiBQcm9taXNlPFQ+IHtcbiAgICByZXR1cm4gbGFzdFZhbHVlRnJvbSh0aGlzLl9kb25lJCk7XG4gIH1cbn1cbiJdfQ==
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2luZ2xlLXRhc2stZXhlY3V0b3IuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9leGVjdXRvci9zaW5nbGUtdGFzay1leGVjdXRvci50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7Ozs7Ozs7R0FRRztBQUVILE9BQU8sRUFBQyxhQUFhLEVBQUUsWUFBWSxFQUFFLGFBQWEsRUFBRSxPQUFPLEVBQUMsTUFBTSxNQUFNLENBQUM7QUFDekUsT0FBTyxFQUFDLGtCQUFrQixFQUFDLE1BQU0scUJBQXFCLENBQUM7QUFDdkQsT0FBTyxFQUFDLFVBQVUsRUFBRSxTQUFTLEVBQUMsTUFBTSxnQkFBZ0IsQ0FBQztBQUNyRCxPQUFPLEVBQUMsY0FBYyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQzdDLE9BQU8sRUFBQyxrQkFBa0IsRUFBQyxNQUFNLDRCQUE0QixDQUFDO0FBRTlEOztHQUVHO0FBQ0gsTUFBTSxDQUFDLE1BQU0sb0JBQW9CLEdBQUcsSUFBSSxjQUFjLENBQXFCLHNCQUFzQixFQUFFO0lBQ2pHLFVBQVUsRUFBRSxNQUFNO0lBQ2xCLE9BQU8sRUFBRSxHQUFHLEVBQUUsQ0FBQyxJQUFJLGtCQUFrQixFQUFFO0NBQ3hDLENBQUMsQ0FBQztBQUVIOzs7Ozs7O0dBT0c7QUFDSCxNQUFNLE9BQU8sa0JBQWtCO0lBSTdCO1FBRlEsV0FBTSxHQUFHLElBQUksT0FBTyxFQUFRLENBQUM7UUFHbkMsSUFBSSxDQUFDLE1BQU07YUFDUixJQUFJO1FBQ0gscUhBQXFIO1FBQ3JILG1IQUFtSDtRQUNuSCxtSEFBbUg7UUFDbkgsK0ZBQStGO1FBQy9GLFNBQVMsQ0FBQyxhQUFhLENBQUMsRUFDeEIsa0JBQWtCLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUMsRUFDMUMsVUFBVSxDQUFDLENBQUMsS0FBSyxFQUFFLE1BQU0sRUFBRSxFQUFFLENBQUMsTUFBTSxDQUFDLEVBQ3JDLGtCQUFrQixFQUFFLENBQ3JCO2FBQ0EsU0FBUyxFQUFFLENBQUM7SUFDakIsQ0FBQztJQUVEOzs7O09BSUc7SUFDSSxNQUFNLENBQUksSUFBc0I7UUFDckMsTUFBTSxLQUFLLEdBQUcsSUFBSSxJQUFJLENBQUksSUFBSSxDQUFDLENBQUM7UUFDaEMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMsS0FBSyxDQUFDLENBQUM7UUFDeEIsT0FBTyxLQUFLLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDdkIsQ0FBQztDQUNGO0FBRUQsTUFBTSxJQUFJO0lBSVIsWUFBb0IsS0FBdUI7UUFBdkIsVUFBSyxHQUFMLEtBQUssQ0FBa0I7UUFGbkMsV0FBTSxHQUFHLElBQUksWUFBWSxFQUFLLENBQUM7SUFHdkMsQ0FBQztJQUVNLEtBQUssQ0FBQyxPQUFPO1FBQ2xCLElBQUksQ0FBQztZQUNILE1BQU0sTUFBTSxHQUFHLE1BQU0sSUFBSSxDQUFDLEtBQUssRUFBRSxDQUFDO1lBQ2xDLElBQUksQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDO1lBQ3pCLElBQUksQ0FBQyxNQUFNLENBQUMsUUFBUSxFQUFFLENBQUM7UUFDekIsQ0FBQztRQUNELE9BQU8sS0FBSyxFQUFFLENBQUM7WUFDYixJQUFJLENBQUMsTUFBTSxDQUFDLEtBQUssQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUMzQixDQUFDO0lBQ0gsQ0FBQztJQUVNLEtBQUs7UUFDVixPQUFPLGFBQWEsQ0FBQyxJQUFJLENBQUMsTUFBTSxDQUFDLENBQUM7SUFDcEMsQ0FBQztDQUNGIiwic291cmNlc0NvbnRlbnQiOlsiLypcbiAqIENvcHlyaWdodCAoYykgMjAxOC0yMDI0IFN3aXNzIEZlZGVyYWwgUmFpbHdheXNcbiAqXG4gKiBUaGlzIHByb2dyYW0gYW5kIHRoZSBhY2NvbXBhbnlpbmcgbWF0ZXJpYWxzIGFyZSBtYWRlXG4gKiBhdmFpbGFibGUgdW5kZXIgdGhlIHRlcm1zIG9mIHRoZSBFY2xpcHNlIFB1YmxpYyBMaWNlbnNlIDIuMFxuICogd2hpY2ggaXMgYXZhaWxhYmxlIGF0IGh0dHBzOi8vd3d3LmVjbGlwc2Uub3JnL2xlZ2FsL2VwbC0yLjAvXG4gKlxuICogU1BEWC1MaWNlbnNlLUlkZW50aWZpZXI6IEVQTC0yLjBcbiAqL1xuXG5pbXBvcnQge2FzYXBTY2hlZHVsZXIsIEFzeW5jU3ViamVjdCwgbGFzdFZhbHVlRnJvbSwgU3ViamVjdH0gZnJvbSAncnhqcyc7XG5pbXBvcnQge3NlcmlhbGl6ZUV4ZWN1dGlvbn0gZnJvbSAnLi4vY29tbW9uL29wZXJhdG9ycyc7XG5pbXBvcnQge2NhdGNoRXJyb3IsIG9ic2VydmVPbn0gZnJvbSAncnhqcy9vcGVyYXRvcnMnO1xuaW1wb3J0IHtJbmplY3Rpb25Ub2tlbn0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge3Rha2VVbnRpbERlc3Ryb3llZH0gZnJvbSAnQGFuZ3VsYXIvY29yZS9yeGpzLWludGVyb3AnO1xuXG4vKipcbiAqIFNlcmlhbGl6ZXMgbmF2aWdhdGlvbiByZXF1ZXN0cyB0byB0aGUgQW5ndWxhciBSb3V0ZXIgdG8gcHJldmVudCB0aGUgY2FuY2VsbGF0aW9uIG9mIHByZXZpb3VzbHkgaW5pdGlhdGVkIGFzeW5jaHJvbm91cyBuYXZpZ2F0aW9ucy5cbiAqL1xuZXhwb3J0IGNvbnN0IEFOR1VMQVJfUk9VVEVSX01VVEVYID0gbmV3IEluamVjdGlvblRva2VuPFNpbmdsZVRhc2tFeGVjdXRvcj4oJ0FOR1VMQVJfUk9VVEVSX01VVEVYJywge1xuICBwcm92aWRlZEluOiAncm9vdCcsXG4gIGZhY3Rvcnk6ICgpID0+IG5ldyBTaW5nbGVUYXNrRXhlY3V0b3IoKSxcbn0pO1xuXG4vKipcbiAqIEFsbG93cyB0aGUgc2VyaWFsIGV4ZWN1dGlvbiBvZiB0YXNrcy5cbiAqXG4gKiBBdCBhbnkgb25lIHRpbWUsIHRoZXJlIGlzIG9ubHkgb25lIHRhc2sgZXhlY3V0aW5nLiBXaGVuIHN1Ym1pdHRpbmcgYSB0YXNrIGFuZCBpZiB0aGVyZSBpcyBhIHRhc2sgYWxyZWFkeSBleGVjdXRpbmcsXG4gKiB0aGUgc3VibWl0dGVkIHRhc2sgd2lsbCBiZSBxdWV1ZWQgZm9yIGxhdGVyIGV4ZWN1dGlvbi5cbiAqXG4gKiBUaGlzIGV4ZWN1dG9yIG11c3QgYmUgY29uc3RydWN0ZWQgd2l0aGluIGFuIGluamVjdGlvbiBjb250ZXh0LiBEZXN0cm95aW5nIHRoZSBpbmplY3Rpb24gY29udGV4dCB3aWxsIGFsc28gZGVzdHJveSB0aGUgZXhlY3V0b3IuXG4gKi9cbmV4cG9ydCBjbGFzcyBTaW5nbGVUYXNrRXhlY3V0b3Ige1xuXG4gIHByaXZhdGUgX3Rhc2skID0gbmV3IFN1YmplY3Q8VGFzaz4oKTtcblxuICBjb25zdHJ1Y3RvcigpIHtcbiAgICB0aGlzLl90YXNrJFxuICAgICAgLnBpcGUoXG4gICAgICAgIC8vIFNjaGVkdWxlIHRoZSB0YXNrIGFzeW5jaHJvbm91c2x5IHNvIHRoYXQgaXQgaXMgbm90IGV4ZWN1dGVkIGlmIHRoZSBleGVjdXRvciBpcyBkZXN0cm95ZWQgaW4gdGhlIHNhbWUgXCJjYWxsIHN0YWNrXCIsXG4gICAgICAgIC8vIGhhcHBlbmluZywgZm9yIGV4YW1wbGUsIHdoZW4gZGVzdHJveWluZyB0aGUgVGVzdGJlZCBpbiB1bml0IHRlc3RzLiBBbmd1bGFyIGRlc3Ryb3lzIGNvbnRleHRzIGZyb20gdGhlIGJvdHRvbSB1cCxcbiAgICAgICAgLy8gaS5lLiwgY2hpbGQgY29udGV4dHMgYXJlIGRlc3Ryb3llZCBiZWZvcmUgcGFyZW50IGNvbnRleHRzLiBJZiBhIHRhc2sgaXMgc2NoZWR1bGVkIGluIGEgZGVzdHJveSBsaWZlY3ljbGUgaG9vayBvZlxuICAgICAgICAvLyBhIGNoaWxkIGNvbnRleHQsIHRoZSB0YXNrIHdvdWxkIHN0aWxsIGJlIGV4ZWN1dGVkIGJlY2F1c2UgdGhlIGV4ZWN1dG9yIGlzIG5vdCBkZXN0cm95ZWQgeWV0LlxuICAgICAgICBvYnNlcnZlT24oYXNhcFNjaGVkdWxlciksXG4gICAgICAgIHNlcmlhbGl6ZUV4ZWN1dGlvbih0YXNrID0+IHRhc2suZXhlY3V0ZSgpKSxcbiAgICAgICAgY2F0Y2hFcnJvcigoZXJyb3IsIGNhdWdodCkgPT4gY2F1Z2h0KSxcbiAgICAgICAgdGFrZVVudGlsRGVzdHJveWVkKCksXG4gICAgICApXG4gICAgICAuc3Vic2NyaWJlKCk7XG4gIH1cblxuICAvKipcbiAgICogU3VibWl0cyBhIHRhc2sgZm9yIHNlcmlhbCBleGVjdXRpb24uXG4gICAqXG4gICAqIFJldHVybnMgYSBQcm9taXNlIHRoYXQgcmVzb2x2ZXMgdG8gdGhlIHJlc3VsdCBvZiB0aGUgcGFzc2VkIHRhc2suXG4gICAqL1xuICBwdWJsaWMgc3VibWl0PFQ+KHRhc2s6ICgpID0+IFByb21pc2U8VD4pOiBQcm9taXNlPFQ+IHtcbiAgICBjb25zdCDJtXRhc2sgPSBuZXcgVGFzazxUPih0YXNrKTtcbiAgICB0aGlzLl90YXNrJC5uZXh0KMm1dGFzayk7XG4gICAgcmV0dXJuIMm1dGFzay5hd2FpdCgpO1xuICB9XG59XG5cbmNsYXNzIFRhc2s8VCA9IGFueT4ge1xuXG4gIHByaXZhdGUgX2RvbmUkID0gbmV3IEFzeW5jU3ViamVjdDxUPigpO1xuXG4gIGNvbnN0cnVjdG9yKHByaXZhdGUgX3dvcms6ICgpID0+IFByb21pc2U8VD4pIHtcbiAgfVxuXG4gIHB1YmxpYyBhc3luYyBleGVjdXRlKCk6IFByb21pc2U8dm9pZD4ge1xuICAgIHRyeSB7XG4gICAgICBjb25zdCByZXN1bHQgPSBhd2FpdCB0aGlzLl93b3JrKCk7XG4gICAgICB0aGlzLl9kb25lJC5uZXh0KHJlc3VsdCk7XG4gICAgICB0aGlzLl9kb25lJC5jb21wbGV0ZSgpO1xuICAgIH1cbiAgICBjYXRjaCAoZXJyb3IpIHtcbiAgICAgIHRoaXMuX2RvbmUkLmVycm9yKGVycm9yKTtcbiAgICB9XG4gIH1cblxuICBwdWJsaWMgYXdhaXQoKTogUHJvbWlzZTxUPiB7XG4gICAgcmV0dXJuIGxhc3RWYWx1ZUZyb20odGhpcy5fZG9uZSQpO1xuICB9XG59XG4iXX0=
@@ -18,7 +18,7 @@ import { DIALOG_ID_PREFIX } from '../../workbench.constants';
18
18
  import { Subject } from 'rxjs';
19
19
  import { throwError } from '../../common/throw-error.util';
20
20
  import { Microfrontends } from '../common/microfrontend.util';
21
- import { SINGLE_NAVIGATION_EXECUTOR } from '../../executor/single-task-executor';
21
+ import { ANGULAR_ROUTER_MUTEX } from '../../executor/single-task-executor';
22
22
  import { Observables } from '@scion/toolkit/util';
23
23
  import { filter, takeUntil } from 'rxjs/operators';
24
24
  import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
@@ -38,7 +38,8 @@ export class MicrofrontendHostDialogComponent {
38
38
  this._dialog = _dialog;
39
39
  this._injector = _injector;
40
40
  this._router = _router;
41
- this._singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);
41
+ /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */
42
+ this._angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);
42
43
  this.outletName = DIALOG_ID_PREFIX.concat(this._dialog.id);
43
44
  }
44
45
  ngOnInit() {
@@ -57,7 +58,7 @@ export class MicrofrontendHostDialogComponent {
57
58
  path = Microfrontends.substituteNamedParameters(path, extras?.params);
58
59
  const outletCommands = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path)) : null);
59
60
  const commands = [{ outlets: { [this.outletName]: outletCommands } }];
60
- return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
61
+ return this._angularRouterMutex.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
61
62
  }
62
63
  createOutletInjector() {
63
64
  this.outletInjector = Injector.create({
@@ -129,4 +130,4 @@ function provideWorkbenchClientDialogHandle(capability, params) {
129
130
  },
130
131
  };
131
132
  }
132
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-dialog.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.html"],"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 {Component, DestroyRef, inject, Injector, Input, OnDestroy, OnInit, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchDialog as WorkbenchClientDialog, WorkbenchDialogCapability} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {ɵWorkbenchDialog} from '../../dialog/ɵworkbench-dialog';\nimport {WorkbenchDialog} from '../../dialog/workbench-dialog';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {DIALOG_ID_PREFIX} from '../../workbench.constants';\nimport {Observable, Subject} from 'rxjs';\nimport {throwError} from '../../common/throw-error.util';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {SINGLE_NAVIGATION_EXECUTOR} from '../../executor/single-task-executor';\nimport {Observables} from '@scion/toolkit/util';\nimport {filter, takeUntil} from 'rxjs/operators';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * Navigates to the microfrontend of a given {@link WorkbenchDialogCapability} via {@link Router}.\n *\n * Unlike {@link MicrofrontendDialogComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * to allow direct integration of the content provided by the workbench host application via the Angular router.\n *\n * This component is designed to be displayed in a workbench dialog.\n */\n@Component({\n  selector: 'wb-microfrontend-host-dialog',\n  styleUrls: ['./microfrontend-host-dialog.component.scss'],\n  templateUrl: './microfrontend-host-dialog.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostDialogComponent implements OnDestroy, OnInit {\n\n  @Input({required: true})\n  public capability!: WorkbenchDialogCapability;\n\n  @Input({required: true})\n  public params!: Map<string, unknown>;\n\n  protected outletName: string;\n  protected outletInjector!: Injector;\n\n  private _singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);\n\n  constructor(private _dialog: ɵWorkbenchDialog,\n              private _injector: Injector,\n              private _router: Router) {\n    this.outletName = DIALOG_ID_PREFIX.concat(this._dialog.id);\n  }\n\n  public ngOnInit(): void {\n    this.setDialogProperties();\n    this.createOutletInjector();\n    this.navigate(this.capability.properties.path, {params: this.params}).then(success => {\n      if (!success) {\n        this._dialog.close(Error('[DialogNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the named outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras?: {params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras?.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[this.outletName]: outletCommands}}];\n    return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  private createOutletInjector(): void {\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchClientDialogHandle(this.capability, this.params),\n        // Prevent injecting the dialog handle in host dialog component.\n        {provide: WorkbenchDialog, useFactory: () => throwError(`[NullInjectorError] No provider for 'WorkbenchDialog'`)},\n      ],\n    });\n  }\n\n  private setDialogProperties(): void {\n    this._dialog.size.width = this.capability.properties.size?.width;\n    this._dialog.size.height = this.capability.properties.size?.height;\n    this._dialog.size.minWidth = this.capability.properties.size?.minWidth;\n    this._dialog.size.maxWidth = this.capability.properties.size?.maxWidth;\n    this._dialog.size.minHeight = this.capability.properties.size?.minHeight;\n    this._dialog.size.maxHeight = this.capability.properties.size?.maxHeight;\n\n    this._dialog.title = Microfrontends.substituteNamedParameters(this.capability.properties.title, this.params);\n    this._dialog.closable = this.capability.properties.closable ?? true;\n    this._dialog.resizable = this.capability.properties.resizable ?? true;\n    this._dialog.padding = this.capability.properties.padding ?? true;\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchDialog} handle to the routed component.\n */\nfunction provideWorkbenchClientDialogHandle(capability: WorkbenchDialogCapability, params: Map<string, unknown>): StaticProvider {\n  return {\n    provide: WorkbenchClientDialog,\n    useFactory: (): WorkbenchClientDialog => {\n      const dialog = inject(ɵWorkbenchDialog);\n      const propertyChange$ = new Subject<'title'>();\n\n      return new class<R = unknown> implements WorkbenchClientDialog<R> {\n        public readonly capability = capability;\n        public readonly params = params;\n\n        public setTitle(title: string | Observable<string>): void {\n          propertyChange$.next('title');\n\n          Observables.coerce(title)\n            .pipe(\n              takeUntil(propertyChange$.pipe(filter(prop => prop === 'title'))),\n              takeUntilDestroyed(dialog.injector.get(DestroyRef)),\n            )\n            .subscribe(title => dialog.title = title);\n        }\n\n        public close(result?: R | Error): void {\n          dialog.close(result);\n        }\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
133
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-dialog.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.html"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAC,SAAS,EAAE,UAAU,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAqB,qBAAqB,EAAiB,MAAM,eAAe,CAAC;AACvI,OAAO,EAAC,eAAe,IAAI,qBAAqB,EAA4B,MAAM,yBAAyB,CAAC;AAC5G,OAAO,EAAC,OAAO,EAAC,MAAM,4BAA4B,CAAC;AAEnD,OAAO,EAAS,YAAY,EAAC,MAAM,iBAAiB,CAAC;AACrD,OAAO,EAAC,gBAAgB,EAAC,MAAM,gCAAgC,CAAC;AAChE,OAAO,EAAC,eAAe,EAAC,MAAM,+BAA+B,CAAC;AAC9D,OAAO,EAAC,gBAAgB,EAAC,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAC,gBAAgB,EAAC,MAAM,2BAA2B,CAAC;AAC3D,OAAO,EAAa,OAAO,EAAC,MAAM,MAAM,CAAC;AACzC,OAAO,EAAC,UAAU,EAAC,MAAM,+BAA+B,CAAC;AACzD,OAAO,EAAC,cAAc,EAAC,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAC,oBAAoB,EAAC,MAAM,qCAAqC,CAAC;AACzE,OAAO,EAAC,WAAW,EAAC,MAAM,qBAAqB,CAAC;AAChD,OAAO,EAAC,MAAM,EAAE,SAAS,EAAC,MAAM,gBAAgB,CAAC;AACjD,OAAO,EAAC,kBAAkB,EAAC,MAAM,4BAA4B,CAAC;;;;AAE9D;;;;;;;GAOG;AAWH,MAAM,OAAO,gCAAgC;IAc3C,YAAoB,OAAyB,EACzB,SAAmB,EACnB,OAAe;QAFf,YAAO,GAAP,OAAO,CAAkB;QACzB,cAAS,GAAT,SAAS,CAAU;QACnB,YAAO,GAAP,OAAO,CAAQ;QALnC,2IAA2I;QACnI,wBAAmB,GAAG,MAAM,CAAC,oBAAoB,CAAC,CAAC;QAKzD,IAAI,CAAC,UAAU,GAAG,gBAAgB,CAAC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;IAC7D,CAAC;IAEM,QAAQ;QACb,IAAI,CAAC,mBAAmB,EAAE,CAAC;QAC3B,IAAI,CAAC,oBAAoB,EAAE,CAAC;QAC5B,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,EAAC,MAAM,EAAE,IAAI,CAAC,MAAM,EAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACnF,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,KAAK,CAAC,mGAAmG,CAAC,CAAC,CAAC;YACjI,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,QAAQ,CAAC,IAAmB,EAAE,MAAoC;QACxE,IAAI,GAAG,cAAc,CAAC,yBAAyB,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,CAAC,CAAC;QAEtE,MAAM,cAAc,GAAoB,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,cAAc,CAAC,IAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC5I,MAAM,QAAQ,GAAa,CAAC,EAAC,OAAO,EAAE,EAAC,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,cAAc,EAAC,EAAC,CAAC,CAAC;QAC5E,OAAO,IAAI,CAAC,mBAAmB,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,QAAQ,EAAE,EAAC,kBAAkB,EAAE,IAAI,EAAE,mBAAmB,EAAE,UAAU,EAAC,CAAC,CAAC,CAAC;IAC7I,CAAC;IAEO,oBAAoB;QAC1B,IAAI,CAAC,cAAc,GAAG,QAAQ,CAAC,MAAM,CAAC;YACpC,MAAM,EAAE,IAAI,CAAC,SAAS;YACtB,SAAS,EAAE,CAAC,kCAAkC,CAAC,IAAI,CAAC,UAAU,EAAE,IAAI,CAAC,MAAM,CAAC;gBAC1E,gEAAgE;gBAChE,EAAC,OAAO,EAAE,eAAe,EAAE,UAAU,EAAE,GAAG,EAAE,CAAC,UAAU,CAAC,uDAAuD,CAAC,EAAC;aAClH;SACF,CAAC,CAAC;IACL,CAAC;IAEO,mBAAmB;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,KAAK,CAAC;QACjE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,MAAM,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,MAAM,CAAC;QACnE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC;QACvE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,QAAQ,CAAC;QACvE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC;QACzE,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,SAAS,CAAC;QAEzE,IAAI,CAAC,OAAO,CAAC,KAAK,GAAG,cAAc,CAAC,yBAAyB,CAAC,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,KAAK,EAAE,IAAI,CAAC,MAAM,CAAC,CAAC;QAC7G,IAAI,CAAC,OAAO,CAAC,QAAQ,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,QAAQ,IAAI,IAAI,CAAC;QACpE,IAAI,CAAC,OAAO,CAAC,SAAS,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,SAAS,IAAI,IAAI,CAAC;QACtE,IAAI,CAAC,OAAO,CAAC,OAAO,GAAG,IAAI,CAAC,UAAU,CAAC,UAAU,CAAC,OAAO,IAAI,IAAI,CAAC;IACpE,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,iCAAiC;IAC/D,CAAC;8GAnEU,gCAAgC;kGAAhC,gCAAgC,gJC7C7C,2KAKA,qHDoCI,YAAY,2JACZ,gBAAgB;;2FAGP,gCAAgC;kBAV5C,SAAS;+BACE,8BAA8B,cAG5B,IAAI,WACP;wBACP,YAAY;wBACZ,gBAAgB;qBACjB;iIAKM,UAAU;sBADhB,KAAK;uBAAC,EAAC,QAAQ,EAAE,IAAI,EAAC;gBAIhB,MAAM;sBADZ,KAAK;uBAAC,EAAC,QAAQ,EAAE,IAAI,EAAC;;AAiEzB;;GAEG;AACH,SAAS,kCAAkC,CAAC,UAAqC,EAAE,MAA4B;IAC7G,OAAO;QACL,OAAO,EAAE,qBAAqB;QAC9B,UAAU,EAAE,GAA0B,EAAE;YACtC,MAAM,MAAM,GAAG,MAAM,CAAC,gBAAgB,CAAC,CAAC;YACxC,MAAM,eAAe,GAAG,IAAI,OAAO,EAAW,CAAC;YAE/C,OAAO,IAAI;gBAAA;oBACO,eAAU,GAAG,UAAU,CAAC;oBACxB,WAAM,GAAG,MAAM,CAAC;gBAoBlC,CAAC;gBAlBQ,QAAQ,CAAC,KAAkC;oBAChD,eAAe,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;oBAE9B,WAAW,CAAC,MAAM,CAAC,KAAK,CAAC;yBACtB,IAAI,CACH,SAAS,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,KAAK,OAAO,CAAC,CAAC,CAAC,EACjE,kBAAkB,CAAC,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,CACpD;yBACA,SAAS,CAAC,KAAK,CAAC,EAAE,CAAC,MAAM,CAAC,KAAK,GAAG,KAAK,CAAC,CAAC;gBAC9C,CAAC;gBAEM,KAAK,CAAC,MAAkB;oBAC7B,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBACvB,CAAC;gBAEM,WAAW;oBAChB,wDAAwD;gBAC1D,CAAC;aACF,CAAC;QACJ,CAAC;KACF,CAAC;AACJ,CAAC","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 {Component, DestroyRef, inject, Injector, Input, OnDestroy, OnInit, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchDialog as WorkbenchClientDialog, WorkbenchDialogCapability} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {ɵWorkbenchDialog} from '../../dialog/ɵworkbench-dialog';\nimport {WorkbenchDialog} from '../../dialog/workbench-dialog';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {DIALOG_ID_PREFIX} from '../../workbench.constants';\nimport {Observable, Subject} from 'rxjs';\nimport {throwError} from '../../common/throw-error.util';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {ANGULAR_ROUTER_MUTEX} from '../../executor/single-task-executor';\nimport {Observables} from '@scion/toolkit/util';\nimport {filter, takeUntil} from 'rxjs/operators';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\n\n/**\n * Navigates to the microfrontend of a given {@link WorkbenchDialogCapability} via {@link Router}.\n *\n * Unlike {@link MicrofrontendDialogComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * to allow direct integration of the content provided by the workbench host application via the Angular router.\n *\n * This component is designed to be displayed in a workbench dialog.\n */\n@Component({\n  selector: 'wb-microfrontend-host-dialog',\n  styleUrls: ['./microfrontend-host-dialog.component.scss'],\n  templateUrl: './microfrontend-host-dialog.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostDialogComponent implements OnDestroy, OnInit {\n\n  @Input({required: true})\n  public capability!: WorkbenchDialogCapability;\n\n  @Input({required: true})\n  public params!: Map<string, unknown>;\n\n  protected outletName: string;\n  protected outletInjector!: Injector;\n\n  /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */\n  private _angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);\n\n  constructor(private _dialog: ɵWorkbenchDialog,\n              private _injector: Injector,\n              private _router: Router) {\n    this.outletName = DIALOG_ID_PREFIX.concat(this._dialog.id);\n  }\n\n  public ngOnInit(): void {\n    this.setDialogProperties();\n    this.createOutletInjector();\n    this.navigate(this.capability.properties.path, {params: this.params}).then(success => {\n      if (!success) {\n        this._dialog.close(Error('[DialogNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the named outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras?: {params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras?.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[this.outletName]: outletCommands}}];\n    return this._angularRouterMutex.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  private createOutletInjector(): void {\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchClientDialogHandle(this.capability, this.params),\n        // Prevent injecting the dialog handle in host dialog component.\n        {provide: WorkbenchDialog, useFactory: () => throwError(`[NullInjectorError] No provider for 'WorkbenchDialog'`)},\n      ],\n    });\n  }\n\n  private setDialogProperties(): void {\n    this._dialog.size.width = this.capability.properties.size?.width;\n    this._dialog.size.height = this.capability.properties.size?.height;\n    this._dialog.size.minWidth = this.capability.properties.size?.minWidth;\n    this._dialog.size.maxWidth = this.capability.properties.size?.maxWidth;\n    this._dialog.size.minHeight = this.capability.properties.size?.minHeight;\n    this._dialog.size.maxHeight = this.capability.properties.size?.maxHeight;\n\n    this._dialog.title = Microfrontends.substituteNamedParameters(this.capability.properties.title, this.params);\n    this._dialog.closable = this.capability.properties.closable ?? true;\n    this._dialog.resizable = this.capability.properties.resizable ?? true;\n    this._dialog.padding = this.capability.properties.padding ?? true;\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchDialog} handle to the routed component.\n */\nfunction provideWorkbenchClientDialogHandle(capability: WorkbenchDialogCapability, params: Map<string, unknown>): StaticProvider {\n  return {\n    provide: WorkbenchClientDialog,\n    useFactory: (): WorkbenchClientDialog => {\n      const dialog = inject(ɵWorkbenchDialog);\n      const propertyChange$ = new Subject<'title'>();\n\n      return new class<R = unknown> implements WorkbenchClientDialog<R> {\n        public readonly capability = capability;\n        public readonly params = params;\n\n        public setTitle(title: string | Observable<string>): void {\n          propertyChange$.next('title');\n\n          Observables.coerce(title)\n            .pipe(\n              takeUntil(propertyChange$.pipe(filter(prop => prop === 'title'))),\n              takeUntilDestroyed(dialog.injector.get(DestroyRef)),\n            )\n            .subscribe(title => dialog.title = title);\n        }\n\n        public close(result?: R | Error): void {\n          dialog.close(result);\n        }\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
@@ -15,7 +15,7 @@ import { NgTemplateOutlet } from '@angular/common';
15
15
  import { MESSAGE_BOX_ID_PREFIX } from '../../workbench.constants';
16
16
  import { UUID } from '@scion/toolkit/uuid';
17
17
  import { Microfrontends } from '../common/microfrontend.util';
18
- import { SINGLE_NAVIGATION_EXECUTOR } from '../../executor/single-task-executor';
18
+ import { ANGULAR_ROUTER_MUTEX } from '../../executor/single-task-executor';
19
19
  import { setStyle } from '../../common/dom.util';
20
20
  import * as i0 from "@angular/core";
21
21
  import * as i1 from "../../dialog/\u0275workbench-dialog";
@@ -34,7 +34,8 @@ export class MicrofrontendHostMessageBoxComponent {
34
34
  this._dialog = _dialog;
35
35
  this._injector = _injector;
36
36
  this._router = _router;
37
- this._singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);
37
+ /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */
38
+ this._angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);
38
39
  this.outletName = MESSAGE_BOX_ID_PREFIX.concat(UUID.randomUUID());
39
40
  }
40
41
  ngOnInit() {
@@ -53,7 +54,7 @@ export class MicrofrontendHostMessageBoxComponent {
53
54
  path = Microfrontends.substituteNamedParameters(path, extras?.params);
54
55
  const outletCommands = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path)) : null);
55
56
  const commands = [{ outlets: { [this.outletName]: outletCommands } }];
56
- return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
57
+ return this._angularRouterMutex.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
57
58
  }
58
59
  createOutletInjector() {
59
60
  this.outletInjector = Injector.create({
@@ -109,4 +110,4 @@ export function provideWorkbenchClientMessageBoxHandle(capability, params) {
109
110
  },
110
111
  };
111
112
  }
112
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-message-box.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.html"],"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 {Component, ElementRef, inject, Injector, Input, OnDestroy, OnInit, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchMessageBox, WorkbenchMessageBoxCapability} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {MESSAGE_BOX_ID_PREFIX} from '../../workbench.constants';\nimport {UUID} from '@scion/toolkit/uuid';\nimport {ɵWorkbenchDialog} from '../../dialog/ɵworkbench-dialog';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {SINGLE_NAVIGATION_EXECUTOR} from '../../executor/single-task-executor';\nimport {setStyle} from '../../common/dom.util';\n\n/**\n * Navigates to the microfrontend of a given {@link WorkbenchMessageBoxCapability} via {@link Router}.\n *\n * Unlike {@link MicrofrontendMessageBoxComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * to allow direct integration of the content provided by the workbench host application via the Angular router.\n *\n * This component is designed to be displayed in a workbench message box.\n */\n@Component({\n  selector: 'wb-microfrontend-host-message-box',\n  styleUrls: ['./microfrontend-host-message-box.component.scss'],\n  templateUrl: './microfrontend-host-message-box.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostMessageBoxComponent implements OnDestroy, OnInit {\n\n  @Input({required: true})\n  public capability!: WorkbenchMessageBoxCapability;\n\n  @Input({required: true})\n  public params!: Map<string, unknown>;\n\n  protected outletName: string;\n  protected outletInjector!: Injector;\n\n  private _singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);\n\n  constructor(private _host: ElementRef<HTMLElement>,\n              private _dialog: ɵWorkbenchDialog,\n              private _injector: Injector,\n              private _router: Router) {\n    this.outletName = MESSAGE_BOX_ID_PREFIX.concat(UUID.randomUUID());\n  }\n\n  public ngOnInit(): void {\n    this.setSizeProperties();\n    this.createOutletInjector();\n    this.navigate(this.capability.properties.path, {params: this.params}).then(success => {\n      if (!success) {\n        this._dialog.close(Error('[MessageBoxNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the named outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras?: {params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras?.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[this.outletName]: outletCommands}}];\n    return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  private createOutletInjector(): void {\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchClientMessageBoxHandle(this.capability, this.params)],\n    });\n  }\n\n  private setSizeProperties(): void {\n    setStyle(this._host, {\n      'width': this.capability.properties.size?.width ?? null,\n      'min-width': this.capability.properties.size?.minWidth ?? null,\n      'max-width': this.capability.properties.size?.maxWidth ?? null,\n      'height': this.capability.properties.size?.height ?? null,\n      'min-height': this.capability.properties.size?.minHeight ?? null,\n      'max-height': this.capability.properties.size?.maxHeight ?? null,\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchMessageBox} handle to the routed component.\n */\nexport function provideWorkbenchClientMessageBoxHandle(capability: WorkbenchMessageBoxCapability, params: Map<string, unknown>): StaticProvider {\n  return {\n    provide: WorkbenchMessageBox,\n    useFactory: (): WorkbenchMessageBox => {\n\n      return new class implements WorkbenchMessageBox {\n        public readonly capability = capability;\n        public readonly params = params;\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
113
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-message-box.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.html"],"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 {Component, ElementRef, inject, Injector, Input, OnDestroy, OnInit, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchMessageBox, WorkbenchMessageBoxCapability} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {MESSAGE_BOX_ID_PREFIX} from '../../workbench.constants';\nimport {UUID} from '@scion/toolkit/uuid';\nimport {ɵWorkbenchDialog} from '../../dialog/ɵworkbench-dialog';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {ANGULAR_ROUTER_MUTEX} from '../../executor/single-task-executor';\nimport {setStyle} from '../../common/dom.util';\n\n/**\n * Navigates to the microfrontend of a given {@link WorkbenchMessageBoxCapability} via {@link Router}.\n *\n * Unlike {@link MicrofrontendMessageBoxComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * to allow direct integration of the content provided by the workbench host application via the Angular router.\n *\n * This component is designed to be displayed in a workbench message box.\n */\n@Component({\n  selector: 'wb-microfrontend-host-message-box',\n  styleUrls: ['./microfrontend-host-message-box.component.scss'],\n  templateUrl: './microfrontend-host-message-box.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostMessageBoxComponent implements OnDestroy, OnInit {\n\n  @Input({required: true})\n  public capability!: WorkbenchMessageBoxCapability;\n\n  @Input({required: true})\n  public params!: Map<string, unknown>;\n\n  protected outletName: string;\n  protected outletInjector!: Injector;\n\n  /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */\n  private _angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);\n\n  constructor(private _host: ElementRef<HTMLElement>,\n              private _dialog: ɵWorkbenchDialog,\n              private _injector: Injector,\n              private _router: Router) {\n    this.outletName = MESSAGE_BOX_ID_PREFIX.concat(UUID.randomUUID());\n  }\n\n  public ngOnInit(): void {\n    this.setSizeProperties();\n    this.createOutletInjector();\n    this.navigate(this.capability.properties.path, {params: this.params}).then(success => {\n      if (!success) {\n        this._dialog.close(Error('[MessageBoxNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the named outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras?: {params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras?.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[this.outletName]: outletCommands}}];\n    return this._angularRouterMutex.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  private createOutletInjector(): void {\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchClientMessageBoxHandle(this.capability, this.params)],\n    });\n  }\n\n  private setSizeProperties(): void {\n    setStyle(this._host, {\n      'width': this.capability.properties.size?.width ?? null,\n      'min-width': this.capability.properties.size?.minWidth ?? null,\n      'max-width': this.capability.properties.size?.maxWidth ?? null,\n      'height': this.capability.properties.size?.height ?? null,\n      'min-height': this.capability.properties.size?.minHeight ?? null,\n      'max-height': this.capability.properties.size?.maxHeight ?? null,\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchMessageBox} handle to the routed component.\n */\nexport function provideWorkbenchClientMessageBoxHandle(capability: WorkbenchMessageBoxCapability, params: Map<string, unknown>): StaticProvider {\n  return {\n    provide: WorkbenchMessageBox,\n    useFactory: (): WorkbenchMessageBox => {\n\n      return new class implements WorkbenchMessageBox {\n        public readonly capability = capability;\n        public readonly params = params;\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
@@ -16,7 +16,7 @@ import { NgTemplateOutlet } from '@angular/common';
16
16
  import { Defined } from '@scion/toolkit/util';
17
17
  import { POPUP_ID_PREFIX } from '../../workbench.constants';
18
18
  import { Microfrontends } from '../common/microfrontend.util';
19
- import { SINGLE_NAVIGATION_EXECUTOR } from '../../executor/single-task-executor';
19
+ import { ANGULAR_ROUTER_MUTEX } from '../../executor/single-task-executor';
20
20
  import { Objects } from '../../common/objects.util';
21
21
  import * as i0 from "@angular/core";
22
22
  import * as i1 from "../../popup/popup.config";
@@ -31,7 +31,8 @@ export class MicrofrontendHostPopupComponent {
31
31
  constructor(popup, _injector, _router) {
32
32
  this._injector = _injector;
33
33
  this._router = _router;
34
- this._singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);
34
+ /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */
35
+ this._angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);
35
36
  const popupContext = popup.input;
36
37
  const capability = popupContext.capability;
37
38
  const path = Defined.orElseThrow(capability.properties.path, () => Error(`[PopupProviderError] Missing required path for popup capability [application="${capability.metadata.appSymbolicName}", capability=${Objects.toMatrixNotation(capability.qualifier)}]`));
@@ -44,7 +45,7 @@ export class MicrofrontendHostPopupComponent {
44
45
  // Perform navigation in the named router outlet.
45
46
  this.navigate(path, { outletName: this.outletName, params }).then(success => {
46
47
  if (!success) {
47
- popup.closeWithError(Error('[PopupNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));
48
+ popup.close(Error('[PopupNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));
48
49
  }
49
50
  });
50
51
  }
@@ -55,7 +56,7 @@ export class MicrofrontendHostPopupComponent {
55
56
  path = Microfrontends.substituteNamedParameters(path, extras.params);
56
57
  const outletCommands = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path)) : null);
57
58
  const commands = [{ outlets: { [extras.outletName]: outletCommands } }];
58
- return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
59
+ return this._angularRouterMutex.submit(() => this._router.navigate(commands, { skipLocationChange: true, queryParamsHandling: 'preserve' }));
59
60
  }
60
61
  ngOnDestroy() {
61
62
  this.navigate(null, { outletName: this.outletName }).then(); // Remove the outlet from the URL
@@ -84,12 +85,12 @@ function provideWorkbenchPopupHandle(popupContext) {
84
85
  this.params = popupContext.params;
85
86
  this.referrer = popupContext.referrer;
86
87
  }
88
+ setResult(result) {
89
+ popup.setResult(result);
90
+ }
87
91
  close(result) {
88
92
  popup.close(result);
89
93
  }
90
- closeWithError(error) {
91
- popup.closeWithError(error);
92
- }
93
94
  signalReady() {
94
95
  // nothing to do since not an iframe-based microfrontend
95
96
  }
@@ -97,4 +98,4 @@ function provideWorkbenchPopupHandle(popupContext) {
97
98
  },
98
99
  };
99
100
  }
100
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-popup.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.html"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAC,SAAS,EAAE,MAAM,EAAE,QAAQ,EAAa,qBAAqB,EAAiB,MAAM,eAAe,CAAC;AAC5G,OAAO,EAAC,cAAc,EAAgB,MAAM,yBAAyB,CAAC;AACtE,OAAO,EAAC,OAAO,EAAC,MAAM,4BAA4B,CAAC;AAEnD,OAAO,EAAS,YAAY,EAAC,MAAM,iBAAiB,CAAC;AACrD,OAAO,EAAC,KAAK,EAAS,MAAM,0BAA0B,CAAC;AACvD,OAAO,EAAC,gBAAgB,EAAC,MAAM,iBAAiB,CAAC;AACjD,OAAO,EAAC,OAAO,EAAC,MAAM,qBAAqB,CAAC;AAC5C,OAAO,EAAC,eAAe,EAAC,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAC,cAAc,EAAC,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAC,0BAA0B,EAAC,MAAM,qCAAqC,CAAC;AAC/E,OAAO,EAAC,OAAO,EAAC,MAAM,2BAA2B,CAAC;;;;AAElD;;;;;GAKG;AAWH,MAAM,OAAO,+BAA+B;IAO1C,YAAY,KAA4B,EACpB,SAAmB,EACnB,OAAe;QADf,cAAS,GAAT,SAAS,CAAU;QACnB,YAAO,GAAP,OAAO,CAAQ;QAJ3B,8BAAyB,GAAG,MAAM,CAAC,0BAA0B,CAAC,CAAC;QAKrE,MAAM,YAAY,GAAG,KAAK,CAAC,KAAM,CAAC;QAClC,MAAM,UAAU,GAAG,YAAY,CAAC,UAAU,CAAC;QAC3C,MAAM,IAAI,GAAG,OAAO,CAAC,WAAW,CAAC,UAAU,CAAC,UAAU,CAAC,IAAI,EAAE,GAAG,EAAE,CAAC,KAAK,CAAC,iFAAiF,UAAU,CAAC,QAAS,CAAC,eAAe,iBAAiB,OAAO,CAAC,gBAAgB,CAAC,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAC;QACnQ,MAAM,MAAM,GAAG,YAAY,CAAC,MAAM,CAAC;QACnC,IAAI,CAAC,UAAU,GAAG,eAAe,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;QACnD,IAAI,CAAC,cAAc,GAAG,QAAQ,CAAC,MAAM,CAAC;YACpC,MAAM,EAAE,IAAI,CAAC,SAAS;YACtB,SAAS,EAAE,CAAC,2BAA2B,CAAC,YAAY,CAAC,CAAC;SACvD,CAAC,CAAC;QAEH,iDAAiD;QACjD,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAC,UAAU,EAAE,IAAI,CAAC,UAAU,EAAE,MAAM,EAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,EAAE;YACxE,IAAI,CAAC,OAAO,EAAE,CAAC;gBACb,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,kGAAkG,CAAC,CAAC,CAAC;YAClI,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,QAAQ,CAAC,IAAmB,EAAE,MAAuD;QAC3F,IAAI,GAAG,cAAc,CAAC,yBAAyB,CAAC,IAAI,EAAE,MAAM,CAAC,MAAM,CAAC,CAAC;QAErE,MAAM,cAAc,GAAoB,CAAC,IAAI,KAAK,IAAI,CAAC,CAAC,CAAC,qBAAqB,CAAC,IAAI,CAAC,SAAS,EAAE,GAAG,EAAE,CAAC,OAAO,CAAC,cAAc,CAAC,IAAK,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC;QAC5I,MAAM,QAAQ,GAAa,CAAC,EAAC,OAAO,EAAE,EAAC,CAAC,MAAM,CAAC,UAAU,CAAC,EAAE,cAAc,EAAC,EAAC,CAAC,CAAC;QAC9E,OAAO,IAAI,CAAC,yBAAyB,CAAC,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,QAAQ,EAAE,EAAC,kBAAkB,EAAE,IAAI,EAAE,mBAAmB,EAAE,UAAU,EAAC,CAAC,CAAC,CAAC;IACnJ,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,EAAC,UAAU,EAAE,IAAI,CAAC,UAAU,EAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,iCAAiC;IAC9F,CAAC;8GAzCU,+BAA+B;kGAA/B,+BAA+B,uFCvC5C,2KAKA,qHD8BI,YAAY,2JACZ,gBAAgB;;2FAGP,+BAA+B;kBAV3C,SAAS;+BACE,6BAA6B,cAG3B,IAAI,WACP;wBACP,YAAY;wBACZ,gBAAgB;qBACjB;;AA8CH;;GAEG;AACH,SAAS,2BAA2B,CAAC,YAA2B;IAC9D,OAAO;QACL,OAAO,EAAE,cAAc;QACvB,UAAU,EAAE,GAAmB,EAAE;YAC/B,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;YAE5B,OAAO,IAAI;gBAAA;oBACO,eAAU,GAAG,YAAY,CAAC,UAAU,CAAC;oBACrC,WAAM,GAAG,YAAY,CAAC,MAAM,CAAC;oBAC7B,aAAQ,GAAG,YAAY,CAAC,QAAQ,CAAC;gBAanD,CAAC;gBAXQ,KAAK,CAAU,MAAsB;oBAC1C,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;gBACtB,CAAC;gBAEM,cAAc,CAAC,KAAqB;oBACzC,KAAK,CAAC,cAAc,CAAC,KAAK,CAAC,CAAC;gBAC9B,CAAC;gBAEM,WAAW;oBAChB,wDAAwD;gBAC1D,CAAC;aACF,CAAC;QACJ,CAAC;KACF,CAAC;AACJ,CAAC","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 {Component, inject, Injector, OnDestroy, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchPopup, ɵPopupContext} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {Popup, ɵPopup} from '../../popup/popup.config';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {Defined} from '@scion/toolkit/util';\nimport {POPUP_ID_PREFIX} from '../../workbench.constants';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {SINGLE_NAVIGATION_EXECUTOR} from '../../executor/single-task-executor';\nimport {Objects} from '../../common/objects.util';\n\n/**\n * Displays the microfrontend of a popup capability provided by the host inside a workbench popup.\n *\n * Unlike {@link MicrofrontendPopupComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * because integrating the microfrontend directly via Angular router and not via iframe.\n */\n@Component({\n  selector: 'wb-microfrontend-host-popup',\n  styleUrls: ['./microfrontend-host-popup.component.scss'],\n  templateUrl: './microfrontend-host-popup.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostPopupComponent implements OnDestroy {\n\n  public readonly outletName: string;\n  public readonly outletInjector: Injector;\n\n  private _singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);\n\n  constructor(popup: ɵPopup<ɵPopupContext>,\n              private _injector: Injector,\n              private _router: Router) {\n    const popupContext = popup.input!;\n    const capability = popupContext.capability;\n    const path = Defined.orElseThrow(capability.properties.path, () => Error(`[PopupProviderError] Missing required path for popup capability [application=\"${capability.metadata!.appSymbolicName}\", capability=${Objects.toMatrixNotation(capability.qualifier)}]`));\n    const params = popupContext.params;\n    this.outletName = POPUP_ID_PREFIX.concat(popup.id);\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchPopupHandle(popupContext)],\n    });\n\n    // Perform navigation in the named router outlet.\n    this.navigate(path, {outletName: this.outletName, params}).then(success => {\n      if (!success) {\n        popup.closeWithError(Error('[PopupNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the specified outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras: {outletName: string; params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[extras.outletName]: outletCommands}}];\n    return this._singleNavigationExecutor.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null, {outletName: this.outletName}).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchPopup} handle to the routed component.\n */\nfunction provideWorkbenchPopupHandle(popupContext: ɵPopupContext): StaticProvider {\n  return {\n    provide: WorkbenchPopup,\n    useFactory: (): WorkbenchPopup => {\n      const popup = inject(Popup);\n\n      return new class implements WorkbenchPopup {\n        public readonly capability = popupContext.capability;\n        public readonly params = popupContext.params;\n        public readonly referrer = popupContext.referrer;\n\n        public close<R = any>(result?: R | undefined): void {\n          popup.close(result);\n        }\n\n        public closeWithError(error: Error | string): void {\n          popup.closeWithError(error);\n        }\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
101
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-host-popup.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.html"],"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 {Component, inject, Injector, OnDestroy, runInInjectionContext, StaticProvider} from '@angular/core';\nimport {WorkbenchPopup, ɵPopupContext} from '@scion/workbench-client';\nimport {Routing} from '../../routing/routing.util';\nimport {Commands} from '../../routing/routing.model';\nimport {Router, RouterOutlet} from '@angular/router';\nimport {Popup, ɵPopup} from '../../popup/popup.config';\nimport {NgTemplateOutlet} from '@angular/common';\nimport {Defined} from '@scion/toolkit/util';\nimport {POPUP_ID_PREFIX} from '../../workbench.constants';\nimport {Microfrontends} from '../common/microfrontend.util';\nimport {ANGULAR_ROUTER_MUTEX} from '../../executor/single-task-executor';\nimport {Objects} from '../../common/objects.util';\n\n/**\n * Displays the microfrontend of a popup capability provided by the host inside a workbench popup.\n *\n * Unlike {@link MicrofrontendPopupComponent}, this component uses a `<router-outlet>` instead of a `<sci-router-outlet>`\n * because integrating the microfrontend directly via Angular router and not via iframe.\n */\n@Component({\n  selector: 'wb-microfrontend-host-popup',\n  styleUrls: ['./microfrontend-host-popup.component.scss'],\n  templateUrl: './microfrontend-host-popup.component.html',\n  standalone: true,\n  imports: [\n    RouterOutlet,\n    NgTemplateOutlet,\n  ],\n})\nexport class MicrofrontendHostPopupComponent implements OnDestroy {\n\n  public readonly outletName: string;\n  public readonly outletInjector: Injector;\n\n  /** Mutex to serialize Angular Router navigation requests, preventing the cancellation of previously initiated asynchronous navigations. */\n  private _angularRouterMutex = inject(ANGULAR_ROUTER_MUTEX);\n\n  constructor(popup: ɵPopup<ɵPopupContext>,\n              private _injector: Injector,\n              private _router: Router) {\n    const popupContext = popup.input!;\n    const capability = popupContext.capability;\n    const path = Defined.orElseThrow(capability.properties.path, () => Error(`[PopupProviderError] Missing required path for popup capability [application=\"${capability.metadata!.appSymbolicName}\", capability=${Objects.toMatrixNotation(capability.qualifier)}]`));\n    const params = popupContext.params;\n    this.outletName = POPUP_ID_PREFIX.concat(popup.id);\n    this.outletInjector = Injector.create({\n      parent: this._injector,\n      providers: [provideWorkbenchPopupHandle(popupContext)],\n    });\n\n    // Perform navigation in the named router outlet.\n    this.navigate(path, {outletName: this.outletName, params}).then(success => {\n      if (!success) {\n        popup.close(Error('[PopupNavigateError] Navigation canceled, most likely by a route guard or a parallel navigation.'));\n      }\n    });\n  }\n\n  /**\n   * Performs navigation in the specified outlet, substituting path params if any. To clear navigation, pass `null` as the path.\n   */\n  private navigate(path: string | null, extras: {outletName: string; params?: Map<string, any>}): Promise<boolean> {\n    path = Microfrontends.substituteNamedParameters(path, extras.params);\n\n    const outletCommands: Commands | null = (path !== null ? runInInjectionContext(this._injector, () => Routing.pathToCommands(path!)) : null);\n    const commands: Commands = [{outlets: {[extras.outletName]: outletCommands}}];\n    return this._angularRouterMutex.submit(() => this._router.navigate(commands, {skipLocationChange: true, queryParamsHandling: 'preserve'}));\n  }\n\n  public ngOnDestroy(): void {\n    this.navigate(null, {outletName: this.outletName}).then(); // Remove the outlet from the URL\n  }\n}\n\n/**\n * Provides the {WorkbenchPopup} handle to the routed component.\n */\nfunction provideWorkbenchPopupHandle(popupContext: ɵPopupContext): StaticProvider {\n  return {\n    provide: WorkbenchPopup,\n    useFactory: (): WorkbenchPopup => {\n      const popup = inject(Popup);\n\n      return new class<R = unknown> implements WorkbenchPopup {\n        public readonly capability = popupContext.capability;\n        public readonly params = popupContext.params;\n        public readonly referrer = popupContext.referrer;\n\n        public setResult(result?: R): void {\n          popup.setResult(result);\n        }\n\n        public close(result?: R | Error): void {\n          popup.close(result);\n        }\n\n        public signalReady(): void {\n          // nothing to do since not an iframe-based microfrontend\n        }\n      };\n    },\n  };\n}\n","<ng-container *ngTemplateOutlet=\"router_outlet; injector: outletInjector\"/>\n\n<ng-template #router_outlet>\n  <router-outlet [name]=\"outletName\"/>\n</ng-template>\n"]}
@@ -51,13 +51,12 @@ export class MicrofrontendPopupComponent {
51
51
  this._messageClient.observe$(ɵWorkbenchCommands.popupCloseTopic(this.popup.id))
52
52
  .pipe(takeUntilDestroyed(this._destroyRef))
53
53
  .subscribe(closeRequest => {
54
- if (closeRequest.headers.get(ɵWorkbenchPopupMessageHeaders.CLOSE_WITH_ERROR) === true) {
55
- this.popup.closeWithError(closeRequest.body);
56
- }
57
- else {
58
- this.popup.close(closeRequest.body);
59
- }
54
+ this.popup.close(closeRequest.headers.get(ɵWorkbenchPopupMessageHeaders.CLOSE_WITH_ERROR) ? new Error(closeRequest.body) : closeRequest.body);
60
55
  });
56
+ // Listen to popup result requests.
57
+ this._messageClient.observe$(ɵWorkbenchCommands.popupResultTopic(this.popup.id))
58
+ .pipe(takeUntilDestroyed(this._destroyRef))
59
+ .subscribe(result => this.popup.setResult(result.body));
61
60
  // Make the popup context available to embedded content.
62
61
  this.routerOutletElement.nativeElement.setContextValue(ɵPOPUP_CONTEXT, this._popupContext);
63
62
  // Propagate workbench and color theme to the microfrontend.
@@ -77,7 +76,7 @@ export class MicrofrontendPopupComponent {
77
76
  const { detail: focusWithin } = event;
78
77
  // Close the popup on focus loss.
79
78
  if (this._popupContext.closeOnFocusLost && !focusWithin) {
80
- this.popup.close();
79
+ this.popup.close(this.popup.result);
81
80
  }
82
81
  if (focusWithin) {
83
82
  this._host.nativeElement.dispatchEvent(new CustomEvent('sci-microfrontend-focusin', { bubbles: true }));
@@ -113,4 +112,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImpor
113
112
  type: ViewChild,
114
113
  args: ['router_outlet', { static: true }]
115
114
  }] } });
116
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-popup.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2022 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, CUSTOM_ELEMENTS_SCHEMA, DestroyRef, ElementRef, HostBinding, inject, Injector, OnDestroy, OnInit, runInInjectionContext, ViewChild} from '@angular/core';\nimport {ManifestService, MessageClient, MicrofrontendPlatformConfig, OutletRouter, SciRouterOutletElement} from '@scion/microfrontend-platform';\nimport {Logger, LoggerNames} from '../../logging';\nimport {WorkbenchPopupCapability, ɵPOPUP_CONTEXT, ɵPopupContext, ɵWorkbenchCommands, ɵWorkbenchPopupMessageHeaders} from '@scion/workbench-client';\nimport {ɵPopup} from '../../popup/popup.config';\nimport {NgClass, NgComponentOutlet} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {WorkbenchLayoutService} from '../../layout/workbench-layout.service';\nimport {ComponentType} from '@angular/cdk/portal';\nimport {MicrofrontendSplashComponent} from '../microfrontend-splash/microfrontend-splash.component';\nimport {Microfrontends} from '../common/microfrontend.util';\n\n/**\n * Displays the microfrontend of a given {@link WorkbenchPopupCapability}.\n *\n * This component is designed to be displayed in a workbench popup.\n */\n@Component({\n  selector: 'wb-microfrontend-popup',\n  styleUrls: ['./microfrontend-popup.component.scss'],\n  templateUrl: './microfrontend-popup.component.html',\n  standalone: true,\n  imports: [NgClass, NgComponentOutlet],\n  schemas: [CUSTOM_ELEMENTS_SCHEMA], // required because <sci-router-outlet> is a custom element\n})\nexport class MicrofrontendPopupComponent implements OnInit, OnDestroy {\n\n  private _popupContext: ɵPopupContext;\n\n  public popupCapability: WorkbenchPopupCapability;\n\n  /**\n   * Indicates whether a workbench drag operation is in progress, such as when dragging a view or moving a sash.\n   */\n  @HostBinding('class.workbench-drag')\n  public isWorkbenchDrag = false;\n\n  /**\n   * Splash to display until the microfrontend signals readiness.\n   */\n  protected splash: ComponentType<unknown>;\n\n  @ViewChild('router_outlet', {static: true})\n  public routerOutletElement!: ElementRef<SciRouterOutletElement>;\n\n  constructor(public popup: ɵPopup<ɵPopupContext>,\n              private _host: ElementRef<HTMLElement>,\n              private _outletRouter: OutletRouter,\n              private _manifestService: ManifestService,\n              private _messageClient: MessageClient,\n              private _destroyRef: DestroyRef,\n              private _workbenchLayoutService: WorkbenchLayoutService,\n              private _injector: Injector,\n              private _logger: Logger) {\n    this._popupContext = this.popup.input!;\n    this.popupCapability = this._popupContext.capability;\n    this.installWorkbenchDragDetector();\n    this._logger.debug(() => 'Constructing MicrofrontendPopupComponent.', LoggerNames.MICROFRONTEND);\n    this.splash = inject(MicrofrontendPlatformConfig).splash ?? MicrofrontendSplashComponent;\n  }\n\n  public ngOnInit(): void {\n    // Listen to popup close requests.\n    this._messageClient.observe$<any>(ɵWorkbenchCommands.popupCloseTopic(this.popup.id))\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe(closeRequest => {\n        if (closeRequest.headers.get(ɵWorkbenchPopupMessageHeaders.CLOSE_WITH_ERROR) === true) {\n          this.popup.closeWithError(closeRequest.body);\n        }\n        else {\n          this.popup.close(closeRequest.body);\n        }\n      });\n\n    // Make the popup context available to embedded content.\n    this.routerOutletElement.nativeElement.setContextValue(ɵPOPUP_CONTEXT, this._popupContext);\n\n    // Propagate workbench and color theme to the microfrontend.\n    this.propagateWorkbenchTheme();\n\n    // Navigate to the microfrontend.\n    const application = this._manifestService.getApplication(this.popupCapability.metadata!.appSymbolicName);\n    this._logger.debug(() => `Loading microfrontend into workbench popup [app=${this.popupCapability.metadata!.appSymbolicName}, baseUrl=${application.baseUrl}, path=${(this.popupCapability.properties.path)}].`, LoggerNames.MICROFRONTEND, this._popupContext.params, this.popupCapability);\n    this._outletRouter.navigate(this.popupCapability.properties.path, {\n      outlet: this.popup.id,\n      relativeTo: application.baseUrl,\n      params: this._popupContext.params,\n      pushStateToSessionHistoryStack: false,\n      showSplash: this.popupCapability.properties.showSplash,\n    }).then();\n  }\n\n  public onFocusWithin(event: Event): void {\n    const {detail: focusWithin} = event as CustomEvent<boolean>;\n\n    // Close the popup on focus loss.\n    if (this._popupContext.closeOnFocusLost && !focusWithin) {\n      this.popup.close();\n    }\n\n    if (focusWithin) {\n      this._host.nativeElement.dispatchEvent(new CustomEvent('sci-microfrontend-focusin', {bubbles: true}));\n    }\n  }\n\n  /**\n   * Sets the {@link isWorkbenchDrag} property when a workbench drag operation is detected,\n   * such as when dragging a view or moving a sash.\n   */\n  private installWorkbenchDragDetector(): void {\n    this._workbenchLayoutService.dragging$\n      .pipe(takeUntilDestroyed())\n      .subscribe(event => {\n        this.isWorkbenchDrag = (event === 'start');\n      });\n  }\n\n  private propagateWorkbenchTheme(): void {\n    runInInjectionContext(this._injector, () => Microfrontends.propagateTheme(this.routerOutletElement.nativeElement));\n  }\n\n  public ngOnDestroy(): void {\n    this._outletRouter.navigate(null, {outlet: this.popup.id}).then();\n  }\n}\n","<sci-router-outlet #router_outlet\n                   [name]=\"popup.id\"\n                   [attr.data-capabilityid]=\"popupCapability.metadata!.id\"\n                   [attr.data-app]=\"popupCapability.metadata!.appSymbolicName\"\n                   [ngClass]=\"popup.cssClasses\" class=\"e2e-popup\"\n                   keystrokes=\"keydown.escape\"\n                   (focuswithin)=\"onFocusWithin($event)\">\n  <ng-container *ngComponentOutlet=\"splash\"/>\n</sci-router-outlet>\n"]}
115
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"microfrontend-popup.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.ts","../../../../../../../projects/scion/workbench/src/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2022 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, CUSTOM_ELEMENTS_SCHEMA, DestroyRef, ElementRef, HostBinding, inject, Injector, OnDestroy, OnInit, runInInjectionContext, ViewChild} from '@angular/core';\nimport {ManifestService, MessageClient, MicrofrontendPlatformConfig, OutletRouter, SciRouterOutletElement} from '@scion/microfrontend-platform';\nimport {Logger, LoggerNames} from '../../logging';\nimport {WorkbenchPopupCapability, ɵPOPUP_CONTEXT, ɵPopupContext, ɵWorkbenchCommands, ɵWorkbenchPopupMessageHeaders} from '@scion/workbench-client';\nimport {ɵPopup} from '../../popup/popup.config';\nimport {NgClass, NgComponentOutlet} from '@angular/common';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {WorkbenchLayoutService} from '../../layout/workbench-layout.service';\nimport {ComponentType} from '@angular/cdk/portal';\nimport {MicrofrontendSplashComponent} from '../microfrontend-splash/microfrontend-splash.component';\nimport {Microfrontends} from '../common/microfrontend.util';\n\n/**\n * Displays the microfrontend of a given {@link WorkbenchPopupCapability}.\n *\n * This component is designed to be displayed in a workbench popup.\n */\n@Component({\n  selector: 'wb-microfrontend-popup',\n  styleUrls: ['./microfrontend-popup.component.scss'],\n  templateUrl: './microfrontend-popup.component.html',\n  standalone: true,\n  imports: [NgClass, NgComponentOutlet],\n  schemas: [CUSTOM_ELEMENTS_SCHEMA], // required because <sci-router-outlet> is a custom element\n})\nexport class MicrofrontendPopupComponent implements OnInit, OnDestroy {\n\n  private _popupContext: ɵPopupContext;\n\n  public popupCapability: WorkbenchPopupCapability;\n\n  /**\n   * Indicates whether a workbench drag operation is in progress, such as when dragging a view or moving a sash.\n   */\n  @HostBinding('class.workbench-drag')\n  public isWorkbenchDrag = false;\n\n  /**\n   * Splash to display until the microfrontend signals readiness.\n   */\n  protected splash: ComponentType<unknown>;\n\n  @ViewChild('router_outlet', {static: true})\n  public routerOutletElement!: ElementRef<SciRouterOutletElement>;\n\n  constructor(public popup: ɵPopup<ɵPopupContext>,\n              private _host: ElementRef<HTMLElement>,\n              private _outletRouter: OutletRouter,\n              private _manifestService: ManifestService,\n              private _messageClient: MessageClient,\n              private _destroyRef: DestroyRef,\n              private _workbenchLayoutService: WorkbenchLayoutService,\n              private _injector: Injector,\n              private _logger: Logger) {\n    this._popupContext = this.popup.input!;\n    this.popupCapability = this._popupContext.capability;\n    this.installWorkbenchDragDetector();\n    this._logger.debug(() => 'Constructing MicrofrontendPopupComponent.', LoggerNames.MICROFRONTEND);\n    this.splash = inject(MicrofrontendPlatformConfig).splash ?? MicrofrontendSplashComponent;\n  }\n\n  public ngOnInit(): void {\n    // Listen to popup close requests.\n    this._messageClient.observe$<unknown>(ɵWorkbenchCommands.popupCloseTopic(this.popup.id))\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe(closeRequest => {\n        this.popup.close(closeRequest.headers.get(ɵWorkbenchPopupMessageHeaders.CLOSE_WITH_ERROR) ? new Error(closeRequest.body as string) : closeRequest.body);\n      });\n\n    // Listen to popup result requests.\n    this._messageClient.observe$<unknown>(ɵWorkbenchCommands.popupResultTopic(this.popup.id))\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe(result => this.popup.setResult(result.body));\n\n    // Make the popup context available to embedded content.\n    this.routerOutletElement.nativeElement.setContextValue(ɵPOPUP_CONTEXT, this._popupContext);\n\n    // Propagate workbench and color theme to the microfrontend.\n    this.propagateWorkbenchTheme();\n\n    // Navigate to the microfrontend.\n    const application = this._manifestService.getApplication(this.popupCapability.metadata!.appSymbolicName);\n    this._logger.debug(() => `Loading microfrontend into workbench popup [app=${this.popupCapability.metadata!.appSymbolicName}, baseUrl=${application.baseUrl}, path=${(this.popupCapability.properties.path)}].`, LoggerNames.MICROFRONTEND, this._popupContext.params, this.popupCapability);\n    this._outletRouter.navigate(this.popupCapability.properties.path, {\n      outlet: this.popup.id,\n      relativeTo: application.baseUrl,\n      params: this._popupContext.params,\n      pushStateToSessionHistoryStack: false,\n      showSplash: this.popupCapability.properties.showSplash,\n    }).then();\n  }\n\n  public onFocusWithin(event: Event): void {\n    const {detail: focusWithin} = event as CustomEvent<boolean>;\n\n    // Close the popup on focus loss.\n    if (this._popupContext.closeOnFocusLost && !focusWithin) {\n      this.popup.close(this.popup.result);\n    }\n\n    if (focusWithin) {\n      this._host.nativeElement.dispatchEvent(new CustomEvent('sci-microfrontend-focusin', {bubbles: true}));\n    }\n  }\n\n  /**\n   * Sets the {@link isWorkbenchDrag} property when a workbench drag operation is detected,\n   * such as when dragging a view or moving a sash.\n   */\n  private installWorkbenchDragDetector(): void {\n    this._workbenchLayoutService.dragging$\n      .pipe(takeUntilDestroyed())\n      .subscribe(event => {\n        this.isWorkbenchDrag = (event === 'start');\n      });\n  }\n\n  private propagateWorkbenchTheme(): void {\n    runInInjectionContext(this._injector, () => Microfrontends.propagateTheme(this.routerOutletElement.nativeElement));\n  }\n\n  public ngOnDestroy(): void {\n    this._outletRouter.navigate(null, {outlet: this.popup.id}).then();\n  }\n}\n","<sci-router-outlet #router_outlet\n                   [name]=\"popup.id\"\n                   [attr.data-capabilityid]=\"popupCapability.metadata!.id\"\n                   [attr.data-app]=\"popupCapability.metadata!.appSymbolicName\"\n                   [ngClass]=\"popup.cssClasses\" class=\"e2e-popup\"\n                   keystrokes=\"keydown.escape\"\n                   (focuswithin)=\"onFocusWithin($event)\">\n  <ng-container *ngComponentOutlet=\"splash\"/>\n</sci-router-outlet>\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,28 +7,32 @@
7
7
  *
8
8
  * SPDX-License-Identifier: EPL-2.0
9
9
  */
10
- import { Component, Inject, InjectionToken } from '@angular/core';
10
+ import { Component, inject, InjectionToken, isSignal, signal } from '@angular/core';
11
11
  import * as i0 from "@angular/core";
12
12
  export const TEXT = new InjectionToken('TEXT');
13
13
  /**
14
14
  * Component which renders text injected via {@link TEXT} injection token.
15
15
  */
16
16
  export class TextComponent {
17
- constructor(text) {
18
- this.text = text;
17
+ constructor() {
18
+ this.text = coerceText(inject(TEXT));
19
19
  }
20
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: TextComponent, deps: [{ token: TEXT }], target: i0.ɵɵFactoryTarget.Component }); }
21
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: TextComponent, isStandalone: true, selector: "wb-text", ngImport: i0, template: '{{text}}', isInline: true }); }
20
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: TextComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
21
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: TextComponent, isStandalone: true, selector: "wb-text", ngImport: i0, template: '{{text()}}', isInline: true }); }
22
22
  }
23
23
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: TextComponent, decorators: [{
24
24
  type: Component,
25
25
  args: [{
26
26
  selector: 'wb-text',
27
- template: '{{text}}',
27
+ template: '{{text()}}',
28
28
  standalone: true,
29
29
  }]
30
- }], ctorParameters: () => [{ type: undefined, decorators: [{
31
- type: Inject,
32
- args: [TEXT]
33
- }] }] });
34
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9wYXJ0L3ZpZXctY29udGV4dC1tZW51L3RleHQuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLFNBQVMsRUFBRSxNQUFNLEVBQUUsY0FBYyxFQUFDLE1BQU0sZUFBZSxDQUFDOztBQUVoRSxNQUFNLENBQUMsTUFBTSxJQUFJLEdBQUcsSUFBSSxjQUFjLENBQVMsTUFBTSxDQUFDLENBQUM7QUFFdkQ7O0dBRUc7QUFNSCxNQUFNLE9BQU8sYUFBYTtJQUV4QixZQUFpQyxJQUFZO1FBQVosU0FBSSxHQUFKLElBQUksQ0FBUTtJQUM3QyxDQUFDOzhHQUhVLGFBQWEsa0JBRUosSUFBSTtrR0FGYixhQUFhLG1FQUhkLFVBQVU7OzJGQUdULGFBQWE7a0JBTHpCLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLFNBQVM7b0JBQ25CLFFBQVEsRUFBRSxVQUFVO29CQUNwQixVQUFVLEVBQUUsSUFBSTtpQkFDakI7OzBCQUdjLE1BQU07MkJBQUMsSUFBSSIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtMjAyMiBTd2lzcyBGZWRlcmFsIFJhaWx3YXlzXG4gKlxuICogVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscyBhcmUgbWFkZVxuICogYXZhaWxhYmxlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgRWNsaXBzZSBQdWJsaWMgTGljZW5zZSAyLjBcbiAqIHdoaWNoIGlzIGF2YWlsYWJsZSBhdCBodHRwczovL3d3dy5lY2xpcHNlLm9yZy9sZWdhbC9lcGwtMi4wL1xuICpcbiAqIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBFUEwtMi4wXG4gKi9cblxuaW1wb3J0IHtDb21wb25lbnQsIEluamVjdCwgSW5qZWN0aW9uVG9rZW59IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgY29uc3QgVEVYVCA9IG5ldyBJbmplY3Rpb25Ub2tlbjxzdHJpbmc+KCdURVhUJyk7XG5cbi8qKlxuICogQ29tcG9uZW50IHdoaWNoIHJlbmRlcnMgdGV4dCBpbmplY3RlZCB2aWEge0BsaW5rIFRFWFR9IGluamVjdGlvbiB0b2tlbi5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnd2ItdGV4dCcsXG4gIHRlbXBsYXRlOiAne3t0ZXh0fX0nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxufSlcbmV4cG9ydCBjbGFzcyBUZXh0Q29tcG9uZW50IHtcblxuICBjb25zdHJ1Y3RvcihASW5qZWN0KFRFWFQpIHB1YmxpYyB0ZXh0OiBzdHJpbmcpIHtcbiAgfVxufVxuIl19
30
+ }] });
31
+ function coerceText(textOrFn) {
32
+ if (typeof textOrFn === 'function') {
33
+ const text = textOrFn();
34
+ return isSignal(text) ? text : signal(text);
35
+ }
36
+ return signal(textOrFn);
37
+ }
38
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidGV4dC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9wYXJ0L3ZpZXctY29udGV4dC1tZW51L3RleHQuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBOzs7Ozs7OztHQVFHO0FBRUgsT0FBTyxFQUFDLFNBQVMsRUFBRSxNQUFNLEVBQUUsY0FBYyxFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQVMsTUFBTSxlQUFlLENBQUM7O0FBRTFGLE1BQU0sQ0FBQyxNQUFNLElBQUksR0FBRyxJQUFJLGNBQWMsQ0FBMkMsTUFBTSxDQUFDLENBQUM7QUFFekY7O0dBRUc7QUFNSCxNQUFNLE9BQU8sYUFBYTtJQUwxQjtRQU9ZLFNBQUksR0FBRyxVQUFVLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7S0FDM0M7OEdBSFksYUFBYTtrR0FBYixhQUFhLG1FQUhkLFlBQVk7OzJGQUdYLGFBQWE7a0JBTHpCLFNBQVM7bUJBQUM7b0JBQ1QsUUFBUSxFQUFFLFNBQVM7b0JBQ25CLFFBQVEsRUFBRSxZQUFZO29CQUN0QixVQUFVLEVBQUUsSUFBSTtpQkFDakI7O0FBTUQsU0FBUyxVQUFVLENBQUMsUUFBa0Q7SUFDcEUsSUFBSSxPQUFPLFFBQVEsS0FBSyxVQUFVLEVBQUUsQ0FBQztRQUNuQyxNQUFNLElBQUksR0FBRyxRQUFRLEVBQUUsQ0FBQztRQUN4QixPQUFPLFFBQVEsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQyxNQUFNLENBQUMsSUFBSSxDQUFDLENBQUM7SUFDOUMsQ0FBQztJQUNELE9BQU8sTUFBTSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0FBQzFCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKlxuICogQ29weXJpZ2h0IChjKSAyMDE4LTIwMjQgU3dpc3MgRmVkZXJhbCBSYWlsd2F5c1xuICpcbiAqIFRoaXMgcHJvZ3JhbSBhbmQgdGhlIGFjY29tcGFueWluZyBtYXRlcmlhbHMgYXJlIG1hZGVcbiAqIGF2YWlsYWJsZSB1bmRlciB0aGUgdGVybXMgb2YgdGhlIEVjbGlwc2UgUHVibGljIExpY2Vuc2UgMi4wXG4gKiB3aGljaCBpcyBhdmFpbGFibGUgYXQgaHR0cHM6Ly93d3cuZWNsaXBzZS5vcmcvbGVnYWwvZXBsLTIuMC9cbiAqXG4gKiBTUERYLUxpY2Vuc2UtSWRlbnRpZmllcjogRVBMLTIuMFxuICovXG5cbmltcG9ydCB7Q29tcG9uZW50LCBpbmplY3QsIEluamVjdGlvblRva2VuLCBpc1NpZ25hbCwgc2lnbmFsLCBTaWduYWx9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuXG5leHBvcnQgY29uc3QgVEVYVCA9IG5ldyBJbmplY3Rpb25Ub2tlbjxzdHJpbmcgfCAoKCkgPT4gc3RyaW5nIHwgU2lnbmFsPHN0cmluZz4pPignVEVYVCcpO1xuXG4vKipcbiAqIENvbXBvbmVudCB3aGljaCByZW5kZXJzIHRleHQgaW5qZWN0ZWQgdmlhIHtAbGluayBURVhUfSBpbmplY3Rpb24gdG9rZW4uXG4gKi9cbkBDb21wb25lbnQoe1xuICBzZWxlY3RvcjogJ3diLXRleHQnLFxuICB0ZW1wbGF0ZTogJ3t7dGV4dCgpfX0nLFxuICBzdGFuZGFsb25lOiB0cnVlLFxufSlcbmV4cG9ydCBjbGFzcyBUZXh0Q29tcG9uZW50IHtcblxuICBwcm90ZWN0ZWQgdGV4dCA9IGNvZXJjZVRleHQoaW5qZWN0KFRFWFQpKTtcbn1cblxuZnVuY3Rpb24gY29lcmNlVGV4dCh0ZXh0T3JGbjogc3RyaW5nIHwgKCgpID0+IHN0cmluZyB8IFNpZ25hbDxzdHJpbmc+KSk6IFNpZ25hbDxzdHJpbmc+IHtcbiAgaWYgKHR5cGVvZiB0ZXh0T3JGbiA9PT0gJ2Z1bmN0aW9uJykge1xuICAgIGNvbnN0IHRleHQgPSB0ZXh0T3JGbigpO1xuICAgIHJldHVybiBpc1NpZ25hbCh0ZXh0KSA/IHRleHQgOiBzaWduYWwodGV4dCk7XG4gIH1cbiAgcmV0dXJuIHNpZ25hbCh0ZXh0T3JGbik7XG59XG4iXX0=