@veloceapps/sdk 3.0.10 → 3.0.12
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/bundles/veloce-sdk-cms.umd.js +45 -19
- package/bundles/veloce-sdk-cms.umd.js.map +1 -1
- package/bundles/veloce-sdk.umd.js +63 -27
- package/bundles/veloce-sdk.umd.js.map +1 -1
- package/cms/modules/configuration/services/configuration.service.d.ts +3 -1
- package/cms/modules/flow-configuration/services/flow-configuration.service.d.ts +2 -4
- package/esm2015/cms/modules/configuration/services/configuration.service.js +7 -1
- package/esm2015/cms/modules/flow-configuration/services/flow-configuration.service.js +17 -19
- package/esm2015/cms/modules/flow-configuration/services/flow-update.service.js +15 -2
- package/esm2015/src/components/header/cart-overlay/cart-overlay.component.js +31 -26
- package/esm2015/src/components/header/header.component.js +34 -3
- package/esm2015/src/components/header/header.types.js +1 -1
- package/fesm2015/veloce-sdk-cms.js +37 -20
- package/fesm2015/veloce-sdk-cms.js.map +1 -1
- package/fesm2015/veloce-sdk.js +63 -27
- package/fesm2015/veloce-sdk.js.map +1 -1
- package/package.json +1 -1
- package/src/components/header/cart-overlay/cart-overlay.component.d.ts +11 -9
- package/src/components/header/header.component.d.ts +3 -2
- package/src/components/header/header.types.d.ts +9 -0
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { ConfigurationApiService } from '@veloce/api';
|
|
2
|
-
import { LineItem, PricePlanCharge, RuntimeModel } from '@veloce/core';
|
|
2
|
+
import { ConfigurationContext, LineItem, PricePlanCharge, RuntimeModel } from '@veloce/core';
|
|
3
3
|
import { ContextService, QuoteStates } from '@veloce/sdk/runtime';
|
|
4
4
|
import { Dictionary } from 'lodash';
|
|
5
5
|
import { MessageService } from 'primeng/api';
|
|
@@ -27,6 +27,8 @@ export declare class ConfigurationService {
|
|
|
27
27
|
getSnapshot(): LineItem | undefined;
|
|
28
28
|
getRuntimeModel(): RuntimeModel | undefined;
|
|
29
29
|
getRuntimeContext(): RuntimeContext | undefined;
|
|
30
|
+
get contextSnapshot(): ConfigurationContext | null;
|
|
31
|
+
get context$(): Observable<ConfigurationContext | null>;
|
|
30
32
|
get charges$(): Observable<Dictionary<PricePlanCharge>>;
|
|
31
33
|
get chargesSnapshot(): Dictionary<PricePlanCharge>;
|
|
32
34
|
configure(): Observable<LineItem>;
|
|
@@ -20,10 +20,8 @@ export declare class FlowConfigurationService {
|
|
|
20
20
|
calculate(currentState: LineItem[]): void;
|
|
21
21
|
update$(params: FlowUpdateParams): Observable<PriceSummary>;
|
|
22
22
|
update(params: FlowUpdateParams): void;
|
|
23
|
-
delete$(
|
|
24
|
-
|
|
25
|
-
delete(id: string): void;
|
|
26
|
-
deleteAll(): void;
|
|
23
|
+
delete$(ids: string[]): Observable<PriceSummary>;
|
|
24
|
+
delete(ids: string[]): void;
|
|
27
25
|
get(): Observable<LineItem[]>;
|
|
28
26
|
getSnapshot(): LineItem[];
|
|
29
27
|
get charges$(): Observable<Dictionary<PricePlanCharge>>;
|
|
@@ -66,6 +66,12 @@ export class ConfigurationService {
|
|
|
66
66
|
getRuntimeContext() {
|
|
67
67
|
return this.runtimeService.runtimeContext;
|
|
68
68
|
}
|
|
69
|
+
get contextSnapshot() {
|
|
70
|
+
return this.contextService.resolve();
|
|
71
|
+
}
|
|
72
|
+
get context$() {
|
|
73
|
+
return this.contextService.resolve$();
|
|
74
|
+
}
|
|
69
75
|
get charges$() {
|
|
70
76
|
return this.charges.asObservable();
|
|
71
77
|
}
|
|
@@ -140,4 +146,4 @@ ConfigurationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0",
|
|
|
140
146
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: ConfigurationService, decorators: [{
|
|
141
147
|
type: Injectable
|
|
142
148
|
}], ctorParameters: function () { return [{ type: i1.ConfigurationRuntimeService }, { type: i2.ContextService }, { type: i3.ConfigurationApiService }, { type: i4.MessageService }, { type: i5.DialogService }]; } });
|
|
143
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration.service.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/cms/modules/configuration/services/configuration.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ConfigurationApiService } from '@veloce/api';\nimport { ConfirmationComponent } from '@veloce/components';\nimport {\n  ConfigurationContext,\n  ConfigurationMode,\n  ConfigurationRequest,\n  LineItem,\n  PricePlanCharge,\n  RuntimeModel,\n  VlWindow,\n} from '@veloce/core';\nimport { ContextService, QuoteStates, RuntimeStep } from '@veloce/sdk/runtime';\nimport { Dictionary } from 'lodash';\nimport { MessageService } from 'primeng/api';\nimport { DialogService } from 'primeng/dynamicdialog';\nimport { BehaviorSubject, Observable, shareReplay, throwError } from 'rxjs';\nimport { catchError, map, tap } from 'rxjs/operators';\nimport { LineItemWorker } from '../../../types/line-item.types';\nimport { getDefaultLineItem } from '../helpers';\nimport { RuntimeContext } from '../types/configuration.types';\nimport { ConfigurationRuntimeService } from './configuration-runtime.service';\ndeclare const window: VlWindow;\n\n@Injectable()\nexport class ConfigurationService {\n  private mode = ConfigurationMode.SEARCH;\n  private states: QuoteStates = {};\n\n  private lineItem = new BehaviorSubject<LineItem | undefined>(undefined);\n  private charges = new BehaviorSubject<Dictionary<PricePlanCharge>>({});\n\n  constructor(\n    private runtimeService: ConfigurationRuntimeService,\n    private contextService: ContextService,\n    private configurationApiService: ConfigurationApiService,\n    private messageService: MessageService,\n    private dialogService: DialogService,\n  ) {}\n\n  public reset() {\n    this.runtimeService.reset();\n\n    this.states = {};\n    this.lineItem.next(undefined);\n    this.charges.next({});\n  }\n\n  public patch$(lineItem: LineItem): Observable<LineItem> {\n    if (!this.lineItem.value) {\n      return throwError(() => new Error(`Source LineItem not found`));\n    }\n\n    this.states.configurableRamp = new LineItemWorker(this.lineItem.value).replace(lineItem).li;\n    this.states.asset = this.states.configurableRamp\n      ? this.runtimeService.getAsset(this.states.configurableRamp)\n      : undefined;\n\n    return this.configure().pipe(\n      catchError(error => {\n        console.error(error);\n\n        if (!this.runtimeService.uiDefinitionProperties.suppressToastMessages) {\n          this.messageService.add({ severity: 'error', summary: error });\n        }\n\n        // bounce back if configuration call has failed\n        this.lineItem.next(this.lineItem.value ? { ...this.lineItem.value } : undefined);\n\n        return throwError(() => error);\n      }),\n    );\n  }\n\n  public patch(lineItem: LineItem): void {\n    this.patch$(lineItem).subscribe();\n  }\n\n  public updateCurrentStates(update: Partial<QuoteStates>) {\n    this.states = {\n      ...this.states,\n      ...update,\n    };\n  }\n\n  public get(): Observable<LineItem | undefined> {\n    return this.lineItem.asObservable().pipe(shareReplay());\n  }\n\n  public getSnapshot(): LineItem | undefined {\n    return this.lineItem.value ? { ...this.lineItem.value } : undefined;\n  }\n\n  public getRuntimeModel(): RuntimeModel | undefined {\n    return this.runtimeService.runtimeModel;\n  }\n  public getRuntimeContext(): RuntimeContext | undefined {\n    return this.runtimeService.runtimeContext;\n  }\n\n  public get charges$(): Observable<Dictionary<PricePlanCharge>> {\n    return this.charges.asObservable();\n  }\n\n  public get chargesSnapshot(): Dictionary<PricePlanCharge> {\n    return this.charges.value;\n  }\n\n  public configure(): Observable<LineItem> {\n    const runtimeContext = this.getRuntimeContext();\n    const runtimeModel = this.getRuntimeModel();\n    if (!runtimeContext || !runtimeModel) {\n      return throwError(() => new Error('Runtime context/model not initialized'));\n    }\n\n    const uiDefinitionProperties = {\n      ...(runtimeContext.uiDefinition?.properties ?? {}),\n      ...(this.runtimeService.uiDefinitionProperties ?? {}),\n    };\n    const lineItem = this.states.configurableRamp ?? getDefaultLineItem(runtimeContext, uiDefinitionProperties);\n    const configurationRequest = this.createRequest(lineItem);\n    configurationRequest.lineItems = this.states.currentState || [];\n    configurationRequest.asset = this.states.asset;\n    const mainPricingEnabled = runtimeContext.properties?.PricingEnabled;\n    const pricingEnabled = mainPricingEnabled ? mainPricingEnabled === 'true' : uiDefinitionProperties.pricingEnabled;\n    return this.configurationApiService\n      .configureLineItem({ configurationRequest, runtimeModel, pricingEnabled })\n      .pipe(\n        map(({ lineItem, context, charges, deletedLineItems }) => {\n          if (context) {\n            this.contextService.update({ properties: context.properties });\n          }\n          this.charges.next(charges ?? {});\n          if (deletedLineItems?.length) {\n            this.showInactiveProductsConfirmation();\n          }\n          return lineItem;\n        }),\n      )\n      .pipe(\n        tap(lineItem => lineItem && this.lineItem.next(lineItem)),\n        catchError(error =>\n          throwError(() => new Error(error.error?.message || error.message || JSON.stringify(error))),\n        ),\n      );\n  }\n\n  private createRequest(lineItem: LineItem): ConfigurationRequest {\n    return {\n      lineItem,\n      mode: this.mode,\n      step: !this.lineItem.value ? RuntimeStep.START : RuntimeStep.UPDATE,\n      attributeDomainMode: 'ALL',\n      context: this.contextService.resolve() as ConfigurationContext,\n    };\n  }\n\n  private showInactiveProductsConfirmation() {\n    this.dialogService\n      .open(ConfirmationComponent, {\n        dismissableMask: false,\n        closeOnEscape: false,\n        closable: false,\n        showHeader: true,\n        header: `Inactive Products in Quote`,\n        width: '440px',\n        data: {\n          confirmationConfig: {\n            title: ' ',\n            description: 'This quote contains inactive products. Do you want to remove them?',\n            submitBtn: 'Remove products',\n            cancelBtn: 'Back to Quote',\n          },\n        },\n      })\n      .onClose.subscribe(result => {\n        if (!result) {\n          const context = this.contextService.resolve();\n          window['VELO_BACK_FN'].apply(null, [context?.headerId || '']);\n        }\n      });\n  }\n}\n"]}
|
|
149
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"configuration.service.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/cms/modules/configuration/services/configuration.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ConfigurationApiService } from '@veloce/api';\nimport { ConfirmationComponent } from '@veloce/components';\nimport {\n  ConfigurationContext,\n  ConfigurationMode,\n  ConfigurationRequest,\n  LineItem,\n  PricePlanCharge,\n  RuntimeModel,\n  VlWindow,\n} from '@veloce/core';\nimport { ContextService, QuoteStates, RuntimeStep } from '@veloce/sdk/runtime';\nimport { Dictionary } from 'lodash';\nimport { MessageService } from 'primeng/api';\nimport { DialogService } from 'primeng/dynamicdialog';\nimport { BehaviorSubject, Observable, shareReplay, throwError } from 'rxjs';\nimport { catchError, map, tap } from 'rxjs/operators';\nimport { LineItemWorker } from '../../../types/line-item.types';\nimport { getDefaultLineItem } from '../helpers';\nimport { RuntimeContext } from '../types/configuration.types';\nimport { ConfigurationRuntimeService } from './configuration-runtime.service';\ndeclare const window: VlWindow;\n\n@Injectable()\nexport class ConfigurationService {\n  private mode = ConfigurationMode.SEARCH;\n  private states: QuoteStates = {};\n\n  private lineItem = new BehaviorSubject<LineItem | undefined>(undefined);\n  private charges = new BehaviorSubject<Dictionary<PricePlanCharge>>({});\n\n  constructor(\n    private runtimeService: ConfigurationRuntimeService,\n    private contextService: ContextService,\n    private configurationApiService: ConfigurationApiService,\n    private messageService: MessageService,\n    private dialogService: DialogService,\n  ) {}\n\n  public reset() {\n    this.runtimeService.reset();\n\n    this.states = {};\n    this.lineItem.next(undefined);\n    this.charges.next({});\n  }\n\n  public patch$(lineItem: LineItem): Observable<LineItem> {\n    if (!this.lineItem.value) {\n      return throwError(() => new Error(`Source LineItem not found`));\n    }\n\n    this.states.configurableRamp = new LineItemWorker(this.lineItem.value).replace(lineItem).li;\n    this.states.asset = this.states.configurableRamp\n      ? this.runtimeService.getAsset(this.states.configurableRamp)\n      : undefined;\n\n    return this.configure().pipe(\n      catchError(error => {\n        console.error(error);\n\n        if (!this.runtimeService.uiDefinitionProperties.suppressToastMessages) {\n          this.messageService.add({ severity: 'error', summary: error });\n        }\n\n        // bounce back if configuration call has failed\n        this.lineItem.next(this.lineItem.value ? { ...this.lineItem.value } : undefined);\n\n        return throwError(() => error);\n      }),\n    );\n  }\n\n  public patch(lineItem: LineItem): void {\n    this.patch$(lineItem).subscribe();\n  }\n\n  public updateCurrentStates(update: Partial<QuoteStates>) {\n    this.states = {\n      ...this.states,\n      ...update,\n    };\n  }\n\n  public get(): Observable<LineItem | undefined> {\n    return this.lineItem.asObservable().pipe(shareReplay());\n  }\n\n  public getSnapshot(): LineItem | undefined {\n    return this.lineItem.value ? { ...this.lineItem.value } : undefined;\n  }\n\n  public getRuntimeModel(): RuntimeModel | undefined {\n    return this.runtimeService.runtimeModel;\n  }\n\n  public getRuntimeContext(): RuntimeContext | undefined {\n    return this.runtimeService.runtimeContext;\n  }\n\n  public get contextSnapshot(): ConfigurationContext | null {\n    return this.contextService.resolve();\n  }\n\n  public get context$(): Observable<ConfigurationContext | null> {\n    return this.contextService.resolve$();\n  }\n\n  public get charges$(): Observable<Dictionary<PricePlanCharge>> {\n    return this.charges.asObservable();\n  }\n\n  public get chargesSnapshot(): Dictionary<PricePlanCharge> {\n    return this.charges.value;\n  }\n\n  public configure(): Observable<LineItem> {\n    const runtimeContext = this.getRuntimeContext();\n    const runtimeModel = this.getRuntimeModel();\n    if (!runtimeContext || !runtimeModel) {\n      return throwError(() => new Error('Runtime context/model not initialized'));\n    }\n\n    const uiDefinitionProperties = {\n      ...(runtimeContext.uiDefinition?.properties ?? {}),\n      ...(this.runtimeService.uiDefinitionProperties ?? {}),\n    };\n    const lineItem = this.states.configurableRamp ?? getDefaultLineItem(runtimeContext, uiDefinitionProperties);\n    const configurationRequest = this.createRequest(lineItem);\n    configurationRequest.lineItems = this.states.currentState || [];\n    configurationRequest.asset = this.states.asset;\n    const mainPricingEnabled = runtimeContext.properties?.PricingEnabled;\n    const pricingEnabled = mainPricingEnabled ? mainPricingEnabled === 'true' : uiDefinitionProperties.pricingEnabled;\n    return this.configurationApiService\n      .configureLineItem({ configurationRequest, runtimeModel, pricingEnabled })\n      .pipe(\n        map(({ lineItem, context, charges, deletedLineItems }) => {\n          if (context) {\n            this.contextService.update({ properties: context.properties });\n          }\n          this.charges.next(charges ?? {});\n          if (deletedLineItems?.length) {\n            this.showInactiveProductsConfirmation();\n          }\n          return lineItem;\n        }),\n      )\n      .pipe(\n        tap(lineItem => lineItem && this.lineItem.next(lineItem)),\n        catchError(error =>\n          throwError(() => new Error(error.error?.message || error.message || JSON.stringify(error))),\n        ),\n      );\n  }\n\n  private createRequest(lineItem: LineItem): ConfigurationRequest {\n    return {\n      lineItem,\n      mode: this.mode,\n      step: !this.lineItem.value ? RuntimeStep.START : RuntimeStep.UPDATE,\n      attributeDomainMode: 'ALL',\n      context: this.contextService.resolve() as ConfigurationContext,\n    };\n  }\n\n  private showInactiveProductsConfirmation() {\n    this.dialogService\n      .open(ConfirmationComponent, {\n        dismissableMask: false,\n        closeOnEscape: false,\n        closable: false,\n        showHeader: true,\n        header: `Inactive Products in Quote`,\n        width: '440px',\n        data: {\n          confirmationConfig: {\n            title: ' ',\n            description: 'This quote contains inactive products. Do you want to remove them?',\n            submitBtn: 'Remove products',\n            cancelBtn: 'Back to Quote',\n          },\n        },\n      })\n      .onClose.subscribe(result => {\n        if (!result) {\n          const context = this.contextService.resolve();\n          window['VELO_BACK_FN'].apply(null, [context?.headerId || '']);\n        }\n      });\n  }\n}\n"]}
|
|
@@ -26,28 +26,22 @@ export class FlowConfigurationService {
|
|
|
26
26
|
if (context) {
|
|
27
27
|
this.contextService.update({ properties: context.properties });
|
|
28
28
|
}
|
|
29
|
-
}),
|
|
29
|
+
}), this.handleError());
|
|
30
30
|
}
|
|
31
31
|
calculate(currentState) {
|
|
32
32
|
this.calculate$(currentState).subscribe();
|
|
33
33
|
}
|
|
34
34
|
update$(params) {
|
|
35
|
-
return of([]).pipe(map(() => this.lineItems.value.map(li => this.updateService.update(li, params))),
|
|
35
|
+
return of([]).pipe(map(() => this.lineItems.value.map(li => this.updateService.update(li, params))), this.handleError(), switchMap(currentState => this.calculate$(currentState)));
|
|
36
36
|
}
|
|
37
37
|
update(params) {
|
|
38
38
|
this.update$(params).subscribe();
|
|
39
39
|
}
|
|
40
|
-
delete$(
|
|
41
|
-
return of([]).pipe(map(() => this.updateService.delete(this.lineItems.value
|
|
40
|
+
delete$(ids) {
|
|
41
|
+
return of([]).pipe(map(() => ids.reduce((result, id) => this.updateService.delete(result, id), this.lineItems.value)), this.handleError(), switchMap(currentState => this.calculate$(currentState)));
|
|
42
42
|
}
|
|
43
|
-
|
|
44
|
-
|
|
45
|
-
}
|
|
46
|
-
delete(id) {
|
|
47
|
-
this.delete$(id).subscribe();
|
|
48
|
-
}
|
|
49
|
-
deleteAll() {
|
|
50
|
-
this.deleteAll$().subscribe();
|
|
43
|
+
delete(ids) {
|
|
44
|
+
this.delete$(ids).subscribe();
|
|
51
45
|
}
|
|
52
46
|
get() {
|
|
53
47
|
return this.lineItems.asObservable().pipe(shareReplay());
|
|
@@ -67,12 +61,16 @@ export class FlowConfigurationService {
|
|
|
67
61
|
get context$() {
|
|
68
62
|
return this.contextService.resolve$();
|
|
69
63
|
}
|
|
70
|
-
handleError(
|
|
71
|
-
|
|
72
|
-
|
|
73
|
-
|
|
74
|
-
|
|
75
|
-
|
|
64
|
+
handleError() {
|
|
65
|
+
return (source$) => {
|
|
66
|
+
return source$.pipe(catchError(error => {
|
|
67
|
+
console.error(error);
|
|
68
|
+
this.messageService.add({ severity: 'error', summary: error });
|
|
69
|
+
// bounce back if configuration call has failed
|
|
70
|
+
this.lineItems.next(this.lineItems.value.slice());
|
|
71
|
+
return throwError(() => error);
|
|
72
|
+
}));
|
|
73
|
+
};
|
|
76
74
|
}
|
|
77
75
|
}
|
|
78
76
|
FlowConfigurationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: FlowConfigurationService, deps: [{ token: i1.PriceApiService }, { token: i2.ContextService }, { token: i3.MessageService }, { token: i4.FlowUpdateService }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
@@ -80,4 +78,4 @@ FlowConfigurationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0
|
|
|
80
78
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: FlowConfigurationService, decorators: [{
|
|
81
79
|
type: Injectable
|
|
82
80
|
}], ctorParameters: function () { return [{ type: i1.PriceApiService }, { type: i2.ContextService }, { type: i3.MessageService }, { type: i4.FlowUpdateService }]; } });
|
|
83
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
81
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flow-configuration.service.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/cms/modules/flow-configuration/services/flow-configuration.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { PriceApiService } from '@veloce/api';\nimport { ConfigurationContext, LineItem, PricePlanCharge, PriceSummary } from '@veloce/core';\nimport { ContextService } from '@veloce/sdk/runtime';\nimport { Dictionary } from 'lodash';\nimport { MessageService } from 'primeng/api';\nimport { BehaviorSubject, catchError, map, Observable, of, shareReplay, switchMap, tap, throwError } from 'rxjs';\nimport { FlowUpdateParams } from '../types/update.types';\nimport { FlowUpdateService } from './flow-update.service';\n\n@Injectable()\nexport class FlowConfigurationService {\n  private lineItems = new BehaviorSubject<LineItem[]>([]);\n  private charges = new BehaviorSubject<Dictionary<PricePlanCharge>>({});\n\n  constructor(\n    private priceApiService: PriceApiService,\n    private contextService: ContextService,\n    private messageService: MessageService,\n    private updateService: FlowUpdateService,\n  ) {}\n\n  public reset() {\n    this.lineItems.next([]);\n    this.charges.next({});\n  }\n\n  calculate$(currentState: LineItem[]): Observable<PriceSummary> {\n    const context = this.contextService.resolve();\n\n    return this.priceApiService.calculate({ context, currentState }).pipe(\n      tap((result: PriceSummary) => {\n        this.lineItems.next(result.lineItems);\n        this.charges.next(result.charges);\n\n        if (context) {\n          this.contextService.update({ properties: context.properties });\n        }\n      }),\n      this.handleError(),\n    );\n  }\n\n  calculate(currentState: LineItem[]): void {\n    this.calculate$(currentState).subscribe();\n  }\n\n  update$(params: FlowUpdateParams): Observable<PriceSummary> {\n    return of([]).pipe(\n      map(() => this.lineItems.value.map(li => this.updateService.update(li, params))),\n      this.handleError(),\n      switchMap(currentState => this.calculate$(currentState)),\n    );\n  }\n\n  update(params: FlowUpdateParams): void {\n    this.update$(params).subscribe();\n  }\n\n  delete$(ids: string[]): Observable<PriceSummary> {\n    return of([]).pipe(\n      map(() => ids.reduce((result, id) => this.updateService.delete(result, id), this.lineItems.value)),\n      this.handleError(),\n      switchMap(currentState => this.calculate$(currentState)),\n    );\n  }\n\n  delete(ids: string[]): void {\n    this.delete$(ids).subscribe();\n  }\n\n  public get(): Observable<LineItem[]> {\n    return this.lineItems.asObservable().pipe(shareReplay());\n  }\n\n  public getSnapshot(): LineItem[] {\n    return this.lineItems.value.slice();\n  }\n\n  public get charges$(): Observable<Dictionary<PricePlanCharge>> {\n    return this.charges.asObservable();\n  }\n\n  public get chargesSnapshot(): Dictionary<PricePlanCharge> {\n    return this.charges.value;\n  }\n\n  public get contextSnapshot(): ConfigurationContext | null {\n    return this.contextService.resolve();\n  }\n\n  public get context$(): Observable<ConfigurationContext | null> {\n    return this.contextService.resolve$();\n  }\n\n  private handleError() {\n    return <T>(source$: Observable<T>): Observable<T> => {\n      return source$.pipe(\n        catchError(error => {\n          console.error(error);\n          this.messageService.add({ severity: 'error', summary: error });\n\n          // bounce back if configuration call has failed\n          this.lineItems.next(this.lineItems.value.slice());\n\n          return throwError(() => error);\n        }),\n      );\n    };\n  }\n}\n"]}
|
|
@@ -19,7 +19,20 @@ export class FlowUpdateService {
|
|
|
19
19
|
}
|
|
20
20
|
}
|
|
21
21
|
delete(lineItems, id) {
|
|
22
|
-
const
|
|
22
|
+
const idsToRemove = [id];
|
|
23
|
+
const topLevelLineItem = lineItems.find(li => li.id === id);
|
|
24
|
+
if (topLevelLineItem) {
|
|
25
|
+
// find term-related line items (which are only top level)
|
|
26
|
+
// expired term line items won't be deleted
|
|
27
|
+
let foundTermLineItem = topLevelLineItem;
|
|
28
|
+
while (foundTermLineItem) {
|
|
29
|
+
foundTermLineItem = lineItems.find(li => foundTermLineItem && li.rampInstanceId === foundTermLineItem.id);
|
|
30
|
+
if (foundTermLineItem) {
|
|
31
|
+
idsToRemove.push(foundTermLineItem.id);
|
|
32
|
+
}
|
|
33
|
+
}
|
|
34
|
+
}
|
|
35
|
+
const filtered = lineItems.filter(lineItem => !idsToRemove.includes(lineItem.id));
|
|
23
36
|
return filtered.map(lineItem => new LineItemWorker(lineItem).remove(id).li);
|
|
24
37
|
}
|
|
25
38
|
updateQty(rootLineItem, params) {
|
|
@@ -94,4 +107,4 @@ FlowUpdateService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", ve
|
|
|
94
107
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: FlowUpdateService, decorators: [{
|
|
95
108
|
type: Injectable
|
|
96
109
|
}] });
|
|
97
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flow-update.service.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/cms/modules/flow-configuration/services/flow-update.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { LineItem } from '@veloce/core';\nimport moment from 'moment';\nimport { LineItemWorker } from '../../../types/line-item.types';\nimport { findLineItem, findLineItemWithComparator } from '../../../utils/line-item.utils';\nimport { FlowUpdateParams } from '../types/update.types';\n\n@Injectable()\nexport class FlowUpdateService {\n  public update(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    switch (params.attributeType) {\n      case 'QTY':\n        return this.updateQty(rootLineItem, params);\n      case 'EFFECTIVE_START_DATE':\n        return this.updateEffectiveStartDate(rootLineItem, params);\n      case 'END_DATE':\n        return this.updateEndDate(rootLineItem, params);\n      case 'PRICE_ADJUSTMENT':\n        return this.updatePriceAdjustment(rootLineItem, params);\n      default:\n        return rootLineItem;\n    }\n  }\n\n  public delete(lineItems: LineItem[], id: string): LineItem[] {\n    const filtered = lineItems.filter(lineItem => lineItem.id !== id);\n    return filtered.map(lineItem => new LineItemWorker(lineItem).remove(id).li);\n  }\n\n  private updateQty(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    // change quantity allowed only for lineItem\n    if (params.dataType !== 'LINEITEM') {\n      return rootLineItem;\n    }\n\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    if (lineItem.id !== rootLineItem.id) {\n      throw new Error('Quantity can be changed only for root product');\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({ ...lineItem, qty: params.newValue });\n    return updated.li;\n  }\n\n  private updateEffectiveStartDate(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({\n      ...lineItem,\n      effectiveStartDate: moment(params.newValue).format('YYYY-MM-DD'),\n    });\n    return updated.li;\n  }\n\n  private updateEndDate(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({\n      ...lineItem,\n      endDate: moment(params.newValue).format('YYYY-MM-DD'),\n    });\n    return updated.li;\n  }\n\n  private updatePriceAdjustment(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const comparator =\n      params.dataType === 'LINEITEM'\n        ? (li: LineItem) => li.id === params.id\n        : params.dataType === 'CHARGE'\n        ? (li: LineItem) => li.chargeItems.some(({ id }) => id === params.id)\n        : params.dataType === 'GROUP_CHARGE'\n        ? (li: LineItem) => li.chargeGroupItems.some(({ id }) => id === params.id)\n        : undefined;\n\n    if (!comparator) {\n      return rootLineItem;\n    }\n\n    const lineItem = findLineItemWithComparator([rootLineItem], comparator);\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    let updated: LineItem | undefined;\n    if (params.dataType === 'LINEITEM') {\n      updated = {\n        ...lineItem,\n        chargeItems: lineItem.chargeItems.map((chargeItem, index) => {\n          return index === 0 ? { ...chargeItem, priceAdjustment: params.newValue } : chargeItem;\n        }),\n      };\n    } else if (params.dataType === 'CHARGE') {\n      updated = {\n        ...lineItem,\n        chargeItems: lineItem.chargeItems.map(chargeItem => {\n          return chargeItem.id === params.id ? { ...chargeItem, priceAdjustment: params.newValue } : chargeItem;\n        }),\n      };\n    } else if (params.dataType === 'GROUP_CHARGE') {\n      updated = {\n        ...lineItem,\n        chargeGroupItems: lineItem.chargeGroupItems.map(chargeGroupItem => {\n          return chargeGroupItem.id === params.id\n            ? { ...chargeGroupItem, priceAdjustment: params.newValue }\n            : chargeGroupItem;\n        }),\n      };\n    }\n\n    const updatedRootLineItem = updated ? new LineItemWorker(rootLineItem).replace(updated).li : rootLineItem;\n\n    return updatedRootLineItem;\n  }\n}\n"]}
|
|
110
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"flow-update.service.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/cms/modules/flow-configuration/services/flow-update.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { LineItem } from '@veloce/core';\nimport moment from 'moment';\nimport { LineItemWorker } from '../../../types/line-item.types';\nimport { findLineItem, findLineItemWithComparator } from '../../../utils/line-item.utils';\nimport { FlowUpdateParams } from '../types/update.types';\n\n@Injectable()\nexport class FlowUpdateService {\n  public update(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    switch (params.attributeType) {\n      case 'QTY':\n        return this.updateQty(rootLineItem, params);\n      case 'EFFECTIVE_START_DATE':\n        return this.updateEffectiveStartDate(rootLineItem, params);\n      case 'END_DATE':\n        return this.updateEndDate(rootLineItem, params);\n      case 'PRICE_ADJUSTMENT':\n        return this.updatePriceAdjustment(rootLineItem, params);\n      default:\n        return rootLineItem;\n    }\n  }\n\n  public delete(lineItems: LineItem[], id: string): LineItem[] {\n    const idsToRemove: string[] = [id];\n    const topLevelLineItem = lineItems.find(li => li.id === id);\n\n    if (topLevelLineItem) {\n      // find term-related line items (which are only top level)\n      // expired term line items won't be deleted\n      let foundTermLineItem: LineItem | undefined = topLevelLineItem;\n\n      while (foundTermLineItem) {\n        foundTermLineItem = lineItems.find(li => foundTermLineItem && li.rampInstanceId === foundTermLineItem.id);\n        if (foundTermLineItem) {\n          idsToRemove.push(foundTermLineItem.id);\n        }\n      }\n    }\n\n    const filtered = lineItems.filter(lineItem => !idsToRemove.includes(lineItem.id));\n    return filtered.map(lineItem => new LineItemWorker(lineItem).remove(id).li);\n  }\n\n  private updateQty(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    // change quantity allowed only for lineItem\n    if (params.dataType !== 'LINEITEM') {\n      return rootLineItem;\n    }\n\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    if (lineItem.id !== rootLineItem.id) {\n      throw new Error('Quantity can be changed only for root product');\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({ ...lineItem, qty: params.newValue });\n    return updated.li;\n  }\n\n  private updateEffectiveStartDate(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({\n      ...lineItem,\n      effectiveStartDate: moment(params.newValue).format('YYYY-MM-DD'),\n    });\n    return updated.li;\n  }\n\n  private updateEndDate(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const lineItem = findLineItem(params.id, [rootLineItem]);\n\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    const updated = new LineItemWorker(rootLineItem).replace({\n      ...lineItem,\n      endDate: moment(params.newValue).format('YYYY-MM-DD'),\n    });\n    return updated.li;\n  }\n\n  private updatePriceAdjustment(rootLineItem: LineItem, params: FlowUpdateParams): LineItem {\n    const comparator =\n      params.dataType === 'LINEITEM'\n        ? (li: LineItem) => li.id === params.id\n        : params.dataType === 'CHARGE'\n        ? (li: LineItem) => li.chargeItems.some(({ id }) => id === params.id)\n        : params.dataType === 'GROUP_CHARGE'\n        ? (li: LineItem) => li.chargeGroupItems.some(({ id }) => id === params.id)\n        : undefined;\n\n    if (!comparator) {\n      return rootLineItem;\n    }\n\n    const lineItem = findLineItemWithComparator([rootLineItem], comparator);\n    if (!lineItem) {\n      return rootLineItem;\n    }\n\n    let updated: LineItem | undefined;\n    if (params.dataType === 'LINEITEM') {\n      updated = {\n        ...lineItem,\n        chargeItems: lineItem.chargeItems.map((chargeItem, index) => {\n          return index === 0 ? { ...chargeItem, priceAdjustment: params.newValue } : chargeItem;\n        }),\n      };\n    } else if (params.dataType === 'CHARGE') {\n      updated = {\n        ...lineItem,\n        chargeItems: lineItem.chargeItems.map(chargeItem => {\n          return chargeItem.id === params.id ? { ...chargeItem, priceAdjustment: params.newValue } : chargeItem;\n        }),\n      };\n    } else if (params.dataType === 'GROUP_CHARGE') {\n      updated = {\n        ...lineItem,\n        chargeGroupItems: lineItem.chargeGroupItems.map(chargeGroupItem => {\n          return chargeGroupItem.id === params.id\n            ? { ...chargeGroupItem, priceAdjustment: params.newValue }\n            : chargeGroupItem;\n        }),\n      };\n    }\n\n    const updatedRootLineItem = updated ? new LineItemWorker(rootLineItem).replace(updated).li : rootLineItem;\n\n    return updatedRootLineItem;\n  }\n}\n"]}
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { ChangeDetectionStrategy, Component, ViewChild } from '@angular/core';
|
|
1
|
+
import { ChangeDetectionStrategy, Component, Input, ViewChild, } from '@angular/core';
|
|
2
2
|
import { FormControl, FormGroup } from '@angular/forms';
|
|
3
3
|
import { OverlayPanel } from 'primeng/overlaypanel';
|
|
4
4
|
import { BehaviorSubject, catchError, combineLatest, distinctUntilChanged, filter, map, noop, of, Subject, takeUntil, tap, } from 'rxjs';
|
|
@@ -18,10 +18,10 @@ export class HeaderCartOverlayComponent {
|
|
|
18
18
|
this.baseHttpService = baseHttpService;
|
|
19
19
|
this.flowConfiguration = flowConfiguration;
|
|
20
20
|
this.routerService = routerService;
|
|
21
|
+
this.form = new FormGroup({});
|
|
22
|
+
this.shouldUpdate$ = new BehaviorSubject(true);
|
|
21
23
|
this.imagesMap$ = new BehaviorSubject({});
|
|
22
24
|
this.destroyed$ = new Subject();
|
|
23
|
-
this.form = new FormGroup({});
|
|
24
|
-
this.lineItems$ = this.flowConfiguration.get();
|
|
25
25
|
this.readonlyProductId$ = combineLatest([
|
|
26
26
|
this.routerService.route$,
|
|
27
27
|
this.routerService.isConfigurationRoute$(),
|
|
@@ -32,10 +32,15 @@ export class HeaderCartOverlayComponent {
|
|
|
32
32
|
return route.queryParams.productId;
|
|
33
33
|
}));
|
|
34
34
|
}
|
|
35
|
+
ngOnChanges(changes) {
|
|
36
|
+
if (changes.products) {
|
|
37
|
+
this.shouldUpdate$.next(true);
|
|
38
|
+
}
|
|
39
|
+
}
|
|
35
40
|
ngAfterViewInit() {
|
|
36
|
-
combineLatest([this.
|
|
41
|
+
combineLatest([this.readonlyProductId$, this.shouldUpdate$, this.overlayPanel.onShow])
|
|
37
42
|
.pipe(filter(() => this.overlayPanel.overlayVisible), takeUntil(this.destroyed$))
|
|
38
|
-
.subscribe(([
|
|
43
|
+
.subscribe(([readonlyProductId]) => this.updateControls(this.products, readonlyProductId));
|
|
39
44
|
}
|
|
40
45
|
ngOnDestroy() {
|
|
41
46
|
this.destroyed$.next();
|
|
@@ -48,30 +53,28 @@ export class HeaderCartOverlayComponent {
|
|
|
48
53
|
}
|
|
49
54
|
return this.imagesMap$.pipe(map(imagesMap => imagesMap[productId]), distinctUntilChanged());
|
|
50
55
|
}
|
|
51
|
-
isConfigurable(lineItem) {
|
|
52
|
-
return lineItem.properties['#configurable'] === 'true';
|
|
53
|
-
}
|
|
54
56
|
navigateToProductConfiguration(productId) {
|
|
55
57
|
this.overlayPanel.hide();
|
|
56
58
|
this.routerService.navigateToProductConfiguration(productId);
|
|
57
59
|
}
|
|
58
|
-
controlBlurHandler(
|
|
59
|
-
const control = this.form.controls[
|
|
60
|
+
controlBlurHandler(product) {
|
|
61
|
+
const control = this.form.controls[product.id];
|
|
60
62
|
if (control.invalid) {
|
|
61
63
|
return;
|
|
62
64
|
}
|
|
63
65
|
this.flowConfiguration.update({
|
|
64
66
|
dataType: 'LINEITEM',
|
|
65
67
|
attributeType: 'QTY',
|
|
66
|
-
id:
|
|
68
|
+
id: product.id,
|
|
67
69
|
newValue: control.value,
|
|
68
70
|
});
|
|
69
71
|
}
|
|
70
|
-
deleteHandler(
|
|
71
|
-
this.flowConfiguration.delete(
|
|
72
|
+
deleteHandler(product) {
|
|
73
|
+
this.flowConfiguration.delete([product.id]);
|
|
72
74
|
}
|
|
73
75
|
deleteAllHandler() {
|
|
74
|
-
this.
|
|
76
|
+
const productIds = this.products.map(product => product.id);
|
|
77
|
+
this.flowConfiguration.delete(productIds);
|
|
75
78
|
}
|
|
76
79
|
fetchProductImage(productId) {
|
|
77
80
|
this.baseHttpService
|
|
@@ -84,25 +87,25 @@ export class HeaderCartOverlayComponent {
|
|
|
84
87
|
.pipe(map(file => URL.createObjectURL(file)), catchError(() => of('')), tap(url => this.imagesMap$.next(Object.assign(Object.assign({}, this.imagesMap$.value), { [productId]: url }))), takeUntil(this.destroyed$))
|
|
85
88
|
.subscribe();
|
|
86
89
|
}
|
|
87
|
-
updateControls(
|
|
90
|
+
updateControls(products, readonlyProductId) {
|
|
88
91
|
const ids = [];
|
|
89
|
-
|
|
92
|
+
products.forEach(item => {
|
|
90
93
|
var _a;
|
|
91
|
-
if (!
|
|
94
|
+
if (!item.id) {
|
|
92
95
|
return;
|
|
93
96
|
}
|
|
94
|
-
ids.push(
|
|
95
|
-
if (!this.form.controls[
|
|
96
|
-
this.form.addControl(
|
|
97
|
+
ids.push(item.id);
|
|
98
|
+
if (!this.form.controls[item.id]) {
|
|
99
|
+
this.form.addControl(item.id, new FormControl(item.qty, []));
|
|
97
100
|
}
|
|
98
101
|
else {
|
|
99
|
-
(_a = this.form.controls[
|
|
102
|
+
(_a = this.form.controls[item.id]) === null || _a === void 0 ? void 0 : _a.setValue(item.qty);
|
|
100
103
|
}
|
|
101
|
-
if (
|
|
102
|
-
this.form.controls[
|
|
104
|
+
if (item.productId === readonlyProductId) {
|
|
105
|
+
this.form.controls[item.id].disable();
|
|
103
106
|
}
|
|
104
107
|
else {
|
|
105
|
-
this.form.controls[
|
|
108
|
+
this.form.controls[item.id].enable();
|
|
106
109
|
}
|
|
107
110
|
});
|
|
108
111
|
const removedIds = Object.keys(this.form.controls).filter(id => !ids.includes(id));
|
|
@@ -110,7 +113,7 @@ export class HeaderCartOverlayComponent {
|
|
|
110
113
|
}
|
|
111
114
|
}
|
|
112
115
|
HeaderCartOverlayComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: HeaderCartOverlayComponent, deps: [{ token: i1.BaseHttpService }, { token: i2.FlowConfigurationService }, { token: i3.FlowRouterService }], target: i0.ɵɵFactoryTarget.Component });
|
|
113
|
-
HeaderCartOverlayComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", type: HeaderCartOverlayComponent, selector: "vl-header-cart-overlay", viewQueries: [{ propertyName: "overlayPanel", first: true, predicate: OverlayPanel, descendants: 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=\"
|
|
116
|
+
HeaderCartOverlayComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", type: HeaderCartOverlayComponent, selector: "vl-header-cart-overlay", 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 *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 class=\"product__info\">\n <div class=\"product__image-wrapper\">\n <div\n *ngIf=\"getImageUrl(product.productId) | async as imageUrl; else noImage\"\n class=\"product__image\"\n [ngStyle]=\"{ 'background-image': 'url(' + imageUrl + ')' }\"\n ></div>\n </div>\n <div class=\"flex flex-column justify-content-center\">\n <div class=\"word-break\">{{ product.name }}</div>\n <div>\n <p-button\n label=\"Configure\"\n [disabled]=\"!product.configurable || readonlyProductId === product.productId\"\n styleClass=\"p-button-link p-button-sm pl-0\"\n (click)=\"navigateToProductConfiguration(product.productId)\"\n ></p-button>\n <p-button\n label=\"Delete\"\n [disabled]=\"readonlyProductId === product.productId\"\n styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n (click)=\"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\">$0.00</span>\n <span class=\"text-right pt-3\">$0.00</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]=\"readonlyProductId\"\n (click)=\"deleteAllHandler()\"\n ></p-button>\n </div>\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)}.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}\n"], components: [{ type: i4.OverlayPanel, selector: "p-overlayPanel", inputs: ["dismissable", "showCloseIcon", "style", "styleClass", "appendTo", "autoZIndex", "ariaCloseLabel", "baseZIndex", "focusOnShow", "showTransitionOptions", "hideTransitionOptions"], outputs: ["onShow", "onHide"] }, { type: i5.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i6.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: i7.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i8.LetDirective, selector: "[vlLet]", inputs: ["vlLet"] }, { type: i9.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i9.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i9.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { type: i10.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i10.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i10.RequiredValidator, selector: ":not([type=checkbox])[required][formControlName],:not([type=checkbox])[required][formControl],:not([type=checkbox])[required][ngModel]", inputs: ["required"] }], pipes: { "async": i9.AsyncPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
|
114
117
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: HeaderCartOverlayComponent, decorators: [{
|
|
115
118
|
type: Component,
|
|
116
119
|
args: [{
|
|
@@ -122,5 +125,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
|
|
|
122
125
|
}], ctorParameters: function () { return [{ type: i1.BaseHttpService }, { type: i2.FlowConfigurationService }, { type: i3.FlowRouterService }]; }, propDecorators: { overlayPanel: [{
|
|
123
126
|
type: ViewChild,
|
|
124
127
|
args: [OverlayPanel]
|
|
128
|
+
}], products: [{
|
|
129
|
+
type: Input
|
|
125
130
|
}] } });
|
|
126
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-overlay.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-overlay.component.ts","../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-overlay.component.html"],"names":[],"mappings":"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","sourcesContent":["import { AfterViewInit, ChangeDetectionStrategy, Component, OnDestroy, ViewChild } from '@angular/core';\nimport { FormControl, FormGroup } from '@angular/forms';\nimport { BaseHttpService, LineItem } from '@veloce/core';\nimport { FlowConfigurationService } from '@veloce/sdk/cms';\nimport { Dictionary } from 'lodash';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport {\n  BehaviorSubject,\n  catchError,\n  combineLatest,\n  distinctUntilChanged,\n  filter,\n  map,\n  noop,\n  Observable,\n  of,\n  Subject,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { FlowRouterService } from '../../../services/flow-router.service';\n\n@Component({\n  selector: 'vl-header-cart-overlay',\n  templateUrl: './cart-overlay.component.html',\n  styleUrls: ['./cart-overlay.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class HeaderCartOverlayComponent implements AfterViewInit, OnDestroy {\n  @ViewChild(OverlayPanel) overlayPanel!: OverlayPanel;\n\n  public lineItems$: Observable<LineItem[]>;\n  public readonlyProductId$: Observable<string | null>;\n\n  private imagesMap$ = new BehaviorSubject<Dictionary<string>>({});\n  private destroyed$ = new Subject<void>();\n\n  public form = new FormGroup({});\n\n  constructor(\n    private baseHttpService: BaseHttpService,\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\n  ) {\n    this.lineItems$ = this.flowConfiguration.get();\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\n  ngAfterViewInit() {\n    combineLatest([this.lineItems$, this.readonlyProductId$, this.overlayPanel.onShow])\n      .pipe(\n        filter(() => this.overlayPanel.overlayVisible),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe(([lineItems, readonlyProductId]) => this.updateControls(lineItems, readonlyProductId));\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public getImageUrl(productId: string): Observable<string | undefined> {\n    if (this.imagesMap$.value[productId] == null) {\n      this.imagesMap$.next({ ...this.imagesMap$.value, [productId]: '' });\n      this.fetchProductImage(productId);\n    }\n\n    return this.imagesMap$.pipe(\n      map(imagesMap => imagesMap[productId]),\n      distinctUntilChanged(),\n    );\n  }\n\n  public isConfigurable(lineItem: LineItem): boolean {\n    return lineItem.properties['#configurable'] === 'true';\n  }\n\n  public navigateToProductConfiguration(productId: string) {\n    this.overlayPanel.hide();\n    this.routerService.navigateToProductConfiguration(productId);\n  }\n\n  public controlBlurHandler(lineItem: LineItem): void {\n    const control = this.form.controls[lineItem.id];\n    if (control.invalid) {\n      return;\n    }\n\n    this.flowConfiguration.update({\n      dataType: 'LINEITEM',\n      attributeType: 'QTY',\n      id: lineItem.id,\n      newValue: control.value,\n    });\n  }\n\n  public deleteHandler(lineItem: LineItem): void {\n    this.flowConfiguration.delete(lineItem.id);\n  }\n\n  public deleteAllHandler(): void {\n    this.flowConfiguration.deleteAll();\n  }\n\n  private fetchProductImage(productId: string): void {\n    this.baseHttpService\n      .api<Blob>({\n        url: `/products/${productId}/image`,\n        method: 'get',\n        responseType: 'blob',\n        errorHandler: noop,\n      })\n      .pipe(\n        map(file => URL.createObjectURL(file)),\n        catchError(() => of('')),\n        tap(url => this.imagesMap$.next({ ...this.imagesMap$.value, [productId]: url })),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n  }\n\n  private updateControls(lineItems: LineItem[], readonlyProductId: string | null): void {\n    const ids: string[] = [];\n\n    lineItems.forEach(li => {\n      if (!li.id) {\n        return;\n      }\n\n      ids.push(li.id);\n\n      if (!this.form.controls[li.id]) {\n        this.form.addControl(li.id, new FormControl(li.qty, []));\n      } else {\n        this.form.controls[li.id]?.setValue(li.qty);\n      }\n\n      if (li.productId === readonlyProductId) {\n        this.form.controls[li.id].disable();\n      } else {\n        this.form.controls[li.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  }\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=\"lineItems$ | async as lineItems\">\n      <ng-container *ngIf=\"lineItems.length > 0; else empty\">\n        <ng-container *vlLet=\"readonlyProductId$ | async as readonlyProductId\">\n          <h2 class=\"flow-header-overlay__title\">\n            <span>Products ({{ lineItems.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 lineItem of lineItems\">\n              <div class=\"product__info\">\n                <div class=\"product__image-wrapper\">\n                  <div\n                    *ngIf=\"getImageUrl(lineItem.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\">{{ lineItem.name }}</div>\n                  <div>\n                    <p-button\n                      label=\"Configure\"\n                      [disabled]=\"!isConfigurable(lineItem) || readonlyProductId === lineItem.productId\"\n                      styleClass=\"p-button-link p-button-sm pl-0\"\n                      (click)=\"navigateToProductConfiguration(lineItem.productId)\"\n                    ></p-button>\n                    <p-button\n                      label=\"Delete\"\n                      [disabled]=\"readonlyProductId === lineItem.productId\"\n                      styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n                      (click)=\"deleteHandler(lineItem)\"\n                    ></p-button>\n                  </div>\n                </div>\n              </div>\n\n              <p-inputNumber\n                *ngIf=\"form.controls[lineItem.id] as control\"\n                class=\"qty-control\"\n                [formControl]=\"$any(control)\"\n                [min]=\"1\"\n                [required]=\"true\"\n                (onBlur)=\"controlBlurHandler(lineItem)\"\n              ></p-inputNumber>\n              <span class=\"text-right pt-3\">$0.00</span>\n              <span class=\"text-right pt-3\">$0.00</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]=\"readonlyProductId\"\n              (click)=\"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"]}
|
|
131
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"cart-overlay.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-overlay.component.ts","../../../../../../../../libs/sdk/src/components/header/cart-overlay/cart-overlay.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 { BaseHttpService } from '@veloce/core';\nimport { FlowConfigurationService } from '@veloce/sdk/cms';\nimport { Dictionary } from 'lodash';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport {\n  BehaviorSubject,\n  catchError,\n  combineLatest,\n  distinctUntilChanged,\n  filter,\n  map,\n  noop,\n  Observable,\n  of,\n  Subject,\n  takeUntil,\n  tap,\n} from 'rxjs';\nimport { FlowRouterService } from '../../../services/flow-router.service';\nimport { HeaderProduct } from '../header.types';\n\n@Component({\n  selector: 'vl-header-cart-overlay',\n  templateUrl: './cart-overlay.component.html',\n  styleUrls: ['./cart-overlay.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class HeaderCartOverlayComponent 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 imagesMap$ = new BehaviorSubject<Dictionary<string>>({});\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    private baseHttpService: BaseHttpService,\n    private flowConfiguration: FlowConfigurationService,\n    private routerService: FlowRouterService,\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\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes.products) {\n      this.shouldUpdate$.next(true);\n    }\n  }\n\n  ngAfterViewInit() {\n    combineLatest([this.readonlyProductId$, this.shouldUpdate$, this.overlayPanel.onShow])\n      .pipe(\n        filter(() => this.overlayPanel.overlayVisible),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe(([readonlyProductId]) => this.updateControls(this.products, readonlyProductId));\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public getImageUrl(productId: string): Observable<string | undefined> {\n    if (this.imagesMap$.value[productId] == null) {\n      this.imagesMap$.next({ ...this.imagesMap$.value, [productId]: '' });\n      this.fetchProductImage(productId);\n    }\n\n    return this.imagesMap$.pipe(\n      map(imagesMap => imagesMap[productId]),\n      distinctUntilChanged(),\n    );\n  }\n\n  public navigateToProductConfiguration(productId: string) {\n    this.overlayPanel.hide();\n    this.routerService.navigateToProductConfiguration(productId);\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      dataType: 'LINEITEM',\n      attributeType: 'QTY',\n      id: product.id,\n      newValue: control.value,\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 fetchProductImage(productId: string): void {\n    this.baseHttpService\n      .api<Blob>({\n        url: `/products/${productId}/image`,\n        method: 'get',\n        responseType: 'blob',\n        errorHandler: noop,\n      })\n      .pipe(\n        map(file => URL.createObjectURL(file)),\n        catchError(() => of('')),\n        tap(url => this.imagesMap$.next({ ...this.imagesMap$.value, [productId]: url })),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n  }\n\n  private updateControls(products: HeaderProduct[], 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 (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  }\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 *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 class=\"product__info\">\n              <div class=\"product__image-wrapper\">\n                <div\n                  *ngIf=\"getImageUrl(product.productId) | async as imageUrl; else noImage\"\n                  class=\"product__image\"\n                  [ngStyle]=\"{ 'background-image': 'url(' + imageUrl + ')' }\"\n                ></div>\n              </div>\n              <div class=\"flex flex-column justify-content-center\">\n                <div class=\"word-break\">{{ product.name }}</div>\n                <div>\n                  <p-button\n                    label=\"Configure\"\n                    [disabled]=\"!product.configurable || readonlyProductId === product.productId\"\n                    styleClass=\"p-button-link p-button-sm pl-0\"\n                    (click)=\"navigateToProductConfiguration(product.productId)\"\n                  ></p-button>\n                  <p-button\n                    label=\"Delete\"\n                    [disabled]=\"readonlyProductId === product.productId\"\n                    styleClass=\"p-button-link p-button-sm p-button-danger pl-0 pr-0\"\n                    (click)=\"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\">$0.00</span>\n            <span class=\"text-right pt-3\">$0.00</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]=\"readonlyProductId\"\n            (click)=\"deleteAllHandler()\"\n          ></p-button>\n        </div>\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"]}
|