@veloceapps/sdk 5.0.8 → 5.0.10

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.
Files changed (38) hide show
  1. package/bundles/veloce-sdk-cms.umd.js +97 -45
  2. package/bundles/veloce-sdk-cms.umd.js.map +1 -1
  3. package/bundles/veloce-sdk.umd.js +457 -195
  4. package/bundles/veloce-sdk.umd.js.map +1 -1
  5. package/cms/cms.actions.d.ts +4 -2
  6. package/cms/services/launcher.service.d.ts +3 -3
  7. package/cms/types/common.types.d.ts +5 -0
  8. package/cms/types/compilation.types.d.ts +6 -0
  9. package/cms/types/index.d.ts +2 -0
  10. package/cms/types/metrics.types.d.ts +5 -0
  11. package/cms/utils/elements-resolver.d.ts +4 -1
  12. package/cms/vendor-map.d.ts +1 -0
  13. package/esm2015/cms/cms.actions.js +6 -1
  14. package/esm2015/cms/components/preview/preview.component.js +7 -4
  15. package/esm2015/cms/services/launcher.service.js +10 -10
  16. package/esm2015/cms/types/common.types.js +1 -1
  17. package/esm2015/cms/types/compilation.types.js +2 -0
  18. package/esm2015/cms/types/index.js +3 -1
  19. package/esm2015/cms/types/metrics.types.js +2 -0
  20. package/esm2015/cms/utils/element.utils.js +6 -1
  21. package/esm2015/cms/utils/elements-resolver.js +55 -24
  22. package/esm2015/src/components/header/header.component.js +12 -19
  23. package/esm2015/src/components/header/header.module.js +6 -2
  24. package/esm2015/src/components/header/metrics/index.js +2 -0
  25. package/esm2015/src/components/header/metrics/metrics.component.js +216 -0
  26. package/esm2015/src/components/header/metrics/metrics.definitions.js +9 -0
  27. package/esm2015/src/components/header/metrics/metrics.module.js +64 -0
  28. package/fesm2015/veloce-sdk-cms.js +80 -35
  29. package/fesm2015/veloce-sdk-cms.js.map +1 -1
  30. package/fesm2015/veloce-sdk.js +341 -85
  31. package/fesm2015/veloce-sdk.js.map +1 -1
  32. package/package.json +1 -1
  33. package/src/components/header/header.component.d.ts +0 -8
  34. package/src/components/header/header.module.d.ts +2 -1
  35. package/src/components/header/metrics/index.d.ts +1 -0
  36. package/src/components/header/metrics/metrics.component.d.ts +49 -0
  37. package/src/components/header/metrics/metrics.definitions.d.ts +4 -0
  38. package/src/components/header/metrics/metrics.module.d.ts +17 -0
