@veloceapps/sdk 8.0.0-98 → 9.0.0-0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (184) hide show
  1. package/cms/README.md +0 -20
  2. package/cms/cms.actions.d.ts +13 -1
  3. package/cms/components/preview/preview.component.d.ts +2 -3
  4. package/cms/components/preview/preview.types.d.ts +4 -0
  5. package/cms/index.d.ts +1 -0
  6. package/cms/modules/runtime/services/runtime.service.d.ts +1 -1
  7. package/cms/services/index.d.ts +0 -1
  8. package/cms/services/resources.service.d.ts +2 -0
  9. package/cms/types/index.d.ts +0 -1
  10. package/cms/utils/element-metadata-worker.d.ts +11 -0
  11. package/cms/utils/elements-resolver.d.ts +0 -2
  12. package/cms/utils/index.d.ts +3 -0
  13. package/cms/utils/transpilation-worker.d.ts +13 -0
  14. package/cms/vendor-map.d.ts +23 -18
  15. package/core/README.md +20 -0
  16. package/core/modules/configuration/configuration.module.d.ts +2 -1
  17. package/core/modules/configuration/helpers.d.ts +2 -1
  18. package/core/modules/configuration/index.d.ts +1 -1
  19. package/core/modules/configuration/services/configuration-state.service.d.ts +54 -0
  20. package/core/modules/configuration/services/configuration.service.d.ts +6 -6
  21. package/core/modules/configuration/types/configuration.types.d.ts +4 -0
  22. package/core/modules/flow-configuration/flow-configuration.module.d.ts +2 -1
  23. package/core/modules/flow-configuration/services/flow-configuration.service.d.ts +5 -0
  24. package/core/services/context.service.d.ts +2 -0
  25. package/core/services/flow-info.service.d.ts +29 -0
  26. package/core/services/flow-state-configuration.service.d.ts +19 -0
  27. package/core/services/flow-state.service.d.ts +80 -0
  28. package/core/services/index.d.ts +4 -1
  29. package/core/services/quote-draft.service.d.ts +12 -28
  30. package/core/types/flow-customization.types.d.ts +22 -0
  31. package/core/types/flow-state.types.d.ts +11 -0
  32. package/core/types/index.d.ts +3 -1
  33. package/{cms → core}/types/integration.types.d.ts +1 -1
  34. package/core/utils/line-item.utils.d.ts +1 -3
  35. package/esm2020/cms/cms.actions.mjs +21 -1
  36. package/esm2020/cms/components/element-renderer/element-renderer.component.mjs +3 -3
  37. package/esm2020/cms/components/element-tools-panel/element-tools-panel.component.mjs +3 -3
  38. package/esm2020/cms/components/preview/preview.component.mjs +10 -10
  39. package/esm2020/cms/components/preview/preview.module.mjs +2 -2
  40. package/esm2020/cms/components/preview/preview.types.mjs +1 -1
  41. package/esm2020/cms/index.mjs +2 -1
  42. package/esm2020/cms/launcher.module.mjs +4 -4
  43. package/esm2020/cms/modules/migrations/migrations.mjs +2 -2
  44. package/esm2020/cms/modules/migrations/services/migrations.service.mjs +6 -2
  45. package/esm2020/cms/modules/runtime/services/compilation.service.mjs +11 -7
  46. package/esm2020/cms/modules/runtime/services/runtime.service.mjs +4 -4
  47. package/esm2020/cms/plugins/configuration.plugin.mjs +8 -4
  48. package/esm2020/cms/services/index.mjs +1 -2
  49. package/esm2020/cms/services/io-provider.service.mjs +10 -9
  50. package/esm2020/cms/services/resources.service.mjs +10 -6
  51. package/esm2020/cms/services/templates.service.mjs +10 -9
  52. package/esm2020/cms/types/index.mjs +1 -2
  53. package/esm2020/cms/utils/element-metadata-worker.mjs +31 -0
  54. package/esm2020/cms/utils/elements-resolver.mjs +13 -27
  55. package/esm2020/cms/utils/encoding.utils.mjs +3 -2
  56. package/esm2020/cms/utils/index.mjs +4 -1
  57. package/esm2020/cms/utils/path.utils.mjs +3 -3
  58. package/esm2020/cms/utils/script.utils.mjs +19 -15
  59. package/esm2020/cms/utils/transpilation-worker.mjs +52 -0
  60. package/esm2020/cms/vendor-map.mjs +12 -5
  61. package/esm2020/core/core.module.mjs +19 -3
  62. package/esm2020/core/modules/configuration/configuration.module.mjs +7 -23
  63. package/esm2020/core/modules/configuration/helpers.mjs +21 -1
  64. package/esm2020/core/modules/configuration/index.mjs +2 -2
  65. package/esm2020/core/modules/configuration/services/configuration-state.service.mjs +335 -0
  66. package/esm2020/core/modules/configuration/services/configuration.service.mjs +56 -50
  67. package/esm2020/core/modules/configuration/types/configuration.types.mjs +1 -1
  68. package/esm2020/core/modules/flow-configuration/flow-configuration.module.mjs +6 -6
  69. package/esm2020/core/modules/flow-configuration/services/flow-configuration.service.mjs +12 -9
  70. package/esm2020/core/services/context.service.mjs +17 -5
  71. package/esm2020/core/services/flow-info.service.mjs +99 -0
  72. package/esm2020/core/services/flow-state-configuration.service.mjs +53 -0
  73. package/esm2020/core/services/flow-state.service.mjs +512 -0
  74. package/esm2020/core/services/index.mjs +5 -2
  75. package/esm2020/core/services/integration.state.mjs +36 -0
  76. package/esm2020/core/services/product-images.service.mjs +4 -5
  77. package/esm2020/core/services/quote-draft.service.mjs +39 -91
  78. package/esm2020/core/services/runtime-settings.service.mjs +7 -5
  79. package/esm2020/core/types/flow-customization.types.mjs +3 -0
  80. package/esm2020/core/types/flow-state.types.mjs +2 -0
  81. package/esm2020/core/types/index.mjs +4 -2
  82. package/esm2020/core/types/integration.types.mjs +2 -0
  83. package/esm2020/core/utils/line-item.utils.mjs +33 -32
  84. package/esm2020/src/components/doc-gen/doc-gen.component.mjs +52 -55
  85. package/esm2020/src/components/flow-header/flow-header.component.mjs +108 -0
  86. package/esm2020/src/components/flow-header/flow-header.module.mjs +19 -0
  87. package/esm2020/src/components/flow-header/index.mjs +2 -0
  88. package/esm2020/src/components/guided-selling/guided-selling.component.mjs +40 -48
  89. package/esm2020/src/configure-primeng.mjs +33 -0
  90. package/esm2020/src/flow-routing.module.mjs +5 -18
  91. package/esm2020/src/flow.component.mjs +22 -37
  92. package/esm2020/src/flow.module.mjs +7 -5
  93. package/esm2020/src/guards/context.guard.mjs +6 -11
  94. package/esm2020/src/guards/flow.guard.mjs +23 -9
  95. package/esm2020/src/pages/assets/assets.component.mjs +30 -34
  96. package/esm2020/src/pages/catalog/catalog.component.mjs +30 -34
  97. package/esm2020/src/pages/debug/debug.component.mjs +2 -2
  98. package/esm2020/src/pages/product/product.component.mjs +34 -20
  99. package/esm2020/src/pages/record-not-found/record-not-found.component.mjs +6 -3
  100. package/esm2020/src/pages/remote/remote.component.mjs +15 -14
  101. package/esm2020/src/pages/shopping-cart/shopping-cart.component.mjs +31 -35
  102. package/esm2020/src/resolvers/flow.resolver.mjs +37 -55
  103. package/esm2020/src/resolvers/quote.resolver.mjs +55 -67
  104. package/esm2020/src/services/doc-gen.service.mjs +4 -3
  105. package/esm2020/src/services/flow-dialog.service.mjs +21 -6
  106. package/esm2020/src/services/flow-router.service.mjs +56 -15
  107. package/esm2020/src/services/flow.service.mjs +76 -48
  108. package/esm2020/src/services/guided-selling.service.mjs +5 -4
  109. package/esm2020/src/services/index.mjs +3 -1
  110. package/esm2020/src/types/index.mjs +1 -2
  111. package/fesm2015/veloceapps-sdk-cms.mjs +286 -192
  112. package/fesm2015/veloceapps-sdk-cms.mjs.map +1 -1
  113. package/fesm2015/veloceapps-sdk-core.mjs +1195 -427
  114. package/fesm2015/veloceapps-sdk-core.mjs.map +1 -1
  115. package/fesm2015/veloceapps-sdk.mjs +801 -1455
  116. package/fesm2015/veloceapps-sdk.mjs.map +1 -1
  117. package/fesm2020/veloceapps-sdk-cms.mjs +291 -207
  118. package/fesm2020/veloceapps-sdk-cms.mjs.map +1 -1
  119. package/fesm2020/veloceapps-sdk-core.mjs +1176 -454
  120. package/fesm2020/veloceapps-sdk-core.mjs.map +1 -1
  121. package/fesm2020/veloceapps-sdk.mjs +770 -1420
  122. package/fesm2020/veloceapps-sdk.mjs.map +1 -1
  123. package/package.json +8 -7
  124. package/src/components/doc-gen/doc-gen.component.d.ts +7 -10
  125. package/src/components/flow-header/flow-header.component.d.ts +25 -0
  126. package/src/components/flow-header/flow-header.module.d.ts +9 -0
  127. package/src/components/flow-header/index.d.ts +1 -0
  128. package/src/components/guided-selling/guided-selling.component.d.ts +6 -9
  129. package/src/configure-primeng.d.ts +1 -0
  130. package/src/flow-routing.module.d.ts +1 -2
  131. package/src/flow.component.d.ts +7 -12
  132. package/src/flow.module.d.ts +2 -2
  133. package/src/guards/context.guard.d.ts +0 -1
  134. package/src/guards/flow.guard.d.ts +1 -1
  135. package/src/pages/assets/assets.component.d.ts +5 -7
  136. package/src/pages/catalog/catalog.component.d.ts +5 -7
  137. package/src/pages/product/product.component.d.ts +8 -7
  138. package/src/pages/record-not-found/record-not-found.component.d.ts +2 -0
  139. package/src/pages/remote/remote.component.d.ts +2 -3
  140. package/src/pages/shopping-cart/shopping-cart.component.d.ts +5 -7
  141. package/src/resolvers/flow.resolver.d.ts +6 -9
  142. package/src/resolvers/quote.resolver.d.ts +7 -14
  143. package/src/services/doc-gen.service.d.ts +1 -1
  144. package/src/services/flow-dialog.service.d.ts +5 -2
  145. package/src/services/flow-router.service.d.ts +7 -3
  146. package/src/services/flow.service.d.ts +10 -12
  147. package/src/services/guided-selling.service.d.ts +1 -1
  148. package/src/services/index.d.ts +2 -0
  149. package/src/types/index.d.ts +0 -1
  150. package/core/modules/configuration/services/configuration.state.d.ts +0 -30
  151. package/core/services/metric-calculation/metric-calculation.service.d.ts +0 -25
  152. package/core/services/metric-calculation/metric-calculation.types.d.ts +0 -1
  153. package/core/services/metric-calculation/metric-calculation.utils.d.ts +0 -5
  154. package/esm2020/cms/services/integration.state.mjs +0 -37
  155. package/esm2020/cms/types/integration.types.mjs +0 -2
  156. package/esm2020/core/modules/configuration/services/configuration.state.mjs +0 -142
  157. package/esm2020/core/services/metric-calculation/metric-calculation.service.mjs +0 -85
  158. package/esm2020/core/services/metric-calculation/metric-calculation.types.mjs +0 -2
  159. package/esm2020/core/services/metric-calculation/metric-calculation.utils.mjs +0 -42
  160. package/esm2020/src/components/header/cart-overlay/cart-preview.component.mjs +0 -119
  161. package/esm2020/src/components/header/cart-overlay/cart-preview.module.mjs +0 -47
  162. package/esm2020/src/components/header/header.component.mjs +0 -355
  163. package/esm2020/src/components/header/header.module.mjs +0 -52
  164. package/esm2020/src/components/header/header.types.mjs +0 -2
  165. package/esm2020/src/components/header/metrics/index.mjs +0 -2
  166. package/esm2020/src/components/header/metrics/metrics.component.mjs +0 -255
  167. package/esm2020/src/components/header/metrics/metrics.definitions.mjs +0 -2
  168. package/esm2020/src/components/header/metrics/metrics.module.mjs +0 -69
  169. package/esm2020/src/pages/empty-account/empty-account.component.mjs +0 -12
  170. package/esm2020/src/pages/empty-account/empty-account.module.mjs +0 -20
  171. package/esm2020/src/types/flow-customization.types.mjs +0 -3
  172. package/src/components/header/cart-overlay/cart-preview.component.d.ts +0 -36
  173. package/src/components/header/cart-overlay/cart-preview.module.d.ts +0 -14
  174. package/src/components/header/header.component.d.ts +0 -70
  175. package/src/components/header/header.module.d.ts +0 -16
  176. package/src/components/header/header.types.d.ts +0 -20
  177. package/src/components/header/metrics/index.d.ts +0 -1
  178. package/src/components/header/metrics/metrics.component.d.ts +0 -67
  179. package/src/components/header/metrics/metrics.definitions.d.ts +0 -1
  180. package/src/components/header/metrics/metrics.module.d.ts +0 -18
  181. package/src/pages/empty-account/empty-account.component.d.ts +0 -5
  182. package/src/pages/empty-account/empty-account.module.d.ts +0 -10
  183. package/src/types/flow-customization.types.d.ts +0 -12
  184. /package/{cms → core}/services/integration.state.d.ts +0 -0
@@ -1,49 +1,33 @@
1
1
  import * as i4 from '@angular/common';
2
2
  import { CommonModule } from '@angular/common';
3
3
  import * as i0 from '@angular/core';
4
- import { Component, ChangeDetectionStrategy, NgModule, Injectable, InjectionToken, Optional, Inject, ViewChild, Input, inject } from '@angular/core';
5
- import * as i2 from '@veloceapps/api';
4
+ import { Component, ChangeDetectionStrategy, NgModule, Optional, Inject, Injectable, inject } from '@angular/core';
5
+ import * as i1$1 from '@veloceapps/api';
6
6
  import { ApiModule } from '@veloceapps/api';
7
7
  import * as i2$1 from '@veloceapps/components';
8
- import { ToastType, LetDirectiveModule, QuantityControlModule, ErrorTooltipModule, HiddenTextTooltipModule, LoaderModule, EmptyStateModule } from '@veloceapps/components';
8
+ import { ToastType, LoaderModule, LetDirectiveModule } from '@veloceapps/components';
9
9
  import * as i5 from '@veloceapps/sdk/cms';
10
- import { FlowAction, extractElementMetadata, extendElementMetadata, PreviewModule, LauncherModule } from '@veloceapps/sdk/cms';
11
- import * as i1$2 from '@veloceapps/sdk/core';
12
- import { isLineItemModified, getOriginParent, SdkPipesModule, ContextService, QuoteDraftService, ConfigurationService, generateModifiedAssetsMap, SdkCoreModule } from '@veloceapps/sdk/core';
10
+ import { extractElementMetadata, extendElementMetadata, btoaSafe, PreviewModule, FlowAction, LauncherModule } from '@veloceapps/sdk/cms';
11
+ import * as i2 from '@veloceapps/sdk/core';
12
+ import { FLOW_CUSTOMIZATION, ContextService, FlowStateService, FlowInfoService, QuoteDraftService, ConfigurationService, IntegrationState, SdkCoreModule } from '@veloceapps/sdk/core';
13
13
  import * as i3 from 'primeng/button';
14
14
  import { ButtonModule } from 'primeng/button';
15
15
  import * as i1 from 'primeng/dynamicdialog';
16
- import { SalesforceIdUtils, UITemplateComponentType, UITemplateType, ConfigurationContextMode, UUID } from '@veloceapps/core';
17
- import { map, filter, shareReplay, startWith, distinctUntilChanged, BehaviorSubject, Subject, tap, takeUntil, switchMap, of, first, catchError, combineLatest, finalize, noop, forkJoin, from, throwError } from 'rxjs';
18
- import * as i1$1 from '@angular/router';
16
+ import { BehaviorSubject, Subject, filter, first, tap, takeUntil, catchError, of, map, switchMap, shareReplay, startWith, distinctUntilChanged, from, take, combineLatest, forkJoin, throwError } from 'rxjs';
17
+ import * as i1$2 from '@angular/router';
19
18
  import { NavigationEnd, NavigationStart, NavigationCancel, NavigationError, RouterModule } from '@angular/router';
20
- import * as i7 from 'primeng/overlaypanel';
21
- import { OverlayPanel, OverlayPanelModule } from 'primeng/overlaypanel';
22
- import * as i13 from 'primeng/splitbutton';
23
- import { SplitButtonModule } from 'primeng/splitbutton';
24
- import * as i10$1 from 'primeng/tooltip';
25
- import { TooltipModule } from 'primeng/tooltip';
26
- import * as i5$1 from '@angular/forms';
27
- import { FormGroup, FormControl, ReactiveFormsModule, Validators, FormsModule } from '@angular/forms';
28
- import { InputNumberModule } from 'primeng/inputnumber';
29
- import * as i9 from 'primeng/virtualscroller';
30
- import { VirtualScrollerModule } from 'primeng/virtualscroller';
31
- import * as i8 from 'primeng/api';
32
- import * as i5$2 from '@angular/cdk/drag-drop';
33
- import { moveItemInArray, DragDropModule } from '@angular/cdk/drag-drop';
34
- import { requiredValidator, invalidCharactersValidator, standardCharSetWithSpaceRegExp, reservedIdentifierValidator } from '@veloceapps/core/forms';
35
- import { cloneDeep, uniq } from 'lodash';
36
- import { withLatestFrom, tap as tap$1, takeUntil as takeUntil$1, map as map$1 } from 'rxjs/operators';
37
- import * as i9$1 from 'primeng/sidebar';
38
- import { SidebarModule } from 'primeng/sidebar';
39
- import * as i10 from 'primeng/inputtext';
40
- import { InputTextModule } from 'primeng/inputtext';
41
- import * as i11 from 'primeng/checkbox';
42
- import { CheckboxModule } from 'primeng/checkbox';
19
+ import { mapShoppingCartSettings, getMaxRenewalTermsValue, UITemplateType, SalesforceIdUtils, ConfigurationContextMode, isVeloceError, extractErrorDetails, UUID } from '@veloceapps/core';
43
20
  import { HttpErrorResponse, HttpParams } from '@angular/common/http';
21
+ import * as i5$1 from '@angular/forms';
22
+ import { FormGroup, FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';
44
23
  import { DropdownModule } from 'primeng/dropdown';
24
+ import * as i8 from 'primeng/inputtext';
25
+ import { InputTextModule } from 'primeng/inputtext';
45
26
  import * as i6 from 'primeng/radiobutton';
46
27
  import { RadioButtonModule } from 'primeng/radiobutton';
28
+ import * as i2$2 from 'primeng/api';
29
+ import { catchError as catchError$1 } from 'rxjs/operators';
30
+ import { DomHandler } from 'primeng/dom';
47
31
 
48
32
  const VELOCE_FLOW_ROOT_ROUTE = 'VELOCE_FLOW_ROOT_ROUTE';
49
33
 
@@ -91,262 +75,21 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
91
75
  }]
92
76
  }] });
93
77
 
