@veloceapps/sdk 4.0.5 → 4.0.6

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -10,7 +10,9 @@ export declare class QuoteDraftService {
10
10
  private priceApiService;
11
11
  private quoteSubj$;
12
12
  private resetSubj$;
13
- isInitialized: boolean;
13
+ private isInitialized$;
14
+ get isInitialized(): boolean;
15
+ set isInitialized(value: boolean);
14
16
  allPriceLists: PriceList[];
15
17
  assetPriceLists: PriceList[];
16
18
  hasUnsavedChanges: boolean;
@@ -1,7 +1,7 @@
1
1
  import { Injectable } from '@angular/core';
2
2
  import { ConfigurationContextMode } from '@veloce/core';
3
3
  import { BehaviorSubject, combineLatest, noop, zip } from 'rxjs';
4
- import { filter, map, shareReplay, take, tap } from 'rxjs/operators';
4
+ import { filter, map, shareReplay, skip, switchMap, take, tap } from 'rxjs/operators';
5
5
  import * as i0 from "@angular/core";
6
6
  import * as i1 from "./context.service";
7
7
  import * as i2 from "@veloce/api";
@@ -12,12 +12,23 @@ export class QuoteDraftService {
12
12
  this.priceApiService = priceApiService;
13
13
  this.quoteSubj$ = new BehaviorSubject(null);
14
14
  this.resetSubj$ = new BehaviorSubject(true);
15
- this.isInitialized = false;
15
+ this.isInitialized$ = new BehaviorSubject(false);
16
16
  this.allPriceLists = [];
17
17
  this.assetPriceLists = [];
18
18
  this.hasUnsavedChanges = false;
19
19
  this.reset$ = this.resetSubj$.asObservable();
20
20
  this.activePriceList$ = this.context.resolve$().pipe(map(ctx => this.allPriceLists.find(priceList => priceList.id === ctx.properties.PriceListId)), map(priceList => priceList !== null && priceList !== void 0 ? priceList : null));
21
+ this.isInitialized$
22
+ .pipe(filter(isInitialized => isInitialized), switchMap(() => this.quoteSubj$.asObservable()), skip(1), tap(quote => this.markAsUpdated(quote)))
23
+ .subscribe();
24
+ }
25
+ get isInitialized() {
26
+ return this.isInitialized$.getValue();
27
+ }
28
+ set isInitialized(value) {
29
+ if (this.isInitialized !== value) {
30
+ this.isInitialized$.next(value);
31
+ }
21
32
  }
22
33
  reset() {
23
34
  this.resetSubj$.next(true);
@@ -37,7 +48,6 @@ export class QuoteDraftService {
37
48
  return;
38
49
  }
39
50
  this.quoteSubj$.next(Object.assign(Object.assign({}, quoteDraft), { currentState: lineItems }));
40
- this.markAsUpdated();
41
51
  }
42
52
  updateQuoteDraft(update) {
43
53
  const quoteDraft = this.quoteSubj$.value;
@@ -48,7 +58,6 @@ export class QuoteDraftService {
48
58
  this.context.update(update.context);
49
59
  }
50
60
  this.quoteSubj$.next(Object.assign(Object.assign({}, quoteDraft), update));
51
- this.markAsUpdated();
52
61
  }
53
62
  updateByPriceSummary(priceSummary) {
54
63
  const quoteDraft = this.quoteSubj$.value;
@@ -60,7 +69,6 @@ export class QuoteDraftService {
60
69
  return updated !== null && updated !== void 0 ? updated : lineItem;
61
70
  });
62
71
  this.quoteSubj$.next(Object.assign(Object.assign({}, quoteDraft), { currentState: updatedCurrentState, totalPrices: priceSummary.totalPrices, approvalItems: priceSummary.approvalItems }));
63
- this.markAsUpdated();
64
72
  }
65
73
  get quoteDraft$() {
66
74
  return combineLatest([this.quoteSubj$, this.context.resolve$()]).pipe(map(() => this.quoteDraft), filter((quote) => Boolean(quote)), shareReplay());
@@ -176,8 +184,11 @@ export class QuoteDraftService {
176
184
  const ctx = this.context.resolve();
177
185
  return lineItems.filter(li => !li.priceListId || li.priceListId === ctx.properties.PriceListId);
178
186
  }
179
- markAsUpdated() {
180
- if (this.isInitialized && !this.hasUnsavedChanges) {
187
+ markAsUpdated(quote) {
188
+ if ((quote === null || quote === void 0 ? void 0 : quote.context.properties.mode) === ConfigurationContextMode.ACCOUNT) {
189
+ this.hasUnsavedChanges = !!quote && !quote.currentState.every(li => li.actionCode === 'EXIST');
190
+ }
191
+ else {
181
192
  this.hasUnsavedChanges = true;
182
193
  }
183
194
  }
@@ -188,4 +199,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
188
199
  type: Injectable,
189
200
  args: [{ providedIn: 'root' }]
190
201
  }], ctorParameters: function () { return [{ type: i1.ContextService }, { type: i2.QuoteApiService }, { type: i2.PriceApiService }]; } });
191
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quote-draft.service.js","sourceRoot":"","sources":["../../../../../../libs/sdk/core/services/quote-draft.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { PriceApiService, QuoteApiService } from '@veloce/api';\nimport { ConfigurationContextMode, LineItem, PriceList, PriceSummary, QuoteDraft } from '@veloce/core';\nimport { Dictionary } from 'lodash';\nimport { BehaviorSubject, combineLatest, noop, Observable, zip } from 'rxjs';\nimport { filter, map, shareReplay, take, tap } from 'rxjs/operators';\nimport { ContextService } from './context.service';\n\n@Injectable({ providedIn: 'root' })\nexport class QuoteDraftService {\n  private quoteSubj$ = new BehaviorSubject<QuoteDraft | null>(null);\n  private resetSubj$ = new BehaviorSubject(true);\n\n  public isInitialized = false;\n  public allPriceLists: PriceList[] = [];\n  public assetPriceLists: PriceList[] = [];\n  public hasUnsavedChanges = false;\n  public reset$ = this.resetSubj$.asObservable();\n  public activePriceList$: Observable<PriceList | null>;\n\n  constructor(\n    private context: ContextService,\n    private quoteApiService: QuoteApiService,\n    private priceApiService: PriceApiService,\n  ) {\n    this.activePriceList$ = this.context.resolve$().pipe(\n      map(ctx => this.allPriceLists.find(priceList => priceList.id === ctx.properties.PriceListId)),\n      map(priceList => priceList ?? null),\n    );\n  }\n\n  public reset() {\n    this.resetSubj$.next(true);\n    this.quoteSubj$.next(null);\n  }\n\n  public init(quoteId: string, params: Dictionary<string>): Observable<void> {\n    return zip(\n      this.context.resolve$(),\n      this.quoteApiService.getQuoteDraft(quoteId, params),\n      this.priceApiService.getPriceLists(),\n    ).pipe(\n      tap(([context, quote, allPriceLists]) => {\n        this.allPriceLists = allPriceLists;\n        this.quoteSubj$.next(quote);\n\n        this.context.update({\n          ...context,\n          ...quote.context,\n          properties: {\n            ...context.properties,\n            ...quote.context.properties,\n          },\n        });\n\n        this.populateActivePriceLists$();\n      }),\n      map(() => noop()),\n      take(1),\n    );\n  }\n\n  public setCurrentLineItemState(lineItems: LineItem[]): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      currentState: lineItems,\n    });\n\n    this.markAsUpdated();\n  }\n\n  public updateQuoteDraft(update: Partial<QuoteDraft>): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    if (update.context) {\n      this.context.update(update.context);\n    }\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      ...update,\n    });\n\n    this.markAsUpdated();\n  }\n\n  public updateByPriceSummary(priceSummary: PriceSummary): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    const updatedCurrentState = this.currentState.map(lineItem => {\n      const updated = priceSummary.lineItems.find(li => li.id === lineItem.id);\n      return updated ?? lineItem;\n    });\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      currentState: updatedCurrentState,\n      totalPrices: priceSummary.totalPrices,\n      approvalItems: priceSummary.approvalItems,\n    });\n\n    this.markAsUpdated();\n  }\n\n  get quoteDraft$(): Observable<QuoteDraft> {\n    return combineLatest([this.quoteSubj$, this.context.resolve$()]).pipe(\n      map(() => this.quoteDraft),\n      filter((quote): quote is QuoteDraft => Boolean(quote)),\n      shareReplay(),\n    );\n  }\n\n  get quoteDraft(): QuoteDraft | null {\n    const quote = this.quoteSubj$.value;\n\n    if (!quote) {\n      return null;\n    }\n\n    return {\n      ...quote,\n      context: this.context.resolve(),\n    };\n  }\n\n  get quoteDraftForActivePriceList(): QuoteDraft | null {\n    const quoteDraft = this.quoteDraft;\n\n    if (!quoteDraft) {\n      return null;\n    }\n\n    return {\n      ...quoteDraft,\n      initialState: this.filterByActivePriceList(quoteDraft.initialState),\n      currentState: this.filterByActivePriceList(quoteDraft.currentState),\n    };\n  }\n\n  get currentState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(quote => quote.currentState));\n  }\n\n  get currentState(): LineItem[] {\n    return this.quoteDraft?.currentState ?? [];\n  }\n\n  /**\n   * Stream of activeCurrentState\n   */\n  get activeCurrentState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(() => this.activeCurrentState));\n  }\n\n  /**\n   * activeCurrentState is currentState passed through additional filters\n   */\n  get activeCurrentState(): LineItem[] {\n    const ctx = this.context.resolve();\n    let currentState = this.quoteDraft?.currentState ?? [];\n\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      currentState = this.filterByActivePriceList(currentState);\n    }\n\n    return currentState;\n  }\n\n  /**\n   * Stream of activeInitialState\n   */\n  get activeInitialState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(() => this.activeInitialState));\n  }\n\n  /**\n   * activeInitialState is initialState passed through additional filters\n   */\n  get activeInitialState(): LineItem[] {\n    const ctx = this.context.resolve();\n    let initialState = this.quoteDraft?.initialState ?? [];\n\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      initialState = this.filterByActivePriceList(initialState);\n    }\n\n    return initialState;\n  }\n\n  get isStandalone(): boolean {\n    return this.context.resolve().properties.standalone === 'true';\n  }\n\n  get isStandalone$(): Observable<boolean> {\n    return this.context.resolve$().pipe(map(() => this.isStandalone));\n  }\n\n  public isEditMode$(): Observable<boolean> {\n    return this.context.resolve$().pipe(map(() => this.isEditMode()));\n  }\n\n  public isEditMode(): boolean {\n    const context = this.context.resolve();\n\n    if (context.mode === ConfigurationContextMode.ACCOUNT) {\n      return true;\n    }\n\n    if (context.mode === ConfigurationContextMode.QUOTE) {\n      return context.properties.Status === 'Draft';\n    }\n\n    return false;\n  }\n\n  public updateActivePriceList(priceListId: string): void {\n    this.context.update({ properties: { PriceListId: priceListId } });\n  }\n\n  private populateActivePriceLists$(): void {\n    const ctx = this.context.resolve();\n    const quoteDraft = this.quoteDraft;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    // In ACCOUNT mode populate price lists from related assets\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      // Populate list of price lists\n      this.assetPriceLists = quoteDraft.currentState\n        .map(({ priceListId }) => priceListId)\n        .reduce<PriceList[]>((trunk, priceListId) => {\n          if (!priceListId || trunk.some(item => item.id === priceListId)) {\n            return trunk;\n          }\n\n          return [\n            ...trunk,\n            { id: priceListId, name: this.allPriceLists.find(item => item.id === priceListId)?.name ?? '' },\n          ];\n        }, []);\n\n      const activePriceList = this.assetPriceLists[0];\n\n      this.updateActivePriceList(activePriceList?.id);\n\n      // Update PriceListId in context properties\n      // TODO: remove it when backend will have such logic\n      this.context.update({ properties: { PriceListId: activePriceList?.id } });\n    }\n  }\n\n  private filterByActivePriceList(lineItems: LineItem[]): LineItem[] {\n    const ctx = this.context.resolve();\n\n    return lineItems.filter(li => !li.priceListId || li.priceListId === ctx.properties.PriceListId);\n  }\n\n  private markAsUpdated(): void {\n    if (this.isInitialized && !this.hasUnsavedChanges) {\n      this.hasUnsavedChanges = true;\n    }\n  }\n}\n"]}
202
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"quote-draft.service.js","sourceRoot":"","sources":["../../../../../../libs/sdk/core/services/quote-draft.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { PriceApiService, QuoteApiService } from '@veloce/api';\nimport { ConfigurationContextMode, LineItem, PriceList, PriceSummary, QuoteDraft } from '@veloce/core';\nimport { Dictionary } from 'lodash';\nimport { BehaviorSubject, combineLatest, noop, Observable, zip } from 'rxjs';\nimport { filter, map, shareReplay, skip, switchMap, take, tap } from 'rxjs/operators';\nimport { ContextService } from './context.service';\n\n@Injectable({ providedIn: 'root' })\nexport class QuoteDraftService {\n  private quoteSubj$ = new BehaviorSubject<QuoteDraft | null>(null);\n  private resetSubj$ = new BehaviorSubject<boolean>(true);\n  private isInitialized$ = new BehaviorSubject<boolean>(false);\n  public get isInitialized(): boolean {\n    return this.isInitialized$.getValue();\n  }\n  public set isInitialized(value: boolean) {\n    if (this.isInitialized !== value) {\n      this.isInitialized$.next(value);\n    }\n  }\n\n  public allPriceLists: PriceList[] = [];\n  public assetPriceLists: PriceList[] = [];\n  public hasUnsavedChanges = false;\n  public reset$ = this.resetSubj$.asObservable();\n  public activePriceList$: Observable<PriceList | null>;\n\n  constructor(\n    private context: ContextService,\n    private quoteApiService: QuoteApiService,\n    private priceApiService: PriceApiService,\n  ) {\n    this.activePriceList$ = this.context.resolve$().pipe(\n      map(ctx => this.allPriceLists.find(priceList => priceList.id === ctx.properties.PriceListId)),\n      map(priceList => priceList ?? null),\n    );\n\n    this.isInitialized$\n      .pipe(\n        filter(isInitialized => isInitialized),\n        switchMap(() => this.quoteSubj$.asObservable()),\n        skip(1),\n        tap(quote => this.markAsUpdated(quote)),\n      )\n      .subscribe();\n  }\n\n  public reset() {\n    this.resetSubj$.next(true);\n    this.quoteSubj$.next(null);\n  }\n\n  public init(quoteId: string, params: Dictionary<string>): Observable<void> {\n    return zip(\n      this.context.resolve$(),\n      this.quoteApiService.getQuoteDraft(quoteId, params),\n      this.priceApiService.getPriceLists(),\n    ).pipe(\n      tap(([context, quote, allPriceLists]) => {\n        this.allPriceLists = allPriceLists;\n        this.quoteSubj$.next(quote);\n\n        this.context.update({\n          ...context,\n          ...quote.context,\n          properties: {\n            ...context.properties,\n            ...quote.context.properties,\n          },\n        });\n\n        this.populateActivePriceLists$();\n      }),\n      map(() => noop()),\n      take(1),\n    );\n  }\n\n  public setCurrentLineItemState(lineItems: LineItem[]): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      currentState: lineItems,\n    });\n  }\n\n  public updateQuoteDraft(update: Partial<QuoteDraft>): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    if (update.context) {\n      this.context.update(update.context);\n    }\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      ...update,\n    });\n  }\n\n  public updateByPriceSummary(priceSummary: PriceSummary): void {\n    const quoteDraft = this.quoteSubj$.value;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    const updatedCurrentState = this.currentState.map(lineItem => {\n      const updated = priceSummary.lineItems.find(li => li.id === lineItem.id);\n      return updated ?? lineItem;\n    });\n\n    this.quoteSubj$.next({\n      ...quoteDraft,\n      currentState: updatedCurrentState,\n      totalPrices: priceSummary.totalPrices,\n      approvalItems: priceSummary.approvalItems,\n    });\n  }\n\n  get quoteDraft$(): Observable<QuoteDraft> {\n    return combineLatest([this.quoteSubj$, this.context.resolve$()]).pipe(\n      map(() => this.quoteDraft),\n      filter((quote): quote is QuoteDraft => Boolean(quote)),\n      shareReplay(),\n    );\n  }\n\n  get quoteDraft(): QuoteDraft | null {\n    const quote = this.quoteSubj$.value;\n\n    if (!quote) {\n      return null;\n    }\n\n    return {\n      ...quote,\n      context: this.context.resolve(),\n    };\n  }\n\n  get quoteDraftForActivePriceList(): QuoteDraft | null {\n    const quoteDraft = this.quoteDraft;\n\n    if (!quoteDraft) {\n      return null;\n    }\n\n    return {\n      ...quoteDraft,\n      initialState: this.filterByActivePriceList(quoteDraft.initialState),\n      currentState: this.filterByActivePriceList(quoteDraft.currentState),\n    };\n  }\n\n  get currentState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(quote => quote.currentState));\n  }\n\n  get currentState(): LineItem[] {\n    return this.quoteDraft?.currentState ?? [];\n  }\n\n  /**\n   * Stream of activeCurrentState\n   */\n  get activeCurrentState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(() => this.activeCurrentState));\n  }\n\n  /**\n   * activeCurrentState is currentState passed through additional filters\n   */\n  get activeCurrentState(): LineItem[] {\n    const ctx = this.context.resolve();\n    let currentState = this.quoteDraft?.currentState ?? [];\n\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      currentState = this.filterByActivePriceList(currentState);\n    }\n\n    return currentState;\n  }\n\n  /**\n   * Stream of activeInitialState\n   */\n  get activeInitialState$(): Observable<LineItem[]> {\n    return this.quoteDraft$.pipe(map(() => this.activeInitialState));\n  }\n\n  /**\n   * activeInitialState is initialState passed through additional filters\n   */\n  get activeInitialState(): LineItem[] {\n    const ctx = this.context.resolve();\n    let initialState = this.quoteDraft?.initialState ?? [];\n\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      initialState = this.filterByActivePriceList(initialState);\n    }\n\n    return initialState;\n  }\n\n  get isStandalone(): boolean {\n    return this.context.resolve().properties.standalone === 'true';\n  }\n\n  get isStandalone$(): Observable<boolean> {\n    return this.context.resolve$().pipe(map(() => this.isStandalone));\n  }\n\n  public isEditMode$(): Observable<boolean> {\n    return this.context.resolve$().pipe(map(() => this.isEditMode()));\n  }\n\n  public isEditMode(): boolean {\n    const context = this.context.resolve();\n\n    if (context.mode === ConfigurationContextMode.ACCOUNT) {\n      return true;\n    }\n\n    if (context.mode === ConfigurationContextMode.QUOTE) {\n      return context.properties.Status === 'Draft';\n    }\n\n    return false;\n  }\n\n  public updateActivePriceList(priceListId: string): void {\n    this.context.update({ properties: { PriceListId: priceListId } });\n  }\n\n  private populateActivePriceLists$(): void {\n    const ctx = this.context.resolve();\n    const quoteDraft = this.quoteDraft;\n\n    if (!quoteDraft) {\n      return;\n    }\n\n    // In ACCOUNT mode populate price lists from related assets\n    if (ctx.mode === ConfigurationContextMode.ACCOUNT) {\n      // Populate list of price lists\n      this.assetPriceLists = quoteDraft.currentState\n        .map(({ priceListId }) => priceListId)\n        .reduce<PriceList[]>((trunk, priceListId) => {\n          if (!priceListId || trunk.some(item => item.id === priceListId)) {\n            return trunk;\n          }\n\n          return [\n            ...trunk,\n            { id: priceListId, name: this.allPriceLists.find(item => item.id === priceListId)?.name ?? '' },\n          ];\n        }, []);\n\n      const activePriceList = this.assetPriceLists[0];\n\n      this.updateActivePriceList(activePriceList?.id);\n\n      // Update PriceListId in context properties\n      // TODO: remove it when backend will have such logic\n      this.context.update({ properties: { PriceListId: activePriceList?.id } });\n    }\n  }\n\n  private filterByActivePriceList(lineItems: LineItem[]): LineItem[] {\n    const ctx = this.context.resolve();\n\n    return lineItems.filter(li => !li.priceListId || li.priceListId === ctx.properties.PriceListId);\n  }\n\n  private markAsUpdated(quote: QuoteDraft | null): void {\n    if (quote?.context.properties.mode === ConfigurationContextMode.ACCOUNT) {\n      this.hasUnsavedChanges = !!quote && !quote.currentState.every(li => li.actionCode === 'EXIST');\n    } else {\n      this.hasUnsavedChanges = true;\n    }\n  }\n}\n"]}
@@ -103,7 +103,7 @@ export class CartPreviewComponent {
103
103
  }
