@tuki-io/tuki-widgets 0.0.144 → 0.0.146

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.
@@ -5,16 +5,12 @@ export interface UccxServerEntityStats {
5
5
  [key in EntityType]: number;
6
6
  };
7
7
  }
8
- export interface WebexLicenseFeatureUsage {
9
- feature: 'PERSON' | 'ORPHANED_DEVICE';
10
- totalCount: number;
11
- usageCount: number;
12
- unUsedCount: number;
8
+ export interface CcLicense {
9
+ name: string;
10
+ totalUnits: number;
11
+ consumedUnits: number;
13
12
  }
14
13
  export interface WebexLicenseUsageSummary {
15
- webexReadinessLicenseStats: WebexLicenseFeatureUsage[];
16
- totalProfessionalLicenseCount: number;
17
- totalWorkspaceProfessionalLicenseCount: number;
18
- professionalWebexReadinessTotalLicenseStatCoverage: number;
19
- workspaceWebexReadinessTotalLicenseStatCoverage: number;
14
+ licenses: CcLicense[];
15
+ totalExistingLicenses: number;
20
16
  }
@@ -1,5 +1,5 @@
1
1
  import { OnChanges, OnDestroy, OnInit, SimpleChanges } from '@angular/core';
2
- import { ApexChart, ApexDataLabels, ApexLegend, ApexNonAxisChartSeries, ApexPlotOptions } from 'ng-apexcharts';
2
+ import { ApexChart, ApexDataLabels, ApexLegend, ApexNonAxisChartSeries, ApexPlotOptions, ApexTooltip } from 'ng-apexcharts';
3
3
  import * as i0 from "@angular/core";
4
4
  interface LicenseLegendItem {
5
5
  label: string;
@@ -19,6 +19,7 @@ interface LicenseChartOptions {
19
19
  legend: ApexLegend;
20
20
  plotOptions: ApexPlotOptions;
21
21
  dataLabels: ApexDataLabels;
22
+ tooltip: ApexTooltip;
22
23
  }
23
24
  interface LicenseRequirementViewModel {
24
25
  chartOptions: LicenseChartOptions;
@@ -1,5 +1,3 @@
1
1
  ;
2
- ;
3
- ;
4
2
  export {};
5
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2MtcmVhZGluZXNzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHVraS93aWRnZXRzL2NvbnRhY3QtY2VudGVyL2NjLXJlYWRpbmVzcy90eXBlcy9jYy1yZWFkaW5lc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBa0JDLENBQUM7QUFPRCxDQUFDO0FBUUQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEVudGl0eVR5cGUgPVxyXG4gICAgfCAnYXBwbGljYXRpb25zJ1xyXG4gICAgfCAndHJpZ2dlcnMnXHJcbiAgICB8ICdjb250YWN0U2VydmljZVF1ZXVlcydcclxuICAgIHwgJ3Jlc291cmNlcydcclxuICAgIHwgJ3Jlc291cmNlR3JvdXBzJ1xyXG4gICAgfCAnc2tpbGxzJ1xyXG4gICAgfCAneG1sJ1xyXG4gICAgfCAnc2NyaXB0cydcclxuICAgIHwgJ2F1ZGlvUHJvbXB0cydcclxuICAgIHwgJ3RlYW1zJ1xyXG4gICAgLy8gfCAncmVhc29uQ29kZXMnXHJcbiAgICB8ICdwaG9uZWJvb2tzJ1xyXG4gICAgfCAnd3JhcFVwQ29kZXMnO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBVY2N4U2VydmVyRW50aXR5U3RhdHMge1xyXG4gICAgdWNjeElwOiBzdHJpbmc7XHJcbiAgICBjb3VudHM6IHsgW2tleSBpbiBFbnRpdHlUeXBlXTogbnVtYmVyIH1cclxufTtcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgV2ViZXhMaWNlbnNlRmVhdHVyZVVzYWdlIHtcclxuICAgIGZlYXR1cmU6ICdQRVJTT04nIHwgJ09SUEhBTkVEX0RFVklDRScsXHJcbiAgICB0b3RhbENvdW50OiBudW1iZXIsXHJcbiAgICB1c2FnZUNvdW50OiBudW1iZXIsXHJcbiAgICB1blVzZWRDb3VudDogbnVtYmVyXHJcbn07XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIFdlYmV4TGljZW5zZVVzYWdlU3VtbWFyeSB7XHJcbiAgICB3ZWJleFJlYWRpbmVzc0xpY2Vuc2VTdGF0czogV2ViZXhMaWNlbnNlRmVhdHVyZVVzYWdlW10sXHJcbiAgICB0b3RhbFByb2Zlc3Npb25hbExpY2Vuc2VDb3VudDogbnVtYmVyLFxyXG4gICAgdG90YWxXb3Jrc3BhY2VQcm9mZXNzaW9uYWxMaWNlbnNlQ291bnQ6IG51bWJlcixcclxuICAgIHByb2Zlc3Npb25hbFdlYmV4UmVhZGluZXNzVG90YWxMaWNlbnNlU3RhdENvdmVyYWdlOiBudW1iZXIsXHJcbiAgICB3b3Jrc3BhY2VXZWJleFJlYWRpbmVzc1RvdGFsTGljZW5zZVN0YXRDb3ZlcmFnZTogbnVtYmVyXHJcbn07Il19
3
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2MtcmVhZGluZXNzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHVraS93aWRnZXRzL2NvbnRhY3QtY2VudGVyL2NjLXJlYWRpbmVzcy90eXBlcy9jYy1yZWFkaW5lc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBa0JDLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgdHlwZSBFbnRpdHlUeXBlID1cclxuICAgIHwgJ2FwcGxpY2F0aW9ucydcclxuICAgIHwgJ3RyaWdnZXJzJ1xyXG4gICAgfCAnY29udGFjdFNlcnZpY2VRdWV1ZXMnXHJcbiAgICB8ICdyZXNvdXJjZXMnXHJcbiAgICB8ICdyZXNvdXJjZUdyb3VwcydcclxuICAgIHwgJ3NraWxscydcclxuICAgIHwgJ3htbCdcclxuICAgIHwgJ3NjcmlwdHMnXHJcbiAgICB8ICdhdWRpb1Byb21wdHMnXHJcbiAgICB8ICd0ZWFtcydcclxuICAgIC8vIHwgJ3JlYXNvbkNvZGVzJ1xyXG4gICAgfCAncGhvbmVib29rcydcclxuICAgIHwgJ3dyYXBVcENvZGVzJztcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgVWNjeFNlcnZlckVudGl0eVN0YXRzIHtcclxuICAgIHVjY3hJcDogc3RyaW5nO1xyXG4gICAgY291bnRzOiB7IFtrZXkgaW4gRW50aXR5VHlwZV06IG51bWJlciB9XHJcbn07XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIENjTGljZW5zZSB7XHJcbiAgICBuYW1lOiBzdHJpbmc7XHJcbiAgICB0b3RhbFVuaXRzOiBudW1iZXI7XHJcbiAgICBjb25zdW1lZFVuaXRzOiBudW1iZXI7XHJcbn1cclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgV2ViZXhMaWNlbnNlVXNhZ2VTdW1tYXJ5IHtcclxuICAgIGxpY2Vuc2VzOiBDY0xpY2Vuc2VbXTtcclxuICAgIHRvdGFsRXhpc3RpbmdMaWNlbnNlczogbnVtYmVyO1xyXG59Il19
@@ -9,7 +9,8 @@ import * as i4 from "@angular/material/core";
9
9
  import * as i5 from "../../components/card/card.component";
10
10
  const LEGEND_COLORS = {
11
11
  existing: '#16A693',
12
- missing: '#808080'
12
+ missing: '#808080',
13
+ empty: '#E0E0E0'
13
14
  };
14
15
  export class LicencesRequirementComponent {
15
16
  constructor() {
@@ -17,7 +18,7 @@ export class LicencesRequirementComponent {
17
18
  this.viewModel = null;
18
19
  this.loading = false;
19
20
  this.error = false;
20
- this.filters = ['All', 'Professional licenses', 'Workspace licenses'];
21
+ this.filters = ['All'];
21
22
  this.selectedSummaryIndex = 0;
22
23
  this.summary = null;
23
24
  }
@@ -68,6 +69,13 @@ export class LicencesRequirementComponent {
68
69
  else {
69
70
  this.summary = null;
70
71
  }
72
+ // Build filters dynamically from license names
73
+ if (this.summary?.licenses?.length) {
74
+ this.filters = ['All', ...this.summary.licenses.map(l => l.name)];
75
+ }
76
+ else {
77
+ this.filters = ['All'];
78
+ }
71
79
  console.log('Processed summary:', this.summary);
72
80
  this.selectFilter(0);
73
81
  });
@@ -83,38 +91,25 @@ export class LicencesRequirementComponent {
83
91
  }
84
92
  mapSummaryToViewModel(summary, filterIndex) {
85
93
  console.log('Mapping summary to viewModel:', summary, 'filterIndex:', filterIndex);
86
- // Extract data from webexReadinessLicenseStats and license counts
87
- const personStat = summary.webexReadinessLicenseStats?.find(stat => stat.feature === 'PERSON');
88
- const deviceStat = summary.webexReadinessLicenseStats?.find(stat => stat.feature === 'ORPHANED_DEVICE');
89
- const professionalNeeded = personStat?.totalCount ?? 0;
90
- const workspaceNeeded = deviceStat?.totalCount ?? 0;
91
- const professionalOwned = Number(summary.totalProfessionalLicenseCount ?? 0);
92
- const workspaceOwned = Number(summary.totalWorkspaceProfessionalLicenseCount ?? 0);
93
- // Filter data based on selection
94
+ const licenses = summary.licenses || [];
94
95
  let existingCount;
95
96
  let missingCount;
96
97
  let rows;
97
98
  if (filterIndex === 0) {
98
- // All - show both
99
- existingCount = professionalOwned + workspaceOwned;
100
- const totalNeeded = professionalNeeded + workspaceNeeded;
101
- missingCount = Math.max(0, totalNeeded - existingCount);
102
- rows = [
103
- { label: 'Professional licenses', value: professionalNeeded },
104
- { label: 'Workspace licenses', value: workspaceNeeded }
105
- ];
106
- }
107
- else if (filterIndex === 1) {
108
- // Professional licenses only
109
- existingCount = professionalOwned;
110
- missingCount = Math.max(0, professionalNeeded - professionalOwned);
111
- rows = [{ label: 'Professional licenses', value: professionalNeeded }];
99
+ // "All" filter: show totalExistingLicenses vs total consumedUnits
100
+ existingCount = summary.totalExistingLicenses ?? 0;
101
+ missingCount = licenses.reduce((acc, l) => acc + l.consumedUnits, 0);
102
+ rows = licenses.map(l => ({
103
+ label: l.name,
104
+ value: l.consumedUnits
105
+ }));
112
106
  }
113
107
  else {
114
- // Workspace licenses only
115
- existingCount = workspaceOwned;
116
- missingCount = Math.max(0, workspaceNeeded - workspaceOwned);
117
- rows = [{ label: 'Workspace licenses', value: workspaceNeeded }];
108
+ // Specific license filter: show 0 existing vs consumedUnits for that license
109
+ const selectedLicense = licenses[filterIndex - 1];
110
+ existingCount = 0;
111
+ missingCount = selectedLicense?.consumedUnits ?? 0;
112
+ rows = selectedLicense ? [{ label: selectedLicense.name, value: selectedLicense.consumedUnits }] : [];
118
113
  }
119
114
  const totalLicenses = existingCount + missingCount;
120
115
  const legendItems = [
@@ -133,7 +128,7 @@ export class LicencesRequirementComponent {
133
128
  ];
134
129
  console.log('Legend items:', legendItems);
135
130
  console.log('Table rows:', rows);
136
- const totalRequired = rows.reduce((acc, row) => acc + row.value, 0);
131
+ const totalRequired = existingCount + missingCount;
137
132
  return {
138
133
  chartOptions: this.buildChartOptions(legendItems),
139
134
  legendItems,
@@ -142,14 +137,20 @@ export class LicencesRequirementComponent {
142
137
  };
143
138
  }
144
139
  buildChartOptions(legendItems) {
140
+ const totalCount = legendItems.reduce((acc, item) => acc + item.count, 0);
141
+ const isEmpty = totalCount === 0;
142
+ // For empty chart, show a full light gray donut
143
+ const series = isEmpty ? [1] : legendItems.map(item => item.count);
144
+ const labels = isEmpty ? ['No data'] : legendItems.map(item => item.label);
145
+ const colors = isEmpty ? [LEGEND_COLORS.empty] : legendItems.map(item => item.color);
145
146
  return {
146
- series: legendItems.map(item => item.count),
147
+ series,
147
148
  chart: {
148
149
  type: 'donut',
149
150
  height: 220
150
151
  },
151
- labels: legendItems.map(item => item.label),
152
- colors: legendItems.map(item => item.color),
152
+ labels,
153
+ colors,
153
154
  legend: {
154
155
  show: false
155
156
  },
@@ -162,6 +163,9 @@ export class LicencesRequirementComponent {
162
163
  },
163
164
  dataLabels: {
164
165
  enabled: false
166
+ },
167
+ tooltip: {
168
+ enabled: !isEmpty
165
169
  }
166
170
  };
167
171
  }
@@ -175,13 +179,13 @@ export class LicencesRequirementComponent {
175
179
  }
176
180
  }
177
181
  LicencesRequirementComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: LicencesRequirementComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
178
- LicencesRequirementComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: LicencesRequirementComponent, selector: "cc-licenses-requirement", inputs: { customerId: "customerId", token: "token" }, usesOnChanges: true, ngImport: i0, template: "<tk-card title=\"Licenses requirement\">\n <ng-container ngProjectAs=\"card-content\">\n <div\n class=\"widget-body\"\n *ngIf=\"!loading && !error && viewModel; else widgetState\"\n >\n <div class=\"left-section\">\n <mat-select\n [(value)]=\"selectedSummaryIndex\"\n (selectionChange)=\"onFilterChange($event.value)\"\n class=\"filter-select\"\n >\n <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n {{ option }}\n </mat-option>\n </mat-select>\n\n <div class=\"chart-section\">\n <div class=\"chart-wrapper\">\n <apx-chart\n [series]=\"viewModel.chartOptions.series\"\n [chart]=\"viewModel.chartOptions.chart\"\n [labels]=\"viewModel.chartOptions.labels\"\n [colors]=\"viewModel.chartOptions.colors\"\n [legend]=\"viewModel.chartOptions.legend\"\n [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n >\n </apx-chart>\n </div>\n <div class=\"legend\">\n <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n <span\n class=\"legend-marker\"\n [style.backgroundColor]=\"item.color\"\n ></span>\n <span class=\"legend-label\">{{ item.label }}</span>\n <span class=\"legend-value\">{{ item.count | number }}</span>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"divider\" aria-hidden=\"true\"></div>\n\n <div class=\"table-section\">\n <div class=\"lic-table-header\">\n <span>License</span>\n <span>Required</span>\n </div>\n <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n <span>{{ row.label }}</span>\n <span>{{ row.value | number }}</span>\n </div>\n <div class=\"table-row total-row\">\n <span>Total</span>\n <span>{{ viewModel.totalRequired | number }}</span>\n </div>\n </div>\n </div>\n\n <ng-template #widgetState>\n <div class=\"state-message\" *ngIf=\"loading\">\n Loading license requirements\u2026\n </div>\n <div class=\"state-message error\" *ngIf=\"!loading && error\">\n Unable to load license requirements.\n </div>\n <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n No license data available.\n </div>\n </ng-template>\n </ng-container>\n</tk-card>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Poppins:400,100,200,300,500,600,800,700,900\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@300;400&display=swap\";:host{display:block;width:100%;box-sizing:border-box}.filter-select{font-family:Inter,Inter,sans-serif;font-size:14px;min-width:140px;border:1px solid #e5e7eb;border-radius:.5rem;padding:.5rem .75rem;background-color:#fff}.filter-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select{border:1px solid #e5e7eb;border-radius:.5rem;background-color:#fff}:host ::ng-deep .filter-select.mat-mdc-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select.mat-focused{border-color:#16a693}:host ::ng-deep .filter-select .mat-mdc-select-trigger{font-size:14px;padding:.5rem .75rem}:host ::ng-deep .filter-select .mat-mdc-select-value{color:#111827}:host ::ng-deep .filter-select .mat-mdc-select-arrow-wrapper{padding-left:.5rem}.widget-body{display:flex;gap:2.5rem;align-items:flex-start}.left-section{flex:2;display:flex;flex-direction:column;gap:1rem}.chart-section{display:flex;gap:1.5rem;align-items:center}.chart-wrapper{width:220px;min-height:220px;flex-shrink:0}.chart-wrapper apx-chart{width:220px!important;height:220px!important}.legend{display:flex;flex-direction:column;gap:.75rem}.legend-item{display:flex;align-items:center;gap:.5rem;font-family:Inter,Inter,sans-serif;font-size:14px}.legend-marker{width:14px;height:14px;border-radius:50%;display:inline-flex}.legend-label{color:#4b5563}.legend-value{margin-left:auto;font-weight:600;color:#111827}.divider{width:1px;min-height:200px;align-self:stretch;background:#e5e7eb}.table-section{flex:3;display:flex;flex-direction:column;gap:.75rem;font-family:Inter,Inter,sans-serif}.lic-table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.lic-table-header{font-weight:600;color:#6b7280;padding-bottom:.5rem;border-bottom:1px solid #e5e7eb}.table-row{color:#111827}.total-row{font-weight:700;margin-top:.5rem;padding-top:.5rem;border-top:1px solid #e5e7eb}.state-message{font-family:Inter,Inter,sans-serif;font-size:14px;color:#6b7280}.state-message.error{color:#b91c1c}@media (max-width: 960px){.widget-body{flex-direction:column}.divider{display:none}.chart-section{justify-content:space-between}}:host ::ng-deep .apexcharts-legend{display:none!important}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.ChartComponent, selector: "apx-chart", inputs: ["chart", "annotations", "colors", "dataLabels", "series", "stroke", "labels", "legend", "markers", "noData", "fill", "tooltip", "plotOptions", "responsive", "xaxis", "yaxis", "forecastDataPoints", "grid", "states", "title", "subtitle", "theme", "autoUpdateSeries"] }, { kind: "component", type: i3.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex"], exportAs: ["matSelect"] }, { kind: "component", type: i4.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "component", type: i5.CardComponent, selector: "tk-card", inputs: ["title"] }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }] });
182
+ LicencesRequirementComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: LicencesRequirementComponent, selector: "cc-licenses-requirement", inputs: { customerId: "customerId", token: "token" }, usesOnChanges: true, ngImport: i0, template: "<tk-card title=\"Licenses requirement\">\n <ng-container ngProjectAs=\"card-content\">\n <div\n class=\"widget-body\"\n *ngIf=\"!loading && !error && viewModel; else widgetState\"\n >\n <div class=\"left-section\">\n <mat-select\n [(value)]=\"selectedSummaryIndex\"\n (selectionChange)=\"onFilterChange($event.value)\"\n class=\"filter-select\"\n >\n <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n {{ option }}\n </mat-option>\n </mat-select>\n\n <div class=\"chart-section\">\n <div class=\"chart-wrapper\">\n <apx-chart\n [series]=\"viewModel.chartOptions.series\"\n [chart]=\"viewModel.chartOptions.chart\"\n [labels]=\"viewModel.chartOptions.labels\"\n [colors]=\"viewModel.chartOptions.colors\"\n [legend]=\"viewModel.chartOptions.legend\"\n [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n [tooltip]=\"viewModel.chartOptions.tooltip\"\n >\n </apx-chart>\n </div>\n <div class=\"legend\">\n <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n <span\n class=\"legend-marker\"\n [style.backgroundColor]=\"item.color\"\n ></span>\n <span class=\"legend-label\">{{ item.label }}</span>\n <span class=\"legend-value\">{{ item.count | number }}</span>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"divider\" aria-hidden=\"true\"></div>\n\n <div class=\"table-section\">\n <div class=\"lic-table-header\">\n <span>License</span>\n <span>Required</span>\n </div>\n <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n <span>{{ row.label }}</span>\n <span>{{ row.value | number }}</span>\n </div>\n <div class=\"table-row total-row\">\n <span>Total</span>\n <span>{{ viewModel.totalRequired | number }}</span>\n </div>\n </div>\n </div>\n\n <ng-template #widgetState>\n <div class=\"state-message\" *ngIf=\"loading\">\n Loading license requirements\u2026\n </div>\n <div class=\"state-message error\" *ngIf=\"!loading && error\">\n Unable to load license requirements.\n </div>\n <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n No license data available.\n </div>\n </ng-template>\n </ng-container>\n</tk-card>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Poppins:400,100,200,300,500,600,800,700,900\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@300;400&display=swap\";:host{display:block;width:100%;box-sizing:border-box}.filter-select{font-family:Inter,Inter,sans-serif;font-size:14px;min-width:140px;border:1px solid #e5e7eb;border-radius:.5rem;padding:.5rem .75rem;background-color:#fff}.filter-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select{border:1px solid #e5e7eb;border-radius:.5rem;background-color:#fff}:host ::ng-deep .filter-select.mat-mdc-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select.mat-focused{border-color:#16a693}:host ::ng-deep .filter-select .mat-mdc-select-trigger{font-size:14px;padding:.5rem .75rem}:host ::ng-deep .filter-select .mat-mdc-select-value{color:#111827}:host ::ng-deep .filter-select .mat-mdc-select-arrow-wrapper{padding-left:.5rem}.widget-body{display:flex;gap:2.5rem;align-items:flex-start}.left-section{flex:2;display:flex;flex-direction:column;gap:1rem}.chart-section{display:flex;gap:1.5rem;align-items:center}.chart-wrapper{width:220px;min-height:220px;flex-shrink:0}.chart-wrapper apx-chart{width:220px!important;height:220px!important}.legend{display:flex;flex-direction:column;gap:.75rem}.legend-item{display:flex;align-items:center;gap:.5rem;font-family:Inter,Inter,sans-serif;font-size:14px}.legend-marker{width:14px;height:14px;border-radius:50%;display:inline-flex}.legend-label{color:#4b5563}.legend-value{margin-left:auto;font-weight:600;color:#111827}.divider{width:1px;min-height:200px;align-self:stretch;background:#e5e7eb}.table-section{flex:3;display:flex;flex-direction:column;gap:.75rem;font-family:Inter,Inter,sans-serif}.lic-table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.lic-table-header{font-weight:600;color:#6b7280;padding-bottom:.5rem;border-bottom:1px solid #e5e7eb}.table-row{color:#111827}.total-row{font-weight:700;margin-top:.5rem;padding-top:.5rem;border-top:1px solid #e5e7eb}.state-message{font-family:Inter,Inter,sans-serif;font-size:14px;color:#6b7280}.state-message.error{color:#b91c1c}@media (max-width: 960px){.widget-body{flex-direction:column}.divider{display:none}.chart-section{justify-content:space-between}}:host ::ng-deep .apexcharts-legend{display:none!important}\n"], dependencies: [{ kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "component", type: i2.ChartComponent, selector: "apx-chart", inputs: ["chart", "annotations", "colors", "dataLabels", "series", "stroke", "labels", "legend", "markers", "noData", "fill", "tooltip", "plotOptions", "responsive", "xaxis", "yaxis", "forecastDataPoints", "grid", "states", "title", "subtitle", "theme", "autoUpdateSeries"] }, { kind: "component", type: i3.MatSelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex"], exportAs: ["matSelect"] }, { kind: "component", type: i4.MatOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "component", type: i5.CardComponent, selector: "tk-card", inputs: ["title"] }, { kind: "pipe", type: i1.DecimalPipe, name: "number" }] });
179
183
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: LicencesRequirementComponent, decorators: [{
180
184
  type: Component,
181
- args: [{ selector: 'cc-licenses-requirement', template: "<tk-card title=\"Licenses requirement\">\n <ng-container ngProjectAs=\"card-content\">\n <div\n class=\"widget-body\"\n *ngIf=\"!loading && !error && viewModel; else widgetState\"\n >\n <div class=\"left-section\">\n <mat-select\n [(value)]=\"selectedSummaryIndex\"\n (selectionChange)=\"onFilterChange($event.value)\"\n class=\"filter-select\"\n >\n <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n {{ option }}\n </mat-option>\n </mat-select>\n\n <div class=\"chart-section\">\n <div class=\"chart-wrapper\">\n <apx-chart\n [series]=\"viewModel.chartOptions.series\"\n [chart]=\"viewModel.chartOptions.chart\"\n [labels]=\"viewModel.chartOptions.labels\"\n [colors]=\"viewModel.chartOptions.colors\"\n [legend]=\"viewModel.chartOptions.legend\"\n [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n >\n </apx-chart>\n </div>\n <div class=\"legend\">\n <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n <span\n class=\"legend-marker\"\n [style.backgroundColor]=\"item.color\"\n ></span>\n <span class=\"legend-label\">{{ item.label }}</span>\n <span class=\"legend-value\">{{ item.count | number }}</span>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"divider\" aria-hidden=\"true\"></div>\n\n <div class=\"table-section\">\n <div class=\"lic-table-header\">\n <span>License</span>\n <span>Required</span>\n </div>\n <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n <span>{{ row.label }}</span>\n <span>{{ row.value | number }}</span>\n </div>\n <div class=\"table-row total-row\">\n <span>Total</span>\n <span>{{ viewModel.totalRequired | number }}</span>\n </div>\n </div>\n </div>\n\n <ng-template #widgetState>\n <div class=\"state-message\" *ngIf=\"loading\">\n Loading license requirements\u2026\n </div>\n <div class=\"state-message error\" *ngIf=\"!loading && error\">\n Unable to load license requirements.\n </div>\n <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n No license data available.\n </div>\n </ng-template>\n </ng-container>\n</tk-card>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Poppins:400,100,200,300,500,600,800,700,900\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@300;400&display=swap\";:host{display:block;width:100%;box-sizing:border-box}.filter-select{font-family:Inter,Inter,sans-serif;font-size:14px;min-width:140px;border:1px solid #e5e7eb;border-radius:.5rem;padding:.5rem .75rem;background-color:#fff}.filter-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select{border:1px solid #e5e7eb;border-radius:.5rem;background-color:#fff}:host ::ng-deep .filter-select.mat-mdc-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select.mat-focused{border-color:#16a693}:host ::ng-deep .filter-select .mat-mdc-select-trigger{font-size:14px;padding:.5rem .75rem}:host ::ng-deep .filter-select .mat-mdc-select-value{color:#111827}:host ::ng-deep .filter-select .mat-mdc-select-arrow-wrapper{padding-left:.5rem}.widget-body{display:flex;gap:2.5rem;align-items:flex-start}.left-section{flex:2;display:flex;flex-direction:column;gap:1rem}.chart-section{display:flex;gap:1.5rem;align-items:center}.chart-wrapper{width:220px;min-height:220px;flex-shrink:0}.chart-wrapper apx-chart{width:220px!important;height:220px!important}.legend{display:flex;flex-direction:column;gap:.75rem}.legend-item{display:flex;align-items:center;gap:.5rem;font-family:Inter,Inter,sans-serif;font-size:14px}.legend-marker{width:14px;height:14px;border-radius:50%;display:inline-flex}.legend-label{color:#4b5563}.legend-value{margin-left:auto;font-weight:600;color:#111827}.divider{width:1px;min-height:200px;align-self:stretch;background:#e5e7eb}.table-section{flex:3;display:flex;flex-direction:column;gap:.75rem;font-family:Inter,Inter,sans-serif}.lic-table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.lic-table-header{font-weight:600;color:#6b7280;padding-bottom:.5rem;border-bottom:1px solid #e5e7eb}.table-row{color:#111827}.total-row{font-weight:700;margin-top:.5rem;padding-top:.5rem;border-top:1px solid #e5e7eb}.state-message{font-family:Inter,Inter,sans-serif;font-size:14px;color:#6b7280}.state-message.error{color:#b91c1c}@media (max-width: 960px){.widget-body{flex-direction:column}.divider{display:none}.chart-section{justify-content:space-between}}:host ::ng-deep .apexcharts-legend{display:none!important}\n"] }]
185
+ args: [{ selector: 'cc-licenses-requirement', template: "<tk-card title=\"Licenses requirement\">\n <ng-container ngProjectAs=\"card-content\">\n <div\n class=\"widget-body\"\n *ngIf=\"!loading && !error && viewModel; else widgetState\"\n >\n <div class=\"left-section\">\n <mat-select\n [(value)]=\"selectedSummaryIndex\"\n (selectionChange)=\"onFilterChange($event.value)\"\n class=\"filter-select\"\n >\n <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n {{ option }}\n </mat-option>\n </mat-select>\n\n <div class=\"chart-section\">\n <div class=\"chart-wrapper\">\n <apx-chart\n [series]=\"viewModel.chartOptions.series\"\n [chart]=\"viewModel.chartOptions.chart\"\n [labels]=\"viewModel.chartOptions.labels\"\n [colors]=\"viewModel.chartOptions.colors\"\n [legend]=\"viewModel.chartOptions.legend\"\n [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n [tooltip]=\"viewModel.chartOptions.tooltip\"\n >\n </apx-chart>\n </div>\n <div class=\"legend\">\n <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n <span\n class=\"legend-marker\"\n [style.backgroundColor]=\"item.color\"\n ></span>\n <span class=\"legend-label\">{{ item.label }}</span>\n <span class=\"legend-value\">{{ item.count | number }}</span>\n </div>\n </div>\n </div>\n </div>\n\n <div class=\"divider\" aria-hidden=\"true\"></div>\n\n <div class=\"table-section\">\n <div class=\"lic-table-header\">\n <span>License</span>\n <span>Required</span>\n </div>\n <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n <span>{{ row.label }}</span>\n <span>{{ row.value | number }}</span>\n </div>\n <div class=\"table-row total-row\">\n <span>Total</span>\n <span>{{ viewModel.totalRequired | number }}</span>\n </div>\n </div>\n </div>\n\n <ng-template #widgetState>\n <div class=\"state-message\" *ngIf=\"loading\">\n Loading license requirements\u2026\n </div>\n <div class=\"state-message error\" *ngIf=\"!loading && error\">\n Unable to load license requirements.\n </div>\n <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n No license data available.\n </div>\n </ng-template>\n </ng-container>\n</tk-card>\n", styles: ["@import\"https://fonts.googleapis.com/css?family=Poppins:400,100,200,300,500,600,800,700,900\";@import\"https://fonts.googleapis.com/css2?family=Inter:wght@300;400&display=swap\";:host{display:block;width:100%;box-sizing:border-box}.filter-select{font-family:Inter,Inter,sans-serif;font-size:14px;min-width:140px;border:1px solid #e5e7eb;border-radius:.5rem;padding:.5rem .75rem;background-color:#fff}.filter-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select{border:1px solid #e5e7eb;border-radius:.5rem;background-color:#fff}:host ::ng-deep .filter-select.mat-mdc-select:hover{border-color:#d1d5db}:host ::ng-deep .filter-select.mat-mdc-select.mat-focused{border-color:#16a693}:host ::ng-deep .filter-select .mat-mdc-select-trigger{font-size:14px;padding:.5rem .75rem}:host ::ng-deep .filter-select .mat-mdc-select-value{color:#111827}:host ::ng-deep .filter-select .mat-mdc-select-arrow-wrapper{padding-left:.5rem}.widget-body{display:flex;gap:2.5rem;align-items:flex-start}.left-section{flex:2;display:flex;flex-direction:column;gap:1rem}.chart-section{display:flex;gap:1.5rem;align-items:center}.chart-wrapper{width:220px;min-height:220px;flex-shrink:0}.chart-wrapper apx-chart{width:220px!important;height:220px!important}.legend{display:flex;flex-direction:column;gap:.75rem}.legend-item{display:flex;align-items:center;gap:.5rem;font-family:Inter,Inter,sans-serif;font-size:14px}.legend-marker{width:14px;height:14px;border-radius:50%;display:inline-flex}.legend-label{color:#4b5563}.legend-value{margin-left:auto;font-weight:600;color:#111827}.divider{width:1px;min-height:200px;align-self:stretch;background:#e5e7eb}.table-section{flex:3;display:flex;flex-direction:column;gap:.75rem;font-family:Inter,Inter,sans-serif}.lic-table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.lic-table-header{font-weight:600;color:#6b7280;padding-bottom:.5rem;border-bottom:1px solid #e5e7eb}.table-row{color:#111827}.total-row{font-weight:700;margin-top:.5rem;padding-top:.5rem;border-top:1px solid #e5e7eb}.state-message{font-family:Inter,Inter,sans-serif;font-size:14px;color:#6b7280}.state-message.error{color:#b91c1c}@media (max-width: 960px){.widget-body{flex-direction:column}.divider{display:none}.chart-section{justify-content:space-between}}:host ::ng-deep .apexcharts-legend{display:none!important}\n"] }]
182
186
  }], propDecorators: { customerId: [{
183
187
  type: Input
184
188
  }], token: [{
185
189
  type: Input
186
190
  }] } });