94
- class FlowRouterService {
95
- constructor(router, route, contextService, integrationState) {
96
- this.router = router;
97
- this.route = route;
98
- this.contextService = contextService;
99
- this.integrationState = integrationState;
100
- this.urlHistory = [];
101
- this.getLastChildRoute = (route) => {
102
- return route.firstChild ? this.getLastChildRoute(route.firstChild) : route;
103
- };
104
- this.getNthChildRoute = (route, index) => {
105
- if (index <= 0) {
106
- return route;
107
- }
108
- return route.firstChild ? this.getNthChildRoute(route.firstChild, index - 1) : route;
109
- };
110
- this.getLastChildRouteSnapshot = (route) => {
111
- return route.firstChild ? this.getLastChildRouteSnapshot(route.firstChild) : route;
112
- };
113
- this.watchLastChildRoute$ = (route) => {
114
- return this.routeChange$.pipe(map(() => this.getLastChildRouteSnapshot(route.snapshot)));
115
- };
116
- this.getLastChildParams = (route) => {
117
- return route.firstChild ? this.getLastChildParams(route.firstChild) : route.params;
118
- };
119
- this.watchLastChildParams$ = (route) => {
120
- return this.routeChange$.pipe(map(() => this.getLastChildParams(route.snapshot)));
121
- };
122
- this.getFlowSubpath$ = () => {
123
- return this.route$.pipe(map(route => {
124
- const url = route.url.join('/');
125
- const flowRootUrl = this.getFlowRootPath(route);
126
- return url.replace(flowRootUrl, '');
127
- }));
128
- };
129
- this.routeChange$ = this.router.events.pipe(filter(e => e instanceof NavigationEnd), shareReplay());
130
- this.lastChildParams$ = this.watchLastChildParams$(this.route).pipe(startWith(this.getLastChildParams(this.route.snapshot)), shareReplay());
131
- this.lastChildRoute$ = this.watchLastChildRoute$(this.route).pipe(startWith(this.getLastChildRouteSnapshot(this.route.snapshot)), shareReplay());
132
- this.loading$ = this.router.events.pipe(filter(e => e instanceof NavigationStart ||
133
- e instanceof NavigationCancel ||
134
- e instanceof NavigationEnd ||
135
- e instanceof NavigationError), map(e => e instanceof NavigationStart), startWith(false), distinctUntilChanged());
136
- this.routeChange$.subscribe(() => {
137
- this.urlHistory.push(this.router.url);
138
- // Close Guided Selling when user navigates to another flow page
139
- this.integrationState.dispatch(FlowAction.CloseGuidedSelling({ keepState: true }));
140
- });
141
- }
142
- getFlowRootRoute(route) {
143
- const path = [...route.pathFromRoot];
144
- while (path.length) {
145
- const parent = path.pop();
146
- if (!parent) {
147
- break;
148
- }
149
- if (parent.routeConfig?.id === VELOCE_FLOW_ROOT_ROUTE) {
150
- return parent;
151
- }
152
- }
153
- return;
154
- }
155
- getFlowRootPath(route) {
156
- const rootRoute = this.getFlowRootRoute(route);
157
- if (!rootRoute) {
158
- return '';
159
- }
160
- const path = rootRoute.pathFromRoot
161
- .map(r => r.url[0])
162
- .filter(Boolean)
163
- .join('/');
164
- return '/' + path;
165
- }
166
- get route$() {
167
- return this.lastChildRoute$;
168
- }
169
- get params$() {
170
- return this.lastChildParams$;
171
- }
172
- isConfigurationRoute$() {
173
- return this.getFlowSubpath$().pipe(map(url => url.startsWith('product')));
174
- }
175
- isCartRoute$() {
176
- return this.getFlowSubpath$().pipe(map(url => url.startsWith('cart')));
177
- }
178
- isCatalogRoute$() {
179
- return this.getFlowSubpath$().pipe(map(url => url.startsWith('catalog')));
180
- }
181
- isAssetsRoute$() {
182
- return this.getFlowSubpath$().pipe(map(url => url.startsWith('assets')));
183
- }
184
- navigateBack() {
185
- const prevUrl = this.urlHistory[this.urlHistory.length - 2];
186
- if (prevUrl) {
187
- this.router.navigateByUrl(prevUrl);
188
- }
189
- }
190
- navigateToProductConfiguration(productId, lineItemId) {
191
- this.contextService.update({ properties: { productId, lineItemId } });
192
- const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
193
- const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
194
- this.router.navigate([flowRouteUrl, 'product'], {
195
- queryParams: { ...routeSnapshot.queryParams, productId },
196
- });
197
- }
198
- navigateToShoppingCart() {
199
- const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
200
- const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
201
- this.router.navigate([flowRouteUrl, 'cart'], { queryParams: routeSnapshot.queryParams });
202
- }
203
- navigateToCatalog() {
204
- const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
205
- const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
206
- this.router.navigate([flowRouteUrl, 'catalog'], { queryParams: routeSnapshot.queryParams });
207
- }
208
- navigateToAssets() {
209
- const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
210
- const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
211
- this.router.navigate([flowRouteUrl, 'assets'], { queryParams: routeSnapshot.queryParams });
212
- }
213
- switchObject(id) {
214
- const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
215
- const route = this.getLastChildRoute(this.route);
216
- const objName = SalesforceIdUtils.getSfObjectNameById(id);
217
- const queryParams = {
218
- quoteId: objName === 'Quote' ? id : undefined,
219
- accountId: objName === 'Account' ? id : undefined,
220
- orderId: objName === 'Order' ? id : undefined,
221
- };
222
- this.router.navigate([], { relativeTo: route, queryParams: { ...routeSnapshot.queryParams, ...queryParams } });
223
- }
224
- }
225
- FlowRouterService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, deps: [{ token: i1$1.Router }, { token: i1$1.ActivatedRoute }, { token: i1$2.ContextService }, { token: i5.IntegrationState }], target: i0.ɵɵFactoryTarget.Injectable });
226
- FlowRouterService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, providedIn: 'root' });
227
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, decorators: [{
228
- type: Injectable,
229
- args: [{ providedIn: 'root' }]
230
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: i1$1.ActivatedRoute }, { type: i1$2.ContextService }, { type: i5.IntegrationState }]; } });
231
-
232
- class FlowService {
233
- get flow() {
234
- return this.flowSubj$.value;
235
- }
236
- constructor(integrationState, flowRouterService, quoteDraftService, configurationService, flowConfigurationService, flowsApiService) {
237
- this.integrationState = integrationState;
238
- this.flowRouterService = flowRouterService;
239
- this.quoteDraftService = quoteDraftService;
240
- this.configurationService = configurationService;
241
- this.flowConfigurationService = flowConfigurationService;
242
- this.flowsApiService = flowsApiService;
243
- this.flowSubj$ = new BehaviorSubject(null);
244
- this.flow$ = this.flowSubj$.asObservable();
245
- this.cleanup$ = new Subject();
246
- }
247
- cleanup() {
248
- this.cleanup$.next();
249
- this.flowSubj$.next(null);
250
- }
251
- setFlow(flow) {
252
- this.flowSubj$.next(flow ?? null);
253
- }
254
- setFlowById$(flowId) {
255
- return this.flowsApiService.getFlow(flowId).pipe(tap(flow => this.flowSubj$.next(flow ?? null)), map(flow => Boolean(flow)));
256
- }
257
- initSubscriptions() {
258
- this.integrationState
259
- .listen$(FlowAction.FLOW_CONFIGURE_PRODUCT)
260
- .pipe(tap(payload => {
261
- const productId = payload.productId ??
262
- this.quoteDraftService.currentState.find(li => li.id === payload.lineItemId)?.productId;
263
- if (productId) {
264
- this.flowRouterService.navigateToProductConfiguration(productId, payload.lineItemId);
265
- }
266
- else {
267
- console.warn("Parameter 'productId' is needed to start configuration");
268
- }
269
- }), takeUntil(this.cleanup$))
270
- .subscribe();
271
- this.integrationState
272
- .listen$(FlowAction.FLOW_SWITCH_OBJECT)
273
- .pipe(tap(payload => {
274
- this.flowRouterService.switchObject(payload.id);
275
- }), takeUntil(this.cleanup$))
276
- .subscribe();
277
- this.integrationState
278
- .listen$(FlowAction.FLOW_NAVIGATE_BACK)
279
- .pipe(tap(() => this.flowRouterService.navigateBack()), takeUntil(this.cleanup$))
280
- .subscribe();
281
- this.integrationState
282
- .listen$(FlowAction.FLOW_NAVIGATE_TO_CATALOG)
283
- .pipe(tap(() => this.flowRouterService.navigateToCatalog()), takeUntil(this.cleanup$))
284
- .subscribe();
285
- this.integrationState
286
- .listen$(FlowAction.FLOW_NAVIGATE_TO_SHOPPING_CART)
287
- .pipe(tap(() => this.flowRouterService.navigateToShoppingCart()), takeUntil(this.cleanup$))
288
- .subscribe();
289
- this.integrationState
290
- .listen$(FlowAction.FLOW_APPLY_PRODUCT_CONFIGURATION)
291
- .pipe(switchMap(() => {
292
- const quoteDraft = this.quoteDraftService.quoteDraft;
293
- const lineItem = this.configurationService.getSnapshot();
294
- if (!quoteDraft || !lineItem) {
295
- return of(undefined);
296
- }
297
- const isNewLineItem = quoteDraft.currentState.every(li => li.id !== lineItem.id);
298
- const assetId = lineItem.assetId || lineItem.openOrderLineItemId;
299
- let updatedState;
300
- if (isNewLineItem) {
301
- updatedState = [...quoteDraft.currentState, lineItem];
302
- }
303
- else {
304
- updatedState = quoteDraft.currentState.map(li => (li.id === lineItem.id ? lineItem : li));
305
- }
306
- return this.flowConfigurationService.calculate$({ ...quoteDraft, currentState: updatedState }).pipe(tap(() => {
307
- if (assetId) {
308
- const modifiedAssets = this.integrationState.state.modifiedAssets ?? {};
309
- this.integrationState.patchState({
310
- modifiedAssets: { ...modifiedAssets, [assetId]: true },
311
- });
312
- }
313
- }));
314
- }), tap(() => {
315
- this.configurationService.hasUnsavedChanges = false;
316
- this.flowRouterService.navigateToShoppingCart();
317
- }), takeUntil(this.cleanup$))
318
- .subscribe();
319
- this.updateFlowPath();
320
- }
321
- updateFlowPath() {
322
- this.flowRouterService
323
- .getFlowSubpath$()
324
- .pipe(map(path => path.split('/')?.[0]), takeUntil(this.cleanup$))
325
- .subscribe(flowPath => this.integrationState.patchState({ flowPath }));
326
- }
327
- }
328
- FlowService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService, deps: [{ token: i5.IntegrationState }, { token: FlowRouterService }, { token: i1$2.QuoteDraftService }, { token: i1$2.ConfigurationService }, { token: i1$2.FlowConfigurationService }, { token: i2.FlowsApiService }], target: i0.ɵɵFactoryTarget.Injectable });
329
- FlowService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService });
330
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService, decorators: [{
331
- type: Injectable
332
- }], ctorParameters: function () { return [{ type: i5.IntegrationState }, { type: FlowRouterService }, { type: i1$2.QuoteDraftService }, { type: i1$2.ConfigurationService }, { type: i1$2.FlowConfigurationService }, { type: i2.FlowsApiService }]; } });
333
-
334
- const FLOW_CUSTOMIZATION = new InjectionToken('FLOW_CUSTOMIZATION');
335
-
336
78
  class DocGenComponent {
337
- constructor(quoteDraftService, contextService, templatesApi, toastService, flowService, customizationService) {
338
- this.quoteDraftService = quoteDraftService;
339
- this.contextService = contextService;
79
+ constructor(templatesApi, contextService, flowStateService, flowInfo, toastService, customizationService) {
340
80
  this.templatesApi = templatesApi;
81
+ this.contextService = contextService;
82
+ this.flowStateService = flowStateService;
83
+ this.flowInfo = flowInfo;
341
84
  this.toastService = toastService;
342
- this.flowService = flowService;
343
85
  this.customizationService = customizationService;
344
86
  this.uiDefinition$ = new BehaviorSubject(null);
345
87
  this.templateApiName = '';
346
88
  this.destroy$ = new Subject();
347
- // initialize when quote draft requested
348
- this.quoteDraftService.quoteDraft$
349
- .pipe(first(), tap(() => this.initialize()), takeUntil(this.destroy$))
89
+ // wait until flow is initialized
90
+ this.flowStateService
91
+ .isInitialized$()
92
+ .pipe(filter(Boolean), first(), tap(() => this.initialize()), takeUntil(this.destroy$))
350
93
  .subscribe();
351
94
  }
352
95
  ngOnDestroy() {
@@ -354,7 +97,7 @@ class DocGenComponent {
354
97
  this.destroy$.complete();
355
98
  }
356
99
  initialize() {
357
- this.templateApiName = this.flowService.flow?.properties.templates.docGen ?? '';
100
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.docGen ?? '';
358
101
  this.generateUIDefinition$()
359
102
  .pipe(tap(uiDef => {
360
103
  if (!uiDef) {
@@ -368,78 +111,78 @@ class DocGenComponent {
368
111
  }), takeUntil(this.destroy$))
369
112
  .subscribe();
370
113
  }
371
- getTemplateRootComponent$(template) {
372
- return this.templatesApi
373
- .fetchComponents$(template.id)
374
- .pipe(map(components => components.find(c => c.type === UITemplateComponentType.ROOT) ?? undefined));
375
- }
376
- getLocalTemplateComponentMeta$() {
377
- if (!this.customizationService?.getTemplateComponent) {
114
+ getLocalMeta$() {
115
+ if (!this.customizationService?.getTemplateComponents) {
378
116
  return of(undefined);
379
117
  }
380
- return this.customizationService?.getTemplateComponent(this.templateApiName).pipe(map(component => {
381
- if (!component) {
118
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
119
+ if (!components) {
382
120
  return;
383
121
  }
384
- return {
122
+ return components.map(component => ({
385
123
  html: component.html,
386
124
  css: component.css,
387
125
  js: component.js,
388
126
  json: component.json,
389
- };
127
+ }));
390
128
  }));
391
129
  }
392
- getDocGenComponentMeta$() {
393
- return this.templatesApi.fetchTemplates$().pipe(map(templates => {
394
- const template = templates.find(template => template.type === UITemplateType.DOCGEN && template.name === this.templateApiName);
395
- return template ?? templates[0];
396
- }), switchMap(template => (template ? this.getTemplateRootComponent$(template) : of(undefined))), switchMap(component => component ? this.templatesApi.fetchComponentAttachments$(component.uiTemplateId, component) : of(undefined)));
130
+ getOrgMeta$() {
131
+ const template = this.flowInfo.templates.DOCGEN;
132
+ if (!template) {
133
+ return of(undefined);
134
+ }
135
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
397
136
  }
398
137
  generateUIDefinition$() {
399
138
  return of(undefined).pipe(tap(() => {
400
139
  if (!this.templateApiName) {
401
- throw new Error("Flow Query parameter 'docGenTemplateApiName' is missing.");
140
+ throw new Error("Flow 'docGen' template is not defined.");
402
141
  }
403
- }), switchMap(() => this.getLocalTemplateComponentMeta$()), switchMap(meta => (meta ? of(meta) : this.getDocGenComponentMeta$())), map(meta => {
404
- if (!meta) {
142
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
143
+ if (!metaList) {
405
144
  return;
406
145
  }
407
146
  const headerId = this.contextService.resolve().properties.Id ?? null;
408
- let script = meta.js;
409
- if (script) {
410
- const metadata = extractElementMetadata(script);
411
- script = extendElementMetadata(script, {
147
+ const patchedMetaList = metaList.map(component => {
148
+ if (!component.js) {
149
+ return component;
150
+ }
151
+ const metadata = extractElementMetadata(component.js);
152
+ const script = extendElementMetadata(component.js, {
412
153
  inputs: {
413
154
  ...metadata.inputs,
414
155
  Id: headerId ? `"${headerId}"` : null,
415
156
  },
416
157
  });
417
- }
158
+ return {
159
+ ...component,
160
+ js: script,
161
+ };
162
+ });
418
163
  const uiDef = {
419
164
  name: '',
420
165
  createdTimestamp: 0,
421
166
  primary: true,
422
167
  type: 'DEFAULT',
423
168
  version: 2,
424
- children: [
425
- {
426
- children: [],
427
- template: meta.html && btoa(meta.html),
428
- script: script && btoa(script),
429
- styles: meta.css && btoa(meta.css),
430
- },
431
- ],
169
+ children: patchedMetaList.map(child => ({
170
+ children: [],
171
+ template: child.html && btoaSafe(child.html),
172
+ script: child.js && btoaSafe(child.js),
173
+ styles: child.css && btoaSafe(child.css),
174
+ })),
432
175
  };
433
176
  return uiDef;
434
177
  }));
435
178
  }
436
179
  }
437
- DocGenComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DocGenComponent, deps: [{ token: i1$2.QuoteDraftService }, { token: i1$2.ContextService }, { token: i2.UITemplatesApiService }, { token: i2$1.ToastService }, { token: FlowService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
180
+ DocGenComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DocGenComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i2.ContextService }, { token: i2.FlowStateService }, { token: i2.FlowInfoService }, { token: i2$1.ToastService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
438
181
  DocGenComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DocGenComponent, selector: "vl-flow-doc-gen", ngImport: i0, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
439
182
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DocGenComponent, decorators: [{
440
183
  type: Component,
441
184
  args: [{ selector: 'vl-flow-doc-gen', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"] }]
442
- }], ctorParameters: function () { return [{ type: i1$2.QuoteDraftService }, { type: i1$2.ContextService }, { type: i2.UITemplatesApiService }, { type: i2$1.ToastService }, { type: FlowService }, { type: undefined, decorators: [{
185
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i2.ContextService }, { type: i2.FlowStateService }, { type: i2.FlowInfoService }, { type: i2$1.ToastService }, { type: undefined, decorators: [{
443
186
  type: Optional
444
187
  }, {
445
188
  type: Inject,
@@ -461,18 +204,19 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
461
204
  }] });
462
205
 
463
206
  class GuidedSellingComponent {
464
- constructor(quoteDraftService, templatesApi, toastService, flowService, customizationService) {
465
- this.quoteDraftService = quoteDraftService;
207
+ constructor(templatesApi, flowStateService, flowInfo, toastService, customizationService) {
466
208
  this.templatesApi = templatesApi;
209
+ this.flowStateService = flowStateService;
210
+ this.flowInfo = flowInfo;
467
211
  this.toastService = toastService;
468
- this.flowService = flowService;
469
212
  this.customizationService = customizationService;
470
213
  this.uiDefinition$ = new BehaviorSubject(null);
471
214
  this.templateApiName = '';
472
215
  this.destroy$ = new Subject();
473
- // initialize when quote draft requested
474
- this.quoteDraftService.quoteDraft$
475
- .pipe(first(), tap(() => this.initialize()), takeUntil(this.destroy$))
216
+ // wait until flow is initialized
217
+ this.flowStateService
218
+ .isInitialized$()
219
+ .pipe(filter(Boolean), first(), tap(() => this.initialize()), takeUntil(this.destroy$))
476
220
  .subscribe();
477
221
  }
478
222
  ngOnDestroy() {
@@ -480,7 +224,7 @@ class GuidedSellingComponent {
480
224
  this.destroy$.complete();
481
225
  }
482
226
  initialize() {
483
- this.templateApiName = this.flowService.flow?.properties.templates.guidedSelling ?? '';
227
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.guidedSelling ?? '';
484
228
  this.generateUIDefinition$()
485
229
  .pipe(tap(uiDef => {
486
230
  if (!uiDef) {
@@ -494,40 +238,36 @@ class GuidedSellingComponent {
494
238
  }), takeUntil(this.destroy$))
495
239
  .subscribe();
496
240
  }
497
- getTemplateRootComponent$(template) {
498
- return this.templatesApi
499
- .fetchComponents$(template.id)
500
- .pipe(map(components => components.find(c => c.type === UITemplateComponentType.ROOT) ?? undefined));
501
- }
502
- getLocalTemplateComponentMeta$() {
503
- if (!this.customizationService?.getTemplateComponent) {
241
+ getLocalMeta$() {
242
+ if (!this.customizationService?.getTemplateComponents) {
504
243
  return of(undefined);
505
244
  }
506
- return this.customizationService?.getTemplateComponent(this.templateApiName).pipe(map(component => {
507
- if (!component) {
245
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
246
+ if (!components) {
508
247
  return;
509
248
  }
510
- return {
249
+ return components.map(component => ({
511
250
  html: component.html,
512
251
  css: component.css,
513
252
  js: component.js,
514
253
  json: component.json,
515
- };
254
+ }));
516
255
  }));
517
256
  }
518
- getGuidedSellingComponentMeta$() {
519
- return this.templatesApi.fetchTemplates$().pipe(map(templates => {
520
- const template = templates.find(template => template.type === UITemplateType.GUIDED_SELLING && template.name === this.templateApiName);
521
- return template ?? templates[0];
522
- }), switchMap(template => (template ? this.getTemplateRootComponent$(template) : of(undefined))), switchMap(component => component ? this.templatesApi.fetchComponentAttachments$(component.uiTemplateId, component) : of(undefined)));
257
+ getOrgMeta$() {
258
+ const template = this.flowInfo.templates.GUIDED_SELLING;
259
+ if (!template) {
260
+ return of(undefined);
261
+ }
262
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
523
263
  }
524
264
  generateUIDefinition$() {
525
265
  return of(undefined).pipe(tap(() => {
526
266
  if (!this.templateApiName) {
527
- throw new Error("Flow Query parameter 'guidedSellingTemplateApiName' is missing.");
267
+ throw new Error("Flow 'guidedSelling' template is not defined.");
528
268
  }
529
- }), switchMap(() => this.getLocalTemplateComponentMeta$()), switchMap(meta => (meta ? of(meta) : this.getGuidedSellingComponentMeta$())), map(meta => {
530
- if (!meta) {
269
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
270
+ if (!metaList) {
531
271
  return;
532
272
  }
533
273
  const uiDef = {
@@ -536,25 +276,23 @@ class GuidedSellingComponent {
536
276
  primary: true,
537
277
  type: 'DEFAULT',
538
278
  version: 2,
539
- children: [
540
- {
541
- children: [],
542
- template: meta.html && btoa(meta.html),
543
- script: meta.js && btoa(meta.js),
544
- styles: meta.css && btoa(meta.css),
545
- },
546
- ],
279
+ children: metaList.map(meta => ({
280
+ children: [],
281
+ template: meta.html && btoaSafe(meta.html),
282
+ script: meta.js && btoaSafe(meta.js),
283
+ styles: meta.css && btoaSafe(meta.css),
284
+ })),
547
285
  };
548
286
  return uiDef;
549
287
  }));
550
288
  }
551
289
  }
552
- GuidedSellingComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: GuidedSellingComponent, deps: [{ token: i1$2.QuoteDraftService }, { token: i2.UITemplatesApiService }, { token: i2$1.ToastService }, { token: FlowService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
290
+ GuidedSellingComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: GuidedSellingComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i2.FlowStateService }, { token: i2.FlowInfoService }, { token: i2$1.ToastService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
553
291
  GuidedSellingComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: GuidedSellingComponent, selector: "vl-flow-guided-selling", ngImport: i0, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
554
292
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: GuidedSellingComponent, decorators: [{
555
293
  type: Component,
556
294
  args: [{ selector: 'vl-flow-guided-selling', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"] }]
557
- }], ctorParameters: function () { return [{ type: i1$2.QuoteDraftService }, { type: i2.UITemplatesApiService }, { type: i2$1.ToastService }, { type: FlowService }, { type: undefined, decorators: [{
295
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i2.FlowStateService }, { type: i2.FlowInfoService }, { type: i2$1.ToastService }, { type: undefined, decorators: [{
558
296
  type: Optional
559
297
  }, {
560
298
  type: Inject,
@@ -575,150 +313,121 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
575
313
  }]
576
314
  }] });
577
315
 
578
- class CartPreviewComponent {
579
- constructor(flowConfiguration, routerService, productImagesService, quoteDraftService) {
580
- this.flowConfiguration = flowConfiguration;
581
- this.routerService = routerService;
582
- this.productImagesService = productImagesService;
583
- this.quoteDraftService = quoteDraftService;
584
- this.productRowHeight = 65;
585
- this.form = new FormGroup({});
586
- this.hasTermInProducts = false;
587
- this.scrollHeight = 0;
588
- this.shouldUpdate$ = new BehaviorSubject(true);
589
- this.destroyed$ = new Subject();
590
- this.lockedProductId$ = combineLatest([this.routerService.route$, this.routerService.isConfigurationRoute$()]).pipe(map(([route, isConfigurationRoute]) => {
591
- if (!isConfigurationRoute) {
592
- return null;
593
- }
594
- return route.queryParams['productId'];
595
- }));
596
- this.isEditMode$ = this.quoteDraftService.isEditMode$();
597
- }
598
- ngOnChanges(changes) {
599
- const { products } = changes;
600
- if (products) {
601
- this.shouldUpdate$.next(true);
602
- this.scrollHeight = Math.min(this.products.length * this.productRowHeight, 460);
603
- }
604
- }
605
- ngAfterViewInit() {
606
- combineLatest([this.lockedProductId$, this.isEditMode$, this.shouldUpdate$, this.overlayPanel.onShow])
607
- .pipe(filter(() => this.overlayPanel.overlayVisible), takeUntil(this.destroyed$))
608
- .subscribe(([lockedProductId, isEditMode]) => this.updateControls(this.products, isEditMode, lockedProductId));
316
+ class FlowHeaderComponent {
317
+ constructor(templatesApi, flowStateService, flowInfo, toastService, customizationService) {
318
+ this.templatesApi = templatesApi;
319
+ this.flowStateService = flowStateService;
320
+ this.flowInfo = flowInfo;
321
+ this.toastService = toastService;
322
+ this.customizationService = customizationService;
323
+ this.uiDefinition$ = new BehaviorSubject(null);
324
+ this.templateApiName = '';
325
+ this.destroy$ = new Subject();
326
+ // wait until flow is initialized
327
+ this.flowStateService
328
+ .isInitialized$()
329
+ .pipe(filter(Boolean), first(), tap(() => this.initialize()), takeUntil(this.destroy$))
330
+ .subscribe();
609
331
  }
610
332
  ngOnDestroy() {
611
- this.destroyed$.next();
612
- this.destroyed$.complete();
613
- }
614
- getImageUrl(productId) {
615
- return this.productImagesService.getImageUrl$(productId);
616
- }
617
- navigateToProductConfiguration(productId, lineItemId) {
618
- this.overlayPanel.hide();
619
- this.routerService.navigateToProductConfiguration(productId, lineItemId);
620
- }
621
- controlBlurHandler(product) {
622
- const control = this.form.get(product.id);
623
- if (!control || control.invalid) {
624
- return;
625
- }
626
- this.flowConfiguration.update([
627
- {
628
- dataType: 'LINEITEM',
629
- attributeType: 'QTY',
630
- id: product.id,
631
- newValue: control.value,
632
- },
633
- ]);
634
- }
635
- deleteHandler(product) {
636
- this.flowConfiguration.delete([product.id]);
333
+ this.destroy$.next();
334
+ this.destroy$.complete();
637
335
  }
638
- deleteAllHandler() {
639
- const productIds = this.products.map(product => product.id);
640
- this.flowConfiguration.delete(productIds);
336
+ initialize() {
337
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.flowHeader ?? '';
338
+ this.generateUIDefinition$()
339
+ .pipe(tap(uiDef => {
340
+ if (!uiDef) {
341
+ throw new Error(`Component with name '${this.templateApiName}' not found.`);
342
+ }
343
+ this.uiDefinition$.next(uiDef);
344
+ }), catchError(err => {
345
+ const message = 'Failed to resolve Flow Header component. ' + (err.message ?? '');
346
+ this.toastService.add({ severity: ToastType.error, summary: message, sticky: true });
347
+ return of();
348
+ }), takeUntil(this.destroy$))
349
+ .subscribe();
641
350
  }
642
- updateControls(products, isEditMode, lockedProductId) {
643
- const ids = [];
644
- products.forEach(item => {
645
- if (!item.id) {
351
+ getLocalMeta$() {
352
+ if (!this.customizationService?.getTemplateComponents) {
353
+ return of(undefined);
354
+ }
355
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
356
+ if (!components) {
646
357
  return;
647
358
  }
648
- ids.push(item.id);
649
- let control = this.form.get(item.id);
650
- if (!control) {
651
- control = new FormControl(item.qty, []);
652
- this.form.addControl(item.id, control);
653
- }
654
- else {
655
- control.setValue(item.qty);
656
- }
657
- if (!isEditMode || item.deleted || item.productId === lockedProductId) {
658
- control.disable();
359
+ return components.map(component => ({
360
+ html: component.html,
361
+ css: component.css,
362
+ js: component.js,
363
+ json: component.json,
364
+ }));
365
+ }));
366
+ }
367
+ getOrgMeta$() {
368
+ const template = this.flowInfo.templates.FLOW_HEADER;
369
+ if (!template) {
370
+ return of(undefined);
371
+ }
372
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
373
+ }
374
+ generateUIDefinition$() {
375
+ return of(undefined).pipe(tap(() => {
376
+ if (!this.templateApiName) {
377
+ throw new Error("Flow 'flowHeader' template is not defined.");
659
378
  }
660
- else {
661
- control.enable();
379
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
380
+ if (!metaList) {
381
+ return;
662
382
  }
663
- });
664
- const removedIds = Object.keys(this.form.controls).filter(id => !ids.includes(id));
665
- removedIds.forEach(id => this.form.removeControl(id));
666
- this.hasTermInProducts = products.some((item) => item.hasTerm);
383
+ const uiDef = {
384
+ name: '',
385
+ createdTimestamp: 0,
386
+ primary: true,
387
+ type: 'DEFAULT',
388
+ version: 2,
389
+ children: metaList.map(meta => ({
390
+ children: [],
391
+ template: meta.html && btoaSafe(meta.html),
392
+ script: meta.js && btoaSafe(meta.js),
393
+ styles: meta.css && btoaSafe(meta.css),
394
+ })),
395
+ };
396
+ return uiDef;
397
+ }));
667
398
  }
668
399
  }
669
- CartPreviewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewComponent, deps: [{ token: i1$2.FlowConfigurationService }, { token: FlowRouterService }, { token: i1$2.ProductImagesService }, { token: i1$2.QuoteDraftService }], target: i0.ɵɵFactoryTarget.Component });
670
- CartPreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: CartPreviewComponent, selector: "vl-cart-preview", inputs: { products: "products" }, viewQueries: [{ propertyName: "overlayPanel", first: true, predicate: OverlayPanel, descendants: true }], usesOnChanges: true, ngImport: i0, template: "<p-overlayPanel\n styleClass=\"catalog-overlay flow-header-overlay center no-padding\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\" *vlLet=\"lockedProductId$ | async as lockedProductId\">\n <ng-container *vlLet=\"isEditMode$ | async as isEditMode\">\n <ng-container *ngIf=\"products.length > 0; else empty\">\n <h2 class=\"flow-header-overlay__title\">\n <span>Products ({{ products.length }})</span>\n <i class=\"vl-icon vl-cross close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <p-virtualScroller [value]=\"products\" scrollHeight=\"{{ scrollHeight }}px\" [itemSize]=\"productRowHeight\">\n <ng-template pTemplate=\"item\" let-product>\n <div\n class=\"product item\"\n *vlLet=\"!isEditMode || product.deleted as isReadonlyProduct\"\n [class.readonly]=\"isReadonlyProduct\"\n [class.deleted]=\"product.deleted\"\n >\n <ng-container *vlLet=\"lockedProductId === product.productId as isLockedProduct\">\n <div class=\"product__info\">\n <div class=\"product__image-wrapper\">\n <div\n *ngIf=\"getImageUrl(product.productId) | async as imageUrl; else noImage\"\n class=\"product__image\"\n [ngStyle]=\"{ 'background-image': 'url(' + imageUrl + ')' }\"\n ></div>\n </div>\n <div class=\"flex flex-column justify-content-center\">\n <div class=\"word-break product-name\" [class.line-through]=\"product.deleted\">\n {{ product.name }}\n </div>\n <div class=\"actions\" *ngIf=\"!isReadonlyProduct\">\n <p-button\n label=\"Configure\"\n [disabled]=\"isLockedProduct || !product.configurable || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm\"\n (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n ></p-button>\n <p-button\n label=\"Delete\"\n [disabled]=\"isLockedProduct || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm p-button-secondary\"\n (onClick)=\"deleteHandler(product)\"\n ></p-button>\n </div>\n </div>\n </div>\n\n <ng-container *ngIf=\"form.get(product.id) as control\">\n <div class=\"qty\" *ngIf=\"!isReadonlyProduct; else readonlyQty\">\n <vl-quantity-control\n [formControl]=\"$any(control)\"\n (valueChange)=\"controlBlurHandler(product)\"\n ></vl-quantity-control>\n </div>\n <ng-template #readonlyQty\n ><span class=\"text-right\">{{ control.value }}</span></ng-template\n >\n </ng-container>\n </ng-container>\n </div>\n </ng-template>\n </p-virtualScroller>\n\n <ng-template #noImage>\n <i class=\"vl-icon vl-no-image no-image-icon\"></i>\n </ng-template>\n\n <div class=\"flex footer\">\n <p-button\n label=\"Clear Cart\"\n styleClass=\"p-button-link p-button-sm pl-0 pr-0\"\n [disabled]=\"!isEditMode || !!lockedProductId || hasTermInProducts\"\n (onClick)=\"deleteAllHandler()\"\n ></p-button>\n </div>\n </ng-container>\n </ng-container>\n\n <ng-template #empty>\n <h2 class=\"flow-header-overlay__title\">\n <span>Empty Cart</span>\n <i class=\"vl-icon vl-cross close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <span class=\"empty-state\">\n <i class=\"vl-icon vl-bag\"></i>\n There are no products added to the Shopping Cart yet.\n </span>\n </ng-template>\n </div>\n </ng-template>\n</p-overlayPanel>\n", styles: ["p-virtualscroller ::ng-deep .p-virtualscroller-header{background:none;padding:0;border:none;font-weight:unset}.flow-header-overlay *{font-family:var(--cg-font-family)}.flow-header-overlay__wrapper{display:flex;flex-direction:column;width:460px;max-height:600px}.flow-header-overlay__wrapper .close-icon{cursor:pointer}.flow-header-overlay__title{display:flex;justify-content:space-between;align-items:center;margin:0 0 8px;font-size:16px;font-style:normal;font-weight:500;line-height:20px}.empty-state{background:var(--cg-bg-color);padding:16px;display:flex;gap:8px;justify-content:center;font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}.product{display:grid;grid-template-columns:auto 100px;padding:8px 0;border-bottom:1px solid var(--cg-border-color)}.product.readonly{align-items:center}.product.deleted{opacity:.5}.product.item{height:65px;overflow:hidden}.product__info{display:flex;gap:16px}.product__image-wrapper{flex-shrink:0;height:48px;width:48px;display:flex;justify-content:center;align-items:center;background:var(--cg-bg-color-hover);border-radius:4px}.product__image{background-size:contain;background-repeat:no-repeat;background-position:center;height:calc(100% - 12px);width:calc(100% - 12px)}.product .product-name{margin-bottom:4px;font-size:14px;font-weight:500;line-height:20px}.product .actions{display:flex;gap:16px}.footer{padding-top:8px}.word-break{word-break:break-word}.no-image-icon{height:24px;width:24px}.qty{display:flex;align-items:center}.flow-header-overlay__wrapper ::ng-deep .p-button-link{padding:0;font-size:12px;font-weight:400;line-height:18px;letter-spacing:.3px;color:var(--cg-primary-color)}.flow-header-overlay__wrapper ::ng-deep .p-button-link:hover:enabled{color:var(--cg-text-color-secondary)}.flow-header-overlay__wrapper ::ng-deep .p-button-link:hover:enabled .p-button-label{text-decoration:none}.flow-header-overlay__wrapper ::ng-deep .p-button-link.p-button-secondary{color:var(--cg-text-color-secondary)}.flow-header-overlay__wrapper ::ng-deep .p-button-link.p-button-secondary:hover:enabled{color:var(--cg-primary-color)}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "directive", type: i5$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5$1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: i7.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { kind: "directive", type: i8.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i2$1.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { kind: "component", type: i9.VirtualScroller, selector: "p-virtualScroller", inputs: ["value", "itemSize", "style", "styleClass", "scrollHeight", "lazy", "rows", "minBufferPx", "maxBufferPx", "delay", "trackBy", "totalRecords", "first", "cache"], outputs: ["onLazyLoad"] }, { kind: "component", type: i2$1.QuantityControlComponent, selector: "vl-quantity-control", outputs: ["valueChange"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
671
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewComponent, decorators: [{
400
+ FlowHeaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i2.FlowStateService }, { token: i2.FlowInfoService }, { token: i2$1.ToastService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
401
+ FlowHeaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FlowHeaderComponent, selector: "vl-flow-new-header", ngImport: i0, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
402
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderComponent, decorators: [{
672
403
  type: Component,
673
- args: [{ selector: 'vl-cart-preview', changeDetection: ChangeDetectionStrategy.OnPush, template: "<p-overlayPanel\n styleClass=\"catalog-overlay flow-header-overlay center no-padding\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\" *vlLet=\"lockedProductId$ | async as lockedProductId\">\n <ng-container *vlLet=\"isEditMode$ | async as isEditMode\">\n <ng-container *ngIf=\"products.length > 0; else empty\">\n <h2 class=\"flow-header-overlay__title\">\n <span>Products ({{ products.length }})</span>\n <i class=\"vl-icon vl-cross close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <p-virtualScroller [value]=\"products\" scrollHeight=\"{{ scrollHeight }}px\" [itemSize]=\"productRowHeight\">\n <ng-template pTemplate=\"item\" let-product>\n <div\n class=\"product item\"\n *vlLet=\"!isEditMode || product.deleted as isReadonlyProduct\"\n [class.readonly]=\"isReadonlyProduct\"\n [class.deleted]=\"product.deleted\"\n >\n <ng-container *vlLet=\"lockedProductId === product.productId as isLockedProduct\">\n <div class=\"product__info\">\n <div class=\"product__image-wrapper\">\n <div\n *ngIf=\"getImageUrl(product.productId) | async as imageUrl; else noImage\"\n class=\"product__image\"\n [ngStyle]=\"{ 'background-image': 'url(' + imageUrl + ')' }\"\n ></div>\n </div>\n <div class=\"flex flex-column justify-content-center\">\n <div class=\"word-break product-name\" [class.line-through]=\"product.deleted\">\n {{ product.name }}\n </div>\n <div class=\"actions\" *ngIf=\"!isReadonlyProduct\">\n <p-button\n label=\"Configure\"\n [disabled]=\"isLockedProduct || !product.configurable || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm\"\n (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n ></p-button>\n <p-button\n label=\"Delete\"\n [disabled]=\"isLockedProduct || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm p-button-secondary\"\n (onClick)=\"deleteHandler(product)\"\n ></p-button>\n </div>\n </div>\n </div>\n\n <ng-container *ngIf=\"form.get(product.id) as control\">\n <div class=\"qty\" *ngIf=\"!isReadonlyProduct; else readonlyQty\">\n <vl-quantity-control\n [formControl]=\"$any(control)\"\n (valueChange)=\"controlBlurHandler(product)\"\n ></vl-quantity-control>\n </div>\n <ng-template #readonlyQty\n ><span class=\"text-right\">{{ control.value }}</span></ng-template\n >\n </ng-container>\n </ng-container>\n </div>\n </ng-template>\n </p-virtualScroller>\n\n <ng-template #noImage>\n <i class=\"vl-icon vl-no-image no-image-icon\"></i>\n </ng-template>\n\n <div class=\"flex footer\">\n <p-button\n label=\"Clear Cart\"\n styleClass=\"p-button-link p-button-sm pl-0 pr-0\"\n [disabled]=\"!isEditMode || !!lockedProductId || hasTermInProducts\"\n (onClick)=\"deleteAllHandler()\"\n ></p-button>\n </div>\n </ng-container>\n </ng-container>\n\n <ng-template #empty>\n <h2 class=\"flow-header-overlay__title\">\n <span>Empty Cart</span>\n <i class=\"vl-icon vl-cross close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <span class=\"empty-state\">\n <i class=\"vl-icon vl-bag\"></i>\n There are no products added to the Shopping Cart yet.\n </span>\n </ng-template>\n </div>\n </ng-template>\n</p-overlayPanel>\n", styles: ["p-virtualscroller ::ng-deep .p-virtualscroller-header{background:none;padding:0;border:none;font-weight:unset}.flow-header-overlay *{font-family:var(--cg-font-family)}.flow-header-overlay__wrapper{display:flex;flex-direction:column;width:460px;max-height:600px}.flow-header-overlay__wrapper .close-icon{cursor:pointer}.flow-header-overlay__title{display:flex;justify-content:space-between;align-items:center;margin:0 0 8px;font-size:16px;font-style:normal;font-weight:500;line-height:20px}.empty-state{background:var(--cg-bg-color);padding:16px;display:flex;gap:8px;justify-content:center;font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}.product{display:grid;grid-template-columns:auto 100px;padding:8px 0;border-bottom:1px solid var(--cg-border-color)}.product.readonly{align-items:center}.product.deleted{opacity:.5}.product.item{height:65px;overflow:hidden}.product__info{display:flex;gap:16px}.product__image-wrapper{flex-shrink:0;height:48px;width:48px;display:flex;justify-content:center;align-items:center;background:var(--cg-bg-color-hover);border-radius:4px}.product__image{background-size:contain;background-repeat:no-repeat;background-position:center;height:calc(100% - 12px);width:calc(100% - 12px)}.product .product-name{margin-bottom:4px;font-size:14px;font-weight:500;line-height:20px}.product .actions{display:flex;gap:16px}.footer{padding-top:8px}.word-break{word-break:break-word}.no-image-icon{height:24px;width:24px}.qty{display:flex;align-items:center}.flow-header-overlay__wrapper ::ng-deep .p-button-link{padding:0;font-size:12px;font-weight:400;line-height:18px;letter-spacing:.3px;color:var(--cg-primary-color)}.flow-header-overlay__wrapper ::ng-deep .p-button-link:hover:enabled{color:var(--cg-text-color-secondary)}.flow-header-overlay__wrapper ::ng-deep .p-button-link:hover:enabled .p-button-label{text-decoration:none}.flow-header-overlay__wrapper ::ng-deep .p-button-link.p-button-secondary{color:var(--cg-text-color-secondary)}.flow-header-overlay__wrapper ::ng-deep .p-button-link.p-button-secondary:hover:enabled{color:var(--cg-primary-color)}\n"] }]
674
- }], ctorParameters: function () { return [{ type: i1$2.FlowConfigurationService }, { type: FlowRouterService }, { type: i1$2.ProductImagesService }, { type: i1$2.QuoteDraftService }]; }, propDecorators: { overlayPanel: [{
675
- type: ViewChild,
676
- args: [OverlayPanel]
677
- }], products: [{
678
- type: Input
679
- }] } });
404
+ args: [{ selector: 'vl-flow-new-header', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"uiDefinition$ | async as uiDefinition\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n</ng-container>\n", styles: [":host,vl-cms-preview{display:contents}\n"] }]
405
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i2.FlowStateService }, { type: i2.FlowInfoService }, { type: i2$1.ToastService }, { type: undefined, decorators: [{
406
+ type: Optional
407
+ }, {
408
+ type: Inject,
409
+ args: [FLOW_CUSTOMIZATION]
410
+ }] }]; } });
680
411
 
681
- class CartPreviewModule {
412
+ class FlowNewHeaderModule {
682
413
  }
683
- CartPreviewModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
684
- CartPreviewModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewModule, declarations: [CartPreviewComponent], imports: [CommonModule,
685
- ReactiveFormsModule,
686
- ButtonModule,
687
- OverlayPanelModule,
688
- LetDirectiveModule,
689
- InputNumberModule,
690
- VirtualScrollerModule,
691
- QuantityControlModule], exports: [CartPreviewComponent] });
692
- CartPreviewModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewModule, imports: [CommonModule,
693
- ReactiveFormsModule,
694
- ButtonModule,
695
- OverlayPanelModule,
696
- LetDirectiveModule,
697
- InputNumberModule,
698
- VirtualScrollerModule,
699
- QuantityControlModule] });
700
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CartPreviewModule, decorators: [{
414
+ FlowNewHeaderModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowNewHeaderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
415
+ FlowNewHeaderModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: FlowNewHeaderModule, declarations: [FlowHeaderComponent], imports: [CommonModule, PreviewModule], exports: [FlowHeaderComponent] });
416
+ FlowNewHeaderModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowNewHeaderModule, imports: [CommonModule, PreviewModule] });
417
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowNewHeaderModule, decorators: [{
701
418
  type: NgModule,
702
419
  args: [{
703
- declarations: [CartPreviewComponent],
704
- imports: [
705
- CommonModule,
706
- ReactiveFormsModule,
707
- ButtonModule,
708
- OverlayPanelModule,
709
- LetDirectiveModule,
710
- InputNumberModule,
711
- VirtualScrollerModule,
712
- QuantityControlModule,
713
- ],
714
- exports: [CartPreviewComponent],
420
+ declarations: [FlowHeaderComponent],
421
+ imports: [CommonModule, PreviewModule],
422
+ exports: [FlowHeaderComponent],
715
423
  }]
716
424
  }] });
717
425
 
718
426
  class FlowDialogService {
719
- constructor(dialogService, contextService) {
427
+ constructor(dialogService, contextService, runtimeSettings) {
720
428
  this.dialogService = dialogService;
721
429
  this.contextService = contextService;
430
+ this.runtimeSettings = runtimeSettings;
722
431
  }
723
432
  show(config) {
724
433
  return this.dialogService.open(FlowDialogComponent, {
@@ -740,7 +449,7 @@ class FlowDialogService {
740
449
  }
741
450
  showReadonlyModeDialog() {
742
451
  const ctx = this.contextService.resolve();
743
- const objectName = ctx.mode ? ctx.mode[0].toUpperCase() + ctx.mode.substring(1).toLowerCase() : 'Object';
452
+ const objectName = ctx.mode ? ctx.mode[0]?.toUpperCase() + ctx.mode.substring(1).toLowerCase() : 'Object';
744
453
  return this.show({
745
454
  title: 'Error',
746
455
  description: `${objectName} Cannot be Saved`,
@@ -827,674 +536,336 @@ class FlowDialogService {
827
536
  secondaryButton: 'Cancel',
828
537
  });
829
538
  }
539
+ showTermsLimitReachedDialog() {
540
+ const shoppingCartSettings = mapShoppingCartSettings(this.runtimeSettings.getShoppingCartSettings() || []);
541
+ const maxRenewalTerms = getMaxRenewalTermsValue(shoppingCartSettings);
542
+ return this.show({
543
+ title: 'Terms Limit Reached',
544
+ description: `You have reached the term quantity limit: ${maxRenewalTerms}. You can increase the limit in the Shopping Cart Settings.`,
545
+ primaryButton: 'Ok',
546
+ });
547
+ }
548
+ showDialog(dialog) {
549
+ const dialogFunction = this[dialog].bind(this);
550
+ return dialogFunction(dialog);
551
+ }
830
552
  }
831
- FlowDialogService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDialogService, deps: [{ token: i1.DialogService }, { token: i1$2.ContextService }], target: i0.ɵɵFactoryTarget.Injectable });
553
+ FlowDialogService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDialogService, deps: [{ token: i1.DialogService }, { token: i2.ContextService }, { token: i2.RuntimeSettingsService }], target: i0.ɵɵFactoryTarget.Injectable });
832
554
  FlowDialogService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDialogService });
833
555
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDialogService, decorators: [{
834
556
  type: Injectable
835
- }], ctorParameters: function () { return [{ type: i1.DialogService }, { type: i1$2.ContextService }]; } });
557
+ }], ctorParameters: function () { return [{ type: i1.DialogService }, { type: i2.ContextService }, { type: i2.RuntimeSettingsService }]; } });
836
558
 
837
- const METRICS_STORAGE_KEY = 'vl-metrics';
838
-
839
- class MetricsComponent {
840
- constructor(contextService, quoteDraftService, cdr, metricsCalculationService, flowConfiguration, shoppingCartSettingsApiService, runtimeSettings) {
559
+ class FlowRouterService {
560
+ constructor(router, route, contextService, integrationState, flowInfoService, flowStateService) {
561
+ this.router = router;
562
+ this.route = route;
841
563
  this.contextService = contextService;
842
- this.quoteDraftService = quoteDraftService;
843
- this.cdr = cdr;
844
- this.metricsCalculationService = metricsCalculationService;
845
- this.flowConfiguration = flowConfiguration;
846
- this.shoppingCartSettingsApiService = shoppingCartSettingsApiService;
847
- this.runtimeSettings = runtimeSettings;
848
- this.emptyStateMetrics = [
849
- { name: 'MRR', value: 0 },
850
- { name: 'E.MRR', value: 0 },
851
- { name: 'NRR', value: 0 },
852
- { name: 'E.NRR', value: 0 },
853
- ];
854
- this.visibleSelectedMetrics = [];
855
- this.restSelectedMetrics = [];
856
- this.metrics = [];
857
- this.filteredMetrics = [];
858
- this.sidebarVisible = false;
859
- this.searchControl = new FormControl('');
860
- this.isFocused = false;
861
- this.nameControl = new FormControl('', [
862
- requiredValidator,
863
- invalidCharactersValidator(standardCharSetWithSpaceRegExp),
864
- reservedIdentifierValidator,
865
- Validators.maxLength(80),
866
- ]);
867
- this.activeMetricRules = [];
868
- this.activeMetricRuleNames = [];
869
- this.defaultMetricRuleNames = [];
870
- this.metricKeys = [];
871
- this.lastSavedMetrics = [];
872
- this.destroyed$ = new Subject();
873
- this.currencySymbol = this.runtimeSettings.getCurrencySettings().symbol;
874
- }
875
- ngOnInit() {
876
- this.activeMetricRules = this.flowConfiguration.activeMetricsSnapshot.filter(metricRule => metricRule.metrics.some(metric => !!metric.totalName));
877
- const activeMetricRuleNames = this.activeMetricRules.map(metricRule => metricRule.title);
878
- this.shoppingCartSettingsApiService
879
- .getSettingsMap()
880
- .pipe(withLatestFrom(this.quoteDraftService.currentState$), tap$1(([settings, lineItems]) => {
881
- this.defaultMetricRuleNames = (settings?.DEFAULT_HEADER_METRICS || []).filter(ruleName => activeMetricRuleNames.includes(ruleName));
882
- this.metricKeys = this.collectMetricKeys(lineItems);
883
- this.activeMetricRuleNames = this.getActiveMetricRuleNames();
884
- }), takeUntil$1(this.destroyed$))
885
- .subscribe(() => {
886
- this.collectAvailableMetrics();
887
- });
888
- this.searchControl.valueChanges.pipe(takeUntil$1(this.destroyed$), startWith('')).subscribe(query => {
889
- this.filteredMetrics = this.metrics.filter(({ name }) => {
890
- return (name || '').toLowerCase().includes(query?.toLowerCase());
891
- });
564
+ this.integrationState = integrationState;
565
+ this.flowInfoService = flowInfoService;
566
+ this.flowStateService = flowStateService;
567
+ this.urlHistory = [];
568
+ this.getLastChildRoute = (route) => {
569
+ return route.firstChild ? this.getLastChildRoute(route.firstChild) : route;
570
+ };
571
+ this.getNthChildRoute = (route, index) => {
572
+ if (index <= 0) {
573
+ return route;
574
+ }
575
+ return route.firstChild ? this.getNthChildRoute(route.firstChild, index - 1) : route;
576
+ };
577
+ this.getLastChildRouteSnapshot = (route) => {
578
+ return route.firstChild ? this.getLastChildRouteSnapshot(route.firstChild) : route;
579
+ };
580
+ this.watchLastChildRoute$ = (route) => {
581
+ return this.routeChange$.pipe(map(() => this.getLastChildRouteSnapshot(route.snapshot)));
582
+ };
583
+ this.getLastChildParams = (route) => {
584
+ return route.firstChild ? this.getLastChildParams(route.firstChild) : route.params;
585
+ };
586
+ this.watchLastChildParams$ = (route) => {
587
+ return this.routeChange$.pipe(map(() => this.getLastChildParams(route.snapshot)));
588
+ };
589
+ this.getFlowSubpath$ = () => {
590
+ return this.route$.pipe(map(route => {
591
+ const url = route.url.join('/');
592
+ const flowRootUrl = this.getFlowRootPath(route);
593
+ return url.replace(flowRootUrl, '');
594
+ }));
595
+ };
596
+ this.routeChange$ = this.router.events.pipe(filter(e => e instanceof NavigationEnd), shareReplay());
597
+ this.lastChildParams$ = this.watchLastChildParams$(this.route).pipe(startWith(this.getLastChildParams(this.route.snapshot)), shareReplay());
598
+ this.lastChildRoute$ = this.watchLastChildRoute$(this.route).pipe(startWith(this.getLastChildRouteSnapshot(this.route.snapshot)), shareReplay());
599
+ this.loading$ = this.router.events.pipe(filter(e => e instanceof NavigationStart ||
600
+ e instanceof NavigationCancel ||
601
+ e instanceof NavigationEnd ||
602
+ e instanceof NavigationError), map(e => e instanceof NavigationStart), startWith(false), distinctUntilChanged());
603
+ this.routeChange$.subscribe(() => {
604
+ this.urlHistory.push(this.router.url);
605
+ // Close Guided Selling when user navigates to another flow page
606
+ this.integrationState.dispatch(FlowAction.CloseGuidedSelling({ keepState: true }));
892
607
  });
893
- this.metricsCalculationService.onMetricsUpdate$
894
- .pipe(takeUntil$1(this.destroyed$))
895
- .subscribe(() => this.cdr.detectChanges());
896
608
  }
897
- ngOnDestroy() {
898
- this.destroyed$.next();
899
- this.destroyed$.complete();
900
- }
901
- toggleOverlay(event, overlay, target) {
902
- event.stopPropagation();
903
- if (this.restSelectedMetrics.length) {
904
- if (!overlay.overlayVisible) {
905
- document.dispatchEvent(new Event('click'));
609
+ getFlowRootRoute(route) {
610
+ const path = [...route.pathFromRoot];
611
+ while (path.length) {
612
+ const parent = path.pop();
613
+ if (!parent) {
614
+ break;
615
+ }
616
+ if (parent.routeConfig?.id === VELOCE_FLOW_ROOT_ROUTE) {
617
+ return parent;
906
618
  }
907
- overlay.toggle(event, target);
908
619
  }
909
- else {
910
- this.openSidebar();
620
+ return;
621
+ }
622
+ getFlowRootPath(route) {
623
+ const rootRoute = this.getFlowRootRoute(route);
624
+ if (!rootRoute) {
625
+ return '';
911
626
  }
627
+ const path = rootRoute.pathFromRoot
628
+ .map(r => r.url[0])
629
+ .filter(Boolean)
630
+ .join('/');
631
+ return '/' + path;
912
632
  }
913
- openSidebar() {
914
- this.overlayPanel.hide();
915
- this.sidebarVisible = true;
633
+ get route$() {
634
+ return this.lastChildRoute$;
916
635
  }
917
- closeSidebar() {
918
- this.sidebarVisible = false;
919
- this.metrics = cloneDeep(this.lastSavedMetrics);
920
- this.editingMetric = undefined;
921
- this.searchControl.setValue('');
636
+ get params$() {
637
+ return this.lastChildParams$;
922
638
  }
923
- getMetricValue(key) {
924
- const context = this.contextService.resolve();
925
- if (key.startsWith('Effective_')) {
926
- const correctKey = key.replace('Effective_', '');
927
- return this.toNumber(context.properties[correctKey]) - this.toNumber(context.properties[`Previous${correctKey}`]);
928
- }
929
- else {
930
- return this.metricsCalculationService.getMetricValue(key);
931
- }
639
+ get params() {
640
+ return this.getLastChildParams(this.route.snapshot);
932
641
  }
933
- save() {
934
- localStorage.setItem(METRICS_STORAGE_KEY, JSON.stringify(this.metrics));
935
- this.setMetrics();
936
- this.lastSavedMetrics = cloneDeep(this.metrics);
937
- this.closeSidebar();
642
+ isConfigurationRoute$() {
643
+ return this.getFlowSubpath$().pipe(map(url => url.startsWith('product')));
938
644
  }
939
- changeMetricOrder(event) {
940
- moveItemInArray(this.metrics, event.previousIndex, event.currentIndex);
941
- this.filteredMetrics = this.metrics;
645
+ isCartRoute$() {
646
+ return this.getFlowSubpath$().pipe(map(url => url.startsWith('cart')));
942
647
  }
943
- editMetric(metric) {
944
- if (this.editingMetric && this.nameControl.invalid) {
945
- return;
648
+ isCatalogRoute$() {
649
+ return this.getFlowSubpath$().pipe(map(url => url.startsWith('catalog')));
650
+ }
651
+ isAssetsRoute$() {
652
+ return this.getFlowSubpath$().pipe(map(url => url.startsWith('assets')));
653
+ }
654
+ navigateBack() {
655
+ const prevUrl = this.urlHistory[this.urlHistory.length - 2];
656
+ if (prevUrl) {
657
+ this.router.navigateByUrl(prevUrl);
946
658
  }
947
- this.editingMetric = metric;
948
- this.nameControl.setValue(metric.name);
949
659
  }
950
- saveMetric() {
951
- if (this.nameControl.invalid) {
952
- return;
660
+ navigateTo(path, productId, lineItemId) {
661
+ if (path === 'shopping-cart') {
662
+ this.navigateToShoppingCart();
953
663
  }
954
- if (this.editingMetric) {
955
- this.editingMetric.name = this.nameControl.value?.trim();
664
+ else if (path === 'catalog') {
665
+ this.navigateToCatalog();
666
+ }
667
+ else if (path === 'assets') {
668
+ this.navigateToAssets();
669
+ }
670
+ else if (path === 'product' && productId) {
671
+ this.navigateToProductConfiguration(productId, lineItemId);
956
672
  }
957
- this.editingMetric = undefined;
958
673
  }
959
- cancelMetric() {
960
- this.editingMetric = undefined;
674
+ navigateToProductConfiguration(productId, lineItemId) {
675
+ let updateContext$;
676
+ if (this.flowInfoService.isLegacy) {
677
+ updateContext$ = of(undefined).pipe(tap(() => {
678
+ this.contextService.update({ properties: { productId, lineItemId: lineItemId ?? '' } });
679
+ }));
680
+ }
681
+ else {
682
+ updateContext$ = this.flowStateService.dispatch$(UITemplateType.FLOW_ENGINE, 'UPDATE_CONTEXT_PROPERTIES', {
683
+ productId,
684
+ lineItemId: lineItemId ?? '',
685
+ });
686
+ }
687
+ updateContext$
688
+ .pipe(tap(() => {
689
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
690
+ const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
691
+ this.router.navigate([flowRouteUrl, 'product'], {
692
+ queryParams: { ...routeSnapshot.queryParams, productId },
693
+ });
694
+ }))
695
+ .subscribe();
961
696
  }
962
- resetToDefault() {
963
- this.collectAvailableMetrics(true);
697
+ navigateToShoppingCart() {
698
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
699
+ const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
700
+ this.router.navigate([flowRouteUrl, 'cart'], { queryParams: routeSnapshot.queryParams });
964
701
  }
965
- onFocus() {
966
- this.isFocused = true;
702
+ navigateToCatalog() {
703
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
704
+ const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
705
+ this.router.navigate([flowRouteUrl, 'catalog'], { queryParams: routeSnapshot.queryParams });
967
706
  }
968
- onBlur() {
969
- this.isFocused = false;
707
+ navigateToAssets() {
708
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
709
+ const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
710
+ this.router.navigate([flowRouteUrl, 'assets'], { queryParams: routeSnapshot.queryParams });
970
711
  }
971
- clearSearch(event) {
972
- event.stopPropagation();
973
- this.searchControl.setValue('');
712
+ showErrorPage$(message, details) {
713
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
714
+ const flowRouteUrl = this.getFlowRootPath(routeSnapshot);
715
+ return from(this.router.navigate([flowRouteUrl, '404'], {
716
+ state: { message, type: 'error', ...(details && { details }) },
717
+ replaceUrl: true,
718
+ })).pipe(map(() => false));
974
719
  }
975
- collectMetricKeys(lineItems) {
976
- const keys = [];
977
- lineItems.forEach(lineItem => {
978
- keys.push(...Object.keys(lineItem.totalMetrics || {}));
979
- keys.push(...this.collectMetricKeys(lineItem.lineItems));
980
- });
981
- return uniq(keys);
982
- }
983
- setMetrics() {
984
- const metrics = cloneDeep(this.metrics.filter(({ visible }) => visible));
985
- this.visibleSelectedMetrics = metrics.slice(0, 4);
986
- this.restSelectedMetrics = metrics.slice(4);
987
- }
988
- collectAvailableMetrics(resetToDefault = false) {
989
- const storedMetrics = JSON.parse(localStorage.getItem(METRICS_STORAGE_KEY) || '[]');
990
- const metricKeys = this.metricKeys.slice();
991
- const isDefaultConfiguration = !storedMetrics.length || resetToDefault;
992
- this.metrics = (isDefaultConfiguration ? this.activeMetricRuleNames : storedMetrics).reduce((result, metric) => {
993
- const metricKeyIndex = metricKeys.findIndex(key => key === metric.key);
994
- if (metricKeyIndex >= 0) {
995
- metricKeys.splice(metricKeyIndex, 1);
996
- }
997
- if (isDefaultConfiguration && this.defaultMetricRuleNames.includes(metric.name)) {
998
- metric.visible = true;
999
- }
1000
- return [...result, metric];
1001
- }, []);
1002
- this.metrics = [
1003
- ...this.metrics,
1004
- ...metricKeys.map(metricKey => {
1005
- return { key: metricKey, name: this.getMetricName(metricKey), visible: false };
1006
- }),
1007
- ];
1008
- // sort the metrics by the order of default metric rule names from the shopping cart settings
1009
- this.metrics = this.sortMetrics();
1010
- this.metrics = cloneDeep(this.metrics);
1011
- this.searchControl.setValue('');
1012
- if (!resetToDefault) {
1013
- this.lastSavedMetrics = cloneDeep(this.metrics);
1014
- this.setMetrics();
1015
- }
1016
- this.cdr.markForCheck();
1017
- }
1018
- getMetricName(key) {
1019
- return (this.activeMetricRules.find(metricRule => metricRule.metrics.find(metric => metric.totalName === key))?.title ||
1020
- key);
1021
- }
1022
- toNumber(value) {
1023
- return +(value || 0);
1024
- }
1025
- getActiveMetricRuleNames() {
1026
- return this.activeMetricRules.reduce((result, metricRule) => {
1027
- const name = metricRule.title;
1028
- const visible = this.defaultMetricRuleNames.includes(name);
1029
- metricRule.metrics.forEach(metric => {
1030
- result.push({
1031
- key: metric.totalName || metric.name,
1032
- name,
1033
- visible,
1034
- });
1035
- });
1036
- return result;
1037
- }, []);
1038
- }
1039
- /**
1040
- * Sorts the Metrics array by the order of the default metric rule names from the shopping cart settings.
1041
- * If a metric is included in the list of default metrics, it raises the metric to the top of the metrics array.
1042
- */
1043
- sortMetrics() {
1044
- return this.metrics.slice().sort((a, b) => {
1045
- const indexA = this.defaultMetricRuleNames.indexOf(a.name);
1046
- const indexB = this.defaultMetricRuleNames.indexOf(b.name);
1047
- if (indexA === -1 && indexB === -1) {
1048
- return 0;
1049
- }
1050
- if (indexA === -1) {
1051
- return 1;
1052
- }
1053
- if (indexB === -1) {
1054
- return -1;
1055
- }
1056
- return indexA - indexB;
1057
- });
720
+ switchObject(id) {
721
+ const routeSnapshot = this.getLastChildRouteSnapshot(this.route.snapshot);
722
+ const route = this.getLastChildRoute(this.route);
723
+ const objName = SalesforceIdUtils.getSfObjectNameById(id);
724
+ const queryParams = {
725
+ quoteId: objName === 'Quote' ? id : undefined,
726
+ accountId: objName === 'Account' ? id : undefined,
727
+ orderId: objName === 'Order' ? id : undefined,
728
+ };
729
+ this.router.navigate([], { relativeTo: route, queryParams: { ...routeSnapshot.queryParams, ...queryParams } });
1058
730
  }
1059
731
  }
1060
- MetricsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MetricsComponent, deps: [{ token: i1$2.ContextService }, { token: i1$2.QuoteDraftService }, { token: i0.ChangeDetectorRef }, { token: i1$2.MetricsCalculationService }, { token: i1$2.FlowConfigurationService }, { token: i2.ShoppingCartSettingsApiService }, { token: i1$2.RuntimeSettingsService }], target: i0.ɵɵFactoryTarget.Component });
1061
- MetricsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: MetricsComponent, selector: "vl-metrics", viewQueries: [{ propertyName: "overlayPanel", first: true, predicate: OverlayPanel, descendants: true }], ngImport: i0, template: "<div class=\"header-metrics\" (click)=\"toggleOverlay($event, overlayPanel, metricsTarget)\">\n <div class=\"metrics\">\n <div *ngFor=\"let metric of visibleSelectedMetrics\" class=\"metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ getMetricValue(metric.key) | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ getMetricValue(metric.key) | vlPrice }}\n </span>\n </div>\n\n <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ metric.value | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.value | vlPrice }}\n </span>\n </div>\n </ng-container>\n </div>\n\n <div #metricsTarget class=\"metrics-overlay-target\" [class.expanded]=\"overlayPanel?.overlayVisible\">\n <i class=\"vl-icon vl-arrow-down\"></i>\n </div>\n</div>\n\n<p-overlayPanel\n styleClass=\"catalog-overlay metrics-overlay-container right no-padding\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"overlay-metrics\">\n <div *ngFor=\"let metric of restSelectedMetrics\" class=\"metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ getMetricValue(metric.key) | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ getMetricValue(metric.key) | vlPrice }}\n </span>\n </div>\n </div>\n <button class=\"config\" (click)=\"openSidebar()\">\n Metrics Config\n <i class=\"vl-icon vl-arrow-right\"></i>\n </button>\n </ng-template>\n</p-overlayPanel>\n\n<p-sidebar\n [visible]=\"sidebarVisible\"\n (visibleChange)=\"closeSidebar()\"\n position=\"right\"\n [baseZIndex]=\"1000\"\n [style]=\"{ width: '300px' }\"\n>\n <div class=\"container\">\n <div class=\"header\">\n Quote Metrics Config\n <i class=\"vl-icon vl-cross\" (click)=\"closeSidebar()\"></i>\n </div>\n\n <label class=\"search-container\" [class.active]=\"searchControl.value || isFocused\">\n <i class=\"vl-icon vl-search\"></i>\n <input\n data-test-id=\"search\"\n [formControl]=\"searchControl\"\n pInputText\n placeholder=\"Search for metric\u2026\"\n class=\"w-full search-input\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n />\n <i *ngIf=\"searchControl.value\" class=\"vl-icon vl-cross\" (click)=\"clearSearch($event)\"></i>\n </label>\n\n <div class=\"content\">\n <div class=\"sidebar-metrics\" cdkDropList (cdkDropListDropped)=\"changeMetricOrder($event)\">\n <div\n *ngFor=\"let metric of filteredMetrics\"\n class=\"sidebar-metric\"\n [class.edit]=\"metric.key === editingMetric?.key\"\n cdkDrag\n [cdkDragDisabled]=\"!!searchControl.value\"\n >\n <div class=\"drag-icon\">\n <i class=\"vl-icon vl-drag\" cdkDragHandle></i>\n </div>\n <ng-container\n [ngTemplateOutlet]=\"metricTemplate\"\n [ngTemplateOutletContext]=\"{ metric: metric }\"\n ></ng-container>\n </div>\n </div>\n\n <div class=\"empty-state\" *ngIf=\"!filteredMetrics.length\">There are no items matching your search criteria</div>\n </div>\n <div class=\"footer\">\n <p-button\n label=\"Reset to Default\"\n styleClass=\"p-button-outlined p-button-sm\"\n (onClick)=\"resetToDefault()\"\n ></p-button>\n <p-button label=\"Save\" styleClass=\"p-button p-button-filled p-button-sm\" (onClick)=\"save()\"></p-button>\n </div>\n </div>\n</p-sidebar>\n\n<ng-template #metricTemplate let-metric=\"metric\">\n <div *ngIf=\"editingMetric?.key !== metric.key\" class=\"preview-state\">\n <p-checkbox\n [ngModel]=\"metric.visible\"\n (ngModelChange)=\"metric.visible = !metric.visible\"\n [binary]=\"true\"\n checkboxIcon=\"vl-icon vl-check\"\n ></p-checkbox>\n <div class=\"title\">{{ metric.name }}</div>\n </div>\n\n <div *ngIf=\"editingMetric?.key === metric.key\" class=\"edit-state\">\n <input data-test-id=\"name\" [formControl]=\"nameControl\" pInputText placeholder=\"Metric name\" class=\"w-full\" />\n <vl-error-tooltip [tooltip]=\"nameControl | error : 'name'\" [visible]=\"!!nameControl.errors\"></vl-error-tooltip>\n </div>\n <div class=\"actions\">\n <div *ngIf=\"editingMetric?.key !== metric.key\" (click)=\"editMetric(metric)\" class=\"action edit\">\n <i class=\"vl-icon vl-edit\"></i>\n </div>\n <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"cancelMetric()\" class=\"action cancel\">\n <i class=\"vl-icon vl-cross\"></i>\n </div>\n <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"saveMetric()\" class=\"action save\">\n <i class=\"vl-icon vl-check\"></i>\n </div>\n </div>\n</ng-template>\n", styles: [":host .header-metrics{display:flex;gap:8px;cursor:pointer}:host .header-metrics *{font-family:var(--cg-font-family)}:host .header-metrics .metrics{min-width:130px;max-width:260px;display:flex;flex-wrap:wrap;justify-content:flex-start;gap:0 8px;height:32px;align-items:flex-start}:host .header-metrics .metric{flex:1;flex-basis:114px;max-width:122px;display:flex;justify-content:space-between;align-items:flex-end;font-size:12px;line-height:16px;letter-spacing:.3px}:host .header-metrics .metric .name{color:#ffffffb3;max-width:30px;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}:host .header-metrics .metric .value{overflow:hidden;text-overflow:ellipsis;max-width:7rem}:host .header-metrics .metric.empty-state-metric{filter:blur(1.5px)}:host .metrics-overlay-target{display:flex;align-items:center;width:16px;border-radius:4px;background:rgba(255,255,255,.1)}:host .metrics-overlay-target .vl-icon{color:var(--cg-white)}:host .metrics-overlay-target.expanded .vl-icon:before{transform:rotate(180deg)}:host ::ng-deep .p-sidebar-header{display:none}:host ::ng-deep .p-sidebar-content{padding:0;height:100%}::ng-deep .p-overlaypanel.metrics-overlay-container{width:280px}::ng-deep .p-overlaypanel.metrics-overlay-container *{font-family:var(--cg-font-family)}::ng-deep .p-overlaypanel.metrics-overlay-container .p-overlaypanel-content{padding:0;margin-top:0}::ng-deep .p-overlaypanel.metrics-overlay-container .overlay-metrics{padding:16px;max-height:112px;overflow:auto}::ng-deep .p-overlaypanel.metrics-overlay-container .metric{display:flex;font-size:12px;line-height:16px;letter-spacing:.3px;align-items:flex-end}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .name{color:var(--cg-text-color-secondary);max-width:50%;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .value{overflow:hidden;text-overflow:ellipsis}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .filler:before{background-image:radial-gradient(circle at 1px 1px,var(--cg-text-color-secondary) 1px,transparent 0)}::ng-deep .p-overlaypanel.metrics-overlay-container .config{height:48px;display:flex;align-items:center;justify-content:space-between;gap:10px;padding:8px 8px 8px 16px;cursor:pointer;background:var(--cg-white);outline:none;border:none;border-top:1px solid var(--cg-border-color);width:100%;font-size:12px;line-height:16px;letter-spacing:.3px}::ng-deep .p-overlaypanel.metrics-overlay-container .config .vl-icon-chevron-down{color:var(--cg-primary-color)}.filler{flex:1;margin:0 4px 7px;min-width:7px}.filler:before{display:block;width:100%;height:2px;content:\"\";background-image:radial-gradient(circle at 1px 1px,rgba(255,255,255,.7) 1px,transparent 0);background-size:5px 2px}.container{display:flex;flex-direction:column;height:100%}.container .header{box-sizing:content-box;padding:24px 24px 0;margin-bottom:16px;height:32px;flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;font-family:var(--vl-font-family);font-size:20px;font-style:normal;font-weight:500;line-height:24px;letter-spacing:-.6px;color:var(--vl-accent-color)}.container .header i{cursor:pointer}.container .search-container{cursor:pointer;display:flex;height:32px;margin:0 24px 8px;padding:8px;background:var(--cg-bg-color);border:1px solid transparent;border-radius:32px}.container .search-container:hover,.container .search-container.active{background:var(--cg-white);border-color:var(--cg-primary-color)}.container .search-container .vl-icon{flex:0 0 auto}.container .search-container .search-input{width:100%;height:16px;padding:0 8px;border:none;font-size:12px;line-height:16px;letter-spacing:.3px;color:var(--cg-black-color);background:none}.container .search-container .search-input::placeholder{font-weight:400;font-size:12px;line-height:16px;letter-spacing:.3px;color:var(--cg-text-color-secondary)}.container .content{padding:0 24px;display:flex;flex-direction:column;flex:1;overflow:hidden}.container .footer{flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;gap:16px;padding:16px 24px;border-top:1px solid var(--cg-border-color)}.container .footer p-button{flex:1}.container .footer p-button ::ng-deep button{border-radius:24px;width:100%;font-size:12px;font-family:var(--cg-font-family);font-weight:300;line-height:16px;letter-spacing:.3px;white-space:nowrap}.container .footer p-button ::ng-deep .p-button-outlined{background-color:var(--cg-white);color:var(--cg-primary-color)}.container .footer p-button ::ng-deep .p-button-outlined:hover{background-color:var(--cg-primary-color);color:var(--cg-white)}.container .footer p-button ::ng-deep .p-button-filled{background-color:var(--cg-primary-color);color:var(--cg-white);border:1px solid var(--cg-primary-color)}.container .footer p-button ::ng-deep .p-button-filled:hover{background-color:var(--cg-primary-color-hover);border-color:var(--cg-primary-color-hover)}.container app-empty-state::ng-deep{margin-bottom:12px}.container app-empty-state::ng-deep .container{width:100%;padding-bottom:0}.sidebar-metrics{width:100%;overflow:auto;padding:1px 0}.empty-state{display:flex;align-items:center;justify-content:center;padding:16px;color:var(--cg-text-color-secondary);font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}.sidebar-metric{display:flex;align-items:center;height:32px;position:relative;border-bottom:1px solid var(--cg-border-color)}.sidebar-metric.edit{border-bottom-color:var(--cg-primary-color)}.sidebar-metric.edit .drag-icon{display:none}.sidebar-metric:hover:not(.edit){background:var(--cg-bg-color)}.sidebar-metric:hover:not(.edit) .edit{visibility:visible}.sidebar-metric p-checkbox{margin-right:8px}.sidebar-metric .drag-icon{width:16px;height:16px;margin-right:8px}.sidebar-metric .drag-icon i{cursor:pointer;color:var(--cg-text-color-disabled)}.sidebar-metric.cdk-drag-disabled .drag-icon i{cursor:default}.sidebar-metric .title{font-weight:400;font-size:12px;line-height:16px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-webkit-user-select:text;user-select:text;cursor:text;color:var(--cg-black)}.sidebar-metric .preview-state,.sidebar-metric .edit-state{position:relative;margin-right:8px;min-width:0;display:flex;align-items:center;flex:1}.sidebar-metric .preview-state input,.sidebar-metric .edit-state input{padding-left:48px;border:none;background:transparent;width:100%}.sidebar-metric .actions{display:flex;gap:4px}.sidebar-metric .action{cursor:pointer;width:24px;height:24px;display:flex;align-items:center;justify-content:center;border:1px solid transparent;border-radius:4px}.sidebar-metric .action:hover{background:var(--cg-primary-color);color:var(--cg-white)}.sidebar-metric .edit{visibility:hidden}.sidebar-metric .save{border-color:var(--cg-primary-color);border-radius:4px}.sidebar-metric ::ng-deep .p-checkbox{width:16px;height:16px}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box{width:16px;height:16px;border-width:1px;border-color:var(--cg-border-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box:hover{border-color:var(--cg-primary-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box.p-highlight{background:var(--cg-primary-color);border-color:var(--cg-primary-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box .p-checkbox-icon{flex:0 0 auto;width:12px;height:12px}.cdk-drag-preview{z-index:10001!important;box-sizing:border-box;border-radius:4px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}::ng-deep .metric-tooltip .p-tooltip-text{font-size:11px;line-height:14px;letter-spacing:.3px;background:var(--cg-primary-color-hover)}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet", "ngTemplateOutletInjector"] }, { kind: "directive", type: i5$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5$1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i5$1.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "directive", type: i5$2.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "cdkDropListData", "cdkDropListOrientation", "id", "cdkDropListLockAxis", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListAutoScrollDisabled", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { kind: "directive", type: i5$2.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragData", "cdkDragLockAxis", "cdkDragRootElement", "cdkDragBoundary", "cdkDragStartDelay", "cdkDragFreeDragPosition", "cdkDragDisabled", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragPreviewContainer"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { kind: "directive", type: i5$2.CdkDragHandle, selector: "[cdkDragHandle]", inputs: ["cdkDragHandleDisabled"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: i7.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { kind: "directive", type: i8.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "component", type: i9$1.Sidebar, selector: "p-sidebar", inputs: ["appendTo", "blockScroll", "style", "styleClass", "ariaCloseLabel", "autoZIndex", "baseZIndex", "modal", "dismissible", "showCloseIcon", "closeOnEscape", "transitionOptions", "visible", "position", "fullScreen"], outputs: ["onShow", "onHide", "visibleChange"] }, { kind: "directive", type: i10.InputText, selector: "[pInputText]" }, { kind: "component", type: i11.Checkbox, selector: "p-checkbox", inputs: ["value", "name", "disabled", "binary", "label", "ariaLabelledBy", "ariaLabel", "tabindex", "inputId", "style", "styleClass", "labelStyleClass", "formControl", "checkboxIcon", "readonly", "required", "trueValue", "falseValue"], outputs: ["onChange"] }, { kind: "component", type: i2$1.ErrorTooltipComponent, selector: "vl-error-tooltip", inputs: ["tooltip", "visible", "top"] }, { kind: "directive", type: i2$1.AppHiddenTextTooltipDirective, selector: "[vlHiddenTextTooltip]", inputs: ["vlHiddenTextTooltip"] }, { kind: "pipe", type: i2$1.ErrorPipe, name: "error" }, { kind: "pipe", type: i1$2.PricePipe, name: "vlPrice" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1062
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MetricsComponent, decorators: [{
1063
- type: Component,
1064
- args: [{ selector: 'vl-metrics', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"header-metrics\" (click)=\"toggleOverlay($event, overlayPanel, metricsTarget)\">\n <div class=\"metrics\">\n <div *ngFor=\"let metric of visibleSelectedMetrics\" class=\"metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ getMetricValue(metric.key) | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ getMetricValue(metric.key) | vlPrice }}\n </span>\n </div>\n\n <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ metric.value | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.value | vlPrice }}\n </span>\n </div>\n </ng-container>\n </div>\n\n <div #metricsTarget class=\"metrics-overlay-target\" [class.expanded]=\"overlayPanel?.overlayVisible\">\n <i class=\"vl-icon vl-arrow-down\"></i>\n </div>\n</div>\n\n<p-overlayPanel\n styleClass=\"catalog-overlay metrics-overlay-container right no-padding\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"overlay-metrics\">\n <div *ngFor=\"let metric of restSelectedMetrics\" class=\"metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"{{ getMetricValue(metric.key) | vlPrice }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ getMetricValue(metric.key) | vlPrice }}\n </span>\n </div>\n </div>\n <button class=\"config\" (click)=\"openSidebar()\">\n Metrics Config\n <i class=\"vl-icon vl-arrow-right\"></i>\n </button>\n </ng-template>\n</p-overlayPanel>\n\n<p-sidebar\n [visible]=\"sidebarVisible\"\n (visibleChange)=\"closeSidebar()\"\n position=\"right\"\n [baseZIndex]=\"1000\"\n [style]=\"{ width: '300px' }\"\n>\n <div class=\"container\">\n <div class=\"header\">\n Quote Metrics Config\n <i class=\"vl-icon vl-cross\" (click)=\"closeSidebar()\"></i>\n </div>\n\n <label class=\"search-container\" [class.active]=\"searchControl.value || isFocused\">\n <i class=\"vl-icon vl-search\"></i>\n <input\n data-test-id=\"search\"\n [formControl]=\"searchControl\"\n pInputText\n placeholder=\"Search for metric\u2026\"\n class=\"w-full search-input\"\n (focus)=\"onFocus()\"\n (blur)=\"onBlur()\"\n />\n <i *ngIf=\"searchControl.value\" class=\"vl-icon vl-cross\" (click)=\"clearSearch($event)\"></i>\n </label>\n\n <div class=\"content\">\n <div class=\"sidebar-metrics\" cdkDropList (cdkDropListDropped)=\"changeMetricOrder($event)\">\n <div\n *ngFor=\"let metric of filteredMetrics\"\n class=\"sidebar-metric\"\n [class.edit]=\"metric.key === editingMetric?.key\"\n cdkDrag\n [cdkDragDisabled]=\"!!searchControl.value\"\n >\n <div class=\"drag-icon\">\n <i class=\"vl-icon vl-drag\" cdkDragHandle></i>\n </div>\n <ng-container\n [ngTemplateOutlet]=\"metricTemplate\"\n [ngTemplateOutletContext]=\"{ metric: metric }\"\n ></ng-container>\n </div>\n </div>\n\n <div class=\"empty-state\" *ngIf=\"!filteredMetrics.length\">There are no items matching your search criteria</div>\n </div>\n <div class=\"footer\">\n <p-button\n label=\"Reset to Default\"\n styleClass=\"p-button-outlined p-button-sm\"\n (onClick)=\"resetToDefault()\"\n ></p-button>\n <p-button label=\"Save\" styleClass=\"p-button p-button-filled p-button-sm\" (onClick)=\"save()\"></p-button>\n </div>\n </div>\n</p-sidebar>\n\n<ng-template #metricTemplate let-metric=\"metric\">\n <div *ngIf=\"editingMetric?.key !== metric.key\" class=\"preview-state\">\n <p-checkbox\n [ngModel]=\"metric.visible\"\n (ngModelChange)=\"metric.visible = !metric.visible\"\n [binary]=\"true\"\n checkboxIcon=\"vl-icon vl-check\"\n ></p-checkbox>\n <div class=\"title\">{{ metric.name }}</div>\n </div>\n\n <div *ngIf=\"editingMetric?.key === metric.key\" class=\"edit-state\">\n <input data-test-id=\"name\" [formControl]=\"nameControl\" pInputText placeholder=\"Metric name\" class=\"w-full\" />\n <vl-error-tooltip [tooltip]=\"nameControl | error : 'name'\" [visible]=\"!!nameControl.errors\"></vl-error-tooltip>\n </div>\n <div class=\"actions\">\n <div *ngIf=\"editingMetric?.key !== metric.key\" (click)=\"editMetric(metric)\" class=\"action edit\">\n <i class=\"vl-icon vl-edit\"></i>\n </div>\n <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"cancelMetric()\" class=\"action cancel\">\n <i class=\"vl-icon vl-cross\"></i>\n </div>\n <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"saveMetric()\" class=\"action save\">\n <i class=\"vl-icon vl-check\"></i>\n </div>\n </div>\n</ng-template>\n", styles: [":host .header-metrics{display:flex;gap:8px;cursor:pointer}:host .header-metrics *{font-family:var(--cg-font-family)}:host .header-metrics .metrics{min-width:130px;max-width:260px;display:flex;flex-wrap:wrap;justify-content:flex-start;gap:0 8px;height:32px;align-items:flex-start}:host .header-metrics .metric{flex:1;flex-basis:114px;max-width:122px;display:flex;justify-content:space-between;align-items:flex-end;font-size:12px;line-height:16px;letter-spacing:.3px}:host .header-metrics .metric .name{color:#ffffffb3;max-width:30px;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}:host .header-metrics .metric .value{overflow:hidden;text-overflow:ellipsis;max-width:7rem}:host .header-metrics .metric.empty-state-metric{filter:blur(1.5px)}:host .metrics-overlay-target{display:flex;align-items:center;width:16px;border-radius:4px;background:rgba(255,255,255,.1)}:host .metrics-overlay-target .vl-icon{color:var(--cg-white)}:host .metrics-overlay-target.expanded .vl-icon:before{transform:rotate(180deg)}:host ::ng-deep .p-sidebar-header{display:none}:host ::ng-deep .p-sidebar-content{padding:0;height:100%}::ng-deep .p-overlaypanel.metrics-overlay-container{width:280px}::ng-deep .p-overlaypanel.metrics-overlay-container *{font-family:var(--cg-font-family)}::ng-deep .p-overlaypanel.metrics-overlay-container .p-overlaypanel-content{padding:0;margin-top:0}::ng-deep .p-overlaypanel.metrics-overlay-container .overlay-metrics{padding:16px;max-height:112px;overflow:auto}::ng-deep .p-overlaypanel.metrics-overlay-container .metric{display:flex;font-size:12px;line-height:16px;letter-spacing:.3px;align-items:flex-end}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .name{color:var(--cg-text-color-secondary);max-width:50%;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .value{overflow:hidden;text-overflow:ellipsis}::ng-deep .p-overlaypanel.metrics-overlay-container .metric .filler:before{background-image:radial-gradient(circle at 1px 1px,var(--cg-text-color-secondary) 1px,transparent 0)}::ng-deep .p-overlaypanel.metrics-overlay-container .config{height:48px;display:flex;align-items:center;justify-content:space-between;gap:10px;padding:8px 8px 8px 16px;cursor:pointer;background:var(--cg-white);outline:none;border:none;border-top:1px solid var(--cg-border-color);width:100%;font-size:12px;line-height:16px;letter-spacing:.3px}::ng-deep .p-overlaypanel.metrics-overlay-container .config .vl-icon-chevron-down{color:var(--cg-primary-color)}.filler{flex:1;margin:0 4px 7px;min-width:7px}.filler:before{display:block;width:100%;height:2px;content:\"\";background-image:radial-gradient(circle at 1px 1px,rgba(255,255,255,.7) 1px,transparent 0);background-size:5px 2px}.container{display:flex;flex-direction:column;height:100%}.container .header{box-sizing:content-box;padding:24px 24px 0;margin-bottom:16px;height:32px;flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;font-family:var(--vl-font-family);font-size:20px;font-style:normal;font-weight:500;line-height:24px;letter-spacing:-.6px;color:var(--vl-accent-color)}.container .header i{cursor:pointer}.container .search-container{cursor:pointer;display:flex;height:32px;margin:0 24px 8px;padding:8px;background:var(--cg-bg-color);border:1px solid transparent;border-radius:32px}.container .search-container:hover,.container .search-container.active{background:var(--cg-white);border-color:var(--cg-primary-color)}.container .search-container .vl-icon{flex:0 0 auto}.container .search-container .search-input{width:100%;height:16px;padding:0 8px;border:none;font-size:12px;line-height:16px;letter-spacing:.3px;color:var(--cg-black-color);background:none}.container .search-container .search-input::placeholder{font-weight:400;font-size:12px;line-height:16px;letter-spacing:.3px;color:var(--cg-text-color-secondary)}.container .content{padding:0 24px;display:flex;flex-direction:column;flex:1;overflow:hidden}.container .footer{flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;gap:16px;padding:16px 24px;border-top:1px solid var(--cg-border-color)}.container .footer p-button{flex:1}.container .footer p-button ::ng-deep button{border-radius:24px;width:100%;font-size:12px;font-family:var(--cg-font-family);font-weight:300;line-height:16px;letter-spacing:.3px;white-space:nowrap}.container .footer p-button ::ng-deep .p-button-outlined{background-color:var(--cg-white);color:var(--cg-primary-color)}.container .footer p-button ::ng-deep .p-button-outlined:hover{background-color:var(--cg-primary-color);color:var(--cg-white)}.container .footer p-button ::ng-deep .p-button-filled{background-color:var(--cg-primary-color);color:var(--cg-white);border:1px solid var(--cg-primary-color)}.container .footer p-button ::ng-deep .p-button-filled:hover{background-color:var(--cg-primary-color-hover);border-color:var(--cg-primary-color-hover)}.container app-empty-state::ng-deep{margin-bottom:12px}.container app-empty-state::ng-deep .container{width:100%;padding-bottom:0}.sidebar-metrics{width:100%;overflow:auto;padding:1px 0}.empty-state{display:flex;align-items:center;justify-content:center;padding:16px;color:var(--cg-text-color-secondary);font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}.sidebar-metric{display:flex;align-items:center;height:32px;position:relative;border-bottom:1px solid var(--cg-border-color)}.sidebar-metric.edit{border-bottom-color:var(--cg-primary-color)}.sidebar-metric.edit .drag-icon{display:none}.sidebar-metric:hover:not(.edit){background:var(--cg-bg-color)}.sidebar-metric:hover:not(.edit) .edit{visibility:visible}.sidebar-metric p-checkbox{margin-right:8px}.sidebar-metric .drag-icon{width:16px;height:16px;margin-right:8px}.sidebar-metric .drag-icon i{cursor:pointer;color:var(--cg-text-color-disabled)}.sidebar-metric.cdk-drag-disabled .drag-icon i{cursor:default}.sidebar-metric .title{font-weight:400;font-size:12px;line-height:16px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-webkit-user-select:text;user-select:text;cursor:text;color:var(--cg-black)}.sidebar-metric .preview-state,.sidebar-metric .edit-state{position:relative;margin-right:8px;min-width:0;display:flex;align-items:center;flex:1}.sidebar-metric .preview-state input,.sidebar-metric .edit-state input{padding-left:48px;border:none;background:transparent;width:100%}.sidebar-metric .actions{display:flex;gap:4px}.sidebar-metric .action{cursor:pointer;width:24px;height:24px;display:flex;align-items:center;justify-content:center;border:1px solid transparent;border-radius:4px}.sidebar-metric .action:hover{background:var(--cg-primary-color);color:var(--cg-white)}.sidebar-metric .edit{visibility:hidden}.sidebar-metric .save{border-color:var(--cg-primary-color);border-radius:4px}.sidebar-metric ::ng-deep .p-checkbox{width:16px;height:16px}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box{width:16px;height:16px;border-width:1px;border-color:var(--cg-border-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box:hover{border-color:var(--cg-primary-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box.p-highlight{background:var(--cg-primary-color);border-color:var(--cg-primary-color)}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box .p-checkbox-icon{flex:0 0 auto;width:12px;height:12px}.cdk-drag-preview{z-index:10001!important;box-sizing:border-box;border-radius:4px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}::ng-deep .metric-tooltip .p-tooltip-text{font-size:11px;line-height:14px;letter-spacing:.3px;background:var(--cg-primary-color-hover)}\n"] }]
1065
- }], ctorParameters: function () { return [{ type: i1$2.ContextService }, { type: i1$2.QuoteDraftService }, { type: i0.ChangeDetectorRef }, { type: i1$2.MetricsCalculationService }, { type: i1$2.FlowConfigurationService }, { type: i2.ShoppingCartSettingsApiService }, { type: i1$2.RuntimeSettingsService }]; }, propDecorators: { overlayPanel: [{
1066
- type: ViewChild,
1067
- args: [OverlayPanel]
1068
- }] } });
732
+ FlowRouterService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, deps: [{ token: i1$2.Router }, { token: i1$2.ActivatedRoute }, { token: i2.ContextService }, { token: i2.IntegrationState }, { token: i2.FlowInfoService }, { token: i2.FlowStateService }], target: i0.ɵɵFactoryTarget.Injectable });
733
+ FlowRouterService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, providedIn: 'root' });
734
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRouterService, decorators: [{
735
+ type: Injectable,
736
+ args: [{ providedIn: 'root' }]
737
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: i1$2.ActivatedRoute }, { type: i2.ContextService }, { type: i2.IntegrationState }, { type: i2.FlowInfoService }, { type: i2.FlowStateService }]; } });
1069
738
 
1070
- class FlowHeaderComponent {
1071
- constructor(contextService, quoteDraftService, quoteApiService, sfApiService, flowConfiguration, routerService, dialogService, integrationState) {
1072
- this.contextService = contextService;
1073
- this.quoteDraftService = quoteDraftService;
1074
- this.quoteApiService = quoteApiService;
1075
- this.sfApiService = sfApiService;
1076
- this.flowConfiguration = flowConfiguration;
1077
- this.routerService = routerService;
1078
- this.dialogService = dialogService;
739
+ class FlowService {
740
+ constructor(integrationState, flowRouterService, quoteDraftService, configurationService, configurationStateService, flowDialogService, flowConfigurationService, flowInfoService, flowStateService) {
1079
741
  this.integrationState = integrationState;
1080
- this.isSaveInProgress$ = new BehaviorSubject(false);
1081
- this.isSubmitInProgress$ = new BehaviorSubject(false);
1082
- this.dateFormat = 'MM.dd.yyyy';
1083
- this.assetPriceLists$ = new BehaviorSubject([]);
1084
- this.mode$ = new BehaviorSubject(ConfigurationContextMode.TEST);
1085
- this.destroyed$ = new Subject();
1086
- this.contextProperties$ = this.contextService.resolve$().pipe(map(ctx => ctx.properties));
1087
- this.objectName$ = this.mode$.pipe(map(mode => mode?.toLocaleLowerCase() ?? ''));
1088
- this.activePriceList$ = this.quoteDraftService.activePriceList$;
1089
- this.isReadonlyMode$ = this.quoteDraftService.hasProducts$.pipe(map(() => this.isReadonlyMode()));
1090
- this.status$ = this.contextService.resolve$().pipe(map(context => context.properties.Status ?? ''));
1091
- this.isEditMode$ = this.quoteDraftService.isEditMode$();
1092
- this.assetsCount$ = this.getAssets$().pipe(map(lineItems => lineItems.length));
1093
- this.products$ = this.getDraftItems$().pipe(map(lineItems => this.generateProducts(lineItems)));
1094
- this.isConfigurationRoute$ = this.routerService.isConfigurationRoute$();
1095
- this.isCartRoute$ = this.routerService.isCartRoute$();
1096
- this.isCatalogRoute$ = this.routerService.isCatalogRoute$();
1097
- this.isAssetsRoute$ = this.routerService.isAssetsRoute$();
1098
- this.objectDetails$ = this.getObjectDetails$();
1099
- }
1100
- ngOnInit() {
1101
- this.contextService
1102
- .resolve$()
1103
- .pipe(takeUntil(this.destroyed$))
1104
- .subscribe(ctx => this.mode$.next(ctx.mode));
1105
- // update assetPriceLists on quoteDraft change in ACCOUNT mode
1106
- this.quoteDraftService.quoteDraft$
1107
- .pipe(takeUntil(this.destroyed$), filter(() => this.isAccountMode))
1108
- .subscribe(() => this.assetPriceLists$.next(this.quoteDraftService.assetPriceLists));
1109
- }
1110
- ngOnDestroy() {
1111
- this.destroyed$.next();
1112
- this.destroyed$.complete();
1113
- }
1114
- get isAccountMode() {
1115
- return this.mode$.value === ConfigurationContextMode.ACCOUNT;
1116
- }
1117
- get isQuoteMode() {
1118
- return this.mode$.value === ConfigurationContextMode.QUOTE;
1119
- }
1120
- get isAccountMode$() {
1121
- return this.mode$.pipe(map(mode => mode === ConfigurationContextMode.ACCOUNT));
1122
- }
1123
- get isQuoteMode$() {
1124
- return this.mode$.pipe(map(mode => mode === ConfigurationContextMode.QUOTE));
1125
- }
1126
- getSplitButtonActions(isCartRoute) {
1127
- return [
1128
- {
1129
- label: 'Save to Quote',
1130
- command: () => {
1131
- this.saveButtonClickHandler(isCartRoute);
1132
- },
1133
- },
1134
- {
1135
- label: 'Quick Save',
1136
- command: () => {
1137
- this.saveButtonClickHandler(isCartRoute, true);
1138
- },
1139
- },
1140
- {
1141
- label: 'Generate Document',
1142
- command: () => {
1143
- this.docGenButtonClickHandler(isCartRoute);
1144
- },
1145
- },
1146
- {
1147
- label: 'Submit For Approval',
1148
- command: () => {
1149
- this.submitButtonClickHandler(isCartRoute);
1150
- },
1151
- },
1152
- ];
1153
- }
1154
- back(objectId) {
1155
- const targetId = objectId ?? this.contextService.resolve().properties.Id;
1156
- if (targetId) {
1157
- window.VELO_BACK_FN.apply(null, [targetId]);
1158
- }
1159
- }
1160
- getSalesforceObjectLink(objectId) {
1161
- if (!objectId) {
1162
- return '';
1163
- }
1164
- return `${window.location.origin}/${objectId}`;
1165
- }
1166
- navigateToShoppingCart() {
1167
- if (!this.isReadonlyMode()) {
1168
- this.routerService.navigateToShoppingCart();
1169
- }
1170
- }
1171
- navigateToCatalog() {
1172
- if (!this.isReadonlyMode()) {
1173
- this.routerService.navigateToCatalog();
1174
- }
742
+ this.flowRouterService = flowRouterService;
743
+ this.quoteDraftService = quoteDraftService;
744
+ this.configurationService = configurationService;
745
+ this.configurationStateService = configurationStateService;
746
+ this.flowDialogService = flowDialogService;
747
+ this.flowConfigurationService = flowConfigurationService;
748
+ this.flowInfoService = flowInfoService;
749
+ this.flowStateService = flowStateService;
750
+ this.cleanup$ = new Subject();
1175
751
  }
1176
- navigateToAssets() {
1177
- if (!this.isReadonlyMode()) {
1178
- this.routerService.navigateToAssets();
1179
- }
752
+ cleanup() {
753
+ this.cleanup$.next();
1180
754
  }
1181
- docGenButtonClickHandler(isCartRoute) {
1182
- if (this.isAccountMode) {
1183
- if (!this.quoteDraftService.hasUnsavedChanges) {
1184
- this.dialogService.showAccountNoChangesDialog().subscribe();
755
+ initSubscriptions() {
756
+ this.integrationState
757
+ .listen$(FlowAction.FLOW_CONFIGURE_PRODUCT)
758
+ .pipe(switchMap(payload => {
759
+ if (this.flowInfoService.isLegacy) {
760
+ const productId = payload.productId ??
761
+ this.quoteDraftService.currentState.find(li => li.id === payload.lineItemId)?.productId;
762
+ return of({ ...payload, productId });
763
+ }
764
+ return this.prepareConfiguration$(payload.lineItemId).pipe(map(() => payload));
765
+ }), tap(payload => {
766
+ if (payload.productId) {
767
+ this.flowRouterService.navigateToProductConfiguration(payload.productId, payload.lineItemId);
1185
768
  }
1186
769
  else {
1187
- this.dialogService.showDocgenAccountUnsavedChangesDialog().subscribe();
770
+ console.warn("Parameter 'productId' is needed to start configuration");
1188
771
  }
1189
- return;
1190
- }
1191
- if (!this.quoteDraftService.isEditMode()) {
1192
- this.dialogService.showDocgenReadonlyDialog().subscribe();
1193
- return;
1194
- }
1195
- if (!isCartRoute) {
1196
- this.dialogService.showDocgenOutsideShoppingCartDialog().subscribe();
1197
- return;
1198
- }
1199
- let shouldOpen$ = of(true);
1200
- if (this.quoteDraftService.hasUnsavedChanges) {
1201
- shouldOpen$ = this.dialogService.showDocgenUnsavedChangesDialog().pipe(switchMap(confirmed => {
1202
- if (!confirmed) {
1203
- return of(false);
1204
- }
1205
- return this.saveQuote$(true).pipe(map(() => true));
1206
- }));
1207
- }
1208
- shouldOpen$
1209
- .pipe(first(), tap(shouldOpen => shouldOpen && this.integrationState.dispatch(FlowAction.OpenDocGenAction())))
772
+ }), takeUntil(this.cleanup$))
773
+ .subscribe();
774
+ this.integrationState
775
+ .listen$(FlowAction.FLOW_SWITCH_OBJECT)
776
+ .pipe(tap(payload => {
777
+ this.flowRouterService.switchObject(payload.id);
778
+ }), takeUntil(this.cleanup$))
779
+ .subscribe();
780
+ this.integrationState
781
+ .listen$(FlowAction.FLOW_NAVIGATE_BACK)
782
+ .pipe(switchMap(() => this.configurationStateService.cancelConfiguration()), tap(() => this.flowRouterService.navigateBack()), takeUntil(this.cleanup$))
783
+ .subscribe();
784
+ this.integrationState
785
+ .listen$(FlowAction.FLOW_NAVIGATE_TO)
786
+ .pipe(switchMap(payload => this.configurationStateService.cancelConfiguration().pipe(map(() => payload))), tap(payload => {
787
+ if (payload.path === 'product') {
788
+ this.integrationState.dispatch(FlowAction.ConfigureProductAction(payload));
789
+ }
790
+ else {
791
+ this.flowRouterService.navigateTo(payload.path, payload.productId, payload.lineItemId);
792
+ }
793
+ }), takeUntil(this.cleanup$))
1210
794
  .subscribe();
1211
- }
1212
- saveButtonClickHandler(isCartRoute, stayOnPage = false) {
1213
- if (!this.quoteDraftService.isEditMode()) {
1214
- if (this.isQuoteMode) {
1215
- this.dialogService.showQuoteReadonlyModeDialog().subscribe();
795
+ this.integrationState
796
+ .listen$(FlowAction.OPEN_DIALOG)
797
+ .pipe(switchMap(payload => this.flowDialogService.showDialog(payload.dialog).pipe(take(1))), takeUntil(this.cleanup$))
798
+ .subscribe(dialogResult => {
799
+ this.integrationState.patchState({ dialogResult });
800
+ });
801
+ this.integrationState
802
+ .listen$(FlowAction.FLOW_NAVIGATE_TO_CATALOG)
803
+ .pipe(tap(() => this.flowRouterService.navigateToCatalog()), takeUntil(this.cleanup$))
804
+ .subscribe();
805
+ this.integrationState
806
+ .listen$(FlowAction.FLOW_NAVIGATE_TO_SHOPPING_CART)
807
+ .pipe(tap(() => this.flowRouterService.navigateToShoppingCart()), takeUntil(this.cleanup$))
808
+ .subscribe();
809
+ this.integrationState
810
+ .listen$(FlowAction.FLOW_APPLY_PRODUCT_CONFIGURATION)
811
+ .pipe(switchMap(() => {
812
+ if (this.flowInfoService.isLegacy) {
813
+ return this.legacyApplyConfiguration();
1216
814
  }
1217
815
  else {
1218
- this.dialogService.showReadonlyModeDialog().subscribe();
816
+ return this.configurationStateService.saveConfiguration('', true).pipe(switchMap(() => this.configurationStateService.cancelConfiguration()), switchMap(() => this.flowStateService.dispatch$(UITemplateType.FLOW_ENGINE, 'MODIFY_ASSETS', {
817
+ addConfiguringAssetId: true,
818
+ enable: true,
819
+ })));
1219
820
  }
1220
- return;
1221
- }
1222
- if (this.isUpsellQuoteWithoutChanges()) {
1223
- this.dialogService.showAccountNoChangesDialog().subscribe();
1224
- return;
1225
- }
1226
- if (this.isAccountMode && !this.quoteDraftService.hasUnsavedChanges) {
1227
- this.dialogService.showAccountNoChangesDialog().subscribe();
1228
- return;
1229
- }
1230
- if (!isCartRoute) {
1231
- this.dialogService.showQuoteOutsideShoppingCartDialog().subscribe();
1232
- return;
1233
- }
1234
- const lineItems = this.flowConfiguration.getSnapshot();
1235
- if (!lineItems.length) {
1236
- this.dialogService.showEmptyCartDialog().subscribe();
1237
- return;
1238
- }
1239
- this.saveQuote$(stayOnPage).subscribe();
1240
- }
1241
- submitButtonClickHandler(isCartRoute) {
1242
- if (this.isQuoteMode && !this.quoteDraftService.isEditMode()) {
1243
- this.dialogService.showReadonlyQuoteSubmitFailureDialog();
1244
- return;
1245
- }
1246
- if (this.isUpsellQuoteWithoutChanges()) {
1247
- this.dialogService.showAccountNoChangesDialog().subscribe();
1248
- return;
1249
- }
1250
- if (this.isAccountMode) {
1251
- this.dialogService.showAccountSubmitFailureDialog();
1252
- return;
1253
- }
1254
- if (!isCartRoute) {
1255
- this.dialogService.showOutsideShoppingCartQuoteSubmitFailureDialog().subscribe();
1256
- return;
1257
- }
1258
- const quoteDraft = this.quoteDraftService.quoteDraft;
1259
- if (!quoteDraft) {
1260
- return;
1261
- }
1262
- this.isSubmitInProgress$.next(true);
1263
- this.quoteApiService
1264
- .submitQuote(quoteDraft)
1265
- .pipe(tap(({ quoteId }) => {
1266
- this.quoteDraftService.hasUnsavedChanges = false;
1267
- this.back(quoteId);
1268
- }), finalize(() => this.isSubmitInProgress$.next(false)), takeUntil(this.destroyed$))
821
+ }), tap(() => {
822
+ this.configurationService.hasUnsavedChanges = false;
823
+ this.flowRouterService.navigateToShoppingCart();
824
+ }), takeUntil(this.cleanup$))
1269
825
  .subscribe();
826
+ this.updateFlowParams();
1270
827
  }
1271
- selectPriceList(priceListId) {
1272
- this.quoteDraftService.updateActivePriceList(priceListId);
828
+ updateFlowParams() {
829
+ this.flowRouterService
830
+ .getFlowSubpath$()
831
+ .pipe(map(path => path.split('/')?.[0]), takeUntil(this.cleanup$))
832
+ .subscribe(flowPath => this.integrationState.patchState({ flowPath }));
833
+ this.flowRouterService.route$
834
+ .pipe(map(path => path.queryParams['productId']), takeUntil(this.cleanup$))
835
+ .subscribe(productId => this.integrationState.patchState({ productId }));
1273
836
  }
1274
- toggleCartOverlay(cart, event) {
1275
- event.stopPropagation();
1276
- if (!this.isReadonlyMode()) {
1277
- if (!cart.overlayPanel.overlayVisible) {
1278
- document.dispatchEvent(new Event('click'));
1279
- }
1280
- cart?.overlayPanel.toggle(event);
837
+ prepareConfiguration$(lineItemId) {
838
+ if (!lineItemId) {
839
+ return of(undefined);
1281
840
  }
841
+ return this.flowStateService.dispatch$(UITemplateType.FLOW_ENGINE, 'UPDATE_PRICE_LIST', {
842
+ lineItemId,
843
+ });
1282
844
  }
1283
- saveQuote$(stayOnPage = false) {
1284
- const quoteDraft = this.quoteDraftService.quoteDraftForActivePriceList;
1285
- if (!quoteDraft) {
845
+ legacyApplyConfiguration() {
846
+ const quoteDraft = this.quoteDraftService.quoteDraft;
847
+ const lineItem = this.configurationService.getSnapshot();
848
+ if (!quoteDraft || !lineItem) {
1286
849
  return of(undefined);
1287
850
  }
1288
- this.isSaveInProgress$.next(true);
1289
- return this.quoteApiService.upsertQuote(quoteDraft).pipe(tap(({ quoteId }) => {
1290
- this.quoteDraftService.hasUnsavedChanges = false;
1291
- if (!stayOnPage) {
1292
- this.back(quoteId);
1293
- }
1294
- }), finalize(() => this.isSaveInProgress$.next(false)), map(noop), takeUntil(this.destroyed$));
1295
- }
1296
- queryName$(objectName, id) {
1297
- if (!id) {
1298
- return of('');
851
+ const isNewLineItem = quoteDraft.currentState.every(li => li.id !== lineItem.id);
852
+ let updatedState;
853
+ if (isNewLineItem) {
854
+ updatedState = [...quoteDraft.currentState, lineItem];
1299
855
  }
1300
- const searchRequest = {
1301
- count: 1,
1302
- rawCondition: `Id = '${id}'`,
1303
- fields: ['Name'],
1304
- };
1305
- return this.sfApiService.query(searchRequest, objectName).pipe(map(result => result[0]?.Name ?? ''), takeUntil(this.destroyed$));
1306
- }
1307
- getObjectDetails$() {
1308
- return this.contextService.resolve$().pipe(distinctUntilChanged((prevCtx, currCtx) => prevCtx.properties.Id === currCtx.properties.Id), map(ctx => {
1309
- const isAccountMode = ctx.mode === ConfigurationContextMode.ACCOUNT;
1310
- const isQuoteMode = ctx.mode === ConfigurationContextMode.QUOTE;
1311
- return {
1312
- accountId: isAccountMode ? ctx.properties.Id : ctx.properties.AccountId,
1313
- opportunityId: ctx.properties.OpportunityId,
1314
- quoteId: isQuoteMode ? ctx.properties.Id : undefined,
1315
- quoteName: isQuoteMode ? ctx.properties.Name : undefined,
1316
- quoteNumber: isQuoteMode ? ctx.properties.QuoteNumber : undefined,
1317
- };
1318
- }), switchMap(details => combineLatest([
1319
- of(details),
1320
- this.queryName$('Account', details.accountId),
1321
- this.queryName$('Opportunity', details.opportunityId),
1322
- ])), map(([details, accountName, opportunityName]) => ({ ...details, accountName, opportunityName })));
1323
- }
1324
- formatMetric(value) {
1325
- return (value ? Number(value) : 0).toFixed(2);
1326
- }
1327
- getDraftItems$() {
1328
- return combineLatest([
1329
- this.flowConfiguration.get(),
1330
- this.integrationState.state$.pipe(map(state => state.modifiedAssets), distinctUntilChanged()),
1331
- ]).pipe(map(([lineItems, modifiedAssets]) => {
1332
- const currentTerms = lineItems.reduce((acc, li) => {
1333
- const assetId = li.assetId || li.openOrderLineItemId;
1334
- const isModified = isLineItemModified(li);
1335
- const currentTerm = getOriginParent(lineItems, li) ?? li;
1336
- if (isModified || (assetId && modifiedAssets?.[assetId])) {
1337
- acc.push(currentTerm);
1338
- }
1339
- return acc;
1340
- }, []);
1341
- const uniqCurrentTerms = currentTerms.filter((value, index, array) => array.indexOf(value) === index);
1342
- return uniqCurrentTerms;
1343
- }));
1344
- }
1345
- getAssets$() {
1346
- return this.flowConfiguration
1347
- .get()
1348
- .pipe(map(lineItems => lineItems.filter(li => (!!li.assetId || !!li.openOrderLineItemId) && !li.rampInstanceId)));
1349
- }
1350
- generateProducts(lineItems) {
1351
- const date = new Date();
1352
- date.setHours(0, 0, 0, 0);
1353
- const now = date.getTime();
1354
- const termParentIds = lineItems.map(li => li.rampInstanceId).filter(Boolean);
1355
- const lineItemsRampInstanceIdMap = lineItems.reduce((acc, li) => {
1356
- acc[li.rampInstanceId ?? ''] = li;
1357
- return acc;
1358
- }, {});
1359
- return lineItems.reduce((result, li) => {
1360
- // find main term line item
1361
- if (li.rampInstanceId) {
1362
- return result;
1363
- }
1364
- // find current term line item
1365
- let target = li;
1366
- while (target && target.endDate && new Date(target.endDate).getTime() <= now) {
1367
- target = lineItemsRampInstanceIdMap[li.id];
1368
- }
1369
- if (target && target.productId) {
1370
- result.push({
1371
- id: target.id,
1372
- productId: target.productId,
1373
- name: target.name,
1374
- configurable: target.properties['#configurable'] === 'true',
1375
- deleted: target.actionCode === 'DELETE',
1376
- hasTerm: termParentIds.includes(target.id),
1377
- qty: target.qty,
1378
- mrr: this.formatMetric(target.properties.VDM_Total_MRR),
1379
- nrr: this.formatMetric(target.properties.VDM_Total_NRR),
1380
- });
1381
- }
1382
- return result;
1383
- }, []);
1384
- }
1385
- isReadonlyMode() {
1386
- return (this.quoteDraftService.quoteDraft?.context.properties.mode === ConfigurationContextMode.ACCOUNT &&
1387
- !this.quoteDraftService.hasProducts);
1388
- }
1389
- isUpsellQuoteWithoutChanges() {
1390
- const noUpdates = this.quoteDraftService.activeCurrentState.every(lineItem => lineItem.actionCode === 'EXIST');
1391
- const sameNumberOfProducts = this.quoteDraftService.getInitialCurrentState().length === this.quoteDraftService.activeCurrentState.length;
1392
- return noUpdates && sameNumberOfProducts;
856
+ else {
857
+ updatedState = quoteDraft.currentState.map(li => (li.id === lineItem.id ? lineItem : li));
858
+ }
859
+ return this.flowConfigurationService.calculate$({ ...quoteDraft, currentState: updatedState });
1393
860
  }
1394
861
  }
1395
- FlowHeaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderComponent, deps: [{ token: i1$2.ContextService }, { token: i1$2.QuoteDraftService }, { token: i2.QuoteApiService }, { token: i2.SalesforceApiService }, { token: i1$2.FlowConfigurationService }, { token: FlowRouterService }, { token: FlowDialogService }, { token: i5.IntegrationState }], target: i0.ɵɵFactoryTarget.Component });
1396
- FlowHeaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FlowHeaderComponent, selector: "vl-flow-header", ngImport: i0, template: "<ng-container *ngIf=\"contextProperties$ | async as contextProperties\">\n <ng-container *ngIf=\"objectDetails$ | async as details\">\n <div class=\"flow-info\">\n <nav class=\"nav-item nav-back\" (click)=\"back()\">\n <i class=\"nav-icon vl-icon vl-icon-back\"></i>\n </nav>\n\n <ng-container *ngIf=\"isAccountMode$ | async\">\n <nav\n class=\"account-name nav-item\"\n [pTooltip]=\"contextProperties.Name ?? ''\"\n tooltipPosition=\"bottom\"\n [showDelay]=\"1000\"\n >\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n </nav>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode$ | async\">\n <span class=\"nav-item\">Quote #{{ details.quoteNumber }}</span>\n\n <span class=\"dot-separator\"></span>\n\n <nav class=\"nav-item\" (click)=\"quoteDetails.toggle($event)\">\n <span>{{ status$ | async }}</span>\n\n <i *ngIf=\"!quoteDetails.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"catalog-overlay quote-info-overlay\" #quoteDetails>\n <ng-template pTemplate>\n <div class=\"quote-info\">\n <div class=\"title\">\n <span>Quote Information</span>\n <span class=\"close-action\" (click)=\"quoteDetails.hide()\">\n <i class=\"vl-icon vl-cross\"></i>\n </span>\n </div>\n\n <div class=\"details\">\n <div class=\"details-row\">\n <div class=\"label\">Account Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n {{ details.accountName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Opportunity Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n {{ details.opportunityName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Quote Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n {{ details.quoteName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Quote Number</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n {{ details.quoteNumber }}\n </a>\n </div>\n </div>\n </div>\n </div>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n </div>\n\n <div class=\"flow-navigation\">\n <ng-container *vlLet=\"isReadonlyMode$ | async as isReadonlyMode\">\n <nav\n class=\"nav-item\"\n *vlLet=\"assetsCount$ | async as assetsCount\"\n [ngClass]=\"{ active: (isAssetsRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToAssets()\"\n >\n Assets\n <div *ngIf=\"assetsCount\" class=\"counter\">{{ assetsCount }}</div>\n </nav>\n <nav\n class=\"nav-item\"\n [ngClass]=\"{ active: (isCatalogRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToCatalog()\"\n >\n Catalog\n </nav>\n <nav\n *vlLet=\"isConfigurationRoute$ | async as isConfigurationRoute\"\n class=\"nav-item\"\n [ngClass]=\"{ active: isConfigurationRoute, disabled: !isConfigurationRoute }\"\n >\n Configurator\n </nav>\n <ng-container *vlLet=\"products$ | async as products\">\n <nav\n class=\"nav-item\"\n [ngClass]=\"{ active: (isCartRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToShoppingCart()\"\n >\n Shopping Cart\n <div *ngIf=\"products?.length\" class=\"counter\">{{ products?.length }}</div>\n\n <nav\n class=\"nav-popover-toggle\"\n [ngClass]=\"{ active: (isCartRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"toggleCartOverlay(cart, $event)\"\n >\n <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n </nav>\n\n <vl-cart-preview #cart [products]=\"products ?? []\"></vl-cart-preview>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"flow-controls\">\n <ng-container *ngIf=\"(isReadonlyMode$ | async) === false\">\n <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n <div class=\"price-plan\">\n <ng-container *vlLet=\"assetPriceLists$ | async as assetPriceLists\">\n <ng-container\n *ngIf=\"(isAccountMode$ | async) && assetPriceLists && assetPriceLists.length > 1; else singlePriceList\"\n >\n <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n <span>{{ priceList?.name }}</span>\n <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"price-list-overlay\" #priceListsOverlay>\n <ng-template pTemplate>\n <span\n *ngFor=\"let option of assetPriceLists\"\n class=\"price-list-option\"\n [class.active]=\"priceList?.id === option.id\"\n (click)=\"selectPriceList(option.id); priceListsOverlay.hide()\"\n >\n {{ option.name }}\n </span>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n </ng-container>\n\n <ng-template #singlePriceList>\n <div>{{ priceList?.name }}</div>\n </ng-template>\n\n <div *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date : dateFormat }}</div>\n </div>\n </ng-container>\n </ng-container>\n\n <vl-metrics></vl-metrics>\n\n <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n <ng-container *vlLet=\"isReadonlyMode$ | async as isReadonlyMode\">\n <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n <p-splitButton\n *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n #splitButton\n label=\"Save to Quote\"\n (onClick)=\"saveButtonClickHandler(isCartRoute === true)\"\n [model]=\"getSplitButtonActions(isCartRoute === true)\"\n [disabled]=\"isReadonlyMode === true\"\n styleClass=\"catalog-split-button p-button-outlined\"\n [class.opened]=\"splitButton?.menu?.visible\"\n icon=\"vl-icon vl-arrow-down\"\n [menuStyle]=\"{ width: '144px' }\"\n >\n </p-splitButton>\n\n <p-button\n *ngIf=\"isSaveInProgress\"\n class=\"save-button\"\n styleClass=\"p-button\"\n label=\"Saving\"\n [loading]=\"true\"\n ></p-button>\n\n <p-button\n *ngIf=\"isSubmitInProgress\"\n class=\"submit-button\"\n styleClass=\"p-button\"\n label=\"Submitting\"\n [loading]=\"true\"\n ></p-button>\n </ng-container>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n</ng-container>\n", styles: [":host{font-family:var(--cg-font-family);display:flex;align-items:center;height:48px;width:100%;background-color:var(--cg-primary-color);color:#fff;padding:0 16px;flex-shrink:0}:host *{font-family:var(--cg-font-family)}::ng-deep .p-overlaypanel.quote-info-overlay{margin-left:-16px}.quote-info{width:348px}.quote-info .title{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px;font-size:16px;font-style:normal;font-weight:500;line-height:20px}.quote-info .title .close-action{padding:4px}.quote-info .title .vl-icon{cursor:pointer}.quote-info .details .details-row{display:flex;align-items:center}.quote-info .details .details-row .label,.quote-info .details .details-row .value{flex:1;padding:8px;color:var(--cg-black)}.quote-info .details .details-row .label{font-size:12px;font-weight:500;line-height:16px;letter-spacing:.2px}.quote-info .details .details-row .value{text-align:right}.quote-info .details .details-row a{color:var(--cg-black);text-decoration:none;font-size:12px;line-height:18px;letter-spacing:.3px;border-bottom:1px solid var(--cg-black)}:host ::ng-deep .catalog-split-button .p-splitbutton-defaultbutton{padding:8px 0 8px 16px;border-top-left-radius:24px;border-bottom-left-radius:24px;color:var(--cg-white);background:var(--cg-primary-color);border-color:var(--cg-white)}:host ::ng-deep .catalog-split-button .p-splitbutton-defaultbutton .p-button-icon{display:none}:host ::ng-deep .catalog-split-button .p-splitbutton-menubutton{width:16px;box-sizing:content-box;padding:8px;border-top-right-radius:24px;border-bottom-right-radius:24px;border-left:none;color:var(--cg-white);background:var(--cg-primary-color);border-color:var(--cg-white)}:host ::ng-deep .catalog-split-button .p-splitbutton-menubutton:hover:enabled{border-left:none}:host ::ng-deep .catalog-split-button .p-button{line-height:16px}:host ::ng-deep .catalog-split-button .p-button-label{font-size:12px;font-weight:300}:host ::ng-deep .catalog-split-button:hover .p-splitbutton-defaultbutton,:host ::ng-deep .catalog-split-button:hover .p-splitbutton-menubutton{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button:hover .p-splitbutton-defaultbutton:hover,:host ::ng-deep .catalog-split-button:hover .p-splitbutton-menubutton:hover{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay{overflow:hidden;margin-top:4px;padding:0;border-radius:8px;box-shadow:0 8px 32px #0000001a}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem:not(:last-child){border-bottom:1px solid var(--cg-border-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link{padding:8px;color:var(--cg-primary-color);white-space:nowrap;font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link .p-menuitem-text{color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link:hover{background:var(--cg-bg-color)}:host ::ng-deep p-splitButton.opened .p-button-icon{transform:rotate(180deg)}:host ::ng-deep p-splitButton.opened .p-splitbutton-defaultbutton,:host ::ng-deep p-splitButton.opened .p-splitbutton-menubutton{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .save-button,:host ::ng-deep .submit-button{height:32px;width:136px;display:flex}:host ::ng-deep .save-button .p-button,:host ::ng-deep .submit-button .p-button{border-radius:24px;width:100%;height:100%;background:rgba(255,255,255,.3);opacity:1;border:none}:host ::ng-deep .save-button .p-button .p-button-label,:host ::ng-deep .submit-button .p-button .p-button-label{font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}.vl-icon{display:inline-block}.flow-info{flex-shrink:0;display:flex;gap:8px;align-items:center}.flow-info .nav-item{font-weight:400;font-size:12px;line-height:16px;letter-spacing:.3px}.flow-info .nav-popover-toggle{margin-left:-8px}.flow-info .nav-item:not(.disabled):hover,.flow-info .nav-popover-toggle:not(.disabled):hover{opacity:.6}nav{display:flex;align-items:center;cursor:pointer;height:32px}nav.disabled{opacity:.6;cursor:default}nav .icon-with-margin{margin:0 4px}nav a{color:#fff}nav.account-name{margin-left:4px;max-width:200px;overflow:hidden;text-overflow:ellipsis}nav.nav-popover-toggle{width:16px;display:flex;justify-content:center}nav.nav-popover-toggle i{pointer-events:none;margin:0}nav i{pointer-events:none}.metrics__row{display:flex;justify-content:space-between;gap:2px}.dot-separator:after{content:\"\";display:block;width:2px;height:2px;border-radius:50%;background:#fff}.slash-separator:after{content:\"\";display:block;background:#fff;width:1px;height:16px}.flow-navigation{margin-left:16px;margin-right:auto;display:flex;gap:8px;justify-content:center}.flow-navigation .nav-item{padding:8px 16px;border-radius:32px;color:#ffffff80;font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}.flow-navigation .nav-item:not(.disabled):hover:not(.active){color:var(--cg-white);background:rgba(255,255,255,.1)}.flow-navigation .nav-item:not(.disabled).active{color:var(--cg-white)}.flow-navigation .cart-nav-container{display:flex}.flow-navigation .nav-popover-toggle{margin-left:8px}.price-list-option{padding:8px;color:var(--vl-primary-color);cursor:pointer}.price-list-option.active,.price-list-option:hover{background:var(--vl-secondary-nav-bg)}.flow-controls{flex-shrink:0;display:flex;align-items:center;gap:8px;font-size:12px;line-height:16px}.flow-controls .price-plan{color:#ffffffb3}.counter{margin-left:8px;height:16px;min-width:16px;text-align:center;background:rgba(255,255,255,.2);border-radius:16px;padding:0 4px}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i7.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { kind: "directive", type: i8.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { kind: "directive", type: i2$1.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { kind: "directive", type: i10$1.Tooltip, selector: "[pTooltip]", inputs: ["tooltipPosition", "tooltipEvent", "appendTo", "positionStyle", "tooltipStyleClass", "tooltipZIndex", "escape", "showDelay", "hideDelay", "life", "positionTop", "positionLeft", "pTooltip", "tooltipDisabled", "tooltipOptions"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: CartPreviewComponent, selector: "vl-cart-preview", inputs: ["products"] }, { kind: "component", type: i13.SplitButton, selector: "p-splitButton", inputs: ["model", "icon", "iconPos", "label", "style", "styleClass", "menuStyle", "menuStyleClass", "disabled", "tabindex", "appendTo", "dir", "expandAriaLabel", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onClick", "onDropdownClick"] }, { kind: "component", type: MetricsComponent, selector: "vl-metrics" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }, { kind: "pipe", type: i4.DatePipe, name: "date" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1397
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderComponent, decorators: [{
1398
- type: Component,
1399
- args: [{ selector: 'vl-flow-header', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"contextProperties$ | async as contextProperties\">\n <ng-container *ngIf=\"objectDetails$ | async as details\">\n <div class=\"flow-info\">\n <nav class=\"nav-item nav-back\" (click)=\"back()\">\n <i class=\"nav-icon vl-icon vl-icon-back\"></i>\n </nav>\n\n <ng-container *ngIf=\"isAccountMode$ | async\">\n <nav\n class=\"account-name nav-item\"\n [pTooltip]=\"contextProperties.Name ?? ''\"\n tooltipPosition=\"bottom\"\n [showDelay]=\"1000\"\n >\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n </nav>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode$ | async\">\n <span class=\"nav-item\">Quote #{{ details.quoteNumber }}</span>\n\n <span class=\"dot-separator\"></span>\n\n <nav class=\"nav-item\" (click)=\"quoteDetails.toggle($event)\">\n <span>{{ status$ | async }}</span>\n\n <i *ngIf=\"!quoteDetails.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"catalog-overlay quote-info-overlay\" #quoteDetails>\n <ng-template pTemplate>\n <div class=\"quote-info\">\n <div class=\"title\">\n <span>Quote Information</span>\n <span class=\"close-action\" (click)=\"quoteDetails.hide()\">\n <i class=\"vl-icon vl-cross\"></i>\n </span>\n </div>\n\n <div class=\"details\">\n <div class=\"details-row\">\n <div class=\"label\">Account Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n {{ details.accountName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Opportunity Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n {{ details.opportunityName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Quote Name</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n {{ details.quoteName }}\n </a>\n </div>\n </div>\n <div class=\"details-row\">\n <div class=\"label\">Quote Number</div>\n <div class=\"value\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n {{ details.quoteNumber }}\n </a>\n </div>\n </div>\n </div>\n </div>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n </div>\n\n <div class=\"flow-navigation\">\n <ng-container *vlLet=\"isReadonlyMode$ | async as isReadonlyMode\">\n <nav\n class=\"nav-item\"\n *vlLet=\"assetsCount$ | async as assetsCount\"\n [ngClass]=\"{ active: (isAssetsRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToAssets()\"\n >\n Assets\n <div *ngIf=\"assetsCount\" class=\"counter\">{{ assetsCount }}</div>\n </nav>\n <nav\n class=\"nav-item\"\n [ngClass]=\"{ active: (isCatalogRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToCatalog()\"\n >\n Catalog\n </nav>\n <nav\n *vlLet=\"isConfigurationRoute$ | async as isConfigurationRoute\"\n class=\"nav-item\"\n [ngClass]=\"{ active: isConfigurationRoute, disabled: !isConfigurationRoute }\"\n >\n Configurator\n </nav>\n <ng-container *vlLet=\"products$ | async as products\">\n <nav\n class=\"nav-item\"\n [ngClass]=\"{ active: (isCartRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"navigateToShoppingCart()\"\n >\n Shopping Cart\n <div *ngIf=\"products?.length\" class=\"counter\">{{ products?.length }}</div>\n\n <nav\n class=\"nav-popover-toggle\"\n [ngClass]=\"{ active: (isCartRoute$ | async), disabled: isReadonlyMode }\"\n (click)=\"toggleCartOverlay(cart, $event)\"\n >\n <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n </nav>\n\n <vl-cart-preview #cart [products]=\"products ?? []\"></vl-cart-preview>\n </ng-container>\n </ng-container>\n </div>\n\n <div class=\"flow-controls\">\n <ng-container *ngIf=\"(isReadonlyMode$ | async) === false\">\n <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n <div class=\"price-plan\">\n <ng-container *vlLet=\"assetPriceLists$ | async as assetPriceLists\">\n <ng-container\n *ngIf=\"(isAccountMode$ | async) && assetPriceLists && assetPriceLists.length > 1; else singlePriceList\"\n >\n <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n <span>{{ priceList?.name }}</span>\n <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-arrow-down icon-with-margin\"></i>\n <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-arrow-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"price-list-overlay\" #priceListsOverlay>\n <ng-template pTemplate>\n <span\n *ngFor=\"let option of assetPriceLists\"\n class=\"price-list-option\"\n [class.active]=\"priceList?.id === option.id\"\n (click)=\"selectPriceList(option.id); priceListsOverlay.hide()\"\n >\n {{ option.name }}\n </span>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n </ng-container>\n\n <ng-template #singlePriceList>\n <div>{{ priceList?.name }}</div>\n </ng-template>\n\n <div *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date : dateFormat }}</div>\n </div>\n </ng-container>\n </ng-container>\n\n <vl-metrics></vl-metrics>\n\n <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n <ng-container *vlLet=\"isReadonlyMode$ | async as isReadonlyMode\">\n <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n <p-splitButton\n *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n #splitButton\n label=\"Save to Quote\"\n (onClick)=\"saveButtonClickHandler(isCartRoute === true)\"\n [model]=\"getSplitButtonActions(isCartRoute === true)\"\n [disabled]=\"isReadonlyMode === true\"\n styleClass=\"catalog-split-button p-button-outlined\"\n [class.opened]=\"splitButton?.menu?.visible\"\n icon=\"vl-icon vl-arrow-down\"\n [menuStyle]=\"{ width: '144px' }\"\n >\n </p-splitButton>\n\n <p-button\n *ngIf=\"isSaveInProgress\"\n class=\"save-button\"\n styleClass=\"p-button\"\n label=\"Saving\"\n [loading]=\"true\"\n ></p-button>\n\n <p-button\n *ngIf=\"isSubmitInProgress\"\n class=\"submit-button\"\n styleClass=\"p-button\"\n label=\"Submitting\"\n [loading]=\"true\"\n ></p-button>\n </ng-container>\n </ng-container>\n </ng-container>\n </ng-container>\n </div>\n </ng-container>\n</ng-container>\n", styles: [":host{font-family:var(--cg-font-family);display:flex;align-items:center;height:48px;width:100%;background-color:var(--cg-primary-color);color:#fff;padding:0 16px;flex-shrink:0}:host *{font-family:var(--cg-font-family)}::ng-deep .p-overlaypanel.quote-info-overlay{margin-left:-16px}.quote-info{width:348px}.quote-info .title{display:flex;justify-content:space-between;align-items:center;margin-bottom:8px;font-size:16px;font-style:normal;font-weight:500;line-height:20px}.quote-info .title .close-action{padding:4px}.quote-info .title .vl-icon{cursor:pointer}.quote-info .details .details-row{display:flex;align-items:center}.quote-info .details .details-row .label,.quote-info .details .details-row .value{flex:1;padding:8px;color:var(--cg-black)}.quote-info .details .details-row .label{font-size:12px;font-weight:500;line-height:16px;letter-spacing:.2px}.quote-info .details .details-row .value{text-align:right}.quote-info .details .details-row a{color:var(--cg-black);text-decoration:none;font-size:12px;line-height:18px;letter-spacing:.3px;border-bottom:1px solid var(--cg-black)}:host ::ng-deep .catalog-split-button .p-splitbutton-defaultbutton{padding:8px 0 8px 16px;border-top-left-radius:24px;border-bottom-left-radius:24px;color:var(--cg-white);background:var(--cg-primary-color);border-color:var(--cg-white)}:host ::ng-deep .catalog-split-button .p-splitbutton-defaultbutton .p-button-icon{display:none}:host ::ng-deep .catalog-split-button .p-splitbutton-menubutton{width:16px;box-sizing:content-box;padding:8px;border-top-right-radius:24px;border-bottom-right-radius:24px;border-left:none;color:var(--cg-white);background:var(--cg-primary-color);border-color:var(--cg-white)}:host ::ng-deep .catalog-split-button .p-splitbutton-menubutton:hover:enabled{border-left:none}:host ::ng-deep .catalog-split-button .p-button{line-height:16px}:host ::ng-deep .catalog-split-button .p-button-label{font-size:12px;font-weight:300}:host ::ng-deep .catalog-split-button:hover .p-splitbutton-defaultbutton,:host ::ng-deep .catalog-split-button:hover .p-splitbutton-menubutton{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button:hover .p-splitbutton-defaultbutton:hover,:host ::ng-deep .catalog-split-button:hover .p-splitbutton-menubutton:hover{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay{overflow:hidden;margin-top:4px;padding:0;border-radius:8px;box-shadow:0 8px 32px #0000001a}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem:not(:last-child){border-bottom:1px solid var(--cg-border-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link{padding:8px;color:var(--cg-primary-color);white-space:nowrap;font-size:12px;font-weight:300;line-height:16px;letter-spacing:.3px}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link .p-menuitem-text{color:var(--cg-primary-color)}:host ::ng-deep .catalog-split-button .p-menu-overlay .p-menuitem-link:hover{background:var(--cg-bg-color)}:host ::ng-deep p-splitButton.opened .p-button-icon{transform:rotate(180deg)}:host ::ng-deep p-splitButton.opened .p-splitbutton-defaultbutton,:host ::ng-deep p-splitButton.opened .p-splitbutton-menubutton{color:var(--cg-primary-color);background:var(--cg-white);border-color:var(--cg-primary-color)}:host ::ng-deep .save-button,:host ::ng-deep .submit-button{height:32px;width:136px;display:flex}:host ::ng-deep .save-button .p-button,:host ::ng-deep .submit-button .p-button{border-radius:24px;width:100%;height:100%;background:rgba(255,255,255,.3);opacity:1;border:none}:host ::ng-deep .save-button .p-button .p-button-label,:host ::ng-deep .submit-button .p-button .p-button-label{font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}.vl-icon{display:inline-block}.flow-info{flex-shrink:0;display:flex;gap:8px;align-items:center}.flow-info .nav-item{font-weight:400;font-size:12px;line-height:16px;letter-spacing:.3px}.flow-info .nav-popover-toggle{margin-left:-8px}.flow-info .nav-item:not(.disabled):hover,.flow-info .nav-popover-toggle:not(.disabled):hover{opacity:.6}nav{display:flex;align-items:center;cursor:pointer;height:32px}nav.disabled{opacity:.6;cursor:default}nav .icon-with-margin{margin:0 4px}nav a{color:#fff}nav.account-name{margin-left:4px;max-width:200px;overflow:hidden;text-overflow:ellipsis}nav.nav-popover-toggle{width:16px;display:flex;justify-content:center}nav.nav-popover-toggle i{pointer-events:none;margin:0}nav i{pointer-events:none}.metrics__row{display:flex;justify-content:space-between;gap:2px}.dot-separator:after{content:\"\";display:block;width:2px;height:2px;border-radius:50%;background:#fff}.slash-separator:after{content:\"\";display:block;background:#fff;width:1px;height:16px}.flow-navigation{margin-left:16px;margin-right:auto;display:flex;gap:8px;justify-content:center}.flow-navigation .nav-item{padding:8px 16px;border-radius:32px;color:#ffffff80;font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}.flow-navigation .nav-item:not(.disabled):hover:not(.active){color:var(--cg-white);background:rgba(255,255,255,.1)}.flow-navigation .nav-item:not(.disabled).active{color:var(--cg-white)}.flow-navigation .cart-nav-container{display:flex}.flow-navigation .nav-popover-toggle{margin-left:8px}.price-list-option{padding:8px;color:var(--vl-primary-color);cursor:pointer}.price-list-option.active,.price-list-option:hover{background:var(--vl-secondary-nav-bg)}.flow-controls{flex-shrink:0;display:flex;align-items:center;gap:8px;font-size:12px;line-height:16px}.flow-controls .price-plan{color:#ffffffb3}.counter{margin-left:8px;height:16px;min-width:16px;text-align:center;background:rgba(255,255,255,.2);border-radius:16px;padding:0 4px}\n"] }]
1400
- }], ctorParameters: function () { return [{ type: i1$2.ContextService }, { type: i1$2.QuoteDraftService }, { type: i2.QuoteApiService }, { type: i2.SalesforceApiService }, { type: i1$2.FlowConfigurationService }, { type: FlowRouterService }, { type: FlowDialogService }, { type: i5.IntegrationState }]; } });
1401
-
1402
- class MetricsModule {
1403
- }
1404
- MetricsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MetricsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
1405
- MetricsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: MetricsModule, declarations: [MetricsComponent], imports: [CommonModule,
1406
- FormsModule,
1407
- ReactiveFormsModule,
1408
- DragDropModule,
1409
- ButtonModule,
1410
- OverlayPanelModule,
1411
- SidebarModule,
1412
- InputTextModule,
1413
- CheckboxModule,
1414
- LetDirectiveModule,
1415
- VirtualScrollerModule,
1416
- ErrorTooltipModule,
1417
- HiddenTextTooltipModule,
1418
- SdkPipesModule], exports: [MetricsComponent] });
1419
- MetricsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MetricsModule, imports: [CommonModule,
1420
- FormsModule,
1421
- ReactiveFormsModule,
1422
- DragDropModule,
1423
- ButtonModule,
1424
- OverlayPanelModule,
1425
- SidebarModule,
1426
- InputTextModule,
1427
- CheckboxModule,
1428
- LetDirectiveModule,
1429
- VirtualScrollerModule,
1430
- ErrorTooltipModule,
1431
- HiddenTextTooltipModule,
1432
- SdkPipesModule] });
1433
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: MetricsModule, decorators: [{
1434
- type: NgModule,
1435
- args: [{
1436
- declarations: [MetricsComponent],
1437
- imports: [
1438
- CommonModule,
1439
- FormsModule,
1440
- ReactiveFormsModule,
1441
- DragDropModule,
1442
- ButtonModule,
1443
- OverlayPanelModule,
1444
- SidebarModule,
1445
- InputTextModule,
1446
- CheckboxModule,
1447
- LetDirectiveModule,
1448
- VirtualScrollerModule,
1449
- ErrorTooltipModule,
1450
- HiddenTextTooltipModule,
1451
- SdkPipesModule,
1452
- ],
1453
- exports: [MetricsComponent],
1454
- }]
1455
- }] });
1456
-
1457
- class FlowHeaderModule {
1458
- }
1459
- FlowHeaderModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
1460
- FlowHeaderModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderModule, declarations: [FlowHeaderComponent], imports: [CommonModule,
1461
- OverlayPanelModule,
1462
- LetDirectiveModule,
1463
- TooltipModule,
1464
- ButtonModule,
1465
- CartPreviewModule,
1466
- SplitButtonModule,
1467
- MetricsModule,
1468
- SdkPipesModule], exports: [FlowHeaderComponent] });
1469
- FlowHeaderModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderModule, imports: [CommonModule,
1470
- OverlayPanelModule,
1471
- LetDirectiveModule,
1472
- TooltipModule,
1473
- ButtonModule,
1474
- CartPreviewModule,
1475
- SplitButtonModule,
1476
- MetricsModule,
1477
- SdkPipesModule] });
1478
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowHeaderModule, decorators: [{
1479
- type: NgModule,
1480
- args: [{
1481
- declarations: [FlowHeaderComponent],
1482
- imports: [
1483
- CommonModule,
1484
- OverlayPanelModule,
1485
- LetDirectiveModule,
1486
- TooltipModule,
1487
- ButtonModule,
1488
- CartPreviewModule,
1489
- SplitButtonModule,
1490
- MetricsModule,
1491
- SdkPipesModule,
1492
- ],
1493
- exports: [FlowHeaderComponent],
1494
- }]
1495
- }] });
862
+ FlowService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService, deps: [{ token: i5.IntegrationState }, { token: FlowRouterService }, { token: i2.QuoteDraftService }, { token: i2.ConfigurationService }, { token: i2.ConfigurationStateService }, { token: FlowDialogService }, { token: i2.FlowConfigurationService }, { token: i2.FlowInfoService }, { token: i2.FlowStateService }], target: i0.ɵɵFactoryTarget.Injectable });
863
+ FlowService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService });
864
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowService, decorators: [{
865
+ type: Injectable
866
+ }], ctorParameters: function () { return [{ type: i5.IntegrationState }, { type: FlowRouterService }, { type: i2.QuoteDraftService }, { type: i2.ConfigurationService }, { type: i2.ConfigurationStateService }, { type: FlowDialogService }, { type: i2.FlowConfigurationService }, { type: i2.FlowInfoService }, { type: i2.FlowStateService }]; } });
1496
867
 
1497
- class FlowDocGenService {
868
+ class FlowGuidedSellingService {
1498
869
  constructor(integrationState) {
1499
870
  this.integrationState = integrationState;
1500
871
  this.cleanup$ = new Subject();
@@ -1507,53 +878,40 @@ class FlowDocGenService {
1507
878
  }
1508
879
  initSubscriptions() {
1509
880
  this.integrationState
1510
- .listen$(FlowAction.FLOW_OPEN_DOC_GEN)
881
+ .listen$(FlowAction.OPEN_GUIDED_SELLING)
1511
882
  .pipe(tap(() => this.isVisibleSubj$.next(true)), takeUntil(this.cleanup$))
1512
883
  .subscribe();
1513
884
  this.integrationState
1514
- .listen$(FlowAction.FLOW_CLOSE_DOC_GEN)
885
+ .listen$(FlowAction.CLOSE_GUIDED_SELLING)
1515
886
  .pipe(tap(() => this.isVisibleSubj$.next(false)), takeUntil(this.cleanup$))
1516
887
  .subscribe();
1517
888
  }
1518
889
  }
1519
- FlowDocGenService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService, deps: [{ token: i5.IntegrationState }], target: i0.ɵɵFactoryTarget.Injectable });
1520
- FlowDocGenService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService });
1521
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService, decorators: [{
890
+ FlowGuidedSellingService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService, deps: [{ token: i2.IntegrationState }], target: i0.ɵɵFactoryTarget.Injectable });
891
+ FlowGuidedSellingService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService });
892
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService, decorators: [{
1522
893
  type: Injectable
1523
- }], ctorParameters: function () { return [{ type: i5.IntegrationState }]; } });
894
+ }], ctorParameters: function () { return [{ type: i2.IntegrationState }]; } });
1524
895
 
1525
896
  class FlowComponent {
1526
- constructor(routerService, quoteDraftService, flowService, docGenService, integrationState, contextService) {
897
+ constructor(routerService, flowService, flowInfo, guidedSellingService) {
1527
898
  this.routerService = routerService;
1528
- this.quoteDraftService = quoteDraftService;
1529
899
  this.flowService = flowService;
1530
- this.docGenService = docGenService;
1531
- this.integrationState = integrationState;
1532
- this.contextService = contextService;
900
+ this.flowInfo = flowInfo;
901
+ this.guidedSellingService = guidedSellingService;
1533
902
  this.isLoading$ = this.routerService.loading$;
1534
- this.showHeader$ = combineLatest([
1535
- this.routerService.route$,
1536
- this.quoteDraftService.isStandalone$,
1537
- this.contextService.isInitialized$,
1538
- ]).pipe(map(([route, isStandalone, isContextInited]) => !!isContextInited && this.quoteDraftService.isInitialized && route.data['showHeader'] && !isStandalone));
1539
- this.isStandalone$ = this.quoteDraftService.isStandalone$;
903
+ this.showHeader$ = combineLatest([this.routerService.route$, this.flowInfo.flow$]).pipe(map(([route, flow]) => route.data['showHeader'] && !flow?.properties.standalone));
904
+ this.isStandalone$ = this.flowInfo.flow$.pipe(map(flow => Boolean(flow?.properties.standalone)));
905
+ this.guidedSellingVisible$ = this.guidedSellingService.isVisible$;
1540
906
  this.flowService.initSubscriptions();
1541
907
  }
1542
- ngOnInit() {
1543
- this.integrationState.clear();
1544
- }
1545
- ngOnDestroy() {
1546
- this.flowService.cleanup();
1547
- this.docGenService.cleanup();
1548
- this.contextService.delete();
1549
- }
1550
908
  }
1551
- FlowComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowComponent, deps: [{ token: FlowRouterService }, { token: i1$2.QuoteDraftService }, { token: FlowService }, { token: FlowDocGenService }, { token: i5.IntegrationState }, { token: i1$2.ContextService }], target: i0.ɵɵFactoryTarget.Component });
1552
- FlowComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FlowComponent, selector: "vl-flow", ngImport: i0, template: "<vl-flow-header *ngIf=\"showHeader$ | async\"></vl-flow-header>\n\n<div class=\"flow-content\">\n <div class=\"loading-overlay\" *ngIf=\"isLoading$ | async\">\n <vl-loader label=\"LOADING\"></vl-loader>\n </div>\n\n <router-outlet></router-outlet>\n\n <vl-flow-guided-selling *ngIf=\"(isStandalone$ | async) !== true\"></vl-flow-guided-selling>\n</div>\n\n<ng-container *ngIf=\"(isStandalone$ | async) !== true\">\n <vl-flow-doc-gen></vl-flow-doc-gen>\n</ng-container>\n", styles: [":host{display:flex;flex-direction:column;width:100%;height:100%}.flow-content{flex-grow:1;position:relative;overflow:hidden}.loading-overlay{position:absolute;height:100%;width:100%;inset:0;background-color:#fff;z-index:999}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1$1.RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "component", type: FlowHeaderComponent, selector: "vl-flow-header" }, { kind: "component", type: DocGenComponent, selector: "vl-flow-doc-gen" }, { kind: "component", type: GuidedSellingComponent, selector: "vl-flow-guided-selling" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
909
+ FlowComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowComponent, deps: [{ token: FlowRouterService }, { token: FlowService }, { token: i2.FlowInfoService }, { token: FlowGuidedSellingService }], target: i0.ɵɵFactoryTarget.Component });
910
+ FlowComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: FlowComponent, selector: "vl-flow", ngImport: i0, template: "<vl-flow-new-header *ngIf=\"showHeader$ | async\"></vl-flow-new-header>\n\n<div class=\"flow-content\">\n <div class=\"loading-overlay\" *ngIf=\"isLoading$ | async\">\n <vl-loader label=\"LOADING\"></vl-loader>\n </div>\n\n <router-outlet></router-outlet>\n\n <div class=\"guided-selling\" [ngClass]=\"{ hidden: (guidedSellingVisible$ | async) === false }\">\n <vl-flow-guided-selling *ngIf=\"(isStandalone$ | async) !== true\"></vl-flow-guided-selling>\n </div>\n</div>\n\n<ng-container *ngIf=\"(isStandalone$ | async) !== true\">\n <vl-flow-doc-gen></vl-flow-doc-gen>\n</ng-container>\n", styles: [":host{display:flex;flex-direction:column;width:100%;height:100%}.flow-content{flex-grow:1;position:relative;overflow:hidden}.loading-overlay{position:absolute;height:100%;width:100%;inset:0;background-color:#fff;z-index:999}.guided-selling{position:absolute;top:0;width:100%;height:100%;z-index:100}.hidden{display:none}\n"], dependencies: [{ kind: "directive", type: i4.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1$2.RouterOutlet, selector: "router-outlet", inputs: ["name"], outputs: ["activate", "deactivate", "attach", "detach"], exportAs: ["outlet"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "component", type: FlowHeaderComponent, selector: "vl-flow-new-header" }, { kind: "component", type: DocGenComponent, selector: "vl-flow-doc-gen" }, { kind: "component", type: GuidedSellingComponent, selector: "vl-flow-guided-selling" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1553
911
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowComponent, decorators: [{
1554
912
  type: Component,
1555
- args: [{ selector: 'vl-flow', changeDetection: ChangeDetectionStrategy.OnPush, template: "<vl-flow-header *ngIf=\"showHeader$ | async\"></vl-flow-header>\n\n<div class=\"flow-content\">\n <div class=\"loading-overlay\" *ngIf=\"isLoading$ | async\">\n <vl-loader label=\"LOADING\"></vl-loader>\n </div>\n\n <router-outlet></router-outlet>\n\n <vl-flow-guided-selling *ngIf=\"(isStandalone$ | async) !== true\"></vl-flow-guided-selling>\n</div>\n\n<ng-container *ngIf=\"(isStandalone$ | async) !== true\">\n <vl-flow-doc-gen></vl-flow-doc-gen>\n</ng-container>\n", styles: [":host{display:flex;flex-direction:column;width:100%;height:100%}.flow-content{flex-grow:1;position:relative;overflow:hidden}.loading-overlay{position:absolute;height:100%;width:100%;inset:0;background-color:#fff;z-index:999}\n"] }]
1556
- }], ctorParameters: function () { return [{ type: FlowRouterService }, { type: i1$2.QuoteDraftService }, { type: FlowService }, { type: FlowDocGenService }, { type: i5.IntegrationState }, { type: i1$2.ContextService }]; } });
913
+ args: [{ selector: 'vl-flow', changeDetection: ChangeDetectionStrategy.OnPush, template: "<vl-flow-new-header *ngIf=\"showHeader$ | async\"></vl-flow-new-header>\n\n<div class=\"flow-content\">\n <div class=\"loading-overlay\" *ngIf=\"isLoading$ | async\">\n <vl-loader label=\"LOADING\"></vl-loader>\n </div>\n\n <router-outlet></router-outlet>\n\n <div class=\"guided-selling\" [ngClass]=\"{ hidden: (guidedSellingVisible$ | async) === false }\">\n <vl-flow-guided-selling *ngIf=\"(isStandalone$ | async) !== true\"></vl-flow-guided-selling>\n </div>\n</div>\n\n<ng-container *ngIf=\"(isStandalone$ | async) !== true\">\n <vl-flow-doc-gen></vl-flow-doc-gen>\n</ng-container>\n", styles: [":host{display:flex;flex-direction:column;width:100%;height:100%}.flow-content{flex-grow:1;position:relative;overflow:hidden}.loading-overlay{position:absolute;height:100%;width:100%;inset:0;background-color:#fff;z-index:999}.guided-selling{position:absolute;top:0;width:100%;height:100%;z-index:100}.hidden{display:none}\n"] }]
914
+ }], ctorParameters: function () { return [{ type: FlowRouterService }, { type: FlowService }, { type: i2.FlowInfoService }, { type: FlowGuidedSellingService }]; } });
1557
915
 
1558
916
  const getFlowObjectIdPropertyName = (id) => {
1559
917
  const objectName = SalesforceIdUtils.getSfObjectNameById(id);
@@ -1593,7 +951,7 @@ class ContextGuard {
1593
951
  const mode = this.getConfigurationContextMode(accountId, quoteId, orderId, rpcMessage);
1594
952
  // Restrict if mode is not defined
1595
953
  if (mode == null) {
1596
- return this.handleError(route, 'Mode is undefined');
954
+ return this.routerService.showErrorPage$('Mode is undefined');
1597
955
  }
1598
956
  const headerId = accountId || quoteId || orderId || this.rpcMessageId || 'empty-for-test-mode';
1599
957
  // Allow if context is already initialized with the same headerId
@@ -1605,7 +963,6 @@ class ContextGuard {
1605
963
  }
1606
964
  // Initialize context and runtime settings
1607
965
  return forkJoin([this.contextService.create(headerId, mode), this.runtimeSettingsService.create()]).pipe(tap(([context]) => {
1608
- // Update context with queryParams
1609
966
  this.contextService.update({
1610
967
  ...context,
1611
968
  properties: {
@@ -1618,7 +975,8 @@ class ContextGuard {
1618
975
  this.runtimeSettingsService.initCurrency(context.properties['CurrencyIsoCode']);
1619
976
  }), map(() => true), catchError(e => {
1620
977
  const message = e instanceof HttpErrorResponse ? e.error.message : e;
1621
- return this.handleError(route, message);
978
+ const errorDetails = isVeloceError(e.error) ? extractErrorDetails(e.error) : [];
979
+ return this.routerService.showErrorPage$(message, errorDetails);
1622
980
  }));
1623
981
  }
1624
982
  canActivate(route) {
@@ -1651,36 +1009,42 @@ class ContextGuard {
1651
1009
  }
1652
1010
  return;
1653
1011
  }
1654
- handleError(route, message) {
1655
- this.contextService.delete();
1656
- const parentUrl = this.routerService.getFlowRootPath(route);
1657
- return from(this.router.navigate([parentUrl, '404'], { state: { message } })).pipe(map(() => false));
1658
- }
1659
1012
  }
1660
- ContextGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ContextGuard, deps: [{ token: i1$1.Router }, { token: FlowRouterService }, { token: i1$2.ContextService }, { token: i1$2.RuntimeSettingsService }], target: i0.ɵɵFactoryTarget.Injectable });
1013
+ ContextGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ContextGuard, deps: [{ token: i1$2.Router }, { token: FlowRouterService }, { token: i2.ContextService }, { token: i2.RuntimeSettingsService }], target: i0.ɵɵFactoryTarget.Injectable });
1661
1014
  ContextGuard.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ContextGuard });
1662
1015
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ContextGuard, decorators: [{
1663
1016
  type: Injectable
1664
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: FlowRouterService }, { type: i1$2.ContextService }, { type: i1$2.RuntimeSettingsService }]; } });
1017
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: FlowRouterService }, { type: i2.ContextService }, { type: i2.RuntimeSettingsService }]; } });
1665
1018
 