104
104
  }
105
105
  CartPreviewComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: CartPreviewComponent, deps: [{ token: i1.FlowConfigurationService }, { token: i2.FlowRouterService }, { token: i1.ProductImagesService }, { token: i1.QuoteDraftService }], target: i0.ɵɵFactoryTarget.Component });
106
- CartPreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", 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=\"navigation-settings-overlay flow-header-overlay center\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\" *vlLet=\"readonlyProductId$ | async as readonlyProductId\">\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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <div class=\"product header\">\n <span>Product</span>\n <span class=\"text-right\">Qty</span>\n <span class=\"text-right\">MRR</span>\n <span class=\"text-right\">NRR</span>\n </div>\n\n <div class=\"scrollable-wrapper\">\n <div class=\"product\" *ngFor=\"let product of products\">\n <div\n class=\"product__info\"\n *vlLet=\"!isEditMode || product.deleted || readonlyProductId === product.productId as readonlyProduct\"\n >\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 }}</div>\n <div>\n <p-button\n label=\"Configure\"\n [disabled]=\"readonlyProduct || !product.configurable || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm pl-0\"\n (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n ></p-button>\n <p-button\n label=\"Delete\"\n [disabled]=\"readonlyProduct || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n (onClick)=\"deleteHandler(product)\"\n ></p-button>\n </div>\n </div>\n </div>\n\n <span>\n <p-inputNumber\n *ngIf=\"form.controls[product.id] as control\"\n class=\"qty-control\"\n [formControl]=\"$any(control)\"\n [min]=\"1\"\n [required]=\"true\"\n (onBlur)=\"controlBlurHandler(product)\"\n ></p-inputNumber>\n </span>\n <span class=\"text-right pt-3\">${{ product.mrr }}</span>\n <span class=\"text-right pt-3\">${{ product.nrr }}</span>\n </div>\n\n <ng-template #noImage>\n <i class=\"vl-icon vl-icon-no-image no-image-icon\"></i>\n </ng-template>\n </div>\n\n <div class=\"flex justify-content-end mt-3\">\n <p-button\n label=\"Clear cart\"\n styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n [disabled]=\"!isEditMode || !!readonlyProductId || 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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <span class=\"empty-state\">There are no products added to the Shopping Cart yet.</span>\n </ng-template>\n </div>\n </ng-template>\n</p-overlayPanel>\n", styles: [".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 16px}.empty-state{color:var(--vl-text-color-deep-accent)}.scrollable-wrapper{overflow:auto}.product{display:grid;grid-template-columns:auto 60px 80px 80px;padding:16px 0}.product:not(.header){border-bottom:1px solid var(--vl-border-color)}.product.header{color:var(--vl-text-color-deep-accent);padding-bottom:0}.product__info{display:flex;grid-gap:16px;gap:16px}.product__image-wrapper{flex-shrink:0;height:64px;width:64px;display:flex;justify-content:center;align-items:center;background:var(--vl-primary-nav-overlay-bg);border-radius:4px}.product__image{background-size:contain;background-repeat:no-repeat;background-position:center;height:calc(100% - 12px);width:calc(100% - 12px)}.product .qty-control ::ng-deep .p-inputnumber-input{align-self:flex-start;text-align:right;width:100%}.word-break{word-break:break-word}.no-image-icon{color:#b4d1ef;height:18px;width:18px}:host ::ng-deep .p-overlaypanel.navigation-settings-overlay{margin-top:20px}\n"], components: [{ type: i3.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { type: i4.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i5.InputNumber, selector: "p-inputNumber", inputs: ["showButtons", "format", "buttonLayout", "inputId", "styleClass", "style", "placeholder", "size", "maxlength", "tabindex", "title", "ariaLabel", "ariaRequired", "name", "required", "autocomplete", "min", "max", "incrementButtonClass", "decrementButtonClass", "incrementButtonIcon", "decrementButtonIcon", "readonly", "step", "allowEmpty", "locale", "localeMatcher", "mode", "currency", "currencyDisplay", "useGrouping", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "inputStyle", "inputStyleClass", "disabled"], outputs: ["onInput", "onFocus", "onBlur", "onKeyDown"] }], directives: [{ type: i6.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i7.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i8.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i9.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i9.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }], pipes: { "async": i8.AsyncPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
106
+ CartPreviewComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", 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=\"navigation-settings-overlay flow-header-overlay center\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\" *vlLet=\"readonlyProductId$ | async as readonlyProductId\">\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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <div class=\"product header\">\n <span>Product</span>\n <span class=\"text-right\">Qty</span>\n <span class=\"text-right\">MRR</span>\n <span class=\"text-right\">NRR</span>\n </div>\n\n <div class=\"scrollable-wrapper\">\n <div class=\"product\" *ngFor=\"let product of products\">\n <div\n class=\"product__info\"\n *vlLet=\"!isEditMode || product.deleted || readonlyProductId === product.productId as readonlyProduct\"\n >\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 }}</div>\n <div>\n <p-button\n label=\"Configure\"\n [disabled]=\"readonlyProduct || !product.configurable || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm pl-0\"\n (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n ></p-button>\n <p-button\n label=\"Delete\"\n [disabled]=\"readonlyProduct || product.hasTerm\"\n styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n (onClick)=\"deleteHandler(product)\"\n ></p-button>\n </div>\n </div>\n </div>\n\n <span>\n <p-inputNumber\n *ngIf=\"form.controls[product.id] as control\"\n class=\"qty-control\"\n [formControl]=\"$any(control)\"\n [min]=\"1\"\n [required]=\"true\"\n (onBlur)=\"controlBlurHandler(product)\"\n ></p-inputNumber>\n </span>\n <span class=\"text-right pt-3\">${{ product.mrr }}</span>\n <span class=\"text-right pt-3\">${{ product.nrr }}</span>\n </div>\n\n <ng-template #noImage>\n <i class=\"vl-icon vl-icon-no-image no-image-icon\"></i>\n </ng-template>\n </div>\n\n <div class=\"flex justify-content-end mt-3 footer\">\n <p-button\n label=\"Clear cart\"\n styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n [disabled]=\"!isEditMode || !!readonlyProductId || 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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n </h2>\n\n <span class=\"empty-state\">There are no products added to the Shopping Cart yet.</span>\n </ng-template>\n </div>\n </ng-template>\n</p-overlayPanel>\n", styles: [".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;padding:16px}.empty-state{color:var(--vl-text-color-deep-accent);padding:0 16px 16px}.scrollable-wrapper{overflow:auto}.product{display:grid;grid-template-columns:auto 60px 80px 80px;padding:16px}.product:not(.header){border-bottom:1px solid var(--vl-border-color)}.product.header{color:var(--vl-text-color-deep-accent);padding-bottom:0}.product__info{display:flex;grid-gap:16px;gap:16px}.product__image-wrapper{flex-shrink:0;height:64px;width:64px;display:flex;justify-content:center;align-items:center;background:var(--vl-primary-nav-overlay-bg);border-radius:4px}.product__image{background-size:contain;background-repeat:no-repeat;background-position:center;height:calc(100% - 12px);width:calc(100% - 12px)}.product .qty-control ::ng-deep .p-inputnumber-input{align-self:flex-start;text-align:right;width:100%}.footer{padding:0 16px 16px}.word-break{word-break:break-word}.no-image-icon{color:#b4d1ef;height:18px;width:18px}:host ::ng-deep .p-overlaypanel.navigation-settings-overlay{margin-top:20px}\n"], components: [{ type: i3.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { type: i4.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i5.InputNumber, selector: "p-inputNumber", inputs: ["showButtons", "format", "buttonLayout", "inputId", "styleClass", "style", "placeholder", "size", "maxlength", "tabindex", "title", "ariaLabel", "ariaRequired", "name", "required", "autocomplete", "min", "max", "incrementButtonClass", "decrementButtonClass", "incrementButtonIcon", "decrementButtonIcon", "readonly", "step", "allowEmpty", "locale", "localeMatcher", "mode", "currency", "currencyDisplay", "useGrouping", "minFractionDigits", "maxFractionDigits", "prefix", "suffix", "inputStyle", "inputStyleClass", "disabled"], outputs: ["onInput", "onFocus", "onBlur", "onKeyDown"] }], directives: [{ type: i6.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i7.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i8.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i9.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i9.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i9.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }], pipes: { "async": i8.AsyncPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
107
107
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: CartPreviewComponent, decorators: [{
108
108
  type: Component,
109
109
  args: [{
@@ -118,4 +118,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
118
118
  }], products: [{
119
119
  type: Input
120
120
  }] } });
121
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-preview.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-preview.component.ts","../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-preview.component.html"],"names":[],"mappings":"AAAA,OAAO,EAEL,uBAAuB,EACvB,SAAS,EACT,KAAK,EAIL,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,WAAW,EAAE,SAAS,EAAE,MAAM,gBAAgB,CAAC;AAExD,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,aAAa,EAAE,MAAM,EAAE,GAAG,EAAc,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;;;;;;;;;;;AAUnG,MAAM,OAAO,oBAAoB;IAc/B,YACU,iBAA2C,EAC3C,aAAgC,EAChC,oBAA0C,EAC1C,iBAAoC;QAHpC,sBAAiB,GAAjB,iBAAiB,CAA0B;QAC3C,kBAAa,GAAb,aAAa,CAAmB;QAChC,yBAAoB,GAApB,oBAAoB,CAAsB;QAC1C,sBAAiB,GAAjB,iBAAiB,CAAmB;QAbvC,SAAI,GAAG,IAAI,SAAS,CAAC,EAAE,CAAC,CAAC;QAGxB,kBAAa,GAAG,IAAI,eAAe,CAAC,IAAI,CAAC,CAAC;QAC1C,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;QAGlC,sBAAiB,GAAG,KAAK,CAAC;QAQ/B,IAAI,CAAC,kBAAkB,GAAG,aAAa,CAAC;YACtC,IAAI,CAAC,aAAa,CAAC,MAAM;YACzB,IAAI,CAAC,aAAa,CAAC,qBAAqB,EAAE;SAC3C,CAAC,CAAC,IAAI,CACL,GAAG,CAAC,CAAC,CAAC,KAAK,EAAE,oBAAoB,CAAC,EAAE,EAAE;YACpC,IAAI,CAAC,oBAAoB,EAAE;gBACzB,OAAO,IAAI,CAAC;aACb;YAED,OAAO,KAAK,CAAC,WAAW,CAAC,SAAS,CAAC;QACrC,CAAC,CAAC,CACH,CAAC;QAEF,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;IAC1D,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,MAAM,EAAE,QAAQ,EAAE,GAAG,OAAO,CAAC;QAE7B,IAAI,QAAQ,EAAE;YACZ,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SAC/B;IACH,CAAC;IAED,eAAe;QACb,aAAa,CAAC,CAAC,IAAI,CAAC,kBAAkB,EAAE,IAAI,CAAC,WAAW,EAAE,IAAI,CAAC,aAAa,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;aACrG,IAAI,CACH,MAAM,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,cAAc,CAAC,EAC9C,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,CAAC,CAAC,CAAC,iBAAiB,EAAE,UAAU,CAAC,EAAE,EAAE,CAC7C,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,QAAQ,EAAE,UAAU,EAAE,iBAAiB,CAAC,CAClE,CAAC;IACN,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAEM,WAAW,CAAC,SAAiB;QAClC,OAAO,IAAI,CAAC,oBAAoB,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC;IAC3D,CAAC;IAEM,8BAA8B,CAAC,SAAiB,EAAE,UAAkB;QACzE,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,aAAa,CAAC,8BAA8B,CAAC,SAAS,EAAE,UAAU,CAAC,CAAC;IAC3E,CAAC;IAEM,kBAAkB,CAAC,OAAsB;QAC9C,MAAM,OAAO,GAAG,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC/C,IAAI,OAAO,CAAC,OAAO,EAAE;YACnB,OAAO;SACR;QAED,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC;YAC5B;gBACE,QAAQ,EAAE,UAAU;gBACpB,aAAa,EAAE,KAAK;gBACpB,EAAE,EAAE,OAAO,CAAC,EAAE;gBACd,QAAQ,EAAE,OAAO,CAAC,KAAK;aACxB;SACF,CAAC,CAAC;IACL,CAAC;IAEM,aAAa,CAAC,OAAsB;QACzC,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC,CAAC;IAC9C,CAAC;IAEM,gBAAgB;QACrB,MAAM,UAAU,GAAG,IAAI,CAAC,QAAQ,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;QAC5D,IAAI,CAAC,iBAAiB,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;IAC5C,CAAC;IAEO,cAAc,CAAC,QAAyB,EAAE,UAAmB,EAAE,iBAAgC;QACrG,MAAM,GAAG,GAAa,EAAE,CAAC;QAEzB,QAAQ,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;;YACtB,IAAI,CAAC,IAAI,CAAC,EAAE,EAAE;gBACZ,OAAO;aACR;YAED,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;YAElB,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,EAAE;gBAChC,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,EAAE,EAAE,IAAI,WAAW,CAAC,IAAI,CAAC,GAAG,EAAE,EAAE,CAAC,CAAC,CAAC;aAC9D;iBAAM;gBACL,MAAA,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,0CAAE,QAAQ,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;aACjD;YAED,IAAI,CAAC,UAAU,IAAI,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,SAAS,KAAK,iBAAiB,EAAE;gBACvE,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,OAAO,EAAE,CAAC;aACvC;iBAAM;gBACL,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC,MAAM,EAAE,CAAC;aACtC;QACH,CAAC,CAAC,CAAC;QAEH,MAAM,UAAU,GAAG,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC,MAAM,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC;QACnF,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC,EAAE,CAAC,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC,CAAC,CAAC;QACtD,IAAI,CAAC,iBAAiB,GAAG,QAAQ,CAAC,IAAI,CAAC,CAAC,IAAmB,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;IAChF,CAAC;;kHAxHU,oBAAoB;sGAApB,oBAAoB,uIACpB,YAAY,qECxBzB,m+HA8FA;4FDvEa,oBAAoB;kBANhC,SAAS;mBAAC;oBACT,QAAQ,EAAE,iBAAiB;oBAC3B,WAAW,EAAE,+BAA+B;oBAC5C,SAAS,EAAE,CAAC,+BAA+B,CAAC;oBAC5C,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;kNAE0B,YAAY;sBAApC,SAAS;uBAAC,YAAY;gBAEd,QAAQ;sBAAhB,KAAK","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { FlowConfigurationService, ProductImagesService, QuoteDraftService } from '@veloce/sdk/core';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport { BehaviorSubject, combineLatest, filter, map, Observable, Subject, takeUntil } from 'rxjs';\nimport { FlowRouterService } from '../../../services/flow-router.service';\nimport { HeaderProduct } from '../header.types';\n\n@Component({\n  selector: 'vl-cart-preview',\n  templateUrl: './cart-preview.component.html',\n  styleUrls: ['./cart-preview.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CartPreviewComponent implements OnChanges, AfterViewInit, OnDestroy {\n  @ViewChild(OverlayPanel) overlayPanel!: OverlayPanel;\n\n  @Input() products!: HeaderProduct[];\n\n  public form = new FormGroup({});\n  public readonlyProductId$: Observable<string | null>;\n\n  private shouldUpdate$ = new BehaviorSubject(true);\n  private destroyed$ = new Subject<void>();\n\n  public isEditMode$: Observable<boolean>;\n  public hasTermInProducts = false;\n\n  constructor(\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\n    private productImagesService: ProductImagesService,\n    private quoteDraftService: QuoteDraftService,\n  ) {\n    this.readonlyProductId$ = combineLatest([\n      this.routerService.route$,\n      this.routerService.isConfigurationRoute$(),\n    ]).pipe(\n      map(([route, isConfigurationRoute]) => {\n        if (!isConfigurationRoute) {\n          return null;\n        }\n\n        return route.queryParams.productId;\n      }),\n    );\n\n    this.isEditMode$ = this.quoteDraftService.isEditMode$();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const { products } = changes;\n\n    if (products) {\n      this.shouldUpdate$.next(true);\n    }\n  }\n\n  ngAfterViewInit() {\n    combineLatest([this.readonlyProductId$, this.isEditMode$, this.shouldUpdate$, this.overlayPanel.onShow])\n      .pipe(\n        filter(() => this.overlayPanel.overlayVisible),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe(([readonlyProductId, isEditMode]) =>\n        this.updateControls(this.products, isEditMode, readonlyProductId),\n      );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public getImageUrl(productId: string): Observable<string | null> {\n    return this.productImagesService.getImageUrl$(productId);\n  }\n\n  public navigateToProductConfiguration(productId: string, lineItemId: string) {\n    this.overlayPanel.hide();\n    this.routerService.navigateToProductConfiguration(productId, lineItemId);\n  }\n\n  public controlBlurHandler(product: HeaderProduct): void {\n    const control = this.form.controls[product.id];\n    if (control.invalid) {\n      return;\n    }\n\n    this.flowConfiguration.update([\n      {\n        dataType: 'LINEITEM',\n        attributeType: 'QTY',\n        id: product.id,\n        newValue: control.value,\n      },\n    ]);\n  }\n\n  public deleteHandler(product: HeaderProduct): void {\n    this.flowConfiguration.delete([product.id]);\n  }\n\n  public deleteAllHandler(): void {\n    const productIds = this.products.map(product => product.id);\n    this.flowConfiguration.delete(productIds);\n  }\n\n  private updateControls(products: HeaderProduct[], isEditMode: boolean, readonlyProductId: string | null): void {\n    const ids: string[] = [];\n\n    products.forEach(item => {\n      if (!item.id) {\n        return;\n      }\n\n      ids.push(item.id);\n\n      if (!this.form.controls[item.id]) {\n        this.form.addControl(item.id, new FormControl(item.qty, []));\n      } else {\n        this.form.controls[item.id]?.setValue(item.qty);\n      }\n\n      if (!isEditMode || item.deleted || item.productId === readonlyProductId) {\n        this.form.controls[item.id].disable();\n      } else {\n        this.form.controls[item.id].enable();\n      }\n    });\n\n    const removedIds = Object.keys(this.form.controls).filter(id => !ids.includes(id));\n    removedIds.forEach(id => this.form.removeControl(id));\n    this.hasTermInProducts = products.some((item: HeaderProduct) => item.hasTerm);\n  }\n}\n","<p-overlayPanel\n  styleClass=\"navigation-settings-overlay flow-header-overlay center\"\n  showTransitionOptions=\"0ms\"\n  hideTransitionOptions=\"0ms\"\n>\n  <ng-template pTemplate>\n    <div class=\"flow-header-overlay__wrapper\" *vlLet=\"readonlyProductId$ | async as readonlyProductId\">\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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n          </h2>\n\n          <div class=\"product header\">\n            <span>Product</span>\n            <span class=\"text-right\">Qty</span>\n            <span class=\"text-right\">MRR</span>\n            <span class=\"text-right\">NRR</span>\n          </div>\n\n          <div class=\"scrollable-wrapper\">\n            <div class=\"product\" *ngFor=\"let product of products\">\n              <div\n                class=\"product__info\"\n                *vlLet=\"!isEditMode || product.deleted || readonlyProductId === product.productId as readonlyProduct\"\n              >\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 }}</div>\n                  <div>\n                    <p-button\n                      label=\"Configure\"\n                      [disabled]=\"readonlyProduct || !product.configurable || product.hasTerm\"\n                      styleClass=\"p-button-link p-button-sm pl-0\"\n                      (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n                    ></p-button>\n                    <p-button\n                      label=\"Delete\"\n                      [disabled]=\"readonlyProduct || product.hasTerm\"\n                      styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n                      (onClick)=\"deleteHandler(product)\"\n                    ></p-button>\n                  </div>\n                </div>\n              </div>\n\n              <span>\n                <p-inputNumber\n                  *ngIf=\"form.controls[product.id] as control\"\n                  class=\"qty-control\"\n                  [formControl]=\"$any(control)\"\n                  [min]=\"1\"\n                  [required]=\"true\"\n                  (onBlur)=\"controlBlurHandler(product)\"\n                ></p-inputNumber>\n              </span>\n              <span class=\"text-right pt-3\">${{ product.mrr }}</span>\n              <span class=\"text-right pt-3\">${{ product.nrr }}</span>\n            </div>\n\n            <ng-template #noImage>\n              <i class=\"vl-icon vl-icon-no-image no-image-icon\"></i>\n            </ng-template>\n          </div>\n\n          <div class=\"flex justify-content-end mt-3\">\n            <p-button\n              label=\"Clear cart\"\n              styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n              [disabled]=\"!isEditMode || !!readonlyProductId || 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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n        </h2>\n\n        <span class=\"empty-state\">There are no products added to the Shopping Cart yet.</span>\n      </ng-template>\n    </div>\n  </ng-template>\n</p-overlayPanel>\n"]}
121
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-preview.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-preview.component.ts","../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-preview.component.html"],"names":[],"mappings":"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","sourcesContent":["import {\n  AfterViewInit,\n  ChangeDetectionStrategy,\n  Component,\n  Input,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { FlowConfigurationService, ProductImagesService, QuoteDraftService } from '@veloce/sdk/core';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport { BehaviorSubject, combineLatest, filter, map, Observable, Subject, takeUntil } from 'rxjs';\nimport { FlowRouterService } from '../../../services/flow-router.service';\nimport { HeaderProduct } from '../header.types';\n\n@Component({\n  selector: 'vl-cart-preview',\n  templateUrl: './cart-preview.component.html',\n  styleUrls: ['./cart-preview.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class CartPreviewComponent implements OnChanges, AfterViewInit, OnDestroy {\n  @ViewChild(OverlayPanel) overlayPanel!: OverlayPanel;\n\n  @Input() products!: HeaderProduct[];\n\n  public form = new FormGroup({});\n  public readonlyProductId$: Observable<string | null>;\n\n  private shouldUpdate$ = new BehaviorSubject(true);\n  private destroyed$ = new Subject<void>();\n\n  public isEditMode$: Observable<boolean>;\n  public hasTermInProducts = false;\n\n  constructor(\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\n    private productImagesService: ProductImagesService,\n    private quoteDraftService: QuoteDraftService,\n  ) {\n    this.readonlyProductId$ = combineLatest([\n      this.routerService.route$,\n      this.routerService.isConfigurationRoute$(),\n    ]).pipe(\n      map(([route, isConfigurationRoute]) => {\n        if (!isConfigurationRoute) {\n          return null;\n        }\n\n        return route.queryParams.productId;\n      }),\n    );\n\n    this.isEditMode$ = this.quoteDraftService.isEditMode$();\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    const { products } = changes;\n\n    if (products) {\n      this.shouldUpdate$.next(true);\n    }\n  }\n\n  ngAfterViewInit() {\n    combineLatest([this.readonlyProductId$, this.isEditMode$, this.shouldUpdate$, this.overlayPanel.onShow])\n      .pipe(\n        filter(() => this.overlayPanel.overlayVisible),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe(([readonlyProductId, isEditMode]) =>\n        this.updateControls(this.products, isEditMode, readonlyProductId),\n      );\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public getImageUrl(productId: string): Observable<string | null> {\n    return this.productImagesService.getImageUrl$(productId);\n  }\n\n  public navigateToProductConfiguration(productId: string, lineItemId: string) {\n    this.overlayPanel.hide();\n    this.routerService.navigateToProductConfiguration(productId, lineItemId);\n  }\n\n  public controlBlurHandler(product: HeaderProduct): void {\n    const control = this.form.controls[product.id];\n    if (control.invalid) {\n      return;\n    }\n\n    this.flowConfiguration.update([\n      {\n        dataType: 'LINEITEM',\n        attributeType: 'QTY',\n        id: product.id,\n        newValue: control.value,\n      },\n    ]);\n  }\n\n  public deleteHandler(product: HeaderProduct): void {\n    this.flowConfiguration.delete([product.id]);\n  }\n\n  public deleteAllHandler(): void {\n    const productIds = this.products.map(product => product.id);\n    this.flowConfiguration.delete(productIds);\n  }\n\n  private updateControls(products: HeaderProduct[], isEditMode: boolean, readonlyProductId: string | null): void {\n    const ids: string[] = [];\n\n    products.forEach(item => {\n      if (!item.id) {\n        return;\n      }\n\n      ids.push(item.id);\n\n      if (!this.form.controls[item.id]) {\n        this.form.addControl(item.id, new FormControl(item.qty, []));\n      } else {\n        this.form.controls[item.id]?.setValue(item.qty);\n      }\n\n      if (!isEditMode || item.deleted || item.productId === readonlyProductId) {\n        this.form.controls[item.id].disable();\n      } else {\n        this.form.controls[item.id].enable();\n      }\n    });\n\n    const removedIds = Object.keys(this.form.controls).filter(id => !ids.includes(id));\n    removedIds.forEach(id => this.form.removeControl(id));\n    this.hasTermInProducts = products.some((item: HeaderProduct) => item.hasTerm);\n  }\n}\n","<p-overlayPanel\n  styleClass=\"navigation-settings-overlay flow-header-overlay center\"\n  showTransitionOptions=\"0ms\"\n  hideTransitionOptions=\"0ms\"\n>\n  <ng-template pTemplate>\n    <div class=\"flow-header-overlay__wrapper\" *vlLet=\"readonlyProductId$ | async as readonlyProductId\">\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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n          </h2>\n\n          <div class=\"product header\">\n            <span>Product</span>\n            <span class=\"text-right\">Qty</span>\n            <span class=\"text-right\">MRR</span>\n            <span class=\"text-right\">NRR</span>\n          </div>\n\n          <div class=\"scrollable-wrapper\">\n            <div class=\"product\" *ngFor=\"let product of products\">\n              <div\n                class=\"product__info\"\n                *vlLet=\"!isEditMode || product.deleted || readonlyProductId === product.productId as readonlyProduct\"\n              >\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 }}</div>\n                  <div>\n                    <p-button\n                      label=\"Configure\"\n                      [disabled]=\"readonlyProduct || !product.configurable || product.hasTerm\"\n                      styleClass=\"p-button-link p-button-sm pl-0\"\n                      (onClick)=\"navigateToProductConfiguration(product.productId, product.id)\"\n                    ></p-button>\n                    <p-button\n                      label=\"Delete\"\n                      [disabled]=\"readonlyProduct || product.hasTerm\"\n                      styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n                      (onClick)=\"deleteHandler(product)\"\n                    ></p-button>\n                  </div>\n                </div>\n              </div>\n\n              <span>\n                <p-inputNumber\n                  *ngIf=\"form.controls[product.id] as control\"\n                  class=\"qty-control\"\n                  [formControl]=\"$any(control)\"\n                  [min]=\"1\"\n                  [required]=\"true\"\n                  (onBlur)=\"controlBlurHandler(product)\"\n                ></p-inputNumber>\n              </span>\n              <span class=\"text-right pt-3\">${{ product.mrr }}</span>\n              <span class=\"text-right pt-3\">${{ product.nrr }}</span>\n            </div>\n\n            <ng-template #noImage>\n              <i class=\"vl-icon vl-icon-no-image no-image-icon\"></i>\n            </ng-template>\n          </div>\n\n          <div class=\"flex justify-content-end mt-3 footer\">\n            <p-button\n              label=\"Clear cart\"\n              styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n              [disabled]=\"!isEditMode || !!readonlyProductId || 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-icon-close close-icon\" (click)=\"overlayPanel.hide()\"></i>\n        </h2>\n\n        <span class=\"empty-state\">There are no products added to the Shopping Cart yet.</span>\n      </ng-template>\n    </div>\n  </ng-template>\n</p-overlayPanel>\n"]}
@@ -251,7 +251,7 @@ export class FlowHeaderComponent {
251
251
  }
252
252
  }
253
253
  FlowHeaderComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: FlowHeaderComponent, deps: [{ token: i1.ContextService }, { token: i1.QuoteDraftService }, { token: i2.QuoteApiService }, { token: i2.SalesforceApiService }, { token: i1.FlowConfigurationService }, { token: i3.FlowRouterService }, { token: i4.FlowDialogService }, { token: i5.IntegrationState }], target: i0.ɵɵFactoryTarget.Component });
254
- FlowHeaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", type: FlowHeaderComponent, selector: "vl-flow-header", ngImport: i0, template: "<div class=\"flow-info\" *vlLet=\"objectDetails$ | async as details\">\n <nav class=\"nav-item nav-back\" (click)=\"back()\">\n <i class=\"nav-icon vl-icon vl-icon-arrow-left\"></i>\n\n <span> Back to {{ objectName | titlecase }}</span>\n </nav>\n\n <ng-container *ngIf=\"isAccountMode\">\n <span class=\"dot-separator\"></span>\n\n <span>Account name</span>\n\n <nav class=\"account-name\" [pTooltip]=\"contextProperties.Name ?? ''\" tooltipPosition=\"bottom\" [showDelay]=\"1000\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n </nav>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode\">\n <span class=\"dot-separator\"></span>\n\n <span>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-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"navigation-settings-overlay flow-header-overlay center\" #quoteDetails>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\">\n <h2 class=\"flow-header-overlay__title\">\n <span>Quote Information</span>\n <i class=\"vl-icon vl-icon-close close-icon\" (click)=\"quoteDetails.hide()\"></i>\n </h2>\n\n <ul class=\"info-list\">\n <li class=\"info-list__row\">\n <span>Account Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n <span>{{ details.accountName }}</span>\n </a>\n </li>\n <li class=\"info-list__row\">\n <span>Opportunity Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n <span>{{ details.opportunityName }}</span>\n </a>\n </li>\n <li class=\"info-list__row\">\n <span>Quote Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n <span>{{ details.quoteName }}</span>\n </a>\n </li>\n </ul>\n </div>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n</div>\n\n<div class=\"flow-navigation\">\n <nav class=\"nav-item\" [ngClass]=\"{ active: isCatalogRoute$ | async }\" (click)=\"navigateToCatalog()\">Catalog</nav>\n <nav class=\"nav-item disabled\" [ngClass]=\"{ active: isConfigurationRoute$ | async }\">Configurator</nav>\n <ng-container *vlLet=\"products$ | async as products\">\n <nav class=\"nav-item\" [ngClass]=\"{ active: isCartRoute$ | async }\" (click)=\"navigateToShoppingCart()\">\n Shopping Cart ({{ products.length }})\n </nav>\n\n <nav class=\"nav-popover-toggle active\" (click)=\"cart?.overlayPanel?.toggle($event)\">\n <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n </nav>\n\n <vl-cart-preview #cart [products]=\"products\"></vl-cart-preview>\n </ng-container>\n</div>\n\n<div class=\"flow-controls\" *vlLet=\"objectDetails$ | async as details\">\n <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n <ng-container *ngIf=\"isAccountMode && assetPriceLists.length > 1\">\n <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n <span>{{ priceList?.name }}</span>\n <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-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 >{{ option.name }}</span\n >\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode\">\n <span>{{ priceList?.name }}</span>\n </ng-container>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode\">\n <span *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date: 'MM.dd.yyyy' }}</span>\n\n <span class=\"slash-separator\"></span>\n\n <span class=\"metrics\">\n <div class=\"metrics__row\">\n MRR: <span class=\"font-semibold\">${{ totalMRR$ | async }}</span>\n </div>\n <div class=\"metrics__row\">\n NRR: <span class=\"font-semibold\">${{ totalNRR$ | async }}</span>\n </div>\n </span>\n </ng-container>\n\n <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n <p-button\n class=\"doc-gen-button\"\n icon=\"vl-icon vl-icon-doc-gen\"\n (onClick)=\"docGenButtonClickHandler(isCartRoute)\"\n ></p-button>\n\n <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n <p-splitButton\n *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n label=\"Save to Quote\"\n (onClick)=\"saveButtonClickHandler(isCartRoute)\"\n [model]=\"getSplitButtonActions(isCartRoute)\"\n styleClass=\"p-button-outlined\"\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</div>\n", styles: [":host{display:flex;align-items:center;height:48px;width:100%;background-color:var(--vl-primary-color);color:#fff;padding:0 32px;flex-shrink:0}::ng-deep .p-overlaypanel.flow-header-overlay .p-overlaypanel-content{background-color:#fff;padding:16px}::ng-deep .p-overlaypanel.flow-header-overlay.left:before{left:6px!important}::ng-deep .p-overlaypanel.flow-header-overlay.right:before{right:6px!important}::ng-deep .p-overlaypanel.flow-header-overlay.left .p-overlaypanel-content{margin-left:-16px}::ng-deep .p-overlaypanel.flow-header-overlay.right .p-overlaypanel-content{margin-right:-16px}::ng-deep .p-overlaypanel.flow-header-overlay:before{background-color:#fff}::ng-deep .p-overlaypanel.price-list-overlay .p-overlaypanel-content{border-radius:5px;border-color:var(--vl-border-color);padding:0;display:flex;flex-direction:column;max-height:140px;overflow:auto}:host ::ng-deep p-splitButton{height:32px;width:146px;display:flex}:host ::ng-deep p-splitButton .p-splitbutton{display:flex;font-size:12px;border-radius:2px;height:100%;width:100%;-webkit-backface-visibility:hidden}:host ::ng-deep p-splitButton .p-splitbutton .p-button:focus{box-shadow:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button{background-color:#fff;color:var(--vl-primary-color);border:1px solid #fff}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button:enabled:hover{background-color:var(--vl-secondary-nav-bg);color:var(--vl-primary-color);border:1px solid var(--vl-secondary-nav-bg);outline:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-menubutton{background-color:var(--vl-secondary-nav-bg);border-radius:0 2px 2px 0;border:1px solid var(--vl-secondary-nav-bg);border-left:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-menubutton:hover{background-color:var(--vl-secondary-nav-bg);border:1px solid var(--vl-secondary-nav-bg);border-left:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-defaultbutton{border-radius:2px 0 0 2px}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-defaultbutton .p-button-label{font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}:host ::ng-deep p-splitButton .p-splitbutton .p-menu{display:flex;width:auto;margin-top:2px;border-radius:4px;border:1px solid var(--vl-border-color);box-shadow:0 4px 20px #2767c11a}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list{display:flex;flex-direction:column}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem{display:flex;width:100%;height:32px}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link{padding:8px;width:100%;height:100%}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link:not(.p-disabled):hover{background-color:var(--vl-secondary-nav-bg)}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link:not(.p-disabled):hover .p-menuitem-text{color:var(--vl-primary-color)}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link .p-menuitem-text{line-height:16px;font-size:12px;letter-spacing:.3px;white-space:nowrap}:host ::ng-deep .doc-gen-button{height:32px;width:32px;display:inline-flex}:host ::ng-deep .doc-gen-button .p-button{border:1px solid #fff;border-radius:2px;width:100%;height:100%}:host ::ng-deep .save-button,:host ::ng-deep .submit-button{height:32px;width:146px;display:flex}:host ::ng-deep .save-button .p-button,:host ::ng-deep .submit-button .p-button{border-radius:3px;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;grid-gap:8px;gap:8px;align-items:center}.flow-info .nav-popover-toggle{margin-left:-8px}.flow-info .nav-back{font-weight:bold}.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;padding:4px 0}nav.disabled{opacity:.6;cursor:default}nav .nav-icon{margin-right:10px}nav .icon-with-margin{margin:0 4px}nav a{color:#fff}nav.account-name{margin-left:4px;display:block;max-width:200px;overflow:hidden;text-overflow:ellipsis}nav.nav-popover-toggle{width:24px;display:flex;justify-content:center}nav.nav-popover-toggle i{pointer-events:none;margin:0}nav i{pointer-events:none}.dot-separator:after{content:\"\";display:block;width:4px;height:4px;border-radius:50%;background:#fff}.slash-separator:after{content:\"\";display:block;background:#fff;width:1px;height:16px}.flow-header-overlay__wrapper{width:360px}.flow-header-overlay__wrapper .close-icon{cursor:pointer}.flow-header-overlay__wrapper .info-list{list-style:none;padding:0;font-size:12px}.flow-header-overlay__wrapper .info-list__row{padding:8px 0;display:flex;justify-content:space-between}.flow-header-overlay__wrapper .info-list__row a{text-align:right}.flow-header-overlay__title{display:flex;justify-content:space-between;align-items:center;margin:0 0 24px}.flow-navigation{flex-grow:1;display:flex;grid-gap:16px;gap:16px;justify-content:center;font-weight:600}.flow-navigation .cart-nav-container{display:flex}.flow-navigation .nav-popover-toggle{margin-left:-14px}.flow-navigation .nav-item,.flow-navigation .nav-popover-toggle{opacity:.6}.flow-navigation .nav-item.active,.flow-navigation .nav-item:not(.disabled):hover,.flow-navigation .nav-popover-toggle.active,.flow-navigation .nav-popover-toggle:not(.disabled):hover{opacity:1}.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;grid-gap:8px;gap:8px}\n"], components: [{ type: i6.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { type: i7.CartPreviewComponent, selector: "vl-cart-preview", inputs: ["products"] }, { type: i8.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i9.SplitButton, selector: "p-splitButton", inputs: ["model", "icon", "iconPos", "label", "style", "styleClass", "menuStyle", "menuStyleClass", "disabled", "tabindex", "appendTo", "dir", "expandAriaLabel", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onClick", "onDropdownClick"] }], directives: [{ type: i10.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { type: i11.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i12.Tooltip, selector: "[pTooltip]", inputs: ["tooltipPosition", "tooltipEvent", "appendTo", "positionStyle", "tooltipStyleClass", "tooltipZIndex", "escape", "showDelay", "hideDelay", "life", "positionTop", "positionLeft", "pTooltip", "tooltipDisabled", "tooltipOptions"] }, { type: i13.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i11.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i11.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i11.AsyncPipe, "titlecase": i11.TitleCasePipe, "date": i11.DatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
254
+ FlowHeaderComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", type: FlowHeaderComponent, selector: "vl-flow-header", ngImport: i0, template: "<div class=\"flow-info\" *vlLet=\"objectDetails$ | async as details\">\n <nav class=\"nav-item nav-back\" (click)=\"back()\">\n <i class=\"nav-icon vl-icon vl-icon-arrow-left\"></i>\n\n <span> Back to {{ objectName | titlecase }}</span>\n </nav>\n\n <ng-container *ngIf=\"isAccountMode\">\n <span class=\"dot-separator\"></span>\n\n <span>Account name</span>\n\n <nav class=\"account-name\" [pTooltip]=\"contextProperties.Name ?? ''\" tooltipPosition=\"bottom\" [showDelay]=\"1000\">\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n </nav>\n </ng-container>\n\n <ng-container *ngIf=\"isQuoteMode\">\n <span class=\"dot-separator\"></span>\n\n <span>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-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n </nav>\n\n <p-overlayPanel styleClass=\"navigation-settings-overlay flow-header-overlay center\" #quoteDetails>\n <ng-template pTemplate>\n <div class=\"flow-header-overlay__wrapper\">\n <h2 class=\"flow-header-overlay__title\">\n <span>Quote Information</span>\n <i class=\"vl-icon vl-icon-close close-icon\" (click)=\"quoteDetails.hide()\"></i>\n </h2>\n\n <ul class=\"info-list\">\n <li class=\"info-list__row\">\n <span>Account Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n <span>{{ details.accountName }}</span>\n </a>\n </li>\n <li class=\"info-list__row\">\n <span>Opportunity Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n <span>{{ details.opportunityName }}</span>\n </a>\n </li>\n <li class=\"info-list__row\">\n <span>Quote Name:</span>\n <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n <span>{{ details.quoteName }}</span>\n </a>\n </li>\n </ul>\n </div>\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n</div>\n\n<div class=\"flow-navigation\">\n <nav class=\"nav-item\" [ngClass]=\"{ active: isCatalogRoute$ | async }\" (click)=\"navigateToCatalog()\">Catalog</nav>\n <nav class=\"nav-item disabled\" [ngClass]=\"{ active: isConfigurationRoute$ | async }\">Configurator</nav>\n <ng-container *vlLet=\"products$ | async as products\">\n <nav class=\"nav-item\" [ngClass]=\"{ active: isCartRoute$ | async }\" (click)=\"navigateToShoppingCart()\">\n Shopping Cart ({{ products.length }})\n </nav>\n\n <nav class=\"nav-popover-toggle active\" (click)=\"cart?.overlayPanel?.toggle($event)\">\n <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n </nav>\n\n <vl-cart-preview #cart [products]=\"products\"></vl-cart-preview>\n </ng-container>\n</div>\n\n<div class=\"flow-controls\" *vlLet=\"objectDetails$ | async as details\">\n <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n <div>\n <ng-container *ngIf=\"isAccountMode && assetPriceLists.length > 1; else singlePriceList\">\n <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n <span>{{ priceList?.name }}</span>\n <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-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 >{{ option.name }}</span\n >\n </ng-template>\n </p-overlayPanel>\n </ng-container>\n\n <ng-template #singlePriceList>\n <div>{{ priceList?.name }}</div>\n </ng-template>\n\n <ng-container *ngIf=\"isQuoteMode\">\n <div *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date: 'MM.dd.yyyy' }}</div>\n </ng-container>\n </div>\n </ng-container>\n\n <span class=\"metrics\">\n <div class=\"metrics__row\">\n <span>MRR:</span>\n <span class=\"font-semibold\">${{ totalMRR$ | async }}</span>\n </div>\n <div class=\"metrics__row\">\n <span>NRR:</span>\n <span class=\"font-semibold\">${{ totalNRR$ | async }}</span>\n </div>\n </span>\n\n <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n <p-button\n class=\"doc-gen-button\"\n icon=\"vl-icon vl-icon-doc-gen\"\n (onClick)=\"docGenButtonClickHandler(isCartRoute)\"\n ></p-button>\n\n <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n <p-splitButton\n *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n label=\"Save to Quote\"\n (onClick)=\"saveButtonClickHandler(isCartRoute)\"\n [model]=\"getSplitButtonActions(isCartRoute)\"\n styleClass=\"p-button-outlined\"\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</div>\n", styles: [":host{display:flex;align-items:center;height:48px;width:100%;background-color:var(--vl-primary-color);color:#fff;padding:0 32px;flex-shrink:0}::ng-deep .p-overlaypanel.flow-header-overlay .p-overlaypanel-content{background-color:#fff;padding:0}::ng-deep .p-overlaypanel.flow-header-overlay.left:before{left:6px!important}::ng-deep .p-overlaypanel.flow-header-overlay.right:before{right:6px!important}::ng-deep .p-overlaypanel.flow-header-overlay.left .p-overlaypanel-content{margin-left:-16px}::ng-deep .p-overlaypanel.flow-header-overlay.right .p-overlaypanel-content{margin-right:-16px}::ng-deep .p-overlaypanel.flow-header-overlay:before{background-color:#fff}::ng-deep .p-overlaypanel.price-list-overlay .p-overlaypanel-content{border-radius:5px;border-color:var(--vl-border-color);padding:0;display:flex;flex-direction:column;max-height:140px;overflow:auto}:host ::ng-deep p-splitButton{height:32px;width:146px;display:flex}:host ::ng-deep p-splitButton .p-splitbutton{display:flex;font-size:12px;border-radius:2px;height:100%;width:100%;-webkit-backface-visibility:hidden}:host ::ng-deep p-splitButton .p-splitbutton .p-button:focus{box-shadow:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button{background-color:#fff;color:var(--vl-primary-color);border:1px solid #fff}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button:enabled:hover{background-color:var(--vl-secondary-nav-bg);color:var(--vl-primary-color);border:1px solid var(--vl-secondary-nav-bg);outline:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-menubutton{background-color:var(--vl-secondary-nav-bg);border-radius:0 2px 2px 0;border:1px solid var(--vl-secondary-nav-bg);border-left:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-menubutton:hover{background-color:var(--vl-secondary-nav-bg);border:1px solid var(--vl-secondary-nav-bg);border-left:none}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-defaultbutton{border-radius:2px 0 0 2px}:host ::ng-deep p-splitButton .p-splitbutton.p-button-outlined .p-button.p-splitbutton-defaultbutton .p-button-label{font-size:12px;line-height:16px;letter-spacing:.3px;white-space:nowrap}:host ::ng-deep p-splitButton .p-splitbutton .p-menu{display:flex;width:auto;margin-top:2px;border-radius:4px;border:1px solid var(--vl-border-color);box-shadow:0 4px 20px #2767c11a}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list{display:flex;flex-direction:column}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem{display:flex;width:100%;height:32px}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link{padding:8px;width:100%;height:100%}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link:not(.p-disabled):hover{background-color:var(--vl-secondary-nav-bg)}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link:not(.p-disabled):hover .p-menuitem-text{color:var(--vl-primary-color)}:host ::ng-deep p-splitButton .p-splitbutton .p-menu .p-menu-list .p-menuitem .p-menuitem-link .p-menuitem-text{line-height:16px;font-size:12px;letter-spacing:.3px;white-space:nowrap}:host ::ng-deep .doc-gen-button{height:32px;width:32px;display:inline-flex}:host ::ng-deep .doc-gen-button .p-button{border:1px solid #fff;border-radius:2px;width:100%;height:100%}:host ::ng-deep .save-button,:host ::ng-deep .submit-button{height:32px;width:146px;display:flex}:host ::ng-deep .save-button .p-button,:host ::ng-deep .submit-button .p-button{border-radius:3px;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;grid-gap:8px;gap:8px;align-items:center}.flow-info .nav-popover-toggle{margin-left:-8px}.flow-info .nav-back{font-weight:bold}.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:14px}nav.disabled{opacity:.6;cursor:default}nav .nav-icon{margin-right:10px}nav .icon-with-margin{margin:0 4px}nav a{color:#fff}nav.account-name{margin-left:4px;display:block;max-width:200px;overflow:hidden;text-overflow:ellipsis}nav.nav-popover-toggle{width:24px;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;grid-gap:2px;gap:2px}.dot-separator:after{content:\"\";display:block;width:4px;height:4px;border-radius:50%;background:#fff}.slash-separator:after{content:\"\";display:block;background:#fff;width:1px;height:16px}.flow-header-overlay__wrapper{width:360px}.flow-header-overlay__wrapper .close-icon{cursor:pointer}.flow-header-overlay__wrapper .info-list{list-style:none;padding:0;font-size:12px}.flow-header-overlay__wrapper .info-list__row{padding:8px 0;display:flex;justify-content:space-between}.flow-header-overlay__wrapper .info-list__row a{text-align:right}.flow-header-overlay__title{display:flex;justify-content:space-between;align-items:center;margin:0 0 24px}.flow-navigation{flex-grow:1;display:flex;grid-gap:16px;gap:16px;justify-content:center;font-weight:600}.flow-navigation .cart-nav-container{display:flex}.flow-navigation .nav-popover-toggle{margin-left:-14px}.flow-navigation .nav-item,.flow-navigation .nav-popover-toggle{opacity:.6}.flow-navigation .nav-item.active,.flow-navigation .nav-item:not(.disabled):hover,.flow-navigation .nav-popover-toggle.active,.flow-navigation .nav-popover-toggle:not(.disabled):hover{opacity:1}.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;grid-gap:16px;gap:16px}\n"], components: [{ type: i6.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { type: i7.CartPreviewComponent, selector: "vl-cart-preview", inputs: ["products"] }, { type: i8.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i9.SplitButton, selector: "p-splitButton", inputs: ["model", "icon", "iconPos", "label", "style", "styleClass", "menuStyle", "menuStyleClass", "disabled", "tabindex", "appendTo", "dir", "expandAriaLabel", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onClick", "onDropdownClick"] }], directives: [{ type: i10.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { type: i11.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i12.Tooltip, selector: "[pTooltip]", inputs: ["tooltipPosition", "tooltipEvent", "appendTo", "positionStyle", "tooltipStyleClass", "tooltipZIndex", "escape", "showDelay", "hideDelay", "life", "positionTop", "positionLeft", "pTooltip", "tooltipDisabled", "tooltipOptions"] }, { type: i13.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i11.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { type: i11.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }], pipes: { "async": i11.AsyncPipe, "titlecase": i11.TitleCasePipe, "date": i11.DatePipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
255
255
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: FlowHeaderComponent, decorators: [{
256
256
  type: Component,
257
257
  args: [{
@@ -261,4 +261,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
261
261
  changeDetection: ChangeDetectionStrategy.OnPush,
262
262
  }]
263
263
  }], ctorParameters: function () { return [{ type: i1.ContextService }, { type: i1.QuoteDraftService }, { type: i2.QuoteApiService }, { type: i2.SalesforceApiService }, { type: i1.FlowConfigurationService }, { type: i3.FlowRouterService }, { type: i4.FlowDialogService }, { type: i5.IntegrationState }]; } });
264
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header.component.js","sourceRoot":"","sources":["../../../../../../../libs/sdk/src/components/header/header.component.ts","../../../../../../../libs/sdk/src/components/header/header.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,uBAAuB,EAAE,SAAS,EAAqB,MAAM,eAAe,CAAC;AAEtF,OAAO,EACL,wBAAwB,GAMzB,MAAM,cAAc,CAAC;AACtB,OAAO,EAAoB,gBAAgB,EAAE,MAAM,iBAAiB,CAAC;AAGrE,OAAO,EACL,eAAe,EACf,oBAAoB,EACpB,QAAQ,EACR,KAAK,EACL,GAAG,EACH,IAAI,EAEJ,EAAE,EACF,OAAO,EACP,SAAS,EACT,SAAS,EACT,GAAG,GACJ,MAAM,MAAM,CAAC;;;;;;;;;;;;;;;AAad,MAAM,OAAO,mBAAmB;IAoB9B,YACU,cAA8B,EAC9B,iBAAoC,EACpC,eAAgC,EAChC,YAAkC,EAClC,iBAA2C,EAC3C,aAAgC,EAChC,aAAgC,EAChC,gBAAkC;QAPlC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,oBAAe,GAAf,eAAe,CAAiB;QAChC,iBAAY,GAAZ,YAAY,CAAsB;QAClC,sBAAiB,GAAjB,iBAAiB,CAA0B;QAC3C,kBAAa,GAAb,aAAa,CAAmB;QAChC,kBAAa,GAAb,aAAa,CAAmB;QAChC,qBAAgB,GAAhB,gBAAgB,CAAkB;QA3BrC,eAAU,GAAG,EAAE,CAAC;QAKhB,mBAAc,GAAG,IAAI,eAAe,CAAsB,EAAE,CAAC,CAAC;QAM9D,sBAAiB,GAAG,IAAI,eAAe,CAAC,KAAK,CAAC,CAAC;QAC/C,wBAAmB,GAAG,IAAI,eAAe,CAAC,KAAK,CAAC,CAAC;QAKhD,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;QAYvC,MAAM,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;QAE1C,IAAI,CAAC,IAAI,GAAG,GAAG,CAAC,IAAI,CAAC;QACrB,IAAI,CAAC,UAAU,GAAG,GAAG,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC;QACzC,IAAI,CAAC,iBAAiB,GAAG,GAAG,CAAC,UAAU,CAAC;QACxC,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,iBAAiB,CAAC,eAAe,CAAC;QAE9D,IAAI,CAAC,gBAAgB,GAAG,IAAI,CAAC,iBAAiB,CAAC,gBAAgB,CAAC;QAChE,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,OAAO,CAAC,EAAE,WAAC,OAAA,MAAA,OAAO,CAAC,UAAU,CAAC,MAAM,mCAAI,EAAE,CAAA,EAAA,CAAC,CAAC,CAAC;QACpG,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;QACxD,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,iBAAiB,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,SAAS,CAAC,CAAC,CAAC,CAAC;QACvG,IAAI,CAAC,qBAAqB,GAAG,IAAI,CAAC,aAAa,CAAC,qBAAqB,EAAE,CAAC;QACxE,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,YAAY,EAAE,CAAC;QACtD,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,aAAa,CAAC,eAAe,EAAE,CAAC;QAE5D,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,IAAI,CAClD,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAAC,EACxC,oBAAoB,EAAE,EACtB,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CACnC,CAAC;QAEF,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC,cAAc,CAAC,QAAQ,EAAE,CAAC,IAAI,CAClD,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,UAAU,CAAC,aAAa,CAAC,EACxC,oBAAoB,EAAE,EACtB,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC,CACnC,CAAC;IACJ,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,qBAAqB,EAAE,CAAC;IAC/B,CAAC;IAED,WAAW;QACT,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAED,IAAW,aAAa;QACtB,OAAO,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,OAAO,CAAC;IACxD,CAAC;IAED,IAAW,WAAW;QACpB,OAAO,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,KAAK,CAAC;IACtD,CAAC;IAEM,qBAAqB,CAAC,WAAoB;QAC/C,OAAO;YACL;gBACE,KAAK,EAAE,eAAe;gBACtB,OAAO,EAAE,GAAG,EAAE;oBACZ,IAAI,CAAC,sBAAsB,CAAC,WAAW,CAAC,CAAC;gBAC3C,CAAC;aACF;YACD;gBACE,KAAK,EAAE,qBAAqB;gBAC5B,OAAO,EAAE,GAAG,EAAE;oBACZ,IAAI,CAAC,wBAAwB,CAAC,WAAW,CAAC,CAAC;gBAC7C,CAAC;aACF;SACF,CAAC;IACJ,CAAC;IAEM,IAAI,CAAC,QAAiB;QAC3B,MAAM,QAAQ,GAAG,QAAQ,aAAR,QAAQ,cAAR,QAAQ,GAAI,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC,QAAQ,CAAC;QAEpE,IAAI,QAAQ,EAAE;YACZ,MAAM,CAAC,YAAY,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC,QAAQ,CAAC,CAAC,CAAC;SAC7C;IACH,CAAC;IAEM,uBAAuB,CAAC,QAAiB;QAC9C,IAAI,CAAC,QAAQ,EAAE;YACb,OAAO,EAAE,CAAC;SACX;QAED,OAAO,GAAG,MAAM,CAAC,QAAQ,CAAC,MAAM,IAAI,QAAQ,EAAE,CAAC;IACjD,CAAC;IAEM,sBAAsB;QAC3B,IAAI,CAAC,aAAa,CAAC,sBAAsB,EAAE,CAAC;IAC9C,CAAC;IAEM,iBAAiB;QACtB,IAAI,CAAC,aAAa,CAAC,iBAAiB,EAAE,CAAC;IACzC,CAAC;IAEM,wBAAwB,CAAC,WAAoB;QAClD,IAAI,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,OAAO,EAAE;YAClD,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,EAAE;gBAC7C,IAAI,CAAC,aAAa,CAAC,0BAA0B,EAAE,CAAC,SAAS,EAAE,CAAC;aAC7D;iBAAM;gBACL,IAAI,CAAC,aAAa,CAAC,qCAAqC,EAAE,CAAC,SAAS,EAAE,CAAC;aACxE;YACD,OAAO;SACR;QAED,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,EAAE;YACxC,IAAI,CAAC,aAAa,CAAC,wBAAwB,EAAE,CAAC,SAAS,EAAE,CAAC;YAC1D,OAAO;SACR;QAED,IAAI,CAAC,WAAW,EAAE;YAChB,IAAI,CAAC,aAAa,CAAC,mCAAmC,EAAE,CAAC,SAAS,EAAE,CAAC;YACrE,OAAO;SACR;QAED,IAAI,UAAU,GAAG,EAAE,CAAC,IAAI,CAAC,CAAC;QAE1B,IAAI,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,EAAE;YAC5C,UAAU,GAAG,IAAI,CAAC,aAAa,CAAC,8BAA8B,EAAE,CAAC,IAAI,CACnE,SAAS,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC,EACtE,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,CAChB,CAAC;SACH;QAED,UAAU;aACP,IAAI,CACH,KAAK,EAAE,EACP,GAAG,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,gBAAgB,CAAC,QAAQ,CAAC,gBAAgB,EAAE,CAAC,CAAC,CAC9D;aACA,SAAS,EAAE,CAAC;IACjB,CAAC;IAEM,sBAAsB,CAAC,WAAoB;QAChD,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,EAAE;YACxC,IAAI,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,KAAK,EAAE;gBAChD,IAAI,CAAC,aAAa,CAAC,2BAA2B,EAAE,CAAC,SAAS,EAAE,CAAC;aAC9D;iBAAM;gBACL,IAAI,CAAC,aAAa,CAAC,sBAAsB,EAAE,CAAC,SAAS,EAAE,CAAC;aACzD;YACD,OAAO;SACR;QAED,IAAI,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,OAAO,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,EAAE;YAC/F,IAAI,CAAC,aAAa,CAAC,0BAA0B,EAAE,CAAC,SAAS,EAAE,CAAC;YAC5D,OAAO;SACR;QAED,IAAI,CAAC,WAAW,EAAE;YAChB,IAAI,CAAC,aAAa,CAAC,kCAAkC,EAAE,CAAC,SAAS,EAAE,CAAC;YACpE,OAAO;SACR;QAED,MAAM,SAAS,GAAG,IAAI,CAAC,iBAAiB,CAAC,WAAW,EAAE,CAAC;QAEvD,IAAI,CAAC,SAAS,CAAC,MAAM,EAAE;YACrB,IAAI,CAAC,aAAa,CAAC,mBAAmB,EAAE,CAAC,SAAS,EAAE,CAAC;YACrD,OAAO;SACR;QAED,IAAI,CAAC,UAAU,EAAE,CAAC,SAAS,EAAE,CAAC;IAChC,CAAC;IAEM,wBAAwB,CAAC,WAAoB;QAClD,IAAI,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,iBAAiB,CAAC,UAAU,EAAE,EAAE;YACxF,IAAI,CAAC,aAAa,CAAC,oCAAoC,EAAE,CAAC;YAC1D,OAAO;SACR;QAED,IAAI,IAAI,CAAC,IAAI,KAAK,wBAAwB,CAAC,OAAO,EAAE;YAClD,IAAI,CAAC,aAAa,CAAC,8BAA8B,EAAE,CAAC;YACpD,OAAO;SACR;QAED,IAAI,CAAC,WAAW,EAAE;YAChB,IAAI,CAAC,aAAa,CAAC,+CAA+C,EAAE,CAAC,SAAS,EAAE,CAAC;YACjF,OAAO;SACR;QAED,MAAM,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC;QACrD,IAAI,CAAC,UAAU,EAAE;YACf,OAAO;SACR;QAED,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAEpC,IAAI,CAAC,eAAe;aACjB,WAAW,CAAC,UAAU,CAAC;aACvB,IAAI,CACH,GAAG,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YAClB,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,GAAG,KAAK,CAAC;YACjD,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACrB,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,mBAAmB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EACpD,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,EAAE,CAAC;IACjB,CAAC;IAEM,eAAe,CAAC,WAAmB;QACxC,IAAI,CAAC,iBAAiB,CAAC,qBAAqB,CAAC,WAAW,CAAC,CAAC;IAC5D,CAAC;IAEO,UAAU,CAAC,UAAU,GAAG,KAAK;QACnC,MAAM,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,4BAA4B,CAAC;QACvE,IAAI,CAAC,UAAU,EAAE;YACf,OAAO,EAAE,CAAC,SAAS,CAAC,CAAC;SACtB;QAED,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;QAElC,OAAO,IAAI,CAAC,eAAe,CAAC,WAAW,CAAC,UAAU,CAAC,CAAC,IAAI,CACtD,GAAG,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YAClB,IAAI,CAAC,iBAAiB,CAAC,iBAAiB,GAAG,KAAK,CAAC;YAEjD,IAAI,CAAC,UAAU,EAAE;gBACf,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;aACpB;QACH,CAAC,CAAC,EACF,QAAQ,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,EAClD,GAAG,CAAC,IAAI,CAAC,EACT,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B,CAAC;IACJ,CAAC;IAEO,UAAU,CAAC,UAAkB,EAAE,EAAW;QAChD,IAAI,CAAC,EAAE,EAAE;YACP,OAAO,EAAE,CAAC,EAAE,CAAC,CAAC;SACf;QAED,MAAM,aAAa,GAAkB;YACnC,KAAK,EAAE,CAAC;YACR,YAAY,EAAE,SAAS,EAAE,GAAG;YAC5B,MAAM,EAAE,CAAC,MAAM,CAAC;SACjB,CAAC;QAEF,OAAO,IAAI,CAAC,YAAY,CAAC,KAAK,CAAM,aAAa,EAAE,UAAU,CAAC,CAAC,IAAI,CACjE,GAAG,CAAC,MAAM,CAAC,EAAE,eAAC,OAAA,MAAA,MAAA,MAAM,CAAC,CAAC,CAAC,0CAAE,IAAI,mCAAI,EAAE,CAAA,EAAA,CAAC,EACpC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B,CAAC;IACJ,CAAC;IAEO,qBAAqB;;QAC3B,MAAM,SAAS,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC;QACpG,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,aAAa,CAAC;QAC3D,MAAM,OAAO,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC;QACzE,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC;QAC7E,MAAM,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,CAAC,CAAC,MAAA,IAAI,CAAC,iBAAiB,CAAC,WAAW,0CAAE,OAAO,CAAC,KAAK,EAAE,EAAE,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;QAE1G,IAAI,CAAC,cAAc,CAAC,IAAI,iCACnB,IAAI,CAAC,cAAc,CAAC,KAAK,KAC5B,SAAS;YACT,aAAa;YACb,OAAO;YACP,SAAS;YACT,WAAW,IACX,CAAC;QAEH,IAAI,CAAC,UAAU,CAAC,SAAS,EAAE,SAAS,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,EAAE,CAC5D,IAAI,CAAC,cAAc,CAAC,IAAI,iCACnB,IAAI,CAAC,cAAc,CAAC,KAAK,KAC5B,WAAW,IACX,CACH,CAAC;QACF,IAAI,CAAC,UAAU,CAAC,aAAa,EAAE,aAAa,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,EAAE,CACxE,IAAI,CAAC,cAAc,CAAC,IAAI,iCACnB,IAAI,CAAC,cAAc,CAAC,KAAK,KAC5B,eAAe,IACf,CACH,CAAC;IACJ,CAAC;IAEO,YAAY,CAAC,KAAc;QACjC,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;IAChD,CAAC;IAEO,gBAAgB,CAAC,SAAqB;QAC5C,MAAM,IAAI,GAAG,IAAI,IAAI,EAAE,CAAC;QACxB,IAAI,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,CAAC,CAAC;QAC1B,MAAM,GAAG,GAAG,IAAI,CAAC,OAAO,EAAE,CAAC;QAC3B,MAAM,aAAa,GAAG,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,cAAc,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC;QAE7E,OAAO,SAAS,CAAC,MAAM,CAAC,CAAC,MAAuB,EAAE,EAAE,EAAE,EAAE;YACtD,2BAA2B;YAC3B,IAAI,EAAE,CAAC,cAAc,EAAE;gBACrB,OAAO,MAAM,CAAC;aACf;YAED,8BAA8B;YAC9B,IAAI,MAAM,GAAyB,EAAE,CAAC;YACtC,OAAO,MAAM,IAAI,MAAM,CAAC,OAAO,IAAI,IAAI,IAAI,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,OAAO,EAAE,IAAI,GAAG,EAAE;gBAC5E,MAAM,GAAG,SAAS,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,cAAc,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;aAC9D;YAED,IAAI,MAAM,IAAI,MAAM,CAAC,SAAS,EAAE;gBAC9B,OAAO;oBACL,GAAG,MAAM;oBACT;wBACE,EAAE,EAAE,MAAM,CAAC,EAAE;wBACb,SAAS,EAAE,MAAM,CAAC,SAAS;wBAC3B,IAAI,EAAE,MAAM,CAAC,IAAI;wBACjB,YAAY,EAAE,MAAM,CAAC,UAAU,CAAC,eAAe,CAAC,KAAK,MAAM;wBAC3D,OAAO,EAAE,MAAM,CAAC,UAAU,KAAK,QAAQ;wBACvC,OAAO,EAAE,aAAa,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;wBAC1C,GAAG,EAAE,MAAM,CAAC,GAAG;wBACf,GAAG,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC;wBACvD,GAAG,EAAE,IAAI,CAAC,YAAY,CAAC,MAAM,CAAC,UAAU,CAAC,aAAa,CAAC;qBACxD;iBACF,CAAC;aACH;YAED,OAAO,MAAM,CAAC;QAChB,CAAC,EAAE,EAAE,CAAC,CAAC;IACT,CAAC;;iHA7UU,mBAAmB;qGAAnB,mBAAmB,sDCvChC,qwMAiKA;4FD1Ha,mBAAmB;kBAN/B,SAAS;mBAAC;oBACT,QAAQ,EAAE,gBAAgB;oBAC1B,WAAW,EAAE,yBAAyB;oBACtC,SAAS,EAAE,CAAC,yBAAyB,CAAC;oBACtC,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD","sourcesContent":["import { ChangeDetectionStrategy, Component, OnDestroy, OnInit } from '@angular/core';\nimport { QuoteApiService, SalesforceApiService } from '@veloce/api';\nimport {\n  ConfigurationContextMode,\n  ContextProperties,\n  LineItem,\n  PriceList,\n  SearchRequest,\n  VlWindow,\n} from '@veloce/core';\nimport { IntegrationState, OpenDocGenAction } from '@veloce/sdk/cms';\nimport { ContextService, FlowConfigurationService, QuoteDraftService } from '@veloce/sdk/core';\nimport { MenuItem } from 'primeng/api';\nimport {\n  BehaviorSubject,\n  distinctUntilChanged,\n  finalize,\n  first,\n  map,\n  noop,\n  Observable,\n  of,\n  Subject,\n  switchMap,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { FlowDialogService } from '../../services/flow-dialog.service';\nimport { FlowRouterService } from '../../services/flow-router.service';\nimport { HeaderObjectDetails, HeaderProduct } from './header.types';\n\ndeclare const window: VlWindow;\n\n@Component({\n  selector: 'vl-flow-header',\n  templateUrl: './header.component.html',\n  styleUrls: ['./header.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FlowHeaderComponent implements OnInit, OnDestroy {\n  public objectName = '';\n  public contextProperties: ContextProperties;\n  public assetPriceLists: PriceList[];\n  public activePriceList$: Observable<PriceList | null>;\n  public status$: Observable<string>;\n  public objectDetails$ = new BehaviorSubject<HeaderObjectDetails>({});\n  public products$: Observable<HeaderProduct[]>;\n  public isConfigurationRoute$: Observable<boolean>;\n  public isCartRoute$: Observable<boolean>;\n  public isCatalogRoute$: Observable<boolean>;\n  public isEditMode$: Observable<boolean>;\n  public isSaveInProgress$ = new BehaviorSubject(false);\n  public isSubmitInProgress$ = new BehaviorSubject(false);\n  public totalMRR$: Observable<string>;\n  public totalNRR$: Observable<string>;\n\n  private mode?: ConfigurationContextMode;\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    private contextService: ContextService,\n    private quoteDraftService: QuoteDraftService,\n    private quoteApiService: QuoteApiService,\n    private sfApiService: SalesforceApiService,\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\n    private dialogService: FlowDialogService,\n    private integrationState: IntegrationState,\n  ) {\n    const ctx = this.contextService.resolve();\n\n    this.mode = ctx.mode;\n    this.objectName = ctx.mode.toLowerCase();\n    this.contextProperties = ctx.properties;\n    this.assetPriceLists = this.quoteDraftService.assetPriceLists;\n\n    this.activePriceList$ = this.quoteDraftService.activePriceList$;\n    this.status$ = this.contextService.resolve$().pipe(map(context => context.properties.Status ?? ''));\n    this.isEditMode$ = this.quoteDraftService.isEditMode$();\n    this.products$ = this.flowConfiguration.get().pipe(map(lineItems => this.generateProducts(lineItems)));\n    this.isConfigurationRoute$ = this.routerService.isConfigurationRoute$();\n    this.isCartRoute$ = this.routerService.isCartRoute$();\n    this.isCatalogRoute$ = this.routerService.isCatalogRoute$();\n\n    this.totalMRR$ = this.contextService.resolve$().pipe(\n      map(ctx => ctx.properties.VDM_Total_MRR),\n      distinctUntilChanged(),\n      map(mrr => this.formatMetric(mrr)),\n    );\n\n    this.totalNRR$ = this.contextService.resolve$().pipe(\n      map(ctx => ctx.properties.VDM_Total_NRR),\n      distinctUntilChanged(),\n      map(nrr => this.formatMetric(nrr)),\n    );\n  }\n\n  ngOnInit() {\n    this.populateObjectDetails();\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public get isAccountMode() {\n    return this.mode === ConfigurationContextMode.ACCOUNT;\n  }\n\n  public get isQuoteMode() {\n    return this.mode === ConfigurationContextMode.QUOTE;\n  }\n\n  public getSplitButtonActions(isCartRoute: boolean): MenuItem[] {\n    return [\n      {\n        label: 'Save to Quote',\n        command: () => {\n          this.saveButtonClickHandler(isCartRoute);\n        },\n      },\n      {\n        label: 'Submit For Approval',\n        command: () => {\n          this.submitButtonClickHandler(isCartRoute);\n        },\n      },\n    ];\n  }\n\n  public back(objectId?: string): void {\n    const targetId = objectId ?? this.contextService.resolve().headerId;\n\n    if (targetId) {\n      window.VELO_BACK_FN.apply(null, [targetId]);\n    }\n  }\n\n  public getSalesforceObjectLink(objectId?: string): string {\n    if (!objectId) {\n      return '';\n    }\n\n    return `${window.location.origin}/${objectId}`;\n  }\n\n  public navigateToShoppingCart() {\n    this.routerService.navigateToShoppingCart();\n  }\n\n  public navigateToCatalog() {\n    this.routerService.navigateToCatalog();\n  }\n\n  public docGenButtonClickHandler(isCartRoute: boolean): void {\n    if (this.mode === ConfigurationContextMode.ACCOUNT) {\n      if (!this.quoteDraftService.hasUnsavedChanges) {\n        this.dialogService.showAccountNoChangesDialog().subscribe();\n      } else {\n        this.dialogService.showDocgenAccountUnsavedChangesDialog().subscribe();\n      }\n      return;\n    }\n\n    if (!this.quoteDraftService.isEditMode()) {\n      this.dialogService.showDocgenReadonlyDialog().subscribe();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showDocgenOutsideShoppingCartDialog().subscribe();\n      return;\n    }\n\n    let observable = of(true);\n\n    if (this.quoteDraftService.hasUnsavedChanges) {\n      observable = this.dialogService.showDocgenUnsavedChangesDialog().pipe(\n        switchMap(confirmed => (confirmed ? this.saveQuote$(true) : of(true))),\n        map(() => true),\n      );\n    }\n\n    observable\n      .pipe(\n        first(),\n        tap(() => this.integrationState.dispatch(OpenDocGenAction())),\n      )\n      .subscribe();\n  }\n\n  public saveButtonClickHandler(isCartRoute: boolean): void {\n    if (!this.quoteDraftService.isEditMode()) {\n      if (this.mode === ConfigurationContextMode.QUOTE) {\n        this.dialogService.showQuoteReadonlyModeDialog().subscribe();\n      } else {\n        this.dialogService.showReadonlyModeDialog().subscribe();\n      }\n      return;\n    }\n\n    if (this.mode === ConfigurationContextMode.ACCOUNT && !this.quoteDraftService.hasUnsavedChanges) {\n      this.dialogService.showAccountNoChangesDialog().subscribe();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showQuoteOutsideShoppingCartDialog().subscribe();\n      return;\n    }\n\n    const lineItems = this.flowConfiguration.getSnapshot();\n\n    if (!lineItems.length) {\n      this.dialogService.showEmptyCartDialog().subscribe();\n      return;\n    }\n\n    this.saveQuote$().subscribe();\n  }\n\n  public submitButtonClickHandler(isCartRoute: boolean): void {\n    if (this.mode === ConfigurationContextMode.QUOTE && !this.quoteDraftService.isEditMode()) {\n      this.dialogService.showReadonlyQuoteSubmitFailureDialog();\n      return;\n    }\n\n    if (this.mode === ConfigurationContextMode.ACCOUNT) {\n      this.dialogService.showAccountSubmitFailureDialog();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showOutsideShoppingCartQuoteSubmitFailureDialog().subscribe();\n      return;\n    }\n\n    const quoteDraft = this.quoteDraftService.quoteDraft;\n    if (!quoteDraft) {\n      return;\n    }\n\n    this.isSubmitInProgress$.next(true);\n\n    this.quoteApiService\n      .submitQuote(quoteDraft)\n      .pipe(\n        tap(({ quoteId }) => {\n          this.quoteDraftService.hasUnsavedChanges = false;\n          this.back(quoteId);\n        }),\n        finalize(() => this.isSubmitInProgress$.next(false)),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n  }\n\n  public selectPriceList(priceListId: string): void {\n    this.quoteDraftService.updateActivePriceList(priceListId);\n  }\n\n  private saveQuote$(stayOnPage = false): Observable<void> {\n    const quoteDraft = this.quoteDraftService.quoteDraftForActivePriceList;\n    if (!quoteDraft) {\n      return of(undefined);\n    }\n\n    this.isSaveInProgress$.next(true);\n\n    return this.quoteApiService.upsertQuote(quoteDraft).pipe(\n      tap(({ quoteId }) => {\n        this.quoteDraftService.hasUnsavedChanges = false;\n\n        if (!stayOnPage) {\n          this.back(quoteId);\n        }\n      }),\n      finalize(() => this.isSaveInProgress$.next(false)),\n      map(noop),\n      takeUntil(this.destroyed$),\n    );\n  }\n\n  private queryName$(objectName: string, id?: string): Observable<string> {\n    if (!id) {\n      return of('');\n    }\n\n    const searchRequest: SearchRequest = {\n      count: 1,\n      rawCondition: `Id = '${id}'`,\n      fields: ['Name'],\n    };\n\n    return this.sfApiService.query<any>(searchRequest, objectName).pipe(\n      map(result => result[0]?.Name ?? ''),\n      takeUntil(this.destroyed$),\n    );\n  }\n\n  private populateObjectDetails(): void {\n    const accountId = this.isAccountMode ? this.contextProperties.Id : this.contextProperties.AccountId;\n    const opportunityId = this.contextProperties.OpportunityId;\n    const quoteId = this.isQuoteMode ? this.contextProperties.Id : undefined;\n    const quoteName = this.isQuoteMode ? this.contextProperties.Name : undefined;\n    const quoteNumber = this.isQuoteMode ? this.contextProperties.QuoteNumber?.replace(/^0+/, '') : undefined;\n\n    this.objectDetails$.next({\n      ...this.objectDetails$.value,\n      accountId,\n      opportunityId,\n      quoteId,\n      quoteName,\n      quoteNumber,\n    });\n\n    this.queryName$('Account', accountId).subscribe(accountName =>\n      this.objectDetails$.next({\n        ...this.objectDetails$.value,\n        accountName,\n      }),\n    );\n    this.queryName$('Opportunity', opportunityId).subscribe(opportunityName =>\n      this.objectDetails$.next({\n        ...this.objectDetails$.value,\n        opportunityName,\n      }),\n    );\n  }\n\n  private formatMetric(value?: string): string {\n    return (value ? Number(value) : 0).toFixed(2);\n  }\n\n  private generateProducts(lineItems: LineItem[]): HeaderProduct[] {\n    const date = new Date();\n    date.setHours(0, 0, 0, 0);\n    const now = date.getTime();\n    const termParentIds = lineItems.map(li => li.rampInstanceId).filter(Boolean);\n\n    return lineItems.reduce((result: HeaderProduct[], li) => {\n      // find main term line item\n      if (li.rampInstanceId) {\n        return result;\n      }\n\n      // find current term line item\n      let target: LineItem | undefined = li;\n      while (target && target.endDate && new Date(target.endDate).getTime() <= now) {\n        target = lineItems.find(sub => sub.rampInstanceId === li.id);\n      }\n\n      if (target && target.productId) {\n        return [\n          ...result,\n          {\n            id: target.id,\n            productId: target.productId,\n            name: target.name,\n            configurable: target.properties['#configurable'] === 'true',\n            deleted: target.actionCode === 'DELETE',\n            hasTerm: termParentIds.includes(target.id),\n            qty: target.qty,\n            mrr: this.formatMetric(target.properties.VDM_Total_MRR),\n            nrr: this.formatMetric(target.properties.VDM_Total_NRR),\n          },\n        ];\n      }\n\n      return result;\n    }, []);\n  }\n}\n","<div class=\"flow-info\" *vlLet=\"objectDetails$ | async as details\">\n  <nav class=\"nav-item nav-back\" (click)=\"back()\">\n    <i class=\"nav-icon vl-icon vl-icon-arrow-left\"></i>\n\n    <span> Back to {{ objectName | titlecase }}</span>\n  </nav>\n\n  <ng-container *ngIf=\"isAccountMode\">\n    <span class=\"dot-separator\"></span>\n\n    <span>Account name</span>\n\n    <nav class=\"account-name\" [pTooltip]=\"contextProperties.Name ?? ''\" tooltipPosition=\"bottom\" [showDelay]=\"1000\">\n      <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n    </nav>\n  </ng-container>\n\n  <ng-container *ngIf=\"isQuoteMode\">\n    <span class=\"dot-separator\"></span>\n\n    <span>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-icon-chevron-down icon-with-margin\"></i>\n      <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n    </nav>\n\n    <p-overlayPanel styleClass=\"navigation-settings-overlay flow-header-overlay center\" #quoteDetails>\n      <ng-template pTemplate>\n        <div class=\"flow-header-overlay__wrapper\">\n          <h2 class=\"flow-header-overlay__title\">\n            <span>Quote Information</span>\n            <i class=\"vl-icon vl-icon-close close-icon\" (click)=\"quoteDetails.hide()\"></i>\n          </h2>\n\n          <ul class=\"info-list\">\n            <li class=\"info-list__row\">\n              <span>Account Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n                <span>{{ details.accountName }}</span>\n              </a>\n            </li>\n            <li class=\"info-list__row\">\n              <span>Opportunity Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n                <span>{{ details.opportunityName }}</span>\n              </a>\n            </li>\n            <li class=\"info-list__row\">\n              <span>Quote Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n                <span>{{ details.quoteName }}</span>\n              </a>\n            </li>\n          </ul>\n        </div>\n      </ng-template>\n    </p-overlayPanel>\n  </ng-container>\n</div>\n\n<div class=\"flow-navigation\">\n  <nav class=\"nav-item\" [ngClass]=\"{ active: isCatalogRoute$ | async }\" (click)=\"navigateToCatalog()\">Catalog</nav>\n  <nav class=\"nav-item disabled\" [ngClass]=\"{ active: isConfigurationRoute$ | async }\">Configurator</nav>\n  <ng-container *vlLet=\"products$ | async as products\">\n    <nav class=\"nav-item\" [ngClass]=\"{ active: isCartRoute$ | async }\" (click)=\"navigateToShoppingCart()\">\n      Shopping Cart ({{ products.length }})\n    </nav>\n\n    <nav class=\"nav-popover-toggle active\" (click)=\"cart?.overlayPanel?.toggle($event)\">\n      <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n      <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n    </nav>\n\n    <vl-cart-preview #cart [products]=\"products\"></vl-cart-preview>\n  </ng-container>\n</div>\n\n<div class=\"flow-controls\" *vlLet=\"objectDetails$ | async as details\">\n  <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n    <ng-container *ngIf=\"isAccountMode && assetPriceLists.length > 1\">\n      <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n        <span>{{ priceList?.name }}</span>\n        <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n        <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-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            >{{ option.name }}</span\n          >\n        </ng-template>\n      </p-overlayPanel>\n    </ng-container>\n\n    <ng-container *ngIf=\"isQuoteMode\">\n      <span>{{ priceList?.name }}</span>\n    </ng-container>\n  </ng-container>\n\n  <ng-container *ngIf=\"isQuoteMode\">\n    <span *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date: 'MM.dd.yyyy' }}</span>\n\n    <span class=\"slash-separator\"></span>\n\n    <span class=\"metrics\">\n      <div class=\"metrics__row\">\n        MRR: <span class=\"font-semibold\">${{ totalMRR$ | async }}</span>\n      </div>\n      <div class=\"metrics__row\">\n        NRR: <span class=\"font-semibold\">${{ totalNRR$ | async }}</span>\n      </div>\n    </span>\n  </ng-container>\n\n  <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n    <p-button\n      class=\"doc-gen-button\"\n      icon=\"vl-icon vl-icon-doc-gen\"\n      (onClick)=\"docGenButtonClickHandler(isCartRoute)\"\n    ></p-button>\n\n    <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n      <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n        <p-splitButton\n          *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n          label=\"Save to Quote\"\n          (onClick)=\"saveButtonClickHandler(isCartRoute)\"\n          [model]=\"getSplitButtonActions(isCartRoute)\"\n          styleClass=\"p-button-outlined\"\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</div>\n"]}
264
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"header.component.js","sourceRoot":"","sources":["../../../../../../../libs/sdk/src/components/header/header.component.ts","../../../../../../../libs/sdk/src/components/header/header.component.html"],"names":[],"mappings":"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","sourcesContent":["import { ChangeDetectionStrategy, Component, OnDestroy, OnInit } from '@angular/core';\nimport { QuoteApiService, SalesforceApiService } from '@veloce/api';\nimport {\n  ConfigurationContextMode,\n  ContextProperties,\n  LineItem,\n  PriceList,\n  SearchRequest,\n  VlWindow,\n} from '@veloce/core';\nimport { IntegrationState, OpenDocGenAction } from '@veloce/sdk/cms';\nimport { ContextService, FlowConfigurationService, QuoteDraftService } from '@veloce/sdk/core';\nimport { MenuItem } from 'primeng/api';\nimport {\n  BehaviorSubject,\n  distinctUntilChanged,\n  finalize,\n  first,\n  map,\n  noop,\n  Observable,\n  of,\n  Subject,\n  switchMap,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { FlowDialogService } from '../../services/flow-dialog.service';\nimport { FlowRouterService } from '../../services/flow-router.service';\nimport { HeaderObjectDetails, HeaderProduct } from './header.types';\n\ndeclare const window: VlWindow;\n\n@Component({\n  selector: 'vl-flow-header',\n  templateUrl: './header.component.html',\n  styleUrls: ['./header.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class FlowHeaderComponent implements OnInit, OnDestroy {\n  public objectName = '';\n  public contextProperties: ContextProperties;\n  public assetPriceLists: PriceList[];\n  public activePriceList$: Observable<PriceList | null>;\n  public status$: Observable<string>;\n  public objectDetails$ = new BehaviorSubject<HeaderObjectDetails>({});\n  public products$: Observable<HeaderProduct[]>;\n  public isConfigurationRoute$: Observable<boolean>;\n  public isCartRoute$: Observable<boolean>;\n  public isCatalogRoute$: Observable<boolean>;\n  public isEditMode$: Observable<boolean>;\n  public isSaveInProgress$ = new BehaviorSubject(false);\n  public isSubmitInProgress$ = new BehaviorSubject(false);\n  public totalMRR$: Observable<string>;\n  public totalNRR$: Observable<string>;\n\n  private mode?: ConfigurationContextMode;\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    private contextService: ContextService,\n    private quoteDraftService: QuoteDraftService,\n    private quoteApiService: QuoteApiService,\n    private sfApiService: SalesforceApiService,\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\n    private dialogService: FlowDialogService,\n    private integrationState: IntegrationState,\n  ) {\n    const ctx = this.contextService.resolve();\n\n    this.mode = ctx.mode;\n    this.objectName = ctx.mode.toLowerCase();\n    this.contextProperties = ctx.properties;\n    this.assetPriceLists = this.quoteDraftService.assetPriceLists;\n\n    this.activePriceList$ = this.quoteDraftService.activePriceList$;\n    this.status$ = this.contextService.resolve$().pipe(map(context => context.properties.Status ?? ''));\n    this.isEditMode$ = this.quoteDraftService.isEditMode$();\n    this.products$ = this.flowConfiguration.get().pipe(map(lineItems => this.generateProducts(lineItems)));\n    this.isConfigurationRoute$ = this.routerService.isConfigurationRoute$();\n    this.isCartRoute$ = this.routerService.isCartRoute$();\n    this.isCatalogRoute$ = this.routerService.isCatalogRoute$();\n\n    this.totalMRR$ = this.contextService.resolve$().pipe(\n      map(ctx => ctx.properties.VDM_Total_MRR),\n      distinctUntilChanged(),\n      map(mrr => this.formatMetric(mrr)),\n    );\n\n    this.totalNRR$ = this.contextService.resolve$().pipe(\n      map(ctx => ctx.properties.VDM_Total_NRR),\n      distinctUntilChanged(),\n      map(nrr => this.formatMetric(nrr)),\n    );\n  }\n\n  ngOnInit() {\n    this.populateObjectDetails();\n  }\n\n  ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public get isAccountMode() {\n    return this.mode === ConfigurationContextMode.ACCOUNT;\n  }\n\n  public get isQuoteMode() {\n    return this.mode === ConfigurationContextMode.QUOTE;\n  }\n\n  public getSplitButtonActions(isCartRoute: boolean): MenuItem[] {\n    return [\n      {\n        label: 'Save to Quote',\n        command: () => {\n          this.saveButtonClickHandler(isCartRoute);\n        },\n      },\n      {\n        label: 'Submit For Approval',\n        command: () => {\n          this.submitButtonClickHandler(isCartRoute);\n        },\n      },\n    ];\n  }\n\n  public back(objectId?: string): void {\n    const targetId = objectId ?? this.contextService.resolve().headerId;\n\n    if (targetId) {\n      window.VELO_BACK_FN.apply(null, [targetId]);\n    }\n  }\n\n  public getSalesforceObjectLink(objectId?: string): string {\n    if (!objectId) {\n      return '';\n    }\n\n    return `${window.location.origin}/${objectId}`;\n  }\n\n  public navigateToShoppingCart() {\n    this.routerService.navigateToShoppingCart();\n  }\n\n  public navigateToCatalog() {\n    this.routerService.navigateToCatalog();\n  }\n\n  public docGenButtonClickHandler(isCartRoute: boolean): void {\n    if (this.mode === ConfigurationContextMode.ACCOUNT) {\n      if (!this.quoteDraftService.hasUnsavedChanges) {\n        this.dialogService.showAccountNoChangesDialog().subscribe();\n      } else {\n        this.dialogService.showDocgenAccountUnsavedChangesDialog().subscribe();\n      }\n      return;\n    }\n\n    if (!this.quoteDraftService.isEditMode()) {\n      this.dialogService.showDocgenReadonlyDialog().subscribe();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showDocgenOutsideShoppingCartDialog().subscribe();\n      return;\n    }\n\n    let observable = of(true);\n\n    if (this.quoteDraftService.hasUnsavedChanges) {\n      observable = this.dialogService.showDocgenUnsavedChangesDialog().pipe(\n        switchMap(confirmed => (confirmed ? this.saveQuote$(true) : of(true))),\n        map(() => true),\n      );\n    }\n\n    observable\n      .pipe(\n        first(),\n        tap(() => this.integrationState.dispatch(OpenDocGenAction())),\n      )\n      .subscribe();\n  }\n\n  public saveButtonClickHandler(isCartRoute: boolean): void {\n    if (!this.quoteDraftService.isEditMode()) {\n      if (this.mode === ConfigurationContextMode.QUOTE) {\n        this.dialogService.showQuoteReadonlyModeDialog().subscribe();\n      } else {\n        this.dialogService.showReadonlyModeDialog().subscribe();\n      }\n      return;\n    }\n\n    if (this.mode === ConfigurationContextMode.ACCOUNT && !this.quoteDraftService.hasUnsavedChanges) {\n      this.dialogService.showAccountNoChangesDialog().subscribe();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showQuoteOutsideShoppingCartDialog().subscribe();\n      return;\n    }\n\n    const lineItems = this.flowConfiguration.getSnapshot();\n\n    if (!lineItems.length) {\n      this.dialogService.showEmptyCartDialog().subscribe();\n      return;\n    }\n\n    this.saveQuote$().subscribe();\n  }\n\n  public submitButtonClickHandler(isCartRoute: boolean): void {\n    if (this.mode === ConfigurationContextMode.QUOTE && !this.quoteDraftService.isEditMode()) {\n      this.dialogService.showReadonlyQuoteSubmitFailureDialog();\n      return;\n    }\n\n    if (this.mode === ConfigurationContextMode.ACCOUNT) {\n      this.dialogService.showAccountSubmitFailureDialog();\n      return;\n    }\n\n    if (!isCartRoute) {\n      this.dialogService.showOutsideShoppingCartQuoteSubmitFailureDialog().subscribe();\n      return;\n    }\n\n    const quoteDraft = this.quoteDraftService.quoteDraft;\n    if (!quoteDraft) {\n      return;\n    }\n\n    this.isSubmitInProgress$.next(true);\n\n    this.quoteApiService\n      .submitQuote(quoteDraft)\n      .pipe(\n        tap(({ quoteId }) => {\n          this.quoteDraftService.hasUnsavedChanges = false;\n          this.back(quoteId);\n        }),\n        finalize(() => this.isSubmitInProgress$.next(false)),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n  }\n\n  public selectPriceList(priceListId: string): void {\n    this.quoteDraftService.updateActivePriceList(priceListId);\n  }\n\n  private saveQuote$(stayOnPage = false): Observable<void> {\n    const quoteDraft = this.quoteDraftService.quoteDraftForActivePriceList;\n    if (!quoteDraft) {\n      return of(undefined);\n    }\n\n    this.isSaveInProgress$.next(true);\n\n    return this.quoteApiService.upsertQuote(quoteDraft).pipe(\n      tap(({ quoteId }) => {\n        this.quoteDraftService.hasUnsavedChanges = false;\n\n        if (!stayOnPage) {\n          this.back(quoteId);\n        }\n      }),\n      finalize(() => this.isSaveInProgress$.next(false)),\n      map(noop),\n      takeUntil(this.destroyed$),\n    );\n  }\n\n  private queryName$(objectName: string, id?: string): Observable<string> {\n    if (!id) {\n      return of('');\n    }\n\n    const searchRequest: SearchRequest = {\n      count: 1,\n      rawCondition: `Id = '${id}'`,\n      fields: ['Name'],\n    };\n\n    return this.sfApiService.query<any>(searchRequest, objectName).pipe(\n      map(result => result[0]?.Name ?? ''),\n      takeUntil(this.destroyed$),\n    );\n  }\n\n  private populateObjectDetails(): void {\n    const accountId = this.isAccountMode ? this.contextProperties.Id : this.contextProperties.AccountId;\n    const opportunityId = this.contextProperties.OpportunityId;\n    const quoteId = this.isQuoteMode ? this.contextProperties.Id : undefined;\n    const quoteName = this.isQuoteMode ? this.contextProperties.Name : undefined;\n    const quoteNumber = this.isQuoteMode ? this.contextProperties.QuoteNumber?.replace(/^0+/, '') : undefined;\n\n    this.objectDetails$.next({\n      ...this.objectDetails$.value,\n      accountId,\n      opportunityId,\n      quoteId,\n      quoteName,\n      quoteNumber,\n    });\n\n    this.queryName$('Account', accountId).subscribe(accountName =>\n      this.objectDetails$.next({\n        ...this.objectDetails$.value,\n        accountName,\n      }),\n    );\n    this.queryName$('Opportunity', opportunityId).subscribe(opportunityName =>\n      this.objectDetails$.next({\n        ...this.objectDetails$.value,\n        opportunityName,\n      }),\n    );\n  }\n\n  private formatMetric(value?: string): string {\n    return (value ? Number(value) : 0).toFixed(2);\n  }\n\n  private generateProducts(lineItems: LineItem[]): HeaderProduct[] {\n    const date = new Date();\n    date.setHours(0, 0, 0, 0);\n    const now = date.getTime();\n    const termParentIds = lineItems.map(li => li.rampInstanceId).filter(Boolean);\n\n    return lineItems.reduce((result: HeaderProduct[], li) => {\n      // find main term line item\n      if (li.rampInstanceId) {\n        return result;\n      }\n\n      // find current term line item\n      let target: LineItem | undefined = li;\n      while (target && target.endDate && new Date(target.endDate).getTime() <= now) {\n        target = lineItems.find(sub => sub.rampInstanceId === li.id);\n      }\n\n      if (target && target.productId) {\n        return [\n          ...result,\n          {\n            id: target.id,\n            productId: target.productId,\n            name: target.name,\n            configurable: target.properties['#configurable'] === 'true',\n            deleted: target.actionCode === 'DELETE',\n            hasTerm: termParentIds.includes(target.id),\n            qty: target.qty,\n            mrr: this.formatMetric(target.properties.VDM_Total_MRR),\n            nrr: this.formatMetric(target.properties.VDM_Total_NRR),\n          },\n        ];\n      }\n\n      return result;\n    }, []);\n  }\n}\n","<div class=\"flow-info\" *vlLet=\"objectDetails$ | async as details\">\n  <nav class=\"nav-item nav-back\" (click)=\"back()\">\n    <i class=\"nav-icon vl-icon vl-icon-arrow-left\"></i>\n\n    <span> Back to {{ objectName | titlecase }}</span>\n  </nav>\n\n  <ng-container *ngIf=\"isAccountMode\">\n    <span class=\"dot-separator\"></span>\n\n    <span>Account name</span>\n\n    <nav class=\"account-name\" [pTooltip]=\"contextProperties.Name ?? ''\" tooltipPosition=\"bottom\" [showDelay]=\"1000\">\n      <a target=\"_blank\" [href]=\"getSalesforceObjectLink(contextProperties.Id)\">{{ contextProperties.Name }}</a>\n    </nav>\n  </ng-container>\n\n  <ng-container *ngIf=\"isQuoteMode\">\n    <span class=\"dot-separator\"></span>\n\n    <span>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-icon-chevron-down icon-with-margin\"></i>\n      <i *ngIf=\"quoteDetails.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n    </nav>\n\n    <p-overlayPanel styleClass=\"navigation-settings-overlay flow-header-overlay center\" #quoteDetails>\n      <ng-template pTemplate>\n        <div class=\"flow-header-overlay__wrapper\">\n          <h2 class=\"flow-header-overlay__title\">\n            <span>Quote Information</span>\n            <i class=\"vl-icon vl-icon-close close-icon\" (click)=\"quoteDetails.hide()\"></i>\n          </h2>\n\n          <ul class=\"info-list\">\n            <li class=\"info-list__row\">\n              <span>Account Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.accountId)\">\n                <span>{{ details.accountName }}</span>\n              </a>\n            </li>\n            <li class=\"info-list__row\">\n              <span>Opportunity Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.opportunityId)\">\n                <span>{{ details.opportunityName }}</span>\n              </a>\n            </li>\n            <li class=\"info-list__row\">\n              <span>Quote Name:</span>\n              <a target=\"_blank\" [href]=\"getSalesforceObjectLink(details.quoteId)\">\n                <span>{{ details.quoteName }}</span>\n              </a>\n            </li>\n          </ul>\n        </div>\n      </ng-template>\n    </p-overlayPanel>\n  </ng-container>\n</div>\n\n<div class=\"flow-navigation\">\n  <nav class=\"nav-item\" [ngClass]=\"{ active: isCatalogRoute$ | async }\" (click)=\"navigateToCatalog()\">Catalog</nav>\n  <nav class=\"nav-item disabled\" [ngClass]=\"{ active: isConfigurationRoute$ | async }\">Configurator</nav>\n  <ng-container *vlLet=\"products$ | async as products\">\n    <nav class=\"nav-item\" [ngClass]=\"{ active: isCartRoute$ | async }\" (click)=\"navigateToShoppingCart()\">\n      Shopping Cart ({{ products.length }})\n    </nav>\n\n    <nav class=\"nav-popover-toggle active\" (click)=\"cart?.overlayPanel?.toggle($event)\">\n      <i *ngIf=\"!cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n      <i *ngIf=\"cart?.overlayPanel?.overlayVisible\" class=\"vl-icon vl-icon-chevron-up icon-with-margin\"></i>\n    </nav>\n\n    <vl-cart-preview #cart [products]=\"products\"></vl-cart-preview>\n  </ng-container>\n</div>\n\n<div class=\"flow-controls\" *vlLet=\"objectDetails$ | async as details\">\n  <ng-container *vlLet=\"activePriceList$ | async as priceList\">\n    <div>\n      <ng-container *ngIf=\"isAccountMode && assetPriceLists.length > 1; else singlePriceList\">\n        <nav class=\"nav-item\" (click)=\"priceListsOverlay?.toggle($event)\">\n          <span>{{ priceList?.name }}</span>\n          <i *ngIf=\"!priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-down icon-with-margin\"></i>\n          <i *ngIf=\"priceListsOverlay?.overlayVisible\" class=\"vl-icon vl-icon-chevron-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              >{{ option.name }}</span\n            >\n          </ng-template>\n        </p-overlayPanel>\n      </ng-container>\n\n      <ng-template #singlePriceList>\n        <div>{{ priceList?.name }}</div>\n      </ng-template>\n\n      <ng-container *ngIf=\"isQuoteMode\">\n        <div *ngIf=\"contextProperties.StartDate\">{{ contextProperties.StartDate | date: 'MM.dd.yyyy' }}</div>\n      </ng-container>\n    </div>\n  </ng-container>\n\n  <span class=\"metrics\">\n    <div class=\"metrics__row\">\n      <span>MRR:</span>\n      <span class=\"font-semibold\">${{ totalMRR$ | async }}</span>\n    </div>\n    <div class=\"metrics__row\">\n      <span>NRR:</span>\n      <span class=\"font-semibold\">${{ totalNRR$ | async }}</span>\n    </div>\n  </span>\n\n  <ng-container *vlLet=\"isCartRoute$ | async as isCartRoute\">\n    <p-button\n      class=\"doc-gen-button\"\n      icon=\"vl-icon vl-icon-doc-gen\"\n      (onClick)=\"docGenButtonClickHandler(isCartRoute)\"\n    ></p-button>\n\n    <ng-container *vlLet=\"isSaveInProgress$ | async as isSaveInProgress\">\n      <ng-container *vlLet=\"isSubmitInProgress$ | async as isSubmitInProgress\">\n        <p-splitButton\n          *ngIf=\"!isSaveInProgress && !isSubmitInProgress\"\n          label=\"Save to Quote\"\n          (onClick)=\"saveButtonClickHandler(isCartRoute)\"\n          [model]=\"getSplitButtonActions(isCartRoute)\"\n          styleClass=\"p-button-outlined\"\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</div>\n"]}