187
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"licences-requirement.component.js","sourceRoot":"","sources":["../../../../../../../../projects/tuki/widgets/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.ts","../../../../../../../../projects/tuki/widgets/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, OnDestroy, OnInit, SimpleChanges, inject } from '@angular/core';\nimport { ApexChart, ApexDataLabels, ApexLegend, ApexNonAxisChartSeries, ApexPlotOptions } from 'ng-apexcharts';\nimport { catchError, finalize, of, Subscription } from 'rxjs';\nimport { CcReadinessService } from '../../cc-readiness.service';\nimport { WebexLicenseUsageSummary } from '../../types/cc-readiness';\n\ninterface LicenseLegendItem {\n  label: string;\n  count: number;\n  percentage: number;\n  color: string;\n}\n\ninterface LicenseTableRow {\n  label: string;\n  value: number;\n}\n\ninterface LicenseChartOptions {\n  series: ApexNonAxisChartSeries;\n  chart: ApexChart;\n  labels: string[];\n  colors: string[];\n  legend: ApexLegend;\n  plotOptions: ApexPlotOptions;\n  dataLabels: ApexDataLabels;\n}\n\ninterface LicenseRequirementViewModel {\n  chartOptions: LicenseChartOptions;\n  legendItems: LicenseLegendItem[];\n  rows: LicenseTableRow[];\n  totalRequired: number;\n}\n\nconst LEGEND_COLORS = {\n  existing: '#16A693',\n  missing: '#808080'\n};\n\n@Component({\n  selector: 'cc-licenses-requirement',\n  templateUrl: './licences-requirement.component.html',\n  styleUrls: ['./licences-requirement.component.scss']\n})\nexport class LicencesRequirementComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() customerId!: number;\n  @Input() token!: string;\n\n  private readonly ccReadinessService = inject(CcReadinessService);\n  private subscription?: Subscription;\n\n  viewModel: LicenseRequirementViewModel | null = null;\n  loading = false;\n  error = false;\n\n  filters: string[] = ['All', 'Professional licenses', 'Workspace licenses'];\n  selectedSummaryIndex = 0;\n\n  private summary: WebexLicenseUsageSummary | null = null;\n\n  ngOnInit(): void {\n    this.loadSummaries();\n  }\n\n  ngOnDestroy(): void {\n    this.subscription?.unsubscribe();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if ((changes['customerId'] && !changes['customerId'].firstChange) ||\n      (changes['token'] && !changes['token'].firstChange)) {\n      this.loadSummaries();\n    }\n  }\n\n  onFilterChange(index: number): void {\n    if (!isNaN(index) && index >= 0 && index < this.filters.length) {\n      this.selectFilter(index);\n    }\n  }\n\n  private loadSummaries(): void {\n    const numericCustomerId = Number(this.customerId);\n    if (!numericCustomerId || !this.token) {\n      return;\n    }\n\n    this.subscription?.unsubscribe();\n    this.loading = true;\n    this.error = false;\n    this.viewModel = null;\n    this.selectedSummaryIndex = 0;\n\n    this.ccReadinessService.setToken(this.token);\n    this.subscription = this.ccReadinessService.fetchWebexLicenseUsageSummary(numericCustomerId)\n      .pipe(\n        catchError(error => {\n          console.error('Unable to load Webex license usage summary', error);\n          this.error = true;\n          return of<WebexLicenseUsageSummary[]>([]);\n        }),\n        finalize(() => this.loading = false)\n      )\n      .subscribe(items => {\n        console.log('Raw API response:', items);\n        // Handle both array and single object responses\n        if (Array.isArray(items) && items.length > 0) {\n          this.summary = items[0];\n        } else if (items && typeof items === 'object' && !Array.isArray(items)) {\n          // If single object returned, use it directly\n          this.summary = items as unknown as WebexLicenseUsageSummary;\n        } else {\n          this.summary = null;\n        }\n\n        console.log('Processed summary:', this.summary);\n        this.selectFilter(0);\n      });\n  }\n\n  private selectFilter(index: number): void {\n    this.selectedSummaryIndex = index;\n    if (!this.summary) {\n      this.viewModel = null;\n      return;\n    }\n    this.viewModel = this.mapSummaryToViewModel(this.summary, index);\n    console.log('Created viewModel for filter index', index, ':', this.viewModel);\n  }\n\n  private mapSummaryToViewModel(summary: WebexLicenseUsageSummary, filterIndex: number): LicenseRequirementViewModel {\n    console.log('Mapping summary to viewModel:', summary, 'filterIndex:', filterIndex);\n\n    // Extract data from webexReadinessLicenseStats and license counts\n    const personStat = summary.webexReadinessLicenseStats?.find(stat => stat.feature === 'PERSON');\n    const deviceStat = summary.webexReadinessLicenseStats?.find(stat => stat.feature === 'ORPHANED_DEVICE');\n\n    const professionalNeeded = personStat?.totalCount ?? 0;\n    const workspaceNeeded = deviceStat?.totalCount ?? 0;\n\n    const professionalOwned = Number(summary.totalProfessionalLicenseCount ?? 0);\n    const workspaceOwned = Number(summary.totalWorkspaceProfessionalLicenseCount ?? 0);\n\n    // Filter data based on selection\n    let existingCount: number;\n    let missingCount: number;\n    let rows: LicenseTableRow[];\n\n    if (filterIndex === 0) {\n      // All - show both\n      existingCount = professionalOwned + workspaceOwned;\n      const totalNeeded = professionalNeeded + workspaceNeeded;\n      missingCount = Math.max(0, totalNeeded - existingCount);\n\n      rows = [\n        { label: 'Professional licenses', value: professionalNeeded },\n        { label: 'Workspace licenses', value: workspaceNeeded }\n      ];\n    } else if (filterIndex === 1) {\n      // Professional licenses only\n      existingCount = professionalOwned;\n      missingCount = Math.max(0, professionalNeeded - professionalOwned);\n\n      rows = [{ label: 'Professional licenses', value: professionalNeeded }];\n    } else {\n      // Workspace licenses only\n      existingCount = workspaceOwned;\n      missingCount = Math.max(0, workspaceNeeded - workspaceOwned);\n\n      rows = [{ label: 'Workspace licenses', value: workspaceNeeded }];\n    }\n\n    const totalLicenses = existingCount + missingCount;\n\n    const legendItems: LicenseLegendItem[] = [\n      {\n        label: 'Existing licenses',\n        count: existingCount,\n        percentage: totalLicenses > 0 ? (existingCount / totalLicenses) * 100 : 0,\n        color: LEGEND_COLORS.existing\n      },\n      {\n        label: 'Missing licenses',\n        count: missingCount,\n        percentage: totalLicenses > 0 ? (missingCount / totalLicenses) * 100 : 0,\n        color: LEGEND_COLORS.missing\n      }\n    ];\n\n    console.log('Legend items:', legendItems);\n    console.log('Table rows:', rows);\n\n    const totalRequired = rows.reduce((acc, row) => acc + row.value, 0);\n\n    return {\n      chartOptions: this.buildChartOptions(legendItems),\n      legendItems,\n      rows,\n      totalRequired\n    };\n  }\n\n  private buildChartOptions(legendItems: LicenseLegendItem[]): LicenseChartOptions {\n    return {\n      series: legendItems.map(item => item.count),\n      chart: {\n        type: 'donut',\n        height: 220\n      },\n      labels: legendItems.map(item => item.label),\n      colors: legendItems.map(item => item.color),\n      legend: {\n        show: false\n      },\n      plotOptions: {\n        pie: {\n          donut: {\n            size: '75%'\n          }\n        }\n      },\n      dataLabels: {\n        enabled: false\n      }\n    };\n  }\n\n  private normalizePercentage(value: number): number {\n    if (!isFinite(value)) {\n      return 0;\n    }\n    // Convert decimal to percentage (0.9487 → 94.9)\n    const percentage = value * 100;\n    return Math.round(percentage * 10) / 10;\n  }\n}\n","<tk-card title=\"Licenses requirement\">\n  <ng-container ngProjectAs=\"card-content\">\n    <div\n      class=\"widget-body\"\n      *ngIf=\"!loading && !error && viewModel; else widgetState\"\n    >\n      <div class=\"left-section\">\n        <mat-select\n          [(value)]=\"selectedSummaryIndex\"\n          (selectionChange)=\"onFilterChange($event.value)\"\n          class=\"filter-select\"\n        >\n          <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n            {{ option }}\n          </mat-option>\n        </mat-select>\n\n        <div class=\"chart-section\">\n          <div class=\"chart-wrapper\">\n            <apx-chart\n              [series]=\"viewModel.chartOptions.series\"\n              [chart]=\"viewModel.chartOptions.chart\"\n              [labels]=\"viewModel.chartOptions.labels\"\n              [colors]=\"viewModel.chartOptions.colors\"\n              [legend]=\"viewModel.chartOptions.legend\"\n              [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n              [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n            >\n            </apx-chart>\n          </div>\n          <div class=\"legend\">\n            <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n              <span\n                class=\"legend-marker\"\n                [style.backgroundColor]=\"item.color\"\n              ></span>\n              <span class=\"legend-label\">{{ item.label }}</span>\n              <span class=\"legend-value\">{{ item.count | number }}</span>\n            </div>\n          </div>\n        </div>\n      </div>\n\n      <div class=\"divider\" aria-hidden=\"true\"></div>\n\n      <div class=\"table-section\">\n        <div class=\"lic-table-header\">\n          <span>License</span>\n          <span>Required</span>\n        </div>\n        <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n          <span>{{ row.label }}</span>\n          <span>{{ row.value | number }}</span>\n        </div>\n        <div class=\"table-row total-row\">\n          <span>Total</span>\n          <span>{{ viewModel.totalRequired | number }}</span>\n        </div>\n      </div>\n    </div>\n\n    <ng-template #widgetState>\n      <div class=\"state-message\" *ngIf=\"loading\">\n        Loading license requirements…\n      </div>\n      <div class=\"state-message error\" *ngIf=\"!loading && error\">\n        Unable to load license requirements.\n      </div>\n      <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n        No license data available.\n      </div>\n    </ng-template>\n  </ng-container>\n</tk-card>\n"]}
191
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"licences-requirement.component.js","sourceRoot":"","sources":["../../../../../../../../projects/tuki/widgets/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.ts","../../../../../../../../projects/tuki/widgets/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.html"],"names":[],"mappings":"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","sourcesContent":["import { Component, Input, OnChanges, OnDestroy, OnInit, SimpleChanges, inject } from '@angular/core';\nimport { ApexChart, ApexDataLabels, ApexLegend, ApexNonAxisChartSeries, ApexPlotOptions, ApexTooltip } from 'ng-apexcharts';\nimport { catchError, finalize, of, Subscription } from 'rxjs';\nimport { CcReadinessService } from '../../cc-readiness.service';\nimport { WebexLicenseUsageSummary } from '../../types/cc-readiness';\n\ninterface LicenseLegendItem {\n  label: string;\n  count: number;\n  percentage: number;\n  color: string;\n}\n\ninterface LicenseTableRow {\n  label: string;\n  value: number;\n}\n\ninterface LicenseChartOptions {\n  series: ApexNonAxisChartSeries;\n  chart: ApexChart;\n  labels: string[];\n  colors: string[];\n  legend: ApexLegend;\n  plotOptions: ApexPlotOptions;\n  dataLabels: ApexDataLabels;\n  tooltip: ApexTooltip;\n}\n\ninterface LicenseRequirementViewModel {\n  chartOptions: LicenseChartOptions;\n  legendItems: LicenseLegendItem[];\n  rows: LicenseTableRow[];\n  totalRequired: number;\n}\n\nconst LEGEND_COLORS = {\n  existing: '#16A693',\n  missing: '#808080',\n  empty: '#E0E0E0'\n};\n\n@Component({\n  selector: 'cc-licenses-requirement',\n  templateUrl: './licences-requirement.component.html',\n  styleUrls: ['./licences-requirement.component.scss']\n})\nexport class LicencesRequirementComponent implements OnInit, OnDestroy, OnChanges {\n  @Input() customerId!: number;\n  @Input() token!: string;\n\n  private readonly ccReadinessService = inject(CcReadinessService);\n  private subscription?: Subscription;\n\n  viewModel: LicenseRequirementViewModel | null = null;\n  loading = false;\n  error = false;\n\n  filters: string[] = ['All'];\n  selectedSummaryIndex = 0;\n\n  private summary: WebexLicenseUsageSummary | null = null;\n\n  ngOnInit(): void {\n    this.loadSummaries();\n  }\n\n  ngOnDestroy(): void {\n    this.subscription?.unsubscribe();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if ((changes['customerId'] && !changes['customerId'].firstChange) ||\n      (changes['token'] && !changes['token'].firstChange)) {\n      this.loadSummaries();\n    }\n  }\n\n  onFilterChange(index: number): void {\n    if (!isNaN(index) && index >= 0 && index < this.filters.length) {\n      this.selectFilter(index);\n    }\n  }\n\n  private loadSummaries(): void {\n    const numericCustomerId = Number(this.customerId);\n    if (!numericCustomerId || !this.token) {\n      return;\n    }\n\n    this.subscription?.unsubscribe();\n    this.loading = true;\n    this.error = false;\n    this.viewModel = null;\n    this.selectedSummaryIndex = 0;\n\n    this.ccReadinessService.setToken(this.token);\n    this.subscription = this.ccReadinessService.fetchWebexLicenseUsageSummary(numericCustomerId)\n      .pipe(\n        catchError(error => {\n          console.error('Unable to load Webex license usage summary', error);\n          this.error = true;\n          return of<WebexLicenseUsageSummary[]>([]);\n        }),\n        finalize(() => this.loading = false)\n      )\n      .subscribe(items => {\n        console.log('Raw API response:', items);\n        // Handle both array and single object responses\n        if (Array.isArray(items) && items.length > 0) {\n          this.summary = items[0];\n        } else if (items && typeof items === 'object' && !Array.isArray(items)) {\n          // If single object returned, use it directly\n          this.summary = items as unknown as WebexLicenseUsageSummary;\n        } else {\n          this.summary = null;\n        }\n\n        // Build filters dynamically from license names\n        if (this.summary?.licenses?.length) {\n          this.filters = ['All', ...this.summary.licenses.map(l => l.name)];\n        } else {\n          this.filters = ['All'];\n        }\n\n        console.log('Processed summary:', this.summary);\n        this.selectFilter(0);\n      });\n  }\n\n  private selectFilter(index: number): void {\n    this.selectedSummaryIndex = index;\n    if (!this.summary) {\n      this.viewModel = null;\n      return;\n    }\n    this.viewModel = this.mapSummaryToViewModel(this.summary, index);\n    console.log('Created viewModel for filter index', index, ':', this.viewModel);\n  }\n\n  private mapSummaryToViewModel(summary: WebexLicenseUsageSummary, filterIndex: number): LicenseRequirementViewModel {\n    console.log('Mapping summary to viewModel:', summary, 'filterIndex:', filterIndex);\n\n    const licenses = summary.licenses || [];\n\n    let existingCount: number;\n    let missingCount: number;\n    let rows: LicenseTableRow[];\n\n    if (filterIndex === 0) {\n      // \"All\" filter: show totalExistingLicenses vs total consumedUnits\n      existingCount = summary.totalExistingLicenses ?? 0;\n      missingCount = licenses.reduce((acc, l) => acc + l.consumedUnits, 0);\n      rows = licenses.map(l => ({\n        label: l.name,\n        value: l.consumedUnits\n      }));\n    } else {\n      // Specific license filter: show 0 existing vs consumedUnits for that license\n      const selectedLicense = licenses[filterIndex - 1];\n      existingCount = 0;\n      missingCount = selectedLicense?.consumedUnits ?? 0;\n      rows = selectedLicense ? [{ label: selectedLicense.name, value: selectedLicense.consumedUnits }] : [];\n    }\n\n    const totalLicenses = existingCount + missingCount;\n\n    const legendItems: LicenseLegendItem[] = [\n      {\n        label: 'Existing licenses',\n        count: existingCount,\n        percentage: totalLicenses > 0 ? (existingCount / totalLicenses) * 100 : 0,\n        color: LEGEND_COLORS.existing\n      },\n      {\n        label: 'Missing licenses',\n        count: missingCount,\n        percentage: totalLicenses > 0 ? (missingCount / totalLicenses) * 100 : 0,\n        color: LEGEND_COLORS.missing\n      }\n    ];\n\n    console.log('Legend items:', legendItems);\n    console.log('Table rows:', rows);\n\n    const totalRequired = existingCount + missingCount;\n\n    return {\n      chartOptions: this.buildChartOptions(legendItems),\n      legendItems,\n      rows,\n      totalRequired\n    };\n  }\n\n  private buildChartOptions(legendItems: LicenseLegendItem[]): LicenseChartOptions {\n    const totalCount = legendItems.reduce((acc, item) => acc + item.count, 0);\n    const isEmpty = totalCount === 0;\n\n    // For empty chart, show a full light gray donut\n    const series = isEmpty ? [1] : legendItems.map(item => item.count);\n    const labels = isEmpty ? ['No data'] : legendItems.map(item => item.label);\n    const colors = isEmpty ? [LEGEND_COLORS.empty] : legendItems.map(item => item.color);\n\n    return {\n      series,\n      chart: {\n        type: 'donut',\n        height: 220\n      },\n      labels,\n      colors,\n      legend: {\n        show: false\n      },\n      plotOptions: {\n        pie: {\n          donut: {\n            size: '75%'\n          }\n        }\n      },\n      dataLabels: {\n        enabled: false\n      },\n      tooltip: {\n        enabled: !isEmpty\n      }\n    };\n  }\n\n  private normalizePercentage(value: number): number {\n    if (!isFinite(value)) {\n      return 0;\n    }\n    // Convert decimal to percentage (0.9487 → 94.9)\n    const percentage = value * 100;\n    return Math.round(percentage * 10) / 10;\n  }\n}\n","<tk-card title=\"Licenses requirement\">\n  <ng-container ngProjectAs=\"card-content\">\n    <div\n      class=\"widget-body\"\n      *ngIf=\"!loading && !error && viewModel; else widgetState\"\n    >\n      <div class=\"left-section\">\n        <mat-select\n          [(value)]=\"selectedSummaryIndex\"\n          (selectionChange)=\"onFilterChange($event.value)\"\n          class=\"filter-select\"\n        >\n          <mat-option *ngFor=\"let option of filters; index as i\" [value]=\"i\">\n            {{ option }}\n          </mat-option>\n        </mat-select>\n\n        <div class=\"chart-section\">\n          <div class=\"chart-wrapper\">\n            <apx-chart\n              [series]=\"viewModel.chartOptions.series\"\n              [chart]=\"viewModel.chartOptions.chart\"\n              [labels]=\"viewModel.chartOptions.labels\"\n              [colors]=\"viewModel.chartOptions.colors\"\n              [legend]=\"viewModel.chartOptions.legend\"\n              [plotOptions]=\"viewModel.chartOptions.plotOptions\"\n              [dataLabels]=\"viewModel.chartOptions.dataLabels\"\n              [tooltip]=\"viewModel.chartOptions.tooltip\"\n            >\n            </apx-chart>\n          </div>\n          <div class=\"legend\">\n            <div class=\"legend-item\" *ngFor=\"let item of viewModel.legendItems\">\n              <span\n                class=\"legend-marker\"\n                [style.backgroundColor]=\"item.color\"\n              ></span>\n              <span class=\"legend-label\">{{ item.label }}</span>\n              <span class=\"legend-value\">{{ item.count | number }}</span>\n            </div>\n          </div>\n        </div>\n      </div>\n\n      <div class=\"divider\" aria-hidden=\"true\"></div>\n\n      <div class=\"table-section\">\n        <div class=\"lic-table-header\">\n          <span>License</span>\n          <span>Required</span>\n        </div>\n        <div class=\"table-row\" *ngFor=\"let row of viewModel.rows\">\n          <span>{{ row.label }}</span>\n          <span>{{ row.value | number }}</span>\n        </div>\n        <div class=\"table-row total-row\">\n          <span>Total</span>\n          <span>{{ viewModel.totalRequired | number }}</span>\n        </div>\n      </div>\n    </div>\n\n    <ng-template #widgetState>\n      <div class=\"state-message\" *ngIf=\"loading\">\n        Loading license requirements…\n      </div>\n      <div class=\"state-message error\" *ngIf=\"!loading && error\">\n        Unable to load license requirements.\n      </div>\n      <div class=\"state-message\" *ngIf=\"!loading && !error && !viewModel\">\n        No license data available.\n      </div>\n    </ng-template>\n  </ng-container>\n</tk-card>\n"]}
@@ -1,7 +1,7 @@
1
1
  export const API = {
2
2
  READINESS: {
3
3
  UCCX_SERVER_ENTITY_STATS: (customerId) => `/api/uccx/migration/customer/${customerId}/entity-counts`,
4
- LICENSE_USAGE_SUMMARY: (customerId) => `/api/webexreadiness/customers/${customerId}/callinglicesnsecountperfeature`,
4
+ LICENSE_USAGE_SUMMARY: (customerId) => `/api/uccx/migration/customer/${customerId}/cc-license-stats`,
5
5
  }
6
6
  };