1666
- const initFlow = (route) => {
1019
+ const keepFlowInitialized = (route) => {
1667
1020
  const contextService = inject(ContextService);
1668
- const flowService = inject(FlowService);
1021
+ const flowState = inject(FlowStateService);
1022
+ const flowInfoService = inject(FlowInfoService);
1023
+ const routerService = inject(FlowRouterService);
1669
1024
  const quoteDraft = inject(QuoteDraftService);
1670
1025
  const configurationService = inject(ConfigurationService);
1026
+ const integrationState = inject(IntegrationState);
1671
1027
  const { flowId } = route.queryParams;
1672
- const flow = flowService.flow;
1028
+ const flow = flowInfoService.flow;
1673
1029
  if (flow && flow?.id === flowId) {
1674
1030
  return true;
1675
1031
  }
1676
1032
  // Cleanup
1677
- contextService.delete();
1678
- configurationService.reset();
1033
+ flowState.cleanup();
1679
1034
  quoteDraft.reset();
1035
+ configurationService.reset();
1036
+ integrationState.clear();
1037
+ flowInfoService.cleanup();
1038
+ contextService.delete();
1680
1039
  if (!flowId) {
1681
1040
  return true;
1682
1041
  }
1683
- return flowService.setFlowById$(flowId);
1042
+ const params = { ...route.queryParams, ...flow?.properties.queryParams };
1043
+ return flowInfoService.init$(flowId, params).pipe(map(() => true), catchError(e => {
1044
+ const message = e instanceof HttpErrorResponse ? e.error.message : e;
1045
+ const errorDetails = isVeloceError(e.error) ? extractErrorDetails(e.error) : [];
1046
+ return routerService.showErrorPage$(message, errorDetails);
1047
+ }));
1684
1048
  };
1685
1049
 
1686
1050
  class ProductUnloadGuard {
@@ -1717,11 +1081,11 @@ class ProductUnloadGuard {
1717
1081
  }));
