@scion/workbench 18.0.0-beta.3 → 18.0.0-beta.5
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/esm2022/lib/common/class-list.mjs +67 -29
- package/esm2022/lib/common/operators.mjs +1 -12
- package/esm2022/lib/content-projection/content-as-overlay.component.mjs +3 -3
- package/esm2022/lib/dialog/dialog-header/dialog-header.component.mjs +3 -8
- package/esm2022/lib/dialog/workbench-dialog.component.mjs +22 -20
- package/esm2022/lib/dialog/workbench-dialog.mjs +61 -1
- package/esm2022/lib/dialog//311/265workbench-dialog.mjs +71 -26
- package/esm2022/lib/dialog//311/265workbench-dialog.service.mjs +25 -23
- package/esm2022/lib/executor/latest-task-executor.mjs +55 -0
- package/esm2022/lib/executor/single-task-executor.mjs +7 -17
- package/esm2022/lib/layout/main-area-layout/main-area-layout.component.mjs +5 -6
- package/esm2022/lib/layout/migration/model/workbench-layout-migration-v5.model.mjs +1 -1
- package/esm2022/lib/layout/migration/workbench-layout-migration-v3.service.mjs +3 -3
- package/esm2022/lib/layout/stringifier.mjs +2 -2
- package/esm2022/lib/layout/workbench-layout.component.mjs +4 -5
- 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 +6 -7
- package/esm2022/lib/layout//311/265workbench-layout.factory.mjs +3 -2
- package/esm2022/lib/layout//311/265workbench-layout.mjs +34 -31
- package/esm2022/lib/message-box/message-box-footer/message-box-footer.component.mjs +3 -3
- package/esm2022/lib/message-box/message-box-header/message-box-header.component.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/common/microfrontend.util.mjs +5 -6
- package/esm2022/lib/microfrontend-platform/microfrontend-dialog/microfrontend-dialog.component.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.mjs +14 -6
- package/esm2022/lib/microfrontend-platform/microfrontend-host-message-box/microfrontend-host-message-box.component.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/microfrontend-host-popup/microfrontend-host-popup.component.mjs +5 -5
- package/esm2022/lib/microfrontend-platform/microfrontend-message-box/microfrontend-message-box.component.mjs +3 -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 +3 -3
- 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 -12
- package/esm2022/lib/microfrontend-platform/microfrontend-view/microfrontend-view.component.mjs +18 -16
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-intent-handler.interceptor.mjs +13 -14
- package/esm2022/lib/microfrontend-platform/routing/microfrontend-view-routes.mjs +3 -3
- package/esm2022/lib/microfrontend-platform/workbench-microfrontend-support.mjs +4 -4
- package/esm2022/lib/notification/notification-list.component.mjs +4 -8
- package/esm2022/lib/notification/notification.component.mjs +4 -5
- package/esm2022/lib/page-not-found/page-not-found.component.mjs +3 -3
- package/esm2022/lib/part/part-action-bar/part-action-bar.component.mjs +7 -13
- package/esm2022/lib/part/part-action-bar/part-action.directive.mjs +4 -4
- package/esm2022/lib/part/part-bar/part-bar.component.mjs +56 -75
- package/esm2022/lib/part/part-portal.pipe.mjs +8 -8
- package/esm2022/lib/part/part.component.mjs +32 -40
- 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 +27 -28
- package/esm2022/lib/part/view-list/view-list.component.mjs +41 -48
- package/esm2022/lib/part/view-list-button/view-list-button.component.mjs +20 -24
- package/esm2022/lib/part/view-list-item/view-list-item.component.mjs +15 -17
- package/esm2022/lib/part/view-tab/view-tab.component.mjs +77 -88
- 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 +14 -15
- 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-perspective.model.mjs +1 -1
- package/esm2022/lib/perspective/workbench-perspective.registry.mjs +11 -43
- package/esm2022/lib/perspective/workbench-perspective.service.mjs +27 -28
- package/esm2022/lib/perspective//311/265workbench-perspective.model.mjs +53 -47
- package/esm2022/lib/popup/popup.component.mjs +3 -3
- package/esm2022/lib/popup/popup.config.mjs +23 -13
- package/esm2022/lib/popup/popup.service.mjs +48 -45
- package/esm2022/lib/portal/wb-component-portal.mjs +2 -2
- package/esm2022/lib/registry/workbench-object-registry.mjs +38 -17
- package/esm2022/lib/routing/public_api.mjs +2 -1
- package/esm2022/lib/routing/routing.model.mjs +1 -1
- package/esm2022/lib/routing/routing.util.mjs +210 -0
- package/esm2022/lib/routing/workbench-auxiliary-route-installer.service.mjs +19 -3
- package/esm2022/lib/routing/workbench-dialog-differ.mjs +3 -3
- 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 +132 -144
- package/esm2022/lib/routing/workbench-view-outlet-differ.mjs +3 -3
- package/esm2022/lib/routing//311/265workbench-router.service.mjs +52 -40
- package/esm2022/lib/startup/splash/splash.component.mjs +3 -3
- package/esm2022/lib/startup/workbench-launcher.service.mjs +7 -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 +7 -7
- package/esm2022/lib/view/view-portal.pipe.mjs +7 -7
- package/esm2022/lib/view/view.component.mjs +22 -34
- 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 +136 -150
- package/esm2022/lib/view-dnd/view-drag.service.mjs +1 -1
- package/esm2022/lib/view-dnd/view-tab-drag-image-renderer.service.mjs +18 -20
- package/esm2022/lib/workbench.component.mjs +3 -3
- package/esm2022/lib/workbench.model.mjs +1 -1
- package/esm2022/lib/workbench.service.mjs +1 -1
- package/esm2022/lib//311/265workbench.service.mjs +31 -42
- package/fesm2022/scion-workbench.mjs +2028 -1953
- package/fesm2022/scion-workbench.mjs.map +1 -1
- package/lib/common/class-list.d.ts +35 -22
- package/lib/common/operators.d.ts +1 -6
- 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 +36 -17
- 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 +3 -6
- package/lib/layout/migration/model/workbench-layout-migration-v5.model.d.ts +4 -0
- package/lib/layout/workbench-layout.d.ts +9 -6
- package/lib/layout/workbench-layout.model.d.ts +2 -0
- package/lib/layout/workbench-layout.service.d.ts +4 -4
- package/lib/layout//311/265workbench-layout.d.ts +12 -13
- package/lib/layout//311/265workbench-layout.factory.d.ts +3 -2
- package/lib/microfrontend-platform/microfrontend-host-dialog/microfrontend-host-dialog.component.d.ts +1 -5
- package/lib/microfrontend-platform/microfrontend-popup/microfrontend-popup-intent-handler.interceptor.d.ts +4 -8
- package/lib/microfrontend-platform/microfrontend-view/microfrontend-view-command-handler.service.d.ts +5 -8
- package/lib/microfrontend-platform/routing/microfrontend-view-intent-handler.interceptor.d.ts +3 -7
- package/lib/notification/notification-list.component.d.ts +0 -2
- package/lib/part/part-action-bar/part-action-bar.component.d.ts +4 -1
- package/lib/part/part-action-bar/part-action.directive.d.ts +2 -2
- package/lib/part/part-bar/part-bar.component.d.ts +8 -12
- 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/view-menu.directive.d.ts +1 -1
- package/lib/part/view-context-menu/view-menu.service.d.ts +8 -12
- 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 +17 -28
- 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 +24 -17
- package/lib/perspective/workbench-perspective.model.d.ts +2 -8
- package/lib/perspective/workbench-perspective.registry.d.ts +3 -26
- package/lib/perspective/workbench-perspective.service.d.ts +8 -14
- package/lib/perspective//311/265workbench-perspective.model.d.ts +26 -21
- package/lib/popup/popup.config.d.ts +2 -0
- package/lib/popup/popup.service.d.ts +12 -14
- package/lib/registry/workbench-object-registry.d.ts +10 -6
- package/lib/routing/public_api.d.ts +2 -1
- package/lib/routing/routing.model.d.ts +25 -13
- package/lib/routing/{router.util.d.ts → routing.util.d.ts} +24 -1
- package/lib/routing/workbench-auxiliary-route-installer.service.d.ts +7 -0
- 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 -57
- package/lib/routing//311/265workbench-router.service.d.ts +10 -16
- 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-portal.pipe.d.ts +1 -3
- package/lib/view/view.component.d.ts +2 -4
- package/lib/view/workbench-view.model.d.ts +49 -35
- package/lib/view/workbench-view.registry.d.ts +3 -27
- package/lib/view//311/265workbench-view.model.d.ts +51 -61
- package/lib/view-dnd/view-drag.service.d.ts +5 -2
- package/lib/workbench.model.d.ts +6 -2
- package/lib/workbench.service.d.ts +58 -56
- package/lib//311/265workbench.service.d.ts +18 -31
- package/package.json +1 -1
- 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/part/part-action-bar/part-action-filter.pipe.mjs +0 -26
- package/esm2022/lib/routing/router.util.mjs +0 -126
- 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/part/part-action-bar/part-action-filter.pipe.d.ts +0 -11
|
@@ -1,30 +1,37 @@
|
|
|
1
|
+
/*
|
|
2
|
+
* Copyright (c) 2018-2024 Swiss Federal Railways
|
|
3
|
+
*
|
|
4
|
+
* This program and the accompanying materials are made
|
|
5
|
+
* available under the terms of the Eclipse Public License 2.0
|
|
6
|
+
* which is available at https://www.eclipse.org/legal/epl-2.0/
|
|
7
|
+
*
|
|
8
|
+
* SPDX-License-Identifier: EPL-2.0
|
|
9
|
+
*/
|
|
10
|
+
import { Router } from '@angular/router';
|
|
1
11
|
import { Defined, Observables } from '@scion/toolkit/util';
|
|
2
|
-
import { inject, Injectable, NgZone, runInInjectionContext } from '@angular/core';
|
|
12
|
+
import { inject, Injectable, Injector, NgZone, runInInjectionContext } from '@angular/core';
|
|
13
|
+
import { WorkbenchLayoutService } from '../layout/workbench-layout.service';
|
|
3
14
|
import { MAIN_AREA_LAYOUT_QUERY_PARAM } from '../workbench.constants';
|
|
4
15
|
import { SINGLE_NAVIGATION_EXECUTOR } from '../executor/single-task-executor';
|
|
5
16
|
import { firstValueFrom } from 'rxjs';
|
|
6
17
|
import { WorkbenchNavigationalStates } from './workbench-navigational-states';
|
|
7
|
-
import {
|
|
18
|
+
import { Routing } from './routing.util';
|
|
8
19
|
import { UrlSegmentMatcher } from './url-segment-matcher';
|
|
9
20
|
import { Objects } from '../common/objects.util';
|
|
21
|
+
import { WORKBENCH_VIEW_REGISTRY } from '../view/workbench-view.registry';
|
|
22
|
+
import { Logger } from '../logging';
|
|
10
23
|
import * as i0 from "@angular/core";
|
|
11
|
-
import * as i1 from "@angular/router";
|
|
12
|
-
import * as i2 from "../layout/workbench-layout.service";
|
|
13
|
-
import * as i3 from "../view/workbench-view.registry";
|
|
14
|
-
import * as i4 from "../logging";
|
|
15
24
|
/** @inheritDoc */
|
|
16
25
|
export class ɵWorkbenchRouter {
|
|
17
|
-
constructor(
|
|
18
|
-
this._router =
|
|
19
|
-
this._workbenchLayoutService =
|
|
20
|
-
this._workbenchViewRegistry =
|
|
21
|
-
this._injector =
|
|
22
|
-
this._logger =
|
|
23
|
-
this._zone =
|
|
26
|
+
constructor() {
|
|
27
|
+
this._router = inject(Router);
|
|
28
|
+
this._workbenchLayoutService = inject(WorkbenchLayoutService);
|
|
29
|
+
this._workbenchViewRegistry = inject(WORKBENCH_VIEW_REGISTRY);
|
|
30
|
+
this._injector = inject(Injector);
|
|
31
|
+
this._logger = inject(Logger);
|
|
32
|
+
this._zone = inject(NgZone);
|
|
24
33
|
this._singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);
|
|
25
|
-
/**
|
|
26
|
-
* Holds the current navigational context during a workbench navigation, or `null` if no navigation is in progress.
|
|
27
|
-
*/
|
|
34
|
+
/** Holds the current navigational context during a workbench navigation, or `null` if no navigation is in progress. */
|
|
28
35
|
this._currentNavigationContext = null;
|
|
29
36
|
// Instruct the Angular router to process navigations that do not change the current URL, i.e., when only updating navigation state.
|
|
30
37
|
// For example, the workbench grid is passed to the navigation as state, not as a query parameter. Without this flag set, changes to
|
|
@@ -44,11 +51,11 @@ export class ɵWorkbenchRouter {
|
|
|
44
51
|
return this._singleNavigationExecutor.submit(async () => {
|
|
45
52
|
// Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.
|
|
46
53
|
// Otherwise, this navigation would override the initial layout as given in the URL.
|
|
47
|
-
if (!this._workbenchLayoutService.layout) {
|
|
54
|
+
if (!this._workbenchLayoutService.layout()) {
|
|
48
55
|
await this.waitForInitialLayout();
|
|
49
56
|
}
|
|
50
57
|
// Let the navigator compute the new workbench layout.
|
|
51
|
-
const currentLayout = this._workbenchLayoutService.layout;
|
|
58
|
+
const currentLayout = this._workbenchLayoutService.layout();
|
|
52
59
|
let newLayout = await runInInjectionContext(this._injector, () => navigateFn(currentLayout));
|
|
53
60
|
if (!newLayout) {
|
|
54
61
|
return true;
|
|
@@ -56,12 +63,10 @@ export class ɵWorkbenchRouter {
|
|
|
56
63
|
// Remove views marked for removal, invoking `CanClose` guard if implemented.
|
|
57
64
|
newLayout = await newLayout.removeViewsMarkedForRemoval(viewUid => this.canCloseView(viewUid));
|
|
58
65
|
// Create extras with workbench navigation instructions.
|
|
59
|
-
extras = createNavigationExtras(newLayout, extras);
|
|
60
|
-
// Create the new URL tree.
|
|
61
|
-
const commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());
|
|
62
|
-
const urlTree = this._router.createUrlTree(commands, extras);
|
|
66
|
+
extras = createNavigationExtras({ newLayout, currentLayout }, extras);
|
|
63
67
|
// Perform the navigation.
|
|
64
|
-
|
|
68
|
+
const commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());
|
|
69
|
+
if (!(await this._router.navigate(commands, extras))) {
|
|
65
70
|
return false;
|
|
66
71
|
}
|
|
67
72
|
// Block subsequent navigation(s) until Angular has flushed the changed layout to the DOM.
|
|
@@ -99,11 +104,11 @@ export class ɵWorkbenchRouter {
|
|
|
99
104
|
return this._singleNavigationExecutor.submit(async () => {
|
|
100
105
|
// Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.
|
|
101
106
|
// Otherwise, would override the initial layout as given in the URL.
|
|
102
|
-
if (!this._workbenchLayoutService.layout) {
|
|
107
|
+
if (!this._workbenchLayoutService.layout()) {
|
|
103
108
|
await this.waitForInitialLayout();
|
|
104
109
|
}
|
|
105
110
|
// Let the navigator compute the new workbench layout.
|
|
106
|
-
const currentLayout = this._workbenchLayoutService.layout;
|
|
111
|
+
const currentLayout = this._workbenchLayoutService.layout();
|
|
107
112
|
let newLayout = await runInInjectionContext(this._injector, () => onNavigate(currentLayout));
|
|
108
113
|
if (!newLayout) {
|
|
109
114
|
return null;
|
|
@@ -111,7 +116,7 @@ export class ɵWorkbenchRouter {
|
|
|
111
116
|
// Remove views marked for removal.
|
|
112
117
|
newLayout = await newLayout.removeViewsMarkedForRemoval();
|
|
113
118
|
// Create extras with workbench navigation instructions.
|
|
114
|
-
extras = createNavigationExtras(newLayout, extras);
|
|
119
|
+
extras = createNavigationExtras({ newLayout, currentLayout }, extras);
|
|
115
120
|
// Create the new URL tree.
|
|
116
121
|
const commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());
|
|
117
122
|
return this._router.createUrlTree(commands, extras);
|
|
@@ -121,11 +126,11 @@ export class ɵWorkbenchRouter {
|
|
|
121
126
|
* Decides if given view can be closed, invoking `CanClose` guard if implemented.
|
|
122
127
|
*/
|
|
123
128
|
async canCloseView(viewUid) {
|
|
124
|
-
const view = this._workbenchViewRegistry.
|
|
129
|
+
const view = this._workbenchViewRegistry.objects().find(view => view.uid === viewUid);
|
|
125
130
|
if (!view) {
|
|
126
131
|
return true;
|
|
127
132
|
}
|
|
128
|
-
if (!view.closable) {
|
|
133
|
+
if (!view.closable()) {
|
|
129
134
|
return false;
|
|
130
135
|
}
|
|
131
136
|
// Test if the view implements `CanClose` guard.
|
|
@@ -164,13 +169,13 @@ export class ɵWorkbenchRouter {
|
|
|
164
169
|
async waitForInitialLayout() {
|
|
165
170
|
await firstValueFrom(this._workbenchLayoutService.layout$);
|
|
166
171
|
}
|
|
167
|
-
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: ɵWorkbenchRouter, deps: [
|
|
172
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: ɵWorkbenchRouter, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
168
173
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: ɵWorkbenchRouter, providedIn: 'root' }); }
|
|
169
174
|
}
|
|
170
175
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: ɵWorkbenchRouter, decorators: [{
|
|
171
176
|
type: Injectable,
|
|
172
177
|
args: [{ providedIn: 'root' }]
|
|
173
|
-
}], ctorParameters: () => [
|
|
178
|
+
}], ctorParameters: () => [] });
|
|
174
179
|
/**
|
|
175
180
|
* Creates a navigation to navigate view(s) based on the provided array of commands and extras.
|
|
176
181
|
*/
|
|
@@ -183,7 +188,7 @@ function createNavigationFromCommands(commands, extras) {
|
|
|
183
188
|
}
|
|
184
189
|
return layout.removeView(extras.target);
|
|
185
190
|
}
|
|
186
|
-
const urlSegments =
|
|
191
|
+
const urlSegments = Routing.commandsToSegments(commands, { relativeTo: extras.relativeTo });
|
|
187
192
|
return layout
|
|
188
193
|
.views({
|
|
189
194
|
partId: extras.partId,
|
|
@@ -198,7 +203,7 @@ function createNavigationFromCommands(commands, extras) {
|
|
|
198
203
|
return addView(layout.computeNextViewId(), layout);
|
|
199
204
|
}
|
|
200
205
|
case 'auto': {
|
|
201
|
-
const urlSegments =
|
|
206
|
+
const urlSegments = Routing.commandsToSegments(commands, { relativeTo: extras.relativeTo });
|
|
202
207
|
const views = layout.views({
|
|
203
208
|
partId: extras.partId,
|
|
204
209
|
segments: new UrlSegmentMatcher(urlSegments, { matchMatrixParams: false, matchWildcardPath: false }),
|
|
@@ -239,11 +244,13 @@ function createNavigationFromCommands(commands, extras) {
|
|
|
239
244
|
partId: partId,
|
|
240
245
|
position: extras.position ?? 'after-active-view',
|
|
241
246
|
activateView: extras.activate ?? true,
|
|
247
|
+
activatePart: extras.activate ?? true,
|
|
242
248
|
})
|
|
243
249
|
.navigateView(viewId, commands, {
|
|
244
250
|
relativeTo: extras.relativeTo,
|
|
245
251
|
hint: extras.hint,
|
|
246
252
|
cssClass: extras.cssClass,
|
|
253
|
+
data: extras.data,
|
|
247
254
|
state: extras.state,
|
|
248
255
|
});
|
|
249
256
|
}
|
|
@@ -252,12 +259,13 @@ function createNavigationFromCommands(commands, extras) {
|
|
|
252
259
|
*/
|
|
253
260
|
function updateView(viewId, layout) {
|
|
254
261
|
if (extras.activate ?? true) {
|
|
255
|
-
layout = layout.activateView(viewId);
|
|
262
|
+
layout = layout.activateView(viewId, { activatePart: true });
|
|
256
263
|
}
|
|
257
264
|
return layout.navigateView(viewId, commands, {
|
|
258
265
|
relativeTo: extras.relativeTo,
|
|
259
266
|
hint: extras.hint,
|
|
260
267
|
cssClass: extras.cssClass,
|
|
268
|
+
data: extras.data,
|
|
261
269
|
state: extras.state,
|
|
262
270
|
});
|
|
263
271
|
}
|
|
@@ -265,8 +273,9 @@ function createNavigationFromCommands(commands, extras) {
|
|
|
265
273
|
/**
|
|
266
274
|
* Creates navigation extras with workbench navigation instructions.
|
|
267
275
|
*/
|
|
268
|
-
function createNavigationExtras(
|
|
269
|
-
const {
|
|
276
|
+
function createNavigationExtras(layouts, extras) {
|
|
277
|
+
const { newLayout, currentLayout } = layouts;
|
|
278
|
+
const { workbenchGrid, mainAreaGrid } = newLayout.serialize({ excludeViewMarkedForRemoval: true });
|
|
270
279
|
return {
|
|
271
280
|
...extras,
|
|
272
281
|
// Instruct the Angular router to process the navigation even if the URL does not change, e.g., when changing the workbench grid which is not contained in the URL.
|
|
@@ -276,8 +285,11 @@ function createNavigationExtras(layout, extras) {
|
|
|
276
285
|
// Associate workbench-specific state with the navigation.
|
|
277
286
|
state: WorkbenchNavigationalStates.create({
|
|
278
287
|
workbenchGrid: workbenchGrid,
|
|
279
|
-
|
|
280
|
-
|
|
288
|
+
// Fall back to the current layout's perspective if not specified by the new layout,
|
|
289
|
+
// e.g., the navigator provides a new layout to replace the current layout.
|
|
290
|
+
perspectiveId: newLayout.perspectiveId ?? currentLayout?.perspectiveId,
|
|
291
|
+
maximized: newLayout.maximized,
|
|
292
|
+
navigationStates: newLayout.navigationStates(),
|
|
281
293
|
}),
|
|
282
294
|
// Add the main area as query parameter.
|
|
283
295
|
queryParams: { ...extras?.queryParams, [MAIN_AREA_LAYOUT_QUERY_PARAM]: mainAreaGrid },
|
|
@@ -297,7 +309,7 @@ function computeNavigationCommands(previousViewOutlets, nextViewOutlets) {
|
|
|
297
309
|
viewIds.forEach(viewId => {
|
|
298
310
|
// Test if the view was added to the layout.
|
|
299
311
|
if (!previousViewOutletMap.has(viewId)) {
|
|
300
|
-
commands.set(viewId,
|
|
312
|
+
commands.set(viewId, Routing.segmentsToCommands(nextViewOutletMap.get(viewId)));
|
|
301
313
|
}
|
|
302
314
|
// Test if the view was removed from the layout.
|
|
303
315
|
else if (!nextViewOutletMap.has(viewId)) {
|
|
@@ -305,10 +317,10 @@ function computeNavigationCommands(previousViewOutlets, nextViewOutlets) {
|
|
|
305
317
|
}
|
|
306
318
|
// Test if the view was updated.
|
|
307
319
|
else if (!new UrlSegmentMatcher(previousViewOutletMap.get(viewId), { matchMatrixParams: true, matchWildcardPath: false }).matches(nextViewOutletMap.get(viewId))) {
|
|
308
|
-
commands.set(viewId,
|
|
320
|
+
commands.set(viewId, Routing.segmentsToCommands(nextViewOutletMap.get(viewId)));
|
|
309
321
|
}
|
|
310
322
|
});
|
|
311
323
|
// Add view commands to the 'outlets' property to be interpreted by the Angular router.
|
|
312
324
|
return commands.size ? [{ outlets: Object.fromEntries(commands) }] : [];
|
|
313
325
|
}
|
|
314
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ɵworkbench-router.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/routing/ɵworkbench-router.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2024 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {NavigationExtras, Router, UrlSegment, UrlTree} from '@angular/router';\nimport {WorkbenchRouter} from './workbench-router.service';\nimport {Defined, Observables} from '@scion/toolkit/util';\nimport {inject, Injectable, Injector, NgZone, runInInjectionContext} from '@angular/core';\nimport {WorkbenchLayoutService} from '../layout/workbench-layout.service';\nimport {MAIN_AREA_LAYOUT_QUERY_PARAM} from '../workbench.constants';\nimport {SINGLE_NAVIGATION_EXECUTOR} from '../executor/single-task-executor';\nimport {firstValueFrom} from 'rxjs';\nimport {WorkbenchNavigationalStates} from './workbench-navigational-states';\nimport {ɵWorkbenchLayout} from '../layout/ɵworkbench-layout';\nimport {RouterUtils} from './router.util';\nimport {Commands, ViewOutlets, WorkbenchNavigationContext, WorkbenchNavigationExtras} from './routing.model';\nimport {ViewId} from '../view/workbench-view.model';\nimport {UrlSegmentMatcher} from './url-segment-matcher';\nimport {Objects} from '../common/objects.util';\nimport {WorkbenchViewRegistry} from '../view/workbench-view.registry';\nimport {Logger} from '../logging';\nimport {CanClose} from '../workbench.model';\n\n/** @inheritDoc */\n@Injectable({providedIn: 'root'})\nexport class ɵWorkbenchRouter implements WorkbenchRouter {\n\n  private _singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);\n\n  /**\n   * Holds the current navigational context during a workbench navigation, or `null` if no navigation is in progress.\n   */\n  private _currentNavigationContext: WorkbenchNavigationContext | null = null;\n\n  constructor(private _router: Router,\n              private _workbenchLayoutService: WorkbenchLayoutService,\n              private _workbenchViewRegistry: WorkbenchViewRegistry,\n              private _injector: Injector,\n              private _logger: Logger,\n              private _zone: NgZone) {\n    // Instruct the Angular router to process navigations that do not change the current URL, i.e., when only updating navigation state.\n    // For example, the workbench grid is passed to the navigation as state, not as a query parameter. Without this flag set, changes to\n    // the workbench grid would not be added to the browsing history stack.\n    // Although the `onSameUrlNavigation` flag can be set per navigation via the navigation extras, this is not sufficient because the\n    // Angular router ignores it when navigating back and forth in the browsing history.\n    // See Angular router.ts#setUpLocationChangeListener and router.ts#navigateToSyncWithBrowser\n    this._router.onSameUrlNavigation = 'reload';\n  }\n\n  /** @inheritDoc */\n  public navigate(commands: Commands, extras?: WorkbenchNavigationExtras): Promise<boolean>;\n  public navigate(navigateFn: ɵNavigateFn, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean>;\n  public navigate(commandsOrNavigateFn: Commands | ɵNavigateFn, extras?: WorkbenchNavigationExtras | Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean>;\n  public navigate(commandsOrNavigateFn: Commands | ɵNavigateFn, extras?: WorkbenchNavigationExtras | Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean> {\n    // Ensure to run in Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.navigate(commandsOrNavigateFn, extras));\n    }\n\n    const navigateFn = typeof commandsOrNavigateFn === 'function' ? commandsOrNavigateFn : createNavigationFromCommands(commandsOrNavigateFn, extras ?? {});\n\n    // Serialize navigation requests to prevent race conditions when modifying the currently active workbench layout.\n    return this._singleNavigationExecutor.submit(async () => {\n      // Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.\n      // Otherwise, this navigation would override the initial layout as given in the URL.\n      if (!this._workbenchLayoutService.layout) {\n        await this.waitForInitialLayout();\n      }\n\n      // Let the navigator compute the new workbench layout.\n      const currentLayout = this._workbenchLayoutService.layout!;\n      let newLayout: ɵWorkbenchLayout | null = await runInInjectionContext(this._injector, () => navigateFn(currentLayout)) as ɵWorkbenchLayout;\n      if (!newLayout) {\n        return true;\n      }\n\n      // Remove views marked for removal, invoking `CanClose` guard if implemented.\n      newLayout = await newLayout.removeViewsMarkedForRemoval(viewUid => this.canCloseView(viewUid));\n\n      // Create extras with workbench navigation instructions.\n      extras = createNavigationExtras(newLayout, extras);\n\n      // Create the new URL tree.\n      const commands: Commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());\n      const urlTree = this._router.createUrlTree(commands, extras);\n\n      // Perform the navigation.\n      if (!(await this._router.navigateByUrl(urlTree, extras))) {\n        return false;\n      }\n\n      // Block subsequent navigation(s) until Angular has flushed the changed layout to the DOM.\n      await firstValueFrom(this._zone.onStable);\n      return true;\n    });\n  }\n\n  /**\n   * Performs changes to the current workbench layout.\n   *\n   * Unlike {@link navigate}, does not perform the navigation but returns the URL tree instead.\n   *\n   * The router will invoke the callback and pass the current layout for modification. The callback can call `inject` to get any required dependencies.\n   *\n   * The following example adds a part to the left of the main area, inserts a view and navigates it.\n   *\n   * ```ts\n   * inject(WorkbenchRouter).createUrlTree(layout => layout\n   *   .addPart('left', {relativeTo: MAIN_AREA, align: 'left'})\n   *   .addView('navigator', {partId: 'left'})\n   *   .navigateView('navigator', ['path/to/view'])\n   *   .activateView('navigator'),\n   * );\n   * ```\n   *\n   * @param onNavigate - Specifies the callback to modify the layout.\n   * @param extras - Controls how to perform the navigation.\n   * @see NavigateFn\n   */\n  public async createUrlTree(onNavigate: (layout: ɵWorkbenchLayout) => Promise<ɵWorkbenchLayout | null> | ɵWorkbenchLayout | null, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<UrlTree | null> {\n    // Ensure to run in Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.createUrlTree(onNavigate, extras));\n    }\n\n    return this._singleNavigationExecutor.submit(async () => {\n      // Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.\n      // Otherwise, would override the initial layout as given in the URL.\n      if (!this._workbenchLayoutService.layout) {\n        await this.waitForInitialLayout();\n      }\n\n      // Let the navigator compute the new workbench layout.\n      const currentLayout = this._workbenchLayoutService.layout!;\n      let newLayout: ɵWorkbenchLayout | null = await runInInjectionContext(this._injector, () => onNavigate(currentLayout));\n      if (!newLayout) {\n        return null;\n      }\n\n      // Remove views marked for removal.\n      newLayout = await newLayout.removeViewsMarkedForRemoval();\n\n      // Create extras with workbench navigation instructions.\n      extras = createNavigationExtras(newLayout, extras);\n\n      // Create the new URL tree.\n      const commands: Commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());\n      return this._router.createUrlTree(commands, extras);\n    });\n  }\n\n  /**\n   * Decides if given view can be closed, invoking `CanClose` guard if implemented.\n   */\n  private async canCloseView(viewUid: string): Promise<boolean> {\n    const view = this._workbenchViewRegistry.views.find(view => view.uid === viewUid);\n    if (!view) {\n      return true;\n    }\n    if (!view.closable) {\n      return false;\n    }\n\n    // Test if the view implements `CanClose` guard.\n    const component = view.getComponent() as CanClose | null;\n    if (typeof component?.canClose !== 'function') {\n      return true;\n    }\n\n    // Invoke `CanClose` guard to decide if to close the view.\n    try {\n      const canClose = runInInjectionContext(view.getComponentInjector()!, () => component.canClose());\n      return await firstValueFrom(Observables.coerce(canClose), {defaultValue: true});\n    }\n    catch (error) {\n      this._logger.error(`Unhandled error while invoking 'CanClose' guard of view '${view.id}'.`, error);\n      return true;\n    }\n  }\n\n  /**\n   * Returns the context of the current workbench navigation, when being invoked during navigation, or throws an error otherwise.\n   */\n  public getCurrentNavigationContext(): WorkbenchNavigationContext {\n    if (!this._currentNavigationContext) {\n      throw Error('[NavigateError] Navigation context not available because no navigation is in progress.');\n    }\n    return this._currentNavigationContext;\n  }\n\n  /**\n   * Sets navigational contextual data.\n   */\n  public setCurrentNavigationContext(context: WorkbenchNavigationContext | null): void {\n    this._currentNavigationContext = context;\n  }\n\n  /**\n   * Blocks until the initial layout is available, i.e. after completion of Angular's initial navigation.\n   */\n  private async waitForInitialLayout(): Promise<void> {\n    await firstValueFrom(this._workbenchLayoutService.layout$);\n  }\n}\n\n/**\n * Creates a navigation to navigate view(s) based on the provided array of commands and extras.\n */\nfunction createNavigationFromCommands(commands: Commands, extras: WorkbenchNavigationExtras): ɵNavigateFn {\n  return (layout: ɵWorkbenchLayout): ɵWorkbenchLayout | null => {\n    if (extras.close) {\n      if (extras.target) {\n        if (commands.length || extras.hint || extras.partId || extras.relativeTo) {\n          throw Error('[NavigateError] Commands, hint, part or relativeTo must not be set when closing a view by id.');\n        }\n        return layout.removeView(extras.target);\n      }\n      const urlSegments = RouterUtils.commandsToSegments(commands, {relativeTo: extras.relativeTo});\n      return layout\n        .views({\n          partId: extras.partId,\n          segments: new UrlSegmentMatcher(urlSegments, {matchMatrixParams: false, matchWildcardPath: true}),\n          navigationHint: extras.hint ?? null,\n        })\n        .reduce((layout, view) => layout.removeView(view.id), layout);\n    }\n    else {\n      switch (extras.target ?? 'auto') {\n        case 'blank': {\n          return addView(layout.computeNextViewId(), layout);\n        }\n        case 'auto': {\n          const urlSegments = RouterUtils.commandsToSegments(commands, {relativeTo: extras.relativeTo});\n          const views = layout.views({\n            partId: extras.partId,\n            segments: new UrlSegmentMatcher(urlSegments, {matchMatrixParams: false, matchWildcardPath: false}),\n            navigationHint: extras.hint ?? null,\n          });\n          if (views.length) {\n            return views.reduce((layout, view) => updateView(view.id, layout), layout);\n          }\n          else {\n            return addView(layout.computeNextViewId(), layout);\n          }\n        }\n        default: {\n          const viewId = extras.target!;\n          if (layout.hasView(viewId)) {\n            return updateView(viewId, layout);\n          }\n          else {\n            return addView(viewId, layout);\n          }\n        }\n      }\n    }\n  };\n\n  /**\n   * Adds the specified view to the layout.\n   */\n  function addView(viewId: string, layout: ɵWorkbenchLayout): ɵWorkbenchLayout {\n    // Default to the active part if not specified or not in the layout, with the active part of the main area taking precedence.\n    const partId = ((): string => {\n      if (extras.partId && layout.hasPart(extras.partId)) {\n        return extras.partId;\n      }\n      return layout.activePart({grid: 'mainArea'})?.id ?? layout.activePart({grid: 'workbench'}).id;\n    })();\n\n    return layout\n      .addView(viewId, {\n        partId: partId,\n        position: extras.position ?? 'after-active-view',\n        activateView: extras.activate ?? true,\n      })\n      .navigateView(viewId, commands, {\n        relativeTo: extras.relativeTo,\n        hint: extras.hint,\n        cssClass: extras.cssClass,\n        state: extras.state,\n      });\n  }\n\n  /**\n   * Updates the specified view.\n   */\n  function updateView(viewId: string, layout: ɵWorkbenchLayout): ɵWorkbenchLayout {\n    if (extras.activate ?? true) {\n      layout = layout.activateView(viewId);\n    }\n\n    return layout.navigateView(viewId, commands, {\n      relativeTo: extras.relativeTo,\n      hint: extras.hint,\n      cssClass: extras.cssClass,\n      state: extras.state,\n    });\n  }\n}\n\n/**\n * Creates navigation extras with workbench navigation instructions.\n */\nfunction createNavigationExtras(layout: ɵWorkbenchLayout, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): NavigationExtras {\n  const {workbenchGrid, mainAreaGrid} = layout.serialize({excludeViewMarkedForRemoval: true});\n\n  return {\n    ...extras,\n    // Instruct the Angular router to process the navigation even if the URL does not change, e.g., when changing the workbench grid which is not contained in the URL.\n    onSameUrlNavigation: 'reload',\n    // Unset `relativeTo` because commands are already normalized to their absolute form.\n    relativeTo: null,\n    // Associate workbench-specific state with the navigation.\n    state: WorkbenchNavigationalStates.create({\n      workbenchGrid: workbenchGrid,\n      maximized: layout.maximized,\n      viewStates: layout.viewStates(),\n    }),\n    // Add the main area as query parameter.\n    queryParams: {...extras?.queryParams, [MAIN_AREA_LAYOUT_QUERY_PARAM]: mainAreaGrid},\n    // Merge with existing query params unless specified an explicit strategy, e.g., for migrating an outdated layout URL.\n    // Note that `null` is a valid strategy for clearing existing query params, so do not use the nullish coalescing operator (??).\n    queryParamsHandling: Defined.orElse(extras?.queryParamsHandling, 'merge'),\n  };\n}\n\n/**\n * Computes commands that can be passed to the Angular router to navigate view outlets and to remove view outlets of removed views.\n */\nfunction computeNavigationCommands(previousViewOutlets: ViewOutlets, nextViewOutlets: ViewOutlets): [{outlets: {[outlet: ViewId]: Commands | null}}] | [] {\n  const previousViewOutletMap = new Map<ViewId, UrlSegment[]>(Objects.entries(previousViewOutlets));\n  const nextViewOutletMap = new Map<ViewId, UrlSegment[]>(Objects.entries(nextViewOutlets));\n\n  const commands = new Map<ViewId, Commands | null>();\n  const viewIds = new Set<ViewId>([...previousViewOutletMap.keys(), ...nextViewOutletMap.keys()]);\n\n  viewIds.forEach(viewId => {\n    // Test if the view was added to the layout.\n    if (!previousViewOutletMap.has(viewId)) {\n      commands.set(viewId, RouterUtils.segmentsToCommands(nextViewOutletMap.get(viewId)!));\n    }\n    // Test if the view was removed from the layout.\n    else if (!nextViewOutletMap.has(viewId)) {\n      commands.set(viewId, null);\n    }\n    // Test if the view was updated.\n    else if (!new UrlSegmentMatcher(previousViewOutletMap.get(viewId)!, {matchMatrixParams: true, matchWildcardPath: false}).matches(nextViewOutletMap.get(viewId)!)) {\n      commands.set(viewId, RouterUtils.segmentsToCommands(nextViewOutletMap.get(viewId)!));\n    }\n  });\n\n  // Add view commands to the 'outlets' property to be interpreted by the Angular router.\n  return commands.size ? [{outlets: Object.fromEntries(commands)}] : [];\n}\n\n/**\n * @inheritDoc\n * @see NavigateFn\n */\nexport type ɵNavigateFn = (layout: ɵWorkbenchLayout) => Promise<ɵWorkbenchLayout | null> | ɵWorkbenchLayout | null;\n"]}
|
|
326
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ɵworkbench-router.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/routing/ɵworkbench-router.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2024 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {NavigationExtras, Router, UrlSegment, UrlTree} from '@angular/router';\nimport {WorkbenchRouter} from './workbench-router.service';\nimport {Defined, Observables} from '@scion/toolkit/util';\nimport {inject, Injectable, Injector, NgZone, runInInjectionContext} from '@angular/core';\nimport {WorkbenchLayoutService} from '../layout/workbench-layout.service';\nimport {MAIN_AREA_LAYOUT_QUERY_PARAM} from '../workbench.constants';\nimport {SINGLE_NAVIGATION_EXECUTOR} from '../executor/single-task-executor';\nimport {firstValueFrom} from 'rxjs';\nimport {WorkbenchNavigationalStates} from './workbench-navigational-states';\nimport {ɵWorkbenchLayout} from '../layout/ɵworkbench-layout';\nimport {Routing} from './routing.util';\nimport {Commands, ViewOutlets, WorkbenchNavigationContext, WorkbenchNavigationExtras} from './routing.model';\nimport {ViewId} from '../view/workbench-view.model';\nimport {UrlSegmentMatcher} from './url-segment-matcher';\nimport {Objects} from '../common/objects.util';\nimport {WORKBENCH_VIEW_REGISTRY} from '../view/workbench-view.registry';\nimport {Logger} from '../logging';\nimport {CanClose} from '../workbench.model';\n\n/** @inheritDoc */\n@Injectable({providedIn: 'root'})\nexport class ɵWorkbenchRouter implements WorkbenchRouter {\n\n  private readonly _router = inject(Router);\n  private readonly _workbenchLayoutService = inject(WorkbenchLayoutService);\n  private readonly _workbenchViewRegistry = inject(WORKBENCH_VIEW_REGISTRY);\n  private readonly _injector = inject(Injector);\n  private readonly _logger = inject(Logger);\n  private readonly _zone = inject(NgZone);\n  private readonly _singleNavigationExecutor = inject(SINGLE_NAVIGATION_EXECUTOR);\n\n  /** Holds the current navigational context during a workbench navigation, or `null` if no navigation is in progress. */\n  private _currentNavigationContext: WorkbenchNavigationContext | null = null;\n\n  constructor() {\n    // Instruct the Angular router to process navigations that do not change the current URL, i.e., when only updating navigation state.\n    // For example, the workbench grid is passed to the navigation as state, not as a query parameter. Without this flag set, changes to\n    // the workbench grid would not be added to the browsing history stack.\n    // Although the `onSameUrlNavigation` flag can be set per navigation via the navigation extras, this is not sufficient because the\n    // Angular router ignores it when navigating back and forth in the browsing history.\n    // See Angular router.ts#setUpLocationChangeListener and router.ts#navigateToSyncWithBrowser\n    this._router.onSameUrlNavigation = 'reload';\n  }\n\n  /** @inheritDoc */\n  public navigate(commands: Commands, extras?: WorkbenchNavigationExtras): Promise<boolean>;\n  public navigate(navigateFn: ɵNavigateFn, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean>;\n  public navigate(commandsOrNavigateFn: Commands | ɵNavigateFn, extras?: WorkbenchNavigationExtras | Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean>;\n  public navigate(commandsOrNavigateFn: Commands | ɵNavigateFn, extras?: WorkbenchNavigationExtras | Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<boolean> {\n    // Ensure to run in Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.navigate(commandsOrNavigateFn, extras));\n    }\n\n    const navigateFn = typeof commandsOrNavigateFn === 'function' ? commandsOrNavigateFn : createNavigationFromCommands(commandsOrNavigateFn, extras ?? {});\n\n    // Serialize navigation requests to prevent race conditions when modifying the currently active workbench layout.\n    return this._singleNavigationExecutor.submit(async () => {\n      // Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.\n      // Otherwise, this navigation would override the initial layout as given in the URL.\n      if (!this._workbenchLayoutService.layout()) {\n        await this.waitForInitialLayout();\n      }\n\n      // Let the navigator compute the new workbench layout.\n      const currentLayout = this._workbenchLayoutService.layout()!;\n      let newLayout: ɵWorkbenchLayout | null = await runInInjectionContext(this._injector, () => navigateFn(currentLayout)) as ɵWorkbenchLayout;\n      if (!newLayout) {\n        return true;\n      }\n\n      // Remove views marked for removal, invoking `CanClose` guard if implemented.\n      newLayout = await newLayout.removeViewsMarkedForRemoval(viewUid => this.canCloseView(viewUid));\n\n      // Create extras with workbench navigation instructions.\n      extras = createNavigationExtras({newLayout, currentLayout}, extras);\n\n      // Perform the navigation.\n      const commands: Commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());\n      if (!(await this._router.navigate(commands, extras))) {\n        return false;\n      }\n\n      // Block subsequent navigation(s) until Angular has flushed the changed layout to the DOM.\n      await firstValueFrom(this._zone.onStable);\n      return true;\n    });\n  }\n\n  /**\n   * Performs changes to the current workbench layout.\n   *\n   * Unlike {@link navigate}, does not perform the navigation but returns the URL tree instead.\n   *\n   * The router will invoke the callback and pass the current layout for modification. The callback can call `inject` to get any required dependencies.\n   *\n   * The following example adds a part to the left of the main area, inserts a view and navigates it.\n   *\n   * ```ts\n   * inject(WorkbenchRouter).createUrlTree(layout => layout\n   *   .addPart('left', {relativeTo: MAIN_AREA, align: 'left'})\n   *   .addView('navigator', {partId: 'left'})\n   *   .navigateView('navigator', ['path/to/view'])\n   *   .activateView('navigator'),\n   * );\n   * ```\n   *\n   * @param onNavigate - Specifies the callback to modify the layout.\n   * @param extras - Controls how to perform the navigation.\n   * @see NavigateFn\n   */\n  public async createUrlTree(onNavigate: (layout: ɵWorkbenchLayout) => Promise<ɵWorkbenchLayout | null> | ɵWorkbenchLayout | null, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): Promise<UrlTree | null> {\n    // Ensure to run in Angular zone.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.createUrlTree(onNavigate, extras));\n    }\n\n    return this._singleNavigationExecutor.submit(async () => {\n      // Wait until the initial layout is available, i.e., after completion of Angular's initial navigation.\n      // Otherwise, would override the initial layout as given in the URL.\n      if (!this._workbenchLayoutService.layout()) {\n        await this.waitForInitialLayout();\n      }\n\n      // Let the navigator compute the new workbench layout.\n      const currentLayout = this._workbenchLayoutService.layout()!;\n      let newLayout: ɵWorkbenchLayout | null = await runInInjectionContext(this._injector, () => onNavigate(currentLayout));\n      if (!newLayout) {\n        return null;\n      }\n\n      // Remove views marked for removal.\n      newLayout = await newLayout.removeViewsMarkedForRemoval();\n\n      // Create extras with workbench navigation instructions.\n      extras = createNavigationExtras({newLayout, currentLayout}, extras);\n\n      // Create the new URL tree.\n      const commands: Commands = computeNavigationCommands(currentLayout.viewOutlets(), newLayout.viewOutlets());\n      return this._router.createUrlTree(commands, extras);\n    });\n  }\n\n  /**\n   * Decides if given view can be closed, invoking `CanClose` guard if implemented.\n   */\n  private async canCloseView(viewUid: string): Promise<boolean> {\n    const view = this._workbenchViewRegistry.objects().find(view => view.uid === viewUid);\n    if (!view) {\n      return true;\n    }\n    if (!view.closable()) {\n      return false;\n    }\n\n    // Test if the view implements `CanClose` guard.\n    const component = view.getComponent() as CanClose | null;\n    if (typeof component?.canClose !== 'function') {\n      return true;\n    }\n\n    // Invoke `CanClose` guard to decide if to close the view.\n    try {\n      const canClose = runInInjectionContext(view.getComponentInjector()!, () => component.canClose());\n      return await firstValueFrom(Observables.coerce(canClose), {defaultValue: true});\n    }\n    catch (error) {\n      this._logger.error(`Unhandled error while invoking 'CanClose' guard of view '${view.id}'.`, error);\n      return true;\n    }\n  }\n\n  /**\n   * Returns the context of the current workbench navigation, when being invoked during navigation, or throws an error otherwise.\n   */\n  public getCurrentNavigationContext(): WorkbenchNavigationContext {\n    if (!this._currentNavigationContext) {\n      throw Error('[NavigateError] Navigation context not available because no navigation is in progress.');\n    }\n    return this._currentNavigationContext;\n  }\n\n  /**\n   * Sets navigational contextual data.\n   */\n  public setCurrentNavigationContext(context: WorkbenchNavigationContext | null): void {\n    this._currentNavigationContext = context;\n  }\n\n  /**\n   * Blocks until the initial layout is available, i.e. after completion of Angular's initial navigation.\n   */\n  private async waitForInitialLayout(): Promise<void> {\n    await firstValueFrom(this._workbenchLayoutService.layout$);\n  }\n}\n\n/**\n * Creates a navigation to navigate view(s) based on the provided array of commands and extras.\n */\nfunction createNavigationFromCommands(commands: Commands, extras: WorkbenchNavigationExtras): ɵNavigateFn {\n  return (layout: ɵWorkbenchLayout): ɵWorkbenchLayout | null => {\n    if (extras.close) {\n      if (extras.target) {\n        if (commands.length || extras.hint || extras.partId || extras.relativeTo) {\n          throw Error('[NavigateError] Commands, hint, part or relativeTo must not be set when closing a view by id.');\n        }\n        return layout.removeView(extras.target);\n      }\n      const urlSegments = Routing.commandsToSegments(commands, {relativeTo: extras.relativeTo});\n      return layout\n        .views({\n          partId: extras.partId,\n          segments: new UrlSegmentMatcher(urlSegments, {matchMatrixParams: false, matchWildcardPath: true}),\n          navigationHint: extras.hint ?? null,\n        })\n        .reduce((layout, view) => layout.removeView(view.id), layout);\n    }\n    else {\n      switch (extras.target ?? 'auto') {\n        case 'blank': {\n          return addView(layout.computeNextViewId(), layout);\n        }\n        case 'auto': {\n          const urlSegments = Routing.commandsToSegments(commands, {relativeTo: extras.relativeTo});\n          const views = layout.views({\n            partId: extras.partId,\n            segments: new UrlSegmentMatcher(urlSegments, {matchMatrixParams: false, matchWildcardPath: false}),\n            navigationHint: extras.hint ?? null,\n          });\n          if (views.length) {\n            return views.reduce((layout, view) => updateView(view.id, layout), layout);\n          }\n          else {\n            return addView(layout.computeNextViewId(), layout);\n          }\n        }\n        default: {\n          const viewId = extras.target!;\n          if (layout.hasView(viewId)) {\n            return updateView(viewId, layout);\n          }\n          else {\n            return addView(viewId, layout);\n          }\n        }\n      }\n    }\n  };\n\n  /**\n   * Adds the specified view to the layout.\n   */\n  function addView(viewId: string, layout: ɵWorkbenchLayout): ɵWorkbenchLayout {\n    // Default to the active part if not specified or not in the layout, with the active part of the main area taking precedence.\n    const partId = ((): string => {\n      if (extras.partId && layout.hasPart(extras.partId)) {\n        return extras.partId;\n      }\n      return layout.activePart({grid: 'mainArea'})?.id ?? layout.activePart({grid: 'workbench'}).id;\n    })();\n\n    return layout\n      .addView(viewId, {\n        partId: partId,\n        position: extras.position ?? 'after-active-view',\n        activateView: extras.activate ?? true,\n        activatePart: extras.activate ?? true,\n      })\n      .navigateView(viewId, commands, {\n        relativeTo: extras.relativeTo,\n        hint: extras.hint,\n        cssClass: extras.cssClass,\n        data: extras.data,\n        state: extras.state,\n      });\n  }\n\n  /**\n   * Updates the specified view.\n   */\n  function updateView(viewId: string, layout: ɵWorkbenchLayout): ɵWorkbenchLayout {\n    if (extras.activate ?? true) {\n      layout = layout.activateView(viewId, {activatePart: true});\n    }\n\n    return layout.navigateView(viewId, commands, {\n      relativeTo: extras.relativeTo,\n      hint: extras.hint,\n      cssClass: extras.cssClass,\n      data: extras.data,\n      state: extras.state,\n    });\n  }\n}\n\n/**\n * Creates navigation extras with workbench navigation instructions.\n */\nfunction createNavigationExtras(layouts: {newLayout: ɵWorkbenchLayout; currentLayout?: ɵWorkbenchLayout | undefined}, extras?: Omit<NavigationExtras, 'relativeTo' | 'state'>): NavigationExtras {\n  const {newLayout, currentLayout} = layouts;\n  const {workbenchGrid, mainAreaGrid} = newLayout.serialize({excludeViewMarkedForRemoval: true});\n\n  return {\n    ...extras,\n    // Instruct the Angular router to process the navigation even if the URL does not change, e.g., when changing the workbench grid which is not contained in the URL.\n    onSameUrlNavigation: 'reload',\n    // Unset `relativeTo` because commands are already normalized to their absolute form.\n    relativeTo: null,\n    // Associate workbench-specific state with the navigation.\n    state: WorkbenchNavigationalStates.create({\n      workbenchGrid: workbenchGrid,\n      // Fall back to the current layout's perspective if not specified by the new layout,\n      // e.g., the navigator provides a new layout to replace the current layout.\n      perspectiveId: newLayout.perspectiveId ?? currentLayout?.perspectiveId,\n      maximized: newLayout.maximized,\n      navigationStates: newLayout.navigationStates(),\n    }),\n    // Add the main area as query parameter.\n    queryParams: {...extras?.queryParams, [MAIN_AREA_LAYOUT_QUERY_PARAM]: mainAreaGrid},\n    // Merge with existing query params unless specified an explicit strategy, e.g., for migrating an outdated layout URL.\n    // Note that `null` is a valid strategy for clearing existing query params, so do not use the nullish coalescing operator (??).\n    queryParamsHandling: Defined.orElse(extras?.queryParamsHandling, 'merge'),\n  };\n}\n\n/**\n * Computes commands that can be passed to the Angular router to navigate view outlets and to remove view outlets of removed views.\n */\nfunction computeNavigationCommands(previousViewOutlets: ViewOutlets, nextViewOutlets: ViewOutlets): [{outlets: {[outlet: ViewId]: Commands | null}}] | [] {\n  const previousViewOutletMap = new Map<ViewId, UrlSegment[]>(Objects.entries(previousViewOutlets));\n  const nextViewOutletMap = new Map<ViewId, UrlSegment[]>(Objects.entries(nextViewOutlets));\n\n  const commands = new Map<ViewId, Commands | null>();\n  const viewIds = new Set<ViewId>([...previousViewOutletMap.keys(), ...nextViewOutletMap.keys()]);\n\n  viewIds.forEach(viewId => {\n    // Test if the view was added to the layout.\n    if (!previousViewOutletMap.has(viewId)) {\n      commands.set(viewId, Routing.segmentsToCommands(nextViewOutletMap.get(viewId)!));\n    }\n    // Test if the view was removed from the layout.\n    else if (!nextViewOutletMap.has(viewId)) {\n      commands.set(viewId, null);\n    }\n    // Test if the view was updated.\n    else if (!new UrlSegmentMatcher(previousViewOutletMap.get(viewId)!, {matchMatrixParams: true, matchWildcardPath: false}).matches(nextViewOutletMap.get(viewId)!)) {\n      commands.set(viewId, Routing.segmentsToCommands(nextViewOutletMap.get(viewId)!));\n    }\n  });\n\n  // Add view commands to the 'outlets' property to be interpreted by the Angular router.\n  return commands.size ? [{outlets: Object.fromEntries(commands)}] : [];\n}\n\n/**\n * @inheritDoc\n * @see NavigateFn\n */\nexport type ɵNavigateFn = (layout: ɵWorkbenchLayout) => Promise<ɵWorkbenchLayout | null> | ɵWorkbenchLayout | null;\n"]}
|
|
@@ -15,10 +15,10 @@ import * as i0 from "@angular/core";
|
|
|
15
15
|
*/
|
|
16
16
|
export class SplashComponent {
|
|
17
17
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: SplashComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
18
|
-
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: SplashComponent, isStandalone: true, selector: "wb-splash", ngImport: i0, template: "<sci-throbber type=\"ellipsis\"
|
|
18
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.0.2", type: SplashComponent, isStandalone: true, selector: "wb-splash", ngImport: i0, template: "<sci-throbber type=\"ellipsis\"/>\n", styles: [":host{display:grid;justify-content:center;margin-top:3em}:host>sci-throbber{--sci-throbber-size: 80px}\n"], dependencies: [{ kind: "component", type: SciThrobberComponent, selector: "sci-throbber", inputs: ["type"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush }); }
|
|
19
19
|
}
|
|
20
20
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: SplashComponent, decorators: [{
|
|
21
21
|
type: Component,
|
|
22
|
-
args: [{ selector: 'wb-splash', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [SciThrobberComponent], template: "<sci-throbber type=\"ellipsis\"
|
|
22
|
+
args: [{ selector: 'wb-splash', changeDetection: ChangeDetectionStrategy.OnPush, standalone: true, imports: [SciThrobberComponent], template: "<sci-throbber type=\"ellipsis\"/>\n", styles: [":host{display:grid;justify-content:center;margin-top:3em}:host>sci-throbber{--sci-throbber-size: 80px}\n"] }]
|
|
23
23
|
}] });
|
|
24
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
24
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic3BsYXNoLmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3NjaW9uL3dvcmtiZW5jaC9zcmMvbGliL3N0YXJ0dXAvc3BsYXNoL3NwbGFzaC5jb21wb25lbnQudHMiLCIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy9zY2lvbi93b3JrYmVuY2gvc3JjL2xpYi9zdGFydHVwL3NwbGFzaC9zcGxhc2guY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUE7Ozs7Ozs7O0dBUUc7QUFFSCxPQUFPLEVBQUMsdUJBQXVCLEVBQUUsU0FBUyxFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQ2pFLE9BQU8sRUFBQyxvQkFBb0IsRUFBQyxNQUFNLDRCQUE0QixDQUFDOztBQUVoRTs7R0FFRztBQVNILE1BQU0sT0FBTyxlQUFlOzhHQUFmLGVBQWU7a0dBQWYsZUFBZSxxRUN4QjVCLHFDQUNBLGtLRHFCWSxvQkFBb0I7OzJGQUVuQixlQUFlO2tCQVIzQixTQUFTOytCQUNFLFdBQVcsbUJBR0osdUJBQXVCLENBQUMsTUFBTSxjQUNuQyxJQUFJLFdBQ1AsQ0FBQyxvQkFBb0IsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qXG4gKiBDb3B5cmlnaHQgKGMpIDIwMTgtMjAyMiBTd2lzcyBGZWRlcmFsIFJhaWx3YXlzXG4gKlxuICogVGhpcyBwcm9ncmFtIGFuZCB0aGUgYWNjb21wYW55aW5nIG1hdGVyaWFscyBhcmUgbWFkZVxuICogYXZhaWxhYmxlIHVuZGVyIHRoZSB0ZXJtcyBvZiB0aGUgRWNsaXBzZSBQdWJsaWMgTGljZW5zZSAyLjBcbiAqIHdoaWNoIGlzIGF2YWlsYWJsZSBhdCBodHRwczovL3d3dy5lY2xpcHNlLm9yZy9sZWdhbC9lcGwtMi4wL1xuICpcbiAqIFNQRFgtTGljZW5zZS1JZGVudGlmaWVyOiBFUEwtMi4wXG4gKi9cblxuaW1wb3J0IHtDaGFuZ2VEZXRlY3Rpb25TdHJhdGVneSwgQ29tcG9uZW50fSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7U2NpVGhyb2JiZXJDb21wb25lbnR9IGZyb20gJ0BzY2lvbi9jb21wb25lbnRzL3Rocm9iYmVyJztcblxuLyoqXG4gKiBEZWZhdWx0IHNwbGFzaCBkaXNwbGF5ZWQgd2hpbGUgc3RhcnRpbmcgdGhlIHdvcmtiZW5jaC5cbiAqL1xuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAnd2Itc3BsYXNoJyxcbiAgdGVtcGxhdGVVcmw6ICcuL3NwbGFzaC5jb21wb25lbnQuaHRtbCcsXG4gIHN0eWxlVXJsczogWycuL3NwbGFzaC5jb21wb25lbnQuc2NzcyddLFxuICBjaGFuZ2VEZXRlY3Rpb246IENoYW5nZURldGVjdGlvblN0cmF0ZWd5Lk9uUHVzaCxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW1NjaVRocm9iYmVyQ29tcG9uZW50XSxcbn0pXG5leHBvcnQgY2xhc3MgU3BsYXNoQ29tcG9uZW50IHtcbn1cbiIsIjxzY2ktdGhyb2JiZXIgdHlwZT1cImVsbGlwc2lzXCIvPlxuIl19
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { APP_INITIALIZER, ApplicationInitStatus, inject, Injectable, makeEnvironmentProviders, NgZone } from '@angular/core';
|
|
1
|
+
import { APP_INITIALIZER, ApplicationInitStatus, inject, Injectable, makeEnvironmentProviders, NgZone, signal } from '@angular/core';
|
|
2
2
|
import { runWorkbenchInitializers, WORKBENCH_POST_STARTUP, WORKBENCH_PRE_STARTUP, WORKBENCH_STARTUP } from './workbench-initializer';
|
|
3
3
|
import { LoggerNames } from '../logging';
|
|
4
4
|
import * as i0 from "@angular/core";
|
|
@@ -120,21 +120,18 @@ var StartupState;
|
|
|
120
120
|
*/
|
|
121
121
|
export class WorkbenchStartup {
|
|
122
122
|
constructor() {
|
|
123
|
-
|
|
123
|
+
/**
|
|
124
|
+
* Signals when the workbench completed startup.
|
|
125
|
+
*/
|
|
126
|
+
this.isStarted = signal(false);
|
|
124
127
|
/**
|
|
125
128
|
* Promise that resolves when the workbench has completed the startup.
|
|
126
129
|
*/
|
|
127
130
|
this.whenStarted = new Promise(resolve => this.notifyStarted = () => {
|
|
128
|
-
this.
|
|
131
|
+
this.isStarted.set(true);
|
|
129
132
|
resolve(true);
|
|
130
133
|
});
|
|
131
134
|
}
|
|
132
|
-
/**
|
|
133
|
-
* Returns whether the workbench completed startup.
|
|
134
|
-
*/
|
|
135
|
-
isStarted() {
|
|
136
|
-
return this._started;
|
|
137
|
-
}
|
|
138
135
|
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: WorkbenchStartup, deps: [], target: i0.ɵɵFactoryTarget.Injectable }); }
|
|
139
136
|
static { this.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: WorkbenchStartup, providedIn: 'root' }); }
|
|
140
137
|
}
|
|
@@ -158,4 +155,4 @@ export function provideWorkbenchLauncher(workbenchConfig) {
|
|
|
158
155
|
multi: true,
|
|
159
156
|
}]);
|
|
160
157
|
}
|
|
161
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workbench-launcher.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/startup/workbench-launcher.service.ts"],"names":[],"mappings":"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","sourcesContent":["/*\n * Copyright (c) 2018-2022 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {WorkbenchConfig} from '../workbench-config';\nimport {APP_INITIALIZER, ApplicationInitStatus, EnvironmentProviders, inject, Injectable, Injector, makeEnvironmentProviders, NgZone} from '@angular/core';\nimport {runWorkbenchInitializers, WORKBENCH_POST_STARTUP, WORKBENCH_PRE_STARTUP, WORKBENCH_STARTUP} from './workbench-initializer';\nimport {Logger, LoggerNames} from '../logging';\n\n/**\n * Provides API to launch the SCION Workbench.\n *\n * During workbench startup, the launcher runs registered workbench initializers and waits for them to complete.\n * Workbench initializers may execute in parallel. To register a workbench initializer, provide a class implementing\n * {@link WorkbenchInitializer} as a multi-provider using the DI class token {@link WorkbenchInitializer}.\n *\n * The SCION Workbench supports the following launchers:\n *\n *  - **APP_INITIALIZER**\n *   Launches the workbench in an Angular `APP_INITIALIZER`, which is before bootstrapping the app component.\n *\n * - **LAZY** (which is the default)\n *   Launches the workbench at the latest when bootstrapping the workbench root component `<wb-workbench>`.\n *\n *   With this strategy, you are flexible when to start the workbench. You can start the workbench explicitly by\n *   calling {@link WorkbenchLauncher#launch}, e.g., to launch the workbench from a route guard or app initializer,\n *   or start it automatically when adding the workbench root component `<wb-workbench>` to the Angular component\n *   tree.\n *\n * The workbench component displays a startup splash until completed startup. You can configure a custom splash in\n * {@link WorkbenchConfig#startup#splash}. When launching the workbench in an Angular `APP_INITIALIZER`, no splash will\n * display since the workbench will start upfront.\n */\n@Injectable({providedIn: 'root'})\nexport class WorkbenchLauncher {\n\n  private _state: StartupState = StartupState.Stopped;\n\n  constructor(private _startup: WorkbenchStartup,\n              private _logger: Logger,\n              private _zone: NgZone,\n              private _injector: Injector) {\n  }\n\n  /**\n   * Launches the SCION Workbench.\n   *\n   * This method represents the initial entry point that should be called to launch the workbench. The launcher runs\n   * registered workbench initializers and waits for them to complete. Calling this method has no effect if the workbench\n   * is already starting or has started.\n   *\n   * ### Splash\n   * When mounting the workbench root component `<wb-workbench>` to the DOM before the workbench startup has finished,\n   * the workbench will display a startup splash until completed startup.\n   *\n   * ### Startup Lifecycle Hooks\n   * The SCION Workbench defines a number of injection tokens (also called DI tokens) as hooks into the workbench's startup process.\n   * Hooks are called at defined points during startup, enabling the application's controlled initialization.\n   *\n   * The application can associate one or more initializers with any of these DI tokens. An initializer can be any object and is to\n   * be provided as a multi-provider. If the initializer is a function or instance of {@link WorkbenchInitializer} and returns a Promise,\n   * the workbench waits for the Promise to be resolved before proceeding with the startup. Initializers can call `inject` to get any\n   * required dependencies. Initializers associated with the same DI token may run in parallel.\n   *\n   * Following DI tokens are available as hooks into the workbench's startup process, listed in the order in which they are injected and\n   * executed.\n   *\n   * - {@link WORKBENCH_PRE_STARTUP}\n   * - {@link WORKBENCH_STARTUP}\n   * - {@link MICROFRONTEND_PLATFORM_PRE_STARTUP}\n   * - {@link MICROFRONTEND_PLATFORM_POST_STARTUP}\n   * - {@link WORKBENCH_POST_STARTUP}\n   *\n   * ### Example of how to hook into the startup process.\n   *\n   * ```ts\n   * {\n   *   provide: WORKBENCH_STARTUP,\n   *   multi: true,\n   *   useValue: () => inject(Initializer).init(),\n   * }\n   * ```\n   *\n   * @return A Promise that resolves when the workbench has completed the startup or that rejects if the startup failed.\n   */\n  public async launch(): Promise<true> {\n    // Ensure to run in the Angular zone to check the workbench for changes even if called from outside the Angular zone, e.g. in unit tests.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.launch());\n    }\n\n    switch (this._state) {\n      case StartupState.Stopped: {\n        this._logger.debug(() => `Starting Workbench. Waiting for workbench initializers to complete. [launcher=${this._injector.get(ApplicationInitStatus).done ? 'LAZY' : 'APP_INITIALIZER'}]`, LoggerNames.LIFECYCLE);\n        this._state = StartupState.Starting;\n        await runWorkbenchInitializers(WORKBENCH_PRE_STARTUP, this._injector);\n        await runWorkbenchInitializers(WORKBENCH_STARTUP, this._injector);\n        this._state = StartupState.Started;\n        await runWorkbenchInitializers(WORKBENCH_POST_STARTUP, this._injector);\n        this._logger.debug(() => 'Workbench started.', LoggerNames.LIFECYCLE);\n        this._startup.notifyStarted();\n        return this._startup.whenStarted;\n      }\n      case StartupState.Starting:\n      case StartupState.Started: {\n        return this._startup.whenStarted;\n      }\n      default: {\n        throw Error(`[WorkbenchStartupError] Illegal startup state: ${this._state}`);\n      }\n    }\n  }\n}\n\nenum StartupState {\n  Stopped, Starting, Started\n}\n\n/**\n * Allows waiting for the workbench startup to complete.\n */\n@Injectable({providedIn: 'root'})\nexport class WorkbenchStartup {\n\n  private _started = false;\n\n  /* @internal */\n  public notifyStarted!: () => void;\n\n  /**\n   * Promise that resolves when the workbench has completed the startup.\n   */\n  public readonly whenStarted = new Promise<true>(resolve => this.notifyStarted = () => {\n    this._started = true;\n    resolve(true);\n  });\n\n  /**\n   * Returns whether the workbench completed startup.\n   */\n  public isStarted(): boolean {\n    return this._started;\n  }\n}\n\n/**\n * Provides a set of DI providers for launching the workbench.\n */\nexport function provideWorkbenchLauncher(workbenchConfig: WorkbenchConfig): EnvironmentProviders | [] {\n  if (workbenchConfig.startup?.launcher !== 'APP_INITIALIZER') {\n    return [];\n  }\n\n  return makeEnvironmentProviders([{\n    provide: APP_INITIALIZER,\n    useFactory: () => {\n      const launcher = inject(WorkbenchLauncher);\n      return () => launcher.launch();\n    },\n    multi: true,\n  }]);\n}\n"]}
|
|
158
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"workbench-launcher.service.js","sourceRoot":"","sources":["../../../../../../projects/scion/workbench/src/lib/startup/workbench-launcher.service.ts"],"names":[],"mappings":"AAWA,OAAO,EAAC,eAAe,EAAE,qBAAqB,EAAwB,MAAM,EAAE,UAAU,EAAY,wBAAwB,EAAE,MAAM,EAAE,MAAM,EAAC,MAAM,eAAe,CAAC;AACnK,OAAO,EAAC,wBAAwB,EAAE,sBAAsB,EAAE,qBAAqB,EAAE,iBAAiB,EAAC,MAAM,yBAAyB,CAAC;AACnI,OAAO,EAAS,WAAW,EAAC,MAAM,YAAY,CAAC;;;AAE/C;;;;;;;;;;;;;;;;;;;;;;;GAuBG;AAEH,MAAM,OAAO,iBAAiB;IAI5B,YAAoB,QAA0B,EAC1B,OAAe,EACf,KAAa,EACb,SAAmB;QAHnB,aAAQ,GAAR,QAAQ,CAAkB;QAC1B,YAAO,GAAP,OAAO,CAAQ;QACf,UAAK,GAAL,KAAK,CAAQ;QACb,cAAS,GAAT,SAAS,CAAU;QAL/B,WAAM,GAAiB,YAAY,CAAC,OAAO,CAAC;IAMpD,CAAC;IAED;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;OAwCG;IACI,KAAK,CAAC,MAAM;QACjB,yIAAyI;QACzI,IAAI,CAAC,MAAM,CAAC,eAAe,EAAE,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;QAC7C,CAAC;QAED,QAAQ,IAAI,CAAC,MAAM,EAAE,CAAC;YACpB,KAAK,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC1B,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,iFAAiF,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,qBAAqB,CAAC,CAAC,IAAI,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,iBAAiB,GAAG,EAAE,WAAW,CAAC,SAAS,CAAC,CAAC;gBACjN,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC,QAAQ,CAAC;gBACpC,MAAM,wBAAwB,CAAC,qBAAqB,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;gBACtE,MAAM,wBAAwB,CAAC,iBAAiB,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;gBAClE,IAAI,CAAC,MAAM,GAAG,YAAY,CAAC,OAAO,CAAC;gBACnC,MAAM,wBAAwB,CAAC,sBAAsB,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC;gBACvE,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,GAAG,EAAE,CAAC,oBAAoB,EAAE,WAAW,CAAC,SAAS,CAAC,CAAC;gBACtE,IAAI,CAAC,QAAQ,CAAC,aAAa,EAAE,CAAC;gBAC9B,OAAO,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC;YACnC,CAAC;YACD,KAAK,YAAY,CAAC,QAAQ,CAAC;YAC3B,KAAK,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC;gBAC1B,OAAO,IAAI,CAAC,QAAQ,CAAC,WAAW,CAAC;YACnC,CAAC;YACD,OAAO,CAAC,CAAC,CAAC;gBACR,MAAM,KAAK,CAAC,kDAAkD,IAAI,CAAC,MAAM,EAAE,CAAC,CAAC;YAC/E,CAAC;QACH,CAAC;IACH,CAAC;8GA7EU,iBAAiB;kHAAjB,iBAAiB,cADL,MAAM;;2FAClB,iBAAiB;kBAD7B,UAAU;mBAAC,EAAC,UAAU,EAAE,MAAM,EAAC;;AAiFhC,IAAK,YAEJ;AAFD,WAAK,YAAY;IACf,qDAAO,CAAA;IAAE,uDAAQ,CAAA;IAAE,qDAAO,CAAA;AAC5B,CAAC,EAFI,YAAY,KAAZ,YAAY,QAEhB;AAED;;GAEG;AAEH,MAAM,OAAO,gBAAgB;IAD7B;QAGE;;WAEG;QACa,cAAS,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;QAK1C;;WAEG;QACa,gBAAW,GAAG,IAAI,OAAO,CAAO,OAAO,CAAC,EAAE,CAAC,IAAI,CAAC,aAAa,GAAG,GAAG,EAAE;YACnF,IAAI,CAAC,SAAS,CAAC,GAAG,CAAC,IAAI,CAAC,CAAC;YACzB,OAAO,CAAC,IAAI,CAAC,CAAC;QAChB,CAAC,CAAC,CAAC;KACJ;8GAjBY,gBAAgB;kHAAhB,gBAAgB,cADJ,MAAM;;2FAClB,gBAAgB;kBAD5B,UAAU;mBAAC,EAAC,UAAU,EAAE,MAAM,EAAC;;AAoBhC;;GAEG;AACH,MAAM,UAAU,wBAAwB,CAAC,eAAgC;IACvE,IAAI,eAAe,CAAC,OAAO,EAAE,QAAQ,KAAK,iBAAiB,EAAE,CAAC;QAC5D,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,OAAO,wBAAwB,CAAC,CAAC;YAC/B,OAAO,EAAE,eAAe;YACxB,UAAU,EAAE,GAAG,EAAE;gBACf,MAAM,QAAQ,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;gBAC3C,OAAO,GAAG,EAAE,CAAC,QAAQ,CAAC,MAAM,EAAE,CAAC;YACjC,CAAC;YACD,KAAK,EAAE,IAAI;SACZ,CAAC,CAAC,CAAC;AACN,CAAC","sourcesContent":["/*\n * Copyright (c) 2018-2022 Swiss Federal Railways\n *\n * This program and the accompanying materials are made\n * available under the terms of the Eclipse Public License 2.0\n * which is available at https://www.eclipse.org/legal/epl-2.0/\n *\n * SPDX-License-Identifier: EPL-2.0\n */\n\nimport {WorkbenchConfig} from '../workbench-config';\nimport {APP_INITIALIZER, ApplicationInitStatus, EnvironmentProviders, inject, Injectable, Injector, makeEnvironmentProviders, NgZone, signal} from '@angular/core';\nimport {runWorkbenchInitializers, WORKBENCH_POST_STARTUP, WORKBENCH_PRE_STARTUP, WORKBENCH_STARTUP} from './workbench-initializer';\nimport {Logger, LoggerNames} from '../logging';\n\n/**\n * Provides API to launch the SCION Workbench.\n *\n * During workbench startup, the launcher runs registered workbench initializers and waits for them to complete.\n * Workbench initializers may execute in parallel. To register a workbench initializer, provide a class implementing\n * {@link WorkbenchInitializer} as a multi-provider using the DI class token {@link WorkbenchInitializer}.\n *\n * The SCION Workbench supports the following launchers:\n *\n *  - **APP_INITIALIZER**\n *   Launches the workbench in an Angular `APP_INITIALIZER`, which is before bootstrapping the app component.\n *\n * - **LAZY** (which is the default)\n *   Launches the workbench at the latest when bootstrapping the workbench root component `<wb-workbench>`.\n *\n *   With this strategy, you are flexible when to start the workbench. You can start the workbench explicitly by\n *   calling {@link WorkbenchLauncher#launch}, e.g., to launch the workbench from a route guard or app initializer,\n *   or start it automatically when adding the workbench root component `<wb-workbench>` to the Angular component\n *   tree.\n *\n * The workbench component displays a startup splash until completed startup. You can configure a custom splash in\n * {@link WorkbenchConfig#startup#splash}. When launching the workbench in an Angular `APP_INITIALIZER`, no splash will\n * display since the workbench will start upfront.\n */\n@Injectable({providedIn: 'root'})\nexport class WorkbenchLauncher {\n\n  private _state: StartupState = StartupState.Stopped;\n\n  constructor(private _startup: WorkbenchStartup,\n              private _logger: Logger,\n              private _zone: NgZone,\n              private _injector: Injector) {\n  }\n\n  /**\n   * Launches the SCION Workbench.\n   *\n   * This method represents the initial entry point that should be called to launch the workbench. The launcher runs\n   * registered workbench initializers and waits for them to complete. Calling this method has no effect if the workbench\n   * is already starting or has started.\n   *\n   * ### Splash\n   * When mounting the workbench root component `<wb-workbench>` to the DOM before the workbench startup has finished,\n   * the workbench will display a startup splash until completed startup.\n   *\n   * ### Startup Lifecycle Hooks\n   * The SCION Workbench defines a number of injection tokens (also called DI tokens) as hooks into the workbench's startup process.\n   * Hooks are called at defined points during startup, enabling the application's controlled initialization.\n   *\n   * The application can associate one or more initializers with any of these DI tokens. An initializer can be any object and is to\n   * be provided as a multi-provider. If the initializer is a function or instance of {@link WorkbenchInitializer} and returns a Promise,\n   * the workbench waits for the Promise to be resolved before proceeding with the startup. Initializers can call `inject` to get any\n   * required dependencies. Initializers associated with the same DI token may run in parallel.\n   *\n   * Following DI tokens are available as hooks into the workbench's startup process, listed in the order in which they are injected and\n   * executed.\n   *\n   * - {@link WORKBENCH_PRE_STARTUP}\n   * - {@link WORKBENCH_STARTUP}\n   * - {@link MICROFRONTEND_PLATFORM_PRE_STARTUP}\n   * - {@link MICROFRONTEND_PLATFORM_POST_STARTUP}\n   * - {@link WORKBENCH_POST_STARTUP}\n   *\n   * ### Example of how to hook into the startup process.\n   *\n   * ```ts\n   * {\n   *   provide: WORKBENCH_STARTUP,\n   *   multi: true,\n   *   useValue: () => inject(Initializer).init(),\n   * }\n   * ```\n   *\n   * @return A Promise that resolves when the workbench has completed the startup or that rejects if the startup failed.\n   */\n  public async launch(): Promise<true> {\n    // Ensure to run in the Angular zone to check the workbench for changes even if called from outside the Angular zone, e.g. in unit tests.\n    if (!NgZone.isInAngularZone()) {\n      return this._zone.run(() => this.launch());\n    }\n\n    switch (this._state) {\n      case StartupState.Stopped: {\n        this._logger.debug(() => `Starting Workbench. Waiting for workbench initializers to complete. [launcher=${this._injector.get(ApplicationInitStatus).done ? 'LAZY' : 'APP_INITIALIZER'}]`, LoggerNames.LIFECYCLE);\n        this._state = StartupState.Starting;\n        await runWorkbenchInitializers(WORKBENCH_PRE_STARTUP, this._injector);\n        await runWorkbenchInitializers(WORKBENCH_STARTUP, this._injector);\n        this._state = StartupState.Started;\n        await runWorkbenchInitializers(WORKBENCH_POST_STARTUP, this._injector);\n        this._logger.debug(() => 'Workbench started.', LoggerNames.LIFECYCLE);\n        this._startup.notifyStarted();\n        return this._startup.whenStarted;\n      }\n      case StartupState.Starting:\n      case StartupState.Started: {\n        return this._startup.whenStarted;\n      }\n      default: {\n        throw Error(`[WorkbenchStartupError] Illegal startup state: ${this._state}`);\n      }\n    }\n  }\n}\n\nenum StartupState {\n  Stopped, Starting, Started\n}\n\n/**\n * Allows waiting for the workbench startup to complete.\n */\n@Injectable({providedIn: 'root'})\nexport class WorkbenchStartup {\n\n  /**\n   * Signals when the workbench completed startup.\n   */\n  public readonly isStarted = signal(false);\n\n  /* @internal */\n  public notifyStarted!: () => void;\n\n  /**\n   * Promise that resolves when the workbench has completed the startup.\n   */\n  public readonly whenStarted = new Promise<true>(resolve => this.notifyStarted = () => {\n    this.isStarted.set(true);\n    resolve(true);\n  });\n}\n\n/**\n * Provides a set of DI providers for launching the workbench.\n */\nexport function provideWorkbenchLauncher(workbenchConfig: WorkbenchConfig): EnvironmentProviders | [] {\n  if (workbenchConfig.startup?.launcher !== 'APP_INITIALIZER') {\n    return [];\n  }\n\n  return makeEnvironmentProviders([{\n    provide: APP_INITIALIZER,\n    useFactory: () => {\n      const launcher = inject(WorkbenchLauncher);\n      return () => launcher.launch();\n    },\n    multi: true,\n  }]);\n}\n"]}
|
|
@@ -37,7 +37,7 @@ export function provideWorkbenchForTest(config) {
|
|
|
37
37
|
provideWorkbench(config),
|
|
38
38
|
provideNoopAnimations(),
|
|
39
39
|
{ provide: ActivationInstantProvider, useClass: SequenceInstantProvider },
|
|
40
|
-
{ provide: MAIN_AREA_INITIAL_PART_ID, useValue: 'main' },
|
|
40
|
+
{ provide: MAIN_AREA_INITIAL_PART_ID, useValue: config?.mainAreaInitialPartId ?? 'main' },
|
|
41
41
|
{ provide: ComponentFixtureAutoDetect, useValue: true },
|
|
42
42
|
{ provide: ENVIRONMENT_INITIALIZER, multi: true, useValue: () => inject(Router).initialNavigation() },
|
|
43
43
|
{ provide: ENVIRONMENT_INITIALIZER, multi: true, useValue: () => localStorage.clear() },
|
|
@@ -57,4 +57,4 @@ class SequenceInstantProvider {
|
|
|
57
57
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.0.2", ngImport: i0, type: SequenceInstantProvider, decorators: [{
|
|
58
58
|
type: Injectable
|
|
59
59
|
}] });
|
|
60
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
60
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoid29ya2JlbmNoLnByb3ZpZGVyLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvc2Npb24vd29ya2JlbmNoL3NyYy9saWIvdGVzdGluZy93b3JrYmVuY2gucHJvdmlkZXIudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQ0EsT0FBTyxFQUFDLHVCQUF1QixFQUF3QixNQUFNLEVBQUUsVUFBVSxFQUFFLHdCQUF3QixFQUFDLE1BQU0sZUFBZSxDQUFDO0FBQzFILE9BQU8sRUFBQyxnQkFBZ0IsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ3ZELE9BQU8sRUFBQyxxQkFBcUIsRUFBQyxNQUFNLHNDQUFzQyxDQUFDO0FBQzNFLE9BQU8sRUFBQyx5QkFBeUIsRUFBQyxNQUFNLGdDQUFnQyxDQUFDO0FBQ3pFLE9BQU8sRUFBQyx5QkFBeUIsRUFBQyxNQUFNLDZCQUE2QixDQUFDO0FBQ3RFLE9BQU8sRUFBQywwQkFBMEIsRUFBQyxNQUFNLHVCQUF1QixDQUFDO0FBQ2pFLE9BQU8sRUFBQyxNQUFNLEVBQUMsTUFBTSxpQkFBaUIsQ0FBQzs7QUFFdkM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0F5Qkc7QUFDSCxNQUFNLFVBQVUsdUJBQXVCLENBQUMsTUFBMkQ7SUFDakcsT0FBTyx3QkFBd0IsQ0FBQztRQUM5QixnQkFBZ0IsQ0FBQyxNQUFNLENBQUM7UUFDeEIscUJBQXFCLEVBQUU7UUFDdkIsRUFBQyxPQUFPLEVBQUUseUJBQXlCLEVBQUUsUUFBUSxFQUFFLHVCQUF1QixFQUFDO1FBQ3ZFLEVBQUMsT0FBTyxFQUFFLHlCQUF5QixFQUFFLFFBQVEsRUFBRSxNQUFNLEVBQUUscUJBQXFCLElBQUksTUFBTSxFQUFDO1FBQ3ZGLEVBQUMsT0FBTyxFQUFFLDBCQUEwQixFQUFFLFFBQVEsRUFBRSxJQUFJLEVBQUM7UUFDckQsRUFBQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsTUFBTSxDQUFDLE1BQU0sQ0FBQyxDQUFDLGlCQUFpQixFQUFFLEVBQUM7UUFDbkcsRUFBQyxPQUFPLEVBQUUsdUJBQXVCLEVBQUUsS0FBSyxFQUFFLElBQUksRUFBRSxRQUFRLEVBQUUsR0FBRyxFQUFFLENBQUMsWUFBWSxDQUFDLEtBQUssRUFBRSxFQUFDO1FBQ3JGLEVBQUMsT0FBTyxFQUFFLHVCQUF1QixFQUFFLEtBQUssRUFBRSxJQUFJLEVBQUUsUUFBUSxFQUFFLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FBQyxJQUFJLEdBQUcsRUFBRSxFQUFDO0tBQ2xGLENBQUMsQ0FBQztBQUNMLENBQUM7QUFFRCxNQUNNLHVCQUF1QjtJQUQ3QjtRQUdVLGNBQVMsR0FBRyxDQUFDLENBQUM7S0FLdkI7SUFIUSxHQUFHO1FBQ1IsT0FBTyxJQUFJLENBQUMsU0FBUyxFQUFFLENBQUM7SUFDMUIsQ0FBQzs4R0FORyx1QkFBdUI7a0hBQXZCLHVCQUF1Qjs7MkZBQXZCLHVCQUF1QjtrQkFENUIsVUFBVSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7V29ya2JlbmNoQ29uZmlnfSBmcm9tICcuLi93b3JrYmVuY2gtY29uZmlnJztcbmltcG9ydCB7RU5WSVJPTk1FTlRfSU5JVElBTElaRVIsIEVudmlyb25tZW50UHJvdmlkZXJzLCBpbmplY3QsIEluamVjdGFibGUsIG1ha2VFbnZpcm9ubWVudFByb3ZpZGVyc30gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5pbXBvcnQge3Byb3ZpZGVXb3JrYmVuY2h9IGZyb20gJy4uL3dvcmtiZW5jaC5wcm92aWRlcic7XG5pbXBvcnQge3Byb3ZpZGVOb29wQW5pbWF0aW9uc30gZnJvbSAnQGFuZ3VsYXIvcGxhdGZvcm0tYnJvd3Nlci9hbmltYXRpb25zJztcbmltcG9ydCB7QWN0aXZhdGlvbkluc3RhbnRQcm92aWRlcn0gZnJvbSAnLi4vYWN0aXZhdGlvbi1pbnN0YW50LnByb3ZpZGVyJztcbmltcG9ydCB7TUFJTl9BUkVBX0lOSVRJQUxfUEFSVF9JRH0gZnJvbSAnLi4vbGF5b3V0L8m1d29ya2JlbmNoLWxheW91dCc7XG5pbXBvcnQge0NvbXBvbmVudEZpeHR1cmVBdXRvRGV0ZWN0fSBmcm9tICdAYW5ndWxhci9jb3JlL3Rlc3RpbmcnO1xuaW1wb3J0IHtSb3V0ZXJ9IGZyb20gJ0Bhbmd1bGFyL3JvdXRlcic7XG5cbi8qKlxuICogUmV0dXJucyB0aGUgc2V0IG9mIGRlcGVuZGVuY3ktaW5qZWN0aW9uIHByb3ZpZGVycyB0byBlbmFibGUgYW5kIGNvbmZpZ3VyZSB0aGUgU0NJT04gV29ya2JlbmNoIGluIHRlc3RzLlxuICpcbiAqIERvZXMgdGhlIGZvbGxvd2luZzpcbiAqIC0gcHJvdmlkZXMgdGhlIHdvcmtiZW5jaCB3aXRoIGdpdmVuIGNvbmZpZ1xuICogLSBjb25maWd1cmVzIHRoZSB3b3JrYmVuY2ggdG8gbmFtZSB0aGUgaW5pdGlhbCBwYXJ0ICdtYWluJ1xuICogLSBpbnN0YWxscyBhIHNlcXVlbmNlIGZvciBhY3RpdmF0aW9uIGluc3RhbnRzXG4gKiAtIGNvbmZpZ3VyZXMgdGhlIHRlc3RiZWQgdG8gYXV0by1kZXRlY3QgY2hhbmdlc1xuICogLSBkaXNhYmxlcyBhbmltYXRpb25zXG4gKiAtIHBlcmZvcm1zIHRoZSBpbml0aWFsIG5hdmlnYXRpb25cbiAqXG4gKiAjIyMgVXNhZ2VcbiAqXG4gKiBgYGBcbiAqIGJlZm9yZUVhY2goKCkgPT4ge1xuICogICBUZXN0QmVkLmNvbmZpZ3VyZVRlc3RpbmdNb2R1bGUoe1xuICogICAgICBpbXBvcnRzOiBbXG4gKiAgICAgICAgV29ya2JlbmNoVGVzdGluZ01vZHVsZS5mb3JUZXN0KCksXG4gKiAgICAgICAgUm91dGVyVGVzdGluZ01vZHVsZS53aXRoUm91dGVzKFtcbiAqICAgICAgICAgIHtwYXRoOiAndGVzdC12aWV3JywgY29tcG9uZW50OiBUZXN0Q29tcG9uZW50fSxcbiAqICAgICAgICBdKSxcbiAqICAgICAgXSxcbiAqICAgIH0pO1xuICogfSk7XG4gKiBgYGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIHByb3ZpZGVXb3JrYmVuY2hGb3JUZXN0KGNvbmZpZz86IFdvcmtiZW5jaENvbmZpZyAmIHttYWluQXJlYUluaXRpYWxQYXJ0SWQ/OiBzdHJpbmd9KTogRW52aXJvbm1lbnRQcm92aWRlcnMge1xuICByZXR1cm4gbWFrZUVudmlyb25tZW50UHJvdmlkZXJzKFtcbiAgICBwcm92aWRlV29ya2JlbmNoKGNvbmZpZyksXG4gICAgcHJvdmlkZU5vb3BBbmltYXRpb25zKCksXG4gICAge3Byb3ZpZGU6IEFjdGl2YXRpb25JbnN0YW50UHJvdmlkZXIsIHVzZUNsYXNzOiBTZXF1ZW5jZUluc3RhbnRQcm92aWRlcn0sXG4gICAge3Byb3ZpZGU6IE1BSU5fQVJFQV9JTklUSUFMX1BBUlRfSUQsIHVzZVZhbHVlOiBjb25maWc/Lm1haW5BcmVhSW5pdGlhbFBhcnRJZCA/PyAnbWFpbid9LFxuICAgIHtwcm92aWRlOiBDb21wb25lbnRGaXh0dXJlQXV0b0RldGVjdCwgdXNlVmFsdWU6IHRydWV9LFxuICAgIHtwcm92aWRlOiBFTlZJUk9OTUVOVF9JTklUSUFMSVpFUiwgbXVsdGk6IHRydWUsIHVzZVZhbHVlOiAoKSA9PiBpbmplY3QoUm91dGVyKS5pbml0aWFsTmF2aWdhdGlvbigpfSxcbiAgICB7cHJvdmlkZTogRU5WSVJPTk1FTlRfSU5JVElBTElaRVIsIG11bHRpOiB0cnVlLCB1c2VWYWx1ZTogKCkgPT4gbG9jYWxTdG9yYWdlLmNsZWFyKCl9LFxuICAgIHtwcm92aWRlOiBFTlZJUk9OTUVOVF9JTklUSUFMSVpFUiwgbXVsdGk6IHRydWUsIHVzZVZhbHVlOiAoKSA9PiB3aW5kb3cubmFtZSA9ICcnfSxcbiAgXSk7XG59XG5cbkBJbmplY3RhYmxlKC8qIERPIE5PVCBQUk9WSURFIHZpYSAncHJvdmlkZWRJbicgbWV0YWRhdGEgYXMgcmVnaXN0ZXJlZCBvbmx5IGlmIHVzaW5nIGBXb3JrYmVuY2hUZXN0aW5nTW9kdWxlYC4gKi8pXG5jbGFzcyBTZXF1ZW5jZUluc3RhbnRQcm92aWRlciBpbXBsZW1lbnRzIEFjdGl2YXRpb25JbnN0YW50UHJvdmlkZXIge1xuXG4gIHByaXZhdGUgX3NlcXVlbmNlID0gMDtcblxuICBwdWJsaWMgbm93KCk6IG51bWJlciB7XG4gICAgcmV0dXJuIHRoaXMuX3NlcXVlbmNlKys7XG4gIH1cbn1cbiJdfQ==
|