@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.
- package/bundles/veloce-sdk-cms.umd.js +97 -45
- package/bundles/veloce-sdk-cms.umd.js.map +1 -1
- package/bundles/veloce-sdk.umd.js +457 -195
- package/bundles/veloce-sdk.umd.js.map +1 -1
- package/cms/cms.actions.d.ts +4 -2
- package/cms/services/launcher.service.d.ts +3 -3
- package/cms/types/common.types.d.ts +5 -0
- package/cms/types/compilation.types.d.ts +6 -0
- package/cms/types/index.d.ts +2 -0
- package/cms/types/metrics.types.d.ts +5 -0
- package/cms/utils/elements-resolver.d.ts +4 -1
- package/cms/vendor-map.d.ts +1 -0
- package/esm2015/cms/cms.actions.js +6 -1
- package/esm2015/cms/components/preview/preview.component.js +7 -4
- package/esm2015/cms/services/launcher.service.js +10 -10
- package/esm2015/cms/types/common.types.js +1 -1
- package/esm2015/cms/types/compilation.types.js +2 -0
- package/esm2015/cms/types/index.js +3 -1
- package/esm2015/cms/types/metrics.types.js +2 -0
- package/esm2015/cms/utils/element.utils.js +6 -1
- package/esm2015/cms/utils/elements-resolver.js +55 -24
- package/esm2015/src/components/header/header.component.js +12 -19
- package/esm2015/src/components/header/header.module.js +6 -2
- package/esm2015/src/components/header/metrics/index.js +2 -0
- package/esm2015/src/components/header/metrics/metrics.component.js +216 -0
- package/esm2015/src/components/header/metrics/metrics.definitions.js +9 -0
- package/esm2015/src/components/header/metrics/metrics.module.js +64 -0
- package/fesm2015/veloce-sdk-cms.js +80 -35
- package/fesm2015/veloce-sdk-cms.js.map +1 -1
- package/fesm2015/veloce-sdk.js +341 -85
- package/fesm2015/veloce-sdk.js.map +1 -1
- package/package.json +1 -1
- package/src/components/header/header.component.d.ts +0 -8
- package/src/components/header/header.module.d.ts +2 -1
- package/src/components/header/metrics/index.d.ts +1 -0
- package/src/components/header/metrics/metrics.component.d.ts +49 -0
- package/src/components/header/metrics/metrics.definitions.d.ts +4 -0
- 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
|
-
|
1076
|
-
|
1077
|
-
|
1078
|
-
|
1079
|
-
|
1080
|
-
|
1081
|
-
|
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.
|
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(
|
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(
|
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
|
-
|
1167
|
-
|
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(
|
1177
|
-
|
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(
|
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, ...
|
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
|
-
|
1271
|
-
const compilation$ = this.launcherService.compileModule(
|
1312
|
+
const elements = this.elementToMetadataSafe(children);
|
1313
|
+
const compilation$ = this.launcherService.compileModule(uiDefinitionMeta, elements);
|
1272
1314
|
forkJoin([compilation$, this.initializeConfiguration$()])
|
1273
|
-
.pipe(tap(() =>
|
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
|