1718
1082
  }
1719
1083
  }
1720
- ProductUnloadGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductUnloadGuard, deps: [{ token: i1$1.Router }, { token: i1$2.ContextService }, { token: i1$2.QuoteDraftService }, { token: i1$2.ConfigurationService }, { token: FlowDialogService }], target: i0.ɵɵFactoryTarget.Injectable });
1084
+ ProductUnloadGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductUnloadGuard, deps: [{ token: i1$2.Router }, { token: i2.ContextService }, { token: i2.QuoteDraftService }, { token: i2.ConfigurationService }, { token: FlowDialogService }], target: i0.ɵɵFactoryTarget.Injectable });
1721
1085
  ProductUnloadGuard.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductUnloadGuard });
1722
1086
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductUnloadGuard, decorators: [{
1723
1087
  type: Injectable
1724
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: i1$2.ContextService }, { type: i1$2.QuoteDraftService }, { type: i1$2.ConfigurationService }, { type: FlowDialogService }]; } });
1088
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: i2.ContextService }, { type: i2.QuoteDraftService }, { type: i2.ConfigurationService }, { type: FlowDialogService }]; } });
1725
1089
 
1726
1090
  class RootGuard {
1727
1091
  constructor(router, routerService) {
@@ -1756,25 +1120,25 @@ class RootGuard {
1756
1120
  return true;
1757
1121
  }
1758
1122
  }
1759
- RootGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RootGuard, deps: [{ token: i1$1.Router }, { token: FlowRouterService }], target: i0.ɵɵFactoryTarget.Injectable });
1123
+ RootGuard.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RootGuard, deps: [{ token: i1$2.Router }, { token: FlowRouterService }], target: i0.ɵɵFactoryTarget.Injectable });
1760
1124
  RootGuard.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RootGuard, providedIn: 'root' });
