@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
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
/*
|
|
2
|
-
* Copyright (c) 2018-
|
|
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,43 +7,43 @@
|
|
|
7
7
|
*
|
|
8
8
|
* SPDX-License-Identifier: EPL-2.0
|
|
9
9
|
*/
|
|
10
|
-
import { Component, ElementRef, HostListener,
|
|
10
|
+
import { Component, computed, effect, ElementRef, HostListener, inject, NgZone, signal, untracked, viewChild, viewChildren } from '@angular/core';
|
|
11
11
|
import { ViewTabComponent } from '../view-tab/view-tab.component';
|
|
12
|
-
import {
|
|
13
|
-
import {
|
|
14
|
-
import {
|
|
12
|
+
import { map, mergeMap, takeUntil } from 'rxjs/operators';
|
|
13
|
+
import { from, fromEvent, merge, Observable, Subject, timer } from 'rxjs';
|
|
14
|
+
import { ViewTabDragImageRenderer } from '../../view-dnd/view-tab-drag-image-renderer.service';
|
|
15
|
+
import { ViewDragService } from '../../view-dnd/view-drag.service';
|
|
15
16
|
import { getCssTranslation, setCssClass, setCssVariable, unsetCssClass, unsetCssVariable } from '../../common/dom.util';
|
|
16
|
-
import {
|
|
17
|
+
import { ɵWorkbenchPart } from '../ɵworkbench-part.model';
|
|
18
|
+
import { filterArray, subscribeIn } from '@scion/toolkit/operators';
|
|
17
19
|
import { SciViewportComponent } from '@scion/components/viewport';
|
|
18
|
-
import {
|
|
19
|
-
import {
|
|
20
|
+
import { ɵWorkbenchRouter } from '../../routing/ɵworkbench-router.service';
|
|
21
|
+
import { boundingClientRect, dimension } from '@scion/components/dimension';
|
|
20
22
|
import { PartActionBarComponent } from '../part-action-bar/part-action-bar.component';
|
|
21
23
|
import { ViewListButtonComponent } from '../view-list-button/view-list-button.component';
|
|
22
24
|
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
23
|
-
import { fromDimension$ } from '@scion/toolkit/observable';
|
|
24
25
|
import { WORKBENCH_ID } from '../../workbench-id';
|
|
26
|
+
import { clamp } from '../../common/math.util';
|
|
25
27
|
import * as i0 from "@angular/core";
|
|
26
|
-
import * as i1 from "../../layout/workbench-layout.service";
|
|
27
|
-
import * as i2 from "../../routing/\u0275workbench-router.service";
|
|
28
|
-
import * as i3 from "../../view-dnd/view-tab-drag-image-renderer.service";
|
|
29
|
-
import * as i4 from "../\u0275workbench-part.model";
|
|
30
|
-
import * as i5 from "../../view-dnd/view-drag.service";
|
|
31
|
-
import * as i6 from "@scion/components/dimension";
|
|
32
28
|
/**
|
|
33
29
|
* Renders view tabs and actions of a {@link WorkbenchPart}.
|
|
34
30
|
*
|
|
35
31
|
* Tabs are added to a viewport, which the user can scroll if not enough space. The viewport grows with its tabs,
|
|
36
32
|
* allowing actions to be placed directly after the last tab or on the right side.
|
|
37
33
|
*
|
|
38
|
-
* ## Drag
|
|
34
|
+
* ## Drag and Drop
|
|
35
|
+
* This component subscribes to drag events to visualize tabs during a drag operation. The drag operation is initiated in {@link ViewTabComponent}.
|
|
36
|
+
*
|
|
39
37
|
* We use native drag and drop to support dragging views to other windows.
|
|
40
38
|
*
|
|
41
|
-
*
|
|
42
|
-
* -
|
|
43
|
-
* -
|
|
44
|
-
* -
|
|
45
|
-
* -
|
|
46
|
-
* -
|
|
39
|
+
* Following events are triggered:
|
|
40
|
+
* - dragstart: when start dragging a tab (only received in the tabbar where the drag operation has started)
|
|
41
|
+
* - dragend: when end dragging a tab, either via drop or cancel (only received in the tabbar where the drag operation has started)
|
|
42
|
+
* - dragenter: when a tab enters the tabbar
|
|
43
|
+
* - dragover: when dragging a tab over the tabbar, every few hundred milliseconds
|
|
44
|
+
* - dragleave: when a tab leaves the tabbar
|
|
45
|
+
* - drop: when the user drops a tab in the tabbar
|
|
46
|
+
*
|
|
47
47
|
* - Note that `dragleave` event is not fired on drop action. Therefore, always handle both events, `dragleave` and `drop` events, respectively.
|
|
48
48
|
*
|
|
49
49
|
* ## Terminology
|
|
@@ -53,34 +53,34 @@ import * as i6 from "@scion/components/dimension";
|
|
|
53
53
|
* e.g., to snap the drag image to the tabbar when dragging near it.
|
|
54
54
|
*/
|
|
55
55
|
export class PartBarComponent {
|
|
56
|
-
|
|
57
|
-
|
|
58
|
-
.pipe(startWith(queryList), takeUntilDestroyed(this._destroyRef))
|
|
59
|
-
.subscribe(queryList => this._viewTabs$.next(queryList.toArray()));
|
|
60
|
-
}
|
|
61
|
-
constructor(host, _workbenchId, _workbenchLayoutService, _router, _viewTabDragImageRenderer, _part, _viewDragService, _destroyRef, _zone) {
|
|
62
|
-
this._workbenchId = _workbenchId;
|
|
63
|
-
this._workbenchLayoutService = _workbenchLayoutService;
|
|
64
|
-
this._router = _router;
|
|
65
|
-
this._viewTabDragImageRenderer = _viewTabDragImageRenderer;
|
|
66
|
-
this._part = _part;
|
|
67
|
-
this._viewDragService = _viewDragService;
|
|
68
|
-
this._destroyRef = _destroyRef;
|
|
69
|
-
this._zone = _zone;
|
|
56
|
+
constructor() {
|
|
57
|
+
this._host = inject((ElementRef)).nativeElement;
|
|
70
58
|
this._viewportChange$ = new Subject();
|
|
71
|
-
this.
|
|
72
|
-
this.
|
|
73
|
-
this.
|
|
74
|
-
this.
|
|
59
|
+
this._workbenchId = inject(WORKBENCH_ID);
|
|
60
|
+
this._router = inject(ɵWorkbenchRouter);
|
|
61
|
+
this._viewTabDragImageRenderer = inject(ViewTabDragImageRenderer);
|
|
62
|
+
this._viewDragService = inject(ViewDragService);
|
|
63
|
+
this._zone = inject(NgZone);
|
|
64
|
+
this._viewTabs = viewChildren(ViewTabComponent);
|
|
65
|
+
this._hostBoundingClientRect = boundingClientRect(inject(ElementRef));
|
|
66
|
+
this._viewportComponent = viewChild.required(SciViewportComponent);
|
|
67
|
+
this._viewportComponentElement = viewChild.required(SciViewportComponent, { read: (ElementRef) });
|
|
68
|
+
this._tabCornerRadiusElement = viewChild.required('tab_corner_radius', { read: (ElementRef) });
|
|
69
|
+
this._paddingInlineElement = viewChild.required('padding_inline', { read: (ElementRef) });
|
|
70
|
+
this._fillerElement = viewChild.required('filler');
|
|
71
|
+
this._viewportBoundingBox = boundingClientRect(this._viewportComponentElement);
|
|
72
|
+
this.part = inject(ɵWorkbenchPart);
|
|
73
|
+
this.startActions = computed(() => this.part.actions().filter(action => action.align !== 'end'));
|
|
74
|
+
this.endActions = computed(() => this.part.actions().filter(action => action.align === 'end'));
|
|
75
75
|
/**
|
|
76
76
|
* Reference to the tab before which to insert the drag source on drop, or `end` if dropping it after the last tab.
|
|
77
77
|
*/
|
|
78
|
-
this.dropTargetViewTab = null;
|
|
78
|
+
this.dropTargetViewTab = signal(null);
|
|
79
79
|
/**
|
|
80
80
|
* Reference to the tab where the drag operation was started.
|
|
81
81
|
* This reference is only set if the drag operation started on a tab of this tabbar.
|
|
82
82
|
*/
|
|
83
|
-
this.dragSourceViewTab = null;
|
|
83
|
+
this.dragSourceViewTab = signal(null);
|
|
84
84
|
/**
|
|
85
85
|
* Transfer data of the tab being dragged over this tabbar.
|
|
86
86
|
*/
|
|
@@ -89,6 +89,10 @@ export class PartBarComponent {
|
|
|
89
89
|
* Function for snapping the drag image to the tabbar when dragging near it.
|
|
90
90
|
*/
|
|
91
91
|
this._constrainFn = null;
|
|
92
|
+
/**
|
|
93
|
+
* Signals when unsetting drag state.
|
|
94
|
+
*/
|
|
95
|
+
this._onUnsetDragState = signal(undefined, { equal: () => false });
|
|
92
96
|
/**
|
|
93
97
|
* Tabbar indents where to display the rounded bottom corners of the first and last view tab.
|
|
94
98
|
*/
|
|
@@ -96,171 +100,186 @@ export class PartBarComponent {
|
|
|
96
100
|
left: 0,
|
|
97
101
|
right: 0,
|
|
98
102
|
};
|
|
99
|
-
|
|
100
|
-
* Notifies when the drag animation has finished, i.e., tabs have transitioned to their final position.
|
|
101
|
-
*/
|
|
102
|
-
this._dragAnimationStable$ = interval(0, animationFrameScheduler)
|
|
103
|
-
.pipe(filter(() => this.isDragAnimationStable()), subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)), observeInside(continueFn => this._zone.run(continueFn)), map(() => undefined));
|
|
104
|
-
this._host = host.nativeElement;
|
|
105
|
-
this.dragover$ = this._viewDragService.tabbarDragOver$.pipe(map(partId => partId === this._part.id));
|
|
106
|
-
}
|
|
107
|
-
ngOnInit() {
|
|
103
|
+
this._maxViewportWidth = this.calculateMaxViewportWidth();
|
|
108
104
|
this.installActiveViewScroller();
|
|
109
105
|
this.installScrolledIntoViewUpdater();
|
|
110
106
|
this.installViewDragListener();
|
|
111
|
-
this.
|
|
112
|
-
this.
|
|
107
|
+
this.installTabbarIndentSizeTracker();
|
|
108
|
+
this.installViewportChangeTracker();
|
|
113
109
|
}
|
|
114
110
|
onDoubleClick(event) {
|
|
115
|
-
if (this.
|
|
111
|
+
if (this.part.isInMainArea) {
|
|
116
112
|
this._router.navigate(layout => layout.toggleMaximized()).then();
|
|
117
113
|
}
|
|
118
114
|
event.stopPropagation();
|
|
119
115
|
}
|
|
120
|
-
get viewIds$() {
|
|
121
|
-
return this._part.viewIds$;
|
|
122
|
-
}
|
|
123
|
-
onTabbarViewportDimensionChange() {
|
|
124
|
-
this._viewportChange$.next();
|
|
125
|
-
}
|
|
126
|
-
onScroll() {
|
|
127
|
-
this._viewportChange$.next();
|
|
128
|
-
}
|
|
129
116
|
/**
|
|
130
|
-
* Method invoked when the user starts dragging a tab of this tabbar.
|
|
117
|
+
* Method invoked when the user starts dragging a tab of this tabbar, only in the tabbar where the drag started.
|
|
131
118
|
*/
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
137
|
-
|
|
138
|
-
const dragSourceViewTab = this._viewTabs[dragSourceIndex];
|
|
139
|
-
const dropTargetViewTab = this._viewTabs[dragSourceIndex + 1] ?? 'end';
|
|
140
|
-
// Constrain function for snapping the drag image to the tabbar when dragging near it.
|
|
119
|
+
onTabDragStart() {
|
|
120
|
+
// Memoize drag data.
|
|
121
|
+
this._dragData = this._viewDragService.viewDragData();
|
|
122
|
+
// Indicate dragging over this tabbar.
|
|
123
|
+
this._viewDragService.signalTabbarDragEnter(this.part.id);
|
|
124
|
+
// Snap drag image to tabbar when dragging near it.
|
|
141
125
|
this._constrainFn = this.createDragImageConstrainFn();
|
|
142
126
|
this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);
|
|
143
|
-
|
|
144
|
-
|
|
145
|
-
//
|
|
146
|
-
|
|
147
|
-
|
|
148
|
-
|
|
149
|
-
|
|
150
|
-
|
|
151
|
-
|
|
152
|
-
});
|
|
153
|
-
});
|
|
154
|
-
// When dragging a tab very quickly out of this tabbar, only the `dragstart` event gets fired, but not the `dragenter` event.
|
|
155
|
-
// Therefore, we subscribe to global `dragenter` events and manually invoke `onTabbarDragLeave` to correctly display the tabs and the active view.
|
|
156
|
-
this._viewDragService.viewDrag$(window, { eventType: ['dragenter'] })
|
|
157
|
-
.pipe(take(1), takeUntil(this._dragend$), takeUntilDestroyed(this._destroyRef))
|
|
127
|
+
// Disable pointer events to not break the drag operation (in Firefox, but not Chrome and Edge).
|
|
128
|
+
requestAnimationFrame(() => setCssClass(this._host, 'pointer-events-disabled'));
|
|
129
|
+
// Set CSS class to not animate start dragging.
|
|
130
|
+
setCssClass(this._host, 'on-drag-start');
|
|
131
|
+
this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-start'));
|
|
132
|
+
// Check if the user continues dragging over this tabbar.
|
|
133
|
+
// If not, the user has quickly dragged the tab out of the tabbar, so manually call `onTabbarDragEnter` and `onTabbarDragLeave`.
|
|
134
|
+
fromEvent(window, 'dragenter', { once: true })
|
|
135
|
+
.pipe(takeUntil(merge(fromEvent(this._host, 'dragenter'), fromEvent(this._host, 'dragend'))))
|
|
158
136
|
.subscribe(event => {
|
|
159
|
-
|
|
160
|
-
|
|
161
|
-
if ((this._host.compareDocumentPosition(event.target) & Node.DOCUMENT_POSITION_CONTAINED_BY) === 0) {
|
|
162
|
-
this.onTabbarDragLeave();
|
|
163
|
-
}
|
|
137
|
+
this.onTabbarDragEnter(event);
|
|
138
|
+
this.onTabbarDragLeave(event);
|
|
164
139
|
});
|
|
165
140
|
}
|
|
166
141
|
/**
|
|
167
|
-
* Method invoked when the user ends dragging a tab of this tabbar.
|
|
168
|
-
*
|
|
169
|
-
* Invoked only in the tabbar where the drag operation started, regardless of whether the drag operation was completed or canceled.
|
|
142
|
+
* Method invoked when the user ends dragging a tab of this tabbar, either by drop or cancel, only in the tabbar where the drag started.
|
|
170
143
|
*/
|
|
171
|
-
|
|
172
|
-
|
|
173
|
-
|
|
144
|
+
onTabDragEnd(event) {
|
|
145
|
+
// Unset drag state only on cancel, not on drop, to prevent tabs from temporarily reverting to their pre-drag state. Drag state is reset in `onTabMoved` after a drop.
|
|
146
|
+
if (event.dataTransfer.dropEffect === 'none') {
|
|
147
|
+
// Reactivate the drag source view on cancel.
|
|
148
|
+
this.dragSourceViewTab()?.view().activate({ skipLocationChange: true }).then();
|
|
149
|
+
// Unset drag state.
|
|
150
|
+
this.unsetDragState();
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
/**
|
|
154
|
+
* Method invoked when the layout has changed after a drop. The drop may not be related to a tab of this tabbar.
|
|
155
|
+
*/
|
|
156
|
+
onTabMoved(event) {
|
|
157
|
+
// Unset drag state only if no new drag has started since the drop.
|
|
158
|
+
if (this._dragData?.uid === event.dragData?.uid) {
|
|
159
|
+
this.unsetDragState();
|
|
160
|
+
}
|
|
174
161
|
}
|
|
175
162
|
/**
|
|
176
163
|
* Method invoked when the user drags a tab into this tabbar.
|
|
177
164
|
*/
|
|
178
165
|
onTabbarDragEnter(event) {
|
|
179
|
-
|
|
180
|
-
this._viewDragService.
|
|
181
|
-
|
|
182
|
-
|
|
183
|
-
|
|
184
|
-
|
|
185
|
-
|
|
166
|
+
// Memoize drag data.
|
|
167
|
+
const dragData = this._dragData = this._viewDragService.viewDragData();
|
|
168
|
+
// Indicate dragging over this tabbar.
|
|
169
|
+
this._viewDragService.signalTabbarDragEnter(this.part.id);
|
|
170
|
+
// Snap drag image to tabbar when dragging near it.
|
|
171
|
+
this._constrainFn = this.createDragImageConstrainFn();
|
|
172
|
+
this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);
|
|
173
|
+
// Set drag source, but only if started the drag operation in this part.
|
|
174
|
+
// We do not set the drag source in `dragstart` to not break the drag operation (in Chrome and Edge, but not Firefox).
|
|
175
|
+
if (this._host.classList.contains('on-drag-start')) {
|
|
176
|
+
this.dragSourceViewTab.set(this._viewTabs().find(viewTab => viewTab.viewId === dragData.viewId) ?? null);
|
|
177
|
+
setCssVariable(this._host, { '--ɵpart-bar-drag-placeholder-width': `${dragData.viewTabWidth}px` });
|
|
186
178
|
}
|
|
187
|
-
// Locate
|
|
188
|
-
this.dropTargetViewTab
|
|
189
|
-
//
|
|
190
|
-
setCssClass(this._host, '
|
|
191
|
-
|
|
192
|
-
|
|
193
|
-
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
'--ɵpart-bar-drag-image-placeholder-width': `${this._dragData.viewTabWidth}px`,
|
|
197
|
-
});
|
|
179
|
+
// Locate the tab the user is dragging over.
|
|
180
|
+
this.dropTargetViewTab.set(this.computeDropTarget(event));
|
|
181
|
+
// Disable pointer events on the tabbar for a smoother drag and drop experience when quickly swapping tabs.
|
|
182
|
+
setCssClass(this._host, 'pointer-events-disabled');
|
|
183
|
+
// Set CSS class to animate entering the tabbar.
|
|
184
|
+
setCssClass(this._host, 'on-drag-enter');
|
|
185
|
+
this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-enter'));
|
|
186
|
+
// Inject drag source width into CSS.
|
|
187
|
+
setCssVariable(this._host, { '--ɵpart-bar-drag-source-width': `${dragData.viewTabWidth}px` });
|
|
198
188
|
}
|
|
199
189
|
/**
|
|
200
190
|
* Method invoked when the user drags a tab out of this tabbar.
|
|
201
191
|
*/
|
|
202
|
-
onTabbarDragLeave() {
|
|
203
|
-
|
|
204
|
-
//
|
|
205
|
-
|
|
206
|
-
|
|
207
|
-
.
|
|
208
|
-
.subscribe({ complete: () => unsetCssClass(this._host, 'drag-leave') });
|
|
209
|
-
this.unsetDragState({ unsetDragSource: false });
|
|
210
|
-
// Activate the view next to the view being dragged out of this tabbar. But, do not push the navigation into browsing history stack.
|
|
211
|
-
if (this.dragSourceViewTab && this.dragSourceViewTab.active) {
|
|
212
|
-
this._router.navigate(layout => layout.activateAdjacentView(this.dragSourceViewTab.viewId), { skipLocationChange: true }).then();
|
|
192
|
+
onTabbarDragLeave(event) {
|
|
193
|
+
// Set CSS class to animate leaving the tabbar, but not when canceling the drag operation for instant reset.
|
|
194
|
+
// Pointer coordinates are not set when the drag operation is canceled or when dragging the tab out of the window.
|
|
195
|
+
if (event.screenX || event.screenY) {
|
|
196
|
+
setCssClass(this._host, 'on-drag-leave');
|
|
197
|
+
this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-leave'));
|
|
213
198
|
}
|
|
199
|
+
// Activate the adjacent tab, only if the drag operation is not canceled.
|
|
200
|
+
// The drag event does not provide information about cancelation. Therefore, we wait for a macrotask, canceling it when receiving `dragend`, i.e., the drag operation has been canceled.
|
|
201
|
+
if (this.dragSourceViewTab()?.active) {
|
|
202
|
+
timer(0)
|
|
203
|
+
.pipe(takeUntil(fromEvent(window, 'dragend')))
|
|
204
|
+
.subscribe(() => this._router.navigate(layout => layout.activateAdjacentView(this.dragSourceViewTab().viewId), { skipLocationChange: true }));
|
|
205
|
+
}
|
|
206
|
+
// Clean up drag state.
|
|
207
|
+
this.unsetDragState({ unsetDragSource: false, unsetDragData: false });
|
|
214
208
|
}
|
|
215
209
|
/**
|
|
216
|
-
* Method invoked
|
|
210
|
+
* Method invoked when the user is dragging a tab over this tabbar.
|
|
217
211
|
*/
|
|
218
|
-
onTabbarDragOver(event) {
|
|
212
|
+
onTabbarDragOver(event, dragDistance, scrollDistance) {
|
|
213
|
+
NgZone.assertNotInAngularZone();
|
|
219
214
|
event.preventDefault(); // allow view drop
|
|
220
|
-
// Locate the tab
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
|
|
225
|
-
|
|
215
|
+
// Locate the tab the user is dragging over.
|
|
216
|
+
// - Run in animation frame for better performance, most noticeable on lower CPU power.
|
|
217
|
+
// - Do not wait for the drag animation to complete for a smoother drag and drop experience when quickly moving the tab in the tabbar.
|
|
218
|
+
requestAnimationFrame(() => {
|
|
219
|
+
// Skip if the drag operation has ended.
|
|
220
|
+
if (this._viewDragService.isDragOverTabbar !== this.part.id) {
|
|
221
|
+
return;
|
|
222
|
+
}
|
|
223
|
+
// Ignore the drag distance if `drag-enter` animation is in progress to avoid incorrect drop target computation when dragging a tab from the left.
|
|
224
|
+
if (this._host.classList.contains('on-drag-enter')) {
|
|
225
|
+
this.dropTargetViewTab.set(this.computeDropTarget(event, 0));
|
|
226
|
+
}
|
|
227
|
+
// Compute drop target only when moving the drag pointer to prevent tabs from sliding back and forth (while animation is in progress)
|
|
228
|
+
else if (dragDistance) {
|
|
229
|
+
this.dropTargetViewTab.set(this.computeDropTarget(event, dragDistance));
|
|
230
|
+
}
|
|
231
|
+
// Compute drop target when scrolling the viewport, i.e., not moving the drag pointer but scrolling the viewport because placed the drag pointer at the viewport edges.
|
|
232
|
+
else if (scrollDistance) {
|
|
233
|
+
this.dropTargetViewTab.set(this.computeDropTarget(event, scrollDistance));
|
|
234
|
+
}
|
|
235
|
+
// Set CSS class to animate dragging over the tabbar.
|
|
236
|
+
setCssClass(this._host, 'drag-over');
|
|
237
|
+
// Calculate the new size of the drag source placeholder.
|
|
238
|
+
// The placeholder expands the viewport when dragging a tab to the right, moving part actions along with the dragover event.
|
|
239
|
+
setCssVariable(this._host, { '--ɵpart-bar-drag-placeholder-width': `${this.calculateDragPlaceholderWidth(event)}px` });
|
|
240
|
+
});
|
|
226
241
|
}
|
|
227
242
|
/**
|
|
228
243
|
* Method invoked when the user drops a tab in this tabbar.
|
|
229
244
|
*/
|
|
230
|
-
onTabbarDrop() {
|
|
245
|
+
async onTabbarDrop() {
|
|
246
|
+
const dragData = this._dragData;
|
|
247
|
+
const dropTargetViewTab = this.dropTargetViewTab();
|
|
248
|
+
const dropPosition = dropTargetViewTab === 'end' ? 'end' : this._viewTabs().indexOf(dropTargetViewTab);
|
|
231
249
|
this._viewDragService.dispatchViewMoveEvent({
|
|
232
250
|
source: {
|
|
233
|
-
workbenchId:
|
|
234
|
-
partId:
|
|
235
|
-
viewId:
|
|
236
|
-
alternativeViewId:
|
|
237
|
-
viewUrlSegments:
|
|
238
|
-
navigationHint:
|
|
239
|
-
|
|
251
|
+
workbenchId: dragData.workbenchId,
|
|
252
|
+
partId: dragData.partId,
|
|
253
|
+
viewId: dragData.viewId,
|
|
254
|
+
alternativeViewId: dragData.alternativeViewId,
|
|
255
|
+
viewUrlSegments: dragData.viewUrlSegments,
|
|
256
|
+
navigationHint: dragData.navigationHint,
|
|
257
|
+
navigationData: dragData.navigationData,
|
|
258
|
+
classList: dragData.classList,
|
|
240
259
|
},
|
|
241
260
|
target: {
|
|
242
261
|
workbenchId: this._workbenchId,
|
|
243
|
-
position:
|
|
244
|
-
elementId: this.
|
|
262
|
+
position: dropPosition,
|
|
263
|
+
elementId: this.part.id,
|
|
245
264
|
},
|
|
246
|
-
|
|
247
|
-
// Wait undoing the drag state until flushed tab changes to the DOM, i.e., after the layout routing cycle completes.
|
|
248
|
-
// If we were to undo the drag state immediately during the course of the drop event, the tab layout would temporarily
|
|
249
|
-
// revert to the state before the drag operation, resulting in an ugly flicker.
|
|
250
|
-
firstValueFrom(this._workbenchLayoutService.onLayoutChange$).then(() => {
|
|
251
|
-
this.unsetDragState();
|
|
265
|
+
dragData: dragData,
|
|
252
266
|
});
|
|
253
267
|
}
|
|
254
268
|
/**
|
|
255
269
|
* Locates the tab before which the dragged tab is to be inserted when it is dropped. Dragging beyond the last tab returns `end`.
|
|
256
270
|
*/
|
|
257
|
-
computeDropTarget(event) {
|
|
258
|
-
const
|
|
259
|
-
const viewTabs = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab);
|
|
271
|
+
computeDropTarget(event, dragDistanceX = 0) {
|
|
272
|
+
const dragImage = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData, event);
|
|
273
|
+
const viewTabs = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab());
|
|
260
274
|
for (const viewTab of viewTabs) {
|
|
261
|
-
const
|
|
262
|
-
const
|
|
263
|
-
|
|
275
|
+
const { left, width } = viewTab.boundingClientRect();
|
|
276
|
+
const viewTabHCenter = left + (width / 2);
|
|
277
|
+
// Moving the pointer to the left.
|
|
278
|
+
if (dragDistanceX <= 0 && dragImage.left < viewTabHCenter) {
|
|
279
|
+
return viewTab;
|
|
280
|
+
}
|
|
281
|
+
// Moving the pinter to the right.
|
|
282
|
+
if (dragDistanceX > 0 && dragImage.right < viewTabHCenter) {
|
|
264
283
|
return viewTab;
|
|
265
284
|
}
|
|
266
285
|
}
|
|
@@ -270,49 +289,32 @@ export class PartBarComponent {
|
|
|
270
289
|
* Creates the function for snapping the drag image to the tabbar when dragging near it.
|
|
271
290
|
*/
|
|
272
291
|
createDragImageConstrainFn() {
|
|
273
|
-
|
|
274
|
-
|
|
275
|
-
|
|
276
|
-
|
|
277
|
-
return new ViewDragImageRect({
|
|
278
|
-
x: Math.min(Math.max(hostLeft + this._tabbarIndent.left, rect.x), hostLeft + maxViewportWidth - rect.width - this._tabbarIndent.right),
|
|
279
|
-
y: viewportBoundingBox.top,
|
|
280
|
-
width: rect.width,
|
|
281
|
-
height: viewportBoundingBox.height, // fit into tabbar, e.g., when dragging a tab into a tabbar of a different height
|
|
282
|
-
});
|
|
283
|
-
};
|
|
292
|
+
return (dragImage) => new DOMRect(clamp(dragImage.x, {
|
|
293
|
+
min: this._hostBoundingClientRect().left + this._tabbarIndent.left,
|
|
294
|
+
max: this._hostBoundingClientRect().left + this._maxViewportWidth() - dragImage.width - this._tabbarIndent.right,
|
|
295
|
+
}), this._viewportBoundingBox().top, dragImage.width, this._viewportBoundingBox().height);
|
|
284
296
|
}
|
|
285
297
|
/**
|
|
286
|
-
* Calculates the maximum width of the tabbar viewport before an overflow would occur
|
|
287
|
-
*
|
|
288
|
-
* To calculate the effective width, we need to simulate an overflow in the DOM. Therefore, call this method only from `dragstart` and not
|
|
289
|
-
* from `dragover` and disable animations.
|
|
298
|
+
* Calculates the maximum width of the tabbar viewport before an overflow would occur.
|
|
299
|
+
* Use to constrain the drag image to the tabbar bounds.
|
|
290
300
|
*/
|
|
291
301
|
calculateMaxViewportWidth() {
|
|
292
|
-
const
|
|
293
|
-
|
|
294
|
-
setCssClass(this._host, 'calculating-max-viewport-width');
|
|
295
|
-
setCssVariable(this._host, { '--ɵpart-bar-drag-image-placeholder-width': `${this._host.clientWidth}px` });
|
|
296
|
-
try {
|
|
297
|
-
return this._viewportElement.nativeElement.getBoundingClientRect().width;
|
|
298
|
-
}
|
|
299
|
-
finally {
|
|
300
|
-
setCssVariable(this._host, { '--ɵpart-bar-drag-image-placeholder-width': currentWidth || null });
|
|
301
|
-
unsetCssClass(this._host, 'calculating-max-viewport-width');
|
|
302
|
-
}
|
|
302
|
+
const fillerDimension = dimension(this._fillerElement);
|
|
303
|
+
return computed(() => this._viewportBoundingBox().width + fillerDimension().offsetWidth);
|
|
303
304
|
}
|
|
304
305
|
/**
|
|
305
|
-
* Calculates the width for the drag
|
|
306
|
+
* Calculates the width for the drag placeholder.
|
|
306
307
|
*
|
|
308
|
+
* The placeholder expands the viewport when dragging a tab to the right, moving part actions along with the dragover event.
|
|
309
|
+
*
|
|
310
|
+
* Calculation:
|
|
307
311
|
* - When dragging over tabs, the width is equal to the width of the drag source.
|
|
308
|
-
* - When dragging to the right of the last tab, the width of the placeholder is
|
|
309
|
-
* last tab and the position of the dragover event. Applying this width will resize the viewport, moving part actions along
|
|
310
|
-
* with the dragover event.
|
|
312
|
+
* - When dragging to the right of the last tab, the width of the placeholder is the distance between the last tab and the position of the drag pointer.
|
|
311
313
|
*/
|
|
312
|
-
|
|
314
|
+
calculateDragPlaceholderWidth(event) {
|
|
313
315
|
const viewDragImageRect = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData, event);
|
|
314
|
-
const lastViewTab = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab).at(-1);
|
|
315
|
-
const lastViewTabRight = lastViewTab?.
|
|
316
|
+
const lastViewTab = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab()).at(-1);
|
|
317
|
+
const lastViewTabRight = lastViewTab?.boundingClientRect().right ?? this._hostBoundingClientRect().left + this._tabbarIndent.left;
|
|
316
318
|
return Math.max(lastViewTabRight + viewDragImageRect.width, viewDragImageRect.right) - lastViewTabRight;
|
|
317
319
|
}
|
|
318
320
|
/**
|
|
@@ -320,11 +322,12 @@ export class PartBarComponent {
|
|
|
320
322
|
* The animation is considered stable when all tabs have moved to their final position.
|
|
321
323
|
*/
|
|
322
324
|
isDragAnimationStable() {
|
|
323
|
-
|
|
324
|
-
|
|
325
|
+
const dropTargetViewTab = this.dropTargetViewTab();
|
|
326
|
+
if (this._viewDragService.isDragOverTabbar !== this.part.id) {
|
|
327
|
+
return this._viewTabs().every(viewTab => getCssTranslation(viewTab.host).translateX === 'none');
|
|
325
328
|
}
|
|
326
|
-
const viewTabs = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab);
|
|
327
|
-
const dropTargetIndex =
|
|
329
|
+
const viewTabs = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab());
|
|
330
|
+
const dropTargetIndex = dropTargetViewTab === 'end' ? viewTabs.length : viewTabs.indexOf(dropTargetViewTab);
|
|
328
331
|
const dragSourceWidth = this._dragData.viewTabWidth;
|
|
329
332
|
return viewTabs.every((viewTab, i) => {
|
|
330
333
|
const viewTabTranslateX = getCssTranslation(viewTab.host).translateX;
|
|
@@ -339,33 +342,63 @@ export class PartBarComponent {
|
|
|
339
342
|
return true;
|
|
340
343
|
});
|
|
341
344
|
}
|
|
345
|
+
/**
|
|
346
|
+
* Cleans up the drag state, unsetting drag data and drag source based on provided options.
|
|
347
|
+
*/
|
|
342
348
|
unsetDragState(options) {
|
|
343
|
-
|
|
344
|
-
|
|
345
|
-
}
|
|
346
|
-
this.dropTargetViewTab = null;
|
|
347
|
-
this._dragData = null;
|
|
349
|
+
this.dropTargetViewTab.set(null);
|
|
350
|
+
this._onUnsetDragState.set();
|
|
348
351
|
this._viewTabDragImageRenderer.unsetConstrainDragImageRectFn(this._constrainFn);
|
|
352
|
+
this._viewDragService.signalTabbarDragLeave(this.part.id);
|
|
349
353
|
this._constrainFn = null;
|
|
350
|
-
unsetCssClass(this._host, 'drag-over');
|
|
351
|
-
unsetCssVariable(this._host, '--ɵpart-bar-drag-source-width', '--ɵpart-bar-drag-
|
|
352
|
-
|
|
354
|
+
unsetCssClass(this._host, 'drag-over', 'pointer-events-disabled');
|
|
355
|
+
unsetCssVariable(this._host, '--ɵpart-bar-drag-source-width', '--ɵpart-bar-drag-placeholder-width');
|
|
356
|
+
if (options?.unsetDragSource ?? true) {
|
|
357
|
+
this.dragSourceViewTab.set(null);
|
|
358
|
+
}
|
|
359
|
+
if (options?.unsetDragData ?? true) {
|
|
360
|
+
this._dragData = null;
|
|
361
|
+
}
|
|
353
362
|
}
|
|
354
|
-
|
|
355
|
-
|
|
363
|
+
/**
|
|
364
|
+
* Executes given function when the drag animation has finished, i.e., tabs have transitioned to their final position.
|
|
365
|
+
*/
|
|
366
|
+
onDragAnimationStable(callback) {
|
|
367
|
+
if (NgZone.isInAngularZone()) {
|
|
368
|
+
this._zone.runOutsideAngular(() => this.onDragAnimationStable(callback));
|
|
369
|
+
return;
|
|
370
|
+
}
|
|
371
|
+
requestAnimationFrame(() => this.isDragAnimationStable() ? callback() : this.onDragAnimationStable(callback));
|
|
356
372
|
}
|
|
357
373
|
/**
|
|
358
374
|
* Scrolls the tab of the active view into view.
|
|
359
375
|
*/
|
|
360
376
|
installActiveViewScroller() {
|
|
361
|
-
|
|
362
|
-
|
|
363
|
-
.
|
|
364
|
-
// There may be no active view
|
|
365
|
-
|
|
366
|
-
|
|
367
|
-
this._viewportComponent.scrollIntoView(activeViewTab.host);
|
|
377
|
+
effect(() => {
|
|
378
|
+
// Track the active view.
|
|
379
|
+
const activeViewId = this.part.activeViewId();
|
|
380
|
+
// There may be no active view, e.g., if no view has been activated, or when dragging the last view out of the tabbar.
|
|
381
|
+
if (!activeViewId) {
|
|
382
|
+
return;
|
|
368
383
|
}
|
|
384
|
+
// Do not track rendered tabs in general to prevent unwanted scrolling when opening or closing inactive tabs.
|
|
385
|
+
const viewTabs = untracked(() => this._viewTabs());
|
|
386
|
+
// Get a reference to the currently rendered active tab, waiting if necessary by tracking the rendered tabs.
|
|
387
|
+
const activeViewTab = viewTabs.find(viewTab => viewTab.viewId === activeViewId);
|
|
388
|
+
if (!activeViewTab) {
|
|
389
|
+
this._viewTabs(); // Track rendered tabs to re-execute once rendered.
|
|
390
|
+
return;
|
|
391
|
+
}
|
|
392
|
+
// Track navigation of the active view, scrolling its tab into view if necessary.
|
|
393
|
+
activeViewTab.view().navigationId();
|
|
394
|
+
// Track unsetting drag state to ensure the active view is scrolled into view after drop or cancel.
|
|
395
|
+
this._onUnsetDragState();
|
|
396
|
+
// Scroll the tab into view if scrolled out of view.
|
|
397
|
+
untracked(() => requestAnimationFrame(() => {
|
|
398
|
+
if (!this._viewportComponent().isElementInView(activeViewTab.host, 'full')) {
|
|
399
|
+
this._viewportComponent().scrollIntoView(activeViewTab.host);
|
|
400
|
+
}
|
|
401
|
+
}));
|
|
369
402
|
});
|
|
370
403
|
}
|
|
371
404
|
/**
|
|
@@ -373,87 +406,123 @@ export class PartBarComponent {
|
|
|
373
406
|
*/
|
|
374
407
|
installScrolledIntoViewUpdater() {
|
|
375
408
|
this._viewportChange$
|
|
376
|
-
.pipe(map(() => this._viewTabs), filterArray(viewTab => viewTab !== this.dragSourceViewTab), // skip drag source as always scrolled out of view
|
|
377
|
-
mergeMap(viewTabs => from(viewTabs)), takeUntilDestroyed(
|
|
409
|
+
.pipe(map(() => this._viewTabs()), filterArray(viewTab => viewTab !== this.dragSourceViewTab()), // skip drag source as always scrolled out of view
|
|
410
|
+
mergeMap(viewTabs => from(viewTabs)), takeUntilDestroyed())
|
|
378
411
|
.subscribe(viewTab => {
|
|
379
|
-
viewTab.view.scrolledIntoView = this._viewportComponent.isElementInView(viewTab.host, 'full');
|
|
412
|
+
viewTab.view().scrolledIntoView = this._viewportComponent().isElementInView(viewTab.host, 'full');
|
|
380
413
|
});
|
|
381
414
|
}
|
|
415
|
+
/**
|
|
416
|
+
* Subscribes to drag events.
|
|
417
|
+
*/
|
|
382
418
|
installViewDragListener() {
|
|
419
|
+
const zone = inject(NgZone);
|
|
420
|
+
// Subscribe to drag events relevant for this tabbar.
|
|
383
421
|
this._viewDragService.viewDrag$(this._host)
|
|
384
|
-
.pipe(
|
|
385
|
-
|
|
422
|
+
.pipe(subscribeIn(fn => zone.runOutsideAngular(fn)), calculateDistance(() => this._viewportComponent()), // provide viewport lazily because not injected yet
|
|
423
|
+
takeUntilDestroyed())
|
|
424
|
+
.subscribe(({ event, dragDistance, scrollDistance }) => {
|
|
386
425
|
switch (event.type) {
|
|
426
|
+
case 'dragstart':
|
|
427
|
+
this.onTabDragStart();
|
|
428
|
+
break;
|
|
429
|
+
case 'dragend':
|
|
430
|
+
this.onTabDragEnd(event);
|
|
431
|
+
break;
|
|
387
432
|
case 'dragenter':
|
|
388
433
|
this.onTabbarDragEnter(event);
|
|
389
434
|
break;
|
|
390
435
|
case 'dragover':
|
|
391
|
-
this.onTabbarDragOver(event);
|
|
436
|
+
this.onTabbarDragOver(event, dragDistance, scrollDistance);
|
|
392
437
|
break;
|
|
393
438
|
case 'dragleave':
|
|
394
|
-
this.onTabbarDragLeave();
|
|
439
|
+
this.onTabbarDragLeave(event);
|
|
395
440
|
break;
|
|
396
441
|
case 'drop':
|
|
397
|
-
this.onTabbarDrop();
|
|
442
|
+
this.onTabbarDrop().then();
|
|
398
443
|
break;
|
|
399
444
|
}
|
|
400
445
|
});
|
|
446
|
+
// Get notified when a view has moved in the layout.
|
|
447
|
+
this._viewDragService.viewMoved$
|
|
448
|
+
.pipe(subscribeIn(fn => zone.runOutsideAngular(fn)), takeUntilDestroyed())
|
|
449
|
+
.subscribe(event => this.onTabMoved(event));
|
|
450
|
+
/**
|
|
451
|
+
* Calculates the distance the drag pointer and viewport have moved since the last drag event.
|
|
452
|
+
*/
|
|
453
|
+
function calculateDistance(viewport) {
|
|
454
|
+
return source$ => new Observable(observer => {
|
|
455
|
+
let lastScreenX;
|
|
456
|
+
let lastScrollLeft;
|
|
457
|
+
const subscription = source$.subscribe(event => {
|
|
458
|
+
observer.next({
|
|
459
|
+
event,
|
|
460
|
+
dragDistance: event.screenX - (lastScreenX ?? event.screenX),
|
|
461
|
+
scrollDistance: viewport().scrollLeft - (lastScrollLeft ?? viewport().scrollLeft),
|
|
462
|
+
});
|
|
463
|
+
lastScreenX = event.screenX;
|
|
464
|
+
lastScrollLeft = viewport().scrollLeft;
|
|
465
|
+
});
|
|
466
|
+
return () => subscription.unsubscribe();
|
|
467
|
+
});
|
|
468
|
+
}
|
|
401
469
|
}
|
|
402
|
-
|
|
403
|
-
|
|
404
|
-
|
|
405
|
-
|
|
470
|
+
/**
|
|
471
|
+
* Emits on {@link _viewportChange$} when the viewport size or scroll position changes.
|
|
472
|
+
*/
|
|
473
|
+
installViewportChangeTracker() {
|
|
474
|
+
const viewportSize = dimension(computed(() => this._viewportComponentElement().nativeElement));
|
|
475
|
+
const viewportClientSize = dimension(computed(() => this._viewportComponent().viewportClientElement));
|
|
476
|
+
const zone = inject(NgZone);
|
|
477
|
+
// Detect resizing.
|
|
478
|
+
effect(() => {
|
|
479
|
+
// Track viewport and viewport client sizes.
|
|
480
|
+
viewportSize();
|
|
481
|
+
viewportClientSize();
|
|
482
|
+
untracked(() => this._viewportChange$.next());
|
|
483
|
+
});
|
|
484
|
+
// Detect scrolling.
|
|
485
|
+
effect(onCleanup => {
|
|
486
|
+
const viewport = this._viewportComponent().viewportElement;
|
|
487
|
+
untracked(() => {
|
|
488
|
+
const subscription = fromEvent(viewport, 'scroll')
|
|
489
|
+
.pipe(subscribeIn(fn => zone.runOutsideAngular(fn)))
|
|
490
|
+
.subscribe(() => this._viewportChange$.next());
|
|
491
|
+
onCleanup(() => subscription.unsubscribe());
|
|
492
|
+
});
|
|
493
|
+
});
|
|
406
494
|
}
|
|
407
|
-
|
|
408
|
-
|
|
409
|
-
|
|
410
|
-
|
|
411
|
-
|
|
412
|
-
|
|
413
|
-
|
|
495
|
+
/**
|
|
496
|
+
* Tracks the tabbar indents.
|
|
497
|
+
*/
|
|
498
|
+
installTabbarIndentSizeTracker() {
|
|
499
|
+
const tabCornerRadiusDimension = dimension(this._tabCornerRadiusElement);
|
|
500
|
+
const paddingInlineDimension = dimension(this._paddingInlineElement);
|
|
501
|
+
effect(() => {
|
|
502
|
+
const indent = tabCornerRadiusDimension().clientWidth + paddingInlineDimension().clientWidth;
|
|
503
|
+
this._tabbarIndent = { left: indent, right: indent };
|
|
504
|
+
setCssVariable(this._host, {
|
|
505
|
+
'--ɵpart-bar-indent-left': `${this._tabbarIndent.left}px`,
|
|
506
|
+
'--ɵpart-bar-indent-right': `${this._tabbarIndent.right}px`,
|
|
507
|
+
});
|
|
414
508
|
});
|
|
415
509
|
}
|
|
416
|
-
|
|
417
|
-
|
|
510
|
+
ngOnDestroy() {
|
|
511
|
+
this.unsetDragState();
|
|
512
|
+
}
|
|
513
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PartBarComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
514
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.0.2", type: PartBarComponent, isStandalone: true, selector: "wb-part-bar", host: { listeners: { "dblclick": "onDoubleClick($event)" } }, viewQueries: [{ propertyName: "_viewTabs", predicate: ViewTabComponent, descendants: true, isSignal: true }, { propertyName: "_viewportComponent", first: true, predicate: SciViewportComponent, descendants: true, isSignal: true }, { propertyName: "_viewportComponentElement", first: true, predicate: SciViewportComponent, descendants: true, read: ElementRef, isSignal: true }, { propertyName: "_tabCornerRadiusElement", first: true, predicate: ["tab_corner_radius"], descendants: true, read: ElementRef, isSignal: true }, { propertyName: "_paddingInlineElement", first: true, predicate: ["padding_inline"], descendants: true, read: ElementRef, isSignal: true }, { propertyName: "_fillerElement", first: true, predicate: ["filler"], descendants: true, isSignal: true }], ngImport: i0, template: "<sci-viewport class=\"tabbar e2e-tabbar\" scrollbarStyle=\"hidden\">\n @for (viewId of part.viewIds(); track viewId) {\n @if (viewtab === dropTargetViewTab()) {\n <div class=\"divider\"></div>\n }\n <wb-view-tab #viewtab\n [viewId]=\"viewId\"\n [class.drag-source]=\"viewtab === dragSourceViewTab()\"\n [class.drop-target]=\"viewtab === dropTargetViewTab()\"/>\n <div class=\"divider\"></div>\n }\n\n <!-- Placeholder for expanding the viewport when dragging a tab to the right, moving part actions along with the dragover event. -->\n <div class=\"drag-placeholder\"></div>\n</sci-viewport>\n\n<!-- Actions -->\n@if (startActions().length) {\n <wb-part-action-bar [actions]=\"startActions()\" class=\"start\"/>\n}\n<div class=\"filler\" #filler></div>\n@if (endActions().length) {\n <wb-part-action-bar [actions]=\"endActions()\" class=\"end\"/>\n}\n\n<!-- View List Dropdown -->\n<wb-view-list-button/>\n\n<!-- Element for injecting the corner radius into the component. -->\n<div class=\"tab-corner-radius\" #tab_corner_radius></div>\n<!-- Element for injecting the tabbar padding into the component. -->\n<div class=\"padding-inline\" #padding_inline></div>\n", styles: ["@charset \"UTF-8\";:host{--\\275part-bar-indent-left: 0;--\\275part-bar-indent-right: 0;--\\275part-bar-drag-source-width: 0;--\\275part-bar-drag-placeholder-width: 0;display:flex;color:var(--sci-workbench-tab-text-color);background-color:var(--sci-workbench-part-bar-background-color);border-top:var(--sci-workbench-part-bar-padding-top) solid transparent;background-clip:border-box;height:var(--sci-workbench-tab-height);box-sizing:content-box;position:relative;container-name:viewtab;container-type:size}wb-workbench:has(wb-main-area-layout) wb-part:not(.main-area) :host{background-color:var(--sci-workbench-part-peripheral-bar-background-color)}:host>sci-viewport.tabbar{flex:initial}:host>sci-viewport.tabbar::part(content){display:flex;padding-left:var(--\\275part-bar-indent-left);padding-right:var(--\\275part-bar-indent-right)}:host>sci-viewport.tabbar>wb-view-tab{flex:none}:host>sci-viewport.tabbar>wb-view-tab.drag-source{display:none}:host>sci-viewport.tabbar>div.divider{flex:none;display:flex;position:relative;width:0}:host>sci-viewport.tabbar>div.divider:after{flex:none;align-self:center;content:\"\";position:absolute;width:1px;left:-1px;height:65%;background-color:var(--sci-workbench-part-bar-divider-color)}:host>sci-viewport.tabbar>div.drag-placeholder{flex:none;width:var(--\\275part-bar-drag-placeholder-width)}:host>sci-viewport.tabbar>wb-view-tab.drop-target,:host>sci-viewport.tabbar wb-view-tab.drop-target~:is(wb-view-tab,div.divider),:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.drop-target){transform:translate(var(--\\275part-bar-drag-source-width))}:host>sci-viewport.tabbar>wb-view-tab:is(.active,.drag-source)+div.divider{display:none}:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.active:not(.drag-source)){display:none}:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.drag-source:not(.active)){display:none}:host:not(.drag-over)>sci-viewport.tabbar:not(:has(wb-view-tab:not(.drag-source)))::part(content){padding-right:0}:host>wb-part-action-bar.start{flex:none;padding-left:var(--sci-workbench-part-bar-actions-padding-inline)}:host>div.filler{flex:auto}:host>wb-part-action-bar.end{flex:none;padding-right:var(--sci-workbench-part-bar-actions-padding-inline)}:host>wb-view-list-button{flex:none}:host:before{content:\"\";position:absolute;bottom:0;left:0;right:0;height:1px;background-color:var(--sci-workbench-tab-border-color)}:host>div.tab-corner-radius{position:absolute;visibility:hidden;width:var(--sci-workbench-tab-border-radius)}:host>div.padding-inline{position:absolute;visibility:hidden;width:var(--sci-workbench-part-bar-padding-inline)}:host.on-drag-enter:not(.on-drag-start) wb-view-tab,:host.on-drag-enter:not(.on-drag-start) div.divider,:host.drag-over wb-view-tab,:host.drag-over div.divider,:host.on-drag-leave wb-view-tab,:host.on-drag-leave div.divider{transition:transform 175ms cubic-bezier(0,0,.2,1)}:host.on-drag-enter:not(.on-drag-start) div.drag-placeholder,:host.on-drag-leave div.drag-placeholder{transition:width 175ms cubic-bezier(0,0,.2,1)}:host:not(:has(wb-view-tab.drag-source)):not(:has(wb-view-tab.drop-target)):not(.drag-over) div.drag-placeholder{display:none}:host.pointer-events-disabled>sci-viewport.tabbar::part(content){pointer-events:none}\n"], dependencies: [{ kind: "component", type: SciViewportComponent, selector: "sci-viewport", inputs: ["scrollbarStyle"], outputs: ["scroll"] }, { kind: "component", type: ViewTabComponent, selector: "wb-view-tab", inputs: ["viewId"] }, { kind: "component", type: PartActionBarComponent, selector: "wb-part-action-bar", inputs: ["actions"] }, { kind: "component", type: ViewListButtonComponent, selector: "wb-view-list-button" }] }); }
|
|
418
515
|
}
|
|
419
516
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: PartBarComponent, decorators: [{
|
|
420
517
|
type: Component,
|
|
421
518
|
args: [{ selector: 'wb-part-bar', standalone: true, imports: [
|
|
422
|
-
NgFor,
|
|
423
|
-
AsyncPipe,
|
|
424
519
|
SciViewportComponent,
|
|
425
|
-
SciDimensionModule,
|
|
426
520
|
ViewTabComponent,
|
|
427
521
|
PartActionBarComponent,
|
|
428
522
|
ViewListButtonComponent,
|
|
429
|
-
|
|
430
|
-
|
|
431
|
-
}], ctorParameters: () => [{ type: i0.ElementRef }, { type: undefined, decorators: [{
|
|
432
|
-
type: Inject,
|
|
433
|
-
args: [WORKBENCH_ID]
|
|
434
|
-
}] }, { type: i1.WorkbenchLayoutService }, { type: i2.ɵWorkbenchRouter }, { type: i3.ViewTabDragImageRenderer }, { type: i4.ɵWorkbenchPart }, { type: i5.ViewDragService }, { type: i0.DestroyRef }, { type: i0.NgZone }], propDecorators: { _viewportElement: [{
|
|
435
|
-
type: ViewChild,
|
|
436
|
-
args: [SciViewportComponent, { static: true, read: ElementRef }]
|
|
437
|
-
}], _viewportComponent: [{
|
|
438
|
-
type: ViewChild,
|
|
439
|
-
args: [SciViewportComponent, { static: true }]
|
|
440
|
-
}], _tabCornerRadiusElement: [{
|
|
441
|
-
type: ViewChild,
|
|
442
|
-
args: ['tab_corner_radius', { static: true }]
|
|
443
|
-
}], _paddingInlineElement: [{
|
|
444
|
-
type: ViewChild,
|
|
445
|
-
args: ['padding_inline', { static: true }]
|
|
446
|
-
}], injectViewTabs: [{
|
|
447
|
-
type: ViewChildren,
|
|
448
|
-
args: [ViewTabComponent]
|
|
449
|
-
}], onDoubleClick: [{
|
|
523
|
+
], template: "<sci-viewport class=\"tabbar e2e-tabbar\" scrollbarStyle=\"hidden\">\n @for (viewId of part.viewIds(); track viewId) {\n @if (viewtab === dropTargetViewTab()) {\n <div class=\"divider\"></div>\n }\n <wb-view-tab #viewtab\n [viewId]=\"viewId\"\n [class.drag-source]=\"viewtab === dragSourceViewTab()\"\n [class.drop-target]=\"viewtab === dropTargetViewTab()\"/>\n <div class=\"divider\"></div>\n }\n\n <!-- Placeholder for expanding the viewport when dragging a tab to the right, moving part actions along with the dragover event. -->\n <div class=\"drag-placeholder\"></div>\n</sci-viewport>\n\n<!-- Actions -->\n@if (startActions().length) {\n <wb-part-action-bar [actions]=\"startActions()\" class=\"start\"/>\n}\n<div class=\"filler\" #filler></div>\n@if (endActions().length) {\n <wb-part-action-bar [actions]=\"endActions()\" class=\"end\"/>\n}\n\n<!-- View List Dropdown -->\n<wb-view-list-button/>\n\n<!-- Element for injecting the corner radius into the component. -->\n<div class=\"tab-corner-radius\" #tab_corner_radius></div>\n<!-- Element for injecting the tabbar padding into the component. -->\n<div class=\"padding-inline\" #padding_inline></div>\n", styles: ["@charset \"UTF-8\";:host{--\\275part-bar-indent-left: 0;--\\275part-bar-indent-right: 0;--\\275part-bar-drag-source-width: 0;--\\275part-bar-drag-placeholder-width: 0;display:flex;color:var(--sci-workbench-tab-text-color);background-color:var(--sci-workbench-part-bar-background-color);border-top:var(--sci-workbench-part-bar-padding-top) solid transparent;background-clip:border-box;height:var(--sci-workbench-tab-height);box-sizing:content-box;position:relative;container-name:viewtab;container-type:size}wb-workbench:has(wb-main-area-layout) wb-part:not(.main-area) :host{background-color:var(--sci-workbench-part-peripheral-bar-background-color)}:host>sci-viewport.tabbar{flex:initial}:host>sci-viewport.tabbar::part(content){display:flex;padding-left:var(--\\275part-bar-indent-left);padding-right:var(--\\275part-bar-indent-right)}:host>sci-viewport.tabbar>wb-view-tab{flex:none}:host>sci-viewport.tabbar>wb-view-tab.drag-source{display:none}:host>sci-viewport.tabbar>div.divider{flex:none;display:flex;position:relative;width:0}:host>sci-viewport.tabbar>div.divider:after{flex:none;align-self:center;content:\"\";position:absolute;width:1px;left:-1px;height:65%;background-color:var(--sci-workbench-part-bar-divider-color)}:host>sci-viewport.tabbar>div.drag-placeholder{flex:none;width:var(--\\275part-bar-drag-placeholder-width)}:host>sci-viewport.tabbar>wb-view-tab.drop-target,:host>sci-viewport.tabbar wb-view-tab.drop-target~:is(wb-view-tab,div.divider),:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.drop-target){transform:translate(var(--\\275part-bar-drag-source-width))}:host>sci-viewport.tabbar>wb-view-tab:is(.active,.drag-source)+div.divider{display:none}:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.active:not(.drag-source)){display:none}:host>sci-viewport.tabbar>div.divider:has(+wb-view-tab.drag-source:not(.active)){display:none}:host:not(.drag-over)>sci-viewport.tabbar:not(:has(wb-view-tab:not(.drag-source)))::part(content){padding-right:0}:host>wb-part-action-bar.start{flex:none;padding-left:var(--sci-workbench-part-bar-actions-padding-inline)}:host>div.filler{flex:auto}:host>wb-part-action-bar.end{flex:none;padding-right:var(--sci-workbench-part-bar-actions-padding-inline)}:host>wb-view-list-button{flex:none}:host:before{content:\"\";position:absolute;bottom:0;left:0;right:0;height:1px;background-color:var(--sci-workbench-tab-border-color)}:host>div.tab-corner-radius{position:absolute;visibility:hidden;width:var(--sci-workbench-tab-border-radius)}:host>div.padding-inline{position:absolute;visibility:hidden;width:var(--sci-workbench-part-bar-padding-inline)}:host.on-drag-enter:not(.on-drag-start) wb-view-tab,:host.on-drag-enter:not(.on-drag-start) div.divider,:host.drag-over wb-view-tab,:host.drag-over div.divider,:host.on-drag-leave wb-view-tab,:host.on-drag-leave div.divider{transition:transform 175ms cubic-bezier(0,0,.2,1)}:host.on-drag-enter:not(.on-drag-start) div.drag-placeholder,:host.on-drag-leave div.drag-placeholder{transition:width 175ms cubic-bezier(0,0,.2,1)}:host:not(:has(wb-view-tab.drag-source)):not(:has(wb-view-tab.drop-target)):not(.drag-over) div.drag-placeholder{display:none}:host.pointer-events-disabled>sci-viewport.tabbar::part(content){pointer-events:none}\n"] }]
|
|
524
|
+
}], ctorParameters: () => [], propDecorators: { onDoubleClick: [{
|
|
450
525
|
type: HostListener,
|
|
451
526
|
args: ['dblclick', ['$event']]
|
|
452
|
-
}], onViewDragStart: [{
|
|
453
|
-
type: HostListener,
|
|
454
|
-
args: ['dragstart', ['$event']]
|
|
455
|
-
}], onViewDragEnd: [{
|
|
456
|
-
type: HostListener,
|
|
457
|
-
args: ['dragend']
|
|
458
527
|
}] } });
|
|
459
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"part-bar.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/part/part-bar/part-bar.component.ts","../../../../../../../projects/scion/workbench/src/lib/part/part-bar/part-bar.component.html"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2023 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {Component, DestroyRef, ElementRef, HostListener, Inject, NgZone, OnInit, QueryList, ViewChild, ViewChildren} from '@angular/core';\nimport {ViewTabComponent} from '../view-tab/view-tab.component';\nimport {WorkbenchLayoutService} from '../../layout/workbench-layout.service';\nimport {filter, map, mergeMap, startWith, switchMap, take, takeUntil} from 'rxjs/operators';\nimport {animationFrameScheduler, BehaviorSubject, combineLatest, firstValueFrom, from, interval, Observable, Subject} from 'rxjs';\nimport {ConstrainFn, ViewDragImageRect, ViewTabDragImageRenderer} from '../../view-dnd/view-tab-drag-image-renderer.service';\nimport {ViewDragData, ViewDragService} from '../../view-dnd/view-drag.service';\nimport {getCssTranslation, setCssClass, setCssVariable, unsetCssClass, unsetCssVariable} from '../../common/dom.util';\nimport {ɵWorkbenchPart} from '../ɵworkbench-part.model';\nimport {filterArray, mapArray, observeInside, subscribeInside} from '@scion/toolkit/operators';\nimport {SciViewportComponent} from '@scion/components/viewport';\nimport {ɵWorkbenchRouter} from '../../routing/ɵworkbench-router.service';\nimport {SciDimensionModule} from '@scion/components/dimension';\nimport {AsyncPipe, NgFor, NgIf} from '@angular/common';\nimport {PartActionBarComponent} from '../part-action-bar/part-action-bar.component';\nimport {ViewListButtonComponent} from '../view-list-button/view-list-button.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {fromDimension$} from '@scion/toolkit/observable';\nimport {WORKBENCH_ID} from '../../workbench-id';\nimport {ViewId} from '../../view/workbench-view.model';\n\n/**\n * Renders view tabs and actions of a {@link WorkbenchPart}.\n *\n * Tabs are added to a viewport, which the user can scroll if not enough space. The viewport grows with its tabs,\n * allowing actions to be placed directly after the last tab or on the right side.\n *\n * ## Drag Events\n * We use native drag and drop to support dragging views to other windows.\n *\n * By listening to drag events, the tabbar visualizes tabs during a drag operation. The drag operation is initiated in {@link ViewTabComponent}.\n * - `dragstart` and `dragend` events are fired in the tabbar where the drag operation was initiated.\n * - `dragenter` event is fired when a tab enters the tabbar.\n * - `dragover` events are fired when dragging a tab over the tabbar.\n * - `dragleave` event is fired when a tab leaves the tabbar.\n * - `drop` event is fired when the user drops a tab in the tabbar.\n * - Note that `dragleave` event is not fired on drop action. Therefore, always handle both events, `dragleave` and `drop` events, respectively.\n *\n * ## Terminology\n * - Drag source: Tab that is being dragged.\n * - Drop target: Tab before which to insert the drag source on drop, or `end` if dropping it after the last tab.\n * - Drag image:  Image for the tab to be dragged. We do not use the native drag image support to control drag image position and size,\n *                e.g., to snap the drag image to the tabbar when dragging near it.\n */\n@Component({\n  selector: 'wb-part-bar',\n  templateUrl: './part-bar.component.html',\n  styleUrls: ['./part-bar.component.scss'],\n  standalone: true,\n  imports: [\n    NgFor,\n    AsyncPipe,\n    SciViewportComponent,\n    SciDimensionModule,\n    ViewTabComponent,\n    PartActionBarComponent,\n    ViewListButtonComponent,\n    NgIf,\n  ],\n})\nexport class PartBarComponent implements OnInit {\n\n  private readonly _host: HTMLElement;\n  private readonly _viewportChange$ = new Subject<void>();\n  private readonly _viewTabs$ = new BehaviorSubject<ViewTabComponent[]>([]);\n  private readonly _dragenter$ = new Subject<void>();\n  private readonly _dragleave$ = new Subject<void>();\n  private readonly _dragend$ = new Subject<void>();\n  public readonly dragover$: Observable<boolean>;\n\n  @ViewChild(SciViewportComponent, {static: true, read: ElementRef})\n  private _viewportElement!: ElementRef<HTMLElement>;\n\n  @ViewChild(SciViewportComponent, {static: true})\n  private _viewportComponent!: SciViewportComponent;\n\n  @ViewChild('tab_corner_radius', {static: true})\n  private _tabCornerRadiusElement!: ElementRef<HTMLElement>;\n\n  @ViewChild('padding_inline', {static: true})\n  private _paddingInlineElement!: ElementRef<HTMLElement>;\n\n  @ViewChildren(ViewTabComponent)\n  public set injectViewTabs(queryList: QueryList<ViewTabComponent>) {\n    queryList.changes\n      .pipe(startWith(queryList), takeUntilDestroyed(this._destroyRef))\n      .subscribe(queryList => this._viewTabs$.next(queryList.toArray()));\n  }\n\n  /**\n   * Reference to the tab before which to insert the drag source on drop, or `end` if dropping it after the last tab.\n   */\n  public dropTargetViewTab: ViewTabComponent | 'end' | null = null;\n\n  /**\n   * Reference to the tab where the drag operation was started.\n   * This reference is only set if the drag operation started on a tab of this tabbar.\n   */\n  public dragSourceViewTab: ViewTabComponent | null = null;\n\n  /**\n   * Transfer data of the tab being dragged over this tabbar.\n   */\n  private _dragData: ViewDragData | null = null;\n\n  /**\n   * Function for snapping the drag image to the tabbar when dragging near it.\n   */\n  private _constrainFn: ConstrainFn | null = null;\n\n  /**\n   * Tabbar indents where to display the rounded bottom corners of the first and last view tab.\n   */\n  private _tabbarIndent = {\n    left: 0,\n    right: 0,\n  };\n\n  /**\n   * Notifies when the drag animation has finished, i.e., tabs have transitioned to their final position.\n   */\n  private _dragAnimationStable$ = interval(0, animationFrameScheduler)\n    .pipe(\n      filter(() => this.isDragAnimationStable()),\n      subscribeInside(continueFn => this._zone.runOutsideAngular(continueFn)),\n      observeInside(continueFn => this._zone.run(continueFn)),\n      map(() => undefined as void),\n    );\n\n  constructor(host: ElementRef<HTMLElement>,\n              @Inject(WORKBENCH_ID) private _workbenchId: string,\n              private _workbenchLayoutService: WorkbenchLayoutService,\n              private _router: ɵWorkbenchRouter,\n              private _viewTabDragImageRenderer: ViewTabDragImageRenderer,\n              private _part: ɵWorkbenchPart,\n              private _viewDragService: ViewDragService,\n              private _destroyRef: DestroyRef,\n              private _zone: NgZone) {\n    this._host = host.nativeElement;\n    this.dragover$ = this._viewDragService.tabbarDragOver$.pipe(map(partId => partId === this._part.id));\n  }\n\n  public ngOnInit(): void {\n    this.installActiveViewScroller();\n    this.installScrolledIntoViewUpdater();\n    this.installViewDragListener();\n    this.installTabbarIndentSizeDetector();\n    this.installViewportClientSizeDetector();\n  }\n\n  @HostListener('dblclick', ['$event'])\n  public onDoubleClick(event: MouseEvent): void {\n    if (this._part.isInMainArea) {\n      this._router.navigate(layout => layout.toggleMaximized()).then();\n    }\n    event.stopPropagation();\n  }\n\n  public get viewIds$(): Observable<ViewId[]> {\n    return this._part.viewIds$;\n  }\n\n  public onTabbarViewportDimensionChange(): void {\n    this._viewportChange$.next();\n  }\n\n  public onScroll(): void {\n    this._viewportChange$.next();\n  }\n\n  /**\n   * Method invoked when the user starts dragging a tab of this tabbar.\n   */\n  @HostListener('dragstart', ['$event'])\n  public onViewDragStart(event: DragEvent): void {\n    if (!this._viewDragService.isViewDragEvent(event)) {\n      return;\n    }\n\n    const dragData = this._viewDragService.viewDragData!;\n    const dragSourceIndex = this._viewTabs.findIndex(viewTab => viewTab.viewId === dragData.viewId);\n    const dragSourceViewTab = this._viewTabs[dragSourceIndex];\n    const dropTargetViewTab = this._viewTabs[dragSourceIndex + 1] ?? 'end';\n\n    // Constrain function for snapping the drag image to the tabbar when dragging near it.\n    this._constrainFn = this.createDragImageConstrainFn();\n    this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);\n    this._viewDragService.setTabbarDragover(this._part.id);\n\n    // When start dragging a view tab, we remove it from the tabbar by setting its `display` to `none` and render its drag image in `ViewTabDragImageRenderer` instead.\n    // However, do not unset the `display` of the drag source during `dragstart`. Otherwise, the native drag operation would be corrupted in Chrome and Edge (but not in Firefox).\n    animationFrameScheduler.schedule(() => {\n      this.dragSourceViewTab = dragSourceViewTab;\n      this.dropTargetViewTab = dropTargetViewTab; // Set drop target to avoid tab animation.\n      setCssVariable(this._host, {\n        '--ɵpart-bar-drag-source-width': `${dragData.viewTabWidth}px`,\n        '--ɵpart-bar-drag-image-placeholder-width': `${dragData.viewTabWidth}px`,\n      });\n    });\n\n    // When dragging a tab very quickly out of this tabbar, only the `dragstart` event gets fired, but not the `dragenter` event.\n    // Therefore, we subscribe to global `dragenter` events and manually invoke `onTabbarDragLeave` to correctly display the tabs and the active view.\n    this._viewDragService.viewDrag$(window, {eventType: ['dragenter']})\n      .pipe(take(1), takeUntil(this._dragend$), takeUntilDestroyed(this._destroyRef))\n      .subscribe(event => {\n        // Check if the user continues dragging over this tabbar.\n        // If not, the user has quickly dragged the tab out of the tabbar and we  manually call `onTabbarDragLeave`.\n        if ((this._host.compareDocumentPosition(event.target as Element) & Node.DOCUMENT_POSITION_CONTAINED_BY) === 0) {\n          this.onTabbarDragLeave();\n        }\n      });\n  }\n\n  /**\n   * Method invoked when the user ends dragging a tab of this tabbar.\n   *\n   * Invoked only in the tabbar where the drag operation started, regardless of whether the drag operation was completed or canceled.\n   */\n  @HostListener('dragend')\n  public onViewDragEnd(): void {\n    this._dragend$.next();\n    this.unsetDragState();\n  }\n\n  /**\n   * Method invoked when the user drags a tab into this tabbar.\n   */\n  private onTabbarDragEnter(event: DragEvent): void {\n    this._dragenter$.next();\n    this._viewDragService.setTabbarDragover(this._part.id);\n    this._dragData = this._viewDragService.viewDragData!;\n\n    // Constrain function for snapping the drag image to the tabbar when dragging near it.\n    if (!this._constrainFn) {\n      this._constrainFn = this.createDragImageConstrainFn();\n      this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);\n    }\n\n    // Locate drop target to have a nice animation for the drag source placeholder.\n    this.dropTargetViewTab = this.computeDropTarget(event);\n\n    // Set the CSS class 'drag-enter' to indicate entering the tabbar.\n    setCssClass(this._host, 'drag-enter', 'drag-over');\n    this._dragAnimationStable$\n      .pipe(take(1), takeUntil(this._dragenter$), takeUntilDestroyed(this._destroyRef))\n      .subscribe({complete: () => unsetCssClass(this._host, 'drag-enter')});\n\n    setCssVariable(this._host, {\n      '--ɵpart-bar-drag-source-width': `${this._dragData.viewTabWidth}px`,\n      '--ɵpart-bar-drag-image-placeholder-width': `${this._dragData.viewTabWidth}px`,\n    });\n  }\n\n  /**\n   * Method invoked when the user drags a tab out of this tabbar.\n   */\n  private onTabbarDragLeave(): void {\n    this._dragleave$.next();\n\n    // Set the CSS class 'drag-leave' to indicate leaving the tabbar.\n    setCssClass(this._host, 'drag-leave');\n    this._dragAnimationStable$\n      .pipe(take(1), takeUntil(this._dragleave$), takeUntilDestroyed(this._destroyRef))\n      .subscribe({complete: () => unsetCssClass(this._host, 'drag-leave')});\n\n    this.unsetDragState({unsetDragSource: false});\n\n    // Activate the view next to the view being dragged out of this tabbar. But, do not push the navigation into browsing history stack.\n    if (this.dragSourceViewTab && this.dragSourceViewTab.active) {\n      this._router.navigate(layout => layout.activateAdjacentView(this.dragSourceViewTab!.viewId), {skipLocationChange: true}).then();\n    }\n  }\n\n  /**\n   * Method invoked while the user is dragging a tab over this tabbar.\n   */\n  private onTabbarDragOver(event: DragEvent): void {\n    event.preventDefault(); // allow view drop\n\n    // Locate the tab over which the user is dragging the tab, but only after pending transitions have ended to prevent tabs from sliding back and forth.\n    if (this.isDragAnimationStable()) {\n      this.dropTargetViewTab = this.computeDropTarget(event);\n      // Synchronize the width of the drag source placeholder with the drag pointer position to move actions along with the pointer.\n      setCssVariable(this._host, {'--ɵpart-bar-drag-image-placeholder-width': `${this.calculateDragImagePlaceholderWidth(event)}px`});\n    }\n  }\n\n  /**\n   * Method invoked when the user drops a tab in this tabbar.\n   */\n  private onTabbarDrop(): void {\n    this._viewDragService.dispatchViewMoveEvent({\n      source: {\n        workbenchId: this._dragData!.workbenchId,\n        partId: this._dragData!.partId,\n        viewId: this._dragData!.viewId,\n        alternativeViewId: this._dragData!.alternativeViewId,\n        viewUrlSegments: this._dragData!.viewUrlSegments,\n        navigationHint: this._dragData!.navigationHint,\n        classList: this._dragData!.classList,\n      },\n      target: {\n        workbenchId: this._workbenchId,\n        position: this.dropTargetViewTab === 'end' ? 'end' : this._viewTabs.indexOf(this.dropTargetViewTab!),\n        elementId: this._part.id,\n      },\n    });\n\n    // Wait undoing the drag state until flushed tab changes to the DOM, i.e., after the layout routing cycle completes.\n    // If we were to undo the drag state immediately during the course of the drop event, the tab layout would temporarily\n    // revert to the state before the drag operation, resulting in an ugly flicker.\n    firstValueFrom(this._workbenchLayoutService.onLayoutChange$).then(() => {\n      this.unsetDragState();\n    });\n  }\n\n  /**\n   * Locates the tab before which the dragged tab is to be inserted when it is dropped. Dragging beyond the last tab returns `end`.\n   */\n  public computeDropTarget(event: DragEvent): ViewTabComponent | 'end' {\n    const viewDragImageRect = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData!, event);\n    const viewTabs = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab);\n\n    for (const viewTab of viewTabs) {\n      const viewTabRect = viewTab.host.getBoundingClientRect();\n      const viewTabMiddle = viewTabRect.left + (viewTabRect.width / 2);\n      if (viewDragImageRect.left < viewTabMiddle && (viewTab !== this.dropTargetViewTab || viewDragImageRect.right < viewTabMiddle)) {\n        return viewTab;\n      }\n    }\n    return 'end';\n  }\n\n  /**\n   * Creates the function for snapping the drag image to the tabbar when dragging near it.\n   */\n  private createDragImageConstrainFn(): ConstrainFn {\n    const hostLeft = this._host.getBoundingClientRect().left;\n    const maxViewportWidth = this.calculateMaxViewportWidth();\n    return (rect: ViewDragImageRect): ViewDragImageRect => {\n      const viewportBoundingBox = this._viewportElement.nativeElement.getBoundingClientRect();\n      return new ViewDragImageRect({\n        x: Math.min(Math.max(hostLeft + this._tabbarIndent.left, rect.x), hostLeft + maxViewportWidth - rect.width - this._tabbarIndent.right),\n        y: viewportBoundingBox.top,\n        width: rect.width,\n        height: viewportBoundingBox.height, // fit into tabbar, e.g., when dragging a tab into a tabbar of a different height\n      });\n    };\n  }\n\n  /**\n   * Calculates the maximum width of the tabbar viewport before an overflow would occur, which can then be used to constrain the drag image.\n   *\n   * To calculate the effective width, we need to simulate an overflow in the DOM. Therefore, call this method only from `dragstart` and not\n   * from `dragover` and disable animations.\n   */\n  private calculateMaxViewportWidth(): number {\n    const currentWidth = this._host.style.getPropertyValue('--ɵpart-bar-drag-image-placeholder-width');\n\n    // Disable animations during calculation and force the viewport to overflow.\n    setCssClass(this._host, 'calculating-max-viewport-width');\n    setCssVariable(this._host, {'--ɵpart-bar-drag-image-placeholder-width': `${this._host.clientWidth}px`});\n    try {\n      return this._viewportElement.nativeElement.getBoundingClientRect().width;\n    }\n    finally {\n      setCssVariable(this._host, {'--ɵpart-bar-drag-image-placeholder-width': currentWidth || null});\n      unsetCssClass(this._host, 'calculating-max-viewport-width');\n    }\n  }\n\n  /**\n   * Calculates the width for the drag image placeholder.\n   *\n   * - When dragging over tabs, the width is equal to the width of the drag source.\n   * - When dragging to the right of the last tab, the width of the placeholder is calculated as the distance between the\n   *   last tab and the position of the dragover event. Applying this width will resize the viewport, moving part actions along\n   *   with the dragover event.\n   */\n  private calculateDragImagePlaceholderWidth(event: DragEvent): number {\n    const viewDragImageRect = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData!, event);\n    const lastViewTab = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab).at(-1);\n    const lastViewTabRight = lastViewTab?.host.getBoundingClientRect().right ?? this._host.getBoundingClientRect().left + this._tabbarIndent.left;\n    return Math.max(lastViewTabRight + viewDragImageRect.width, viewDragImageRect.right) - lastViewTabRight;\n  }\n\n  /**\n   * Indicates whether CSS animations have finished when dragging a tab over the tabbar.\n   * The animation is considered stable when all tabs have moved to their final position.\n   */\n  private isDragAnimationStable(): boolean {\n    if (!this.dropTargetViewTab) {\n      return this._viewTabs.every(viewTab => getCssTranslation(viewTab.host).translateX === 'none');\n    }\n\n    const viewTabs = this._viewTabs.filter(viewTab => viewTab !== this.dragSourceViewTab);\n    const dropTargetIndex = this.dropTargetViewTab === 'end' ? viewTabs.length : viewTabs.indexOf(this.dropTargetViewTab!);\n    const dragSourceWidth = this._dragData!.viewTabWidth;\n\n    return viewTabs.every((viewTab, i) => {\n      const viewTabTranslateX = getCssTranslation(viewTab.host).translateX;\n      // Expect tabs preceding the drop target not to be shifted to the right.\n      if (i < dropTargetIndex && viewTabTranslateX !== 'none') {\n        return false;\n      }\n      // Expect drop target and tabs following the drop target to be shifted to the right.\n      if (i >= dropTargetIndex && Math.floor(Number(viewTabTranslateX)) !== Math.floor(dragSourceWidth)) { // compare floored numbers because `translateX` is not precise\n        return false;\n      }\n      return true;\n    });\n  }\n\n  private unsetDragState(options?: {unsetDragSource?: boolean}): void {\n    if (options?.unsetDragSource ?? true) {\n      this.dragSourceViewTab = null;\n    }\n\n    this.dropTargetViewTab = null;\n    this._dragData = null;\n    this._viewTabDragImageRenderer.unsetConstrainDragImageRectFn(this._constrainFn!);\n    this._constrainFn = null;\n\n    unsetCssClass(this._host, 'drag-over');\n    unsetCssVariable(this._host, '--ɵpart-bar-drag-source-width', '--ɵpart-bar-drag-image-placeholder-width');\n    this._viewDragService.unsetTabbarDragover(this._part.id);\n  }\n\n  private get _viewTabs(): ViewTabComponent[] {\n    return this._viewTabs$.value;\n  }\n\n  /**\n   * Scrolls the tab of the active view into view.\n   */\n  private installActiveViewScroller(): void {\n    this._viewTabs$\n      .pipe(\n        switchMap(viewTabs => combineLatest(viewTabs.map(viewTab => viewTab.view.active$))),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(() => {\n        // There may be no active view in the tabbar, e.g., when dragging the last view out of the tabbar.\n        const activeViewTab = this._viewTabs.find(viewTab => viewTab.active);\n        if (activeViewTab && !this._viewportComponent.isElementInView(activeViewTab.host, 'full')) {\n          this._viewportComponent.scrollIntoView(activeViewTab.host);\n        }\n      });\n  }\n\n  /**\n   * Updates {@link ɵWorkbenchView} when its tab is scrolled into view.\n   */\n  private installScrolledIntoViewUpdater(): void {\n    this._viewportChange$\n      .pipe(\n        map(() => this._viewTabs),\n        filterArray(viewTab => viewTab !== this.dragSourceViewTab), // skip drag source as always scrolled out of view\n        mergeMap(viewTabs => from(viewTabs)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(viewTab => {\n        viewTab.view.scrolledIntoView = this._viewportComponent.isElementInView(viewTab.host, 'full');\n      });\n  }\n\n  private installViewDragListener(): void {\n    this._viewDragService.viewDrag$(this._host)\n      .pipe(takeUntilDestroyed(this._destroyRef))\n      .subscribe((event: DragEvent) => {\n        switch (event.type) {\n          case 'dragenter':\n            this.onTabbarDragEnter(event);\n            break;\n          case 'dragover':\n            this.onTabbarDragOver(event);\n            break;\n          case 'dragleave':\n            this.onTabbarDragLeave();\n            break;\n          case 'drop':\n            this.onTabbarDrop();\n            break;\n        }\n      });\n  }\n\n  private installViewportClientSizeDetector(): void {\n    fromDimension$(this._viewportComponent.viewportClientElement)\n      .pipe(\n        subscribeInside(fn => this._zone.run(fn)),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(() => this._viewportChange$.next());\n  }\n\n  private installTabbarIndentSizeDetector(): void {\n    combineLatest([fromDimension$(this._tabCornerRadiusElement.nativeElement), fromDimension$(this._paddingInlineElement.nativeElement)])\n      .pipe(\n        subscribeInside(fn => this._zone.runOutsideAngular(fn)),\n        mapArray(dimension => dimension.clientWidth),\n        takeUntilDestroyed(this._destroyRef),\n      )\n      .subscribe(([tabCornerRadius, paddingInline]) => {\n        this._tabbarIndent = {left: tabCornerRadius + paddingInline, right: tabCornerRadius + paddingInline};\n        setCssVariable(this._host, {'--ɵpart-bar-indent-left': `${this._tabbarIndent.left}px`});\n        setCssVariable(this._host, {'--ɵpart-bar-indent-right': `${this._tabbarIndent.right}px`});\n      });\n  }\n}\n","<sci-viewport class=\"tabbar\" scrollbarStyle=\"hidden\"\n              [class.dragover]=\"dragover$ | async\"\n              sciDimension (sciDimensionChange)=\"onTabbarViewportDimensionChange()\"\n              (scroll)=\"onScroll()\">\n  <ng-container *ngFor=\"let viewId of viewIds$ | async;\">\n    <div class=\"divider\" *ngIf=\"viewtab === dropTargetViewTab\"></div>\n    <wb-view-tab #viewtab\n                 [viewId]=\"viewId\"\n                 [class.drag-source]=\"viewtab === dragSourceViewTab\"\n                 [class.drop-target]=\"viewtab === dropTargetViewTab\">\n    </wb-view-tab>\n    <div class=\"divider\"></div>\n  </ng-container>\n\n  <!-- Placeholder element to grow the viewport to fit the drag image. -->\n  <div class=\"drag-image-placeholder\"></div>\n</sci-viewport>\n<wb-part-action-bar/>\n<wb-view-list-button/>\n\n<!-- Element for injecting the corner radius into the component. -->\n<div class=\"tab-corner-radius\" #tab_corner_radius></div>\n<!-- Element for injecting the tabbar padding into the component. -->\n<div class=\"padding-inline\" #padding_inline></div>\n"]}
|
|
528
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"part-bar.component.js","sourceRoot":"","sources":["../../../../../../../projects/scion/workbench/src/lib/part/part-bar/part-bar.component.ts","../../../../../../../projects/scion/workbench/src/lib/part/part-bar/part-bar.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, computed, effect, ElementRef, HostListener, inject, NgZone, OnDestroy, Signal, signal, untracked, viewChild, viewChildren} from '@angular/core';\nimport {ViewTabComponent} from '../view-tab/view-tab.component';\nimport {map, mergeMap, takeUntil} from 'rxjs/operators';\nimport {from, fromEvent, merge, Observable, OperatorFunction, Subject, timer} from 'rxjs';\nimport {ConstrainFn, ViewTabDragImageRenderer} from '../../view-dnd/view-tab-drag-image-renderer.service';\nimport {ViewDragData, ViewDragService, ViewMoveEvent} from '../../view-dnd/view-drag.service';\nimport {getCssTranslation, setCssClass, setCssVariable, unsetCssClass, unsetCssVariable} from '../../common/dom.util';\nimport {ɵWorkbenchPart} from '../ɵworkbench-part.model';\nimport {filterArray, subscribeIn} from '@scion/toolkit/operators';\nimport {SciViewportComponent} from '@scion/components/viewport';\nimport {ɵWorkbenchRouter} from '../../routing/ɵworkbench-router.service';\nimport {boundingClientRect, dimension} from '@scion/components/dimension';\nimport {PartActionBarComponent} from '../part-action-bar/part-action-bar.component';\nimport {ViewListButtonComponent} from '../view-list-button/view-list-button.component';\nimport {takeUntilDestroyed} from '@angular/core/rxjs-interop';\nimport {WORKBENCH_ID} from '../../workbench-id';\nimport {clamp} from '../../common/math.util';\n\n/**\n * Renders view tabs and actions of a {@link WorkbenchPart}.\n *\n * Tabs are added to a viewport, which the user can scroll if not enough space. The viewport grows with its tabs,\n * allowing actions to be placed directly after the last tab or on the right side.\n *\n * ## Drag and Drop\n * This component subscribes to drag events to visualize tabs during a drag operation. The drag operation is initiated in {@link ViewTabComponent}.\n *\n * We use native drag and drop to support dragging views to other windows.\n *\n * Following events are triggered:\n * - dragstart: when start dragging a tab (only received in the tabbar where the drag operation has started)\n * - dragend:   when end dragging a tab, either via drop or cancel (only received in the tabbar where the drag operation has started)\n * - dragenter: when a tab enters the tabbar\n * - dragover:  when dragging a tab over the tabbar, every few hundred milliseconds\n * - dragleave: when a tab leaves the tabbar\n * - drop:      when the user drops a tab in the tabbar\n *\n * - Note that `dragleave` event is not fired on drop action. Therefore, always handle both events, `dragleave` and `drop` events, respectively.\n *\n * ## Terminology\n * - Drag source: Tab that is being dragged.\n * - Drop target: Tab before which to insert the drag source on drop, or `end` if dropping it after the last tab.\n * - Drag image:  Image for the tab to be dragged. We do not use the native drag image support to control drag image position and size,\n *                e.g., to snap the drag image to the tabbar when dragging near it.\n */\n@Component({\n  selector: 'wb-part-bar',\n  templateUrl: './part-bar.component.html',\n  styleUrls: ['./part-bar.component.scss'],\n  standalone: true,\n  imports: [\n    SciViewportComponent,\n    ViewTabComponent,\n    PartActionBarComponent,\n    ViewListButtonComponent,\n  ],\n})\nexport class PartBarComponent implements OnDestroy {\n\n  private readonly _host = inject(ElementRef<HTMLElement>).nativeElement;\n  private readonly _viewportChange$ = new Subject<void>();\n  private readonly _workbenchId = inject(WORKBENCH_ID);\n  private readonly _router = inject(ɵWorkbenchRouter);\n  private readonly _viewTabDragImageRenderer = inject(ViewTabDragImageRenderer);\n  private readonly _viewDragService = inject(ViewDragService);\n  private readonly _zone = inject(NgZone);\n\n  private readonly _viewTabs = viewChildren(ViewTabComponent);\n  private readonly _hostBoundingClientRect = boundingClientRect(inject(ElementRef));\n  private readonly _viewportComponent = viewChild.required(SciViewportComponent);\n  private readonly _viewportComponentElement = viewChild.required(SciViewportComponent, {read: ElementRef<HTMLElement>});\n  private readonly _tabCornerRadiusElement = viewChild.required('tab_corner_radius', {read: ElementRef<HTMLElement>});\n  private readonly _paddingInlineElement = viewChild.required('padding_inline', {read: ElementRef<HTMLElement>});\n  private readonly _fillerElement = viewChild.required<ElementRef<HTMLElement>>('filler');\n  private readonly _viewportBoundingBox = boundingClientRect(this._viewportComponentElement);\n  private readonly _maxViewportWidth: Signal<number>;\n\n  protected readonly part = inject(ɵWorkbenchPart);\n  protected readonly startActions = computed(() => this.part.actions().filter(action => action.align !== 'end'));\n  protected readonly endActions = computed(() => this.part.actions().filter(action => action.align === 'end'));\n\n  /**\n   * Reference to the tab before which to insert the drag source on drop, or `end` if dropping it after the last tab.\n   */\n  protected dropTargetViewTab = signal<ViewTabComponent | 'end' | null>(null);\n\n  /**\n   * Reference to the tab where the drag operation was started.\n   * This reference is only set if the drag operation started on a tab of this tabbar.\n   */\n  protected dragSourceViewTab = signal<ViewTabComponent | null>(null);\n\n  /**\n   * Transfer data of the tab being dragged over this tabbar.\n   */\n  private _dragData: ViewDragData | null = null;\n\n  /**\n   * Function for snapping the drag image to the tabbar when dragging near it.\n   */\n  private _constrainFn: ConstrainFn | null = null;\n\n  /**\n   * Signals when unsetting drag state.\n   */\n  private _onUnsetDragState = signal<void>(undefined, {equal: () => false});\n\n  /**\n   * Tabbar indents where to display the rounded bottom corners of the first and last view tab.\n   */\n  private _tabbarIndent = {\n    left: 0,\n    right: 0,\n  };\n\n  constructor() {\n    this._maxViewportWidth = this.calculateMaxViewportWidth();\n    this.installActiveViewScroller();\n    this.installScrolledIntoViewUpdater();\n    this.installViewDragListener();\n    this.installTabbarIndentSizeTracker();\n    this.installViewportChangeTracker();\n  }\n\n  @HostListener('dblclick', ['$event'])\n  protected onDoubleClick(event: MouseEvent): void {\n    if (this.part.isInMainArea) {\n      this._router.navigate(layout => layout.toggleMaximized()).then();\n    }\n    event.stopPropagation();\n  }\n\n  /**\n   * Method invoked when the user starts dragging a tab of this tabbar, only in the tabbar where the drag started.\n   */\n  private onTabDragStart(): void {\n    // Memoize drag data.\n    this._dragData = this._viewDragService.viewDragData()!;\n\n    // Indicate dragging over this tabbar.\n    this._viewDragService.signalTabbarDragEnter(this.part.id);\n\n    // Snap drag image to tabbar when dragging near it.\n    this._constrainFn = this.createDragImageConstrainFn();\n    this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);\n\n    // Disable pointer events to not break the drag operation (in Firefox, but not Chrome and Edge).\n    requestAnimationFrame(() => setCssClass(this._host, 'pointer-events-disabled'));\n\n    // Set CSS class to not animate start dragging.\n    setCssClass(this._host, 'on-drag-start');\n    this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-start'));\n\n    // Check if the user continues dragging over this tabbar.\n    // If not, the user has quickly dragged the tab out of the tabbar, so manually call `onTabbarDragEnter` and `onTabbarDragLeave`.\n    fromEvent<DragEvent>(window, 'dragenter', {once: true})\n      .pipe(takeUntil(merge(fromEvent(this._host, 'dragenter'), fromEvent(this._host, 'dragend'))))\n      .subscribe(event => {\n        this.onTabbarDragEnter(event);\n        this.onTabbarDragLeave(event);\n      });\n  }\n\n  /**\n   * Method invoked when the user ends dragging a tab of this tabbar, either by drop or cancel, only in the tabbar where the drag started.\n   */\n  private onTabDragEnd(event: DragEvent): void {\n    // Unset drag state only on cancel, not on drop, to prevent tabs from temporarily reverting to their pre-drag state. Drag state is reset in `onTabMoved` after a drop.\n    if (event.dataTransfer!.dropEffect === 'none') {\n      // Reactivate the drag source view on cancel.\n      this.dragSourceViewTab()?.view().activate({skipLocationChange: true}).then();\n      // Unset drag state.\n      this.unsetDragState();\n    }\n  }\n\n  /**\n   * Method invoked when the layout has changed after a drop. The drop may not be related to a tab of this tabbar.\n   */\n  private onTabMoved(event: ViewMoveEvent): void {\n    // Unset drag state only if no new drag has started since the drop.\n    if (this._dragData?.uid === event.dragData?.uid) {\n      this.unsetDragState();\n    }\n  }\n\n  /**\n   * Method invoked when the user drags a tab into this tabbar.\n   */\n  private onTabbarDragEnter(event: DragEvent): void {\n    // Memoize drag data.\n    const dragData = this._dragData = this._viewDragService.viewDragData()!;\n\n    // Indicate dragging over this tabbar.\n    this._viewDragService.signalTabbarDragEnter(this.part.id);\n\n    // Snap drag image to tabbar when dragging near it.\n    this._constrainFn = this.createDragImageConstrainFn();\n    this._viewTabDragImageRenderer.setConstrainDragImageRectFn(this._constrainFn);\n\n    // Set drag source, but only if started the drag operation in this part.\n    // We do not set the drag source in `dragstart` to not break the drag operation (in Chrome and Edge, but not Firefox).\n    if (this._host.classList.contains('on-drag-start')) {\n      this.dragSourceViewTab.set(this._viewTabs().find(viewTab => viewTab.viewId === dragData.viewId) ?? null);\n      setCssVariable(this._host, {'--ɵpart-bar-drag-placeholder-width': `${dragData.viewTabWidth}px`});\n    }\n\n    // Locate the tab the user is dragging over.\n    this.dropTargetViewTab.set(this.computeDropTarget(event));\n\n    // Disable pointer events on the tabbar for a smoother drag and drop experience when quickly swapping tabs.\n    setCssClass(this._host, 'pointer-events-disabled');\n\n    // Set CSS class to animate entering the tabbar.\n    setCssClass(this._host, 'on-drag-enter');\n    this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-enter'));\n\n    // Inject drag source width into CSS.\n    setCssVariable(this._host, {'--ɵpart-bar-drag-source-width': `${dragData.viewTabWidth}px`});\n  }\n\n  /**\n   * Method invoked when the user drags a tab out of this tabbar.\n   */\n  private onTabbarDragLeave(event: DragEvent): void {\n    // Set CSS class to animate leaving the tabbar, but not when canceling the drag operation for instant reset.\n    // Pointer coordinates are not set when the drag operation is canceled or when dragging the tab out of the window.\n    if (event.screenX || event.screenY) {\n      setCssClass(this._host, 'on-drag-leave');\n      this.onDragAnimationStable(() => unsetCssClass(this._host, 'on-drag-leave'));\n    }\n\n    // Activate the adjacent tab, only if the drag operation is not canceled.\n    // The drag event does not provide information about cancelation. Therefore, we wait for a macrotask, canceling it when receiving `dragend`, i.e., the drag operation has been canceled.\n    if (this.dragSourceViewTab()?.active) {\n      timer(0)\n        .pipe(takeUntil(fromEvent(window, 'dragend')))\n        .subscribe(() => this._router.navigate(layout => layout.activateAdjacentView(this.dragSourceViewTab()!.viewId), {skipLocationChange: true}));\n    }\n\n    // Clean up drag state.\n    this.unsetDragState({unsetDragSource: false, unsetDragData: false});\n  }\n\n  /**\n   * Method invoked when the user is dragging a tab over this tabbar.\n   */\n  private onTabbarDragOver(event: DragEvent, dragDistance: number, scrollDistance: number): void {\n    NgZone.assertNotInAngularZone();\n    event.preventDefault(); // allow view drop\n\n    // Locate the tab the user is dragging over.\n    // - Run in animation frame for better performance, most noticeable on lower CPU power.\n    // - Do not wait for the drag animation to complete for a smoother drag and drop experience when quickly moving the tab in the tabbar.\n    requestAnimationFrame(() => {\n      // Skip if the drag operation has ended.\n      if (this._viewDragService.isDragOverTabbar !== this.part.id) {\n        return;\n      }\n\n      // Ignore the drag distance if `drag-enter` animation is in progress to avoid incorrect drop target computation when dragging a tab from the left.\n      if (this._host.classList.contains('on-drag-enter')) {\n        this.dropTargetViewTab.set(this.computeDropTarget(event, 0));\n      }\n      // Compute drop target only when moving the drag pointer to prevent tabs from sliding back and forth (while animation is in progress)\n      else if (dragDistance) {\n        this.dropTargetViewTab.set(this.computeDropTarget(event, dragDistance));\n      }\n      // Compute drop target when scrolling the viewport, i.e., not moving the drag pointer but scrolling the viewport because placed the drag pointer at the viewport edges.\n      else if (scrollDistance) {\n        this.dropTargetViewTab.set(this.computeDropTarget(event, scrollDistance));\n      }\n\n      // Set CSS class to animate dragging over the tabbar.\n      setCssClass(this._host, 'drag-over');\n\n      // Calculate the new size of the drag source placeholder.\n      // The placeholder expands the viewport when dragging a tab to the right, moving part actions along with the dragover event.\n      setCssVariable(this._host, {'--ɵpart-bar-drag-placeholder-width': `${this.calculateDragPlaceholderWidth(event)}px`});\n    });\n  }\n\n  /**\n   * Method invoked when the user drops a tab in this tabbar.\n   */\n  private async onTabbarDrop(): Promise<void> {\n    const dragData = this._dragData!;\n    const dropTargetViewTab = this.dropTargetViewTab()!;\n    const dropPosition = dropTargetViewTab === 'end' ? 'end' : this._viewTabs().indexOf(dropTargetViewTab);\n\n    this._viewDragService.dispatchViewMoveEvent({\n      source: {\n        workbenchId: dragData.workbenchId,\n        partId: dragData.partId,\n        viewId: dragData.viewId,\n        alternativeViewId: dragData.alternativeViewId,\n        viewUrlSegments: dragData.viewUrlSegments,\n        navigationHint: dragData.navigationHint,\n        navigationData: dragData.navigationData,\n        classList: dragData.classList,\n      },\n      target: {\n        workbenchId: this._workbenchId,\n        position: dropPosition,\n        elementId: this.part.id,\n      },\n      dragData: dragData,\n    });\n  }\n\n  /**\n   * Locates the tab before which the dragged tab is to be inserted when it is dropped. Dragging beyond the last tab returns `end`.\n   */\n  private computeDropTarget(event: DragEvent, dragDistanceX = 0): ViewTabComponent | 'end' {\n    const dragImage = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData!, event);\n    const viewTabs = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab());\n\n    for (const viewTab of viewTabs) {\n      const {left, width} = viewTab.boundingClientRect();\n      const viewTabHCenter = left + (width / 2);\n\n      // Moving the pointer to the left.\n      if (dragDistanceX <= 0 && dragImage.left < viewTabHCenter) {\n        return viewTab;\n      }\n      // Moving the pinter to the right.\n      if (dragDistanceX > 0 && dragImage.right < viewTabHCenter) {\n        return viewTab;\n      }\n    }\n    return 'end';\n  }\n\n  /**\n   * Creates the function for snapping the drag image to the tabbar when dragging near it.\n   */\n  private createDragImageConstrainFn(): ConstrainFn {\n    return (dragImage: DOMRect): DOMRect => new DOMRect(\n      clamp(dragImage.x, {\n        min: this._hostBoundingClientRect().left + this._tabbarIndent.left,\n        max: this._hostBoundingClientRect().left + this._maxViewportWidth() - dragImage.width - this._tabbarIndent.right,\n      }),\n      this._viewportBoundingBox().top,\n      dragImage.width,\n      this._viewportBoundingBox().height, // fit into tabbar, e.g., when dragging a tab into a tabbar of a different height\n    );\n  }\n\n  /**\n   * Calculates the maximum width of the tabbar viewport before an overflow would occur.\n   * Use to constrain the drag image to the tabbar bounds.\n   */\n  private calculateMaxViewportWidth(): Signal<number> {\n    const fillerDimension = dimension(this._fillerElement);\n    return computed(() => this._viewportBoundingBox().width + fillerDimension().offsetWidth);\n  }\n\n  /**\n   * Calculates the width for the drag placeholder.\n   *\n   * The placeholder expands the viewport when dragging a tab to the right, moving part actions along with the dragover event.\n   *\n   * Calculation:\n   * - When dragging over tabs, the width is equal to the width of the drag source.\n   * - When dragging to the right of the last tab, the width of the placeholder is the distance between the last tab and the position of the drag pointer.\n   */\n  private calculateDragPlaceholderWidth(event: DragEvent): number {\n    const viewDragImageRect = this._viewTabDragImageRenderer.calculateDragImageRect(this._dragData!, event);\n    const lastViewTab = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab()).at(-1);\n    const lastViewTabRight = lastViewTab?.boundingClientRect().right ?? this._hostBoundingClientRect().left + this._tabbarIndent.left;\n    return Math.max(lastViewTabRight + viewDragImageRect.width, viewDragImageRect.right) - lastViewTabRight;\n  }\n\n  /**\n   * Indicates whether CSS animations have finished when dragging a tab over the tabbar.\n   * The animation is considered stable when all tabs have moved to their final position.\n   */\n  private isDragAnimationStable(): boolean {\n    const dropTargetViewTab = this.dropTargetViewTab();\n    if (this._viewDragService.isDragOverTabbar !== this.part.id) {\n      return this._viewTabs().every(viewTab => getCssTranslation(viewTab.host).translateX === 'none');\n    }\n\n    const viewTabs = this._viewTabs().filter(viewTab => viewTab !== this.dragSourceViewTab());\n    const dropTargetIndex = dropTargetViewTab === 'end' ? viewTabs.length : viewTabs.indexOf(dropTargetViewTab!);\n    const dragSourceWidth = this._dragData!.viewTabWidth;\n\n    return viewTabs.every((viewTab, i) => {\n      const viewTabTranslateX = getCssTranslation(viewTab.host).translateX;\n      // Expect tabs preceding the drop target not to be shifted to the right.\n      if (i < dropTargetIndex && viewTabTranslateX !== 'none') {\n        return false;\n      }\n      // Expect drop target and tabs following the drop target to be shifted to the right.\n      if (i >= dropTargetIndex && Math.floor(Number(viewTabTranslateX)) !== Math.floor(dragSourceWidth)) { // compare floored numbers because `translateX` is not precise\n        return false;\n      }\n      return true;\n    });\n  }\n\n  /**\n   * Cleans up the drag state, unsetting drag data and drag source based on provided options.\n   */\n  private unsetDragState(options?: {unsetDragSource?: false; unsetDragData?: false}): void {\n    this.dropTargetViewTab.set(null);\n    this._onUnsetDragState.set();\n    this._viewTabDragImageRenderer.unsetConstrainDragImageRectFn(this._constrainFn!);\n    this._viewDragService.signalTabbarDragLeave(this.part.id);\n    this._constrainFn = null;\n\n    unsetCssClass(this._host, 'drag-over', 'pointer-events-disabled');\n    unsetCssVariable(this._host, '--ɵpart-bar-drag-source-width', '--ɵpart-bar-drag-placeholder-width');\n\n    if (options?.unsetDragSource ?? true) {\n      this.dragSourceViewTab.set(null);\n    }\n    if (options?.unsetDragData ?? true) {\n      this._dragData = null;\n    }\n  }\n\n  /**\n   * Executes given function when the drag animation has finished, i.e., tabs have transitioned to their final position.\n   */\n  private onDragAnimationStable(callback: () => void): void {\n    if (NgZone.isInAngularZone()) {\n      this._zone.runOutsideAngular(() => this.onDragAnimationStable(callback));\n      return;\n    }\n    requestAnimationFrame(() => this.isDragAnimationStable() ? callback() : this.onDragAnimationStable(callback));\n  }\n\n  /**\n   * Scrolls the tab of the active view into view.\n   */\n  private installActiveViewScroller(): void {\n    effect(() => {\n      // Track the active view.\n      const activeViewId = this.part.activeViewId();\n      // There may be no active view, e.g., if no view has been activated, or when dragging the last view out of the tabbar.\n      if (!activeViewId) {\n        return;\n      }\n\n      // Do not track rendered tabs in general to prevent unwanted scrolling when opening or closing inactive tabs.\n      const viewTabs = untracked(() => this._viewTabs());\n\n      // Get a reference to the currently rendered active tab, waiting if necessary by tracking the rendered tabs.\n      const activeViewTab = viewTabs.find(viewTab => viewTab.viewId === activeViewId);\n      if (!activeViewTab) {\n        this._viewTabs(); // Track rendered tabs to re-execute once rendered.\n        return;\n      }\n\n      // Track navigation of the active view, scrolling its tab into view if necessary.\n      activeViewTab.view().navigationId();\n\n      // Track unsetting drag state to ensure the active view is scrolled into view after drop or cancel.\n      this._onUnsetDragState();\n\n      // Scroll the tab into view if scrolled out of view.\n      untracked(() => requestAnimationFrame(() => {\n        if (!this._viewportComponent().isElementInView(activeViewTab.host, 'full')) {\n          this._viewportComponent().scrollIntoView(activeViewTab.host);\n        }\n      }));\n    });\n  }\n\n  /**\n   * Updates {@link ɵWorkbenchView} when its tab is scrolled into view.\n   */\n  private installScrolledIntoViewUpdater(): void {\n    this._viewportChange$\n      .pipe(\n        map(() => this._viewTabs()),\n        filterArray(viewTab => viewTab !== this.dragSourceViewTab()), // skip drag source as always scrolled out of view\n        mergeMap(viewTabs => from(viewTabs)),\n        takeUntilDestroyed(),\n      )\n      .subscribe(viewTab => {\n        viewTab.view().scrolledIntoView = this._viewportComponent().isElementInView(viewTab.host, 'full');\n      });\n  }\n\n  /**\n   * Subscribes to drag events.\n   */\n  private installViewDragListener(): void {\n    const zone = inject(NgZone);\n\n    // Subscribe to drag events relevant for this tabbar.\n    this._viewDragService.viewDrag$(this._host)\n      .pipe(\n        subscribeIn(fn => zone.runOutsideAngular(fn)),\n        calculateDistance(() => this._viewportComponent()), // provide viewport lazily because not injected yet\n        takeUntilDestroyed(),\n      )\n      .subscribe(({event, dragDistance, scrollDistance}) => {\n        switch (event.type) {\n          case 'dragstart':\n            this.onTabDragStart();\n            break;\n          case 'dragend':\n            this.onTabDragEnd(event);\n            break;\n          case 'dragenter':\n            this.onTabbarDragEnter(event);\n            break;\n          case 'dragover':\n            this.onTabbarDragOver(event, dragDistance, scrollDistance);\n            break;\n          case 'dragleave':\n            this.onTabbarDragLeave(event);\n            break;\n          case 'drop':\n            this.onTabbarDrop().then();\n            break;\n        }\n      });\n\n    // Get notified when a view has moved in the layout.\n    this._viewDragService.viewMoved$\n      .pipe(\n        subscribeIn(fn => zone.runOutsideAngular(fn)),\n        takeUntilDestroyed(),\n      )\n      .subscribe(event => this.onTabMoved(event));\n\n    /**\n     * Calculates the distance the drag pointer and viewport have moved since the last drag event.\n     */\n    function calculateDistance(viewport: () => SciViewportComponent): OperatorFunction<DragEvent, DragEventPlusDistance> {\n      return source$ => new Observable<DragEventPlusDistance>(observer => {\n        let lastScreenX: number | undefined;\n        let lastScrollLeft: number | undefined;\n\n        const subscription = source$.subscribe(event => {\n          observer.next({\n            event,\n            dragDistance: event.screenX - (lastScreenX ?? event.screenX),\n            scrollDistance: viewport().scrollLeft - (lastScrollLeft ?? viewport().scrollLeft),\n          });\n          lastScreenX = event.screenX;\n          lastScrollLeft = viewport().scrollLeft;\n        });\n        return () => subscription.unsubscribe();\n      });\n    }\n  }\n\n  /**\n   * Emits on {@link _viewportChange$} when the viewport size or scroll position changes.\n   */\n  private installViewportChangeTracker(): void {\n    const viewportSize = dimension(computed(() => this._viewportComponentElement().nativeElement));\n    const viewportClientSize = dimension(computed(() => this._viewportComponent().viewportClientElement));\n    const zone = inject(NgZone);\n\n    // Detect resizing.\n    effect(() => {\n      // Track viewport and viewport client sizes.\n      viewportSize();\n      viewportClientSize();\n      untracked(() => this._viewportChange$.next());\n    });\n\n    // Detect scrolling.\n    effect(onCleanup => {\n      const viewport = this._viewportComponent().viewportElement;\n      untracked(() => {\n        const subscription = fromEvent(viewport, 'scroll')\n          .pipe(subscribeIn(fn => zone.runOutsideAngular(fn)))\n          .subscribe(() => this._viewportChange$.next());\n        onCleanup(() => subscription.unsubscribe());\n      });\n    });\n  }\n\n  /**\n   * Tracks the tabbar indents.\n   */\n  private installTabbarIndentSizeTracker(): void {\n    const tabCornerRadiusDimension = dimension(this._tabCornerRadiusElement);\n    const paddingInlineDimension = dimension(this._paddingInlineElement);\n\n    effect(() => {\n      const indent = tabCornerRadiusDimension().clientWidth + paddingInlineDimension().clientWidth;\n      this._tabbarIndent = {left: indent, right: indent};\n      setCssVariable(this._host, {\n        '--ɵpart-bar-indent-left': `${this._tabbarIndent.left}px`,\n        '--ɵpart-bar-indent-right': `${this._tabbarIndent.right}px`,\n      });\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this.unsetDragState();\n  }\n}\n\n/**\n * Contains the event plus drag and scroll distance.\n */\ninterface DragEventPlusDistance {\n  /**\n   * Reference to the drag event.\n   */\n  event: DragEvent;\n  /**\n   * Distance the drag pointer has moved since the last drag event.\n   */\n  dragDistance: number;\n  /**\n   * Distance the viewport has scrolled since the last drag event.\n   */\n  scrollDistance: number;\n}\n","<sci-viewport class=\"tabbar e2e-tabbar\" scrollbarStyle=\"hidden\">\n  @for (viewId of part.viewIds(); track viewId) {\n    @if (viewtab === dropTargetViewTab()) {\n      <div class=\"divider\"></div>\n    }\n    <wb-view-tab #viewtab\n                 [viewId]=\"viewId\"\n                 [class.drag-source]=\"viewtab === dragSourceViewTab()\"\n                 [class.drop-target]=\"viewtab === dropTargetViewTab()\"/>\n    <div class=\"divider\"></div>\n  }\n\n  <!-- Placeholder for expanding the viewport when dragging a tab to the right, moving part actions along with the dragover event. -->\n  <div class=\"drag-placeholder\"></div>\n</sci-viewport>\n\n<!-- Actions -->\n@if (startActions().length) {\n  <wb-part-action-bar [actions]=\"startActions()\" class=\"start\"/>\n}\n<div class=\"filler\" #filler></div>\n@if (endActions().length) {\n  <wb-part-action-bar [actions]=\"endActions()\" class=\"end\"/>\n}\n\n<!-- View List Dropdown -->\n<wb-view-list-button/>\n\n<!-- Element for injecting the corner radius into the component. -->\n<div class=\"tab-corner-radius\" #tab_corner_radius></div>\n<!-- Element for injecting the tabbar padding into the component. -->\n<div class=\"padding-inline\" #padding_inline></div>\n"]}
|