@@ -0,0 +1,2 @@
1
+ export * from './metrics.module';
2
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3Nkay9zcmMvY29tcG9uZW50cy9oZWFkZXIvbWV0cmljcy9pbmRleC50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxjQUFjLGtCQUFrQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiZXhwb3J0ICogZnJvbSAnLi9tZXRyaWNzLm1vZHVsZSc7XG4iXX0=
@@ -0,0 +1,216 @@
1
+ import { ChangeDetectionStrategy, Component, ViewChild } from '@angular/core';
2
+ import { FormControl, Validators } from '@angular/forms';
3
+ import { moveItemInArray } from '@angular/cdk/drag-drop';
4
+ import { OverlayPanel } from 'primeng/overlaypanel';
5
+ import { startWith, Subject } from 'rxjs';
6
+ import { takeUntil, tap } from 'rxjs/operators';
7
+ import { cloneDeep, isEqual, uniq } from 'lodash';
8
+ import { FlowAction } from '@veloce/sdk/cms';
9
+ import { defaultMetrics, effectiveMetricKeys, METRICS_STORAGE_KEY } from './metrics.definitions';
10
+ import { invalidCharactersValidator, requiredValidator, reservedIdentifierValidator, standardCharSetWithSpaceRegExp, } from '@veloce/core/forms';
11
+ import * as i0 from "@angular/core";
12
+ import * as i1 from "@veloce/sdk/cms";
13
+ import * as i2 from "@veloce/sdk/core";
14
+ import * as i3 from "primeng/overlaypanel";
15
+ import * as i4 from "primeng/sidebar";
16
+ import * as i5 from "primeng/button";
17
+ import * as i6 from "primeng/checkbox";
18
+ import * as i7 from "@veloce/components";
19
+ import * as i8 from "@angular/common";
20
+ import * as i9 from "primeng/api";
21
+ import * as i10 from "@angular/forms";
22
+ import * as i11 from "primeng/inputtext";
23
+ import * as i12 from "@angular/cdk/drag-drop";
24
+ export class MetricsComponent {
25
+ constructor(integrationState, contextService, quoteDraftService, cdr) {
26
+ this.integrationState = integrationState;
27
+ this.contextService = contextService;
28
+ this.quoteDraftService = quoteDraftService;
29
+ this.cdr = cdr;
30
+ this.emptyStateMetrics = [
31
+ { name: 'MRR', value: 0 },
32
+ { name: 'E.MRR', value: 0 },
33
+ { name: 'NRR', value: 0 },
34
+ { name: 'E.NRR', value: 0 },
35
+ ];
36
+ this.visibleSelectedMetrics = [];
37
+ this.restSelectedMetrics = [];
38
+ this.metrics = [];
39
+ this.filteredMetrics = [];
40
+ this.sidebarVisible = false;
41
+ this.searchControl = new FormControl('');
42
+ this.nameControl = new FormControl('', [
43
+ requiredValidator,
44
+ invalidCharactersValidator(standardCharSetWithSpaceRegExp),
45
+ reservedIdentifierValidator,
46
+ Validators.maxLength(80),
47
+ ]);
48
+ this.defaultMetrics = [];
49
+ this.storedDefaultMetrics = [];
50
+ this.metricKeys = [];
51
+ this.lastSavedMetrics = [];
52
+ this.destroyed$ = new Subject();
53
+ }
54
+ ngOnInit() {
55
+ this.quoteDraftService.currentState$
56
+ .pipe(takeUntil(this.destroyed$), tap(lineItems => {
57
+ this.metricKeys = this.collectMetricKeys(lineItems).reduce((result, key) => {
58
+ result.push(key);
59
+ if (effectiveMetricKeys.includes(key)) {
60
+ result.push(`Effective_${key}`);
61
+ }
62
+ return result;
63
+ }, []);
64
+ this.collectAvailableMetrics();
65
+ }))
66
+ .subscribe();
67
+ this.integrationState
68
+ .listen$(FlowAction.SET_DEFAULT_METRICS)
69
+ .pipe(tap(({ metrics }) => {
70
+ if (isEqual(this.storedDefaultMetrics, metrics)) {
71
+ return;
72
+ }
73
+ this.storedDefaultMetrics = metrics;
74
+ this.collectAvailableMetrics();
75
+ }), takeUntil(this.destroyed$))
76
+ .subscribe();
77
+ this.searchControl.valueChanges.pipe(takeUntil(this.destroyed$), startWith('')).subscribe(query => {
78
+ this.filteredMetrics = this.metrics.filter(({ name }) => {
79
+ return (name || '').toLowerCase().includes(query === null || query === void 0 ? void 0 : query.toLowerCase());
80
+ });
81
+ });
82
+ }
83
+ ngOnDestroy() {
84
+ this.destroyed$.next();
85
+ this.destroyed$.complete();
86
+ }
87
+ openSidebar() {
88
+ this.overlayPanel.hide();
89
+ this.sidebarVisible = true;
90
+ }
91
+ closeSidebar() {
92
+ this.sidebarVisible = false;
93
+ this.metrics = cloneDeep(this.lastSavedMetrics);
94
+ this.editingMetric = undefined;
95
+ this.searchControl.setValue('');
96
+ }
97
+ getMetricValue(key) {
98
+ const context = this.contextService.resolve();
99
+ if (key.startsWith('Effective_')) {
100
+ const correctKey = key.replace('Effective_', '');
101
+ return this.toNumber(context.properties[correctKey]) - this.toNumber(context.properties[`Previous${correctKey}`]);
102
+ }
103
+ else {
104
+ return this.toNumber(context.properties[key]);
105
+ }
106
+ }
107
+ save() {
108
+ localStorage.setItem(METRICS_STORAGE_KEY, JSON.stringify(this.metrics));
109
+ this.setMetrics();
110
+ this.lastSavedMetrics = cloneDeep(this.metrics);
111
+ this.closeSidebar();
112
+ }
113
+ changeMetricOrder(event) {
114
+ moveItemInArray(this.metrics, event.previousIndex, event.currentIndex);
115
+ this.filteredMetrics = this.metrics;
116
+ }
117
+ editMetric(metric) {
118
+ if (this.editingMetric && this.nameControl.invalid) {
119
+ return;
120
+ }
121
+ this.editingMetric = metric;
122
+ this.nameControl.setValue(metric.name);
123
+ }
124
+ saveMetric() {
125
+ var _a;
126
+ if (this.nameControl.invalid) {
127
+ return;
128
+ }
129
+ if (this.editingMetric) {
130
+ this.editingMetric.name = (_a = this.nameControl.value) === null || _a === void 0 ? void 0 : _a.trim();
131
+ }
132
+ this.editingMetric = undefined;
133
+ }
134
+ resetToDefault() {
135
+ this.collectAvailableMetrics(true);
136
+ }
137
+ collectMetricKeys(lineItems) {
138
+ const keys = [];
139
+ lineItems.forEach(lineItem => {
140
+ keys.push(...Object.keys(lineItem.totalMetrics || {}));
141
+ keys.push(...this.collectMetricKeys(lineItem.lineItems));
142
+ });
143
+ return uniq(keys);
144
+ }
145
+ setMetrics() {
146
+ const metrics = cloneDeep(this.metrics.filter(({ visible }) => visible));
147
+ this.visibleSelectedMetrics = metrics.slice(0, 4);
148
+ this.restSelectedMetrics = metrics.slice(4);
149
+ }
150
+ collectAvailableMetrics(resetToDefault = false) {
151
+ this.metrics = [];
152
+ this.defaultMetrics = this.storedDefaultMetrics.length ? this.storedDefaultMetrics : defaultMetrics;
153
+ const storedMetrics = JSON.parse(localStorage.getItem(METRICS_STORAGE_KEY) || '[]');
154
+ const metricKeys = [...this.metricKeys];
155
+ if (!storedMetrics.length || resetToDefault) {
156
+ this.defaultMetrics.forEach(defaultMetric => {
157
+ const metricKeyIndex = metricKeys.findIndex(key => key === defaultMetric.key);
158
+ const storedMetric = storedMetrics.find(({ key }) => key === defaultMetric.key);
159
+ if (storedMetric && !resetToDefault) {
160
+ this.metrics.push(storedMetric);
161
+ }
162
+ else {
163
+ this.metrics.push(defaultMetric);
164
+ }
165
+ if (metricKeyIndex >= 0) {
166
+ metricKeys.splice(metricKeyIndex, 1);
167
+ }
168
+ });
169
+ }
170
+ else {
171
+ this.metrics = [
172
+ ...storedMetrics.reduce((result, metric) => {
173
+ const metricKeyIndex = metricKeys.findIndex(key => key === metric.key);
174
+ if (metricKeyIndex >= 0) {
175
+ metricKeys.splice(metricKeyIndex, 1);
176
+ }
177
+ return [...result, metric];
178
+ }, []),
179
+ ];
180
+ }
181
+ this.metrics = [
182
+ ...this.metrics,
183
+ ...metricKeys.map(metricKey => {
184
+ return { key: metricKey, name: this.getMetricName(metricKey), visible: false };
185
+ }),
186
+ ];
187
+ this.metrics = cloneDeep(this.metrics);
188
+ this.searchControl.setValue('');
189
+ if (!resetToDefault) {
190
+ this.lastSavedMetrics = cloneDeep(this.metrics);
191
+ this.setMetrics();
192
+ }
193
+ this.cdr.markForCheck();
194
+ }
195
+ getMetricName(key) {
196
+ return key.replace(/VDM_|Total_/g, '');
197
+ }
198
+ toNumber(value) {
199
+ return +(value || 0);
200
+ }
201
+ }
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 });
203
+ 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\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n <span class=\"filler\"></span>\n <span class=\"value\">${{ getMetricValue(metric.key) | number: '1.2-2' }}</span>\n </div>\n\n <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n <span class=\"name\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n <span class=\"filler\"></span>\n <span class=\"value\">${{ metric.value | number: '1.2-2' }}</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 class=\"name\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n <span class=\"filler\"></span>\n <span class=\"value\">${{ getMetricValue(metric.key) | number: '1.2-2' }}</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{overflow:hidden;white-space:nowrap;text-overflow:ellipsis}:host .header-metrics .metric .value{font-weight:600}: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{width:160px}::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{overflow:hidden;white-space:nowrap;text-overflow:ellipsis}::ng-deep .metrics-overlay-container .metric .value{font-weight:600}::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)}\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: 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
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsComponent, decorators: [{
205
+ type: Component,
206
+ args: [{
207
+ selector: 'vl-metrics',
208
+ templateUrl: './metrics.component.html',
209
+ styleUrls: ['./metrics.component.scss'],
210
+ changeDetection: ChangeDetectionStrategy.OnPush,
211
+ }]
212
+ }], ctorParameters: function () { return [{ type: i1.IntegrationState }, { type: i2.ContextService }, { type: i2.QuoteDraftService }, { type: i0.ChangeDetectorRef }]; }, propDecorators: { overlayPanel: [{
213
+ type: ViewChild,
214
+ args: [OverlayPanel]
215
+ }] } });
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 { ChangeDetectionStrategy, ChangeDetectorRef, Component, OnDestroy, OnInit, ViewChild } from '@angular/core';\nimport { FormControl, Validators } from '@angular/forms';\nimport { CdkDragDrop, moveItemInArray } from '@angular/cdk/drag-drop';\nimport { OverlayPanel } from 'primeng/overlaypanel';\nimport { startWith, Subject } from 'rxjs';\nimport { takeUntil, tap } from 'rxjs/operators';\nimport { cloneDeep, isEqual, uniq } from 'lodash';\nimport { ContextService, QuoteDraftService } from '@veloce/sdk/core';\nimport { LineItem } from '@veloce/core';\nimport { FlowAction, IntegrationState, Metric } from '@veloce/sdk/cms';\nimport { defaultMetrics, effectiveMetricKeys, METRICS_STORAGE_KEY } from './metrics.definitions';\nimport {\n  invalidCharactersValidator,\n  requiredValidator,\n  reservedIdentifierValidator,\n  standardCharSetWithSpaceRegExp,\n} from '@veloce/core/forms';\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) 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() {\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\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n    <span class=\"filler\"></span>\n    <span class=\"value\">${{ getMetricValue(metric.key) | number: '1.2-2' }}</span>\n  </div>\n\n  <ng-container *ngIf=\"!visibleSelectedMetrics.length\">\n    <div *ngFor=\"let metric of emptyStateMetrics\" class=\"metric empty-state-metric\">\n      <span class=\"name\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n      <span class=\"filler\"></span>\n      <span class=\"value\">${{ metric.value | number: '1.2-2' }}</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 class=\"name\" [attr.title]=\"metric.name\">{{ metric.name }}</span>\n        <span class=\"filler\"></span>\n        <span class=\"value\">${{ getMetricValue(metric.key) | number: '1.2-2' }}</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"]}
@@ -0,0 +1,9 @@
1
+ export const defaultMetrics = [
2
+ { key: 'VDM_Total_MRR', name: 'MRR', visible: true },
3
+ { key: 'Effective_VDM_Total_MRR', name: 'E.MRR', visible: true },
4
+ { key: 'VDM_Total_NRR', name: 'NRR', visible: true },
5
+ { key: 'Effective_VDM_Total_NRR', name: 'E.NRR', visible: true },
6
+ ];
7
+ export const effectiveMetricKeys = ['VDM_Total_MRR', 'VDM_Total_NRR'];
8
+ export const METRICS_STORAGE_KEY = 'vl-metrics';
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljcy5kZWZpbml0aW9ucy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL2xpYnMvc2RrL3NyYy9jb21wb25lbnRzL2hlYWRlci9tZXRyaWNzL21ldHJpY3MuZGVmaW5pdGlvbnMudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsTUFBTSxDQUFDLE1BQU0sY0FBYyxHQUFhO0lBQ3RDLEVBQUUsR0FBRyxFQUFFLGVBQWUsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUU7SUFDcEQsRUFBRSxHQUFHLEVBQUUseUJBQXlCLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFO0lBQ2hFLEVBQUUsR0FBRyxFQUFFLGVBQWUsRUFBRSxJQUFJLEVBQUUsS0FBSyxFQUFFLE9BQU8sRUFBRSxJQUFJLEVBQUU7SUFDcEQsRUFBRSxHQUFHLEVBQUUseUJBQXlCLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFO0NBQ2pFLENBQUM7QUFFRixNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxDQUFDLGVBQWUsRUFBRSxlQUFlLENBQUMsQ0FBQztBQUV0RSxNQUFNLENBQUMsTUFBTSxtQkFBbUIsR0FBRyxZQUFZLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBNZXRyaWMgfSBmcm9tICdAdmVsb2NlL3Nkay9jbXMnO1xuXG5leHBvcnQgY29uc3QgZGVmYXVsdE1ldHJpY3M6IE1ldHJpY1tdID0gW1xuICB7IGtleTogJ1ZETV9Ub3RhbF9NUlInLCBuYW1lOiAnTVJSJywgdmlzaWJsZTogdHJ1ZSB9LFxuICB7IGtleTogJ0VmZmVjdGl2ZV9WRE1fVG90YWxfTVJSJywgbmFtZTogJ0UuTVJSJywgdmlzaWJsZTogdHJ1ZSB9LFxuICB7IGtleTogJ1ZETV9Ub3RhbF9OUlInLCBuYW1lOiAnTlJSJywgdmlzaWJsZTogdHJ1ZSB9LFxuICB7IGtleTogJ0VmZmVjdGl2ZV9WRE1fVG90YWxfTlJSJywgbmFtZTogJ0UuTlJSJywgdmlzaWJsZTogdHJ1ZSB9LFxuXTtcblxuZXhwb3J0IGNvbnN0IGVmZmVjdGl2ZU1ldHJpY0tleXMgPSBbJ1ZETV9Ub3RhbF9NUlInLCAnVkRNX1RvdGFsX05SUiddO1xuXG5leHBvcnQgY29uc3QgTUVUUklDU19TVE9SQUdFX0tFWSA9ICd2bC1tZXRyaWNzJztcbiJdfQ==
@@ -0,0 +1,64 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { NgModule } from '@angular/core';
3
+ import { FormsModule, ReactiveFormsModule } from '@angular/forms';
4
+ import { DragDropModule } from '@angular/cdk/drag-drop';
5
+ import { ButtonModule } from 'primeng/button';
6
+ import { OverlayPanelModule } from 'primeng/overlaypanel';
7
+ import { VirtualScrollerModule } from 'primeng/virtualscroller';
8
+ import { SidebarModule } from 'primeng/sidebar';
9
+ import { InputTextModule } from 'primeng/inputtext';
10
+ import { ErrorTooltipModule, LetDirectiveModule } from '@veloce/components';
11
+ import { MetricsComponent } from './metrics.component';
12
+ import { CheckboxModule } from 'primeng/checkbox';
13
+ import * as i0 from "@angular/core";
14
+ export class MetricsModule {
15
+ }
16
+ MetricsModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
17
+ MetricsModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsModule, declarations: [MetricsComponent], imports: [CommonModule,
18
+ FormsModule,
19
+ ReactiveFormsModule,
20
+ DragDropModule,
21
+ ButtonModule,
22
+ OverlayPanelModule,
23
+ SidebarModule,
24
+ InputTextModule,
25
+ CheckboxModule,
26
+ LetDirectiveModule,
27
+ VirtualScrollerModule,
28
+ ErrorTooltipModule], exports: [MetricsComponent] });
29
+ MetricsModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsModule, imports: [[
30
+ CommonModule,
31
+ FormsModule,
32
+ ReactiveFormsModule,
33
+ DragDropModule,
34
+ ButtonModule,
35
+ OverlayPanelModule,
36
+ SidebarModule,
37
+ InputTextModule,
38
+ CheckboxModule,
39
+ LetDirectiveModule,
40
+ VirtualScrollerModule,
41
+ ErrorTooltipModule,
42
+ ]] });
43
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImport: i0, type: MetricsModule, decorators: [{
44
+ type: NgModule,
45
+ args: [{
46
+ declarations: [MetricsComponent],
47
+ imports: [
48
+ CommonModule,
49
+ FormsModule,
50
+ ReactiveFormsModule,
51
+ DragDropModule,
52
+ ButtonModule,
53
+ OverlayPanelModule,
54
+ SidebarModule,
55
+ InputTextModule,
56
+ CheckboxModule,
57
+ LetDirectiveModule,
58
+ VirtualScrollerModule,
59
+ ErrorTooltipModule,
60
+ ],
61
+ exports: [MetricsComponent],
62
+ }]
63
+ }] });
64
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibWV0cmljcy5tb2R1bGUuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9saWJzL3Nkay9zcmMvY29tcG9uZW50cy9oZWFkZXIvbWV0cmljcy9tZXRyaWNzLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsWUFBWSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDL0MsT0FBTyxFQUFFLFFBQVEsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUN6QyxPQUFPLEVBQUUsV0FBVyxFQUFFLG1CQUFtQixFQUFFLE1BQU0sZ0JBQWdCLENBQUM7QUFDbEUsT0FBTyxFQUFFLGNBQWMsRUFBRSxNQUFNLHdCQUF3QixDQUFDO0FBQ3hELE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxnQkFBZ0IsQ0FBQztBQUM5QyxPQUFPLEVBQUUsa0JBQWtCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUMxRCxPQUFPLEVBQUUscUJBQXFCLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQztBQUNoRSxPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0saUJBQWlCLENBQUM7QUFDaEQsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLG1CQUFtQixDQUFDO0FBQ3BELE9BQU8sRUFBRSxrQkFBa0IsRUFBRSxrQkFBa0IsRUFBRSxNQUFNLG9CQUFvQixDQUFDO0FBQzVFLE9BQU8sRUFBRSxnQkFBZ0IsRUFBRSxNQUFNLHFCQUFxQixDQUFDO0FBQ3ZELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSxrQkFBa0IsQ0FBQzs7QUFvQmxELE1BQU0sT0FBTyxhQUFhOzsyR0FBYixhQUFhOzRHQUFiLGFBQWEsaUJBakJULGdCQUFnQixhQUU3QixZQUFZO1FBQ1osV0FBVztRQUNYLG1CQUFtQjtRQUNuQixjQUFjO1FBQ2QsWUFBWTtRQUNaLGtCQUFrQjtRQUNsQixhQUFhO1FBQ2IsZUFBZTtRQUNmLGNBQWM7UUFDZCxrQkFBa0I7UUFDbEIscUJBQXFCO1FBQ3JCLGtCQUFrQixhQUVWLGdCQUFnQjs0R0FFZixhQUFhLFlBaEJmO1lBQ1AsWUFBWTtZQUNaLFdBQVc7WUFDWCxtQkFBbUI7WUFDbkIsY0FBYztZQUNkLFlBQVk7WUFDWixrQkFBa0I7WUFDbEIsYUFBYTtZQUNiLGVBQWU7WUFDZixjQUFjO1lBQ2Qsa0JBQWtCO1lBQ2xCLHFCQUFxQjtZQUNyQixrQkFBa0I7U0FDbkI7NEZBR1UsYUFBYTtrQkFsQnpCLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFLENBQUMsZ0JBQWdCLENBQUM7b0JBQ2hDLE9BQU8sRUFBRTt3QkFDUCxZQUFZO3dCQUNaLFdBQVc7d0JBQ1gsbUJBQW1CO3dCQUNuQixjQUFjO3dCQUNkLFlBQVk7d0JBQ1osa0JBQWtCO3dCQUNsQixhQUFhO3dCQUNiLGVBQWU7d0JBQ2YsY0FBYzt3QkFDZCxrQkFBa0I7d0JBQ2xCLHFCQUFxQjt3QkFDckIsa0JBQWtCO3FCQUNuQjtvQkFDRCxPQUFPLEVBQUUsQ0FBQyxnQkFBZ0IsQ0FBQztpQkFDNUIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21tb25Nb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24nO1xuaW1wb3J0IHsgTmdNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEZvcm1zTW9kdWxlLCBSZWFjdGl2ZUZvcm1zTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuaW1wb3J0IHsgRHJhZ0Ryb3BNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jZGsvZHJhZy1kcm9wJztcbmltcG9ydCB7IEJ1dHRvbk1vZHVsZSB9IGZyb20gJ3ByaW1lbmcvYnV0dG9uJztcbmltcG9ydCB7IE92ZXJsYXlQYW5lbE1vZHVsZSB9IGZyb20gJ3ByaW1lbmcvb3ZlcmxheXBhbmVsJztcbmltcG9ydCB7IFZpcnR1YWxTY3JvbGxlck1vZHVsZSB9IGZyb20gJ3ByaW1lbmcvdmlydHVhbHNjcm9sbGVyJztcbmltcG9ydCB7IFNpZGViYXJNb2R1bGUgfSBmcm9tICdwcmltZW5nL3NpZGViYXInO1xuaW1wb3J0IHsgSW5wdXRUZXh0TW9kdWxlIH0gZnJvbSAncHJpbWVuZy9pbnB1dHRleHQnO1xuaW1wb3J0IHsgRXJyb3JUb29sdGlwTW9kdWxlLCBMZXREaXJlY3RpdmVNb2R1bGUgfSBmcm9tICdAdmVsb2NlL2NvbXBvbmVudHMnO1xuaW1wb3J0IHsgTWV0cmljc0NvbXBvbmVudCB9IGZyb20gJy4vbWV0cmljcy5jb21wb25lbnQnO1xuaW1wb3J0IHsgQ2hlY2tib3hNb2R1bGUgfSBmcm9tICdwcmltZW5nL2NoZWNrYm94JztcblxuQE5nTW9kdWxlKHtcbiAgZGVjbGFyYXRpb25zOiBbTWV0cmljc0NvbXBvbmVudF0sXG4gIGltcG9ydHM6IFtcbiAgICBDb21tb25Nb2R1bGUsXG4gICAgRm9ybXNNb2R1bGUsXG4gICAgUmVhY3RpdmVGb3Jtc01vZHVsZSxcbiAgICBEcmFnRHJvcE1vZHVsZSxcbiAgICBCdXR0b25Nb2R1bGUsXG4gICAgT3ZlcmxheVBhbmVsTW9kdWxlLFxuICAgIFNpZGViYXJNb2R1bGUsXG4gICAgSW5wdXRUZXh0TW9kdWxlLFxuICAgIENoZWNrYm94TW9kdWxlLFxuICAgIExldERpcmVjdGl2ZU1vZHVsZSxcbiAgICBWaXJ0dWFsU2Nyb2xsZXJNb2R1bGUsXG4gICAgRXJyb3JUb29sdGlwTW9kdWxlLFxuICBdLFxuICBleHBvcnRzOiBbTWV0cmljc0NvbXBvbmVudF0sXG59KVxuZXhwb3J0IGNsYXNzIE1ldHJpY3NNb2R1bGUge31cbiJdfQ==
@@ -4,7 +4,7 @@ import { InjectionToken, Component, ChangeDetectionStrategy, Inject, Injector, I
4
4
  import * as rxjs from 'rxjs';
5
5
  import { BehaviorSubject, Subject, takeUntil, map, filter, from, tap, of, switchMap, forkJoin, catchError } from 'rxjs';
6
6
  import * as lodash from 'lodash';
7
- import { compact, flatten, isArray, pull, set } from 'lodash';
7
+ import { compact, flatten, isArray, pull, set, kebabCase } from 'lodash';
8
8
  import * as i7 from '@angular/common';
9
9
  import { DOCUMENT, CommonModule } from '@angular/common';
10
10
  import { UUID, CoreModule } from '@veloce/core';
@@ -32,6 +32,7 @@ var FlowAction;
32
32
  FlowAction["FLOW_SWITCH_OBJECT"] = "FLOW_SWITCH_OBJECT";
33
33
  FlowAction["REMOTE_APPLY"] = "REMOTE_APPLY";
34
34
  FlowAction["REMOTE_CANCEL"] = "REMOTE_CANCEL";
35
+ FlowAction["SET_DEFAULT_METRICS"] = "SET_DEFAULT_METRICS";
35
36
  })(FlowAction || (FlowAction = {}));
