@scion/workbench 18.0.0-beta.1 → 18.0.0-beta.10
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.
- package/_index.scss +7 -5
- package/design/_workbench-global-styles.scss +18 -0
- package/design/_workbench-icon-font.scss +1 -1
- package/design/_workbench-popup-global-styles.scss +1 -1
- package/design/{_workbench-view-drag-image-global-styles.scss → _workbench-view-global-styles.scss} +1 -2
- package/esm2022/lib/common/class-list.mjs +67 -29
- package/esm2022/lib/common/math.util.mjs +16 -0
- package/esm2022/lib/common/objects.util.mjs +7 -1
- package/esm2022/lib/common/operators.mjs +1 -12
- package/esm2022/lib/common/uid.util.mjs +22 -0
- package/esm2022/lib/content-projection/content-as-overlay.component.mjs +85 -28
- package/esm2022/lib/content-projection/workbench-element-references.mjs +32 -0
- package/esm2022/lib/dialog/dialog-header/dialog-header.component.mjs +3 -8
- package/esm2022/lib/dialog/movable.directive.mjs +3 -3
- package/esm2022/lib/dialog/resizable.directive.mjs +3 -3
- package/esm2022/lib/dialog/workbench-dialog.component.mjs +25 -23
- package/esm2022/lib/dialog/workbench-dialog.mjs +61 -1
- package/esm2022/lib/dialog//311/265workbench-dialog.mjs +98 -60
- package/esm2022/lib/dialog//311/265workbench-dialog.service.mjs +26 -23
- package/esm2022/lib/executor/latest-task-executor.mjs +55 -0
- package/esm2022/lib/executor/single-task-executor.mjs +9 -19
- package/esm2022/lib/filter-field/filter-field.component.mjs +5 -5
- package/esm2022/lib/layout/grid-element/grid-element.component.mjs +8 -18
- package/esm2022/lib/layout/main-area-layout/main-area-layout.component.mjs +18 -21
- package/esm2022/lib/layout/migration/model/workbench-layout-migration-v5.model.mjs +11 -0
- package/esm2022/lib/layout/migration/model/workbench-layout-migration-v6.model.mjs +11 -0
- package/esm2022/lib/layout/migration/workbench-layout-migration-v3.service.mjs +4 -4
- package/esm2022/lib/layout/migration/workbench-layout-migration-v5.service.mjs +67 -0
- package/esm2022/lib/layout/migration/workbench-layout-migration-v6.service.mjs +59 -0
- package/esm2022/lib/layout/stringifier.mjs +70 -0
- package/esm2022/lib/layout/workbench-layout.component.mjs +18 -22
- package/esm2022/lib/layout/workbench-layout.mjs +1 -1
- package/esm2022/lib/layout/workbench-layout.model.mjs +1 -1
- package/esm2022/lib/layout/workbench-layout.service.mjs +26 -24
- package/esm2022/lib/layout/workbench-layouts.util.mjs +1 -9
- package/esm2022/lib/layout/workench-layout-serializer.service.mjs +19 -27
- package/esm2022/lib/layout//311/265workbench-layout.factory.mjs +3 -2
- package/esm2022/lib/layout//311/265workbench-layout.mjs +73 -62
- package/esm2022/lib/logging/logging-support.mjs +7 -13
- package/esm2022/lib/logging//311/265logger.mjs +10 -10
- package/esm2022/lib/message-box/message-box-footer/message-box-footer.component.mjs +7 -6
- package/esm2022/lib/message-box/message-box-header/message-box-header.component.mjs +3 -3
- package/esm2022/lib/message-box/workbench-message-box.component.mjs +2 -2
- package/esm2022/lib/message-box//311/265workbench-message-box.service.mjs +3 -2
- package/esm2022/lib/microfrontend-platform/common/microfrontend.util.mjs +22 -6
- package/esm2022/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.mjs +28 -21
- package/esm2022/lib/microfrontend-platform/initialization/ng-zone-observable-decorator.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/initialization/workbench-host-manifest-interceptor.service.mjs +11 -1
- package/esm2022/lib/microfrontend-platform/manifest-object-cache.service.mjs +63 -0
- package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog-capability-validator.interceptor.mjs +5 -4
- package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog.component.mjs +6 -10
- package/esm2022/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.mjs +18 -9
- package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.mjs +7 -6
- package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.mjs +15 -13
- package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box-capability-validator.interceptor.mjs +3 -2
- package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box.component.mjs +6 -11
- package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-capability-validator.interceptor.mjs +39 -0
- package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-installer.service.mjs +120 -0
- package/esm2022/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-intent-handler.interceptor.mjs +55 -0
- package/esm2022/lib/microfrontend-platform/microfrontend-perspective/workbench-perspective-data.mjs +19 -0
- package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-capability-validator.interceptor.mjs +4 -3
- package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-intent-handler.interceptor.mjs +11 -12
- package/esm2022/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.mjs +12 -28
- package/esm2022/lib/microfrontend-platform/microfrontend-splash/microfrontend-splash.component.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view-command-handler.service.mjs +11 -27
- package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.mjs +91 -71
- package/esm2022/lib/microfrontend-platform/public_api.mjs +2 -1
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-capability-validator.interceptor.mjs +5 -4
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-intent-handler.interceptor.mjs +18 -16
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-routes.mjs +30 -8
- package/esm2022/lib/microfrontend-platform/stable-capability-id-assigner.interceptor.mjs +32 -0
- package/esm2022/lib/microfrontend-platform/workbench-microfrontend-support.mjs +26 -8
- package/esm2022/lib/notification/notification-list.component.mjs +4 -8
- package/esm2022/lib/notification/notification.component.mjs +4 -5
- package/esm2022/lib/notification/notification.service.mjs +5 -4
- package/esm2022/lib/page-not-found/page-not-found.component.mjs +5 -7
- package/esm2022/lib/part/part-action-bar/part-action-bar.component.mjs +11 -19
- package/esm2022/lib/part/part-action-bar/part-action.directive.mjs +4 -4
- package/esm2022/lib/part/part-bar/part-bar.component.mjs +333 -264
- package/esm2022/lib/part/part-portal.pipe.mjs +8 -8
- package/esm2022/lib/part/part.component.mjs +33 -42
- package/esm2022/lib/part/view-context-menu/text.component.mjs +16 -12
- package/esm2022/lib/part/view-context-menu/view-menu.component.mjs +4 -6
- package/esm2022/lib/part/view-context-menu/view-menu.directive.mjs +2 -2
- package/esm2022/lib/part/view-context-menu/view-menu.service.mjs +48 -49
- package/esm2022/lib/part/view-list/view-list.component.mjs +41 -48
- package/esm2022/lib/part/view-list-button/view-list-button.component.mjs +22 -26
- package/esm2022/lib/part/view-list-item/view-list-item.component.mjs +15 -17
- package/esm2022/lib/part/view-tab/view-tab.component.mjs +99 -110
- package/esm2022/lib/part/view-tab-content/view-tab-content.component.mjs +3 -4
- package/esm2022/lib/part/view-tab-drag-image/view-tab-drag-image.component.mjs +16 -17
- package/esm2022/lib/part/workbench-part-action.registry.mjs +18 -42
- package/esm2022/lib/part/workbench-part.model.mjs +10 -1
- package/esm2022/lib/part/workbench-part.registry.mjs +11 -45
- package/esm2022/lib/part//311/265workbench-part.model.mjs +71 -55
- package/esm2022/lib/perspective/workbench-grid-merger.service.mjs +3 -3
- package/esm2022/lib/perspective/workbench-perspective.model.mjs +1 -1
- package/esm2022/lib/perspective/workbench-perspective.registry.mjs +11 -43
- package/esm2022/lib/perspective/workbench-perspective.service.mjs +71 -62
- package/esm2022/lib/perspective/workench-perspective-serializer.service.mjs +2 -2
- package/esm2022/lib/perspective//311/265workbench-perspective.model.mjs +62 -47
- package/esm2022/lib/popup/popup.component.mjs +3 -3
- package/esm2022/lib/popup/popup.config.mjs +27 -25
- package/esm2022/lib/popup/popup.service.mjs +169 -174
- package/esm2022/lib/portal/wb-component-portal.mjs +18 -17
- package/esm2022/lib/public_api.mjs +1 -1
- package/esm2022/lib/registry/workbench-object-registry.mjs +38 -17
- package/esm2022/lib/routing/public_api.mjs +2 -2
- package/esm2022/lib/routing/routing.model.mjs +1 -1
- package/esm2022/lib/routing/routing.util.mjs +215 -0
- package/esm2022/lib/routing/workbench-auxiliary-route-installer.service.mjs +110 -0
- package/esm2022/lib/routing/workbench-dialog-differ.mjs +3 -3
- package/esm2022/lib/routing/workbench-layout-differ.mjs +3 -10
- package/esm2022/lib/routing/workbench-message-box-differ.mjs +3 -3
- package/esm2022/lib/routing/workbench-navigational-states.mjs +2 -2
- package/esm2022/lib/routing/workbench-popup-differ.mjs +3 -3
- package/esm2022/lib/routing/workbench-route-guards.mjs +57 -0
- package/esm2022/lib/routing/workbench-url-observer.service.mjs +140 -148
- package/esm2022/lib/routing/workbench-view-outlet-differ.mjs +58 -0
- package/esm2022/lib/routing//311/265workbench-router.service.mjs +119 -74
- package/esm2022/lib/startup/splash/splash.component.mjs +3 -3
- package/esm2022/lib/startup/workbench-launcher.service.mjs +22 -10
- package/esm2022/lib/testing/workbench.provider.mjs +2 -2
- package/esm2022/lib/theme/workbench-theme-switcher.service.mjs +15 -19
- package/esm2022/lib/view/public_api.mjs +1 -2
- package/esm2022/lib/view/view-move-handler.service.mjs +71 -56
- package/esm2022/lib/view/view-portal.pipe.mjs +7 -7
- package/esm2022/lib/view/view.component.mjs +54 -42
- package/esm2022/lib/view/workbench-view.model.mjs +1 -1
- package/esm2022/lib/view/workbench-view.registry.mjs +11 -45
- package/esm2022/lib/view//311/265workbench-view.model.mjs +183 -153
- package/esm2022/lib/view-dnd/grid-drop-targets.util.mjs +2 -2
- package/esm2022/lib/view-dnd/view-drag.service.mjs +87 -71
- package/esm2022/lib/view-dnd/view-drop-placeholder-renderer.service.mjs +6 -6
- package/esm2022/lib/view-dnd/view-drop-zone.directive.mjs +6 -6
- package/esm2022/lib/view-dnd/view-tab-drag-image-renderer.service.mjs +43 -56
- package/esm2022/lib/workbench-config.mjs +2 -2
- package/esm2022/lib/workbench-id.mjs +3 -3
- package/esm2022/lib/workbench.component.mjs +66 -54
- package/esm2022/lib/workbench.constants.mjs +1 -5
- package/esm2022/lib/workbench.model.mjs +1 -1
- package/esm2022/lib/workbench.provider.mjs +3 -9
- package/esm2022/lib/workbench.service.mjs +1 -1
- package/esm2022/lib//311/265workbench.service.mjs +38 -42
- package/fesm2022/scion-workbench.mjs +4054 -3276
- package/fesm2022/scion-workbench.mjs.map +1 -1
- package/lib/common/class-list.d.ts +35 -22
- package/lib/common/math.util.d.ts +7 -0
- package/lib/common/objects.util.d.ts +4 -0
- package/lib/common/operators.d.ts +1 -6
- package/lib/common/uid.util.d.ts +9 -0
- package/lib/content-projection/content-as-overlay.component.d.ts +39 -22
- package/lib/content-projection/workbench-element-references.d.ts +13 -0
- package/lib/dialog/workbench-dialog.component.d.ts +3 -2
- package/lib/dialog/workbench-dialog.d.ts +57 -13
- package/lib/dialog//311/265workbench-dialog.d.ts +38 -24
- package/lib/dialog//311/265workbench-dialog.service.d.ts +10 -11
- package/lib/executor/latest-task-executor.d.ts +26 -0
- package/lib/executor/single-task-executor.d.ts +5 -8
- package/lib/filter-field/filter-field.component.d.ts +1 -1
- package/lib/layout/grid-element/grid-element.component.d.ts +1 -8
- package/lib/layout/main-area-layout/main-area-layout.component.d.ts +0 -4
- package/lib/layout/migration/model/workbench-layout-migration-v5.model.d.ts +36 -0
- package/lib/layout/migration/model/workbench-layout-migration-v6.model.d.ts +35 -0
- package/lib/layout/migration/workbench-layout-migration-v5.service.d.ts +12 -0
- package/lib/layout/migration/workbench-layout-migration-v6.service.d.ts +12 -0
- package/lib/layout/stringifier.d.ts +26 -0
- package/lib/layout/workbench-layout.component.d.ts +4 -5
- package/lib/layout/workbench-layout.d.ts +9 -6
- package/lib/layout/workbench-layout.model.d.ts +4 -3
- package/lib/layout/workbench-layout.service.d.ts +14 -16
- package/lib/layout/workbench-layouts.util.d.ts +0 -6
- package/lib/layout/workench-layout-serializer.service.d.ts +13 -16
- package/lib/layout//311/265workbench-layout.d.ts +18 -19
- package/lib/layout//311/265workbench-layout.factory.d.ts +3 -2
- package/lib/logging//311/265logger.d.ts +2 -3
- package/lib/microfrontend-platform/common/microfrontend.util.d.ts +5 -1
- package/lib/microfrontend-platform/initialization/microfrontend-platform-initializer.service.d.ts +7 -3
- package/lib/microfrontend-platform/manifest-object-cache.service.d.ts +33 -0
- package/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.d.ts +3 -6
- package/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.d.ts +2 -1
- package/lib/microfrontend-platform/microfrontend-host-message-box/text-message/text-message.component.d.ts +1 -1
- package/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.d.ts +2 -1
- package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-capability-validator.interceptor.d.ts +10 -0
- package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-installer.service.d.ts +24 -0
- package/lib/microfrontend-platform/microfrontend-perspective/microfrontend-perspective-intent-handler.interceptor.d.ts +20 -0
- package/lib/microfrontend-platform/microfrontend-perspective/workbench-perspective-data.d.ts +9 -0
- package/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-intent-handler.interceptor.d.ts +4 -8
- package/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup.component.d.ts +0 -4
- package/lib/microfrontend-platform/microfrontend-view/microfrontend-view-command-handler.service.d.ts +5 -12
- package/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.d.ts +23 -33
- package/lib/microfrontend-platform/public_api.d.ts +1 -0
- package/lib/microfrontend-platform/routing/microfrontend-view-capability-validator.interceptor.d.ts +1 -1
- package/lib/microfrontend-platform/routing/microfrontend-view-intent-handler.interceptor.d.ts +3 -7
- package/lib/microfrontend-platform/routing/microfrontend-view-routes.d.ts +9 -2
- package/lib/microfrontend-platform/stable-capability-id-assigner.interceptor.d.ts +11 -0
- package/lib/notification/notification-list.component.d.ts +0 -2
- package/lib/page-not-found/page-not-found.component.d.ts +1 -0
- package/lib/part/part-action-bar/part-action-bar.component.d.ts +3 -4
- package/lib/part/part-action-bar/part-action.directive.d.ts +2 -2
- package/lib/part/part-bar/part-bar.component.d.ts +73 -63
- package/lib/part/part-portal.pipe.d.ts +3 -5
- package/lib/part/part.component.d.ts +9 -12
- package/lib/part/view-context-menu/text.component.d.ts +3 -4
- package/lib/part/view-context-menu/view-menu.directive.d.ts +1 -1
- package/lib/part/view-context-menu/view-menu.service.d.ts +13 -17
- package/lib/part/view-list/view-list.component.d.ts +8 -21
- package/lib/part/view-list-button/view-list-button.component.d.ts +8 -11
- package/lib/part/view-list-item/view-list-item.component.d.ts +3 -7
- package/lib/part/view-tab/view-tab.component.d.ts +19 -29
- package/lib/part/view-tab-drag-image/view-tab-drag-image.component.d.ts +4 -4
- package/lib/part/workbench-part-action.registry.d.ts +3 -23
- package/lib/part/workbench-part.model.d.ts +8 -27
- package/lib/part/workbench-part.registry.d.ts +3 -26
- package/lib/part//311/265workbench-part.model.d.ts +23 -17
- package/lib/perspective/workbench-perspective.model.d.ts +18 -25
- package/lib/perspective/workbench-perspective.registry.d.ts +3 -26
- package/lib/perspective/workbench-perspective.service.d.ts +16 -14
- package/lib/perspective/workench-perspective-serializer.service.d.ts +1 -1
- package/lib/perspective//311/265workbench-perspective.model.d.ts +30 -21
- package/lib/popup/popup.config.d.ts +11 -10
- package/lib/popup/popup.service.d.ts +15 -28
- package/lib/portal/wb-component-portal.d.ts +21 -5
- package/lib/public_api.d.ts +1 -1
- package/lib/registry/workbench-object-registry.d.ts +10 -6
- package/lib/routing/public_api.d.ts +2 -2
- package/lib/routing/routing.model.d.ts +25 -13
- package/lib/routing/{router.util.d.ts → routing.util.d.ts} +28 -1
- package/lib/routing/{workbench-auxiliary-routes-registrator.service.d.ts → workbench-auxiliary-route-installer.service.d.ts} +10 -3
- package/lib/routing/workbench-layout-differ.d.ts +1 -2
- package/lib/routing/workbench-navigational-states.d.ts +8 -4
- package/lib/{view/workbench-view-route-guards.d.ts → routing/workbench-route-guards.d.ts} +8 -2
- package/lib/routing/workbench-url-observer.service.d.ts +31 -55
- package/lib/routing/workbench-view-outlet-differ.d.ts +32 -0
- package/lib/routing//311/265workbench-router.service.d.ts +22 -18
- package/lib/startup/workbench-launcher.service.d.ts +4 -5
- package/lib/testing/workbench.provider.d.ts +3 -1
- package/lib/theme/workbench-theme-switcher.service.d.ts +4 -6
- package/lib/view/public_api.d.ts +0 -1
- package/lib/view/view-move-handler.service.d.ts +16 -14
- package/lib/view/view-portal.pipe.d.ts +1 -3
- package/lib/view/view.component.d.ts +21 -12
- package/lib/view/workbench-view.model.d.ts +88 -36
- package/lib/view/workbench-view.registry.d.ts +3 -27
- package/lib/view//311/265workbench-view.model.d.ts +62 -71
- package/lib/view-dnd/view-drag.service.d.ts +79 -49
- package/lib/view-dnd/view-drop-placeholder-renderer.service.d.ts +1 -1
- package/lib/view-dnd/view-tab-drag-image-renderer.service.d.ts +12 -36
- package/lib/workbench-config.d.ts +19 -7
- package/lib/workbench.component.d.ts +16 -30
- package/lib/workbench.constants.d.ts +0 -5
- package/lib/workbench.model.d.ts +31 -3
- package/lib/workbench.service.d.ts +58 -56
- package/lib//311/265workbench.service.d.ts +18 -31
- package/package.json +5 -5
- package/esm2022/lib/common/filter-by-predicate.pipe.mjs +0 -27
- package/esm2022/lib/common/filter-by-text.pipe.mjs +0 -39
- package/esm2022/lib/common/uuid.util.mjs +0 -17
- package/esm2022/lib/content-projection/content-projection.directive.mjs +0 -101
- package/esm2022/lib/content-projection/view-container.reference.mjs +0 -68
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-capability-id-assigner.interceptor.mjs +0 -41
- package/esm2022/lib/page-not-found/format-url.pipe.mjs +0 -26
- package/esm2022/lib/part/part-action-bar/part-action-filter.pipe.mjs +0 -26
- package/esm2022/lib/routing/router.util.mjs +0 -126
- package/esm2022/lib/routing/workbench-auxiliary-routes-registrator.service.mjs +0 -94
- package/esm2022/lib/view/workbench-view-route-guards.mjs +0 -47
- package/lib/common/filter-by-predicate.pipe.d.ts +0 -10
- package/lib/common/filter-by-text.pipe.d.ts +0 -12
- package/lib/common/uuid.util.d.ts +0 -8
- package/lib/content-projection/content-projection.directive.d.ts +0 -30
- package/lib/content-projection/view-container.reference.d.ts +0 -42
- package/lib/microfrontend-platform/routing/microfrontend-view-capability-id-assigner.interceptor.d.ts +0 -10
- package/lib/page-not-found/format-url.pipe.d.ts +0 -11
- package/lib/part/part-action-bar/part-action-filter.pipe.d.ts +0 -11
|
@@ -7,29 +7,31 @@
|
|
|
7
7
|
*
|
|
8
8
|
* SPDX-License-Identifier: EPL-2.0
|
|
9
9
|
*/
|
|
10
|
-
import { ElementRef,
|
|
11
|
-
import { OverlayConfig } from '@angular/cdk/overlay';
|
|
12
|
-
import {
|
|
13
|
-
import {
|
|
10
|
+
import { assertNotInReactiveContext, computed, createEnvironmentInjector, DestroyRef, effect, ElementRef, EnvironmentInjector, inject, Injectable, Injector, NgZone, runInInjectionContext, untracked } from '@angular/core';
|
|
11
|
+
import { Overlay, OverlayConfig } from '@angular/cdk/overlay';
|
|
12
|
+
import { firstValueFrom, fromEvent } from 'rxjs';
|
|
13
|
+
import { filter } from 'rxjs/operators';
|
|
14
14
|
import { ComponentPortal } from '@angular/cdk/portal';
|
|
15
|
-
import { Popup, ɵPopup
|
|
16
|
-
import {
|
|
17
|
-
import {
|
|
15
|
+
import { Popup, ɵPopup } from './popup.config';
|
|
16
|
+
import { FocusMonitor } from '@angular/cdk/a11y';
|
|
17
|
+
import { Observables } from '@scion/toolkit/util';
|
|
18
|
+
import { WORKBENCH_VIEW_REGISTRY } from '../view/workbench-view.registry';
|
|
19
|
+
import { fromResize$ } from '@scion/toolkit/observable';
|
|
18
20
|
import { PopupComponent } from './popup.component';
|
|
19
21
|
import { DOCUMENT } from '@angular/common';
|
|
20
|
-
import {
|
|
21
|
-
import {
|
|
22
|
-
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
22
|
+
import { observeIn, subscribeIn } from '@scion/toolkit/operators';
|
|
23
|
+
import { ɵWorkbenchView } from '../view/ɵworkbench-view.model';
|
|
24
|
+
import { takeUntilDestroyed, toObservable, toSignal } from '@angular/core/rxjs-interop';
|
|
23
25
|
import { provideViewContext } from '../view/view-context-provider';
|
|
26
|
+
import { UUID } from '@scion/toolkit/uuid';
|
|
27
|
+
import { boundingClientRect } from '@scion/components/dimension';
|
|
28
|
+
import { coerceElement } from '@angular/cdk/coercion';
|
|
29
|
+
import { clamp } from '../common/math.util';
|
|
24
30
|
import * as i0 from "@angular/core";
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
const NORTH = { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom' };
|
|
30
|
-
const SOUTH = { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top' };
|
|
31
|
-
const WEST = { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center' };
|
|
32
|
-
const EAST = { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center' };
|
|
31
|
+
const NORTH = { originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', panelClass: 'wb-north' };
|
|
32
|
+
const SOUTH = { originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', panelClass: 'wb-south' };
|
|
33
|
+
const WEST = { originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center', panelClass: 'wb-west' };
|
|
34
|
+
const EAST = { originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center', panelClass: 'wb-east' };
|
|
33
35
|
/**
|
|
34
36
|
* Allows displaying a component in a workbench popup.
|
|
35
37
|
*
|
|
@@ -39,14 +41,14 @@ const EAST = { originX: 'end', originY: 'center', overlayX: 'start', overlayY: '
|
|
|
39
41
|
* Unlike views, popups are not part of the persistent Workbench navigation, meaning that popups do not survive a page reload.
|
|
40
42
|
*/
|
|
41
43
|
export class PopupService {
|
|
42
|
-
constructor(
|
|
43
|
-
this.
|
|
44
|
-
this._overlay =
|
|
45
|
-
this._focusManager =
|
|
46
|
-
this._viewRegistry =
|
|
47
|
-
this._zone =
|
|
48
|
-
this._document =
|
|
49
|
-
this._view =
|
|
44
|
+
constructor() {
|
|
45
|
+
this._environmentInjector = inject(EnvironmentInjector);
|
|
46
|
+
this._overlay = inject(Overlay);
|
|
47
|
+
this._focusManager = inject(FocusMonitor);
|
|
48
|
+
this._viewRegistry = inject(WORKBENCH_VIEW_REGISTRY);
|
|
49
|
+
this._zone = inject(NgZone);
|
|
50
|
+
this._document = inject(DOCUMENT);
|
|
51
|
+
this._view = inject(ɵWorkbenchView, { optional: true });
|
|
50
52
|
}
|
|
51
53
|
/**
|
|
52
54
|
* Displays the specified component in a popup.
|
|
@@ -74,19 +76,20 @@ export class PopupService {
|
|
|
74
76
|
* - resolves to `undefined` if closed without a result
|
|
75
77
|
*/
|
|
76
78
|
async open(config) {
|
|
79
|
+
assertNotInReactiveContext(this.open, 'Call WorkbenchPopupService.open() in a non-reactive (non-tracking) context, such as within the untracked() function.');
|
|
77
80
|
// Ensure to run in Angular zone to display the popup even when called from outside of the Angular zone.
|
|
78
81
|
if (!NgZone.isInAngularZone()) {
|
|
79
82
|
return this._zone.run(() => this.open(config));
|
|
80
83
|
}
|
|
81
84
|
const align = config.align || 'north';
|
|
82
85
|
const contextualView = this.resolveContextualView(config);
|
|
83
|
-
const
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
//
|
|
87
|
-
const
|
|
86
|
+
const popup = this.createPopup(config, { view: contextualView });
|
|
87
|
+
const popupDestroyRef = popup.injector.get(DestroyRef);
|
|
88
|
+
const popupOrigin = this.trackPopupOrigin(config, contextualView, popup.injector);
|
|
89
|
+
// Wait for the initial position.
|
|
90
|
+
const initialPosition = await firstValueFrom(toObservable(popupOrigin, { injector: popup.injector }).pipe(filter(Boolean)));
|
|
88
91
|
const overlayPositionStrategy = this._overlay.position()
|
|
89
|
-
.flexibleConnectedTo(
|
|
92
|
+
.flexibleConnectedTo(initialPosition)
|
|
90
93
|
.withFlexibleDimensions(false)
|
|
91
94
|
.withLockedPosition(false) // If locked, the popup won't attempt to reposition itself if not enough space available.
|
|
92
95
|
.withPositions((() => {
|
|
@@ -105,10 +108,7 @@ export class PopupService {
|
|
|
105
108
|
})());
|
|
106
109
|
// Configure the popup overlay.
|
|
107
110
|
const overlayConfig = new OverlayConfig({
|
|
108
|
-
panelClass:
|
|
109
|
-
'wb-popup',
|
|
110
|
-
`wb-${align}`,
|
|
111
|
-
],
|
|
111
|
+
panelClass: 'wb-popup',
|
|
112
112
|
hasBackdrop: false,
|
|
113
113
|
positionStrategy: overlayPositionStrategy,
|
|
114
114
|
scrollStrategy: this._overlay.scrollStrategies.noop(),
|
|
@@ -116,9 +116,8 @@ export class PopupService {
|
|
|
116
116
|
});
|
|
117
117
|
// Construct the popup component and attach it to the DOM.
|
|
118
118
|
const overlayRef = this._overlay.create(overlayConfig);
|
|
119
|
-
const popup = runInInjectionContext(this._injector, () => new ɵPopup(config, referrer));
|
|
120
119
|
const popupPortal = new ComponentPortal(PopupComponent, null, Injector.create({
|
|
121
|
-
parent: config.componentConstructOptions?.injector || this.
|
|
120
|
+
parent: config.componentConstructOptions?.injector || this._environmentInjector,
|
|
122
121
|
providers: [
|
|
123
122
|
{ provide: ɵPopup, useValue: popup },
|
|
124
123
|
{ provide: Popup, useExisting: ɵPopup },
|
|
@@ -131,21 +130,22 @@ export class PopupService {
|
|
|
131
130
|
// Make the popup focusable and request the focus.
|
|
132
131
|
popupElement.setAttribute('tabindex', '-1');
|
|
133
132
|
this._focusManager.focusVia(popupElement, 'program'); // To not close the popup immediately when it opens, if using the 'onFocusLost' strategy.
|
|
134
|
-
//
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
|
|
139
|
-
|
|
140
|
-
|
|
141
|
-
|
|
142
|
-
|
|
133
|
+
// Reposition popup when the anchor is moved.
|
|
134
|
+
effect(() => {
|
|
135
|
+
const origin = popupOrigin();
|
|
136
|
+
untracked(() => {
|
|
137
|
+
if (origin) {
|
|
138
|
+
overlayPositionStrategy.setOrigin(origin);
|
|
139
|
+
overlayRef.updatePosition();
|
|
140
|
+
}
|
|
141
|
+
});
|
|
142
|
+
}, { injector: popup.injector });
|
|
143
|
+
// Reposition popup when resized.
|
|
144
|
+
fromResize$(overlayRef.overlayElement)
|
|
145
|
+
.pipe(observeIn(fn => this._zone.run(fn)), takeUntilDestroyed(popupDestroyRef))
|
|
146
|
+
.subscribe(() => {
|
|
143
147
|
overlayRef.updatePosition();
|
|
144
148
|
});
|
|
145
|
-
// Reposition the popup when its size changes (if necessary).
|
|
146
|
-
fromDimension$(overlayRef.overlayElement)
|
|
147
|
-
.pipe(takeUntilDestroyed(popup.destroyRef))
|
|
148
|
-
.subscribe(() => overlayRef.updatePosition());
|
|
149
149
|
// Close the popup depending on the passed config.
|
|
150
150
|
this.installPopupCloser(config, popupElement, popup, contextualView);
|
|
151
151
|
// Hide the popup when detaching the contextual view, if any.
|
|
@@ -153,62 +153,48 @@ export class PopupService {
|
|
|
153
153
|
this.hidePopupOnViewDetach(overlayRef, contextualView, popup);
|
|
154
154
|
}
|
|
155
155
|
// Dispose the popup when closing it.
|
|
156
|
-
|
|
156
|
+
popupDestroyRef.onDestroy(() => {
|
|
157
157
|
overlayRef.dispose();
|
|
158
158
|
});
|
|
159
159
|
return new Promise((resolve, reject) => {
|
|
160
|
-
|
|
161
|
-
|
|
162
|
-
reject(popup.result.error);
|
|
163
|
-
}
|
|
164
|
-
else {
|
|
165
|
-
resolve(popup.result);
|
|
166
|
-
}
|
|
160
|
+
popupDestroyRef.onDestroy(() => {
|
|
161
|
+
popup.result instanceof Error ? reject(popup.result) : resolve(popup.result);
|
|
167
162
|
});
|
|
168
163
|
});
|
|
169
164
|
}
|
|
170
|
-
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
|
|
178
|
-
break;
|
|
179
|
-
case EAST:
|
|
180
|
-
overlayRef.overlayElement.classList.add('wb-east');
|
|
181
|
-
break;
|
|
182
|
-
case WEST:
|
|
183
|
-
overlayRef.overlayElement.classList.add('wb-west');
|
|
184
|
-
break;
|
|
185
|
-
}
|
|
165
|
+
/**
|
|
166
|
+
* Creates the popup handle.
|
|
167
|
+
*/
|
|
168
|
+
createPopup(config, context) {
|
|
169
|
+
// Construct the handle in an injection context that shares the popup's lifecycle, allowing for automatic cleanup of effects and RxJS interop functions.
|
|
170
|
+
const popupId = config.id ?? UUID.randomUUID();
|
|
171
|
+
const popupEnvironmentInjector = createEnvironmentInjector([provideViewContext(context.view)], this._environmentInjector, `Workbench Popup ${popupId}`);
|
|
172
|
+
return runInInjectionContext(popupEnvironmentInjector, () => new ɵPopup(popupId, config));
|
|
186
173
|
}
|
|
187
174
|
/**
|
|
188
175
|
* Closes the popup depending on the configured popup closing strategy.
|
|
189
176
|
*/
|
|
190
177
|
installPopupCloser(config, popupElement, popup, contextualView) {
|
|
178
|
+
const popupDestroyRef = popup.injector.get(DestroyRef);
|
|
191
179
|
// Close the popup on escape keystroke.
|
|
192
180
|
if (config.closeStrategy?.onEscape ?? true) {
|
|
193
181
|
fromEvent(popupElement, 'keydown')
|
|
194
|
-
.pipe(filter((event) => event.key === 'Escape'),
|
|
182
|
+
.pipe(subscribeIn(fn => this._zone.runOutsideAngular(fn)), filter((event) => event.key === 'Escape'), observeIn(fn => this._zone.run(fn)), takeUntilDestroyed(popupDestroyRef))
|
|
195
183
|
.subscribe(() => popup.close());
|
|
196
184
|
}
|
|
197
185
|
// Close the popup on focus loss.
|
|
198
186
|
if (config.closeStrategy?.onFocusLost ?? true) {
|
|
199
187
|
this._focusManager.monitor(popupElement, true)
|
|
200
|
-
.pipe(filter((focusOrigin) => !focusOrigin), takeUntilDestroyed(
|
|
201
|
-
.subscribe(() => popup.close());
|
|
188
|
+
.pipe(filter((focusOrigin) => !focusOrigin), takeUntilDestroyed(popupDestroyRef))
|
|
189
|
+
.subscribe(() => popup.close(popup.result));
|
|
202
190
|
}
|
|
203
191
|
// Close the popup when closing the view.
|
|
204
192
|
if (contextualView) {
|
|
205
|
-
|
|
206
|
-
.
|
|
207
|
-
|
|
208
|
-
if (!views.some(view => view.id === contextualView.id)) {
|
|
209
|
-
popup.close();
|
|
193
|
+
effect(() => {
|
|
194
|
+
if (!this._viewRegistry.objects().some(view => view.id === contextualView.id)) {
|
|
195
|
+
untracked(() => popup.close());
|
|
210
196
|
}
|
|
211
|
-
});
|
|
197
|
+
}, { injector: popup.injector });
|
|
212
198
|
}
|
|
213
199
|
}
|
|
214
200
|
/**
|
|
@@ -217,55 +203,63 @@ export class PopupService {
|
|
|
217
203
|
*/
|
|
218
204
|
hidePopupOnViewDetach(overlayRef, contextualView, popup) {
|
|
219
205
|
overlayRef.overlayElement.classList.add('wb-view-context');
|
|
206
|
+
const popupDestroyRef = popup.injector.get(DestroyRef);
|
|
220
207
|
let activeElement;
|
|
221
|
-
|
|
222
|
-
.
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
226
|
-
|
|
227
|
-
|
|
228
|
-
|
|
229
|
-
|
|
230
|
-
|
|
231
|
-
|
|
208
|
+
effect(() => {
|
|
209
|
+
const attached = contextualView.portal.attached();
|
|
210
|
+
untracked(() => {
|
|
211
|
+
if (attached) {
|
|
212
|
+
overlayRef.overlayElement.classList.add('wb-view-attached');
|
|
213
|
+
activeElement?.focus();
|
|
214
|
+
}
|
|
215
|
+
else {
|
|
216
|
+
overlayRef.overlayElement.classList.remove('wb-view-attached');
|
|
217
|
+
}
|
|
218
|
+
});
|
|
219
|
+
}, { injector: popup.injector });
|
|
232
220
|
// Track the focus in the popup to restore it when attaching the popup.
|
|
233
221
|
fromEvent(overlayRef.overlayElement, 'focusin')
|
|
234
|
-
.pipe(
|
|
222
|
+
.pipe(subscribeIn(fn => this._zone.runOutsideAngular(fn)), takeUntilDestroyed(popupDestroyRef))
|
|
235
223
|
.subscribe(() => {
|
|
236
224
|
activeElement = this._document.activeElement instanceof HTMLElement ? this._document.activeElement : undefined;
|
|
237
225
|
});
|
|
238
226
|
}
|
|
239
227
|
/**
|
|
240
|
-
*
|
|
241
|
-
*
|
|
242
|
-
* The Observable emits the anchor's initial position, and each time when its position changes.
|
|
228
|
+
* Creates a signal that tracks the position of the popup anchor.
|
|
243
229
|
*/
|
|
244
|
-
|
|
230
|
+
trackPopupOrigin(config, contextualView, injector) {
|
|
231
|
+
const hostBounds = boundingClientRect(contextualView?.portal.componentRef.location.nativeElement ?? document.documentElement, { injector });
|
|
245
232
|
if (config.anchor instanceof Element || config.anchor instanceof ElementRef) {
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
251
|
-
|
|
252
|
-
|
|
253
|
-
|
|
254
|
-
|
|
255
|
-
|
|
256
|
-
|
|
257
|
-
|
|
258
|
-
|
|
259
|
-
|
|
260
|
-
viewportBounds$(view) {
|
|
261
|
-
if (view) {
|
|
262
|
-
return fromDimension$(view.portal.componentRef.location.nativeElement)
|
|
263
|
-
.pipe(map(dimension => dimension.element.getBoundingClientRect()), filter(clientRect => !isNullClientRect(clientRect)), // Omit viewport change if not having a size, for example, if the view is deactivated.
|
|
264
|
-
map(clientRect => ({ x: clientRect.left, y: clientRect.top, width: clientRect.width, height: clientRect.height })), startWithNullBoundsIf(() => !view.active));
|
|
233
|
+
const anchor = coerceElement(config.anchor);
|
|
234
|
+
const anchorBounds = boundingClientRect(anchor, { injector });
|
|
235
|
+
return computed(() => {
|
|
236
|
+
// Maintain position and size when detached to prevent flickering when attached again and to support for virtual scrolling in popup content.
|
|
237
|
+
if (contextualView && !contextualView.portal.attached()) {
|
|
238
|
+
return undefined;
|
|
239
|
+
}
|
|
240
|
+
// IMPORTANT: Track anchor and host bounds only if attached to prevent flickering.
|
|
241
|
+
// The `boundingClientRect` signal does not detect position changes when the element is scrolled out of view.
|
|
242
|
+
// Consequently, if the popup anchor is scrolled out of view and the view is enlarged, the popup may not align
|
|
243
|
+
// with the view boundaries. Therefore, we read the anchor's bounding box directly from the DOM.
|
|
244
|
+
anchorBounds();
|
|
245
|
+
return constrainClientRect(anchor.getBoundingClientRect(), hostBounds());
|
|
246
|
+
}, { equal: isEqualDomRect });
|
|
265
247
|
}
|
|
266
248
|
else {
|
|
267
|
-
|
|
268
|
-
|
|
249
|
+
const anchorBounds = toSignal(Observables.coerce(config.anchor), { injector });
|
|
250
|
+
return computed(() => {
|
|
251
|
+
// Maintain position and size when detached to prevent flickering when attached again and to support for virtual scrolling in popup content.
|
|
252
|
+
if (contextualView && !contextualView.portal.attached()) {
|
|
253
|
+
return undefined;
|
|
254
|
+
}
|
|
255
|
+
// IMPORTANT: Track anchor and host bounds only if attached to prevent flickering.
|
|
256
|
+
if (!anchorBounds()) {
|
|
257
|
+
return undefined;
|
|
258
|
+
}
|
|
259
|
+
const { x, y } = mapToPageCoordinates(anchorBounds(), hostBounds());
|
|
260
|
+
const { width, height } = anchorBounds();
|
|
261
|
+
return constrainClientRect(new DOMRect(x, y, width, height), hostBounds());
|
|
262
|
+
}, { equal: isEqualDomRect });
|
|
269
263
|
}
|
|
270
264
|
}
|
|
271
265
|
/**
|
|
@@ -280,63 +274,64 @@ export class PopupService {
|
|
|
280
274
|
}
|
|
281
275
|
return null;
|
|
282
276
|
}
|
|
283
|
-
|
|
284
|
-
* Maps the passed popup origin, which is relative to the given viewport, to a page coordinate.
|
|
285
|
-
*/
|
|
286
|
-
mapPopupOriginToPageCoordinate(origin, viewportBounds) {
|
|
287
|
-
const xy = origin;
|
|
288
|
-
if (xy.x !== undefined && xy.y !== undefined) {
|
|
289
|
-
return {
|
|
290
|
-
x: viewportBounds.x + xy.x,
|
|
291
|
-
y: viewportBounds.y + xy.y,
|
|
292
|
-
};
|
|
293
|
-
}
|
|
294
|
-
const topLeft = origin;
|
|
295
|
-
if (topLeft.top !== undefined && topLeft.left !== undefined) {
|
|
296
|
-
return {
|
|
297
|
-
x: viewportBounds.x + topLeft.left,
|
|
298
|
-
y: viewportBounds.y + topLeft.top,
|
|
299
|
-
};
|
|
300
|
-
}
|
|
301
|
-
const topRight = origin;
|
|
302
|
-
if (topRight.top !== undefined && topRight.right !== undefined) {
|
|
303
|
-
return {
|
|
304
|
-
x: viewportBounds.x + (viewportBounds.width - topRight.right),
|
|
305
|
-
y: viewportBounds.y + topRight.top,
|
|
306
|
-
};
|
|
307
|
-
}
|
|
308
|
-
const bottomLeft = origin;
|
|
309
|
-
if (bottomLeft.bottom !== undefined && bottomLeft.left !== undefined) {
|
|
310
|
-
return {
|
|
311
|
-
x: viewportBounds.x + bottomLeft.left,
|
|
312
|
-
y: viewportBounds.y + (viewportBounds.height - bottomLeft.bottom),
|
|
313
|
-
};
|
|
314
|
-
}
|
|
315
|
-
const bottomRight = origin;
|
|
316
|
-
if (bottomRight.bottom !== undefined && bottomRight.right !== undefined) {
|
|
317
|
-
return {
|
|
318
|
-
x: viewportBounds.x + (viewportBounds.width - bottomRight.right),
|
|
319
|
-
y: viewportBounds.y + (viewportBounds.height - bottomRight.bottom),
|
|
320
|
-
};
|
|
321
|
-
}
|
|
322
|
-
throw Error('[PopupOriginError] Illegal popup origin; must be "Point", "TopLeftPoint", "TopRightPoint", "BottomLeftPoint" or "BottomRightPoint".');
|
|
323
|
-
}
|
|
324
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PopupService, deps: [{ token: i0.Injector }, { token: i1.Overlay }, { token: i2.FocusMonitor }, { token: i3.WorkbenchViewRegistry }, { token: i0.NgZone }, { token: DOCUMENT }, { token: i4.ɵWorkbenchView, optional: true }], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
277
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PopupService, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
325
278
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PopupService, providedIn: 'root' }); }
|
|
326
279
|
}
|
|
327
280
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PopupService, decorators: [{
|
|
328
281
|
type: Injectable,
|
|
329
282
|
args: [{ providedIn: 'root' }]
|
|
330
|
-
}]
|
|
331
|
-
|
|
332
|
-
|
|
333
|
-
|
|
334
|
-
|
|
335
|
-
|
|
336
|
-
|
|
337
|
-
|
|
283
|
+
}] });
|
|
284
|
+
/**
|
|
285
|
+
* Maps given view coordinates to absolute page coordinates.
|
|
286
|
+
*/
|
|
287
|
+
function mapToPageCoordinates(origin, relativeTo) {
|
|
288
|
+
const xy = origin;
|
|
289
|
+
if (xy.x !== undefined && xy.y !== undefined) {
|
|
290
|
+
return {
|
|
291
|
+
x: relativeTo.x + xy.x,
|
|
292
|
+
y: relativeTo.y + xy.y,
|
|
293
|
+
};
|
|
294
|
+
}
|
|
295
|
+
const topLeft = origin;
|
|
296
|
+
if (topLeft.top !== undefined && topLeft.left !== undefined) {
|
|
297
|
+
return {
|
|
298
|
+
x: relativeTo.x + topLeft.left,
|
|
299
|
+
y: relativeTo.y + topLeft.top,
|
|
300
|
+
};
|
|
301
|
+
}
|
|
302
|
+
const topRight = origin;
|
|
303
|
+
if (topRight.top !== undefined && topRight.right !== undefined) {
|
|
304
|
+
return {
|
|
305
|
+
x: relativeTo.x + (relativeTo.width - topRight.right),
|
|
306
|
+
y: relativeTo.y + topRight.top,
|
|
307
|
+
};
|
|
308
|
+
}
|
|
309
|
+
const bottomLeft = origin;
|
|
310
|
+
if (bottomLeft.bottom !== undefined && bottomLeft.left !== undefined) {
|
|
311
|
+
return {
|
|
312
|
+
x: relativeTo.x + bottomLeft.left,
|
|
313
|
+
y: relativeTo.y + (relativeTo.height - bottomLeft.bottom),
|
|
314
|
+
};
|
|
315
|
+
}
|
|
316
|
+
const bottomRight = origin;
|
|
317
|
+
if (bottomRight.bottom !== undefined && bottomRight.right !== undefined) {
|
|
318
|
+
return {
|
|
319
|
+
x: relativeTo.x + (relativeTo.width - bottomRight.right),
|
|
320
|
+
y: relativeTo.y + (relativeTo.height - bottomRight.bottom),
|
|
321
|
+
};
|
|
322
|
+
}
|
|
323
|
+
throw Error('[PopupOriginError] Illegal popup origin; must be "Point", "TopLeftPoint", "TopRightPoint", "BottomLeftPoint" or "BottomRightPoint".');
|
|
324
|
+
}
|
|
325
|
+
function constrainClientRect(clientRect, constraints) {
|
|
326
|
+
const top = clamp(clientRect.top, { min: constraints.top, max: constraints.bottom });
|
|
327
|
+
const right = clamp(clientRect.right, { min: constraints.left, max: constraints.right });
|
|
328
|
+
const bottom = clamp(clientRect.bottom, { min: constraints.top, max: constraints.bottom });
|
|
329
|
+
const left = clamp(clientRect.left, { min: constraints.left, max: constraints.right });
|
|
330
|
+
const width = right - left;
|
|
331
|
+
const height = bottom - top;
|
|
332
|
+
return new DOMRect(left, top, width, height);
|
|
338
333
|
}
|
|
339
|
-
function
|
|
340
|
-
return
|
|
334
|
+
function isEqualDomRect(a, b) {
|
|
335
|
+
return a?.top === b?.top && a?.right === b?.right && a?.bottom === b?.bottom && a?.left === b?.left;
|
|
341
336
|
}
|
|
342
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popup.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/popup/popup.service.ts"],"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 {ElementRef, Inject, Injectable, Injector, NgZone, Optional, runInInjectionContext} from '@angular/core';\nimport {ConnectedOverlayPositionChange, ConnectedPosition, FlexibleConnectedPositionStrategyOrigin, Overlay, OverlayConfig, OverlayRef} from '@angular/cdk/overlay';\nimport {combineLatestWith, firstValueFrom, fromEvent, identity, MonoTypeOperatorFunction, Observable} from 'rxjs';\nimport {distinctUntilChanged, filter, map, shareReplay, startWith} from 'rxjs/operators';\nimport {ComponentPortal} from '@angular/cdk/portal';\nimport {Popup, PopupConfig, PopupReferrer, ɵPopup, ɵPopupErrorResult} from './popup.config';\nimport {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {Dictionaries, Objects, Observables} from '@scion/toolkit/util';\nimport {WorkbenchViewRegistry} from '../view/workbench-view.registry';\nimport {fromBoundingClientRect$, fromDimension$} from '@scion/toolkit/observable';\nimport {PopupComponent} from './popup.component';\nimport {DOCUMENT} from '@angular/common';\nimport {observeInside, subscribeInside} from '@scion/toolkit/operators';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {BottomLeftPoint, BottomRightPoint, Point, PopupOrigin, TopLeftPoint, TopRightPoint} from './popup.origin';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {provideViewContext} from '../view/view-context-provider';\n\nconst NORTH: ConnectedPosition = {originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom'};\nconst SOUTH: ConnectedPosition = {originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top'};\nconst WEST: ConnectedPosition = {originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center'};\nconst EAST: ConnectedPosition = {originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center'};\n\n/**\n * Allows displaying a component in a workbench popup.\n *\n * A popup is a visual workbench component for displaying content above other content. It is positioned relative to an anchor,\n * which can be either a coordinate or an HTML element. The popup moves when the anchor moves.\n *\n * Unlike views, popups are not part of the persistent Workbench navigation, meaning that popups do not survive a page reload.\n */\n@Injectable({providedIn: 'root'})\nexport class PopupService {\n\n  constructor(private _injector: Injector,\n              private _overlay: Overlay,\n              private _focusManager: FocusMonitor,\n              private _viewRegistry: WorkbenchViewRegistry,\n              private _zone: NgZone,\n              @Inject(DOCUMENT) private _document: Document,\n              @Optional() private _view?: ɵWorkbenchView) {\n  }\n\n  /**\n   * Displays the specified component in a popup.\n   *\n   * To position the popup, provide either a coordinate or an element to serve as the popup anchor.\n   *\n   * If you use an element as the popup anchor, the popup also moves when the anchor element moves. If you use a coordinate\n   * and open the popup in the context of a view, the popup opens relative to the bounds of that view. Otherwise, it\n   * is positioned relative to the page viewport. If you move or resize the view or the page, the popup will also be moved\n   * depending on the pair of coordinates used.\n   *\n   * By setting the alignment of the popup, you can control the region where to open the popup relative to its anchor.\n   *\n   * Optionally, you can pass data to the popup component. The component can inject the popup handle {@link Popup} to\n   * read input data or to close the popup.\n   *\n   * By default, the popup will close on focus loss, or when the user hits the escape key.\n   *\n   * If opening the popup in the context of a view, the popup is bound to the lifecycle of the view, that is, the popup\n   * is displayed only if the view is active and is closed when the view is closed.\n   *\n   * @param   config - Controls popup behavior\n   * @returns a promise that:\n   *          - resolves to the result if closed with a result\n   *          - resolves to `undefined` if closed without a result\n   */\n  public async open<R>(config: PopupConfig): Promise<R> {\n    // Ensure to run in Angular zone to display the popup even when called from outside of the Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.open(config));\n    }\n\n    const align = config.align || 'north';\n    const contextualView = this.resolveContextualView(config);\n    const referrer: PopupReferrer = Dictionaries.withoutUndefinedEntries({\n      viewId: contextualView?.id,\n    });\n\n    // Set up the popup positioning strategy.\n    const popupOrigin$ = this.observePopupOrigin$(config, contextualView).pipe(shareReplay({bufferSize: 1, refCount: false}));\n    const overlayPositionStrategy = this._overlay.position()\n      .flexibleConnectedTo(await firstValueFrom(popupOrigin$))\n      .withFlexibleDimensions(false)\n      .withLockedPosition(false) // If locked, the popup won't attempt to reposition itself if not enough space available.\n      .withPositions(((): ConnectedPosition[] => {\n        switch (align) {\n          case 'north':\n            return [NORTH, SOUTH, WEST, EAST];\n          case 'south':\n            return [SOUTH, NORTH, WEST, EAST];\n          case 'west':\n            return [WEST, EAST, NORTH, SOUTH];\n          case 'east':\n            return [EAST, WEST, NORTH, SOUTH];\n          default:\n            throw Error('[PopupPositionError] Illegal position; must be north, south, west or east');\n        }\n      })());\n\n    // Configure the popup overlay.\n    const overlayConfig = new OverlayConfig({\n      panelClass: [\n        'wb-popup',\n        `wb-${align}`,\n      ],\n      hasBackdrop: false,\n      positionStrategy: overlayPositionStrategy,\n      scrollStrategy: this._overlay.scrollStrategies.noop(),\n      disposeOnNavigation: true,\n    });\n\n    // Construct the popup component and attach it to the DOM.\n    const overlayRef = this._overlay.create(overlayConfig);\n    const popup = runInInjectionContext(this._injector, () => new ɵPopup(config, referrer));\n    const popupPortal = new ComponentPortal(PopupComponent, null, Injector.create({\n      parent: config.componentConstructOptions?.injector || this._injector,\n      providers: [\n        {provide: ɵPopup, useValue: popup},\n        {provide: Popup, useExisting: ɵPopup},\n        provideViewContext(contextualView),\n        ...[config.componentConstructOptions?.providers || []],\n      ],\n    }));\n    const componentRef = overlayRef.attach(popupPortal);\n    const popupElement: HTMLElement = componentRef.location.nativeElement;\n\n    // Make the popup focusable and request the focus.\n    popupElement.setAttribute('tabindex', '-1');\n    this._focusManager.focusVia(popupElement, 'program'); // To not close the popup immediately when it opens, if using the 'onFocusLost' strategy.\n\n    // Synchronize the CSS class that indicates where the popup docks to the anchor; is one of 'wb-north', 'wb-south', 'wb-east', or 'wb-west'.\n    overlayPositionStrategy.positionChanges\n      .pipe(takeUntilDestroyed(popup.destroyRef))\n      .subscribe(change => this.setPopupAlignCssClass(overlayRef, change));\n\n    // Re-position the popup when the origin moves.\n    popupOrigin$\n      .pipe(takeUntilDestroyed(popup.destroyRef))\n      .subscribe((origin: FlexibleConnectedPositionStrategyOrigin) => {\n        overlayPositionStrategy.setOrigin(origin);\n        overlayRef.updatePosition();\n      });\n\n    // Reposition the popup when its size changes (if necessary).\n    fromDimension$(overlayRef.overlayElement)\n      .pipe(takeUntilDestroyed(popup.destroyRef))\n      .subscribe(() => overlayRef.updatePosition());\n\n    // Close the popup depending on the passed config.\n    this.installPopupCloser(config, popupElement, popup, contextualView);\n\n    // Hide the popup when detaching the contextual view, if any.\n    if (contextualView) {\n      this.hidePopupOnViewDetach(overlayRef, contextualView, popup);\n    }\n\n    // Dispose the popup when closing it.\n    popup.destroyRef.onDestroy(() => {\n      overlayRef.dispose();\n    });\n\n    return new Promise<R>((resolve, reject) => {\n      popup.destroyRef.onDestroy(() => {\n        if (popup.result instanceof ɵPopupErrorResult) {\n          reject(popup.result.error);\n        }\n        else {\n          resolve(popup.result as R);\n        }\n      });\n    });\n  }\n\n  private setPopupAlignCssClass(overlayRef: OverlayRef, positionChange: ConnectedOverlayPositionChange): void {\n    overlayRef.overlayElement.classList.remove('wb-north', 'wb-south', 'wb-east', 'wb-west');\n    switch (positionChange.connectionPair) {\n      case NORTH:\n        overlayRef.overlayElement.classList.add('wb-north');\n        break;\n      case SOUTH:\n        overlayRef.overlayElement.classList.add('wb-south');\n        break;\n      case EAST:\n        overlayRef.overlayElement.classList.add('wb-east');\n        break;\n      case WEST:\n        overlayRef.overlayElement.classList.add('wb-west');\n        break;\n    }\n  }\n\n  /**\n   * Closes the popup depending on the configured popup closing strategy.\n   */\n  private installPopupCloser(config: PopupConfig, popupElement: HTMLElement, popup: ɵPopup, contextualView: ɵWorkbenchView | null): void {\n    // Close the popup on escape keystroke.\n    if (config.closeStrategy?.onEscape ?? true) {\n      fromEvent<KeyboardEvent>(popupElement, 'keydown')\n        .pipe(\n          filter((event: KeyboardEvent) => event.key === 'Escape'),\n          subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)),\n          observeInside(continueFn => this._zone.run(continueFn)),\n          takeUntilDestroyed(popup.destroyRef),\n        )\n        .subscribe(() => popup.close());\n    }\n\n    // Close the popup on focus loss.\n    if (config.closeStrategy?.onFocusLost ?? true) {\n      this._focusManager.monitor(popupElement, true)\n        .pipe(\n          filter((focusOrigin: FocusOrigin) => !focusOrigin),\n          takeUntilDestroyed(popup.destroyRef),\n        )\n        .subscribe(() => popup.close());\n    }\n\n    // Close the popup when closing the view.\n    if (contextualView) {\n      this._viewRegistry.views$\n        .pipe(takeUntilDestroyed(popup.destroyRef))\n        .subscribe(views => {\n          if (!views.some(view => view.id === contextualView.id)) {\n            popup.close();\n          }\n        });\n    }\n  }\n\n  /**\n   * Hides the popup when its contextual view is detached, and displays it when it is reattached. Also restores the focus on re-activation.\n   * The contextual view is detached if not active, or located in the peripheral area and the main area is maximized.\n   */\n  private hidePopupOnViewDetach(overlayRef: OverlayRef, contextualView: ɵWorkbenchView, popup: ɵPopup): void {\n    overlayRef.overlayElement.classList.add('wb-view-context');\n\n    let activeElement: HTMLElement | undefined;\n\n    contextualView.portal.attached$\n      .pipe(takeUntilDestroyed(popup.destroyRef))\n      .subscribe(attached => {\n        if (attached) {\n          overlayRef.overlayElement.classList.add('wb-view-attached');\n          activeElement?.focus();\n        }\n        else {\n          overlayRef.overlayElement.classList.remove('wb-view-attached');\n        }\n      });\n\n    // Track the focus in the popup to restore it when attaching the popup.\n    fromEvent(overlayRef.overlayElement, 'focusin')\n      .pipe(\n        subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)),\n        takeUntilDestroyed(popup.destroyRef),\n      )\n      .subscribe(() => {\n        activeElement = this._document.activeElement instanceof HTMLElement ? this._document.activeElement : undefined;\n      });\n  }\n\n  /**\n   * Observes the position of the popup anchor.\n   *\n   * The Observable emits the anchor's initial position, and each time when its position changes.\n   */\n  private observePopupOrigin$(config: PopupConfig, contextualView: ɵWorkbenchView | null): Observable<FlexibleConnectedPositionStrategyOrigin> {\n    if (config.anchor instanceof Element || config.anchor instanceof ElementRef) {\n      return fromBoundingClientRect$(coerceElement<HTMLElement>(config.anchor as HTMLElement))\n        .pipe(map(clientRect => ({x: clientRect.x, y: clientRect.y, width: clientRect.width, height: clientRect.height})));\n    }\n    else {\n      return Observables.coerce(config.anchor)\n        .pipe(\n          combineLatestWith(this.viewportBounds$(contextualView)),\n          map(([popupOrigin, viewportBounds]) => {\n            const {x, y} = this.mapPopupOriginToPageCoordinate(popupOrigin, viewportBounds);\n            return {x, y, width: popupOrigin.width, height: popupOrigin.height};\n          }),\n          distinctUntilChanged(Objects.isEqual),\n        );\n    }\n  }\n\n  /**\n   * Observes the bounds of the viewport (view or page) in which the popup has been opened.\n   */\n  private viewportBounds$(view: ɵWorkbenchView | null): Observable<ViewportBounds> {\n    if (view) {\n      return fromDimension$(view.portal.componentRef.location.nativeElement)\n        .pipe(\n          map(dimension => dimension.element.getBoundingClientRect()),\n          filter(clientRect => !isNullClientRect(clientRect)), // Omit viewport change if not having a size, for example, if the view is deactivated.\n          map(clientRect => ({x: clientRect.left, y: clientRect.top, width: clientRect.width, height: clientRect.height})),\n          startWithNullBoundsIf(() => !view.active), // Ensure initial bounds to be emitted even if the view is inactive. Otherwise, the popup would not be attached to the DOM until the view is activated.\n        );\n    }\n    else {\n      return fromEvent(window, 'resize')\n        .pipe(\n          startWith(undefined as void),\n          map(() => ({x: 0, y: 0, width: window.innerWidth, height: window.innerHeight})),\n        );\n    }\n  }\n\n  /**\n   * Resolves the contextual view to which the popup is bound.\n   */\n  private resolveContextualView(config: PopupConfig): ɵWorkbenchView | null {\n    if (config.context?.viewId) {\n      return this._viewRegistry.get(config.context.viewId);\n    }\n    if (config.context?.viewId === undefined) { // `null` means to open the popup outside the contextual view\n      return this._view ?? null;\n    }\n    return null;\n  }\n\n  /**\n   * Maps the passed popup origin, which is relative to the given viewport, to a page coordinate.\n   */\n  private mapPopupOriginToPageCoordinate(origin: PopupOrigin, viewportBounds: ViewportBounds): Point {\n    const xy = origin as Point;\n    if (xy.x !== undefined && xy.y !== undefined) {\n      return {\n        x: viewportBounds.x + xy.x,\n        y: viewportBounds.y + xy.y,\n      };\n    }\n    const topLeft = origin as TopLeftPoint;\n    if (topLeft.top !== undefined && topLeft.left !== undefined) {\n      return {\n        x: viewportBounds.x + topLeft.left,\n        y: viewportBounds.y + topLeft.top,\n      };\n    }\n    const topRight = origin as TopRightPoint;\n    if (topRight.top !== undefined && topRight.right !== undefined) {\n      return {\n        x: viewportBounds.x + (viewportBounds.width - topRight.right),\n        y: viewportBounds.y + topRight.top,\n      };\n    }\n    const bottomLeft = origin as BottomLeftPoint;\n    if (bottomLeft.bottom !== undefined && bottomLeft.left !== undefined) {\n      return {\n        x: viewportBounds.x + bottomLeft.left,\n        y: viewportBounds.y + (viewportBounds.height - bottomLeft.bottom),\n      };\n    }\n    const bottomRight = origin as BottomRightPoint;\n    if (bottomRight.bottom !== undefined && bottomRight.right !== undefined) {\n      return {\n        x: viewportBounds.x + (viewportBounds.width - bottomRight.right),\n        y: viewportBounds.y + (viewportBounds.height - bottomRight.bottom),\n      };\n    }\n    throw Error('[PopupOriginError] Illegal popup origin; must be \"Point\", \"TopLeftPoint\", \"TopRightPoint\", \"BottomLeftPoint\" or \"BottomRightPoint\".');\n  }\n}\n\nfunction isNullClientRect(clientRect: DOMRect): boolean {\n  return clientRect.top === 0 && clientRect.right === 0 && clientRect.bottom === 0 && clientRect.left === 0;\n}\n\nfunction startWithNullBoundsIf(condition: () => boolean): MonoTypeOperatorFunction<ViewportBounds> {\n  return condition() ? startWith({x: 0, y: 0, width: 0, height: 0}) : identity;\n}\n\ninterface ViewportBounds {\n  x: number;\n  y: number;\n  width: number;\n  height: number;\n}\n"]}
|
|
337
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popup.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/popup/popup.service.ts"],"names":[],"mappings":"AAAA;;;;;;;;GAQG;AAEH,OAAO,EAAC,0BAA0B,EAAE,QAAQ,EAAE,yBAAyB,EAAE,UAAU,EAAE,MAAM,EAAE,UAAU,EAAE,mBAAmB,EAAE,MAAM,EAAE,UAAU,EAAE,QAAQ,EAAE,MAAM,EAAE,qBAAqB,EAAU,SAAS,EAAC,MAAM,eAAe,CAAC;AACnO,OAAO,EAAoB,OAAO,EAAE,aAAa,EAAa,MAAM,sBAAsB,CAAC;AAC3F,OAAO,EAAC,cAAc,EAAE,SAAS,EAAC,MAAM,MAAM,CAAC;AAC/C,OAAO,EAAC,MAAM,EAAC,MAAM,gBAAgB,CAAC;AACtC,OAAO,EAAC,eAAe,EAAC,MAAM,qBAAqB,CAAC;AACpD,OAAO,EAAC,KAAK,EAAe,MAAM,EAAC,MAAM,gBAAgB,CAAC;AAC1D,OAAO,EAAC,YAAY,EAAc,MAAM,mBAAmB,CAAC;AAC5D,OAAO,EAAC,WAAW,EAAC,MAAM,qBAAqB,CAAC;AAChD,OAAO,EAAC,uBAAuB,EAAC,MAAM,iCAAiC,CAAC;AACxE,OAAO,EAAC,WAAW,EAAC,MAAM,2BAA2B,CAAC;AACtD,OAAO,EAAC,cAAc,EAAC,MAAM,mBAAmB,CAAC;AACjD,OAAO,EAAC,QAAQ,EAAC,MAAM,iBAAiB,CAAC;AACzC,OAAO,EAAC,SAAS,EAAE,WAAW,EAAC,MAAM,0BAA0B,CAAC;AAChE,OAAO,EAAC,cAAc,EAAC,MAAM,+BAA+B,CAAC;AAE7D,OAAO,EAAC,kBAAkB,EAAE,YAAY,EAAE,QAAQ,EAAC,MAAM,4BAA4B,CAAC;AACtF,OAAO,EAAC,kBAAkB,EAAC,MAAM,+BAA+B,CAAC;AACjE,OAAO,EAAC,IAAI,EAAC,MAAM,qBAAqB,CAAC;AACzC,OAAO,EAAC,kBAAkB,EAAC,MAAM,6BAA6B,CAAC;AAC/D,OAAO,EAAC,aAAa,EAAC,MAAM,uBAAuB,CAAC;AACpD,OAAO,EAAC,KAAK,EAAC,MAAM,qBAAqB,CAAC;;AAE1C,MAAM,KAAK,GAAsB,EAAC,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC;AACrI,MAAM,KAAK,GAAsB,EAAC,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,UAAU,EAAE,UAAU,EAAC,CAAC;AACrI,MAAM,IAAI,GAAsB,EAAC,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,QAAQ,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAC,CAAC;AAClI,MAAM,IAAI,GAAsB,EAAC,OAAO,EAAE,KAAK,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,OAAO,EAAE,QAAQ,EAAE,QAAQ,EAAE,UAAU,EAAE,SAAS,EAAC,CAAC;AAElI;;;;;;;GAOG;AAEH,MAAM,OAAO,YAAY;IADzB;QAGmB,yBAAoB,GAAG,MAAM,CAAC,mBAAmB,CAAC,CAAC;QACnD,aAAQ,GAAG,MAAM,CAAC,OAAO,CAAC,CAAC;QAC3B,kBAAa,GAAG,MAAM,CAAC,YAAY,CAAC,CAAC;QACrC,kBAAa,GAAG,MAAM,CAAC,uBAAuB,CAAC,CAAC;QAChD,UAAK,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QACvB,cAAS,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;QAC7B,UAAK,GAAG,MAAM,CAAC,cAAc,EAAE,EAAC,QAAQ,EAAE,IAAI,EAAC,CAAC,CAAC;KA8QnE;IA5QC;;;;;;;;;;;;;;;;;;;;;;;;OAwBG;IACI,KAAK,CAAC,IAAI,CAAI,MAAmB;QACtC,0BAA0B,CAAC,IAAI,CAAC,IAAI,EAAE,sHAAsH,CAAC,CAAC;QAE9J,wGAAwG;QACxG,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC,CAAC;QACjD,CAAC;QAED,MAAM,KAAK,GAAG,MAAM,CAAC,KAAK,IAAI,OAAO,CAAC;QACtC,MAAM,cAAc,GAAG,IAAI,CAAC,qBAAqB,CAAC,MAAM,CAAC,CAAC;QAC1D,MAAM,KAAK,GAAG,IAAI,CAAC,WAAW,CAAC,MAAM,EAAE,EAAC,IAAI,EAAE,cAAc,EAAC,CAAC,CAAC;QAC/D,MAAM,eAAe,GAAG,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QACvD,MAAM,WAAW,GAAG,IAAI,CAAC,gBAAgB,CAAC,MAAM,EAAE,cAAc,EAAE,KAAK,CAAC,QAAQ,CAAC,CAAC;QAElF,iCAAiC;QACjC,MAAM,eAAe,GAAG,MAAM,cAAc,CAAC,YAAY,CAAC,WAAW,EAAE,EAAC,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAC,CAAC,CAAC,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;QAE1H,MAAM,uBAAuB,GAAG,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE;aACrD,mBAAmB,CAAC,eAAe,CAAC;aACpC,sBAAsB,CAAC,KAAK,CAAC;aAC7B,kBAAkB,CAAC,KAAK,CAAC,CAAC,yFAAyF;aACnH,aAAa,CAAC,CAAC,GAAwB,EAAE;YACxC,QAAQ,KAAK,EAAE,CAAC;gBACd,KAAK,OAAO;oBACV,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBACpC,KAAK,OAAO;oBACV,OAAO,CAAC,KAAK,EAAE,KAAK,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBACpC,KAAK,MAAM;oBACT,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;gBACpC,KAAK,MAAM;oBACT,OAAO,CAAC,IAAI,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,CAAC,CAAC;gBACpC;oBACE,MAAM,KAAK,CAAC,2EAA2E,CAAC,CAAC;YAC7F,CAAC;QACH,CAAC,CAAC,EAAE,CAAC,CAAC;QAER,+BAA+B;QAC/B,MAAM,aAAa,GAAG,IAAI,aAAa,CAAC;YACtC,UAAU,EAAE,UAAU;YACtB,WAAW,EAAE,KAAK;YAClB,gBAAgB,EAAE,uBAAuB;YACzC,cAAc,EAAE,IAAI,CAAC,QAAQ,CAAC,gBAAgB,CAAC,IAAI,EAAE;YACrD,mBAAmB,EAAE,IAAI;SAC1B,CAAC,CAAC;QAEH,0DAA0D;QAC1D,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,aAAa,CAAC,CAAC;QACvD,MAAM,WAAW,GAAG,IAAI,eAAe,CAAC,cAAc,EAAE,IAAI,EAAE,QAAQ,CAAC,MAAM,CAAC;YAC5E,MAAM,EAAE,MAAM,CAAC,yBAAyB,EAAE,QAAQ,IAAI,IAAI,CAAC,oBAAoB;YAC/E,SAAS,EAAE;gBACT,EAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,KAAK,EAAC;gBAClC,EAAC,OAAO,EAAE,KAAK,EAAE,WAAW,EAAE,MAAM,EAAC;gBACrC,kBAAkB,CAAC,cAAc,CAAC;gBAClC,GAAG,CAAC,MAAM,CAAC,yBAAyB,EAAE,SAAS,IAAI,EAAE,CAAC;aACvD;SACF,CAAC,CAAC,CAAC;QACJ,MAAM,YAAY,GAAG,UAAU,CAAC,MAAM,CAAC,WAAW,CAAC,CAAC;QACpD,MAAM,YAAY,GAAgB,YAAY,CAAC,QAAQ,CAAC,aAAa,CAAC;QAEtE,kDAAkD;QAClD,YAAY,CAAC,YAAY,CAAC,UAAU,EAAE,IAAI,CAAC,CAAC;QAC5C,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,YAAY,EAAE,SAAS,CAAC,CAAC,CAAC,yFAAyF;QAE/I,6CAA6C;QAC7C,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,MAAM,GAAG,WAAW,EAAE,CAAC;YAC7B,SAAS,CAAC,GAAG,EAAE;gBACb,IAAI,MAAM,EAAE,CAAC;oBACX,uBAAuB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;oBAC1C,UAAU,CAAC,cAAc,EAAE,CAAC;gBAC9B,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,EAAC,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAC,CAAC,CAAC;QAE/B,iCAAiC;QACjC,WAAW,CAAC,UAAU,CAAC,cAAc,CAAC;aACnC,IAAI,CACH,SAAS,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,EACnC,kBAAkB,CAAC,eAAe,CAAC,CACpC;aACA,SAAS,CAAC,GAAG,EAAE;YACd,UAAU,CAAC,cAAc,EAAE,CAAC;QAC9B,CAAC,CAAC,CAAC;QAEL,kDAAkD;QAClD,IAAI,CAAC,kBAAkB,CAAC,MAAM,EAAE,YAAY,EAAE,KAAK,EAAE,cAAc,CAAC,CAAC;QAErE,6DAA6D;QAC7D,IAAI,cAAc,EAAE,CAAC;YACnB,IAAI,CAAC,qBAAqB,CAAC,UAAU,EAAE,cAAc,EAAE,KAAK,CAAC,CAAC;QAChE,CAAC;QAED,qCAAqC;QACrC,eAAe,CAAC,SAAS,CAAC,GAAG,EAAE;YAC7B,UAAU,CAAC,OAAO,EAAE,CAAC;QACvB,CAAC,CAAC,CAAC;QAEH,OAAO,IAAI,OAAO,CAAgB,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACpD,eAAe,CAAC,SAAS,CAAC,GAAG,EAAE;gBAC7B,KAAK,CAAC,MAAM,YAAY,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,KAAK,CAAC,MAAW,CAAC,CAAC;YACpF,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;OAEG;IACK,WAAW,CAAI,MAAmB,EAAE,OAAsC;QAChF,wJAAwJ;QACxJ,MAAM,OAAO,GAAG,MAAM,CAAC,EAAE,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;QAC/C,MAAM,wBAAwB,GAAG,yBAAyB,CAAC,CAAC,kBAAkB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,EAAE,IAAI,CAAC,oBAAoB,EAAE,mBAAmB,OAAO,EAAE,CAAC,CAAC;QACxJ,OAAO,qBAAqB,CAAC,wBAAwB,EAAE,GAAG,EAAE,CAAC,IAAI,MAAM,CAAI,OAAO,EAAE,MAAM,CAAC,CAAC,CAAC;IAC/F,CAAC;IAED;;OAEG;IACK,kBAAkB,CAAC,MAAmB,EAAE,YAAyB,EAAE,KAAa,EAAE,cAAqC;QAC7H,MAAM,eAAe,GAAG,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QAEvD,uCAAuC;QACvC,IAAI,MAAM,CAAC,aAAa,EAAE,QAAQ,IAAI,IAAI,EAAE,CAAC;YAC3C,SAAS,CAAgB,YAAY,EAAE,SAAS,CAAC;iBAC9C,IAAI,CACH,WAAW,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,EACnD,MAAM,CAAC,CAAC,KAAoB,EAAE,EAAE,CAAC,KAAK,CAAC,GAAG,KAAK,QAAQ,CAAC,EACxD,SAAS,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,EACnC,kBAAkB,CAAC,eAAe,CAAC,CACpC;iBACA,SAAS,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;QACpC,CAAC;QAED,iCAAiC;QACjC,IAAI,MAAM,CAAC,aAAa,EAAE,WAAW,IAAI,IAAI,EAAE,CAAC;YAC9C,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,YAAY,EAAE,IAAI,CAAC;iBAC3C,IAAI,CACH,MAAM,CAAC,CAAC,WAAwB,EAAE,EAAE,CAAC,CAAC,WAAW,CAAC,EAClD,kBAAkB,CAAC,eAAe,CAAC,CACpC;iBACA,SAAS,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC;QAChD,CAAC;QAED,yCAAyC;QACzC,IAAI,cAAc,EAAE,CAAC;YACnB,MAAM,CAAC,GAAG,EAAE;gBACV,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,OAAO,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,EAAE,KAAK,cAAc,CAAC,EAAE,CAAC,EAAE,CAAC;oBAC9E,SAAS,CAAC,GAAG,EAAE,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,CAAC;gBACjC,CAAC;YACH,CAAC,EAAE,EAAC,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAC,CAAC,CAAC;QACjC,CAAC;IACH,CAAC;IAED;;;OAGG;IACK,qBAAqB,CAAC,UAAsB,EAAE,cAA8B,EAAE,KAAa;QACjG,UAAU,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,CAAC,iBAAiB,CAAC,CAAC;QAE3D,MAAM,eAAe,GAAG,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC;QACvD,IAAI,aAAsC,CAAC;QAE3C,MAAM,CAAC,GAAG,EAAE;YACV,MAAM,QAAQ,GAAG,cAAc,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC;YAClD,SAAS,CAAC,GAAG,EAAE;gBACb,IAAI,QAAQ,EAAE,CAAC;oBACb,UAAU,CAAC,cAAc,CAAC,SAAS,CAAC,GAAG,CAAC,kBAAkB,CAAC,CAAC;oBAC5D,aAAa,EAAE,KAAK,EAAE,CAAC;gBACzB,CAAC;qBACI,CAAC;oBACJ,UAAU,CAAC,cAAc,CAAC,SAAS,CAAC,MAAM,CAAC,kBAAkB,CAAC,CAAC;gBACjE,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,EAAC,QAAQ,EAAE,KAAK,CAAC,QAAQ,EAAC,CAAC,CAAC;QAE/B,uEAAuE;QACvE,SAAS,CAAC,UAAU,CAAC,cAAc,EAAE,SAAS,CAAC;aAC5C,IAAI,CACH,WAAW,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,KAAK,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,EACnD,kBAAkB,CAAC,eAAe,CAAC,CACpC;aACA,SAAS,CAAC,GAAG,EAAE;YACd,aAAa,GAAG,IAAI,CAAC,SAAS,CAAC,aAAa,YAAY,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,CAAC,SAAS,CAAC;QACjH,CAAC,CAAC,CAAC;IACP,CAAC;IAED;;OAEG;IACK,gBAAgB,CAAC,MAAmB,EAAE,cAAqC,EAAE,QAAkB;QACrG,MAAM,UAAU,GAAG,kBAAkB,CAAC,cAAc,EAAE,MAAM,CAAC,YAAY,CAAC,QAAQ,CAAC,aAAa,IAAI,QAAQ,CAAC,eAAe,EAAE,EAAC,QAAQ,EAAC,CAAC,CAAC;QAE1I,IAAI,MAAM,CAAC,MAAM,YAAY,OAAO,IAAI,MAAM,CAAC,MAAM,YAAY,UAAU,EAAE,CAAC;YAC5E,MAAM,MAAM,GAAG,aAAa,CAAC,MAAM,CAAC,MAAM,CAAgB,CAAC;YAC3D,MAAM,YAAY,GAAG,kBAAkB,CAAC,MAAM,EAAE,EAAC,QAAQ,EAAC,CAAC,CAAC;YAC5D,OAAO,QAAQ,CAAC,GAAG,EAAE;gBACnB,4IAA4I;gBAC5I,IAAI,cAAc,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC;oBACxD,OAAO,SAAS,CAAC;gBACnB,CAAC;gBAED,kFAAkF;gBAElF,6GAA6G;gBAC7G,8GAA8G;gBAC9G,gGAAgG;gBAChG,YAAY,EAAE,CAAC;gBAEf,OAAO,mBAAmB,CAAC,MAAM,CAAC,qBAAqB,EAAE,EAAE,UAAU,EAAE,CAAC,CAAC;YAC3E,CAAC,EAAE,EAAC,KAAK,EAAE,cAAc,EAAC,CAAC,CAAC;QAC9B,CAAC;aACI,CAAC;YACJ,MAAM,YAAY,GAAG,QAAQ,CAAC,WAAW,CAAC,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE,EAAC,QAAQ,EAAC,CAAC,CAAC;YAC7E,OAAO,QAAQ,CAAC,GAAG,EAAE;gBACnB,4IAA4I;gBAC5I,IAAI,cAAc,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,QAAQ,EAAE,EAAE,CAAC;oBACxD,OAAO,SAAS,CAAC;gBACnB,CAAC;gBAED,kFAAkF;gBAClF,IAAI,CAAC,YAAY,EAAE,EAAE,CAAC;oBACpB,OAAO,SAAS,CAAC;gBACnB,CAAC;gBAED,MAAM,EAAC,CAAC,EAAE,CAAC,EAAC,GAAG,oBAAoB,CAAC,YAAY,EAAG,EAAE,UAAU,EAAE,CAAC,CAAC;gBACnE,MAAM,EAAC,KAAK,EAAE,MAAM,EAAC,GAAG,YAAY,EAAG,CAAC;gBACxC,OAAO,mBAAmB,CAAC,IAAI,OAAO,CAAC,CAAC,EAAE,CAAC,EAAE,KAAK,EAAE,MAAM,CAAC,EAAE,UAAU,EAAE,CAAC,CAAC;YAC7E,CAAC,EAAE,EAAC,KAAK,EAAE,cAAc,EAAC,CAAC,CAAC;QAC9B,CAAC;IACH,CAAC;IAED;;OAEG;IACK,qBAAqB,CAAC,MAAmB;QAC/C,IAAI,MAAM,CAAC,OAAO,EAAE,MAAM,EAAE,CAAC;YAC3B,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,MAAM,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;QACvD,CAAC;QACD,IAAI,MAAM,CAAC,OAAO,EAAE,MAAM,KAAK,SAAS,EAAE,CAAC,CAAC,6DAA6D;YACvG,OAAO,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC;QAC5B,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;8GArRU,YAAY;kHAAZ,YAAY,cADA,MAAM;;2FAClB,YAAY;kBADxB,UAAU;mBAAC,EAAC,UAAU,EAAE,MAAM,EAAC;;AAyRhC;;GAEG;AACH,SAAS,oBAAoB,CAAC,MAAmB,EAAE,UAAmB;IACpE,MAAM,EAAE,GAAG,MAAe,CAAC;IAC3B,IAAI,EAAE,CAAC,CAAC,KAAK,SAAS,IAAI,EAAE,CAAC,CAAC,KAAK,SAAS,EAAE,CAAC;QAC7C,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;YACtB,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,EAAE,CAAC,CAAC;SACvB,CAAC;IACJ,CAAC;IACD,MAAM,OAAO,GAAG,MAAsB,CAAC;IACvC,IAAI,OAAO,CAAC,GAAG,KAAK,SAAS,IAAI,OAAO,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;QAC5D,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,OAAO,CAAC,IAAI;YAC9B,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,OAAO,CAAC,GAAG;SAC9B,CAAC;IACJ,CAAC;IACD,MAAM,QAAQ,GAAG,MAAuB,CAAC;IACzC,IAAI,QAAQ,CAAC,GAAG,KAAK,SAAS,IAAI,QAAQ,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;QAC/D,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,KAAK,GAAG,QAAQ,CAAC,KAAK,CAAC;YACrD,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,QAAQ,CAAC,GAAG;SAC/B,CAAC;IACJ,CAAC;IACD,MAAM,UAAU,GAAG,MAAyB,CAAC;IAC7C,IAAI,UAAU,CAAC,MAAM,KAAK,SAAS,IAAI,UAAU,CAAC,IAAI,KAAK,SAAS,EAAE,CAAC;QACrE,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,UAAU,CAAC,IAAI;YACjC,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,MAAM,GAAG,UAAU,CAAC,MAAM,CAAC;SAC1D,CAAC;IACJ,CAAC;IACD,MAAM,WAAW,GAAG,MAA0B,CAAC;IAC/C,IAAI,WAAW,CAAC,MAAM,KAAK,SAAS,IAAI,WAAW,CAAC,KAAK,KAAK,SAAS,EAAE,CAAC;QACxE,OAAO;YACL,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,KAAK,GAAG,WAAW,CAAC,KAAK,CAAC;YACxD,CAAC,EAAE,UAAU,CAAC,CAAC,GAAG,CAAC,UAAU,CAAC,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC;SAC3D,CAAC;IACJ,CAAC;IACD,MAAM,KAAK,CAAC,qIAAqI,CAAC,CAAC;AACrJ,CAAC;AAED,SAAS,mBAAmB,CAAC,UAAmB,EAAE,WAAoB;IACpE,MAAM,GAAG,GAAG,KAAK,CAAC,UAAU,CAAC,GAAG,EAAE,EAAC,GAAG,EAAE,WAAW,CAAC,GAAG,EAAE,GAAG,EAAE,WAAW,CAAC,MAAM,EAAC,CAAC,CAAC;IACnF,MAAM,KAAK,GAAG,KAAK,CAAC,UAAU,CAAC,KAAK,EAAE,EAAC,GAAG,EAAE,WAAW,CAAC,IAAI,EAAE,GAAG,EAAE,WAAW,CAAC,KAAK,EAAC,CAAC,CAAC;IACvF,MAAM,MAAM,GAAG,KAAK,CAAC,UAAU,CAAC,MAAM,EAAE,EAAC,GAAG,EAAE,WAAW,CAAC,GAAG,EAAE,GAAG,EAAE,WAAW,CAAC,MAAM,EAAC,CAAC,CAAC;IACzF,MAAM,IAAI,GAAG,KAAK,CAAC,UAAU,CAAC,IAAI,EAAE,EAAC,GAAG,EAAE,WAAW,CAAC,IAAI,EAAE,GAAG,EAAE,WAAW,CAAC,KAAK,EAAC,CAAC,CAAC;IACrF,MAAM,KAAK,GAAG,KAAK,GAAG,IAAI,CAAC;IAC3B,MAAM,MAAM,GAAG,MAAM,GAAG,GAAG,CAAC;IAC5B,OAAO,IAAI,OAAO,CAAC,IAAI,EAAE,GAAG,EAAE,KAAK,EAAE,MAAM,CAAC,CAAC;AAC/C,CAAC;AAED,SAAS,cAAc,CAAC,CAAsB,EAAE,CAAsB;IACpE,OAAO,CAAC,EAAE,GAAG,KAAK,CAAC,EAAE,GAAG,IAAI,CAAC,EAAE,KAAK,KAAK,CAAC,EAAE,KAAK,IAAI,CAAC,EAAE,MAAM,KAAK,CAAC,EAAE,MAAM,IAAI,CAAC,EAAE,IAAI,KAAK,CAAC,EAAE,IAAI,CAAC;AACtG,CAAC","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 {assertNotInReactiveContext, computed, createEnvironmentInjector, DestroyRef, effect, ElementRef, EnvironmentInjector, inject, Injectable, Injector, NgZone, runInInjectionContext, Signal, untracked} from '@angular/core';\nimport {ConnectedPosition, Overlay, OverlayConfig, OverlayRef} from '@angular/cdk/overlay';\nimport {firstValueFrom, fromEvent} from 'rxjs';\nimport {filter} from 'rxjs/operators';\nimport {ComponentPortal} from '@angular/cdk/portal';\nimport {Popup, PopupConfig, ɵPopup} from './popup.config';\nimport {FocusMonitor, FocusOrigin} from '@angular/cdk/a11y';\nimport {Observables} from '@scion/toolkit/util';\nimport {WORKBENCH_VIEW_REGISTRY} from '../view/workbench-view.registry';\nimport {fromResize$} from '@scion/toolkit/observable';\nimport {PopupComponent} from './popup.component';\nimport {DOCUMENT} from '@angular/common';\nimport {observeIn, subscribeIn} from '@scion/toolkit/operators';\nimport {ɵWorkbenchView} from '../view/ɵworkbench-view.model';\nimport {BottomLeftPoint, BottomRightPoint, Point, PopupOrigin, TopLeftPoint, TopRightPoint} from './popup.origin';\nimport {takeUntilDestroyed, toObservable, toSignal} from '@angular/core/rxjs-interop';\nimport {provideViewContext} from '../view/view-context-provider';\nimport {UUID} from '@scion/toolkit/uuid';\nimport {boundingClientRect} from '@scion/components/dimension';\nimport {coerceElement} from '@angular/cdk/coercion';\nimport {clamp} from '../common/math.util';\n\nconst NORTH: ConnectedPosition = {originX: 'center', originY: 'top', overlayX: 'center', overlayY: 'bottom', panelClass: 'wb-north'};\nconst SOUTH: ConnectedPosition = {originX: 'center', originY: 'bottom', overlayX: 'center', overlayY: 'top', panelClass: 'wb-south'};\nconst WEST: ConnectedPosition = {originX: 'start', originY: 'center', overlayX: 'end', overlayY: 'center', panelClass: 'wb-west'};\nconst EAST: ConnectedPosition = {originX: 'end', originY: 'center', overlayX: 'start', overlayY: 'center', panelClass: 'wb-east'};\n\n/**\n * Allows displaying a component in a workbench popup.\n *\n * A popup is a visual workbench component for displaying content above other content. It is positioned relative to an anchor,\n * which can be either a coordinate or an HTML element. The popup moves when the anchor moves.\n *\n * Unlike views, popups are not part of the persistent Workbench navigation, meaning that popups do not survive a page reload.\n */\n@Injectable({providedIn: 'root'})\nexport class PopupService {\n\n  private readonly _environmentInjector = inject(EnvironmentInjector);\n  private readonly _overlay = inject(Overlay);\n  private readonly _focusManager = inject(FocusMonitor);\n  private readonly _viewRegistry = inject(WORKBENCH_VIEW_REGISTRY);\n  private readonly _zone = inject(NgZone);\n  private readonly _document = inject(DOCUMENT);\n  private readonly _view = inject(ɵWorkbenchView, {optional: true});\n\n  /**\n   * Displays the specified component in a popup.\n   *\n   * To position the popup, provide either a coordinate or an element to serve as the popup anchor.\n   *\n   * If you use an element as the popup anchor, the popup also moves when the anchor element moves. If you use a coordinate\n   * and open the popup in the context of a view, the popup opens relative to the bounds of that view. Otherwise, it\n   * is positioned relative to the page viewport. If you move or resize the view or the page, the popup will also be moved\n   * depending on the pair of coordinates used.\n   *\n   * By setting the alignment of the popup, you can control the region where to open the popup relative to its anchor.\n   *\n   * Optionally, you can pass data to the popup component. The component can inject the popup handle {@link Popup} to\n   * read input data or to close the popup.\n   *\n   * By default, the popup will close on focus loss, or when the user hits the escape key.\n   *\n   * If opening the popup in the context of a view, the popup is bound to the lifecycle of the view, that is, the popup\n   * is displayed only if the view is active and is closed when the view is closed.\n   *\n   * @param   config - Controls popup behavior\n   * @returns a promise that:\n   *          - resolves to the result if closed with a result\n   *          - resolves to `undefined` if closed without a result\n   */\n  public async open<R>(config: PopupConfig): Promise<R | undefined> {\n    assertNotInReactiveContext(this.open, 'Call WorkbenchPopupService.open() in a non-reactive (non-tracking) context, such as within the untracked() function.');\n\n    // Ensure to run in Angular zone to display the popup even when called from outside of the Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.open(config));\n    }\n\n    const align = config.align || 'north';\n    const contextualView = this.resolveContextualView(config);\n    const popup = this.createPopup(config, {view: contextualView});\n    const popupDestroyRef = popup.injector.get(DestroyRef);\n    const popupOrigin = this.trackPopupOrigin(config, contextualView, popup.injector);\n\n    // Wait for the initial position.\n    const initialPosition = await firstValueFrom(toObservable(popupOrigin, {injector: popup.injector}).pipe(filter(Boolean)));\n\n    const overlayPositionStrategy = this._overlay.position()\n      .flexibleConnectedTo(initialPosition)\n      .withFlexibleDimensions(false)\n      .withLockedPosition(false) // If locked, the popup won't attempt to reposition itself if not enough space available.\n      .withPositions(((): ConnectedPosition[] => {\n        switch (align) {\n          case 'north':\n            return [NORTH, SOUTH, WEST, EAST];\n          case 'south':\n            return [SOUTH, NORTH, WEST, EAST];\n          case 'west':\n            return [WEST, EAST, NORTH, SOUTH];\n          case 'east':\n            return [EAST, WEST, NORTH, SOUTH];\n          default:\n            throw Error('[PopupPositionError] Illegal position; must be north, south, west or east');\n        }\n      })());\n\n    // Configure the popup overlay.\n    const overlayConfig = new OverlayConfig({\n      panelClass: 'wb-popup',\n      hasBackdrop: false,\n      positionStrategy: overlayPositionStrategy,\n      scrollStrategy: this._overlay.scrollStrategies.noop(),\n      disposeOnNavigation: true,\n    });\n\n    // Construct the popup component and attach it to the DOM.\n    const overlayRef = this._overlay.create(overlayConfig);\n    const popupPortal = new ComponentPortal(PopupComponent, null, Injector.create({\n      parent: config.componentConstructOptions?.injector || this._environmentInjector,\n      providers: [\n        {provide: ɵPopup, useValue: popup},\n        {provide: Popup, useExisting: ɵPopup},\n        provideViewContext(contextualView),\n        ...[config.componentConstructOptions?.providers || []],\n      ],\n    }));\n    const componentRef = overlayRef.attach(popupPortal);\n    const popupElement: HTMLElement = componentRef.location.nativeElement;\n\n    // Make the popup focusable and request the focus.\n    popupElement.setAttribute('tabindex', '-1');\n    this._focusManager.focusVia(popupElement, 'program'); // To not close the popup immediately when it opens, if using the 'onFocusLost' strategy.\n\n    // Reposition popup when the anchor is moved.\n    effect(() => {\n      const origin = popupOrigin();\n      untracked(() => {\n        if (origin) {\n          overlayPositionStrategy.setOrigin(origin);\n          overlayRef.updatePosition();\n        }\n      });\n    }, {injector: popup.injector});\n\n    // Reposition popup when resized.\n    fromResize$(overlayRef.overlayElement)\n      .pipe(\n        observeIn(fn => this._zone.run(fn)),\n        takeUntilDestroyed(popupDestroyRef),\n      )\n      .subscribe(() => {\n        overlayRef.updatePosition();\n      });\n\n    // Close the popup depending on the passed config.\n    this.installPopupCloser(config, popupElement, popup, contextualView);\n\n    // Hide the popup when detaching the contextual view, if any.\n    if (contextualView) {\n      this.hidePopupOnViewDetach(overlayRef, contextualView, popup);\n    }\n\n    // Dispose the popup when closing it.\n    popupDestroyRef.onDestroy(() => {\n      overlayRef.dispose();\n    });\n\n    return new Promise<R | undefined>((resolve, reject) => {\n      popupDestroyRef.onDestroy(() => {\n        popup.result instanceof Error ? reject(popup.result) : resolve(popup.result as R);\n      });\n    });\n  }\n\n  /**\n   * Creates the popup handle.\n   */\n  private createPopup<R>(config: PopupConfig, context: {view: ɵWorkbenchView | null}): ɵPopup<R> {\n    // Construct the handle in an injection context that shares the popup's lifecycle, allowing for automatic cleanup of effects and RxJS interop functions.\n    const popupId = config.id ?? UUID.randomUUID();\n    const popupEnvironmentInjector = createEnvironmentInjector([provideViewContext(context.view)], this._environmentInjector, `Workbench Popup ${popupId}`);\n    return runInInjectionContext(popupEnvironmentInjector, () => new ɵPopup<R>(popupId, config));\n  }\n\n  /**\n   * Closes the popup depending on the configured popup closing strategy.\n   */\n  private installPopupCloser(config: PopupConfig, popupElement: HTMLElement, popup: ɵPopup, contextualView: ɵWorkbenchView | null): void {\n    const popupDestroyRef = popup.injector.get(DestroyRef);\n\n    // Close the popup on escape keystroke.\n    if (config.closeStrategy?.onEscape ?? true) {\n      fromEvent<KeyboardEvent>(popupElement, 'keydown')\n        .pipe(\n          subscribeIn(fn => this._zone.runOutsideAngular(fn)),\n          filter((event: KeyboardEvent) => event.key === 'Escape'),\n          observeIn(fn => this._zone.run(fn)),\n          takeUntilDestroyed(popupDestroyRef),\n        )\n        .subscribe(() => popup.close());\n    }\n\n    // Close the popup on focus loss.\n    if (config.closeStrategy?.onFocusLost ?? true) {\n      this._focusManager.monitor(popupElement, true)\n        .pipe(\n          filter((focusOrigin: FocusOrigin) => !focusOrigin),\n          takeUntilDestroyed(popupDestroyRef),\n        )\n        .subscribe(() => popup.close(popup.result));\n    }\n\n    // Close the popup when closing the view.\n    if (contextualView) {\n      effect(() => {\n        if (!this._viewRegistry.objects().some(view => view.id === contextualView.id)) {\n          untracked(() => popup.close());\n        }\n      }, {injector: popup.injector});\n    }\n  }\n\n  /**\n   * Hides the popup when its contextual view is detached, and displays it when it is reattached. Also restores the focus on re-activation.\n   * The contextual view is detached if not active, or located in the peripheral area and the main area is maximized.\n   */\n  private hidePopupOnViewDetach(overlayRef: OverlayRef, contextualView: ɵWorkbenchView, popup: ɵPopup): void {\n    overlayRef.overlayElement.classList.add('wb-view-context');\n\n    const popupDestroyRef = popup.injector.get(DestroyRef);\n    let activeElement: HTMLElement | undefined;\n\n    effect(() => {\n      const attached = contextualView.portal.attached();\n      untracked(() => {\n        if (attached) {\n          overlayRef.overlayElement.classList.add('wb-view-attached');\n          activeElement?.focus();\n        }\n        else {\n          overlayRef.overlayElement.classList.remove('wb-view-attached');\n        }\n      });\n    }, {injector: popup.injector});\n\n    // Track the focus in the popup to restore it when attaching the popup.\n    fromEvent(overlayRef.overlayElement, 'focusin')\n      .pipe(\n        subscribeIn(fn => this._zone.runOutsideAngular(fn)),\n        takeUntilDestroyed(popupDestroyRef),\n      )\n      .subscribe(() => {\n        activeElement = this._document.activeElement instanceof HTMLElement ? this._document.activeElement : undefined;\n      });\n  }\n\n  /**\n   * Creates a signal that tracks the position of the popup anchor.\n   */\n  private trackPopupOrigin(config: PopupConfig, contextualView: ɵWorkbenchView | null, injector: Injector): Signal<DOMRect | undefined> {\n    const hostBounds = boundingClientRect(contextualView?.portal.componentRef.location.nativeElement ?? document.documentElement, {injector});\n\n    if (config.anchor instanceof Element || config.anchor instanceof ElementRef) {\n      const anchor = coerceElement(config.anchor) as HTMLElement;\n      const anchorBounds = boundingClientRect(anchor, {injector});\n      return computed(() => {\n        // Maintain position and size when detached to prevent flickering when attached again and to support for virtual scrolling in popup content.\n        if (contextualView && !contextualView.portal.attached()) {\n          return undefined;\n        }\n\n        // IMPORTANT: Track anchor and host bounds only if attached to prevent flickering.\n\n        // The `boundingClientRect` signal does not detect position changes when the element is scrolled out of view.\n        // Consequently, if the popup anchor is scrolled out of view and the view is enlarged, the popup may not align\n        // with the view boundaries. Therefore, we read the anchor's bounding box directly from the DOM.\n        anchorBounds();\n\n        return constrainClientRect(anchor.getBoundingClientRect(), hostBounds());\n      }, {equal: isEqualDomRect});\n    }\n    else {\n      const anchorBounds = toSignal(Observables.coerce(config.anchor), {injector});\n      return computed(() => {\n        // Maintain position and size when detached to prevent flickering when attached again and to support for virtual scrolling in popup content.\n        if (contextualView && !contextualView.portal.attached()) {\n          return undefined;\n        }\n\n        // IMPORTANT: Track anchor and host bounds only if attached to prevent flickering.\n        if (!anchorBounds()) {\n          return undefined;\n        }\n\n        const {x, y} = mapToPageCoordinates(anchorBounds()!, hostBounds());\n        const {width, height} = anchorBounds()!;\n        return constrainClientRect(new DOMRect(x, y, width, height), hostBounds());\n      }, {equal: isEqualDomRect});\n    }\n  }\n\n  /**\n   * Resolves the contextual view to which the popup is bound.\n   */\n  private resolveContextualView(config: PopupConfig): ɵWorkbenchView | null {\n    if (config.context?.viewId) {\n      return this._viewRegistry.get(config.context.viewId);\n    }\n    if (config.context?.viewId === undefined) { // `null` means to open the popup outside the contextual view\n      return this._view ?? null;\n    }\n    return null;\n  }\n}\n\n/**\n * Maps given view coordinates to absolute page coordinates.\n */\nfunction mapToPageCoordinates(origin: PopupOrigin, relativeTo: DOMRect): Point {\n  const xy = origin as Point;\n  if (xy.x !== undefined && xy.y !== undefined) {\n    return {\n      x: relativeTo.x + xy.x,\n      y: relativeTo.y + xy.y,\n    };\n  }\n  const topLeft = origin as TopLeftPoint;\n  if (topLeft.top !== undefined && topLeft.left !== undefined) {\n    return {\n      x: relativeTo.x + topLeft.left,\n      y: relativeTo.y + topLeft.top,\n    };\n  }\n  const topRight = origin as TopRightPoint;\n  if (topRight.top !== undefined && topRight.right !== undefined) {\n    return {\n      x: relativeTo.x + (relativeTo.width - topRight.right),\n      y: relativeTo.y + topRight.top,\n    };\n  }\n  const bottomLeft = origin as BottomLeftPoint;\n  if (bottomLeft.bottom !== undefined && bottomLeft.left !== undefined) {\n    return {\n      x: relativeTo.x + bottomLeft.left,\n      y: relativeTo.y + (relativeTo.height - bottomLeft.bottom),\n    };\n  }\n  const bottomRight = origin as BottomRightPoint;\n  if (bottomRight.bottom !== undefined && bottomRight.right !== undefined) {\n    return {\n      x: relativeTo.x + (relativeTo.width - bottomRight.right),\n      y: relativeTo.y + (relativeTo.height - bottomRight.bottom),\n    };\n  }\n  throw Error('[PopupOriginError] Illegal popup origin; must be \"Point\", \"TopLeftPoint\", \"TopRightPoint\", \"BottomLeftPoint\" or \"BottomRightPoint\".');\n}\n\nfunction constrainClientRect(clientRect: DOMRect, constraints: DOMRect): DOMRect {\n  const top = clamp(clientRect.top, {min: constraints.top, max: constraints.bottom});\n  const right = clamp(clientRect.right, {min: constraints.left, max: constraints.right});\n  const bottom = clamp(clientRect.bottom, {min: constraints.top, max: constraints.bottom});\n  const left = clamp(clientRect.left, {min: constraints.left, max: constraints.right});\n  const width = right - left;\n  const height = bottom - top;\n  return new DOMRect(left, top, width, height);\n}\n\nfunction isEqualDomRect(a: DOMRect | undefined, b: DOMRect | undefined): boolean {\n  return a?.top === b?.top && a?.right === b?.right && a?.bottom === b?.bottom && a?.left === b?.left;\n}\n"]}
|