@veloceapps/sdk 5.0.13-8 → 6.0.0-0
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/veloceapps-sdk-core.umd.js +155 -32
- package/bundles/veloceapps-sdk-core.umd.js.map +1 -1
- package/bundles/veloceapps-sdk.umd.js +13 -9
- package/bundles/veloceapps-sdk.umd.js.map +1 -1
- package/core/services/index.d.ts +1 -0
- package/core/services/metric-calculation/metric-calculation.service.d.ts +21 -0
- package/core/services/metric-calculation/metric-calculation.types.d.ts +1 -0
- package/core/services/metric-calculation/metric-calculation.utils.d.ts +5 -0
- package/esm2015/core/core.module.js +4 -4
- package/esm2015/core/services/index.js +2 -1
- package/esm2015/core/services/metric-calculation/metric-calculation.service.js +74 -0
- package/esm2015/core/services/metric-calculation/metric-calculation.types.js +2 -0
- package/esm2015/core/services/metric-calculation/metric-calculation.utils.js +42 -0
- package/esm2015/src/components/header/metrics/metrics.component.js +11 -7
- package/esm2015/src/components/header/metrics/metrics.definitions.js +6 -6
- package/fesm2015/veloceapps-sdk-core.js +136 -27
- package/fesm2015/veloceapps-sdk-core.js.map +1 -1
- package/fesm2015/veloceapps-sdk.js +13 -9
- package/fesm2015/veloceapps-sdk.js.map +1 -1
- package/package.json +5 -5
- package/src/components/header/metrics/metrics.component.d.ts +3 -2
package/core/services/index.d.ts
CHANGED
@@ -0,0 +1,21 @@
|
|
1
|
+
import { ConfigurationSettingsApiService } from '@veloceapps/api';
|
2
|
+
import { Observable } from 'rxjs';
|
3
|
+
import { QuoteDraftService } from '../quote-draft.service';
|
4
|
+
import * as i0 from "@angular/core";
|
5
|
+
export declare class MetricsCalculationService {
|
6
|
+
private quoteDraftService;
|
7
|
+
private settingsService;
|
8
|
+
get onMetricsUpdate$(): Observable<void>;
|
9
|
+
private metricsUpdated$;
|
10
|
+
private quoteMetricsSettings;
|
11
|
+
private metricsCalculationMethodMap;
|
12
|
+
private metricsData;
|
13
|
+
constructor(quoteDraftService: QuoteDraftService, settingsService: ConfigurationSettingsApiService);
|
14
|
+
getMetricValue(metric: string): number;
|
15
|
+
private updateMetrics;
|
16
|
+
private calculateMetric;
|
17
|
+
private buildMetricsCalculationMethods;
|
18
|
+
private collectMetricKeys;
|
19
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<MetricsCalculationService, never>;
|
20
|
+
static ɵprov: i0.ɵɵInjectableDeclaration<MetricsCalculationService>;
|
21
|
+
}
|
@@ -0,0 +1 @@
|
|
1
|
+
export declare type MetricCalculationMethod = 'last' | 'first' | 'avg' | 'sum';
|
@@ -0,0 +1,5 @@
|
|
1
|
+
import { LineItem } from '@veloceapps/core';
|
2
|
+
import { MetricCalculationMethod } from './metric-calculation.types';
|
3
|
+
export declare function calculateMetricByMethod(lineItems: LineItem[], metric: string, method: MetricCalculationMethod): number;
|
4
|
+
export declare function getLineItemsByMethod(lineItems: LineItem[], method: MetricCalculationMethod): LineItem[][];
|
5
|
+
export declare function getDateValue(date: string): number;
|
@@ -1,18 +1,18 @@
|
|
1
1
|
import { NgModule } from '@angular/core';
|
2
2
|
import { FlowConfigurationModule } from './modules';
|
3
3
|
import { ConfigurationModule } from './modules/configuration/configuration.module';
|
4
|
-
import { ContextService, ProductImagesService, QuoteDraftService } from './services';
|
4
|
+
import { ContextService, MetricsCalculationService, ProductImagesService, QuoteDraftService } from './services';
|
5
5
|
import * as i0 from "@angular/core";
|
6
6
|
export class SdkCoreModule {
|
7
7
|
}
|
8
8
|
SdkCoreModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: SdkCoreModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
|
9
9
|
SdkCoreModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: SdkCoreModule, imports: [ConfigurationModule, FlowConfigurationModule] });
|
10
|
-
SdkCoreModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: SdkCoreModule, providers: [ContextService, QuoteDraftService, ProductImagesService], imports: [[ConfigurationModule, FlowConfigurationModule]] });
|
10
|
+
SdkCoreModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: SdkCoreModule, providers: [ContextService, QuoteDraftService, ProductImagesService, MetricsCalculationService], imports: [[ConfigurationModule, FlowConfigurationModule]] });
|
11
11
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: SdkCoreModule, decorators: [{
|
12
12
|
type: NgModule,
|
13
13
|
args: [{
|
14
14
|
imports: [ConfigurationModule, FlowConfigurationModule],
|
15
|
-
providers: [ContextService, QuoteDraftService, ProductImagesService],
|
15
|
+
providers: [ContextService, QuoteDraftService, ProductImagesService, MetricsCalculationService],
|
16
16
|
}]
|
17
17
|
}] });
|
18
|
-
//# sourceMappingURL=data:application/json;base64,
|
18
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY29yZS5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi9saWJzL3Nkay9jb3JlL2NvcmUubW9kdWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxRQUFRLEVBQUUsTUFBTSxlQUFlLENBQUM7QUFDekMsT0FBTyxFQUFFLHVCQUF1QixFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQ3BELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLDhDQUE4QyxDQUFDO0FBQ25GLE9BQU8sRUFBRSxjQUFjLEVBQUUseUJBQXlCLEVBQUUsb0JBQW9CLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSxZQUFZLENBQUM7O0FBTWhILE1BQU0sT0FBTyxhQUFhOzsyR0FBYixhQUFhOzRHQUFiLGFBQWEsWUFIZCxtQkFBbUIsRUFBRSx1QkFBdUI7NEdBRzNDLGFBQWEsYUFGYixDQUFDLGNBQWMsRUFBRSxpQkFBaUIsRUFBRSxvQkFBb0IsRUFBRSx5QkFBeUIsQ0FBQyxZQUR0RixDQUFDLG1CQUFtQixFQUFFLHVCQUF1QixDQUFDOzRGQUc1QyxhQUFhO2tCQUp6QixRQUFRO21CQUFDO29CQUNSLE9BQU8sRUFBRSxDQUFDLG1CQUFtQixFQUFFLHVCQUF1QixDQUFDO29CQUN2RCxTQUFTLEVBQUUsQ0FBQyxjQUFjLEVBQUUsaUJBQWlCLEVBQUUsb0JBQW9CLEVBQUUseUJBQXlCLENBQUM7aUJBQ2hHIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZsb3dDb25maWd1cmF0aW9uTW9kdWxlIH0gZnJvbSAnLi9tb2R1bGVzJztcbmltcG9ydCB7IENvbmZpZ3VyYXRpb25Nb2R1bGUgfSBmcm9tICcuL21vZHVsZXMvY29uZmlndXJhdGlvbi9jb25maWd1cmF0aW9uLm1vZHVsZSc7XG5pbXBvcnQgeyBDb250ZXh0U2VydmljZSwgTWV0cmljc0NhbGN1bGF0aW9uU2VydmljZSwgUHJvZHVjdEltYWdlc1NlcnZpY2UsIFF1b3RlRHJhZnRTZXJ2aWNlIH0gZnJvbSAnLi9zZXJ2aWNlcyc7XG5cbkBOZ01vZHVsZSh7XG4gIGltcG9ydHM6IFtDb25maWd1cmF0aW9uTW9kdWxlLCBGbG93Q29uZmlndXJhdGlvbk1vZHVsZV0sXG4gIHByb3ZpZGVyczogW0NvbnRleHRTZXJ2aWNlLCBRdW90ZURyYWZ0U2VydmljZSwgUHJvZHVjdEltYWdlc1NlcnZpY2UsIE1ldHJpY3NDYWxjdWxhdGlvblNlcnZpY2VdLFxufSlcbmV4cG9ydCBjbGFzcyBTZGtDb3JlTW9kdWxlIHt9XG4iXX0=
|
@@ -1,4 +1,5 @@
|
|
1
1
|
export * from './context.service';
|
2
|
+
export * from './metric-calculation/metric-calculation.service';
|
2
3
|
export * from './product-images.service';
|
3
4
|
export * from './quote-draft.service';
|
4
|
-
//# sourceMappingURL=data:application/json;base64,
|
5
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9saWJzL3Nkay9jb3JlL3NlcnZpY2VzL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLGNBQWMsbUJBQW1CLENBQUM7QUFDbEMsY0FBYyxpREFBaUQsQ0FBQztBQUNoRSxjQUFjLDBCQUEwQixDQUFDO0FBQ3pDLGNBQWMsdUJBQXVCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgKiBmcm9tICcuL2NvbnRleHQuc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL21ldHJpYy1jYWxjdWxhdGlvbi9tZXRyaWMtY2FsY3VsYXRpb24uc2VydmljZSc7XG5leHBvcnQgKiBmcm9tICcuL3Byb2R1Y3QtaW1hZ2VzLnNlcnZpY2UnO1xuZXhwb3J0ICogZnJvbSAnLi9xdW90ZS1kcmFmdC5zZXJ2aWNlJztcbiJdfQ==
|
@@ -0,0 +1,74 @@
|
|
1
|
+
import { Injectable } from '@angular/core';
|
2
|
+
import { uniq } from 'lodash';
|
3
|
+
import { Subject, combineLatest, take } from 'rxjs';
|
4
|
+
import { calculateMetricByMethod } from './metric-calculation.utils';
|
5
|
+
import * as i0 from "@angular/core";
|
6
|
+
import * as i1 from "../quote-draft.service";
|
7
|
+
import * as i2 from "@veloceapps/api";
|
8
|
+
export class MetricsCalculationService {
|
9
|
+
constructor(quoteDraftService, settingsService) {
|
10
|
+
this.quoteDraftService = quoteDraftService;
|
11
|
+
this.settingsService = settingsService;
|
12
|
+
this.metricsUpdated$ = new Subject();
|
13
|
+
this.quoteMetricsSettings = {};
|
14
|
+
this.metricsCalculationMethodMap = {};
|
15
|
+
this.metricsData = {};
|
16
|
+
combineLatest([
|
17
|
+
this.quoteDraftService.currentState$,
|
18
|
+
this.settingsService.fetchSetting('QUOTE_LEVEL_METRIC_CALCULATION_METHOD').pipe(take(1)),
|
19
|
+
]).subscribe(([lineItems, setting]) => {
|
20
|
+
let settingsData = {};
|
21
|
+
try {
|
22
|
+
settingsData = JSON.parse((setting === null || setting === void 0 ? void 0 : setting.value) || '{}');
|
23
|
+
}
|
24
|
+
catch (error) {
|
25
|
+
settingsData = {};
|
26
|
+
}
|
27
|
+
this.quoteMetricsSettings = settingsData;
|
28
|
+
this.updateMetrics(lineItems);
|
29
|
+
});
|
30
|
+
}
|
31
|
+
get onMetricsUpdate$() {
|
32
|
+
return this.metricsUpdated$.asObservable();
|
33
|
+
}
|
34
|
+
getMetricValue(metric) {
|
35
|
+
return this.metricsData[metric] || 0;
|
36
|
+
}
|
37
|
+
updateMetrics(lineItems) {
|
38
|
+
const metricKeys = this.collectMetricKeys(lineItems).filter(key => !key.includes('Effective_'));
|
39
|
+
this.metricsCalculationMethodMap = this.buildMetricsCalculationMethods(metricKeys, this.metricsCalculationMethodMap);
|
40
|
+
this.metricsData = metricKeys.reduce((acc, key) => (Object.assign(Object.assign({}, acc), { [key]: this.calculateMetric(lineItems, key) })), {});
|
41
|
+
this.metricsUpdated$.next();
|
42
|
+
}
|
43
|
+
calculateMetric(lineItems, metric) {
|
44
|
+
return calculateMetricByMethod(lineItems, metric, this.metricsCalculationMethodMap[metric] || 'sum');
|
45
|
+
}
|
46
|
+
buildMetricsCalculationMethods(metricKeys, initial) {
|
47
|
+
return metricKeys.reduce((acc, name) => {
|
48
|
+
if (acc[name]) {
|
49
|
+
return acc;
|
50
|
+
}
|
51
|
+
acc = Object.assign(Object.assign({}, acc), { [name]: 'sum' });
|
52
|
+
const settingKey = name.replace(/VDM_|Total_/g, '');
|
53
|
+
if (this.quoteMetricsSettings[settingKey]) {
|
54
|
+
acc = Object.assign(Object.assign({}, acc), { [name]: this.quoteMetricsSettings[settingKey] });
|
55
|
+
}
|
56
|
+
return acc;
|
57
|
+
}, initial);
|
58
|
+
}
|
59
|
+
collectMetricKeys(lineItems) {
|
60
|
+
const keys = [];
|
61
|
+
lineItems.forEach(lineItem => {
|
62
|
+
keys.push(...Object.keys(lineItem.totalMetrics || {}));
|
63
|
+
keys.push(...this.collectMetricKeys(lineItem.lineItems));
|
64
|
+
});
|
65
|
+
return uniq(keys);
|
66
|
+
}
|
67
|
+
}
|
68
|
+
MetricsCalculationService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsCalculationService, deps: [{ token: i1.QuoteDraftService }, { token: i2.ConfigurationSettingsApiService }], target: i0.ɵɵFactoryTarget.Injectable });
|
69
|
+
MetricsCalculationService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsCalculationService, providedIn: 'root' });
|
70
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsCalculationService, decorators: [{
|
71
|
+
type: Injectable,
|
72
|
+
args: [{ providedIn: 'root' }]
|
73
|
+
}], ctorParameters: function () { return [{ type: i1.QuoteDraftService }, { type: i2.ConfigurationSettingsApiService }]; } });
|
74
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metric-calculation.service.js","sourceRoot":"","sources":["../../../../../../../libs/sdk/core/services/metric-calculation/metric-calculation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { ConfigurationSettingsApiService } from '@veloceapps/api';\nimport { LineItem } from '@veloceapps/core';\nimport { Dictionary, uniq } from 'lodash';\nimport { Observable, Subject, combineLatest, take } from 'rxjs';\nimport { QuoteDraftService } from '../quote-draft.service';\nimport { MetricCalculationMethod } from './metric-calculation.types';\nimport { calculateMetricByMethod } from './metric-calculation.utils';\n\n@Injectable({ providedIn: 'root' })\nexport class MetricsCalculationService {\n  public get onMetricsUpdate$(): Observable<void> {\n    return this.metricsUpdated$.asObservable();\n  }\n\n  private metricsUpdated$ = new Subject<void>();\n  private quoteMetricsSettings: Dictionary<MetricCalculationMethod> = {};\n  private metricsCalculationMethodMap: Dictionary<MetricCalculationMethod> = {};\n  private metricsData: Dictionary<number> = {};\n\n  constructor(private quoteDraftService: QuoteDraftService, private settingsService: ConfigurationSettingsApiService) {\n    combineLatest([\n      this.quoteDraftService.currentState$,\n      this.settingsService.fetchSetting('QUOTE_LEVEL_METRIC_CALCULATION_METHOD').pipe(take(1)),\n    ]).subscribe(([lineItems, setting]) => {\n      let settingsData: Dictionary<MetricCalculationMethod> = {};\n      try {\n        settingsData = JSON.parse(setting?.value || '{}');\n      } catch (error) {\n        settingsData = {};\n      }\n\n      this.quoteMetricsSettings = settingsData;\n\n      this.updateMetrics(lineItems);\n    });\n  }\n\n  public getMetricValue(metric: string): number {\n    return this.metricsData[metric] || 0;\n  }\n\n  private updateMetrics(lineItems: LineItem[]): void {\n    const metricKeys = this.collectMetricKeys(lineItems).filter(key => !key.includes('Effective_'));\n    this.metricsCalculationMethodMap = this.buildMetricsCalculationMethods(\n      metricKeys,\n      this.metricsCalculationMethodMap,\n    );\n\n    this.metricsData = metricKeys.reduce(\n      (acc: Dictionary<number>, key: string) => ({ ...acc, [key]: this.calculateMetric(lineItems, key) }),\n      {},\n    );\n\n    this.metricsUpdated$.next();\n  }\n\n  private calculateMetric(lineItems: LineItem[], metric: string): number {\n    return calculateMetricByMethod(lineItems, metric, this.metricsCalculationMethodMap[metric] || 'sum');\n  }\n\n  private buildMetricsCalculationMethods(\n    metricKeys: string[],\n    initial: Dictionary<MetricCalculationMethod>,\n  ): Dictionary<MetricCalculationMethod> {\n    return metricKeys.reduce((acc: Dictionary<MetricCalculationMethod>, name: string) => {\n      if (acc[name]) {\n        return acc;\n      }\n\n      acc = { ...acc, [name]: 'sum' };\n\n      const settingKey = name.replace(/VDM_|Total_/g, '');\n      if (this.quoteMetricsSettings[settingKey]) {\n        acc = { ...acc, [name]: this.quoteMetricsSettings[settingKey] };\n      }\n\n      return acc;\n    }, initial);\n  }\n\n  private collectMetricKeys(lineItems: LineItem[]): string[] {\n    const keys: string[] = [];\n    lineItems.forEach(lineItem => {\n      keys.push(...Object.keys(lineItem.totalMetrics || {}));\n      keys.push(...this.collectMetricKeys(lineItem.lineItems));\n    });\n    return uniq(keys);\n  }\n}\n"]}
|
@@ -0,0 +1,2 @@
|
|
1
|
+
export {};
|
2
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljLWNhbGN1bGF0aW9uLnR5cGVzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9zZGsvY29yZS9zZXJ2aWNlcy9tZXRyaWMtY2FsY3VsYXRpb24vbWV0cmljLWNhbGN1bGF0aW9uLnR5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiIiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgdHlwZSBNZXRyaWNDYWxjdWxhdGlvbk1ldGhvZCA9ICdsYXN0JyB8ICdmaXJzdCcgfCAnYXZnJyB8ICdzdW0nO1xuIl19
|
@@ -0,0 +1,42 @@
|
|
1
|
+
export function calculateMetricByMethod(lineItems, metric, method) {
|
2
|
+
const items = getLineItemsByMethod(lineItems, method);
|
3
|
+
return items.reduce((acc, li) => {
|
4
|
+
let value = li.reduce((accProduct, item) => accProduct + ((item.totalMetrics && item.totalMetrics[metric]) || 0), 0);
|
5
|
+
if (method === 'avg' && li.length > 0) {
|
6
|
+
value /= li.length;
|
7
|
+
}
|
8
|
+
return acc + value;
|
9
|
+
}, 0);
|
10
|
+
}
|
11
|
+
export function getLineItemsByMethod(lineItems, method) {
|
12
|
+
switch (method) {
|
13
|
+
case 'first': {
|
14
|
+
return lineItems.filter(li => !li.rampInstanceId).map(item => [item]);
|
15
|
+
}
|
16
|
+
case 'last': {
|
17
|
+
const rootTermItems = lineItems.filter(li => !li.rampInstanceId);
|
18
|
+
const products = rootTermItems.map(lineItem => [
|
19
|
+
lineItem,
|
20
|
+
...lineItems.filter(li => li.rampInstanceId === lineItem.id),
|
21
|
+
]);
|
22
|
+
return products
|
23
|
+
.map(items => [...items].sort((a, b) => getDateValue(a.endDate || '') - getDateValue(b.endDate || '')).pop())
|
24
|
+
.filter((li) => Boolean(li))
|
25
|
+
.map(item => [item]);
|
26
|
+
}
|
27
|
+
case 'avg': {
|
28
|
+
const rootTermItems = lineItems.filter(li => !li.rampInstanceId);
|
29
|
+
return rootTermItems.map(lineItem => [lineItem, ...lineItems.filter(li => li.rampInstanceId === lineItem.id)]);
|
30
|
+
}
|
31
|
+
case 'sum': {
|
32
|
+
return lineItems.map(item => [item]);
|
33
|
+
}
|
34
|
+
default: {
|
35
|
+
return lineItems.map(item => [item]);
|
36
|
+
}
|
37
|
+
}
|
38
|
+
}
|
39
|
+
export function getDateValue(date) {
|
40
|
+
return date ? new Date(date).getTime() : 0;
|
41
|
+
}
|
42
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljLWNhbGN1bGF0aW9uLnV0aWxzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vbGlicy9zZGsvY29yZS9zZXJ2aWNlcy9tZXRyaWMtY2FsY3VsYXRpb24vbWV0cmljLWNhbGN1bGF0aW9uLnV0aWxzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUdBLE1BQU0sVUFBVSx1QkFBdUIsQ0FDckMsU0FBcUIsRUFDckIsTUFBYyxFQUNkLE1BQStCO0lBRS9CLE1BQU0sS0FBSyxHQUFHLG9CQUFvQixDQUFDLFNBQVMsRUFBRSxNQUFNLENBQUMsQ0FBQztJQUV0RCxPQUFPLEtBQUssQ0FBQyxNQUFNLENBQUMsQ0FBQyxHQUFXLEVBQUUsRUFBYyxFQUFFLEVBQUU7UUFDbEQsSUFBSSxLQUFLLEdBQUcsRUFBRSxDQUFDLE1BQU0sQ0FDbkIsQ0FBQyxVQUFrQixFQUFFLElBQWMsRUFBRSxFQUFFLENBQUMsVUFBVSxHQUFHLENBQUMsQ0FBQyxJQUFJLENBQUMsWUFBWSxJQUFJLElBQUksQ0FBQyxZQUFZLENBQUMsTUFBTSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsRUFDNUcsQ0FBQyxDQUNGLENBQUM7UUFFRixJQUFJLE1BQU0sS0FBSyxLQUFLLElBQUksRUFBRSxDQUFDLE1BQU0sR0FBRyxDQUFDLEVBQUU7WUFDckMsS0FBSyxJQUFJLEVBQUUsQ0FBQyxNQUFNLENBQUM7U0FDcEI7UUFFRCxPQUFPLEdBQUcsR0FBRyxLQUFLLENBQUM7SUFDckIsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDO0FBQ1IsQ0FBQztBQUVELE1BQU0sVUFBVSxvQkFBb0IsQ0FBQyxTQUFxQixFQUFFLE1BQStCO0lBQ3pGLFFBQVEsTUFBTSxFQUFFO1FBQ2QsS0FBSyxPQUFPLENBQUMsQ0FBQztZQUNaLE9BQU8sU0FBUyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLENBQUMsRUFBRSxDQUFDLGNBQWMsQ0FBQyxDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsRUFBRSxDQUFDLENBQUMsSUFBSSxDQUFDLENBQUMsQ0FBQztTQUN2RTtRQUNELEtBQUssTUFBTSxDQUFDLENBQUM7WUFDWCxNQUFNLGFBQWEsR0FBRyxTQUFTLENBQUMsTUFBTSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUMsY0FBYyxDQUFDLENBQUM7WUFDakUsTUFBTSxRQUFRLEdBQUcsYUFBYSxDQUFDLEdBQUcsQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDO2dCQUM3QyxRQUFRO2dCQUNSLEdBQUcsU0FBUyxDQUFDLE1BQU0sQ0FBQyxFQUFFLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxjQUFjLEtBQUssUUFBUSxDQUFDLEVBQUUsQ0FBQzthQUM3RCxDQUFDLENBQUM7WUFFSCxPQUFPLFFBQVE7aUJBQ1osR0FBRyxDQUFDLEtBQUssQ0FBQyxFQUFFLENBQUMsQ0FBQyxHQUFHLEtBQUssQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFDLEVBQUUsRUFBRSxDQUFDLFlBQVksQ0FBQyxDQUFDLENBQUMsT0FBTyxJQUFJLEVBQUUsQ0FBQyxHQUFHLFlBQVksQ0FBQyxDQUFDLENBQUMsT0FBTyxJQUFJLEVBQUUsQ0FBQyxDQUFDLENBQUMsR0FBRyxFQUFFLENBQUM7aUJBQzVHLE1BQU0sQ0FBQyxDQUFDLEVBQUUsRUFBa0IsRUFBRSxDQUFDLE9BQU8sQ0FBQyxFQUFFLENBQUMsQ0FBQztpQkFDM0MsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1NBQ3hCO1FBQ0QsS0FBSyxLQUFLLENBQUMsQ0FBQztZQUNWLE1BQU0sYUFBYSxHQUFHLFNBQVMsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQyxjQUFjLENBQUMsQ0FBQztZQUNqRSxPQUFPLGFBQWEsQ0FBQyxHQUFHLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDLFFBQVEsRUFBRSxHQUFHLFNBQVMsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLEVBQUUsQ0FBQyxFQUFFLENBQUMsY0FBYyxLQUFLLFFBQVEsQ0FBQyxFQUFFLENBQUMsQ0FBQyxDQUFDLENBQUM7U0FDaEg7UUFDRCxLQUFLLEtBQUssQ0FBQyxDQUFDO1lBQ1YsT0FBTyxTQUFTLENBQUMsR0FBRyxDQUFDLElBQUksQ0FBQyxFQUFFLENBQUMsQ0FBQyxJQUFJLENBQUMsQ0FBQyxDQUFDO1NBQ3RDO1FBQ0QsT0FBTyxDQUFDLENBQUM7WUFDUCxPQUFPLFNBQVMsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLEVBQUUsQ0FBQyxDQUFDLElBQUksQ0FBQyxDQUFDLENBQUM7U0FDdEM7S0FDRjtBQUNILENBQUM7QUFFRCxNQUFNLFVBQVUsWUFBWSxDQUFDLElBQVk7SUFDdkMsT0FBTyxJQUFJLENBQUMsQ0FBQyxDQUFDLElBQUksSUFBSSxDQUFDLElBQUksQ0FBQyxDQUFDLE9BQU8sRUFBRSxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUM7QUFDN0MsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IExpbmVJdGVtIH0gZnJvbSAnQHZlbG9jZWFwcHMvY29yZSc7XG5pbXBvcnQgeyBNZXRyaWNDYWxjdWxhdGlvbk1ldGhvZCB9IGZyb20gJy4vbWV0cmljLWNhbGN1bGF0aW9uLnR5cGVzJztcblxuZXhwb3J0IGZ1bmN0aW9uIGNhbGN1bGF0ZU1ldHJpY0J5TWV0aG9kKFxuICBsaW5lSXRlbXM6IExpbmVJdGVtW10sXG4gIG1ldHJpYzogc3RyaW5nLFxuICBtZXRob2Q6IE1ldHJpY0NhbGN1bGF0aW9uTWV0aG9kLFxuKTogbnVtYmVyIHtcbiAgY29uc3QgaXRlbXMgPSBnZXRMaW5lSXRlbXNCeU1ldGhvZChsaW5lSXRlbXMsIG1ldGhvZCk7XG5cbiAgcmV0dXJuIGl0ZW1zLnJlZHVjZSgoYWNjOiBudW1iZXIsIGxpOiBMaW5lSXRlbVtdKSA9PiB7XG4gICAgbGV0IHZhbHVlID0gbGkucmVkdWNlKFxuICAgICAgKGFjY1Byb2R1Y3Q6IG51bWJlciwgaXRlbTogTGluZUl0ZW0pID0+IGFjY1Byb2R1Y3QgKyAoKGl0ZW0udG90YWxNZXRyaWNzICYmIGl0ZW0udG90YWxNZXRyaWNzW21ldHJpY10pIHx8IDApLFxuICAgICAgMCxcbiAgICApO1xuXG4gICAgaWYgKG1ldGhvZCA9PT0gJ2F2ZycgJiYgbGkubGVuZ3RoID4gMCkge1xuICAgICAgdmFsdWUgLz0gbGkubGVuZ3RoO1xuICAgIH1cblxuICAgIHJldHVybiBhY2MgKyB2YWx1ZTtcbiAgfSwgMCk7XG59XG5cbmV4cG9ydCBmdW5jdGlvbiBnZXRMaW5lSXRlbXNCeU1ldGhvZChsaW5lSXRlbXM6IExpbmVJdGVtW10sIG1ldGhvZDogTWV0cmljQ2FsY3VsYXRpb25NZXRob2QpOiBMaW5lSXRlbVtdW10ge1xuICBzd2l0Y2ggKG1ldGhvZCkge1xuICAgIGNhc2UgJ2ZpcnN0Jzoge1xuICAgICAgcmV0dXJuIGxpbmVJdGVtcy5maWx0ZXIobGkgPT4gIWxpLnJhbXBJbnN0YW5jZUlkKS5tYXAoaXRlbSA9PiBbaXRlbV0pO1xuICAgIH1cbiAgICBjYXNlICdsYXN0Jzoge1xuICAgICAgY29uc3Qgcm9vdFRlcm1JdGVtcyA9IGxpbmVJdGVtcy5maWx0ZXIobGkgPT4gIWxpLnJhbXBJbnN0YW5jZUlkKTtcbiAgICAgIGNvbnN0IHByb2R1Y3RzID0gcm9vdFRlcm1JdGVtcy5tYXAobGluZUl0ZW0gPT4gW1xuICAgICAgICBsaW5lSXRlbSxcbiAgICAgICAgLi4ubGluZUl0ZW1zLmZpbHRlcihsaSA9PiBsaS5yYW1wSW5zdGFuY2VJZCA9PT0gbGluZUl0ZW0uaWQpLFxuICAgICAgXSk7XG5cbiAgICAgIHJldHVybiBwcm9kdWN0c1xuICAgICAgICAubWFwKGl0ZW1zID0+IFsuLi5pdGVtc10uc29ydCgoYSwgYikgPT4gZ2V0RGF0ZVZhbHVlKGEuZW5kRGF0ZSB8fCAnJykgLSBnZXREYXRlVmFsdWUoYi5lbmREYXRlIHx8ICcnKSkucG9wKCkpXG4gICAgICAgIC5maWx0ZXIoKGxpKTogbGkgaXMgTGluZUl0ZW0gPT4gQm9vbGVhbihsaSkpXG4gICAgICAgIC5tYXAoaXRlbSA9PiBbaXRlbV0pO1xuICAgIH1cbiAgICBjYXNlICdhdmcnOiB7XG4gICAgICBjb25zdCByb290VGVybUl0ZW1zID0gbGluZUl0ZW1zLmZpbHRlcihsaSA9PiAhbGkucmFtcEluc3RhbmNlSWQpO1xuICAgICAgcmV0dXJuIHJvb3RUZXJtSXRlbXMubWFwKGxpbmVJdGVtID0+IFtsaW5lSXRlbSwgLi4ubGluZUl0ZW1zLmZpbHRlcihsaSA9PiBsaS5yYW1wSW5zdGFuY2VJZCA9PT0gbGluZUl0ZW0uaWQpXSk7XG4gICAgfVxuICAgIGNhc2UgJ3N1bSc6IHtcbiAgICAgIHJldHVybiBsaW5lSXRlbXMubWFwKGl0ZW0gPT4gW2l0ZW1dKTtcbiAgICB9XG4gICAgZGVmYXVsdDoge1xuICAgICAgcmV0dXJuIGxpbmVJdGVtcy5tYXAoaXRlbSA9PiBbaXRlbV0pO1xuICAgIH1cbiAgfVxufVxuXG5leHBvcnQgZnVuY3Rpb24gZ2V0RGF0ZVZhbHVlKGRhdGU6IHN0cmluZyk6IG51bWJlciB7XG4gIHJldHVybiBkYXRlID8gbmV3IERhdGUoZGF0ZSkuZ2V0VGltZSgpIDogMDtcbn1cbiJdfQ==
|
@@ -5,9 +5,9 @@ import { invalidCharactersValidator, requiredValidator, reservedIdentifierValida
|
|
5
5
|
import { FlowAction } from '@veloceapps/sdk/cms';
|
6
6
|
import { cloneDeep, isEqual, uniq } from 'lodash';
|
7
7
|
import { OverlayPanel } from 'primeng/overlaypanel';
|
8
|
-
import {
|
8
|
+
import { Subject, startWith } from 'rxjs';
|
9
9
|
import { takeUntil, tap } from 'rxjs/operators';
|
10
|
-
import { defaultMetrics, effectiveMetricKeys
|
10
|
+
import { METRICS_STORAGE_KEY, defaultMetrics, effectiveMetricKeys } from './metrics.definitions';
|
11
11
|
import * as i0 from "@angular/core";
|
12
12
|
import * as i1 from "@veloceapps/sdk/cms";
|
13
13
|
import * as i2 from "@veloceapps/sdk/core";
|
@@ -22,11 +22,12 @@ import * as i10 from "@angular/forms";
|
|
22
22
|
import * as i11 from "primeng/inputtext";
|
23
23
|
import * as i12 from "@angular/cdk/drag-drop";
|
24
24
|
export class MetricsComponent {
|
25
|
-
constructor(integrationState, contextService, quoteDraftService, cdr) {
|
25
|
+
constructor(integrationState, contextService, quoteDraftService, cdr, metricsCalculationService) {
|
26
26
|
this.integrationState = integrationState;
|
27
27
|
this.contextService = contextService;
|
28
28
|
this.quoteDraftService = quoteDraftService;
|
29
29
|
this.cdr = cdr;
|
30
|
+
this.metricsCalculationService = metricsCalculationService;
|
30
31
|
this.emptyStateMetrics = [
|
31
32
|
{ name: 'MRR', value: 0 },
|
32
33
|
{ name: 'E.MRR', value: 0 },
|
@@ -79,6 +80,9 @@ export class MetricsComponent {
|
|
79
80
|
return (name || '').toLowerCase().includes(query === null || query === void 0 ? void 0 : query.toLowerCase());
|
80
81
|
});
|
81
82
|
});
|
83
|
+
this.metricsCalculationService.onMetricsUpdate$
|
84
|
+
.pipe(takeUntil(this.destroyed$))
|
85
|
+
.subscribe(() => this.cdr.detectChanges());
|
82
86
|
}
|
83
87
|
ngOnDestroy() {
|
84
88
|
this.destroyed$.next();
|
@@ -101,7 +105,7 @@ export class MetricsComponent {
|
|
101
105
|
return this.toNumber(context.properties[correctKey]) - this.toNumber(context.properties[`Previous${correctKey}`]);
|
102
106
|
}
|
103
107
|
else {
|
104
|
-
return this.
|
108
|
+
return this.metricsCalculationService.getMetricValue(key);
|
105
109
|
}
|
106
110
|
}
|
107
111
|
save() {
|
@@ -199,7 +203,7 @@ export class MetricsComponent {
|
|
199
203
|
return +(value || 0);
|
200
204
|
}
|
201
205
|
}
|
202
|
-
MetricsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsComponent, deps: [{ token: i1.IntegrationState }, { token: i2.ContextService }, { token: i2.QuoteDraftService }, { token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component });
|
206
|
+
MetricsComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsComponent, deps: [{ token: i1.IntegrationState }, { token: i2.ContextService }, { token: i2.QuoteDraftService }, { token: i0.ChangeDetectorRef }, { token: i2.MetricsCalculationService }], target: i0.ɵɵFactoryTarget.Component });
|
203
207
|
MetricsComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "12.0.0", version: "12.2.15", type: MetricsComponent, selector: "vl-metrics", viewQueries: [{ propertyName: "overlayPanel", first: true, predicate: OverlayPanel, descendants: true }], ngImport: i0, template: "<div class=\"header-metrics\" (click)=\"overlayPanel.toggle($event)\">\n <div *ngFor=\"let metric of visibleSelectedMetrics\" class=\"metric\">\n <span class=\"name\" [vlHiddenTextTooltip]=\"metric.name\" tooltipStyleClass=\"metric-tooltip\" tooltipPosition=\"bottom\">\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n </span>\n </div>\n\n <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"${{ metric.value | number: '1.2-2' }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n ${{ metric.value | number: '1.2-2' }}\n </span>\n </div>\n </ng-container>\n</div>\n\n<p-overlayPanel\n styleClass=\"metrics-overlay-container center no-padding\"\n showTransitionOptions=\"0ms\"\n hideTransitionOptions=\"0ms\"\n>\n <ng-template pTemplate>\n <div class=\"overlay-metrics\">\n <div *ngFor=\"let metric of restSelectedMetrics\" class=\"metric\">\n <span\n class=\"name\"\n [vlHiddenTextTooltip]=\"metric.name\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n {{ metric.name }}\n </span>\n <span class=\"filler\"></span>\n <span\n class=\"value\"\n vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n tooltipStyleClass=\"metric-tooltip\"\n tooltipPosition=\"bottom\"\n >\n ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n </span>\n </div>\n </div>\n <button class=\"config\" (click)=\"openSidebar()\">\n Metrics Config\n <i class=\"vl-icon vl-icon-chevron-down\"></i>\n </button>\n </ng-template>\n</p-overlayPanel>\n\n<p-sidebar\n [visible]=\"sidebarVisible\"\n (visibleChange)=\"closeSidebar()\"\n position=\"right\"\n [baseZIndex]=\"1000\"\n [style]=\"{ width: '360px' }\"\n>\n <div class=\"container\">\n <div class=\"header\">\n Quote Metrics Config\n <i class=\"vl-icon vl-icon-close\" (click)=\"closeSidebar()\"></i>\n </div>\n\n <div class=\"search-container\">\n <input\n data-test-id=\"search\"\n [formControl]=\"searchControl\"\n pInputText\n placeholder=\"Search for metric\u2026\"\n class=\"w-full\"\n />\n </div>\n\n <div class=\"content\">\n <div class=\"sidebar-metrics\" cdkDropList (cdkDropListDropped)=\"changeMetricOrder($event)\">\n <div\n *ngFor=\"let metric of filteredMetrics\"\n class=\"sidebar-metric\"\n [class.edit]=\"metric.key === editingMetric?.key\"\n cdkDrag\n [cdkDragDisabled]=\"!!searchControl.value\"\n >\n <div class=\"drag-icon\">\n <i class=\"vl-icon vl-icon-reorder\" cdkDragHandle></i>\n </div>\n <ng-container\n [ngTemplateOutlet]=\"metricTemplate\"\n [ngTemplateOutletContext]=\"{ metric: metric }\"\n ></ng-container>\n </div>\n </div>\n\n <div class=\"empty-state\" *ngIf=\"!filteredMetrics.length\">There are no items matching your search criteria</div>\n </div>\n <div class=\"footer\">\n <p-button\n label=\"Reset to default\"\n styleClass=\"p-button-outlined p-button-sm\"\n (onClick)=\"resetToDefault()\"\n ></p-button>\n <p-button label=\"Save\" styleClass=\"p-button p-button-filled p-button-sm\" (onClick)=\"save()\"></p-button>\n </div>\n </div>\n</p-sidebar>\n\n<ng-template #metricTemplate let-metric=\"metric\">\n <div *ngIf=\"editingMetric?.key !== metric.key\" class=\"preview-state\">\n <p-checkbox\n [ngModel]=\"metric.visible\"\n (ngModelChange)=\"metric.visible = !metric.visible\"\n [binary]=\"true\"\n checkboxIcon=\"vl-icon vl-icon-checkmark\"\n ></p-checkbox>\n <div class=\"title\">{{ metric.name }}</div>\n </div>\n\n <div *ngIf=\"editingMetric?.key === metric.key\" class=\"edit-state\">\n <input data-test-id=\"name\" [formControl]=\"nameControl\" pInputText placeholder=\"Metric name\" class=\"w-full\" />\n <vl-error-tooltip [tooltip]=\"nameControl | error: 'name'\" [visible]=\"!!nameControl.errors\"></vl-error-tooltip>\n </div>\n <div *ngIf=\"editingMetric?.key !== metric.key\" (click)=\"editMetric(metric)\" class=\"action edit\">\n <i class=\"vl-icon vl-icon-edit-sm\"></i>\n </div>\n <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"saveMetric()\" class=\"action save\">\n <i class=\"vl-icon vl-icon-checkmark\"></i>\n </div>\n</ng-template>\n", styles: [":host .header-metrics{cursor:pointer;min-width:130px;max-width:260px;display:flex;flex-wrap:wrap;justify-content:flex-start;grid-gap:0 16px;gap:0 16px;height:32px;align-items:flex-start}:host .header-metrics .metric{flex:1;flex-basis:114px;max-width:122px;display:flex;justify-content:space-between;align-items:flex-end;font-size:12px;line-height:16px;letter-spacing:.3px}:host .header-metrics .metric .name{max-width:55px;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}:host .header-metrics .metric .value{font-weight:600;overflow:hidden;text-overflow:ellipsis}:host .header-metrics .metric.empty-state-metric{filter:blur(1.5px)}:host ::ng-deep .p-sidebar-header{display:none}:host ::ng-deep .p-sidebar-content{padding:0 24px;height:100%}::ng-deep .metrics-overlay-container{max-width:300px}::ng-deep .metrics-overlay-container .p-overlaypanel-content{padding:0}::ng-deep .metrics-overlay-container .overlay-metrics{max-height:110px;overflow:auto}::ng-deep .metrics-overlay-container .metric{display:flex;padding:0 12px;font-size:12px;line-height:16px;letter-spacing:.3px;align-items:flex-end}::ng-deep .metrics-overlay-container .metric .name{max-width:50%;flex-shrink:0;overflow:hidden;white-space:nowrap;text-overflow:ellipsis}::ng-deep .metrics-overlay-container .metric .value{font-weight:600;overflow:hidden;text-overflow:ellipsis}::ng-deep .metrics-overlay-container .metric:not(:last-of-type){margin-bottom:4px}::ng-deep .metrics-overlay-container .metric:first-of-type{margin-top:12px}::ng-deep .metrics-overlay-container .metric:last-of-type{margin-bottom:12px}::ng-deep .metrics-overlay-container .config{display:flex;align-items:center;justify-content:space-between;padding:0 12px;cursor:pointer;background:var(--vl-color-light-blue);outline:none;border:none;width:100%;height:32px;font-size:12px;line-height:16px;letter-spacing:.3px}::ng-deep .metrics-overlay-container .config .vl-icon-chevron-down{color:var(--vl-text-color-accent-light);transform:rotate(-90deg)}.filler{flex:1;margin:0 4px 3px;min-width:5px}.filler:before{display:block;width:100%;height:2px;content:\"\";background-image:radial-gradient(circle at 1px 1px,var(--vl-text-color-accent-light) 1px,transparent 0);background-size:3px 2px}.container{display:flex;flex-direction:column;height:100%}.container .header{margin:15px 0 23px;flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;font-weight:600;font-size:18px;line-height:26px;letter-spacing:-.6px;color:var(--vl-accent-color)}.container .header i{cursor:pointer;font-size:15px}.container .search-container{margin-bottom:12px}.container .search-container input{width:100%}.container .content{display:flex;flex-direction:column;flex:1;overflow:hidden}.container .footer{flex:0 0 auto;display:flex;align-items:center;justify-content:space-between;height:60px;padding:0 20px;margin:0 -16px;border-top:1px solid var(--vl-border-color)}.container .footer p-button{flex:1}.container .footer p-button ::ng-deep button{width:100%}.container .footer p-button:not(:last-child){margin-right:16px}.container app-empty-state::ng-deep{margin-bottom:12px}.container app-empty-state::ng-deep .container{width:100%;padding-bottom:0}.sidebar-metrics{width:100%;overflow:auto;padding:1px 0}.empty-state{display:flex;flex:1;align-items:center;justify-content:center;color:var(--vl-text-color-deep-accent);font-size:12px;line-height:16px;letter-spacing:.3px}.sidebar-metric{display:flex;align-items:center;height:32px;position:relative;padding-left:8px;border-radius:2px}.sidebar-metric:hover:not(.edit){background:var(--vl-color-light-blue)}.sidebar-metric:hover:not(.edit) .edit{visibility:visible}.sidebar-metric p-checkbox{margin-right:8px}.sidebar-metric .drag-icon{width:16px;height:16px;margin-right:8px}.sidebar-metric .drag-icon i{cursor:pointer;color:#c5d1e2}.sidebar-metric.cdk-drag-disabled .drag-icon i{cursor:default}.sidebar-metric .title{font-weight:400;font-size:12px;line-height:16px;overflow:hidden;white-space:nowrap;text-overflow:ellipsis;-webkit-user-select:text;-moz-user-select:text;-ms-user-select:text;user-select:text;cursor:text;color:var(--vl-text-color-secondary)}.sidebar-metric .preview-state,.sidebar-metric .edit-state{position:relative;margin-right:8px;min-width:0;display:flex;align-items:center;flex:1}.sidebar-metric .preview-state input,.sidebar-metric .edit-state input{width:100%}.sidebar-metric .action{cursor:pointer;width:32px;height:32px;display:flex;align-items:center;justify-content:center}.sidebar-metric .edit{visibility:hidden}.sidebar-metric .edit i{color:var(--vl-text-color-deep-accent);width:12px;height:12px}.sidebar-metric .save{background:var(--vl-primary-color);border-radius:2px}.sidebar-metric .save i{color:#fff}.sidebar-metric ::ng-deep .p-checkbox{width:16px;height:16px}.sidebar-metric ::ng-deep .p-checkbox .p-checkbox-box{width:16px;height:16px}.cdk-drag-preview{z-index:10001!important;box-sizing:border-box;border-radius:4px;box-shadow:0 5px 5px -3px #0003,0 8px 10px 1px #00000024,0 3px 14px 2px #0000001f}.cdk-drag-placeholder{opacity:0}.cdk-drag-animating{transition:transform .25s cubic-bezier(0,0,.2,1)}::ng-deep .metric-tooltip .p-tooltip-text{font-size:11px;line-height:14px;letter-spacing:.3px;background:var(--vl-text-color-accent)}\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.Sidebar, selector: "p-sidebar", inputs: ["appendTo", "blockScroll", "style", "styleClass", "ariaCloseLabel", "autoZIndex", "baseZIndex", "modal", "dismissible", "showCloseIcon", "closeOnEscape", "transitionOptions", "visible", "position", "fullScreen"], outputs: ["onShow", "onHide", "visibleChange"] }, { type: i5.Button, selector: "p-button", inputs: ["type", "iconPos", "icon", "badge", "label", "disabled", "loading", "loadingIcon", "style", "styleClass", "badgeClass"], outputs: ["onClick", "onFocus", "onBlur"] }, { type: i6.Checkbox, selector: "p-checkbox", inputs: ["value", "name", "disabled", "binary", "label", "ariaLabelledBy", "ariaLabel", "tabindex", "inputId", "style", "styleClass", "labelStyleClass", "formControl", "checkboxIcon", "readonly", "required", "trueValue", "falseValue"], outputs: ["onChange"] }, { type: i7.ErrorTooltipComponent, selector: "vl-error-tooltip", inputs: ["tooltip", "visible"] }], directives: [{ type: i8.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { type: i7.AppHiddenTextTooltipDirective, selector: "[vlHiddenTextTooltip]", inputs: ["vlHiddenTextTooltip"] }, { type: i8.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { type: i9.PrimeTemplate, selector: "[pTemplate]", inputs: ["type", "pTemplate"] }, { type: i10.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { type: i11.InputText, selector: "[pInputText]" }, { type: i10.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { type: i10.FormControlDirective, selector: "[formControl]", inputs: ["disabled", "formControl", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { type: i12.CdkDropList, selector: "[cdkDropList], cdk-drop-list", inputs: ["cdkDropListConnectedTo", "id", "cdkDropListEnterPredicate", "cdkDropListSortPredicate", "cdkDropListDisabled", "cdkDropListSortingDisabled", "cdkDropListAutoScrollDisabled", "cdkDropListOrientation", "cdkDropListLockAxis", "cdkDropListData", "cdkDropListAutoScrollStep"], outputs: ["cdkDropListDropped", "cdkDropListEntered", "cdkDropListExited", "cdkDropListSorted"], exportAs: ["cdkDropList"] }, { type: i12.CdkDrag, selector: "[cdkDrag]", inputs: ["cdkDragDisabled", "cdkDragStartDelay", "cdkDragLockAxis", "cdkDragConstrainPosition", "cdkDragPreviewClass", "cdkDragBoundary", "cdkDragRootElement", "cdkDragPreviewContainer", "cdkDragData", "cdkDragFreeDragPosition"], outputs: ["cdkDragStarted", "cdkDragReleased", "cdkDragEnded", "cdkDragEntered", "cdkDragExited", "cdkDragDropped", "cdkDragMoved"], exportAs: ["cdkDrag"] }, { type: i12.CdkDragHandle, selector: "[cdkDragHandle]", inputs: ["cdkDragHandleDisabled"] }, { type: i8.NgTemplateOutlet, selector: "[ngTemplateOutlet]", inputs: ["ngTemplateOutletContext", "ngTemplateOutlet"] }, { type: i10.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }], pipes: { "number": i8.DecimalPipe, "error": i7.ErrorPipe }, changeDetection: i0.ChangeDetectionStrategy.OnPush });
|
204
208
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsComponent, decorators: [{
|
205
209
|
type: Component,
|
@@ -209,8 +213,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
|
|
209
213
|
styleUrls: ['./metrics.component.scss'],
|
210
214
|
changeDetection: ChangeDetectionStrategy.OnPush,
|
211
215
|
}]
|
212
|
-
}], ctorParameters: function () { return [{ type: i1.IntegrationState }, { type: i2.ContextService }, { type: i2.QuoteDraftService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { overlayPanel: [{
|
216
|
+
}], ctorParameters: function () { return [{ type: i1.IntegrationState }, { type: i2.ContextService }, { type: i2.QuoteDraftService }, { type: i0.ChangeDetectorRef }, { type: i2.MetricsCalculationService }]; }, propDecorators: { overlayPanel: [{
|
213
217
|
type: ViewChild,
|
214
218
|
args: [OverlayPanel]
|
215
219
|
}] } });
|
216
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metrics.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/metrics/metrics.component.ts","../../../../../../../../libs/sdk/src/components/header/metrics/metrics.component.html"],"names":[],"mappings":"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","sourcesContent":["import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit, ViewChild } from '@angular/core';\nimport { FormControl, Validators } from '@angular/forms';\nimport { LineItem } from '@veloceapps/core';\nimport {\n  invalidCharactersValidator,\n  requiredValidator,\n  reservedIdentifierValidator,\n  standardCharSetWithSpaceRegExp,\n} from '@veloceapps/core/forms';\nimport { FlowAction, IntegrationState, Metric } from '@veloceapps/sdk/cms';\nimport { ContextService, QuoteDraftService } from '@veloceapps/sdk/core';\nimport { cloneDeep, isEqual, uniq } from 'lodash';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport { startWith, Subject } from 'rxjs';\nimport { takeUntil, tap } from 'rxjs/operators';\nimport { defaultMetrics, effectiveMetricKeys, METRICS_STORAGE_KEY } from './metrics.definitions';\n\n@Component({\n  selector: 'vl-metrics',\n  templateUrl: './metrics.component.html',\n  styleUrls: ['./metrics.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MetricsComponent implements OnInit, OnDestroy {\n  @ViewChild(OverlayPanel) public overlayPanel!: OverlayPanel;\n\n  public emptyStateMetrics = [\n    { name: 'MRR', value: 0 },\n    { name: 'E.MRR', value: 0 },\n    { name: 'NRR', value: 0 },\n    { name: 'E.NRR', value: 0 },\n  ];\n  public visibleSelectedMetrics: Metric[] = [];\n  public restSelectedMetrics: Metric[] = [];\n  public metrics: Metric[] = [];\n  public filteredMetrics: Metric[] = [];\n  public editingMetric: Metric | undefined;\n  public sidebarVisible = false;\n  public searchControl: FormControl = new FormControl('');\n  public nameControl: FormControl = new FormControl('', [\n    requiredValidator,\n    invalidCharactersValidator(standardCharSetWithSpaceRegExp),\n    reservedIdentifierValidator,\n    Validators.maxLength(80),\n  ]);\n\n  private defaultMetrics: Metric[] = [];\n  private storedDefaultMetrics: Metric[] = [];\n  private metricKeys: string[] = [];\n  private lastSavedMetrics: Metric[] = [];\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    private integrationState: IntegrationState,\n    private contextService: ContextService,\n    private quoteDraftService: QuoteDraftService,\n    private cdr: ChangeDetectorRef,\n  ) {}\n\n  public ngOnInit(): void {\n    this.quoteDraftService.currentState$\n      .pipe(\n        takeUntil(this.destroyed$),\n        tap(lineItems => {\n          this.metricKeys = this.collectMetricKeys(lineItems).reduce((result, key) => {\n            result.push(key);\n            if (effectiveMetricKeys.includes(key)) {\n              result.push(`Effective_${key}`);\n            }\n            return result;\n          }, [] as string[]);\n          this.collectAvailableMetrics();\n        }),\n      )\n      .subscribe();\n\n    this.integrationState\n      .listen$(FlowAction.SET_DEFAULT_METRICS)\n      .pipe(\n        tap(({ metrics }) => {\n          if (isEqual(this.storedDefaultMetrics, metrics)) {\n            return;\n          }\n          this.storedDefaultMetrics = metrics;\n          this.collectAvailableMetrics();\n        }),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n\n    this.searchControl.valueChanges.pipe(takeUntil(this.destroyed$), startWith('')).subscribe(query => {\n      this.filteredMetrics = this.metrics.filter(({ name }) => {\n        return (name || '').toLowerCase().includes(query?.toLowerCase());\n      });\n    });\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public openSidebar(): void {\n    this.overlayPanel.hide();\n    this.sidebarVisible = true;\n  }\n\n  public closeSidebar(): void {\n    this.sidebarVisible = false;\n    this.metrics = cloneDeep(this.lastSavedMetrics);\n    this.editingMetric = undefined;\n    this.searchControl.setValue('');\n  }\n\n  public getMetricValue(key: string): number {\n    const context = this.contextService.resolve();\n    if (key.startsWith('Effective_')) {\n      const correctKey = key.replace('Effective_', '');\n      return this.toNumber(context.properties[correctKey]) - this.toNumber(context.properties[`Previous${correctKey}`]);\n    } else {\n      return this.toNumber(context.properties[key]);\n    }\n  }\n\n  public save(): void {\n    localStorage.setItem(METRICS_STORAGE_KEY, JSON.stringify(this.metrics));\n    this.setMetrics();\n    this.lastSavedMetrics = cloneDeep(this.metrics);\n    this.closeSidebar();\n  }\n\n  public changeMetricOrder(event: CdkDragDrop<string[]>): void {\n    moveItemInArray(this.metrics, event.previousIndex, event.currentIndex);\n    this.filteredMetrics = this.metrics;\n  }\n\n  public editMetric(metric: Metric): void {\n    if (this.editingMetric && this.nameControl.invalid) {\n      return;\n    }\n    this.editingMetric = metric;\n    this.nameControl.setValue(metric.name);\n  }\n\n  public saveMetric(): void {\n    if (this.nameControl.invalid) {\n      return;\n    }\n    if (this.editingMetric) {\n      this.editingMetric.name = this.nameControl.value?.trim();\n    }\n    this.editingMetric = undefined;\n  }\n\n  public resetToDefault(): void {\n    this.collectAvailableMetrics(true);\n  }\n\n  private collectMetricKeys(lineItems: LineItem[]): string[] {\n    const keys: string[] = [];\n    lineItems.forEach(lineItem => {\n      keys.push(...Object.keys(lineItem.totalMetrics || {}));\n      keys.push(...this.collectMetricKeys(lineItem.lineItems));\n    });\n    return uniq(keys);\n  }\n\n  private setMetrics(): void {\n    const metrics = cloneDeep(this.metrics.filter(({ visible }) => visible));\n\n    this.visibleSelectedMetrics = metrics.slice(0, 4);\n    this.restSelectedMetrics = metrics.slice(4);\n  }\n\n  private collectAvailableMetrics(resetToDefault = false): void {\n    this.metrics = [];\n    this.defaultMetrics = this.storedDefaultMetrics.length ? this.storedDefaultMetrics : defaultMetrics;\n    const storedMetrics: Metric[] = JSON.parse(localStorage.getItem(METRICS_STORAGE_KEY) || '[]');\n    const metricKeys = [...this.metricKeys];\n    if (!storedMetrics.length || resetToDefault) {\n      this.defaultMetrics.forEach(defaultMetric => {\n        const metricKeyIndex = metricKeys.findIndex(key => key === defaultMetric.key);\n\n        const storedMetric = storedMetrics.find(({ key }) => key === defaultMetric.key);\n        if (storedMetric && !resetToDefault) {\n          this.metrics.push(storedMetric);\n        } else {\n          this.metrics.push(defaultMetric);\n        }\n        if (metricKeyIndex >= 0) {\n          metricKeys.splice(metricKeyIndex, 1);\n        }\n      });\n    } else {\n      this.metrics = [\n        ...storedMetrics.reduce((result, metric) => {\n          const metricKeyIndex = metricKeys.findIndex(key => key === metric.key);\n\n          if (metricKeyIndex >= 0) {\n            metricKeys.splice(metricKeyIndex, 1);\n          }\n\n          return [...result, metric];\n        }, [] as Metric[]),\n      ];\n    }\n    this.metrics = [\n      ...this.metrics,\n      ...metricKeys.map(metricKey => {\n        return { key: metricKey, name: this.getMetricName(metricKey), visible: false };\n      }),\n    ];\n    this.metrics = cloneDeep(this.metrics);\n    this.searchControl.setValue('');\n    if (!resetToDefault) {\n      this.lastSavedMetrics = cloneDeep(this.metrics);\n      this.setMetrics();\n    }\n    this.cdr.markForCheck();\n  }\n\n  private getMetricName(key: string): string {\n    return key.replace(/VDM_|Total_/g, '');\n  }\n\n  private toNumber(value: string | undefined): number {\n    return +(value || 0);\n  }\n}\n","<div class=\"header-metrics\" (click)=\"overlayPanel.toggle($event)\">\n  <div *ngFor=\"let metric of visibleSelectedMetrics\" class=\"metric\">\n    <span class=\"name\" [vlHiddenTextTooltip]=\"metric.name\" tooltipStyleClass=\"metric-tooltip\" tooltipPosition=\"bottom\">\n      {{ metric.name }}\n    </span>\n    <span class=\"filler\"></span>\n    <span\n      class=\"value\"\n      vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n      tooltipStyleClass=\"metric-tooltip\"\n      tooltipPosition=\"bottom\"\n    >\n      ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n    </span>\n  </div>\n\n  <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n    <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n      <span\n        class=\"name\"\n        [vlHiddenTextTooltip]=\"metric.name\"\n        tooltipStyleClass=\"metric-tooltip\"\n        tooltipPosition=\"bottom\"\n      >\n        {{ metric.name }}\n      </span>\n      <span class=\"filler\"></span>\n      <span\n        class=\"value\"\n        vlHiddenTextTooltip=\"${{ metric.value | number: '1.2-2' }}\"\n        tooltipStyleClass=\"metric-tooltip\"\n        tooltipPosition=\"bottom\"\n      >\n        ${{ metric.value | number: '1.2-2' }}\n      </span>\n    </div>\n  </ng-container>\n</div>\n\n<p-overlayPanel\n  styleClass=\"metrics-overlay-container center no-padding\"\n  showTransitionOptions=\"0ms\"\n  hideTransitionOptions=\"0ms\"\n>\n  <ng-template pTemplate>\n    <div class=\"overlay-metrics\">\n      <div *ngFor=\"let metric of restSelectedMetrics\" class=\"metric\">\n        <span\n          class=\"name\"\n          [vlHiddenTextTooltip]=\"metric.name\"\n          tooltipStyleClass=\"metric-tooltip\"\n          tooltipPosition=\"bottom\"\n        >\n          {{ metric.name }}\n        </span>\n        <span class=\"filler\"></span>\n        <span\n          class=\"value\"\n          vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n          tooltipStyleClass=\"metric-tooltip\"\n          tooltipPosition=\"bottom\"\n        >\n          ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n        </span>\n      </div>\n    </div>\n    <button class=\"config\" (click)=\"openSidebar()\">\n      Metrics Config\n      <i class=\"vl-icon vl-icon-chevron-down\"></i>\n    </button>\n  </ng-template>\n</p-overlayPanel>\n\n<p-sidebar\n  [visible]=\"sidebarVisible\"\n  (visibleChange)=\"closeSidebar()\"\n  position=\"right\"\n  [baseZIndex]=\"1000\"\n  [style]=\"{ width: '360px' }\"\n>\n  <div class=\"container\">\n    <div class=\"header\">\n      Quote Metrics Config\n      <i class=\"vl-icon vl-icon-close\" (click)=\"closeSidebar()\"></i>\n    </div>\n\n    <div class=\"search-container\">\n      <input\n        data-test-id=\"search\"\n        [formControl]=\"searchControl\"\n        pInputText\n        placeholder=\"Search for metric…\"\n        class=\"w-full\"\n      />\n    </div>\n\n    <div class=\"content\">\n      <div class=\"sidebar-metrics\" cdkDropList (cdkDropListDropped)=\"changeMetricOrder($event)\">\n        <div\n          *ngFor=\"let metric of filteredMetrics\"\n          class=\"sidebar-metric\"\n          [class.edit]=\"metric.key === editingMetric?.key\"\n          cdkDrag\n          [cdkDragDisabled]=\"!!searchControl.value\"\n        >\n          <div class=\"drag-icon\">\n            <i class=\"vl-icon vl-icon-reorder\" cdkDragHandle></i>\n          </div>\n          <ng-container\n            [ngTemplateOutlet]=\"metricTemplate\"\n            [ngTemplateOutletContext]=\"{ metric: metric }\"\n          ></ng-container>\n        </div>\n      </div>\n\n      <div class=\"empty-state\" *ngIf=\"!filteredMetrics.length\">There are no items matching your search criteria</div>\n    </div>\n    <div class=\"footer\">\n      <p-button\n        label=\"Reset to default\"\n        styleClass=\"p-button-outlined p-button-sm\"\n        (onClick)=\"resetToDefault()\"\n      ></p-button>\n      <p-button label=\"Save\" styleClass=\"p-button p-button-filled p-button-sm\" (onClick)=\"save()\"></p-button>\n    </div>\n  </div>\n</p-sidebar>\n\n<ng-template #metricTemplate let-metric=\"metric\">\n  <div *ngIf=\"editingMetric?.key !== metric.key\" class=\"preview-state\">\n    <p-checkbox\n      [ngModel]=\"metric.visible\"\n      (ngModelChange)=\"metric.visible = !metric.visible\"\n      [binary]=\"true\"\n      checkboxIcon=\"vl-icon vl-icon-checkmark\"\n    ></p-checkbox>\n    <div class=\"title\">{{ metric.name }}</div>\n  </div>\n\n  <div *ngIf=\"editingMetric?.key === metric.key\" class=\"edit-state\">\n    <input data-test-id=\"name\" [formControl]=\"nameControl\" pInputText placeholder=\"Metric name\" class=\"w-full\" />\n    <vl-error-tooltip [tooltip]=\"nameControl | error: 'name'\" [visible]=\"!!nameControl.errors\"></vl-error-tooltip>\n  </div>\n  <div *ngIf=\"editingMetric?.key !== metric.key\" (click)=\"editMetric(metric)\" class=\"action edit\">\n    <i class=\"vl-icon vl-icon-edit-sm\"></i>\n  </div>\n  <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"saveMetric()\" class=\"action save\">\n    <i class=\"vl-icon vl-icon-checkmark\"></i>\n  </div>\n</ng-template>\n"]}
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"metrics.component.js","sourceRoot":"","sources":["../../../../../../../../libs/sdk/src/components/header/metrics/metrics.component.ts","../../../../../../../../libs/sdk/src/components/header/metrics/metrics.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAe,eAAe,EAAE,MAAM,wBAAwB,CAAC;AACtE,OAAO,EAAE,uBAAuB,EAAqB,SAAS,EAAqB,SAAS,EAAE,MAAM,eAAe,CAAC;AACpH,OAAO,EAAE,WAAW,EAAE,UAAU,EAAE,MAAM,gBAAgB,CAAC;AAEzD,OAAO,EACL,0BAA0B,EAC1B,iBAAiB,EACjB,2BAA2B,EAC3B,8BAA8B,GAC/B,MAAM,wBAAwB,CAAC;AAChC,OAAO,EAAE,UAAU,EAA4B,MAAM,qBAAqB,CAAC;AAE3E,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,IAAI,EAAE,MAAM,QAAQ,CAAC;AAClD,OAAO,EAAE,YAAY,EAAE,MAAM,sBAAsB,CAAC;AACpD,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAC1C,OAAO,EAAE,SAAS,EAAE,GAAG,EAAE,MAAM,gBAAgB,CAAC;AAChD,OAAO,EAAE,mBAAmB,EAAE,cAAc,EAAE,mBAAmB,EAAE,MAAM,uBAAuB,CAAC;;;;;;;;;;;;;;AAQjG,MAAM,OAAO,gBAAgB;IA6B3B,YACU,gBAAkC,EAClC,cAA8B,EAC9B,iBAAoC,EACpC,GAAsB,EACtB,yBAAoD;QAJpD,qBAAgB,GAAhB,gBAAgB,CAAkB;QAClC,mBAAc,GAAd,cAAc,CAAgB;QAC9B,sBAAiB,GAAjB,iBAAiB,CAAmB;QACpC,QAAG,GAAH,GAAG,CAAmB;QACtB,8BAAyB,GAAzB,yBAAyB,CAA2B;QA/BvD,sBAAiB,GAAG;YACzB,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,EAAE;YACzB,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE;YAC3B,EAAE,IAAI,EAAE,KAAK,EAAE,KAAK,EAAE,CAAC,EAAE;YACzB,EAAE,IAAI,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC,EAAE;SAC5B,CAAC;QACK,2BAAsB,GAAa,EAAE,CAAC;QACtC,wBAAmB,GAAa,EAAE,CAAC;QACnC,YAAO,GAAa,EAAE,CAAC;QACvB,oBAAe,GAAa,EAAE,CAAC;QAE/B,mBAAc,GAAG,KAAK,CAAC;QACvB,kBAAa,GAAgB,IAAI,WAAW,CAAC,EAAE,CAAC,CAAC;QACjD,gBAAW,GAAgB,IAAI,WAAW,CAAC,EAAE,EAAE;YACpD,iBAAiB;YACjB,0BAA0B,CAAC,8BAA8B,CAAC;YAC1D,2BAA2B;YAC3B,UAAU,CAAC,SAAS,CAAC,EAAE,CAAC;SACzB,CAAC,CAAC;QAEK,mBAAc,GAAa,EAAE,CAAC;QAC9B,yBAAoB,GAAa,EAAE,CAAC;QACpC,eAAU,GAAa,EAAE,CAAC;QAC1B,qBAAgB,GAAa,EAAE,CAAC;QAChC,eAAU,GAAG,IAAI,OAAO,EAAQ,CAAC;IAQtC,CAAC;IAEG,QAAQ;QACb,IAAI,CAAC,iBAAiB,CAAC,aAAa;aACjC,IAAI,CACH,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,EAC1B,GAAG,CAAC,SAAS,CAAC,EAAE;YACd,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,iBAAiB,CAAC,SAAS,CAAC,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,GAAG,EAAE,EAAE;gBACzE,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBACjB,IAAI,mBAAmB,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE;oBACrC,MAAM,CAAC,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC,CAAC;iBACjC;gBACD,OAAO,MAAM,CAAC;YAChB,CAAC,EAAE,EAAc,CAAC,CAAC;YAEnB,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC,CAAC,CACH;aACA,SAAS,EAAE,CAAC;QAEf,IAAI,CAAC,gBAAgB;aAClB,OAAO,CAAC,UAAU,CAAC,mBAAmB,CAAC;aACvC,IAAI,CACH,GAAG,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE;YAClB,IAAI,OAAO,CAAC,IAAI,CAAC,oBAAoB,EAAE,OAAO,CAAC,EAAE;gBAC/C,OAAO;aACR;YACD,IAAI,CAAC,oBAAoB,GAAG,OAAO,CAAC;YACpC,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC,CAAC,EACF,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAC3B;aACA,SAAS,EAAE,CAAC;QAEf,IAAI,CAAC,aAAa,CAAC,YAAY,CAAC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,EAAE,SAAS,CAAC,EAAE,CAAC,CAAC,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YAChG,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,IAAI,EAAE,EAAE,EAAE;gBACtD,OAAO,CAAC,IAAI,IAAI,EAAE,CAAC,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,KAAK,aAAL,KAAK,uBAAL,KAAK,CAAE,WAAW,EAAE,CAAC,CAAC;YACnE,CAAC,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,yBAAyB,CAAC,gBAAgB;aAC5C,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,UAAU,CAAC,CAAC;aAChC,SAAS,CAAC,GAAG,EAAE,CAAC,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC,CAAC;IAC/C,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,UAAU,CAAC,IAAI,EAAE,CAAC;QACvB,IAAI,CAAC,UAAU,CAAC,QAAQ,EAAE,CAAC;IAC7B,CAAC;IAEM,WAAW;QAChB,IAAI,CAAC,YAAY,CAAC,IAAI,EAAE,CAAC;QACzB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;IAC7B,CAAC;IAEM,YAAY;QACjB,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAC5B,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC,gBAAgB,CAAC,CAAC;QAChD,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;QAC/B,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;IAClC,CAAC;IAEM,cAAc,CAAC,GAAW;QAC/B,MAAM,OAAO,GAAG,IAAI,CAAC,cAAc,CAAC,OAAO,EAAE,CAAC;QAC9C,IAAI,GAAG,CAAC,UAAU,CAAC,YAAY,CAAC,EAAE;YAChC,MAAM,UAAU,GAAG,GAAG,CAAC,OAAO,CAAC,YAAY,EAAE,EAAE,CAAC,CAAC;YACjD,OAAO,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,UAAU,CAAC,UAAU,CAAC,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,OAAO,CAAC,UAAU,CAAC,WAAW,UAAU,EAAE,CAAC,CAAC,CAAC;SACnH;aAAM;YACL,OAAO,IAAI,CAAC,yBAAyB,CAAC,cAAc,CAAC,GAAG,CAAC,CAAC;SAC3D;IACH,CAAC;IAEM,IAAI;QACT,YAAY,CAAC,OAAO,CAAC,mBAAmB,EAAE,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC;QACxE,IAAI,CAAC,UAAU,EAAE,CAAC;QAClB,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QAChD,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAEM,iBAAiB,CAAC,KAA4B;QACnD,eAAe,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,aAAa,EAAE,KAAK,CAAC,YAAY,CAAC,CAAC;QACvE,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC;IACtC,CAAC;IAEM,UAAU,CAAC,MAAc;QAC9B,IAAI,IAAI,CAAC,aAAa,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE;YAClD,OAAO;SACR;QACD,IAAI,CAAC,aAAa,GAAG,MAAM,CAAC;QAC5B,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC;IACzC,CAAC;IAEM,UAAU;;QACf,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,EAAE;YAC5B,OAAO;SACR;QACD,IAAI,IAAI,CAAC,aAAa,EAAE;YACtB,IAAI,CAAC,aAAa,CAAC,IAAI,GAAG,MAAA,IAAI,CAAC,WAAW,CAAC,KAAK,0CAAE,IAAI,EAAE,CAAC;SAC1D;QACD,IAAI,CAAC,aAAa,GAAG,SAAS,CAAC;IACjC,CAAC;IAEM,cAAc;QACnB,IAAI,CAAC,uBAAuB,CAAC,IAAI,CAAC,CAAC;IACrC,CAAC;IAEO,iBAAiB,CAAC,SAAqB;QAC7C,MAAM,IAAI,GAAa,EAAE,CAAC;QAC1B,SAAS,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE;YAC3B,IAAI,CAAC,IAAI,CAAC,GAAG,MAAM,CAAC,IAAI,CAAC,QAAQ,CAAC,YAAY,IAAI,EAAE,CAAC,CAAC,CAAC;YACvD,IAAI,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,SAAS,CAAC,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;QACH,OAAO,IAAI,CAAC,IAAI,CAAC,CAAC;IACpB,CAAC;IAEO,UAAU;QAChB,MAAM,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC,EAAE,OAAO,EAAE,EAAE,EAAE,CAAC,OAAO,CAAC,CAAC,CAAC;QAEzE,IAAI,CAAC,sBAAsB,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;QAClD,IAAI,CAAC,mBAAmB,GAAG,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;IAC9C,CAAC;IAEO,uBAAuB,CAAC,cAAc,GAAG,KAAK;QACpD,IAAI,CAAC,OAAO,GAAG,EAAE,CAAC;QAClB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,oBAAoB,CAAC,MAAM,CAAC,CAAC,CAAC,IAAI,CAAC,oBAAoB,CAAC,CAAC,CAAC,cAAc,CAAC;QACpG,MAAM,aAAa,GAAa,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,OAAO,CAAC,mBAAmB,CAAC,IAAI,IAAI,CAAC,CAAC;QAC9F,MAAM,UAAU,GAAG,CAAC,GAAG,IAAI,CAAC,UAAU,CAAC,CAAC;QACxC,IAAI,CAAC,aAAa,CAAC,MAAM,IAAI,cAAc,EAAE;YAC3C,IAAI,CAAC,cAAc,CAAC,OAAO,CAAC,aAAa,CAAC,EAAE;gBAC1C,MAAM,cAAc,GAAG,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,aAAa,CAAC,GAAG,CAAC,CAAC;gBAE9E,MAAM,YAAY,GAAG,aAAa,CAAC,IAAI,CAAC,CAAC,EAAE,GAAG,EAAE,EAAE,EAAE,CAAC,GAAG,KAAK,aAAa,CAAC,GAAG,CAAC,CAAC;gBAChF,IAAI,YAAY,IAAI,CAAC,cAAc,EAAE;oBACnC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;iBACjC;qBAAM;oBACL,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;iBAClC;gBACD,IAAI,cAAc,IAAI,CAAC,EAAE;oBACvB,UAAU,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;iBACtC;YACH,CAAC,CAAC,CAAC;SACJ;aAAM;YACL,IAAI,CAAC,OAAO,GAAG;gBACb,GAAG,aAAa,CAAC,MAAM,CAAC,CAAC,MAAM,EAAE,MAAM,EAAE,EAAE;oBACzC,MAAM,cAAc,GAAG,UAAU,CAAC,SAAS,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,KAAK,MAAM,CAAC,GAAG,CAAC,CAAC;oBAEvE,IAAI,cAAc,IAAI,CAAC,EAAE;wBACvB,UAAU,CAAC,MAAM,CAAC,cAAc,EAAE,CAAC,CAAC,CAAC;qBACtC;oBAED,OAAO,CAAC,GAAG,MAAM,EAAE,MAAM,CAAC,CAAC;gBAC7B,CAAC,EAAE,EAAc,CAAC;aACnB,CAAC;SACH;QACD,IAAI,CAAC,OAAO,GAAG;YACb,GAAG,IAAI,CAAC,OAAO;YACf,GAAG,UAAU,CAAC,GAAG,CAAC,SAAS,CAAC,EAAE;gBAC5B,OAAO,EAAE,GAAG,EAAE,SAAS,EAAE,IAAI,EAAE,IAAI,CAAC,aAAa,CAAC,SAAS,CAAC,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;YACjF,CAAC,CAAC;SACH,CAAC;QACF,IAAI,CAAC,OAAO,GAAG,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;QACvC,IAAI,CAAC,aAAa,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QAChC,IAAI,CAAC,cAAc,EAAE;YACnB,IAAI,CAAC,gBAAgB,GAAG,SAAS,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC;YAChD,IAAI,CAAC,UAAU,EAAE,CAAC;SACnB;QACD,IAAI,CAAC,GAAG,CAAC,YAAY,EAAE,CAAC;IAC1B,CAAC;IAEO,aAAa,CAAC,GAAW;QAC/B,OAAO,GAAG,CAAC,OAAO,CAAC,cAAc,EAAE,EAAE,CAAC,CAAC;IACzC,CAAC;IAEO,QAAQ,CAAC,KAAyB;QACxC,OAAO,CAAC,CAAC,KAAK,IAAI,CAAC,CAAC,CAAC;IACvB,CAAC;;8GAlNU,gBAAgB;kGAAhB,gBAAgB,gGAChB,YAAY,gDCzBzB,2oKAsJA;4FD9Ha,gBAAgB;kBAN5B,SAAS;mBAAC;oBACT,QAAQ,EAAE,YAAY;oBACtB,WAAW,EAAE,0BAA0B;oBACvC,SAAS,EAAE,CAAC,0BAA0B,CAAC;oBACvC,eAAe,EAAE,uBAAuB,CAAC,MAAM;iBAChD;4OAEiC,YAAY;sBAA3C,SAAS;uBAAC,YAAY","sourcesContent":["import { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit, ViewChild } from '@angular/core';\nimport { FormControl, Validators } from '@angular/forms';\nimport { LineItem } from '@veloceapps/core';\nimport {\n  invalidCharactersValidator,\n  requiredValidator,\n  reservedIdentifierValidator,\n  standardCharSetWithSpaceRegExp,\n} from '@veloceapps/core/forms';\nimport { FlowAction, IntegrationState, Metric } from '@veloceapps/sdk/cms';\nimport { ContextService, MetricsCalculationService, QuoteDraftService } from '@veloceapps/sdk/core';\nimport { cloneDeep, isEqual, uniq } from 'lodash';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport { Subject, startWith } from 'rxjs';\nimport { takeUntil, tap } from 'rxjs/operators';\nimport { METRICS_STORAGE_KEY, defaultMetrics, effectiveMetricKeys } from './metrics.definitions';\n\n@Component({\n  selector: 'vl-metrics',\n  templateUrl: './metrics.component.html',\n  styleUrls: ['./metrics.component.scss'],\n  changeDetection: ChangeDetectionStrategy.OnPush,\n})\nexport class MetricsComponent implements OnInit, OnDestroy {\n  @ViewChild(OverlayPanel) public overlayPanel!: OverlayPanel;\n\n  public emptyStateMetrics = [\n    { name: 'MRR', value: 0 },\n    { name: 'E.MRR', value: 0 },\n    { name: 'NRR', value: 0 },\n    { name: 'E.NRR', value: 0 },\n  ];\n  public visibleSelectedMetrics: Metric[] = [];\n  public restSelectedMetrics: Metric[] = [];\n  public metrics: Metric[] = [];\n  public filteredMetrics: Metric[] = [];\n  public editingMetric: Metric | undefined;\n  public sidebarVisible = false;\n  public searchControl: FormControl = new FormControl('');\n  public nameControl: FormControl = new FormControl('', [\n    requiredValidator,\n    invalidCharactersValidator(standardCharSetWithSpaceRegExp),\n    reservedIdentifierValidator,\n    Validators.maxLength(80),\n  ]);\n\n  private defaultMetrics: Metric[] = [];\n  private storedDefaultMetrics: Metric[] = [];\n  private metricKeys: string[] = [];\n  private lastSavedMetrics: Metric[] = [];\n  private destroyed$ = new Subject<void>();\n\n  constructor(\n    private integrationState: IntegrationState,\n    private contextService: ContextService,\n    private quoteDraftService: QuoteDraftService,\n    private cdr: ChangeDetectorRef,\n    private metricsCalculationService: MetricsCalculationService,\n  ) {}\n\n  public ngOnInit(): void {\n    this.quoteDraftService.currentState$\n      .pipe(\n        takeUntil(this.destroyed$),\n        tap(lineItems => {\n          this.metricKeys = this.collectMetricKeys(lineItems).reduce((result, key) => {\n            result.push(key);\n            if (effectiveMetricKeys.includes(key)) {\n              result.push(`Effective_${key}`);\n            }\n            return result;\n          }, [] as string[]);\n\n          this.collectAvailableMetrics();\n        }),\n      )\n      .subscribe();\n\n    this.integrationState\n      .listen$(FlowAction.SET_DEFAULT_METRICS)\n      .pipe(\n        tap(({ metrics }) => {\n          if (isEqual(this.storedDefaultMetrics, metrics)) {\n            return;\n          }\n          this.storedDefaultMetrics = metrics;\n          this.collectAvailableMetrics();\n        }),\n        takeUntil(this.destroyed$),\n      )\n      .subscribe();\n\n    this.searchControl.valueChanges.pipe(takeUntil(this.destroyed$), startWith('')).subscribe(query => {\n      this.filteredMetrics = this.metrics.filter(({ name }) => {\n        return (name || '').toLowerCase().includes(query?.toLowerCase());\n      });\n    });\n\n    this.metricsCalculationService.onMetricsUpdate$\n      .pipe(takeUntil(this.destroyed$))\n      .subscribe(() => this.cdr.detectChanges());\n  }\n\n  public ngOnDestroy(): void {\n    this.destroyed$.next();\n    this.destroyed$.complete();\n  }\n\n  public openSidebar(): void {\n    this.overlayPanel.hide();\n    this.sidebarVisible = true;\n  }\n\n  public closeSidebar(): void {\n    this.sidebarVisible = false;\n    this.metrics = cloneDeep(this.lastSavedMetrics);\n    this.editingMetric = undefined;\n    this.searchControl.setValue('');\n  }\n\n  public getMetricValue(key: string): number {\n    const context = this.contextService.resolve();\n    if (key.startsWith('Effective_')) {\n      const correctKey = key.replace('Effective_', '');\n      return this.toNumber(context.properties[correctKey]) - this.toNumber(context.properties[`Previous${correctKey}`]);\n    } else {\n      return this.metricsCalculationService.getMetricValue(key);\n    }\n  }\n\n  public save(): void {\n    localStorage.setItem(METRICS_STORAGE_KEY, JSON.stringify(this.metrics));\n    this.setMetrics();\n    this.lastSavedMetrics = cloneDeep(this.metrics);\n    this.closeSidebar();\n  }\n\n  public changeMetricOrder(event: CdkDragDrop<string[]>): void {\n    moveItemInArray(this.metrics, event.previousIndex, event.currentIndex);\n    this.filteredMetrics = this.metrics;\n  }\n\n  public editMetric(metric: Metric): void {\n    if (this.editingMetric && this.nameControl.invalid) {\n      return;\n    }\n    this.editingMetric = metric;\n    this.nameControl.setValue(metric.name);\n  }\n\n  public saveMetric(): void {\n    if (this.nameControl.invalid) {\n      return;\n    }\n    if (this.editingMetric) {\n      this.editingMetric.name = this.nameControl.value?.trim();\n    }\n    this.editingMetric = undefined;\n  }\n\n  public resetToDefault(): void {\n    this.collectAvailableMetrics(true);\n  }\n\n  private collectMetricKeys(lineItems: LineItem[]): string[] {\n    const keys: string[] = [];\n    lineItems.forEach(lineItem => {\n      keys.push(...Object.keys(lineItem.totalMetrics || {}));\n      keys.push(...this.collectMetricKeys(lineItem.lineItems));\n    });\n    return uniq(keys);\n  }\n\n  private setMetrics(): void {\n    const metrics = cloneDeep(this.metrics.filter(({ visible }) => visible));\n\n    this.visibleSelectedMetrics = metrics.slice(0, 4);\n    this.restSelectedMetrics = metrics.slice(4);\n  }\n\n  private collectAvailableMetrics(resetToDefault = false): void {\n    this.metrics = [];\n    this.defaultMetrics = this.storedDefaultMetrics.length ? this.storedDefaultMetrics : defaultMetrics;\n    const storedMetrics: Metric[] = JSON.parse(localStorage.getItem(METRICS_STORAGE_KEY) || '[]');\n    const metricKeys = [...this.metricKeys];\n    if (!storedMetrics.length || resetToDefault) {\n      this.defaultMetrics.forEach(defaultMetric => {\n        const metricKeyIndex = metricKeys.findIndex(key => key === defaultMetric.key);\n\n        const storedMetric = storedMetrics.find(({ key }) => key === defaultMetric.key);\n        if (storedMetric && !resetToDefault) {\n          this.metrics.push(storedMetric);\n        } else {\n          this.metrics.push(defaultMetric);\n        }\n        if (metricKeyIndex >= 0) {\n          metricKeys.splice(metricKeyIndex, 1);\n        }\n      });\n    } else {\n      this.metrics = [\n        ...storedMetrics.reduce((result, metric) => {\n          const metricKeyIndex = metricKeys.findIndex(key => key === metric.key);\n\n          if (metricKeyIndex >= 0) {\n            metricKeys.splice(metricKeyIndex, 1);\n          }\n\n          return [...result, metric];\n        }, [] as Metric[]),\n      ];\n    }\n    this.metrics = [\n      ...this.metrics,\n      ...metricKeys.map(metricKey => {\n        return { key: metricKey, name: this.getMetricName(metricKey), visible: false };\n      }),\n    ];\n    this.metrics = cloneDeep(this.metrics);\n    this.searchControl.setValue('');\n    if (!resetToDefault) {\n      this.lastSavedMetrics = cloneDeep(this.metrics);\n      this.setMetrics();\n    }\n    this.cdr.markForCheck();\n  }\n\n  private getMetricName(key: string): string {\n    return key.replace(/VDM_|Total_/g, '');\n  }\n\n  private toNumber(value: string | undefined): number {\n    return +(value || 0);\n  }\n}\n","<div class=\"header-metrics\" (click)=\"overlayPanel.toggle($event)\">\n  <div *ngFor=\"let metric of visibleSelectedMetrics\" class=\"metric\">\n    <span class=\"name\" [vlHiddenTextTooltip]=\"metric.name\" tooltipStyleClass=\"metric-tooltip\" tooltipPosition=\"bottom\">\n      {{ metric.name }}\n    </span>\n    <span class=\"filler\"></span>\n    <span\n      class=\"value\"\n      vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n      tooltipStyleClass=\"metric-tooltip\"\n      tooltipPosition=\"bottom\"\n    >\n      ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n    </span>\n  </div>\n\n  <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n    <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n      <span\n        class=\"name\"\n        [vlHiddenTextTooltip]=\"metric.name\"\n        tooltipStyleClass=\"metric-tooltip\"\n        tooltipPosition=\"bottom\"\n      >\n        {{ metric.name }}\n      </span>\n      <span class=\"filler\"></span>\n      <span\n        class=\"value\"\n        vlHiddenTextTooltip=\"${{ metric.value | number: '1.2-2' }}\"\n        tooltipStyleClass=\"metric-tooltip\"\n        tooltipPosition=\"bottom\"\n      >\n        ${{ metric.value | number: '1.2-2' }}\n      </span>\n    </div>\n  </ng-container>\n</div>\n\n<p-overlayPanel\n  styleClass=\"metrics-overlay-container center no-padding\"\n  showTransitionOptions=\"0ms\"\n  hideTransitionOptions=\"0ms\"\n>\n  <ng-template pTemplate>\n    <div class=\"overlay-metrics\">\n      <div *ngFor=\"let metric of restSelectedMetrics\" class=\"metric\">\n        <span\n          class=\"name\"\n          [vlHiddenTextTooltip]=\"metric.name\"\n          tooltipStyleClass=\"metric-tooltip\"\n          tooltipPosition=\"bottom\"\n        >\n          {{ metric.name }}\n        </span>\n        <span class=\"filler\"></span>\n        <span\n          class=\"value\"\n          vlHiddenTextTooltip=\"${{ getMetricValue(metric.key) | number: '1.2-2' }}\"\n          tooltipStyleClass=\"metric-tooltip\"\n          tooltipPosition=\"bottom\"\n        >\n          ${{ getMetricValue(metric.key) | number: '1.2-2' }}\n        </span>\n      </div>\n    </div>\n    <button class=\"config\" (click)=\"openSidebar()\">\n      Metrics Config\n      <i class=\"vl-icon vl-icon-chevron-down\"></i>\n    </button>\n  </ng-template>\n</p-overlayPanel>\n\n<p-sidebar\n  [visible]=\"sidebarVisible\"\n  (visibleChange)=\"closeSidebar()\"\n  position=\"right\"\n  [baseZIndex]=\"1000\"\n  [style]=\"{ width: '360px' }\"\n>\n  <div class=\"container\">\n    <div class=\"header\">\n      Quote Metrics Config\n      <i class=\"vl-icon vl-icon-close\" (click)=\"closeSidebar()\"></i>\n    </div>\n\n    <div class=\"search-container\">\n      <input\n        data-test-id=\"search\"\n        [formControl]=\"searchControl\"\n        pInputText\n        placeholder=\"Search for metric…\"\n        class=\"w-full\"\n      />\n    </div>\n\n    <div class=\"content\">\n      <div class=\"sidebar-metrics\" cdkDropList (cdkDropListDropped)=\"changeMetricOrder($event)\">\n        <div\n          *ngFor=\"let metric of filteredMetrics\"\n          class=\"sidebar-metric\"\n          [class.edit]=\"metric.key === editingMetric?.key\"\n          cdkDrag\n          [cdkDragDisabled]=\"!!searchControl.value\"\n        >\n          <div class=\"drag-icon\">\n            <i class=\"vl-icon vl-icon-reorder\" cdkDragHandle></i>\n          </div>\n          <ng-container\n            [ngTemplateOutlet]=\"metricTemplate\"\n            [ngTemplateOutletContext]=\"{ metric: metric }\"\n          ></ng-container>\n        </div>\n      </div>\n\n      <div class=\"empty-state\" *ngIf=\"!filteredMetrics.length\">There are no items matching your search criteria</div>\n    </div>\n    <div class=\"footer\">\n      <p-button\n        label=\"Reset to default\"\n        styleClass=\"p-button-outlined p-button-sm\"\n        (onClick)=\"resetToDefault()\"\n      ></p-button>\n      <p-button label=\"Save\" styleClass=\"p-button p-button-filled p-button-sm\" (onClick)=\"save()\"></p-button>\n    </div>\n  </div>\n</p-sidebar>\n\n<ng-template #metricTemplate let-metric=\"metric\">\n  <div *ngIf=\"editingMetric?.key !== metric.key\" class=\"preview-state\">\n    <p-checkbox\n      [ngModel]=\"metric.visible\"\n      (ngModelChange)=\"metric.visible = !metric.visible\"\n      [binary]=\"true\"\n      checkboxIcon=\"vl-icon vl-icon-checkmark\"\n    ></p-checkbox>\n    <div class=\"title\">{{ metric.name }}</div>\n  </div>\n\n  <div *ngIf=\"editingMetric?.key === metric.key\" class=\"edit-state\">\n    <input data-test-id=\"name\" [formControl]=\"nameControl\" pInputText placeholder=\"Metric name\" class=\"w-full\" />\n    <vl-error-tooltip [tooltip]=\"nameControl | error: 'name'\" [visible]=\"!!nameControl.errors\"></vl-error-tooltip>\n  </div>\n  <div *ngIf=\"editingMetric?.key !== metric.key\" (click)=\"editMetric(metric)\" class=\"action edit\">\n    <i class=\"vl-icon vl-icon-edit-sm\"></i>\n  </div>\n  <div *ngIf=\"editingMetric?.key === metric.key\" (click)=\"saveMetric()\" class=\"action save\">\n    <i class=\"vl-icon vl-icon-checkmark\"></i>\n  </div>\n</ng-template>\n"]}
|
@@ -1,9 +1,9 @@
|
|
1
1
|
export const defaultMetrics = [
|
2
|
-
{ key: '
|
3
|
-
{ key: '
|
4
|
-
{ key: '
|
5
|
-
{ key: '
|
2
|
+
{ key: 'Total_VDM_MRR', name: 'MRR', visible: true },
|
3
|
+
{ key: 'Effective_Total_VDM_MRR', name: 'E.MRR', visible: true },
|
4
|
+
{ key: 'Total_VDM_NRR', name: 'NRR', visible: true },
|
5
|
+
{ key: 'Effective_Total_VDM_NRR', name: 'E.NRR', visible: true },
|
6
6
|
];
|
7
|
-
export const effectiveMetricKeys = ['
|
7
|
+
export const effectiveMetricKeys = ['Total_VDM_MRR', 'Total_VDM_NRR'];
|
8
8
|
export const METRICS_STORAGE_KEY = 'vl-metrics';
|
9
|
-
//# sourceMappingURL=data:application/json;base64,
|
9
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljcy5kZWZpbml0aW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvc2RrL3NyYy9jb21wb25lbnRzL2hlYWRlci9tZXRyaWNzL21ldHJpY3MuZGVmaW5pdGlvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxDQUFDLE1BQU0sY0FBYyxHQUFhO0lBQ3RDLEVBQUUsR0FBRyxFQUFFLGVBQWUsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUU7SUFDcEQsRUFBRSxHQUFHLEVBQUUseUJBQXlCLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFO0lBQ2hFLEVBQUUsR0FBRyxFQUFFLGVBQWUsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUU7SUFDcEQsRUFBRSxHQUFHLEVBQUUseUJBQXlCLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFO0NBQ2pFLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxDQUFDLGVBQWUsRUFBRSxlQUFlLENBQUMsQ0FBQztBQUV0RSxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxZQUFZLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNZXRyaWMgfSBmcm9tICdAdmVsb2NlYXBwcy9zZGsvY21zJztcblxuZXhwb3J0IGNvbnN0IGRlZmF1bHRNZXRyaWNzOiBNZXRyaWNbXSA9IFtcbiAgeyBrZXk6ICdUb3RhbF9WRE1fTVJSJywgbmFtZTogJ01SUicsIHZpc2libGU6IHRydWUgfSxcbiAgeyBrZXk6ICdFZmZlY3RpdmVfVG90YWxfVkRNX01SUicsIG5hbWU6ICdFLk1SUicsIHZpc2libGU6IHRydWUgfSxcbiAgeyBrZXk6ICdUb3RhbF9WRE1fTlJSJywgbmFtZTogJ05SUicsIHZpc2libGU6IHRydWUgfSxcbiAgeyBrZXk6ICdFZmZlY3RpdmVfVG90YWxfVkRNX05SUicsIG5hbWU6ICdFLk5SUicsIHZpc2libGU6IHRydWUgfSxcbl07XG5cbmV4cG9ydCBjb25zdCBlZmZlY3RpdmVNZXRyaWNLZXlzID0gWydUb3RhbF9WRE1fTVJSJywgJ1RvdGFsX1ZETV9OUlInXTtcblxuZXhwb3J0IGNvbnN0IE1FVFJJQ1NfU1RPUkFHRV9LRVkgPSAndmwtbWV0cmljcyc7XG4iXX0=
|