36
37
  const ConfigureProductAction = ({ lineItemId, productId, }) => ({
37
38
  type: FlowAction.FLOW_CONFIGURE_PRODUCT,
@@ -62,6 +63,10 @@ const SwitchObjectAction = (payload) => ({
62
63
  type: FlowAction.FLOW_SWITCH_OBJECT,
63
64
  payload,
64
65
  });
66
+ const SetDefaultMetrics = (metrics) => ({
67
+ type: FlowAction.SET_DEFAULT_METRICS,
68
+ payload: { metrics },
69
+ });
65
70
 
66
71
  var cmsActions = /*#__PURE__*/Object.freeze({
67
72
  __proto__: null,
@@ -74,7 +79,8 @@ var cmsActions = /*#__PURE__*/Object.freeze({
74
79
  CloseDocGenAction: CloseDocGenAction,
75
80
  RemoteApplyAction: RemoteApplyAction,
76
81
  RemoteCancelAction: RemoteCancelAction,
77
- SwitchObjectAction: SwitchObjectAction
82
+ SwitchObjectAction: SwitchObjectAction,
83
+ SetDefaultMetrics: SetDefaultMetrics
78
84
  });
79
85
 
80
86
  const DEFAULT_PLUGINS_TOKEN = new InjectionToken('DEFAULT_PLUGINS_TOKEN');
@@ -472,6 +478,7 @@ const metadataToElement = (metadata, recursive = true) => {
472
478
  inputs: metadata.inputs,
473
479
  outputs: metadata.outputs,
474
480
  children: metadata.children.map(({ name }) => name),
481
+ configuredStyles: metadata.configuredStyles,
475
482
  };
476
483
  const normalizedElMetadata = normalizeElementMetadata(elMetadata);
477
484
  if (!metadata.script || !EXPORTED_CLASS_REGEX.test(metadata.script)) {
@@ -536,6 +543,10 @@ const normalizeElementMetadata = (elementMetadata) => {
536
543
  if (!metadata.isShared) {
537
544
  delete metadata.isShared;
538
545
  }
546
+ // configuredStyles
547
+ if (!metadata.configuredStyles) {
548
+ delete metadata.configuredStyles;
549
+ }
539
550
  return metadata;
540
551
  };
541
552
  const extractElementMetadata = (script) => {
@@ -1072,16 +1083,13 @@ class ElementsResolver {
1072
1083
  this.uiDef = uiDef;
1073
1084
  this.renderableElements = [];
1074
1085
  this.sharedElements = [];
1075
- this.elements = this.transpileScripts(elements);
1076
- const flatElements = this.flattenElements(this.elements);
1077
- for (const el of flatElements) {
1078
- if (this.isSharedElement(el)) {
1079
- this.sharedElements.push(el);
1080
- }
1081
- else {
1082
- this.renderableElements.push(el);
1083
- }
1084
- }
1086
+ const transpiledElements = this.transpileScripts(elements);
1087
+ this.sharedElements = this.flattenElements(transpiledElements).filter(el => this.isSharedElement(el));
1088
+ this.elements = transpiledElements.map(el => this.processElementMetadata(el)).filter(Boolean);
1089
+ this.renderableElements = this.getRenderableElements(this.elements);
1090
+ }
1091
+ getNgComponents() {
1092
+ return this.renderableElements.map(el => this.resolveElement(el)).filter(Boolean);
1085
1093
  }
1086
1094
  transpile(el) {
1087
1095
  var _a;
@@ -1104,8 +1112,19 @@ class ElementsResolver {
1104
1112
  isSharedElement(el) {
1105
1113
  return Boolean(el.isShared) && el.type !== 'REFERENCE';
1106
1114
  }
1115
+ getRenderableElements(elements) {
1116
+ const renderable = [];
1117
+ for (const el of elements) {
1118
+ if (!this.isSharedElement(el)) {
1119
+ const children = el.children.filter(child => !this.isSharedElement(child));
1120
+ const renderableChildren = this.getRenderableElements(children);
1121
+ renderable.push(Object.assign(Object.assign({}, el), { children }), ...renderableChildren);
1122
+ }
1123
+ }
1124
+ return renderable;
1125
+ }
1107
1126
  getSharedElement(element) {
1108
- if (element.type !== 'REFERENCE') {
1127
+ if (!element.reference) {
1109
1128
  return;
1110
1129
  }
1111
1130
  return this.sharedElements.find(el => element.reference === el.name);
@@ -1122,39 +1141,53 @@ class ElementsResolver {
1122
1141
  }
1123
1142
  processElementMetadata(sourceElement) {
1124
1143
  var _a, _b;
1125
- const sharedElement = this.getSharedElement(sourceElement);
1126
1144
  let finalElement;
1127
1145
  if (sourceElement.type === 'REFERENCE') {
1146
+ const sharedElement = this.getSharedElement(sourceElement);
1128
1147
  if (!sharedElement) {
1129
1148
  console.warn(`Shared element "${sourceElement.reference}" not found`);
1130
1149
  return;
1131
1150
  }
1132
- finalElement = Object.assign(Object.assign({}, sourceElement), { type: sharedElement.type, template: sharedElement.template, styles: ((_a = sharedElement.styles) !== null && _a !== void 0 ? _a : '') + '\n' + ((_b = sourceElement.styles) !== null && _b !== void 0 ? _b : ''), inputs: Object.assign(Object.assign({}, sharedElement.inputs), sourceElement.inputs), outputs: Object.assign(Object.assign({}, sharedElement.outputs), sourceElement.outputs) });
1151
+ finalElement = Object.assign(Object.assign({}, sourceElement), { children: this.getSharedChildren(sharedElement.children, sourceElement.path), type: sharedElement.type, template: sharedElement.template, styles: ((_a = sharedElement.styles) !== null && _a !== void 0 ? _a : '') + '\n' + ((_b = sourceElement.styles) !== null && _b !== void 0 ? _b : ''), inputs: Object.assign(Object.assign({}, sharedElement.inputs), sourceElement.inputs), outputs: Object.assign(Object.assign({}, sharedElement.outputs), sourceElement.outputs) });
1133
1152
  }
1134
1153
  else {
1135
1154
  finalElement = sourceElement;
1155
+ delete sourceElement.reference;
1136
1156
  }
1137
1157
  finalElement.template = this.resolveElementTemplate(finalElement);
1138
1158
  finalElement.styles = this.resolveElementStyles(finalElement);
1139
- return finalElement;
1159
+ return Object.assign(Object.assign({}, finalElement), { children: finalElement.children
1160
+ .map(child => this.processElementMetadata(child))
1161
+ .filter(Boolean) });
1162
+ }
1163
+ getSharedChildren(children, parentPath) {
1164
+ return children.map(c => {
1165
+ if (!c.path) {
1166
+ return c;
1167
+ }
1168
+ const [elName] = c.path.split('/').reverse();
1169
+ const path = parentPath + '/' + elName;
1170
+ return Object.assign(Object.assign({}, c), { path, children: this.getSharedChildren(c.children, path) });
1171
+ });
1140
1172
  }
1141
- resolveElement(sourceElement) {
1173
+ resolveElement(element) {
1142
1174
  var _a;
1143
- const element = this.processElementMetadata(sourceElement);
1144
- if (!element) {
1145
- return;
1146
- }
1147
1175
  const config = CONFIG[element.type];
1148
1176
  const defaultPlugins = (_a = DEFAULT_PLUGINS[this.uiDef.type]) !== null && _a !== void 0 ? _a : [];
1149
1177
  if (!config) {
1150
1178
  console.warn(`Unknown element type "${element.type}"`);
1151
1179
  return;
1152
1180
  }
1181
+ const configuredStyles = this.convertInlineStylesToCSS(element.configuredStyles);
1182
+ if (configuredStyles) {
1183
+ // order is important: styles written by user in CSS code should has higher priority
1184
+ element.styles = configuredStyles + element.styles;
1185
+ }
1153
1186
  const component = Object.assign(Object.assign({ selector: 'vl-element', template: element.template }, (element.styles ? { styles: [element.styles] } : {})), { providers: [
1154
1187
  { provide: DEFAULT_PLUGINS_TOKEN, useValue: defaultPlugins },
1155
1188
  { provide: UI_DEFINITION_METADATA, useValue: this.uiDef },
1156
1189
  { provide: ELEMENT_METADATA, useValue: element },
1157
- { provide: SHARED_ELEMENT_METADATA, useValue: this.getSharedElement(sourceElement) },
1190
+ { provide: SHARED_ELEMENT_METADATA, useValue: this.getSharedElement(element) },
1158
1191
  { provide: ELEMENT_CONFIG, useValue: config },
1159
1192
  { provide: VENDOR_MAP, useValue: vendorMap },
1160
1193
  ] });
@@ -1163,8 +1196,17 @@ class ElementsResolver {
1163
1196
  set(cmp, 'path', element.path);
1164
1197
  return cmp;
1165
1198
  }
1166
- getNgComponents() {
1167
- return this.renderableElements.map(el => this.resolveElement(el)).filter(Boolean);
1199
+ convertInlineStylesToCSS(styles) {
1200
+ if (!styles) {
1201
+ return '';
1202
+ }
1203
+ const entries = Object.entries(styles);
1204
+ let result = ':host {';
1205
+ entries.forEach(([style, value]) => {
1206
+ result += `${kebabCase(style)}:${value};`;
1207
+ });
1208
+ result += '}';
1209
+ return result;
1168
1210
  }
1169
1211
  }
1170
1212
 
@@ -1173,12 +1215,14 @@ class LauncherService {
1173
1215
  this.compiler = compiler;
1174
1216
  this.dynamicModuleService = dynamicModuleService;
1175
1217
  }
1176
- compileModule(elements, uiDefs) {
1177
- this.module = this.getModule(elements, uiDefs);
1218
+ compileModule(uiDef, elements) {
1219
+ const elementsResolver = new ElementsResolver(uiDef, elements);
1220
+ this.dynamicModuleService.elementsTree = elementsResolver.elements;
1221
+ this.module = this.getModule(elementsResolver.getNgComponents());
1178
1222
  return from(this.compiler.compileModuleAndAllComponentsAsync(this.module)).pipe(tap(m => {
1179
1223
  this.dynamicModuleService.componentFactories = m.componentFactories;
1180
1224
  this.moduleInstance = m;
1181
- }));
1225
+ }), map(m => ({ module: m, elements: elementsResolver.elements })));
1182
1226
  }
1183
1227
  destroy() {
1184
1228
  if (this.moduleInstance) {
@@ -1190,16 +1234,14 @@ class LauncherService {
1190
1234
  this.module = undefined;
1191
1235
  }
1192
1236
  }
1193
- getModule(elements, uiDef) {
1237
+ getModule(components) {
1194
1238
  const staticComponents = [ElementChildrenComponent, ElementRendererComponent, CustomTemplateDirective];
1195
- const elementsResolver = new ElementsResolver(uiDef, elements);
1196
- this.dynamicModuleService.elementsTree = elementsResolver.elements;
1197
1239
  let DynamicModule = class DynamicModule {
1198
1240
  };
1199
1241
  DynamicModule = __decorate([
1200
1242
  NgModule({
1201
1243
  imports: [CommonModule, FormsModule, ReactiveFormsModule, FederatedModule, DragDropModule],
1202
- declarations: [...staticComponents, ...elementsResolver.getNgComponents()],
1244
+ declarations: [...staticComponents, ...components],
1203
1245
  jit: true,
1204
1246
  })
1205
1247
  ], DynamicModule);
@@ -1267,10 +1309,13 @@ class PreviewComponent {
1267
1309
  return;
1268
1310
  }
1269
1311
  const _a = this.uiDefinition, { children } = _a, uiDefinitionMeta = __rest(_a, ["children"]);
1270
- this.elements = this.elementToMetadataSafe(children);
1271
- const compilation$ = this.launcherService.compileModule(this.elements, uiDefinitionMeta);
1312
+ const elements = this.elementToMetadataSafe(children);
1313
+ const compilation$ = this.launcherService.compileModule(uiDefinitionMeta, elements);
1272
1314
  forkJoin([compilation$, this.initializeConfiguration$()])
1273
- .pipe(tap(() => this.state$.next({ loading: false, failure: false })), catchError(error => {
1315
+ .pipe(tap(([result]) => {
1316
+ this.elements = result.elements;
1317
+ this.state$.next({ loading: false, failure: false });
1318
+ }), catchError(error => {
1274
1319
  var _a, _b;
1275
1320
  console.error(error);
1276
1321
  if (!((_b = (_a = this.uiDefinition) === null || _a === void 0 ? void 0 : _a.properties) === null || _b === void 0 ? void 0 : _b.suppressToastMessages)) {
@@ -1368,5 +1413,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.15", ngImpo
1368
1413
  * Generated bundle index. Do not edit.
1369
1414
  */
1370
1415
 
1371
- export { ApplyProductConfigurationAction, CloseDocGenAction, ConfigureProductAction, DEFAULT_PLUGINS_TOKEN, ELEMENT_CONFIG, ELEMENT_METADATA, ElementComponent, ElementDefinition, ElementsResolver, FlowAction, IntegrationState, LauncherModule, LauncherService, NavigateBackAction, NavigateToCatalogAction, OpenDocGenAction, PreviewComponent, PreviewModule, RemoteApplyAction, RemoteCancelAction, SHARED_ELEMENT_METADATA, SwitchObjectAction, TemplatesService, UI_DEFINITION_METADATA, UiBuildError, VENDOR_MAP, doesElementSupportIO, elementToMetadata, extendElementMetadata, extractElementMetadata, findElementByModule, flattenElements, getAbsolutePath, getElementConfig, isSharedElement, isValidScript, metadataToElement, normalizeElementMetadata, parseBoundPath, parsePath, stringifyElementMetadata };
1416
+ export { ApplyProductConfigurationAction, CloseDocGenAction, ConfigureProductAction, DEFAULT_PLUGINS_TOKEN, ELEMENT_CONFIG, ELEMENT_METADATA, ElementComponent, ElementDefinition, ElementsResolver, FlowAction, IntegrationState, LauncherModule, LauncherService, NavigateBackAction, NavigateToCatalogAction, OpenDocGenAction, PreviewComponent, PreviewModule, RemoteApplyAction, RemoteCancelAction, SHARED_ELEMENT_METADATA, SetDefaultMetrics, SwitchObjectAction, TemplatesService, UI_DEFINITION_METADATA, UiBuildError, VENDOR_MAP, doesElementSupportIO, elementToMetadata, extendElementMetadata, extractElementMetadata, findElementByModule, flattenElements, getAbsolutePath, getElementConfig, isSharedElement, isValidScript, metadataToElement, normalizeElementMetadata, parseBoundPath, parsePath, stringifyElementMetadata };
1372
1417
  //# sourceMappingURL=veloce-sdk-cms.js.map