1761
1125
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RootGuard, decorators: [{
1762
1126
  type: Injectable,
1763
1127
  args: [{ providedIn: 'root' }]
1764
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: FlowRouterService }]; } });
1128
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: FlowRouterService }]; } });
1765
1129
 
1766
1130
  class AssetsComponent {
1767
- constructor(templatesApi, cdr, toastService, flowService, customizationService) {
1131
+ constructor(templatesApi, cdr, toastService, flowInfo, customizationService) {
1768
1132
  this.templatesApi = templatesApi;
1769
1133
  this.cdr = cdr;
1770
1134
  this.toastService = toastService;
1771
- this.flowService = flowService;
1135
+ this.flowInfo = flowInfo;
1772
1136
  this.customizationService = customizationService;
1773
1137
  this.uiDefinition = undefined;
1774
1138
  this.state$ = new BehaviorSubject({ loading: true, failure: false });
1775
1139
  this.templateApiName = '';
1776
1140
  this.destroyed$ = new Subject();
1777
- this.templateApiName = this.flowService.flow?.properties.templates.assets ?? '';
1141
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.assets ?? '';
1778
1142
  }
1779
1143
  ngOnInit() {
1780
1144
  this.generateUIDefinition$()
@@ -1797,37 +1161,36 @@ class AssetsComponent {
1797
1161
  this.destroyed$.next();
1798
1162
  this.destroyed$.complete();
1799
1163
  }
1800
- getTemplateRootComponent$(template) {
1801
- return this.templatesApi
1802
- .fetchComponents$(template.id)
1803
- .pipe(map(components => components.find(c => c.type === UITemplateComponentType.ROOT) ?? undefined));
1804
- }
1805
- getLocalAssetsComponentMeta$() {
1806
- if (!this.customizationService?.getAssetsComponent) {
1164
+ getLocalMeta$() {
1165
+ if (!this.customizationService?.getTemplateComponents) {
1807
1166
  return of(undefined);
1808
1167
  }
1809
- return this.customizationService?.getAssetsComponent(this.templateApiName).pipe(map(component => {
1810
- if (!component) {
1168
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
1169
+ if (!components) {
1811
1170
  return;
1812
1171
  }
1813
- return {
1172
+ return components.map(component => ({
1814
1173
  html: component.html,
1815
1174
  css: component.css,
1816
1175
  js: component.js,
1817
1176
  json: component.json,
1818
- };
1177
+ }));
1819
1178
  }));
1820
1179
  }
1821
- getAssetsComponentMeta$() {
1822
- return this.templatesApi.fetchTemplates$().pipe(map(templates => templates.find(template => template.name === this.templateApiName)), switchMap(template => (template ? this.getTemplateRootComponent$(template) : of(undefined))), switchMap(component => component ? this.templatesApi.fetchComponentAttachments$(component.uiTemplateId, component) : of(undefined)));
1180
+ getOrgMeta$() {
1181
+ const template = this.flowInfo.templates.SHOPPING_CART;
1182
+ if (!template) {
1183
+ return of(undefined);
1184
+ }
1185
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
1823
1186
  }
1824
1187
  generateUIDefinition$() {
1825
1188
  return of(undefined).pipe(tap(() => {
1826
1189
  if (!this.templateApiName) {
1827
- throw new Error("Flow Query parameter 'assetsTemplateApiName' is missing.");
1190
+ throw new Error("Flow 'assets' template is not defined.");
1828
1191
  }
1829
- }), switchMap(() => this.getLocalAssetsComponentMeta$()), switchMap(meta => (meta ? of(meta) : this.getAssetsComponentMeta$())), map(meta => {
1830
- if (!meta) {
1192
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
1193
+ if (!metaList) {
1831
1194
  return;
1832
1195
  }
1833
1196
  const uiDef = {
@@ -1836,25 +1199,23 @@ class AssetsComponent {
1836
1199
  primary: true,
1837
1200
  type: 'DEFAULT',
1838
1201
  version: 2,
1839
- children: [
1840
- {
1841
- children: [],
1842
- template: meta.html && btoa(meta.html),
1843
- script: meta.js && btoa(meta.js),
1844
- styles: meta.css && btoa(meta.css),
1845
- },
1846
- ],
1202
+ children: metaList.map(meta => ({
1203
+ children: [],
1204
+ template: meta.html && btoaSafe(meta.html),
1205
+ script: meta.js && btoaSafe(meta.js),
1206
+ styles: meta.css && btoaSafe(meta.css),
1207
+ })),
1847
1208
  };
1848
1209
  return uiDef;
1849
1210
  }));
1850
1211
  }
1851
1212
  }
1852
- AssetsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: AssetsComponent, deps: [{ token: i2.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: FlowService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1213
+ AssetsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: AssetsComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: i2.FlowInfoService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1853
1214
  AssetsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: AssetsComponent, selector: "vl-flow-assets", ngImport: i0, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n", styles: [""], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1854
1215
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: AssetsComponent, decorators: [{
1855
1216
  type: Component,
1856
1217
  args: [{ selector: 'vl-flow-assets', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n" }]
1857
- }], ctorParameters: function () { return [{ type: i2.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: FlowService }, { type: undefined, decorators: [{
1218
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: i2.FlowInfoService }, { type: undefined, decorators: [{
1858
1219
  type: Optional
1859
1220
  }, {
1860
1221
  type: Inject,
@@ -1876,17 +1237,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
1876
1237
  }] });
1877
1238
 
1878
1239
  class CatalogComponent {
1879
- constructor(templatesApi, cdr, toastService, flowService, customizationService) {
1240
+ constructor(templatesApi, cdr, toastService, flowInfo, customizationService) {
1880
1241
  this.templatesApi = templatesApi;
1881
1242
  this.cdr = cdr;
1882
1243
  this.toastService = toastService;
1883
- this.flowService = flowService;
1244
+ this.flowInfo = flowInfo;
1884
1245
  this.customizationService = customizationService;
1885
1246
  this.uiDefinition = undefined;
1886
1247
  this.state$ = new BehaviorSubject({ loading: true, failure: false });
1887
1248
  this.templateApiName = '';
1888
1249
  this.destroyed$ = new Subject();
1889
- this.templateApiName = this.flowService.flow?.properties.templates.catalog ?? '';
1250
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.catalog ?? '';
1890
1251
  }
1891
1252
  ngOnInit() {
1892
1253
  this.generateUIDefinition$()
@@ -1909,37 +1270,36 @@ class CatalogComponent {
1909
1270
  this.destroyed$.next();
1910
1271
  this.destroyed$.complete();
1911
1272
  }
1912
- getTemplateRootComponent$(template) {
1913
- return this.templatesApi
1914
- .fetchComponents$(template.id)
1915
- .pipe(map(components => components.find(c => c.type === UITemplateComponentType.ROOT) ?? undefined));
1916
- }
1917
- getLocalCatalogComponentMeta$() {
1918
- if (!this.customizationService?.getCatalogComponent) {
1273
+ getLocalMeta$() {
1274
+ if (!this.customizationService?.getTemplateComponents) {
1919
1275
  return of(undefined);
1920
1276
  }
1921
- return this.customizationService?.getCatalogComponent(this.templateApiName).pipe(map(component => {
1922
- if (!component) {
1277
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
1278
+ if (!components) {
1923
1279
  return;
1924
1280
  }
1925
- return {
1281
+ return components.map(component => ({
1926
1282
  html: component.html,
1927
1283
  css: component.css,
1928
1284
  js: component.js,
1929
1285
  json: component.json,
1930
- };
1286
+ }));
1931
1287
  }));
1932
1288
  }
1933
- getCatalogComponentMeta$() {
1934
- return this.templatesApi.fetchTemplates$().pipe(map(templates => templates.find(template => template.name === this.templateApiName)), switchMap(template => (template ? this.getTemplateRootComponent$(template) : of(undefined))), switchMap(component => component ? this.templatesApi.fetchComponentAttachments$(component.uiTemplateId, component) : of(undefined)));
1289
+ getOrgMeta$() {
1290
+ const template = this.flowInfo.templates.CATALOG;
1291
+ if (!template) {
1292
+ return of(undefined);
1293
+ }
1294
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
1935
1295
  }
1936
1296
  generateUIDefinition$() {
1937
1297
  return of(undefined).pipe(tap(() => {
1938
1298
  if (!this.templateApiName) {
1939
- throw new Error("Flow Query parameter 'catalogTemplateApiName' is missing.");
1299
+ throw new Error("Flow 'catalog' template is not defined.");
1940
1300
  }
1941
- }), switchMap(() => this.getLocalCatalogComponentMeta$()), switchMap(meta => (meta ? of(meta) : this.getCatalogComponentMeta$())), map(meta => {
1942
- if (!meta) {
1301
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
1302
+ if (!metaList) {
1943
1303
  return;
1944
1304
  }
1945
1305
  const uiDef = {
@@ -1948,25 +1308,23 @@ class CatalogComponent {
1948
1308
  primary: true,
1949
1309
  type: 'DEFAULT',
1950
1310
  version: 2,
1951
- children: [
1952
- {
1953
- children: [],
1954
- template: meta.html && btoa(meta.html),
1955
- script: meta.js && btoa(meta.js),
1956
- styles: meta.css && btoa(meta.css),
1957
- },
1958
- ],
1311
+ children: metaList.map(meta => ({
1312
+ children: [],
1313
+ template: meta.html && btoaSafe(meta.html),
1314
+ script: meta.js && btoaSafe(meta.js),
1315
+ styles: meta.css && btoaSafe(meta.css),
1316
+ })),
1959
1317
  };
1960
1318
  return uiDef;
1961
1319
  }));
1962
1320
  }
1963
1321
  }
1964
- CatalogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CatalogComponent, deps: [{ token: i2.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: FlowService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1322
+ CatalogComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CatalogComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: i2.FlowInfoService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1965
1323
  CatalogComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: CatalogComponent, selector: "vl-flow-catalog", ngImport: i0, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n", styles: [""], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1966
1324
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: CatalogComponent, decorators: [{
1967
1325
  type: Component,
1968
1326
  args: [{ selector: 'vl-flow-catalog', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n" }]
1969
- }], ctorParameters: function () { return [{ type: i2.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: FlowService }, { type: undefined, decorators: [{
1327
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: i2.FlowInfoService }, { type: undefined, decorators: [{
1970
1328
  type: Optional
1971
1329
  }, {
1972
1330
  type: Inject,
@@ -2021,12 +1379,12 @@ class DebugComponent {
2021
1379
  return new HttpParams({ fromObject: params }).toString();
2022
1380
  }
2023
1381
  }
2024
- DebugComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DebugComponent, deps: [{ token: i2.FlowsApiService }, { token: i1$1.Router }, { token: i1$1.ActivatedRoute }, { token: i1$2.ContextService }, { token: i1$2.QuoteDraftService }], target: i0.ɵɵFactoryTarget.Component });
2025
- DebugComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DebugComponent, selector: "vl-flow-debug", ngImport: i0, template: "<form [formGroup]=\"form\">\n <div class=\"fields-container\">\n <div class=\"field\">\n <label>SF Object ID</label>\n <input formControlName=\"id\" pInputText type=\"text\" />\n </div>\n </div>\n\n <p-button\n styleClass=\"p-button-primary\"\n label=\"Run Flow\"\n [disabled]=\"!form.value.id || !selectedFlow\"\n (onClick)=\"runFlow()\"\n ></p-button>\n</form>\n\n<table>\n <thead>\n <tr>\n <th [width]=\"30\"></th>\n <th [width]=\"160\">ID</th>\n <th [width]=\"160\">Entry Path</th>\n <th>QueryParams</th>\n <th [width]=\"100\">Standalone</th>\n <th [width]=\"100\">Stateful</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let flow of flows$ | async\" (click)=\"selectedFlow = flow\">\n <td><p-radioButton [inputId]=\"flow.id\" name=\"flow\" [value]=\"flow\" [(ngModel)]=\"selectedFlow\"></p-radioButton></td>\n <td>{{ flow.id }}</td>\n <td>{{ flow.properties.entryPath }}</td>\n <td class=\"cell-query-params\">{{ getQueryParamsString(flow.properties.queryParams) }}</td>\n <td>{{ flow.properties.standalone }}</td>\n <td>{{ flow.properties.stateful }}</td>\n </tr>\n </tbody>\n</table>\n", styles: [":host{display:block;padding:24px 54px}form{display:flex;align-items:center;justify-content:space-between}.fields-container{display:flex;gap:24px}.field{display:flex;flex-direction:column;width:200px;flex-shrink:0}:host ::ng-deep .p-dropdown{width:100%}table{width:100%;border-collapse:collapse;table-layout:fixed}tr{cursor:pointer}tr:hover{background-color:#f0f5fa}th{text-align:left;font-weight:600}th,td{padding:0 10px;height:30px;border-bottom:1px solid var(--vl-border-color);margin-right:16px;line-height:18px}.cell-query-params{word-break:break-all}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5$1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i5$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i5$1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i5$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i5$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i6.RadioButton, selector: "p-radioButton", inputs: ["value", "formControlName", "name", "disabled", "label", "tabindex", "inputId", "ariaLabelledBy", "ariaLabel", "style", "styleClass", "labelStyleClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "directive", type: i10.InputText, selector: "[pInputText]" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1382
+ DebugComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DebugComponent, deps: [{ token: i1$1.FlowsApiService }, { token: i1$2.Router }, { token: i1$2.ActivatedRoute }, { token: i2.ContextService }, { token: i2.QuoteDraftService }], target: i0.ɵɵFactoryTarget.Component });
1383
+ DebugComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: DebugComponent, selector: "vl-flow-debug", ngImport: i0, template: "<form [formGroup]=\"form\">\n <div class=\"fields-container\">\n <div class=\"field\">\n <label>SF Object ID</label>\n <input formControlName=\"id\" pInputText type=\"text\" />\n </div>\n </div>\n\n <p-button\n styleClass=\"p-button-primary\"\n label=\"Run Flow\"\n [disabled]=\"!form.value.id || !selectedFlow\"\n (onClick)=\"runFlow()\"\n ></p-button>\n</form>\n\n<table>\n <thead>\n <tr>\n <th [width]=\"30\"></th>\n <th [width]=\"160\">ID</th>\n <th [width]=\"160\">Entry Path</th>\n <th>QueryParams</th>\n <th [width]=\"100\">Standalone</th>\n <th [width]=\"100\">Stateful</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let flow of flows$ | async\" (click)=\"selectedFlow = flow\">\n <td><p-radioButton [inputId]=\"flow.id\" name=\"flow\" [value]=\"flow\" [(ngModel)]=\"selectedFlow\"></p-radioButton></td>\n <td>{{ flow.id }}</td>\n <td>{{ flow.properties.entryPath }}</td>\n <td class=\"cell-query-params\">{{ getQueryParamsString(flow.properties.queryParams) }}</td>\n <td>{{ flow.properties.standalone }}</td>\n <td>{{ flow.properties.stateful }}</td>\n </tr>\n </tbody>\n</table>\n", styles: [":host{display:block;padding:24px 54px}form{display:flex;align-items:center;justify-content:space-between}.fields-container{display:flex;gap:24px}.field{display:flex;flex-direction:column;width:200px;flex-shrink:0}:host ::ng-deep .p-dropdown{width:100%}table{width:100%;border-collapse:collapse;table-layout:fixed}tbody>tr{cursor:pointer}tbody>tr:hover{background-color:#f0f5fa}th{text-align:left;font-weight:600}th,td{padding:0 10px;height:30px;border-bottom:1px solid var(--vl-border-color);margin-right:16px;line-height:18px}.cell-query-params{word-break:break-all}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5$1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i5$1.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5$1.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5$1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i5$1.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "directive", type: i5$1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i5$1.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i6.RadioButton, selector: "p-radioButton", inputs: ["value", "formControlName", "name", "disabled", "label", "tabindex", "inputId", "ariaLabelledBy", "ariaLabel", "style", "styleClass", "labelStyleClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "component", type: i3.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { kind: "directive", type: i8.InputText, selector: "[pInputText]" }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2026
1384
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DebugComponent, decorators: [{
2027
1385
  type: Component,
2028
- args: [{ selector: 'vl-flow-debug', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form [formGroup]=\"form\">\n <div class=\"fields-container\">\n <div class=\"field\">\n <label>SF Object ID</label>\n <input formControlName=\"id\" pInputText type=\"text\" />\n </div>\n </div>\n\n <p-button\n styleClass=\"p-button-primary\"\n label=\"Run Flow\"\n [disabled]=\"!form.value.id || !selectedFlow\"\n (onClick)=\"runFlow()\"\n ></p-button>\n</form>\n\n<table>\n <thead>\n <tr>\n <th [width]=\"30\"></th>\n <th [width]=\"160\">ID</th>\n <th [width]=\"160\">Entry Path</th>\n <th>QueryParams</th>\n <th [width]=\"100\">Standalone</th>\n <th [width]=\"100\">Stateful</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let flow of flows$ | async\" (click)=\"selectedFlow = flow\">\n <td><p-radioButton [inputId]=\"flow.id\" name=\"flow\" [value]=\"flow\" [(ngModel)]=\"selectedFlow\"></p-radioButton></td>\n <td>{{ flow.id }}</td>\n <td>{{ flow.properties.entryPath }}</td>\n <td class=\"cell-query-params\">{{ getQueryParamsString(flow.properties.queryParams) }}</td>\n <td>{{ flow.properties.standalone }}</td>\n <td>{{ flow.properties.stateful }}</td>\n </tr>\n </tbody>\n</table>\n", styles: [":host{display:block;padding:24px 54px}form{display:flex;align-items:center;justify-content:space-between}.fields-container{display:flex;gap:24px}.field{display:flex;flex-direction:column;width:200px;flex-shrink:0}:host ::ng-deep .p-dropdown{width:100%}table{width:100%;border-collapse:collapse;table-layout:fixed}tr{cursor:pointer}tr:hover{background-color:#f0f5fa}th{text-align:left;font-weight:600}th,td{padding:0 10px;height:30px;border-bottom:1px solid var(--vl-border-color);margin-right:16px;line-height:18px}.cell-query-params{word-break:break-all}\n"] }]
2029
- }], ctorParameters: function () { return [{ type: i2.FlowsApiService }, { type: i1$1.Router }, { type: i1$1.ActivatedRoute }, { type: i1$2.ContextService }, { type: i1$2.QuoteDraftService }]; } });
1386
+ args: [{ selector: 'vl-flow-debug', changeDetection: ChangeDetectionStrategy.OnPush, template: "<form [formGroup]=\"form\">\n <div class=\"fields-container\">\n <div class=\"field\">\n <label>SF Object ID</label>\n <input formControlName=\"id\" pInputText type=\"text\" />\n </div>\n </div>\n\n <p-button\n styleClass=\"p-button-primary\"\n label=\"Run Flow\"\n [disabled]=\"!form.value.id || !selectedFlow\"\n (onClick)=\"runFlow()\"\n ></p-button>\n</form>\n\n<table>\n <thead>\n <tr>\n <th [width]=\"30\"></th>\n <th [width]=\"160\">ID</th>\n <th [width]=\"160\">Entry Path</th>\n <th>QueryParams</th>\n <th [width]=\"100\">Standalone</th>\n <th [width]=\"100\">Stateful</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let flow of flows$ | async\" (click)=\"selectedFlow = flow\">\n <td><p-radioButton [inputId]=\"flow.id\" name=\"flow\" [value]=\"flow\" [(ngModel)]=\"selectedFlow\"></p-radioButton></td>\n <td>{{ flow.id }}</td>\n <td>{{ flow.properties.entryPath }}</td>\n <td class=\"cell-query-params\">{{ getQueryParamsString(flow.properties.queryParams) }}</td>\n <td>{{ flow.properties.standalone }}</td>\n <td>{{ flow.properties.stateful }}</td>\n </tr>\n </tbody>\n</table>\n", styles: [":host{display:block;padding:24px 54px}form{display:flex;align-items:center;justify-content:space-between}.fields-container{display:flex;gap:24px}.field{display:flex;flex-direction:column;width:200px;flex-shrink:0}:host ::ng-deep .p-dropdown{width:100%}table{width:100%;border-collapse:collapse;table-layout:fixed}tbody>tr{cursor:pointer}tbody>tr:hover{background-color:#f0f5fa}th{text-align:left;font-weight:600}th,td{padding:0 10px;height:30px;border-bottom:1px solid var(--vl-border-color);margin-right:16px;line-height:18px}.cell-query-params{word-break:break-all}\n"] }]
1387
+ }], ctorParameters: function () { return [{ type: i1$1.FlowsApiService }, { type: i1$2.Router }, { type: i1$2.ActivatedRoute }, { type: i2.ContextService }, { type: i2.QuoteDraftService }]; } });
2030
1388
 
2031
1389
  const routes$1 = [{ path: '', component: DebugComponent }];
2032
1390
  class DebugModule {
@@ -2034,7 +1392,7 @@ class DebugModule {
2034
1392
  DebugModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: DebugModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
2035
1393
  DebugModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: DebugModule, declarations: [DebugComponent], imports: [CommonModule,
2036
1394
  FormsModule,
2037
- ReactiveFormsModule, i1$1.RouterModule, RadioButtonModule,
1395
+ ReactiveFormsModule, i1$2.RouterModule, RadioButtonModule,
2038
1396
  ButtonModule,
2039
1397
  InputTextModule,
2040
1398
  DropdownModule] });
@@ -2063,36 +1421,15 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2063
1421
  }]
2064
1422
  }] });
2065
1423
 
2066
- class EmptyAccountComponent {
2067
- }
2068
- EmptyAccountComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
2069
- EmptyAccountComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: EmptyAccountComponent, selector: "vl-empty-account", ngImport: i0, template: "<vl-empty-state\n primaryText=\"No Orders in your Account\"\n secondaryText=\"To proceed, go to the Shopping Cart from a Quote.\"\n>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"400\" height=\"281\" viewBox=\"0 0 400 281\" fill=\"none\">\n <path d=\"M264.214 26.9277H70.8805V128.186H264.214V26.9277Z\" fill=\"#0466C8\" />\n <path\n d=\"M143 56.5H136.217C128.166 64.4791 120.051 72.5209 112 80.5H118.783C126.834 72.5209 134.949 64.5419 143 56.5Z\"\n fill=\"white\"\n />\n <path\n d=\"M242.932 57.5681V79.4319H128.575L133.724 74.2801L150.429 57.5681H242.932ZM244 56.5H149.989C142.014 64.4791 133.976 72.5209 126 80.5H244V56.5Z\"\n fill=\"white\"\n />\n <path d=\"M129 56.5H81V80.5H104.969C112.958 72.5209 120.948 64.5419 129 56.5Z\" fill=\"white\" />\n <path\n d=\"M82.3 23.2016C82.2366 23.2016 82.1732 23.2644 82.1098 23.2644C81.3494 23.5785 80.5889 23.5785 79.8918 23.2644C79.765 23.2016 79.6383 23.1387 79.4482 23.0759C78.9412 22.7618 78.561 22.322 78.3075 21.7565C77.6104 20.2487 78.1173 18.6151 79.6383 17.8612C81.1592 17.1073 82.9971 17.547 83.6942 19.0549C84.3913 20.6256 83.8843 22.3848 82.3 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M92.2716 23.2016C92.2086 23.2016 92.1457 23.2644 92.0827 23.2644C91.3273 23.5785 90.5718 23.5785 89.8793 23.2644C89.7534 23.2016 89.6275 23.1387 89.4386 23.0759C88.935 22.7618 88.5573 22.322 88.3054 21.7565C87.6129 20.2487 88.1166 18.6151 89.6275 17.8612C91.1384 17.1073 92.9641 17.547 93.6566 19.0549C94.412 20.6256 93.9084 22.3848 92.2716 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M102.3 23.2016C102.237 23.2016 102.173 23.2644 102.11 23.2644C101.349 23.5785 100.589 23.5785 99.8918 23.2644C99.765 23.2016 99.6383 23.1387 99.4482 23.0759C98.9412 22.7618 98.561 22.322 98.3075 21.7565C97.6104 20.2487 98.1174 18.6151 99.6383 17.8612C101.159 17.1073 102.997 17.547 103.694 19.0549C104.391 20.6256 103.884 22.3848 102.3 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M210.126 91.0156H130.44C129.748 91.0156 129.748 92.0848 130.44 92.0848H210.126C210.818 92.0848 210.818 91.0156 210.126 91.0156Z\"\n fill=\"white\"\n />\n <path\n d=\"M197.484 97.5566H142.013C141.321 97.5566 141.321 98.6258 142.013 98.6258H197.484C198.176 98.6258 198.176 97.5566 197.484 97.5566Z\"\n fill=\"white\"\n />\n <path\n d=\"M229.436 35.5H61.5553C60.8149 35.5 60.8149 36.5 61.5553 36.5H229.503C230.176 36.5 230.176 35.5 229.436 35.5Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M270.44 35.3555H256.226C256.163 35.3555 256.163 36.4246 256.226 36.4246H270.44C270.503 36.4246 270.503 35.3555 270.44 35.3555Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M332.704 212.022C332.642 212.022 332.579 212.022 332.516 211.959L280.566 191.582C280.377 191.519 280.252 191.33 280.252 191.141C280.252 190.953 280.314 190.764 280.44 190.638C292.83 181.33 333.333 150.953 335.346 150.953C335.472 150.953 335.66 151.016 335.723 151.141C335.849 151.267 335.849 151.393 335.849 151.519L333.208 211.456C333.208 211.644 333.082 211.77 332.956 211.896C332.893 211.959 332.767 212.022 332.704 212.022ZM281.824 190.953L332.138 210.701L334.717 152.336C329.811 155.166 301.321 176.299 281.824 190.953Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M366.289 231.015C366.226 231.015 366.226 231.015 366.164 231.015L349.371 227.053C349.182 226.99 349.057 226.927 348.994 226.739C348.931 226.613 348.931 226.424 348.994 226.236L357.736 211.141C357.862 210.953 358.05 210.89 358.239 210.89C358.428 210.89 358.616 211.015 358.679 211.204C366.918 229.758 366.792 230.324 366.792 230.512C366.792 230.638 366.667 230.764 366.541 230.89C366.478 230.953 366.415 231.015 366.289 231.015ZM350.314 226.173L365.472 229.758C364.465 227.116 360.755 218.626 358.113 212.651L350.314 226.173Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M199.686 248.751C192.516 229.38 177.799 213.909 159.497 205.67V205.607C159.748 193.343 154.528 181.141 144.654 173.657C133.396 165.104 118.365 162.902 104.591 164.034C78.1132 166.236 53.0189 181.456 38.805 203.846C36.9811 206.739 35.3459 209.758 33.8365 212.839C33.522 213.468 34.4654 214.034 34.7799 213.405C45.4088 191.015 66.1635 174.16 90 167.682C113.711 161.267 146.981 164.726 156.226 191.519C157.736 195.921 158.491 200.575 158.428 205.229C150.943 201.959 142.893 199.946 134.465 199.254C121.447 198.248 103.522 199.506 95.283 211.204C91.761 216.236 90.8805 222.525 93.5849 228.122C96.2893 233.72 101.761 237.745 107.421 239.946C118.868 244.475 132.39 242.462 142.642 235.858C152.642 229.38 158.679 218.688 159.371 206.865C177.17 215.041 191.572 230.072 198.616 249.066C198.931 249.695 199.937 249.443 199.686 248.751ZM152.075 225.481C145.094 235.355 133.019 241.141 121.006 241.267C109.434 241.393 92.7044 234.6 93.0189 220.701C93.2076 213.217 98.9937 207.305 105.346 204.286C112.013 201.141 119.748 200.198 127.044 200.072C137.987 199.883 148.616 202.148 158.302 206.361C158.113 213.154 156.038 219.883 152.075 225.481Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M341.447 271.204H322.767C322.075 271.204 322.075 272.273 322.767 272.273H341.447C342.138 272.336 342.138 271.204 341.447 271.204Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M233.522 75.4814C230.881 89.3179 229.245 103.154 224.025 110.576C213.962 124.915 185.975 137.117 178.428 143.972C170.881 150.764 171.95 151.142 173.019 152.588C173.648 153.469 177.862 151.771 181.132 150.261C177.421 152.211 172.579 155.23 173.711 157.242C174.654 159.003 178.679 157.808 182.201 156.362C178.994 158.123 175.786 160.387 176.981 161.959C178.365 163.783 183.585 161.897 187.107 160.387C184.088 162.022 180.314 164.475 181.635 166.299C183.774 169.192 235.849 149.758 245.157 120.702C248.805 109.318 250.881 94.6009 252.013 80.8274C245.786 79.3808 239.623 77.494 233.522 75.4814Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M182.327 148.437C179.497 150.576 176.226 151.645 172.704 151.582C172.013 151.582 172.013 152.651 172.704 152.651C176.415 152.714 179.937 151.582 182.893 149.318C183.396 148.941 182.893 147.997 182.327 148.437Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M184.906 154.852C183.27 155.607 181.635 156.173 179.937 156.676C178.302 157.117 175.723 158.06 174.151 157.117C173.522 156.802 173.019 157.683 173.585 158.06C175.157 158.94 176.918 158.5 178.553 158.123C180.943 157.62 183.145 156.802 185.346 155.796C186.101 155.481 185.535 154.538 184.906 154.852Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M186.163 158.185C184.277 159.946 180.189 163.217 177.421 161.833C176.792 161.519 176.226 162.462 176.855 162.777C180.252 164.412 184.528 161.141 186.855 158.94C187.421 158.437 186.667 157.682 186.163 158.185Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M253.522 46.0474C253.522 47.1166 253.522 48.3115 253.522 49.6323C253.459 57.3052 253.145 69.1291 252.013 81.6449C251.95 82.148 251.95 82.6512 251.887 83.1543C245.472 82.148 236.981 79.2549 230.629 78.1857C232.893 65.7958 237.736 54.2864 244.151 45.4185C245.157 44.0348 246.226 42.7769 247.421 41.6449L253.522 40.5757C253.522 40.5757 253.522 41.3933 253.585 42.8398C253.522 43.6574 253.522 44.7266 253.522 46.0474Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M184.687 136.932C179.875 137.121 175.253 138.637 170.631 139.837C165.565 141.101 160.5 142.174 155.371 143.122C134.602 146.975 113.264 148.238 92.1786 146.849C74.7659 145.712 55.3904 142.238 43.2332 128.531C32.469 116.403 28.9865 98.1477 34.9384 82.9879C40.7638 68.0807 55.3271 57.6583 70.9668 55.5738V54.5C58.6197 56.0792 46.969 62.7748 39.4341 72.755C29.9996 85.325 28.6066 102.317 34.6218 116.719C41.0804 132.257 55.2004 141.227 71.1568 145.017C80.2747 147.165 89.6458 147.923 98.9537 148.302C109.718 148.744 120.545 148.428 131.246 147.544C143.087 146.533 154.801 144.638 166.388 142.048C172.403 140.658 178.545 138.321 184.687 138.132C190.576 137.942 197.604 140.974 197.921 147.607C197.984 148.302 199.061 148.302 198.997 147.607C198.617 140.216 191.209 136.679 184.687 136.932Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M190 164.091V172.909C190 173.697 191 173.697 191 172.909V164.091C191 163.303 190 163.303 190 164.091Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M199 164.037V172.963C199 173.679 200 173.679 200 172.963V164.037C200 163.321 199 163.321 199 164.037Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M182 166.5V155.275C182 148.205 187.792 142.5 194.969 142.5C202.145 142.5 207.937 148.205 207.937 155.275L208 166.5H182Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M209.483 165.5H180.517C179.828 165.5 179.828 166.5 180.517 166.5H209.483C210.172 166.5 210.172 165.5 209.483 165.5Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M203.711 151.519V161.582C203.711 162.274 204.78 162.274 204.78 161.582V151.519C204.843 150.827 203.711 150.827 203.711 151.519Z\"\n fill=\"white\"\n />\n <path\n d=\"M196.73 134.601C187.421 142.148 183.648 152.274 187.233 153.343C190.818 154.412 203.208 135.67 203.208 135.67L196.73 134.601Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M199.748 139.632C196.855 144.098 193.396 148.374 189.182 151.708L188.239 152.399C187.358 152.84 186.352 152.148 186.478 150.953C186.541 150.45 187.233 149.066 187.421 148.689C187.925 147.557 188.491 146.425 189.057 145.292C190.377 142.777 191.824 140.387 193.459 138.06C193.836 137.494 192.893 136.928 192.516 137.494C191.195 139.443 181.824 152.399 186.855 153.594C189.245 154.16 191.824 150.827 193.27 149.381C196.038 146.55 198.491 143.469 200.692 140.135C201.069 139.569 200.126 139.066 199.748 139.632Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M211.132 135.984C207.736 139.003 203.648 140.89 199.057 141.33C198.365 141.393 198.365 142.462 199.057 142.399C203.899 141.896 208.239 139.947 211.887 136.739C212.39 136.299 211.635 135.544 211.132 135.984Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M238.176 254.224C250.692 253.783 263.208 255.607 275.723 255.796C275.157 254.475 274.654 253.217 274.088 251.897C273.774 251.205 273.459 250.513 273.145 249.821C270.377 243.658 267.484 237.809 264.717 232.274C254.906 235.859 245.031 239.129 234.843 241.331C237.547 246.928 239.12 251.393 238.176 254.224Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M273.899 253.595C276.101 253.783 278.365 253.783 280.566 253.72C284.717 253.658 288.805 253.72 292.956 253.72C299.12 253.783 305.283 253.532 311.384 253.091C307.17 244.161 302.767 235.796 298.553 228.06C289.057 230.639 279.56 232.966 269.937 235.104C273.145 242.525 275.031 249.003 273.899 253.595Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M274.214 250.828C273.836 255.104 271.069 257.997 267.107 259.57C256.352 263.909 259.937 272.84 259.937 272.84H319.874C316.918 265.167 313.648 257.809 310.314 250.765H274.214V250.828Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M307.296 229.758C293.459 203.783 276.289 186.299 281.447 173.72C297.233 135.419 300.818 107.997 291.887 84.3494L267.484 80.0098C267.484 80.0098 269.623 91.5192 268.553 112.337C267.484 133.154 252.201 152.022 250.252 182.337C249.182 199.129 259.937 225.985 266.415 241.582C275.912 238.437 297.547 232.085 307.296 229.758Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M265.786 234.475C252.516 207.934 240.063 187.683 246.415 172.714C263.145 133.406 281.069 109.884 272.075 86.1733C272.075 86.1733 254.843 72.5255 253.774 93.3431C252.704 114.161 218.05 153.846 214.088 183.972C211.321 204.915 223.962 230.513 230.252 245.356C240.252 242.337 256.352 238.815 265.786 234.475Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M282.201 272.84H222.264C222.264 272.84 222.264 260.45 234.654 257.243C237.17 256.614 238.302 255.104 238.491 252.966H274.591C277.296 259.255 279.874 265.922 282.201 272.84Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M274.088 251.896C274.025 252.274 273.962 252.588 273.836 252.966C273.082 255.67 271.195 258.123 268.428 259.444C264.151 261.519 259.119 261.456 256.289 265.922C254.969 268.06 254.402 270.45 254.34 272.84H253.333C253.522 268.311 254.906 264.098 259.182 261.708C262.704 259.758 267.233 259.821 270.252 256.991C271.447 255.922 272.264 254.475 272.704 252.966C273.019 251.959 273.145 250.89 273.082 249.884C273.459 250.513 273.774 251.205 274.088 251.896Z\"\n fill=\"white\"\n />\n <path\n d=\"M281.195 256.739C279.497 255.67 277.799 254.601 276.101 253.532C275.535 253.154 274.969 254.098 275.535 254.475C277.233 255.544 278.931 256.614 280.629 257.683C281.258 258.06 281.761 257.117 281.195 256.739Z\"\n fill=\"white\"\n />\n <path\n d=\"M244.717 257.809C242.893 257.242 241.132 256.739 239.308 256.173C238.616 255.985 238.365 256.991 238.994 257.243C240.818 257.809 242.579 258.312 244.403 258.878C245.094 259.066 245.346 258.06 244.717 257.809Z\"\n fill=\"white\"\n />\n <path\n d=\"M262.516 231.267C261.069 227.682 259.623 224.098 258.239 220.513C257.987 219.884 256.918 220.135 257.17 220.827C258.616 224.412 260.063 227.997 261.447 231.582C261.698 232.211 262.767 231.959 262.516 231.267Z\"\n fill=\"white\"\n />\n <path\n d=\"M268.616 110.073C268.491 110.513 268.365 110.953 268.302 111.394C265.786 121.645 263.208 131.897 260.692 142.148C260.503 142.84 259.497 142.526 259.623 141.834C262.264 131.331 264.843 120.765 267.484 110.262C267.61 109.821 267.736 109.381 267.799 108.878C268.05 109.318 268.365 109.696 268.616 110.073Z\"\n fill=\"white\"\n />\n <path\n d=\"M268.616 110.073C268.491 110.513 268.365 110.953 268.302 111.393C268.05 111.016 267.736 110.639 267.484 110.261C266.478 108.878 265.409 107.494 264.402 106.11C263.962 105.544 264.906 105.041 265.346 105.544C266.163 106.676 266.981 107.808 267.862 108.878C268.05 109.318 268.365 109.695 268.616 110.073Z\"\n fill=\"white\"\n />\n <path\n d=\"M251.698 43.0913C238.994 45.7328 251.509 69.5693 253.836 93.3429L291.887 84.3492C291.887 84.3492 276.415 37.997 251.698 43.0913Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M251.384 79.8839C249.308 73.6574 247.296 67.4939 245.22 61.2675C245.031 60.6386 243.962 60.8901 244.151 61.582C246.226 67.8084 248.239 73.9719 250.314 80.1983C250.566 80.8273 251.572 80.5128 251.384 79.8839Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M332.075 57.431C322.39 48.3744 311.447 43.5945 299.811 41.2046C297.673 46.9908 295.723 52.9027 293.962 58.8147C295.66 59.3807 297.359 60.0096 298.994 60.6386C323.396 69.6323 326.667 87.1794 330.566 97.6197C334.528 108.06 322.642 122.777 326.604 124.538C330.566 126.362 336.289 113.028 336.289 113.028C336.289 113.028 334.843 135.481 337.736 136.173C340.377 136.865 341.698 131.016 342.264 126.488C342.076 130.764 342.327 136.047 344.906 135.859C347.17 135.67 347.862 131.456 347.987 127.494C348.365 130.764 349.245 133.909 351.384 134.098C354.088 134.349 353.962 128.626 353.522 124.349C354.277 128.123 355.66 132.777 358.05 131.896C362.075 130.513 353.962 77.8713 332.075 57.431Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M302.453 41.7707C300.314 47.7455 296.667 58.8147 295.157 64.9782C287.61 60.6386 279.245 57.1795 273.082 53.6575C265.723 49.5065 258.742 47.5569 254.717 46.3619C254.277 46.2361 253.836 46.1103 253.459 46.0474C251.572 45.5443 250.566 45.2298 250.566 45.2298L251.95 43.0286L253.459 40.5757C254.717 40.4499 255.912 40.3871 257.17 40.2613C272.704 38.9405 288.176 38.3745 302.453 41.7707Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M341.761 123.846C341.384 127.117 341.887 134.853 337.673 135.67C336.981 135.796 337.296 136.865 337.987 136.739C339.874 136.362 341.069 134.915 341.572 133.154C342.453 130.261 342.453 126.928 342.83 123.909C342.956 123.154 341.824 123.154 341.761 123.846Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M346.981 122.966C346.918 124.852 346.855 126.739 346.73 128.626C346.667 129.884 347.107 135.544 344.78 135.041C344.088 134.915 343.837 135.922 344.466 136.11C346.352 136.488 347.296 135.104 347.61 133.406C348.176 130.073 347.925 126.362 348.05 122.966C348.113 122.274 347.044 122.274 346.981 122.966Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M353.648 122.022C353.648 121.33 352.579 121.33 352.579 122.022C352.642 124.035 353.774 132.337 351.132 133.029C350.44 133.217 350.755 134.224 351.447 134.098C353.019 133.657 353.648 132.525 353.774 130.953C354.088 128.06 353.711 124.978 353.648 122.022Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M358.868 119.129C358.805 118.437 357.736 118.5 357.799 119.192C357.987 120.953 360.126 130.639 357.547 131.142C356.855 131.267 357.17 132.274 357.862 132.148C359.56 131.834 360 130.387 360 128.815C359.937 125.67 359.245 122.337 358.868 119.129Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M275.346 54.3493C274.088 53.8462 272.83 53.4059 271.635 52.9028C271.006 52.6512 270.692 53.7204 271.321 53.972C272.579 54.4751 273.836 54.9154 275.031 55.4185C275.723 55.6701 276.038 54.6009 275.346 54.3493Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M256.164 41.9592C254.277 37.4938 251.509 31.2045 251.132 31.1416L241.006 33.28C241.006 33.28 245.849 42.9655 248.113 47.5567C250.629 45.4812 253.333 43.6573 256.164 41.9592Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M252.704 28.4372C252.704 28.7517 252.642 29.0661 252.579 29.3806C252.516 29.6951 252.453 30.0095 252.39 30.2611C252.39 30.324 252.327 30.3869 252.327 30.4498C252.264 30.7013 252.201 30.9529 252.138 31.2045C250.943 35.041 247.862 38.0598 243.082 38.1856C233.899 38.3743 230.252 30.7013 231.384 24.412C232.138 20.1982 235.094 16.5504 239.937 16.173C240.126 16.173 240.377 16.1102 240.566 16.1102C245.031 16.0473 248.176 17.6825 250.189 20.0724C250.503 20.5127 250.818 20.9529 251.132 21.3932C251.321 21.7076 251.447 22.0221 251.635 22.3366C251.887 22.8397 252.075 23.3429 252.201 23.846C252.641 25.3554 252.83 26.9278 252.704 28.4372Z\"\n fill=\"#D4E8F7\"\n />\n <path d=\"M236.478 29.3179L238.679 34.9782L234.528 35.2298L236.478 29.3179Z\" fill=\"#002957\" />\n <path\n d=\"M239.182 28.0601L239.434 28.8148C239.686 29.4438 238.616 29.6953 238.365 29.0664L238.113 28.3117C237.924 27.6827 238.994 27.4312 239.182 28.0601Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M233.27 28.9406L233.522 29.6954C233.774 30.3243 232.704 30.5759 232.453 29.9469L232.201 29.1922C232.013 28.5004 233.082 28.2488 233.27 28.9406Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M247.044 27.1166C246.918 27.6827 246.855 28.3116 246.73 28.8776C246.604 29.5695 247.673 29.8839 247.736 29.1921C247.862 28.6261 247.924 27.9971 248.05 27.4311C248.176 26.7393 247.17 26.4877 247.044 27.1166Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M252.201 23.8461C252.013 23.343 251.824 22.8399 251.635 22.3367C251.509 22.0222 251.321 21.7078 251.132 21.3933C250.88 20.9531 250.566 20.5128 250.189 20.0726C248.176 17.6197 245.031 15.9845 240.566 16.1103C240.377 16.1103 240.126 16.1103 239.937 16.1732C238.491 16.299 234.465 16.9279 232.956 18.8147C228.616 20.1355 225.157 23.909 229.308 25.4814C232.453 26.8021 238.176 25.7958 240.88 24.4751C241.132 25.6072 242.83 28.7518 244.151 28.626C245.723 28.4373 245.472 26.6763 244.969 25.6072C244.906 25.4814 244.843 25.3556 244.78 25.2298C245.094 25.0411 245.409 24.8524 245.786 24.7267C247.925 23.7204 250.818 24.6638 251.509 27.1166C251.824 28.3744 251.698 29.7581 250.943 30.7015C250.88 30.8273 251.824 31.2046 252.138 31.2675C252.201 31.016 252.264 30.7644 252.327 30.5128C252.327 30.4499 252.327 30.387 252.39 30.3241C252.453 30.0097 252.516 29.7581 252.579 29.4436C252.642 29.1292 252.641 28.8147 252.704 28.5002C252.767 26.9279 252.641 25.3556 252.201 23.8461Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M257.17 40.2612L256.226 42.7141L254.78 46.3619L253.522 49.5694L251.635 54.4122L246.478 48.1858L244.151 45.4185C245.157 44.0348 246.226 42.777 247.421 41.6449L253.522 40.5757C254.717 40.4499 255.912 40.3241 257.17 40.2612Z\"\n fill=\"#D4E8F7\"\n />\n </svg>\n</vl-empty-state>\n", styles: [":host{display:flex;height:100%}\n"], dependencies: [{ kind: "component", type: i2$1.EmptyStateComponent, selector: "vl-empty-state", inputs: ["primaryText", "secondaryText", "buttonLabel", "icon", "disabled"], outputs: ["buttonClick"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2070
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountComponent, decorators: [{
2071
- type: Component,
2072
- args: [{ selector: 'vl-empty-account', changeDetection: ChangeDetectionStrategy.OnPush, template: "<vl-empty-state\n primaryText=\"No Orders in your Account\"\n secondaryText=\"To proceed, go to the Shopping Cart from a Quote.\"\n>\n <svg xmlns=\"http://www.w3.org/2000/svg\" width=\"400\" height=\"281\" viewBox=\"0 0 400 281\" fill=\"none\">\n <path d=\"M264.214 26.9277H70.8805V128.186H264.214V26.9277Z\" fill=\"#0466C8\" />\n <path\n d=\"M143 56.5H136.217C128.166 64.4791 120.051 72.5209 112 80.5H118.783C126.834 72.5209 134.949 64.5419 143 56.5Z\"\n fill=\"white\"\n />\n <path\n d=\"M242.932 57.5681V79.4319H128.575L133.724 74.2801L150.429 57.5681H242.932ZM244 56.5H149.989C142.014 64.4791 133.976 72.5209 126 80.5H244V56.5Z\"\n fill=\"white\"\n />\n <path d=\"M129 56.5H81V80.5H104.969C112.958 72.5209 120.948 64.5419 129 56.5Z\" fill=\"white\" />\n <path\n d=\"M82.3 23.2016C82.2366 23.2016 82.1732 23.2644 82.1098 23.2644C81.3494 23.5785 80.5889 23.5785 79.8918 23.2644C79.765 23.2016 79.6383 23.1387 79.4482 23.0759C78.9412 22.7618 78.561 22.322 78.3075 21.7565C77.6104 20.2487 78.1173 18.6151 79.6383 17.8612C81.1592 17.1073 82.9971 17.547 83.6942 19.0549C84.3913 20.6256 83.8843 22.3848 82.3 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M92.2716 23.2016C92.2086 23.2016 92.1457 23.2644 92.0827 23.2644C91.3273 23.5785 90.5718 23.5785 89.8793 23.2644C89.7534 23.2016 89.6275 23.1387 89.4386 23.0759C88.935 22.7618 88.5573 22.322 88.3054 21.7565C87.6129 20.2487 88.1166 18.6151 89.6275 17.8612C91.1384 17.1073 92.9641 17.547 93.6566 19.0549C94.412 20.6256 93.9084 22.3848 92.2716 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M102.3 23.2016C102.237 23.2016 102.173 23.2644 102.11 23.2644C101.349 23.5785 100.589 23.5785 99.8918 23.2644C99.765 23.2016 99.6383 23.1387 99.4482 23.0759C98.9412 22.7618 98.561 22.322 98.3075 21.7565C97.6104 20.2487 98.1174 18.6151 99.6383 17.8612C101.159 17.1073 102.997 17.547 103.694 19.0549C104.391 20.6256 103.884 22.3848 102.3 23.2016Z\"\n fill=\"#B4D1EF\"\n />\n <path\n d=\"M210.126 91.0156H130.44C129.748 91.0156 129.748 92.0848 130.44 92.0848H210.126C210.818 92.0848 210.818 91.0156 210.126 91.0156Z\"\n fill=\"white\"\n />\n <path\n d=\"M197.484 97.5566H142.013C141.321 97.5566 141.321 98.6258 142.013 98.6258H197.484C198.176 98.6258 198.176 97.5566 197.484 97.5566Z\"\n fill=\"white\"\n />\n <path\n d=\"M229.436 35.5H61.5553C60.8149 35.5 60.8149 36.5 61.5553 36.5H229.503C230.176 36.5 230.176 35.5 229.436 35.5Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M270.44 35.3555H256.226C256.163 35.3555 256.163 36.4246 256.226 36.4246H270.44C270.503 36.4246 270.503 35.3555 270.44 35.3555Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M332.704 212.022C332.642 212.022 332.579 212.022 332.516 211.959L280.566 191.582C280.377 191.519 280.252 191.33 280.252 191.141C280.252 190.953 280.314 190.764 280.44 190.638C292.83 181.33 333.333 150.953 335.346 150.953C335.472 150.953 335.66 151.016 335.723 151.141C335.849 151.267 335.849 151.393 335.849 151.519L333.208 211.456C333.208 211.644 333.082 211.77 332.956 211.896C332.893 211.959 332.767 212.022 332.704 212.022ZM281.824 190.953L332.138 210.701L334.717 152.336C329.811 155.166 301.321 176.299 281.824 190.953Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M366.289 231.015C366.226 231.015 366.226 231.015 366.164 231.015L349.371 227.053C349.182 226.99 349.057 226.927 348.994 226.739C348.931 226.613 348.931 226.424 348.994 226.236L357.736 211.141C357.862 210.953 358.05 210.89 358.239 210.89C358.428 210.89 358.616 211.015 358.679 211.204C366.918 229.758 366.792 230.324 366.792 230.512C366.792 230.638 366.667 230.764 366.541 230.89C366.478 230.953 366.415 231.015 366.289 231.015ZM350.314 226.173L365.472 229.758C364.465 227.116 360.755 218.626 358.113 212.651L350.314 226.173Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M199.686 248.751C192.516 229.38 177.799 213.909 159.497 205.67V205.607C159.748 193.343 154.528 181.141 144.654 173.657C133.396 165.104 118.365 162.902 104.591 164.034C78.1132 166.236 53.0189 181.456 38.805 203.846C36.9811 206.739 35.3459 209.758 33.8365 212.839C33.522 213.468 34.4654 214.034 34.7799 213.405C45.4088 191.015 66.1635 174.16 90 167.682C113.711 161.267 146.981 164.726 156.226 191.519C157.736 195.921 158.491 200.575 158.428 205.229C150.943 201.959 142.893 199.946 134.465 199.254C121.447 198.248 103.522 199.506 95.283 211.204C91.761 216.236 90.8805 222.525 93.5849 228.122C96.2893 233.72 101.761 237.745 107.421 239.946C118.868 244.475 132.39 242.462 142.642 235.858C152.642 229.38 158.679 218.688 159.371 206.865C177.17 215.041 191.572 230.072 198.616 249.066C198.931 249.695 199.937 249.443 199.686 248.751ZM152.075 225.481C145.094 235.355 133.019 241.141 121.006 241.267C109.434 241.393 92.7044 234.6 93.0189 220.701C93.2076 213.217 98.9937 207.305 105.346 204.286C112.013 201.141 119.748 200.198 127.044 200.072C137.987 199.883 148.616 202.148 158.302 206.361C158.113 213.154 156.038 219.883 152.075 225.481Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M341.447 271.204H322.767C322.075 271.204 322.075 272.273 322.767 272.273H341.447C342.138 272.336 342.138 271.204 341.447 271.204Z\"\n fill=\"#E5ECF4\"\n />\n <path\n d=\"M233.522 75.4814C230.881 89.3179 229.245 103.154 224.025 110.576C213.962 124.915 185.975 137.117 178.428 143.972C170.881 150.764 171.95 151.142 173.019 152.588C173.648 153.469 177.862 151.771 181.132 150.261C177.421 152.211 172.579 155.23 173.711 157.242C174.654 159.003 178.679 157.808 182.201 156.362C178.994 158.123 175.786 160.387 176.981 161.959C178.365 163.783 183.585 161.897 187.107 160.387C184.088 162.022 180.314 164.475 181.635 166.299C183.774 169.192 235.849 149.758 245.157 120.702C248.805 109.318 250.881 94.6009 252.013 80.8274C245.786 79.3808 239.623 77.494 233.522 75.4814Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M182.327 148.437C179.497 150.576 176.226 151.645 172.704 151.582C172.013 151.582 172.013 152.651 172.704 152.651C176.415 152.714 179.937 151.582 182.893 149.318C183.396 148.941 182.893 147.997 182.327 148.437Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M184.906 154.852C183.27 155.607 181.635 156.173 179.937 156.676C178.302 157.117 175.723 158.06 174.151 157.117C173.522 156.802 173.019 157.683 173.585 158.06C175.157 158.94 176.918 158.5 178.553 158.123C180.943 157.62 183.145 156.802 185.346 155.796C186.101 155.481 185.535 154.538 184.906 154.852Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M186.163 158.185C184.277 159.946 180.189 163.217 177.421 161.833C176.792 161.519 176.226 162.462 176.855 162.777C180.252 164.412 184.528 161.141 186.855 158.94C187.421 158.437 186.667 157.682 186.163 158.185Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M253.522 46.0474C253.522 47.1166 253.522 48.3115 253.522 49.6323C253.459 57.3052 253.145 69.1291 252.013 81.6449C251.95 82.148 251.95 82.6512 251.887 83.1543C245.472 82.148 236.981 79.2549 230.629 78.1857C232.893 65.7958 237.736 54.2864 244.151 45.4185C245.157 44.0348 246.226 42.7769 247.421 41.6449L253.522 40.5757C253.522 40.5757 253.522 41.3933 253.585 42.8398C253.522 43.6574 253.522 44.7266 253.522 46.0474Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M184.687 136.932C179.875 137.121 175.253 138.637 170.631 139.837C165.565 141.101 160.5 142.174 155.371 143.122C134.602 146.975 113.264 148.238 92.1786 146.849C74.7659 145.712 55.3904 142.238 43.2332 128.531C32.469 116.403 28.9865 98.1477 34.9384 82.9879C40.7638 68.0807 55.3271 57.6583 70.9668 55.5738V54.5C58.6197 56.0792 46.969 62.7748 39.4341 72.755C29.9996 85.325 28.6066 102.317 34.6218 116.719C41.0804 132.257 55.2004 141.227 71.1568 145.017C80.2747 147.165 89.6458 147.923 98.9537 148.302C109.718 148.744 120.545 148.428 131.246 147.544C143.087 146.533 154.801 144.638 166.388 142.048C172.403 140.658 178.545 138.321 184.687 138.132C190.576 137.942 197.604 140.974 197.921 147.607C197.984 148.302 199.061 148.302 198.997 147.607C198.617 140.216 191.209 136.679 184.687 136.932Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M190 164.091V172.909C190 173.697 191 173.697 191 172.909V164.091C191 163.303 190 163.303 190 164.091Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M199 164.037V172.963C199 173.679 200 173.679 200 172.963V164.037C200 163.321 199 163.321 199 164.037Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M182 166.5V155.275C182 148.205 187.792 142.5 194.969 142.5C202.145 142.5 207.937 148.205 207.937 155.275L208 166.5H182Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M209.483 165.5H180.517C179.828 165.5 179.828 166.5 180.517 166.5H209.483C210.172 166.5 210.172 165.5 209.483 165.5Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M203.711 151.519V161.582C203.711 162.274 204.78 162.274 204.78 161.582V151.519C204.843 150.827 203.711 150.827 203.711 151.519Z\"\n fill=\"white\"\n />\n <path\n d=\"M196.73 134.601C187.421 142.148 183.648 152.274 187.233 153.343C190.818 154.412 203.208 135.67 203.208 135.67L196.73 134.601Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M199.748 139.632C196.855 144.098 193.396 148.374 189.182 151.708L188.239 152.399C187.358 152.84 186.352 152.148 186.478 150.953C186.541 150.45 187.233 149.066 187.421 148.689C187.925 147.557 188.491 146.425 189.057 145.292C190.377 142.777 191.824 140.387 193.459 138.06C193.836 137.494 192.893 136.928 192.516 137.494C191.195 139.443 181.824 152.399 186.855 153.594C189.245 154.16 191.824 150.827 193.27 149.381C196.038 146.55 198.491 143.469 200.692 140.135C201.069 139.569 200.126 139.066 199.748 139.632Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M211.132 135.984C207.736 139.003 203.648 140.89 199.057 141.33C198.365 141.393 198.365 142.462 199.057 142.399C203.899 141.896 208.239 139.947 211.887 136.739C212.39 136.299 211.635 135.544 211.132 135.984Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M238.176 254.224C250.692 253.783 263.208 255.607 275.723 255.796C275.157 254.475 274.654 253.217 274.088 251.897C273.774 251.205 273.459 250.513 273.145 249.821C270.377 243.658 267.484 237.809 264.717 232.274C254.906 235.859 245.031 239.129 234.843 241.331C237.547 246.928 239.12 251.393 238.176 254.224Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M273.899 253.595C276.101 253.783 278.365 253.783 280.566 253.72C284.717 253.658 288.805 253.72 292.956 253.72C299.12 253.783 305.283 253.532 311.384 253.091C307.17 244.161 302.767 235.796 298.553 228.06C289.057 230.639 279.56 232.966 269.937 235.104C273.145 242.525 275.031 249.003 273.899 253.595Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M274.214 250.828C273.836 255.104 271.069 257.997 267.107 259.57C256.352 263.909 259.937 272.84 259.937 272.84H319.874C316.918 265.167 313.648 257.809 310.314 250.765H274.214V250.828Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M307.296 229.758C293.459 203.783 276.289 186.299 281.447 173.72C297.233 135.419 300.818 107.997 291.887 84.3494L267.484 80.0098C267.484 80.0098 269.623 91.5192 268.553 112.337C267.484 133.154 252.201 152.022 250.252 182.337C249.182 199.129 259.937 225.985 266.415 241.582C275.912 238.437 297.547 232.085 307.296 229.758Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M265.786 234.475C252.516 207.934 240.063 187.683 246.415 172.714C263.145 133.406 281.069 109.884 272.075 86.1733C272.075 86.1733 254.843 72.5255 253.774 93.3431C252.704 114.161 218.05 153.846 214.088 183.972C211.321 204.915 223.962 230.513 230.252 245.356C240.252 242.337 256.352 238.815 265.786 234.475Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M282.201 272.84H222.264C222.264 272.84 222.264 260.45 234.654 257.243C237.17 256.614 238.302 255.104 238.491 252.966H274.591C277.296 259.255 279.874 265.922 282.201 272.84Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M274.088 251.896C274.025 252.274 273.962 252.588 273.836 252.966C273.082 255.67 271.195 258.123 268.428 259.444C264.151 261.519 259.119 261.456 256.289 265.922C254.969 268.06 254.402 270.45 254.34 272.84H253.333C253.522 268.311 254.906 264.098 259.182 261.708C262.704 259.758 267.233 259.821 270.252 256.991C271.447 255.922 272.264 254.475 272.704 252.966C273.019 251.959 273.145 250.89 273.082 249.884C273.459 250.513 273.774 251.205 274.088 251.896Z\"\n fill=\"white\"\n />\n <path\n d=\"M281.195 256.739C279.497 255.67 277.799 254.601 276.101 253.532C275.535 253.154 274.969 254.098 275.535 254.475C277.233 255.544 278.931 256.614 280.629 257.683C281.258 258.06 281.761 257.117 281.195 256.739Z\"\n fill=\"white\"\n />\n <path\n d=\"M244.717 257.809C242.893 257.242 241.132 256.739 239.308 256.173C238.616 255.985 238.365 256.991 238.994 257.243C240.818 257.809 242.579 258.312 244.403 258.878C245.094 259.066 245.346 258.06 244.717 257.809Z\"\n fill=\"white\"\n />\n <path\n d=\"M262.516 231.267C261.069 227.682 259.623 224.098 258.239 220.513C257.987 219.884 256.918 220.135 257.17 220.827C258.616 224.412 260.063 227.997 261.447 231.582C261.698 232.211 262.767 231.959 262.516 231.267Z\"\n fill=\"white\"\n />\n <path\n d=\"M268.616 110.073C268.491 110.513 268.365 110.953 268.302 111.394C265.786 121.645 263.208 131.897 260.692 142.148C260.503 142.84 259.497 142.526 259.623 141.834C262.264 131.331 264.843 120.765 267.484 110.262C267.61 109.821 267.736 109.381 267.799 108.878C268.05 109.318 268.365 109.696 268.616 110.073Z\"\n fill=\"white\"\n />\n <path\n d=\"M268.616 110.073C268.491 110.513 268.365 110.953 268.302 111.393C268.05 111.016 267.736 110.639 267.484 110.261C266.478 108.878 265.409 107.494 264.402 106.11C263.962 105.544 264.906 105.041 265.346 105.544C266.163 106.676 266.981 107.808 267.862 108.878C268.05 109.318 268.365 109.695 268.616 110.073Z\"\n fill=\"white\"\n />\n <path\n d=\"M251.698 43.0913C238.994 45.7328 251.509 69.5693 253.836 93.3429L291.887 84.3492C291.887 84.3492 276.415 37.997 251.698 43.0913Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M251.384 79.8839C249.308 73.6574 247.296 67.4939 245.22 61.2675C245.031 60.6386 243.962 60.8901 244.151 61.582C246.226 67.8084 248.239 73.9719 250.314 80.1983C250.566 80.8273 251.572 80.5128 251.384 79.8839Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M332.075 57.431C322.39 48.3744 311.447 43.5945 299.811 41.2046C297.673 46.9908 295.723 52.9027 293.962 58.8147C295.66 59.3807 297.359 60.0096 298.994 60.6386C323.396 69.6323 326.667 87.1794 330.566 97.6197C334.528 108.06 322.642 122.777 326.604 124.538C330.566 126.362 336.289 113.028 336.289 113.028C336.289 113.028 334.843 135.481 337.736 136.173C340.377 136.865 341.698 131.016 342.264 126.488C342.076 130.764 342.327 136.047 344.906 135.859C347.17 135.67 347.862 131.456 347.987 127.494C348.365 130.764 349.245 133.909 351.384 134.098C354.088 134.349 353.962 128.626 353.522 124.349C354.277 128.123 355.66 132.777 358.05 131.896C362.075 130.513 353.962 77.8713 332.075 57.431Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M302.453 41.7707C300.314 47.7455 296.667 58.8147 295.157 64.9782C287.61 60.6386 279.245 57.1795 273.082 53.6575C265.723 49.5065 258.742 47.5569 254.717 46.3619C254.277 46.2361 253.836 46.1103 253.459 46.0474C251.572 45.5443 250.566 45.2298 250.566 45.2298L251.95 43.0286L253.459 40.5757C254.717 40.4499 255.912 40.3871 257.17 40.2613C272.704 38.9405 288.176 38.3745 302.453 41.7707Z\"\n fill=\"#F2F7FC\"\n />\n <path\n d=\"M341.761 123.846C341.384 127.117 341.887 134.853 337.673 135.67C336.981 135.796 337.296 136.865 337.987 136.739C339.874 136.362 341.069 134.915 341.572 133.154C342.453 130.261 342.453 126.928 342.83 123.909C342.956 123.154 341.824 123.154 341.761 123.846Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M346.981 122.966C346.918 124.852 346.855 126.739 346.73 128.626C346.667 129.884 347.107 135.544 344.78 135.041C344.088 134.915 343.837 135.922 344.466 136.11C346.352 136.488 347.296 135.104 347.61 133.406C348.176 130.073 347.925 126.362 348.05 122.966C348.113 122.274 347.044 122.274 346.981 122.966Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M353.648 122.022C353.648 121.33 352.579 121.33 352.579 122.022C352.642 124.035 353.774 132.337 351.132 133.029C350.44 133.217 350.755 134.224 351.447 134.098C353.019 133.657 353.648 132.525 353.774 130.953C354.088 128.06 353.711 124.978 353.648 122.022Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M358.868 119.129C358.805 118.437 357.736 118.5 357.799 119.192C357.987 120.953 360.126 130.639 357.547 131.142C356.855 131.267 357.17 132.274 357.862 132.148C359.56 131.834 360 130.387 360 128.815C359.937 125.67 359.245 122.337 358.868 119.129Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M275.346 54.3493C274.088 53.8462 272.83 53.4059 271.635 52.9028C271.006 52.6512 270.692 53.7204 271.321 53.972C272.579 54.4751 273.836 54.9154 275.031 55.4185C275.723 55.6701 276.038 54.6009 275.346 54.3493Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M256.164 41.9592C254.277 37.4938 251.509 31.2045 251.132 31.1416L241.006 33.28C241.006 33.28 245.849 42.9655 248.113 47.5567C250.629 45.4812 253.333 43.6573 256.164 41.9592Z\"\n fill=\"#D4E8F7\"\n />\n <path\n d=\"M252.704 28.4372C252.704 28.7517 252.642 29.0661 252.579 29.3806C252.516 29.6951 252.453 30.0095 252.39 30.2611C252.39 30.324 252.327 30.3869 252.327 30.4498C252.264 30.7013 252.201 30.9529 252.138 31.2045C250.943 35.041 247.862 38.0598 243.082 38.1856C233.899 38.3743 230.252 30.7013 231.384 24.412C232.138 20.1982 235.094 16.5504 239.937 16.173C240.126 16.173 240.377 16.1102 240.566 16.1102C245.031 16.0473 248.176 17.6825 250.189 20.0724C250.503 20.5127 250.818 20.9529 251.132 21.3932C251.321 21.7076 251.447 22.0221 251.635 22.3366C251.887 22.8397 252.075 23.3429 252.201 23.846C252.641 25.3554 252.83 26.9278 252.704 28.4372Z\"\n fill=\"#D4E8F7\"\n />\n <path d=\"M236.478 29.3179L238.679 34.9782L234.528 35.2298L236.478 29.3179Z\" fill=\"#002957\" />\n <path\n d=\"M239.182 28.0601L239.434 28.8148C239.686 29.4438 238.616 29.6953 238.365 29.0664L238.113 28.3117C237.924 27.6827 238.994 27.4312 239.182 28.0601Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M233.27 28.9406L233.522 29.6954C233.774 30.3243 232.704 30.5759 232.453 29.9469L232.201 29.1922C232.013 28.5004 233.082 28.2488 233.27 28.9406Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M247.044 27.1166C246.918 27.6827 246.855 28.3116 246.73 28.8776C246.604 29.5695 247.673 29.8839 247.736 29.1921C247.862 28.6261 247.924 27.9971 248.05 27.4311C248.176 26.7393 247.17 26.4877 247.044 27.1166Z\"\n fill=\"#0466C8\"\n />\n <path\n d=\"M252.201 23.8461C252.013 23.343 251.824 22.8399 251.635 22.3367C251.509 22.0222 251.321 21.7078 251.132 21.3933C250.88 20.9531 250.566 20.5128 250.189 20.0726C248.176 17.6197 245.031 15.9845 240.566 16.1103C240.377 16.1103 240.126 16.1103 239.937 16.1732C238.491 16.299 234.465 16.9279 232.956 18.8147C228.616 20.1355 225.157 23.909 229.308 25.4814C232.453 26.8021 238.176 25.7958 240.88 24.4751C241.132 25.6072 242.83 28.7518 244.151 28.626C245.723 28.4373 245.472 26.6763 244.969 25.6072C244.906 25.4814 244.843 25.3556 244.78 25.2298C245.094 25.0411 245.409 24.8524 245.786 24.7267C247.925 23.7204 250.818 24.6638 251.509 27.1166C251.824 28.3744 251.698 29.7581 250.943 30.7015C250.88 30.8273 251.824 31.2046 252.138 31.2675C252.201 31.016 252.264 30.7644 252.327 30.5128C252.327 30.4499 252.327 30.387 252.39 30.3241C252.453 30.0097 252.516 29.7581 252.579 29.4436C252.642 29.1292 252.641 28.8147 252.704 28.5002C252.767 26.9279 252.641 25.3556 252.201 23.8461Z\"\n fill=\"#002957\"\n />\n <path\n d=\"M257.17 40.2612L256.226 42.7141L254.78 46.3619L253.522 49.5694L251.635 54.4122L246.478 48.1858L244.151 45.4185C245.157 44.0348 246.226 42.777 247.421 41.6449L253.522 40.5757C254.717 40.4499 255.912 40.3241 257.17 40.2612Z\"\n fill=\"#D4E8F7\"\n />\n </svg>\n</vl-empty-state>\n", styles: [":host{display:flex;height:100%}\n"] }]
2073
- }] });
2074
-
2075
- class EmptyAccountModule {
2076
- }
2077
- EmptyAccountModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
2078
- EmptyAccountModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountModule, declarations: [EmptyAccountComponent], imports: [CommonModule, PreviewModule, LoaderModule, EmptyStateModule], exports: [EmptyAccountComponent] });
2079
- EmptyAccountModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountModule, imports: [CommonModule, PreviewModule, LoaderModule, EmptyStateModule] });
2080
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: EmptyAccountModule, decorators: [{
2081
- type: NgModule,
2082
- args: [{
2083
- declarations: [EmptyAccountComponent],
2084
- imports: [CommonModule, PreviewModule, LoaderModule, EmptyStateModule],
2085
- exports: [EmptyAccountComponent],
2086
- }]
2087
- }] });
2088
-
2089
1424
  class ProductComponent {
2090
- constructor(contextService, configurationRuntimeService, configurationService, configurationState, quoteDraftService, integrationState, customizationService) {
1425
+ constructor(contextService, configurationRuntimeService, configurationService, configurationStateService, quoteDraftService, flowInfoService, flowStateService, integrationState, customizationService) {
2091
1426
  this.contextService = contextService;
2092
1427
  this.configurationRuntimeService = configurationRuntimeService;
2093
1428
  this.configurationService = configurationService;
2094
- this.configurationState = configurationState;
1429
+ this.configurationStateService = configurationStateService;
2095
1430
  this.quoteDraftService = quoteDraftService;
1431
+ this.flowInfoService = flowInfoService;
1432
+ this.flowStateService = flowStateService;
2096
1433
  this.integrationState = integrationState;
2097
1434
  this.customizationService = customizationService;
2098
1435
  this.uiDefinition$ = new BehaviorSubject(undefined);
@@ -2112,18 +1449,28 @@ class ProductComponent {
2112
1449
  }));
2113
1450
  }
2114
1451
  init$() {
2115
- return this.quoteDraftService.quoteDraft$.pipe(first(), switchMap(quote => {
1452
+ let quoteDraft$;
1453
+ if (this.flowInfoService.isLegacy || !this.flowInfoService.isStateful) {
1454
+ quoteDraft$ = this.quoteDraftService.quoteDraft$;
1455
+ }
1456
+ else {
1457
+ quoteDraft$ = of(undefined);
1458
+ }
1459
+ return quoteDraft$.pipe(first(), switchMap(quote => {
2116
1460
  const contextProperties = this.contextService.resolve().properties;
2117
1461
  const productId = contextProperties.productId;
2118
1462
  if (!productId) {
2119
1463
  throw new Error(`Unable to start configuration for 'productId == null'`);
2120
1464
  }
2121
- const lineItemId = this.getLineItemId(quote, productId, contextProperties.lineItemId);
2122
- const currentStateItem = quote.currentState.find(({ id }) => id === lineItemId);
2123
- if (currentStateItem) {
2124
- this.configurationService.setConfigurableRamp(currentStateItem);
1465
+ if (!quote) {
1466
+ const offeringId = contextProperties.offeringId;
1467
+ return this.configurationRuntimeService.init({ productId, offeringId });
2125
1468
  }
2126
- const { offeringId } = currentStateItem ?? {};
1469
+ const lineItem = this.getLineItem(quote, productId, contextProperties.lineItemId);
1470
+ if (lineItem) {
1471
+ this.configurationService.setConfigurableRamp(lineItem);
1472
+ }
1473
+ const { offeringId } = lineItem ?? {};
2127
1474
  return this.configurationRuntimeService.init({ productId, offeringId });
2128
1475
  }), switchMap(() => this.customizeUI$()), tap(() => {
2129
1476
  const uiDefinition = this.configurationRuntimeService.runtimeContext?.uiDefinitionContainer?.source;
@@ -2138,23 +1485,27 @@ class ProductComponent {
2138
1485
  this.configurationRuntimeService.initializationProps.attributesMap =
2139
1486
  this.integrationState.state.guidedSelling;
2140
1487
  }
2141
- }), switchMap(() => this.configurationState.init$()));
1488
+ }), switchMap(() => this.configurationStateService.init$()));
2142
1489
  }
2143
- getLineItemId(quote, productId, lineItemId) {
1490
+ getLineItem(quote, productId, lineItemId) {
2144
1491
  // search by lineItemId first
2145
- let id = quote.currentState.find(li => li.id === lineItemId)?.id;
2146
- if (!id && this.quoteDraftService.isStandalone) {
2147
- id = quote.currentState.find(li => li.productId === productId)?.id;
1492
+ let li = quote.currentState.find(li => li.id === lineItemId);
1493
+ if (!li && this.quoteDraftService.isStandalone) {
1494
+ li = quote.currentState.find(li => li.productId === productId);
1495
+ }
1496
+ // If still not found, is could be an asset
1497
+ if (!li) {
1498
+ li = this.quoteDraftService.assetsState?.currentState.find(li => li.id === lineItemId);
2148
1499
  }
2149
- return id;
1500
+ return li;
2150
1501
  }
2151
1502
  }
2152
- ProductComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductComponent, deps: [{ token: i1$2.ContextService }, { token: i1$2.ConfigurationRuntimeService }, { token: i1$2.ConfigurationService }, { token: i1$2.ConfigurationState }, { token: i1$2.QuoteDraftService }, { token: i5.IntegrationState }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1503
+ ProductComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductComponent, deps: [{ token: i2.ContextService }, { token: i2.ConfigurationRuntimeService }, { token: i2.ConfigurationService }, { token: i2.ConfigurationStateService }, { token: i2.QuoteDraftService }, { token: i2.FlowInfoService }, { token: i2.FlowStateService }, { token: i2.IntegrationState }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
2153
1504
  ProductComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: ProductComponent, selector: "vl-flow-product", ngImport: i0, template: "<vl-cms-preview [uiDefinition]=\"$any(uiDefinition$ | async)\" [config]=\"config\"></vl-cms-preview>\n", styles: [""], dependencies: [{ kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2154
1505
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ProductComponent, decorators: [{
2155
1506
  type: Component,
2156
1507
  args: [{ selector: 'vl-flow-product', changeDetection: ChangeDetectionStrategy.OnPush, template: "<vl-cms-preview [uiDefinition]=\"$any(uiDefinition$ | async)\" [config]=\"config\"></vl-cms-preview>\n" }]
2157
- }], ctorParameters: function () { return [{ type: i1$2.ContextService }, { type: i1$2.ConfigurationRuntimeService }, { type: i1$2.ConfigurationService }, { type: i1$2.ConfigurationState }, { type: i1$2.QuoteDraftService }, { type: i5.IntegrationState }, { type: undefined, decorators: [{
1508
+ }], ctorParameters: function () { return [{ type: i2.ContextService }, { type: i2.ConfigurationRuntimeService }, { type: i2.ConfigurationService }, { type: i2.ConfigurationStateService }, { type: i2.QuoteDraftService }, { type: i2.FlowInfoService }, { type: i2.FlowStateService }, { type: i2.IntegrationState }, { type: undefined, decorators: [{
2158
1509
  type: Optional
2159
1510
  }, {
2160
1511
  type: Inject,
@@ -2180,26 +1531,29 @@ class RecordNotFoundComponent {
2180
1531
  this.router = router;
2181
1532
  this.route = route;
2182
1533
  this.subMessage = '';
1534
+ this.type = '';
2183
1535
  const navigation = this.router.getCurrentNavigation();
2184
1536
  const { state } = navigation?.extras || {};
2185
1537
  this.message = state?.['message'];
1538
+ this.type = state?.['type'] || '';
1539
+ this.details = state?.['details'];
2186
1540
  if (typeof this.message === 'string') {
2187
1541
  this.subMessage = this.message.includes('/describe') ? 'A potential problem with permissions' : '';
2188
1542
  }
2189
1543
  }
2190
1544
  }
2191
- RecordNotFoundComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundComponent, deps: [{ token: i1$1.Router }, { token: i1$1.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Component });
2192
- RecordNotFoundComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: RecordNotFoundComponent, selector: "vl-flow-record-not-found", ngImport: i0, template: "<div class=\"row\">\n <div class=\"col-md-12\">\n <div class=\"message-wrapper\">\n <div class=\"msg\">\n <div *ngIf=\"message; else defaultMessage\" class=\"message-title\">\n <p>{{ message }}</p>\n\n <p *ngIf=\"subMessage\" class=\"message-title\">{{ subMessage }}</p>\n </div>\n\n <ng-template #defaultMessage>Record not found</ng-template>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;padding:24px 54px}\n"], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
1545
+ RecordNotFoundComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundComponent, deps: [{ token: i1$2.Router }, { token: i1$2.ActivatedRoute }], target: i0.ɵɵFactoryTarget.Component });
1546
+ RecordNotFoundComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: RecordNotFoundComponent, selector: "vl-flow-record-not-found", ngImport: i0, template: "<div class=\"row\">\n <div class=\"col-md-12\">\n <div class=\"message-wrapper\">\n <div *ngIf=\"message; else defaultMessage\">\n <p class=\"text message-text\" [class]=\"type\">{{ message }}</p>\n\n <p *ngIf=\"subMessage\" class=\"text\">{{ subMessage }}</p>\n\n <div *ngIf=\"details && details.length > 0\">\n <div *ngFor=\"let detail of details\" class=\"text\">{{ detail }}</div>\n </div>\n </div>\n\n <ng-template #defaultMessage>Record not found</ng-template>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;padding:24px 54px}:host .text{white-space:pre-line;word-break:break-word}:host .message-text.error{color:var(--vl-error-text-color)}\n"], dependencies: [{ kind: "directive", type: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2193
1547
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundComponent, decorators: [{
2194
1548
  type: Component,
2195
- args: [{ selector: 'vl-flow-record-not-found', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"row\">\n <div class=\"col-md-12\">\n <div class=\"message-wrapper\">\n <div class=\"msg\">\n <div *ngIf=\"message; else defaultMessage\" class=\"message-title\">\n <p>{{ message }}</p>\n\n <p *ngIf=\"subMessage\" class=\"message-title\">{{ subMessage }}</p>\n </div>\n\n <ng-template #defaultMessage>Record not found</ng-template>\n </div>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;padding:24px 54px}\n"] }]
2196
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: i1$1.ActivatedRoute }]; } });
1549
+ args: [{ selector: 'vl-flow-record-not-found', changeDetection: ChangeDetectionStrategy.OnPush, template: "<div class=\"row\">\n <div class=\"col-md-12\">\n <div class=\"message-wrapper\">\n <div *ngIf=\"message; else defaultMessage\">\n <p class=\"text message-text\" [class]=\"type\">{{ message }}</p>\n\n <p *ngIf=\"subMessage\" class=\"text\">{{ subMessage }}</p>\n\n <div *ngIf=\"details && details.length > 0\">\n <div *ngFor=\"let detail of details\" class=\"text\">{{ detail }}</div>\n </div>\n </div>\n\n <ng-template #defaultMessage>Record not found</ng-template>\n </div>\n </div>\n</div>\n", styles: [":host{display:block;padding:24px 54px}:host .text{white-space:pre-line;word-break:break-word}:host .message-text.error{color:var(--vl-error-text-color)}\n"] }]
1550
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: i1$2.ActivatedRoute }]; } });
2197
1551
 
2198
1552
  const routes = [{ path: '', component: RecordNotFoundComponent }];
2199
1553
  class RecordNotFoundModule {
2200
1554
  }
2201
1555
  RecordNotFoundModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
2202
- RecordNotFoundModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundModule, declarations: [RecordNotFoundComponent], imports: [CommonModule, i1$1.RouterModule] });
1556
+ RecordNotFoundModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundModule, declarations: [RecordNotFoundComponent], imports: [CommonModule, i1$2.RouterModule] });
2203
1557
  RecordNotFoundModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundModule, imports: [CommonModule, RouterModule.forChild(routes)] });
2204
1558
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RecordNotFoundModule, decorators: [{
2205
1559
  type: NgModule,
@@ -2313,7 +1667,7 @@ class RemoteComponent {
2313
1667
  let items = [lineItem];
2314
1668
  for (let i = 0; i < items.length; i++) {
2315
1669
  const item = items[i];
2316
- const children = options.get(item.id);
1670
+ const children = item && options.get(item.id);
2317
1671
  if (children) {
2318
1672
  item.lineItems = children;
2319
1673
  items = items.concat(children);
@@ -2386,7 +1740,7 @@ class RemoteComponent {
2386
1740
  if (rootOption) {
2387
1741
  const featureOptions = optionConfigurations[rootOption.featureName] ?? (optionConfigurations[rootOption.featureName] = []);
2388
1742
  const originOption = !rootOption.hasChildren
2389
- ? optionConfigurations[rootOption.featureName].find(({ optionId }) => optionId === rootOption.optionId)
1743
+ ? optionConfigurations[rootOption.featureName]?.find(({ optionId }) => optionId === rootOption.optionId)
2390
1744
  : undefined;
2391
1745
  const option = originOption ?? {};
2392
1746
  option.optionId = option.optionId ?? rootOption.optionId;
@@ -2439,8 +1793,9 @@ class RemoteComponent {
2439
1793
  }
2440
1794
  }
2441
1795
  for (const option of notBundleOptions) {
2442
- if (bundleOptionsByOptionId[option.configuredProductId]) {
2443
- bundleOptionsByOptionId[option.configuredProductId].hasChildren = true;
1796
+ const bundleOption = bundleOptionsByOptionId[option.configuredProductId];
1797
+ if (bundleOption) {
1798
+ bundleOption.hasChildren = true;
2444
1799
  }
2445
1800
  }
2446
1801
  }
@@ -2448,7 +1803,7 @@ class RemoteComponent {
2448
1803
  let items = lineItem.lineItems ?? [];
2449
1804
  for (let i = 0; i < items.length; i++) {
2450
1805
  const item = items[i];
2451
- if (item.lineItems) {
1806
+ if (item?.lineItems) {
2452
1807
  items = items.concat(item.lineItems.map(li => ({ ...li, parentLineItem: item })));
2453
1808
  }
2454
1809
  }
@@ -2514,7 +1869,7 @@ class RemoteComponent {
2514
1869
  .forEach(mapping => {
2515
1870
  const attribute = lineItem.attributes.find(a => a.name === mapping.name);
2516
1871
  if (attribute) {
2517
- new Set(mapping.properties[propertyName].split(',')).forEach(k => (result[k] = attribute.value));
1872
+ new Set(mapping.properties[propertyName]?.split(',')).forEach(k => (result[k] = attribute.value));
2518
1873
  }
2519
1874
  });
2520
1875
  return result;
@@ -2530,12 +1885,12 @@ class RemoteComponent {
2530
1885
  return optionConfigurations;
2531
1886
  }
2532
1887
  }
2533
- RemoteComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RemoteComponent, deps: [{ token: i1$2.ContextService }, { token: i1$2.QuoteDraftService }, { token: i1$2.ConfigurationRuntimeService }, { token: i1$2.ConfigurationService }, { token: i1$2.ConfigurationState }, { token: i8.MessageService }, { token: i5.IntegrationState }, { token: i4.Location }], target: i0.ɵɵFactoryTarget.Component });
1888
+ RemoteComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RemoteComponent, deps: [{ token: i2.ContextService }, { token: i2.QuoteDraftService }, { token: i2.ConfigurationRuntimeService }, { token: i2.ConfigurationService }, { token: i2.ConfigurationStateService }, { token: i2$2.MessageService }, { token: i2.IntegrationState }, { token: i4.Location }], target: i0.ɵɵFactoryTarget.Component });
2534
1889
  RemoteComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: RemoteComponent, selector: "vl-flow-remote", ngImport: i0, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n", styles: [""], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2535
1890
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: RemoteComponent, decorators: [{
2536
1891
  type: Component,
2537
1892
  args: [{ selector: 'vl-flow-remote', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n" }]
2538
- }], ctorParameters: function () { return [{ type: i1$2.ContextService }, { type: i1$2.QuoteDraftService }, { type: i1$2.ConfigurationRuntimeService }, { type: i1$2.ConfigurationService }, { type: i1$2.ConfigurationState }, { type: i8.MessageService }, { type: i5.IntegrationState }, { type: i4.Location }]; } });
1893
+ }], ctorParameters: function () { return [{ type: i2.ContextService }, { type: i2.QuoteDraftService }, { type: i2.ConfigurationRuntimeService }, { type: i2.ConfigurationService }, { type: i2.ConfigurationStateService }, { type: i2$2.MessageService }, { type: i2.IntegrationState }, { type: i4.Location }]; } });
2539
1894
 
2540
1895
  class RemoteModule {
2541
1896
  }
@@ -2552,17 +1907,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2552
1907
  }] });
2553
1908
 
2554
1909
  class ShoppingCartComponent {
2555
- constructor(templatesApi, cdr, toastService, flowService, customizationService) {
1910
+ constructor(templatesApi, cdr, toastService, flowInfo, customizationService) {
2556
1911
  this.templatesApi = templatesApi;
2557
1912
  this.cdr = cdr;
2558
1913
  this.toastService = toastService;
2559
- this.flowService = flowService;
1914
+ this.flowInfo = flowInfo;
2560
1915
  this.customizationService = customizationService;
2561
1916
  this.uiDefinition = undefined;
2562
1917
  this.state$ = new BehaviorSubject({ loading: true, failure: false });
2563
1918
  this.templateApiName = '';
2564
1919
  this.destroyed$ = new Subject();
2565
- this.templateApiName = this.flowService.flow?.properties.templates.shoppingCart ?? '';
1920
+ this.templateApiName = this.flowInfo.flow?.properties.templates?.shoppingCart ?? '';
2566
1921
  }
2567
1922
  ngOnInit() {
2568
1923
  this.generateUIDefinition$()
@@ -2573,7 +1928,7 @@ class ShoppingCartComponent {
2573
1928
  this.uiDefinition = uiDef;
2574
1929
  this.state$.next({ loading: false, failure: false });
2575
1930
  }), catchError(err => {
2576
- const message = 'Failed to resolve Shopping cart component. ' + (err.message ?? '');
1931
+ const message = 'Failed to resolve Shopping Cart component. ' + (err.message ?? '');
2577
1932
  this.toastService.add({ severity: ToastType.error, summary: message, sticky: true });
2578
1933
  this.uiDefinition = undefined;
2579
1934
  this.state$.next({ loading: false, failure: true });
@@ -2585,37 +1940,36 @@ class ShoppingCartComponent {
2585
1940
  this.destroyed$.next();
2586
1941
  this.destroyed$.complete();
2587
1942
  }
2588
- getTemplateRootComponent$(template) {
2589
- return this.templatesApi
2590
- .fetchComponents$(template.id)
2591
- .pipe(map(components => components.find(c => c.type === UITemplateComponentType.ROOT) ?? undefined));
2592
- }
2593
- getLocalShoppingCartComponentMeta$() {
2594
- if (!this.customizationService?.getShoppingCartComponent) {
1943
+ getLocalMeta$() {
1944
+ if (!this.customizationService?.getTemplateComponents) {
2595
1945
  return of(undefined);
2596
1946
  }
2597
- return this.customizationService?.getShoppingCartComponent(this.templateApiName).pipe(map(component => {
2598
- if (!component) {
1947
+ return this.customizationService?.getTemplateComponents(this.templateApiName).pipe(map(components => {
1948
+ if (!components) {
2599
1949
  return;
2600
1950
  }
2601
- return {
1951
+ return components.map(component => ({
2602
1952
  html: component.html,
2603
1953
  css: component.css,
2604
1954
  js: component.js,
2605
1955
  json: component.json,
2606
- };
1956
+ }));
2607
1957
  }));
2608
1958
  }
2609
- getShoppingCartComponentMeta$() {
2610
- return this.templatesApi.fetchTemplates$().pipe(map(templates => templates.find(template => template.name === this.templateApiName)), switchMap(template => (template ? this.getTemplateRootComponent$(template) : of(undefined))), switchMap(component => component ? this.templatesApi.fetchComponentAttachments$(component.uiTemplateId, component) : of(undefined)));
1959
+ getOrgMeta$() {
1960
+ const template = this.flowInfo.templates.SHOPPING_CART;
1961
+ if (!template) {
1962
+ return of(undefined);
1963
+ }
1964
+ return this.templatesApi.fetchComponentsAttachments$(template.id);
2611
1965
  }
2612
1966
  generateUIDefinition$() {
2613
1967
  return of(undefined).pipe(tap(() => {
2614
1968
  if (!this.templateApiName) {
2615
- throw new Error("Flow Query parameter 'cartTemplateApiName' is missing.");
1969
+ throw new Error("Flow 'shoppingCart' template is not defined.");
2616
1970
  }
2617
- }), switchMap(() => this.getLocalShoppingCartComponentMeta$()), switchMap(meta => (meta ? of(meta) : this.getShoppingCartComponentMeta$())), map(meta => {
2618
- if (!meta) {
1971
+ }), switchMap(() => this.getLocalMeta$()), switchMap(metaList => (metaList ? of(metaList) : this.getOrgMeta$())), map(metaList => {
1972
+ if (!metaList) {
2619
1973
  return;
2620
1974
  }
2621
1975
  const uiDef = {
@@ -2624,25 +1978,23 @@ class ShoppingCartComponent {
2624
1978
  primary: true,
2625
1979
  type: 'DEFAULT',
2626
1980
  version: 2,
2627
- children: [
2628
- {
2629
- children: [],
2630
- template: meta.html && btoa(meta.html),
2631
- script: meta.js && btoa(meta.js),
2632
- styles: meta.css && btoa(meta.css),
2633
- },
2634
- ],
1981
+ children: metaList.map(meta => ({
1982
+ children: [],
1983
+ template: meta.html && btoaSafe(meta.html),
1984
+ script: meta.js && btoaSafe(meta.js),
1985
+ styles: meta.css && btoaSafe(meta.css),
1986
+ })),
2635
1987
  };
2636
1988
  return uiDef;
2637
1989
  }));
2638
1990
  }
2639
1991
  }
2640
- ShoppingCartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ShoppingCartComponent, deps: [{ token: i2.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: FlowService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
1992
+ ShoppingCartComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ShoppingCartComponent, deps: [{ token: i1$1.UITemplatesApiService }, { token: i0.ChangeDetectorRef }, { token: i2$1.ToastService }, { token: i2.FlowInfoService }, { token: FLOW_CUSTOMIZATION, optional: true }], target: i0.ɵɵFactoryTarget.Component });
2641
1993
  ShoppingCartComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.9", type: ShoppingCartComponent, selector: "vl-flow-shopping-cart", ngImport: i0, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n", styles: [""], dependencies: [{ kind: "directive", type: i4.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i5.PreviewComponent, selector: "vl-cms-preview", inputs: ["uiDefinition", "config"] }, { kind: "component", type: i2$1.LoaderComponent, selector: "vl-loader", inputs: ["label", "overlayVisible"] }, { kind: "pipe", type: i4.AsyncPipe, name: "async" }], changeDetection: i0.ChangeDetectionStrategy.OnPush });
2642
1994
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: ShoppingCartComponent, decorators: [{
2643
1995
  type: Component,
2644
1996
  args: [{ selector: 'vl-flow-shopping-cart', changeDetection: ChangeDetectionStrategy.OnPush, template: "<ng-container *ngIf=\"state$ | async as state\">\n <vl-loader *ngIf=\"state.loading; else content\" [label]=\"'Loading UI'\"></vl-loader>\n\n <ng-template #content>\n <ng-container *ngIf=\"!state.failure\">\n <vl-cms-preview [uiDefinition]=\"uiDefinition\"></vl-cms-preview>\n </ng-container>\n </ng-template>\n</ng-container>\n" }]
2645
- }], ctorParameters: function () { return [{ type: i2.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: FlowService }, { type: undefined, decorators: [{
1997
+ }], ctorParameters: function () { return [{ type: i1$1.UITemplatesApiService }, { type: i0.ChangeDetectorRef }, { type: i2$1.ToastService }, { type: i2.FlowInfoService }, { type: undefined, decorators: [{
2646
1998
  type: Optional
2647
1999
  }, {
2648
2000
  type: Inject,
@@ -2664,144 +2016,120 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2664
2016
  }] });
2665
2017
 
2666
2018
  class FlowResolver {
2667
- constructor(router, flowsApiService, routerService, contextService, flowService) {
2019
+ constructor(router, routerService, contextService, flowInfo) {
2668
2020
  this.router = router;
2669
- this.flowsApiService = flowsApiService;
2670
2021
  this.routerService = routerService;
2671
2022
  this.contextService = contextService;
2672
- this.flowService = flowService;
2673
- }
2674
- handleError(route, message, queryParams) {
2675
- const parentUrl = this.routerService.getFlowRootPath(route);
2676
- return this.router.navigate([parentUrl, '404'], {
2677
- queryParams,
2678
- state: {
2679
- message: message,
2680
- },
2681
- });
2023
+ this.flowInfo = flowInfo;
2682
2024
  }
2683
2025
  resolve(route) {
2684
2026
  const { queryParams } = route;
2685
- const { flowId } = queryParams;
2686
- if (!flowId) {
2687
- return this.handleError(route);
2688
- }
2689
- return this.flowsApiService.getFlow(flowId).pipe(map$1(flow => {
2690
- if (!flow) {
2691
- return this.handleError(route, `Flow with flowId=${flowId} is not defined`);
2692
- }
2693
- const { properties } = flow;
2694
- const { queryParams: flowQueryParams, entryPath } = properties;
2695
- const mergedQueryParams = {
2696
- ...queryParams,
2697
- ...flowQueryParams,
2698
- };
2699
- const contextProperties = Object.entries({
2700
- ...mergedQueryParams,
2701
- ...getDefaultProperties({ flowParams: properties }),
2702
- }).reduce((trunk, [key, value]) => ({ ...trunk, [key]: String(value) }), {});
2703
- this.contextService.update({ properties: contextProperties });
2704
- this.flowService.setFlow(flow);
2705
- const parentUrl = this.routerService.getFlowRootPath(route);
2706
- const entryUrl = String(entryPath ?? '')
2707
- .split('/')
2708
- .filter(Boolean);
2709
- return this.router
2710
- .navigate([parentUrl, ...entryUrl], {
2711
- queryParams: mergedQueryParams,
2712
- replaceUrl: true,
2713
- })
2714
- .catch(e => {
2715
- const message = e instanceof HttpErrorResponse ? e.error.message : e;
2716
- return this.handleError(route, message, mergedQueryParams);
2717
- });
2027
+ const flow = this.flowInfo.flow;
2028
+ if (!flow) {
2029
+ return of(false);
2030
+ }
2031
+ const { properties } = flow;
2032
+ const { queryParams: flowQueryParams, entryPath } = properties;
2033
+ const mergedQueryParams = {
2034
+ ...queryParams,
2035
+ ...flowQueryParams,
2036
+ };
2037
+ const contextProperties = Object.entries({
2038
+ ...mergedQueryParams,
2039
+ ...getDefaultProperties({ flowParams: properties }),
2040
+ }).reduce((trunk, [key, value]) => ({ ...trunk, [key]: String(value) }), {});
2041
+ this.contextService.update({ properties: contextProperties });
2042
+ this.flowInfo.flow = flow;
2043
+ const parentUrl = this.routerService.getFlowRootPath(route);
2044
+ const entryUrl = String(entryPath ?? '')
2045
+ .split('/')
2046
+ .filter(Boolean);
2047
+ return from(this.router.navigate([parentUrl, ...entryUrl], {
2048
+ queryParams: mergedQueryParams,
2049
+ replaceUrl: true,
2050
+ })).pipe(catchError$1(e => {
2051
+ const message = e instanceof HttpErrorResponse ? e.error.message : e;
2052
+ const errorDetails = isVeloceError(e.error) ? extractErrorDetails(e.error) : [];
2053
+ return this.routerService.showErrorPage$(message, errorDetails);
2718
2054
  }));
2719
2055
  }
2720
2056
  }
2721
- FlowResolver.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowResolver, deps: [{ token: i1$1.Router }, { token: i2.FlowsApiService }, { token: FlowRouterService }, { token: i1$2.ContextService }, { token: FlowService }], target: i0.ɵɵFactoryTarget.Injectable });
2057
+ FlowResolver.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowResolver, deps: [{ token: i1$2.Router }, { token: FlowRouterService }, { token: i2.ContextService }, { token: i2.FlowInfoService }], target: i0.ɵɵFactoryTarget.Injectable });
2722
2058
  FlowResolver.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowResolver });
2723
2059
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowResolver, decorators: [{
2724
2060
  type: Injectable
2725
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: i2.FlowsApiService }, { type: FlowRouterService }, { type: i1$2.ContextService }, { type: FlowService }]; } });
2061
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: FlowRouterService }, { type: i2.ContextService }, { type: i2.FlowInfoService }]; } });
2726
2062
 
2727
2063
  class QuoteResolver {
2728
- constructor(router, quoteDraftService, routerService, contextService, flowConfiguration, integrationState, flowService) {
2064
+ constructor(router, quoteDraftService, routerService, contextService, flowInfo, flowStateService) {
2729
2065
  this.router = router;
2730
2066
  this.quoteDraftService = quoteDraftService;
2731
2067
  this.routerService = routerService;
2732
2068
  this.contextService = contextService;
2733
- this.flowConfiguration = flowConfiguration;
2734
- this.integrationState = integrationState;
2735
- this.flowService = flowService;
2736
- }
2737
- handleError(route, message) {
2738
- const parentUrl = this.routerService.getFlowRootPath(route);
2739
- return from(this.router.navigate([parentUrl, '404'], { state: { message } }));
2740
- }
2741
- calculate$() {
2742
- const quoteDraft = this.quoteDraftService.quoteDraft;
2743
- if (!this.quoteDraftService.isStandalone && quoteDraft && !this.accountHasNotAssets()) {
2744
- return this.flowConfiguration.calculate$(quoteDraft);
2745
- }
2746
- return of(undefined);
2069
+ this.flowInfo = flowInfo;
2070
+ this.flowStateService = flowStateService;
2747
2071
  }
2748
2072
  resolve(route) {
2749
- const { headerId } = this.contextService.resolve();
2750
- const quote = this.quoteDraftService.quoteDraft;
2751
- if (quote && quote.quoteId === headerId) {
2073
+ const flow = this.flowInfo.flow;
2074
+ if (!flow) {
2075
+ return of(false);
2076
+ }
2077
+ if (this.flowStateService.isInitialized()) {
2752
2078
  return of(true);
2753
2079
  }
2754
- const queryParams = route.queryParams;
2755
- return this.quoteDraftService.init(headerId, queryParams).pipe(switchMap(() => this.calculate$()), tap(() => {
2756
- const canNavigate = !this.flowService.flow?.properties.suppressInitialNavigation;
2757
- if (!this.quoteDraftService.isStandalone && canNavigate) {
2758
- if (this.shouldNavigateToAssets()) {
2759
- this.changeNavigation('/assets', route);
2760
- }
2761
- else if (!this.quoteDraftService.isInitialized && this.quoteDraftService.hasProducts) {
2762
- this.changeNavigation('/cart', route);
2763
- }
2764
- }
2765
- this.initializeModifiedAssetsMap();
2766
- this.quoteDraftService.isInitialized = true;
2767
- this.quoteDraftService.hasUnsavedChanges = false;
2768
- }), catchError(e => {
2080
+ return this.flowStateService.init$().pipe(switchMap(() => this.checkDynamicNavigation$(route)), catchError(e => {
2769
2081
  const message = e instanceof HttpErrorResponse ? e.error.message : e;
2770
- return this.handleError(route, message);
2082
+ const errorDetails = isVeloceError(e.error) ? extractErrorDetails(e.error) : [];
2083
+ return this.routerService.showErrorPage$(message, errorDetails);
2771
2084
  }));
2772
2085
  }
2773
- changeNavigation(path, route) {
2774
- const parentUrl = this.routerService.getFlowRootPath(route);
2775
- this.router
2776
- .navigate([parentUrl + path], {
2777
- queryParams: route.queryParams,
2778
- replaceUrl: true,
2779
- })
2780
- .catch(e => {
2781
- const message = e instanceof HttpErrorResponse ? e.error.message : e;
2782
- return this.handleError(route, message);
2783
- });
2784
- }
2785
- shouldNavigateToAssets() {
2786
- if (this.contextService.mode === ConfigurationContextMode.ACCOUNT) {
2787
- return true;
2086
+ checkDynamicNavigation$(route) {
2087
+ const flow = this.flowInfo.flow;
2088
+ if (!flow) {
2089
+ return of(true);
2788
2090
  }
2789
- return this.quoteDraftService.quoteDraft?.currentState.some(li => !!li.assetId) ?? false;
2790
- }
2791
- accountHasNotAssets() {
2792
- return this.contextService.mode === ConfigurationContextMode.ACCOUNT && !this.quoteDraftService.hasProducts;
2091
+ if (flow.properties.suppressInitialNavigation || flow.properties.standalone) {
2092
+ return of(true);
2093
+ }
2094
+ return this.getNavigateTo().pipe(switchMap(navigateTo => {
2095
+ if (!navigateTo) {
2096
+ return of(true);
2097
+ }
2098
+ const parentUrl = this.routerService.getFlowRootPath(route);
2099
+ return from(this.router.navigate([parentUrl + navigateTo], {
2100
+ queryParams: route.queryParams,
2101
+ replaceUrl: true,
2102
+ }));
2103
+ }));
2793
2104
  }
2794
- initializeModifiedAssetsMap() {
2795
- this.integrationState.patchState({
2796
- modifiedAssets: generateModifiedAssetsMap(this.quoteDraftService.currentState),
2797
- });
2105
+ getNavigateTo() {
2106
+ const flow = this.flowInfo.flow;
2107
+ let navigateTo;
2108
+ if (flow?.properties.stateful) {
2109
+ return this.flowStateService.select$(UITemplateType.FLOW_ENGINE, 'NAVIGATE_TO').pipe(map(r => {
2110
+ if (r.success) {
2111
+ return r.result;
2112
+ }
2113
+ return '';
2114
+ }));
2115
+ }
2116
+ else {
2117
+ const isAccountMode = this.contextService.mode === ConfigurationContextMode.ACCOUNT;
2118
+ if (isAccountMode || this.quoteDraftService.hasAssets) {
2119
+ navigateTo = '/assets';
2120
+ }
2121
+ else if (this.quoteDraftService.hasProducts) {
2122
+ navigateTo = '/cart';
2123
+ }
2124
+ }
2125
+ return of(navigateTo);
2798
2126
  }
2799
2127
  }
2800
- QuoteResolver.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: QuoteResolver, deps: [{ token: i1$1.Router }, { token: i1$2.QuoteDraftService }, { token: FlowRouterService }, { token: i1$2.ContextService }, { token: i1$2.FlowConfigurationService }, { token: i5.IntegrationState }, { token: FlowService }], target: i0.ɵɵFactoryTarget.Injectable });
2128
+ QuoteResolver.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: QuoteResolver, deps: [{ token: i1$2.Router }, { token: i2.QuoteDraftService }, { token: FlowRouterService }, { token: i2.ContextService }, { token: i2.FlowInfoService }, { token: i2.FlowStateService }], target: i0.ɵɵFactoryTarget.Injectable });
2801
2129
  QuoteResolver.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: QuoteResolver });
2802
2130
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: QuoteResolver, decorators: [{
2803
2131
  type: Injectable
2804
- }], ctorParameters: function () { return [{ type: i1$1.Router }, { type: i1$2.QuoteDraftService }, { type: FlowRouterService }, { type: i1$2.ContextService }, { type: i1$2.FlowConfigurationService }, { type: i5.IntegrationState }, { type: FlowService }]; } });
2132
+ }], ctorParameters: function () { return [{ type: i1$2.Router }, { type: i2.QuoteDraftService }, { type: FlowRouterService }, { type: i2.ContextService }, { type: i2.FlowInfoService }, { type: i2.FlowStateService }]; } });
2805
2133
 
2806
2134
  const rootRoute = {
2807
2135
  id: VELOCE_FLOW_ROOT_ROUTE,
@@ -2812,7 +2140,7 @@ const rootRoute = {
2812
2140
  children: [
2813
2141
  {
2814
2142
  path: '',
2815
- canActivate: [initFlow],
2143
+ canActivate: [keepFlowInitialized],
2816
2144
  runGuardsAndResolvers: 'always',
2817
2145
  children: [
2818
2146
  {
@@ -2855,14 +2183,6 @@ const rootRoute = {
2855
2183
  canActivate: [ContextGuard],
2856
2184
  data: { showHeader: true },
2857
2185
  },
2858
- {
2859
- path: 'empty',
2860
- component: EmptyAccountComponent,
2861
- runGuardsAndResolvers: 'paramsOrQueryParamsChange',
2862
- resolve: { quote: QuoteResolver },
2863
- canActivate: [ContextGuard],
2864
- data: { showHeader: true },
2865
- },
2866
2186
  {
2867
2187
  path: 'remote',
2868
2188
  component: RemoteComponent,
@@ -2885,19 +2205,17 @@ const rootRoute = {
2885
2205
  class FlowRoutingModule {
2886
2206
  }
2887
2207
  FlowRoutingModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRoutingModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
2888
- FlowRoutingModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: FlowRoutingModule, imports: [i1$1.RouterModule, ProductModule,
2208
+ FlowRoutingModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.9", ngImport: i0, type: FlowRoutingModule, imports: [i1$2.RouterModule, ProductModule,
2889
2209
  ShoppingCartModule,
2890
2210
  CatalogModule,
2891
2211
  AssetsModule,
2892
- RemoteModule,
2893
- EmptyAccountModule], exports: [RouterModule] });
2212
+ RemoteModule], exports: [RouterModule] });
2894
2213
  FlowRoutingModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRoutingModule, providers: [FlowRouterService, RootGuard, ContextGuard, ProductUnloadGuard, FlowResolver, QuoteResolver], imports: [RouterModule.forChild([rootRoute]),
2895
2214
  ProductModule,
2896
2215
  ShoppingCartModule,
2897
2216
  CatalogModule,
2898
2217
  AssetsModule,
2899
- RemoteModule,
2900
- EmptyAccountModule, RouterModule] });
2218
+ RemoteModule, RouterModule] });
2901
2219
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowRoutingModule, decorators: [{
2902
2220
  type: NgModule,
2903
2221
  args: [{
@@ -2908,18 +2226,17 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2908
2226
  CatalogModule,
2909
2227
  AssetsModule,
2910
2228
  RemoteModule,
2911
- EmptyAccountModule,
2912
2229
  ],
2913
2230
  exports: [RouterModule],
2914
2231
  providers: [FlowRouterService, RootGuard, ContextGuard, ProductUnloadGuard, FlowResolver, QuoteResolver],
2915
2232
  }]
2916
2233
  }] });
2917
2234
 
2918
- class FlowGuidedSellingService {
2235
+ class FlowDocGenService {
2919
2236
  constructor(integrationState) {
2920
2237
  this.integrationState = integrationState;
2921
2238
  this.cleanup$ = new Subject();
2922
- this.isVisibleSubj$ = new BehaviorSubject(true);
2239
+ this.isVisibleSubj$ = new BehaviorSubject(false);
2923
2240
  this.isVisible$ = this.isVisibleSubj$.asObservable();
2924
2241
  this.initSubscriptions();
2925
2242
  }
@@ -2928,21 +2245,54 @@ class FlowGuidedSellingService {
2928
2245
  }
2929
2246
  initSubscriptions() {
2930
2247
  this.integrationState
2931
- .listen$(FlowAction.OPEN_GUIDED_SELLING)
2248
+ .listen$(FlowAction.FLOW_OPEN_DOC_GEN)
2932
2249
  .pipe(tap(() => this.isVisibleSubj$.next(true)), takeUntil(this.cleanup$))
2933
2250
  .subscribe();
2934
2251
  this.integrationState
2935
- .listen$(FlowAction.CLOSE_GUIDED_SELLING)
2252
+ .listen$(FlowAction.FLOW_CLOSE_DOC_GEN)
2936
2253
  .pipe(tap(() => this.isVisibleSubj$.next(false)), takeUntil(this.cleanup$))
2937
2254
  .subscribe();
2938
2255
  }
2939
2256
  }
2940
- FlowGuidedSellingService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService, deps: [{ token: i5.IntegrationState }], target: i0.ɵɵFactoryTarget.Injectable });
2941
- FlowGuidedSellingService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService });
2942
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowGuidedSellingService, decorators: [{
2257
+ FlowDocGenService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService, deps: [{ token: i2.IntegrationState }], target: i0.ɵɵFactoryTarget.Injectable });
2258
+ FlowDocGenService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService });
2259
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowDocGenService, decorators: [{
2943
2260
  type: Injectable
2944
- }], ctorParameters: function () { return [{ type: i5.IntegrationState }]; } });
2261
+ }], ctorParameters: function () { return [{ type: i2.IntegrationState }]; } });
2262
+
2263
+ const configurePrimengShadowDOM = () => {
2264
+ DomHandler.getScrollableParents = (element) => {
2265
+ const scrollableParents = [];
2266
+ if (element) {
2267
+ const parents = DomHandler.getParents(element).filter((item) => !(item instanceof ShadowRoot));
2268
+ const overflowRegex = /(auto|scroll)/;
2269
+ const overflowCheck = (node) => {
2270
+ const styleDeclaration = window['getComputedStyle'](node, null);
2271
+ return (overflowRegex.test(styleDeclaration.getPropertyValue('overflow')) ||
2272
+ overflowRegex.test(styleDeclaration.getPropertyValue('overflowX')) ||
2273
+ overflowRegex.test(styleDeclaration.getPropertyValue('overflowY')));
2274
+ };
2275
+ for (const parent of parents) {
2276
+ const scrollSelectors = parent.nodeType === 1 && parent.dataset['scrollselectors'];
2277
+ if (scrollSelectors) {
2278
+ const selectors = scrollSelectors.split(',');
2279
+ for (const selector of selectors) {
2280
+ const el = DomHandler.findSingle(parent, selector);
2281
+ if (el && overflowCheck(el)) {
2282
+ scrollableParents.push(el);
2283
+ }
2284
+ }
2285
+ }
2286
+ if (parent.nodeType !== 9 && overflowCheck(parent)) {
2287
+ scrollableParents.push(parent);
2288
+ }
2289
+ }
2290
+ }
2291
+ return scrollableParents;
2292
+ };
2293
+ };
2945
2294
 