7
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmVuZHBvaW50cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R1a2kvd2lkZ2V0cy9jb250YWN0LWNlbnRlci9zaGFyZWQvYXBpLmVuZHBvaW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxHQUFHLEdBQUc7SUFDZixTQUFTLEVBQUU7UUFDUCx3QkFBd0IsRUFBRSxDQUFDLFVBQWtCLEVBQUUsRUFBRSxDQUFDLGdDQUFnQyxVQUFVLGdCQUFnQjtRQUM1RyxxQkFBcUIsRUFBRSxDQUFDLFVBQWtCLEVBQUUsRUFBRSxDQUFDLGlDQUFpQyxVQUFVLGlDQUFpQztLQUM5SDtDQUNKLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgQVBJID0ge1xyXG4gICAgUkVBRElORVNTOiB7XHJcbiAgICAgICAgVUNDWF9TRVJWRVJfRU5USVRZX1NUQVRTOiAoY3VzdG9tZXJJZDogbnVtYmVyKSA9PiBgL2FwaS91Y2N4L21pZ3JhdGlvbi9jdXN0b21lci8ke2N1c3RvbWVySWR9L2VudGl0eS1jb3VudHNgLFxyXG4gICAgICAgIExJQ0VOU0VfVVNBR0VfU1VNTUFSWTogKGN1c3RvbWVySWQ6IG51bWJlcikgPT4gYC9hcGkvd2ViZXhyZWFkaW5lc3MvY3VzdG9tZXJzLyR7Y3VzdG9tZXJJZH0vY2FsbGluZ2xpY2VzbnNlY291bnRwZXJmZWF0dXJlYCxcclxuICAgIH1cclxufSJdfQ==
7
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiYXBpLmVuZHBvaW50cy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R1a2kvd2lkZ2V0cy9jb250YWN0LWNlbnRlci9zaGFyZWQvYXBpLmVuZHBvaW50cy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxNQUFNLENBQUMsTUFBTSxHQUFHLEdBQUc7SUFDZixTQUFTLEVBQUU7UUFDUCx3QkFBd0IsRUFBRSxDQUFDLFVBQWtCLEVBQUUsRUFBRSxDQUFDLGdDQUFnQyxVQUFVLGdCQUFnQjtRQUM1RyxxQkFBcUIsRUFBRSxDQUFDLFVBQWtCLEVBQUUsRUFBRSxDQUFDLGdDQUFnQyxVQUFVLG1CQUFtQjtLQUMvRztDQUNKLENBQUEiLCJzb3VyY2VzQ29udGVudCI6WyJleHBvcnQgY29uc3QgQVBJID0ge1xyXG4gICAgUkVBRElORVNTOiB7XHJcbiAgICAgICAgVUNDWF9TRVJWRVJfRU5USVRZX1NUQVRTOiAoY3VzdG9tZXJJZDogbnVtYmVyKSA9PiBgL2FwaS91Y2N4L21pZ3JhdGlvbi9jdXN0b21lci8ke2N1c3RvbWVySWR9L2VudGl0eS1jb3VudHNgLFxyXG4gICAgICAgIExJQ0VOU0VfVVNBR0VfU1VNTUFSWTogKGN1c3RvbWVySWQ6IG51bWJlcikgPT4gYC9hcGkvdWNjeC9taWdyYXRpb24vY3VzdG9tZXIvJHtjdXN0b21lcklkfS9jYy1saWNlbnNlLXN0YXRzYCxcclxuICAgIH1cclxufSJdfQ==