2295
+ configurePrimengShadowDOM();
2946
2296
  class FlowModule {
2947
2297
  }
2948
2298
  FlowModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.9", ngImport: i0, type: FlowModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
@@ -2952,7 +2302,7 @@ FlowModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15
2952
2302
  ApiModule,
2953
2303
  LauncherModule,
2954
2304
  LoaderModule,
2955
- FlowHeaderModule,
2305
+ FlowNewHeaderModule,
2956
2306
  FlowDialogModule,
2957
2307
  SdkCoreModule,
2958
2308
  DocGenModule,
@@ -2963,7 +2313,7 @@ FlowModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15
2963
2313
  ApiModule,
2964
2314
  LauncherModule,
2965
2315
  LoaderModule,
2966
- FlowHeaderModule,
2316
+ FlowNewHeaderModule,
2967
2317
  FlowDialogModule,
2968
2318
  SdkCoreModule,
2969
2319
  DocGenModule,
@@ -2979,7 +2329,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2979
2329
  ApiModule,
2980
2330
  LauncherModule,
2981
2331
  LoaderModule,
2982
- FlowHeaderModule,
2332
+ FlowNewHeaderModule,
2983
2333
  FlowDialogModule,
2984
2334
  SdkCoreModule,
2985
2335
  DocGenModule,
@@ -2993,5 +2343,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.9", ngImpor
2993
2343
  * Generated bundle index. Do not edit.
2994
2344
  */
2995
2345
 
2996
- export { ContextGuard, FLOW_CUSTOMIZATION, FlowModule, FlowService, VELOCE_FLOW_ROOT_ROUTE, getDefaultProperties, getFlowObjectIdPropertyName };
2346
+ export { ContextGuard, FlowDialogService, FlowModule, FlowRouterService, FlowService, VELOCE_FLOW_ROOT_ROUTE, getDefaultProperties, getFlowObjectIdPropertyName };
2997
2347
  //# sourceMappingURL=veloceapps-sdk.mjs.map