@tuki-io/tuki-widgets 0.0.126 → 0.0.127

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (37) hide show
  1. package/contact-center/cc-readiness/cc-readiness.module.d.ts +19 -0
  2. package/contact-center/cc-readiness/cc-readiness.service.d.ts +12 -0
  3. package/contact-center/cc-readiness/components/card/card.component.d.ts +6 -0
  4. package/contact-center/cc-readiness/types/cc-readiness.d.ts +20 -0
  5. package/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.d.ts +52 -0
  6. package/contact-center/cc-readiness/widgets/summary-grid/summary-grid.component.d.ts +26 -0
  7. package/contact-center/index.d.ts +5 -0
  8. package/contact-center/public-api.d.ts +3 -0
  9. package/contact-center/shared/api.endpoints.d.ts +6 -0
  10. package/contact-center/shared/api.service.d.ts +20 -0
  11. package/contact-center/shared/material.module.d.ts +15 -0
  12. package/contact-center/shared/shared.module.d.ts +10 -0
  13. package/esm2020/contact-center/cc-readiness/cc-readiness.module.mjs +71 -0
  14. package/esm2020/contact-center/cc-readiness/cc-readiness.service.mjs +35 -0
  15. package/esm2020/contact-center/cc-readiness/components/card/card.component.mjs +13 -0
  16. package/esm2020/contact-center/cc-readiness/types/cc-readiness.mjs +5 -0
  17. package/esm2020/contact-center/cc-readiness/widgets/licences-requirement/licences-requirement.component.mjs +187 -0
  18. package/esm2020/contact-center/cc-readiness/widgets/summary-grid/summary-grid.component.mjs +47 -0
  19. package/esm2020/contact-center/public-api.mjs +4 -0
  20. package/esm2020/contact-center/shared/api.endpoints.mjs +7 -0
  21. package/esm2020/contact-center/shared/api.service.mjs +86 -0
  22. package/esm2020/contact-center/shared/material.module.mjs +76 -0
  23. package/esm2020/contact-center/shared/shared.module.mjs +33 -0
  24. package/esm2020/contact-center/tuki-io-tuki-widgets-contact-center.mjs +5 -0
  25. package/esm2020/di2mt/shared/services/api.service.mjs +1 -1
  26. package/esm2020/users-list/src/users-list.component.mjs +3 -3
  27. package/fesm2015/tuki-io-tuki-widgets-contact-center.mjs +433 -0
  28. package/fesm2015/tuki-io-tuki-widgets-contact-center.mjs.map +1 -0
  29. package/fesm2015/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
  30. package/fesm2015/tuki-io-tuki-widgets-users-list.mjs +2 -2
  31. package/fesm2015/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
  32. package/fesm2020/tuki-io-tuki-widgets-contact-center.mjs +430 -0
  33. package/fesm2020/tuki-io-tuki-widgets-contact-center.mjs.map +1 -0
  34. package/fesm2020/tuki-io-tuki-widgets-di2mt.mjs.map +1 -1
  35. package/fesm2020/tuki-io-tuki-widgets-users-list.mjs +2 -2
  36. package/fesm2020/tuki-io-tuki-widgets-users-list.mjs.map +1 -1
  37. package/package.json +9 -1
@@ -0,0 +1,19 @@
1
+ import * as i0 from "@angular/core";
2
+ import * as i1 from "./widgets/licences-requirement/licences-requirement.component";
3
+ import * as i2 from "./widgets/summary-grid/summary-grid.component";
4
+ import * as i3 from "./components/card/card.component";
5
+ import * as i4 from "@angular/common";
6
+ import * as i5 from "@angular/common/http";
7
+ import * as i6 from "ng-apexcharts";
8
+ import * as i7 from "@angular/material/table";
9
+ import * as i8 from "@angular/material/button";
10
+ import * as i9 from "@angular/material/checkbox";
11
+ import * as i10 from "@angular/material/select";
12
+ import * as i11 from "@angular/material/progress-spinner";
13
+ import * as i12 from "@angular/material/core";
14
+ import * as i13 from "@angular/material/paginator";
15
+ export declare class CcReadinessModule {
16
+ static ɵfac: i0.ɵɵFactoryDeclaration<CcReadinessModule, never>;
17
+ static ɵmod: i0.ɵɵNgModuleDeclaration<CcReadinessModule, [typeof i1.LicencesRequirementComponent, typeof i2.SummaryGridComponent, typeof i3.CardComponent], [typeof i4.CommonModule, typeof i5.HttpClientModule, typeof i6.NgApexchartsModule, typeof i7.MatTableModule, typeof i8.MatButtonModule, typeof i9.MatCheckboxModule, typeof i10.MatSelectModule, typeof i11.MatProgressSpinnerModule, typeof i12.MatOptionModule, typeof i13.MatPaginatorModule, typeof i10.MatSelectModule], [typeof i1.LicencesRequirementComponent, typeof i2.SummaryGridComponent]>;
18
+ static ɵinj: i0.ɵɵInjectorDeclaration<CcReadinessModule>;
19
+ }
@@ -0,0 +1,12 @@
1
+ import { UccxServerEntityStats, WebexLicenseUsageSummary } from './types/cc-readiness';
2
+ import { Observable } from 'rxjs';
3
+ import * as i0 from "@angular/core";
4
+ export declare class CcReadinessService {
5
+ private readonly api;
6
+ constructor();
7
+ fetchUccxServerEntityStats(customerId: number): Observable<UccxServerEntityStats[]>;
8
+ fetchWebexLicenseUsageSummary(customerId: number): Observable<WebexLicenseUsageSummary[]>;
9
+ setToken(token: string): void;
10
+ static ɵfac: i0.ɵɵFactoryDeclaration<CcReadinessService, never>;
11
+ static ɵprov: i0.ɵɵInjectableDeclaration<CcReadinessService>;
12
+ }
@@ -0,0 +1,6 @@
1
+ import * as i0 from "@angular/core";
2
+ export declare class CardComponent {
3
+ title: string;
4
+ static ɵfac: i0.ɵɵFactoryDeclaration<CardComponent, never>;
5
+ static ɵcmp: i0.ɵɵComponentDeclaration<CardComponent, "tk-card", never, { "title": "title"; }, {}, never, ["card-content"], false, never>;
6
+ }
@@ -0,0 +1,20 @@
1
+ export type EntityType = 'applications' | 'triggers' | 'contactServiceQueues' | 'resources' | 'resourceGroups' | 'skills' | 'xml' | 'scripts' | 'audioPrompts' | 'teams' | 'phonebooks' | 'wrapUpCodes';
2
+ export interface UccxServerEntityStats {
3
+ uccxIp: string;
4
+ counts: {
5
+ [key in EntityType]: number;
6
+ };
7
+ }
8
+ export interface WebexLicenseFeatureUsage {
9
+ feature: 'PERSON' | 'ORPHANED_DEVICE';
10
+ totalCount: number;
11
+ usageCount: number;
12
+ unUsedCount: number;
13
+ }
14
+ export interface WebexLicenseUsageSummary {
15
+ webexReadinessLicenseStats: WebexLicenseFeatureUsage[];
16
+ totalProfessionalLicenseCount: number;
17
+ totalWorkspaceProfessionalLicenseCount: number;
18
+ professionalWebexReadinessTotalLicenseStatCoverage: number;
19
+ workspaceWebexReadinessTotalLicenseStatCoverage: number;
20
+ }
@@ -0,0 +1,52 @@
1
+ import { OnChanges, OnDestroy, OnInit, SimpleChanges } from '@angular/core';
2
+ import { ApexChart, ApexDataLabels, ApexLegend, ApexNonAxisChartSeries, ApexPlotOptions } from 'ng-apexcharts';
3
+ import * as i0 from "@angular/core";
4
+ interface LicenseLegendItem {
5
+ label: string;
6
+ count: number;
7
+ percentage: number;
8
+ color: string;
9
+ }
10
+ interface LicenseTableRow {
11
+ label: string;
12
+ value: number;
13
+ }
14
+ interface LicenseChartOptions {
15
+ series: ApexNonAxisChartSeries;
16
+ chart: ApexChart;
17
+ labels: string[];
18
+ colors: string[];
19
+ legend: ApexLegend;
20
+ plotOptions: ApexPlotOptions;
21
+ dataLabels: ApexDataLabels;
22
+ }
23
+ interface LicenseRequirementViewModel {
24
+ chartOptions: LicenseChartOptions;
25
+ legendItems: LicenseLegendItem[];
26
+ rows: LicenseTableRow[];
27
+ totalRequired: number;
28
+ }
29
+ export declare class LicencesRequirementComponent implements OnInit, OnDestroy, OnChanges {
30
+ customerId: number;
31
+ token: string;
32
+ private readonly ccReadinessService;
33
+ private subscription?;
34
+ viewModel: LicenseRequirementViewModel | null;
35
+ loading: boolean;
36
+ error: boolean;
37
+ filters: string[];
38
+ selectedSummaryIndex: number;
39
+ private summary;
40
+ ngOnInit(): void;
41
+ ngOnDestroy(): void;
42
+ ngOnChanges(changes: SimpleChanges): void;
43
+ onFilterChange(index: number): void;
44
+ private loadSummaries;
45
+ private selectFilter;
46
+ private mapSummaryToViewModel;
47
+ private buildChartOptions;
48
+ private normalizePercentage;
49
+ static ɵfac: i0.ɵɵFactoryDeclaration<LicencesRequirementComponent, never>;
50
+ static ɵcmp: i0.ɵɵComponentDeclaration<LicencesRequirementComponent, "cc-licenses-requirement", never, { "customerId": "customerId"; "token": "token"; }, {}, never, never, false, never>;
51
+ }
52
+ export {};
@@ -0,0 +1,26 @@
1
+ import { OnInit } from "@angular/core";
2
+ import { Observable } from "rxjs";
3
+ import * as i0 from "@angular/core";
4
+ interface SummaryGridCount {
5
+ name: string;
6
+ value: number;
7
+ }
8
+ interface SummaryGridItem {
9
+ uccxIp: string;
10
+ counts: SummaryGridCount[];
11
+ }
12
+ export declare class SummaryGridComponent implements OnInit {
13
+ customerId: number;
14
+ token: string;
15
+ private readonly ccReadinessService;
16
+ readonly ENTITY_CONFIG: {
17
+ [key: string]: {
18
+ label: string;
19
+ };
20
+ };
21
+ data$: Observable<SummaryGridItem[]>;
22
+ ngOnInit(): void;
23
+ static ɵfac: i0.ɵɵFactoryDeclaration<SummaryGridComponent, never>;
24
+ static ɵcmp: i0.ɵɵComponentDeclaration<SummaryGridComponent, "cc-summary-grid", never, { "customerId": "customerId"; "token": "token"; }, {}, never, never, false, never>;
25
+ }
26
+ export {};
@@ -0,0 +1,5 @@
1
+ /**
2
+ * Generated bundle index. Do not edit.
3
+ */
4
+ /// <amd-module name="@tuki-io/tuki-widgets/contact-center" />
5
+ export * from './public-api';
@@ -0,0 +1,3 @@
1
+ export * from './cc-readiness/cc-readiness.module';
2
+ export * from './cc-readiness/widgets/summary-grid/summary-grid.component';
3
+ export * from './cc-readiness/widgets/licences-requirement/licences-requirement.component';
@@ -0,0 +1,6 @@
1
+ export declare const API: {
2
+ READINESS: {
3
+ UCCX_SERVER_ENTITY_STATS: (customerId: number) => string;
4
+ LICENSE_USAGE_SUMMARY: (customerId: number) => string;
5
+ };
6
+ };
@@ -0,0 +1,20 @@
1
+ import { HttpClient } from '@angular/common/http';
2
+ import { Observable } from 'rxjs';
3
+ import * as i0 from "@angular/core";
4
+ export declare class APIService {
5
+ private httpClient;
6
+ token: any;
7
+ apiUrl: string;
8
+ constructor(httpClient: HttpClient);
9
+ fetch(url: string, params?: any, cache?: boolean): Observable<any>;
10
+ post(url: string, body: any, params?: {}): Observable<any>;
11
+ postExtended(url: string, body?: null, params?: {}, headers?: {}): Observable<import("@angular/common/http").HttpResponse<Object>>;
12
+ put(url: string, body?: null, params?: {}): Observable<Object>;
13
+ delete(url: string, params?: {}): Observable<Object>;
14
+ fetchPagination(url: string, pageSize: number, pageNumber: number, additionalParams?: {}, cache?: boolean): Observable<any>;
15
+ private prepareEncodedParams;
16
+ private getHeaders;
17
+ private getParameterByName;
18
+ static ɵfac: i0.ɵɵFactoryDeclaration<APIService, never>;
19
+ static ɵprov: i0.ɵɵInjectableDeclaration<APIService>;
20
+ }
@@ -0,0 +1,15 @@
1
+ import * as i0 from "@angular/core";
2
+ import * as i1 from "@angular/material/progress-bar";
3
+ import * as i2 from "@angular/material/tooltip";
4
+ import * as i3 from "@angular/material/table";
5
+ import * as i4 from "@angular/material/checkbox";
6
+ import * as i5 from "@angular/material/button";
7
+ import * as i6 from "@angular/material/icon";
8
+ import * as i7 from "@angular/material/form-field";
9
+ import * as i8 from "@angular/material/input";
10
+ import * as i9 from "@angular/material/select";
11
+ export declare class MaterialModule {
12
+ static ɵfac: i0.ɵɵFactoryDeclaration<MaterialModule, never>;
13
+ static ɵmod: i0.ɵɵNgModuleDeclaration<MaterialModule, never, [typeof i1.MatProgressBarModule, typeof i2.MatTooltipModule, typeof i3.MatTableModule, typeof i4.MatCheckboxModule, typeof i5.MatButtonModule, typeof i6.MatIconModule, typeof i7.MatFormFieldModule, typeof i8.MatInputModule, typeof i9.MatSelectModule], [typeof i1.MatProgressBarModule, typeof i2.MatTooltipModule, typeof i3.MatTableModule, typeof i4.MatCheckboxModule, typeof i5.MatButtonModule, typeof i6.MatIconModule, typeof i7.MatFormFieldModule, typeof i8.MatInputModule, typeof i9.MatSelectModule]>;
14
+ static ɵinj: i0.ɵɵInjectorDeclaration<MaterialModule>;
15
+ }
@@ -0,0 +1,10 @@
1
+ import * as i0 from "@angular/core";
2
+ import * as i1 from "@angular/common";
3
+ import * as i2 from "./material.module";
4
+ import * as i3 from "ng-apexcharts";
5
+ import * as i4 from "@angular/common/http";
6
+ export declare class SharedModule {
7
+ static ɵfac: i0.ɵɵFactoryDeclaration<SharedModule, never>;
8
+ static ɵmod: i0.ɵɵNgModuleDeclaration<SharedModule, never, [typeof i1.CommonModule, typeof i2.MaterialModule, typeof i3.NgApexchartsModule, typeof i4.HttpClientModule], [typeof i2.MaterialModule]>;
9
+ static ɵinj: i0.ɵɵInjectorDeclaration<SharedModule>;
10
+ }
@@ -0,0 +1,71 @@
1
+ import { NgModule } from '@angular/core';
2
+ import { CommonModule } from '@angular/common';
3
+ import { HttpClientModule } from '@angular/common/http';
4
+ import { LicencesRequirementComponent } from './widgets/licences-requirement/licences-requirement.component';
5
+ import { SummaryGridComponent } from './widgets/summary-grid/summary-grid.component';
6
+ import { CardComponent } from './components/card/card.component';
7
+ import { NgApexchartsModule } from 'ng-apexcharts';
8
+ import { MatButtonModule } from '@angular/material/button';
9
+ import { MatCheckboxModule } from '@angular/material/checkbox';
10
+ import { MatOptionModule } from '@angular/material/core';
11
+ import { MatPaginatorModule } from '@angular/material/paginator';
12
+ import { MatProgressSpinnerModule } from '@angular/material/progress-spinner';
13
+ import { MatSelectModule } from '@angular/material/select';
14
+ import { MatTableModule } from '@angular/material/table';
15
+ import * as i0 from "@angular/core";
16
+ export class CcReadinessModule {
17
+ }
18
+ CcReadinessModule.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessModule, deps: [], target: i0.ɵɵFactoryTarget.NgModule });
19
+ CcReadinessModule.ɵmod = i0.ɵɵngDeclareNgModule({ minVersion: "14.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessModule, declarations: [LicencesRequirementComponent,
20
+ SummaryGridComponent,
21
+ CardComponent], imports: [CommonModule,
22
+ HttpClientModule,
23
+ NgApexchartsModule,
24
+ MatTableModule,
25
+ MatButtonModule,
26
+ MatCheckboxModule,
27
+ MatSelectModule,
28
+ MatProgressSpinnerModule,
29
+ MatOptionModule,
30
+ MatPaginatorModule,
31
+ MatSelectModule], exports: [LicencesRequirementComponent,
32
+ SummaryGridComponent] });
33
+ CcReadinessModule.ɵinj = i0.ɵɵngDeclareInjector({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessModule, imports: [CommonModule,
34
+ HttpClientModule,
35
+ NgApexchartsModule,
36
+ MatTableModule,
37
+ MatButtonModule,
38
+ MatCheckboxModule,
39
+ MatSelectModule,
40
+ MatProgressSpinnerModule,
41
+ MatOptionModule,
42
+ MatPaginatorModule,
43
+ MatSelectModule] });
44
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessModule, decorators: [{
45
+ type: NgModule,
46
+ args: [{
47
+ declarations: [
48
+ LicencesRequirementComponent,
49
+ SummaryGridComponent,
50
+ CardComponent
51
+ ],
52
+ imports: [
53
+ CommonModule,
54
+ HttpClientModule,
55
+ NgApexchartsModule,
56
+ MatTableModule,
57
+ MatButtonModule,
58
+ MatCheckboxModule,
59
+ MatSelectModule,
60
+ MatProgressSpinnerModule,
61
+ MatOptionModule,
62
+ MatPaginatorModule,
63
+ MatSelectModule
64
+ ],
65
+ exports: [
66
+ LicencesRequirementComponent,
67
+ SummaryGridComponent
68
+ ]
69
+ }]
70
+ }] });
71
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2MtcmVhZGluZXNzLm1vZHVsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R1a2kvd2lkZ2V0cy9jb250YWN0LWNlbnRlci9jYy1yZWFkaW5lc3MvY2MtcmVhZGluZXNzLm1vZHVsZS50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQSxPQUFPLEVBQUUsUUFBUSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQ3pDLE9BQU8sRUFBRSxZQUFZLEVBQUUsTUFBTSxpQkFBaUIsQ0FBQztBQUMvQyxPQUFPLEVBQUUsZ0JBQWdCLEVBQUUsTUFBTSxzQkFBc0IsQ0FBQztBQUN4RCxPQUFPLEVBQUUsNEJBQTRCLEVBQUUsTUFBTSwrREFBK0QsQ0FBQztBQUM3RyxPQUFPLEVBQUUsb0JBQW9CLEVBQUUsTUFBTSwrQ0FBK0MsQ0FBQztBQUNyRixPQUFPLEVBQUUsYUFBYSxFQUFFLE1BQU0sa0NBQWtDLENBQUM7QUFDakUsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sZUFBZSxDQUFDO0FBR25ELE9BQU8sRUFBRSxlQUFlLEVBQUUsTUFBTSwwQkFBMEIsQ0FBQztBQUMzRCxPQUFPLEVBQUUsaUJBQWlCLEVBQUUsTUFBTSw0QkFBNEIsQ0FBQztBQUMvRCxPQUFPLEVBQUUsZUFBZSxFQUFFLE1BQU0sd0JBQXdCLENBQUM7QUFDekQsT0FBTyxFQUFFLGtCQUFrQixFQUFFLE1BQU0sNkJBQTZCLENBQUM7QUFDakUsT0FBTyxFQUFFLHdCQUF3QixFQUFFLE1BQU0sb0NBQW9DLENBQUM7QUFDOUUsT0FBTyxFQUFFLGVBQWUsRUFBRSxNQUFNLDBCQUEwQixDQUFDO0FBQzNELE9BQU8sRUFBRSxjQUFjLEVBQUUsTUFBTSx5QkFBeUIsQ0FBQzs7QUE0QnpELE1BQU0sT0FBTyxpQkFBaUI7OytHQUFqQixpQkFBaUI7Z0hBQWpCLGlCQUFpQixpQkF0QjFCLDRCQUE0QjtRQUM1QixvQkFBb0I7UUFDcEIsYUFBYSxhQUdiLFlBQVk7UUFDWixnQkFBZ0I7UUFDaEIsa0JBQWtCO1FBQ2xCLGNBQWM7UUFDZCxlQUFlO1FBQ2YsaUJBQWlCO1FBQ2pCLGVBQWU7UUFDZix3QkFBd0I7UUFDeEIsZUFBZTtRQUNmLGtCQUFrQjtRQUNsQixlQUFlLGFBR2YsNEJBQTRCO1FBQzVCLG9CQUFvQjtnSEFHWCxpQkFBaUIsWUFqQjFCLFlBQVk7UUFDWixnQkFBZ0I7UUFDaEIsa0JBQWtCO1FBQ2xCLGNBQWM7UUFDZCxlQUFlO1FBQ2YsaUJBQWlCO1FBQ2pCLGVBQWU7UUFDZix3QkFBd0I7UUFDeEIsZUFBZTtRQUNmLGtCQUFrQjtRQUNsQixlQUFlOzRGQU9OLGlCQUFpQjtrQkF4QjdCLFFBQVE7bUJBQUM7b0JBQ1IsWUFBWSxFQUFFO3dCQUNaLDRCQUE0Qjt3QkFDNUIsb0JBQW9CO3dCQUNwQixhQUFhO3FCQUNkO29CQUNELE9BQU8sRUFBRTt3QkFDUCxZQUFZO3dCQUNaLGdCQUFnQjt3QkFDaEIsa0JBQWtCO3dCQUNsQixjQUFjO3dCQUNkLGVBQWU7d0JBQ2YsaUJBQWlCO3dCQUNqQixlQUFlO3dCQUNmLHdCQUF3Qjt3QkFDeEIsZUFBZTt3QkFDZixrQkFBa0I7d0JBQ2xCLGVBQWU7cUJBQ2hCO29CQUNELE9BQU8sRUFBRTt3QkFDUCw0QkFBNEI7d0JBQzVCLG9CQUFvQjtxQkFDckI7aUJBQ0YiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBOZ01vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQ29tbW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvY29tbW9uJztcbmltcG9ydCB7IEh0dHBDbGllbnRNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9jb21tb24vaHR0cCc7XG5pbXBvcnQgeyBMaWNlbmNlc1JlcXVpcmVtZW50Q29tcG9uZW50IH0gZnJvbSAnLi93aWRnZXRzL2xpY2VuY2VzLXJlcXVpcmVtZW50L2xpY2VuY2VzLXJlcXVpcmVtZW50LmNvbXBvbmVudCc7XG5pbXBvcnQgeyBTdW1tYXJ5R3JpZENvbXBvbmVudCB9IGZyb20gJy4vd2lkZ2V0cy9zdW1tYXJ5LWdyaWQvc3VtbWFyeS1ncmlkLmNvbXBvbmVudCc7XG5pbXBvcnQgeyBDYXJkQ29tcG9uZW50IH0gZnJvbSAnLi9jb21wb25lbnRzL2NhcmQvY2FyZC5jb21wb25lbnQnO1xuaW1wb3J0IHsgTmdBcGV4Y2hhcnRzTW9kdWxlIH0gZnJvbSAnbmctYXBleGNoYXJ0cyc7XG5pbXBvcnQgeyBNYXRlcmlhbE1vZHVsZSB9IGZyb20gJy4uL3NoYXJlZC9tYXRlcmlhbC5tb2R1bGUnO1xuaW1wb3J0IHsgU2hhcmVkTW9kdWxlIH0gZnJvbSAnLi4vc2hhcmVkL3NoYXJlZC5tb2R1bGUnO1xuaW1wb3J0IHsgTWF0QnV0dG9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvYnV0dG9uJztcbmltcG9ydCB7IE1hdENoZWNrYm94TW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY2hlY2tib3gnO1xuaW1wb3J0IHsgTWF0T3B0aW9uTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvY29yZSc7XG5pbXBvcnQgeyBNYXRQYWdpbmF0b3JNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9wYWdpbmF0b3InO1xuaW1wb3J0IHsgTWF0UHJvZ3Jlc3NTcGlubmVyTW9kdWxlIH0gZnJvbSAnQGFuZ3VsYXIvbWF0ZXJpYWwvcHJvZ3Jlc3Mtc3Bpbm5lcic7XG5pbXBvcnQgeyBNYXRTZWxlY3RNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC9zZWxlY3QnO1xuaW1wb3J0IHsgTWF0VGFibGVNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9tYXRlcmlhbC90YWJsZSc7XG5cblxuXG5ATmdNb2R1bGUoe1xuICBkZWNsYXJhdGlvbnM6IFtcbiAgICBMaWNlbmNlc1JlcXVpcmVtZW50Q29tcG9uZW50LFxuICAgIFN1bW1hcnlHcmlkQ29tcG9uZW50LFxuICAgIENhcmRDb21wb25lbnRcbiAgXSxcbiAgaW1wb3J0czogW1xuICAgIENvbW1vbk1vZHVsZSxcbiAgICBIdHRwQ2xpZW50TW9kdWxlLFxuICAgIE5nQXBleGNoYXJ0c01vZHVsZSxcbiAgICBNYXRUYWJsZU1vZHVsZSxcbiAgICBNYXRCdXR0b25Nb2R1bGUsXG4gICAgTWF0Q2hlY2tib3hNb2R1bGUsXG4gICAgTWF0U2VsZWN0TW9kdWxlLFxuICAgIE1hdFByb2dyZXNzU3Bpbm5lck1vZHVsZSxcbiAgICBNYXRPcHRpb25Nb2R1bGUsXG4gICAgTWF0UGFnaW5hdG9yTW9kdWxlLFxuICAgIE1hdFNlbGVjdE1vZHVsZVxuICBdLFxuICBleHBvcnRzOiBbXG4gICAgTGljZW5jZXNSZXF1aXJlbWVudENvbXBvbmVudCxcbiAgICBTdW1tYXJ5R3JpZENvbXBvbmVudFxuICBdXG59KVxuZXhwb3J0IGNsYXNzIENjUmVhZGluZXNzTW9kdWxlIHsgfVxuIl19
@@ -0,0 +1,35 @@
1
+ import { inject, Injectable } from '@angular/core';
2
+ import { APIService } from '../shared/api.service';
3
+ import { API } from '../shared/api.endpoints';
4
+ import { map } from 'rxjs';
5
+ import * as i0 from "@angular/core";
6
+ export class CcReadinessService {
7
+ constructor() {
8
+ this.api = inject(APIService);
9
+ }
10
+ fetchUccxServerEntityStats(customerId) {
11
+ return this.api.fetch(API.READINESS.UCCX_SERVER_ENTITY_STATS(customerId))
12
+ .pipe(map((stats) => {
13
+ stats.forEach(stat => {
14
+ // @ts-ignore
15
+ delete stat.counts.reasonCodes;
16
+ });
17
+ return stats;
18
+ }));
19
+ }
20
+ fetchWebexLicenseUsageSummary(customerId) {
21
+ return this.api.fetch(API.READINESS.LICENSE_USAGE_SUMMARY(customerId));
22
+ }
23
+ setToken(token) {
24
+ this.api.token = token;
25
+ }
26
+ }
27
+ CcReadinessService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
28
+ CcReadinessService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessService, providedIn: 'root' });
29
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CcReadinessService, decorators: [{
30
+ type: Injectable,
31
+ args: [{
32
+ providedIn: 'root'
33
+ }]
34
+ }], ctorParameters: function () { return []; } });
35
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2MtcmVhZGluZXNzLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90dWtpL3dpZGdldHMvY29udGFjdC1jZW50ZXIvY2MtcmVhZGluZXNzL2NjLXJlYWRpbmVzcy5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxNQUFNLEVBQVUsVUFBVSxFQUFFLE1BQU0sZUFBZSxDQUFDO0FBQzNELE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSx1QkFBdUIsQ0FBQztBQUVuRCxPQUFPLEVBQUUsR0FBRyxFQUFFLE1BQU0seUJBQXlCLENBQUM7QUFDOUMsT0FBTyxFQUFjLEdBQUcsRUFBMEIsTUFBTSxNQUFNLENBQUM7O0FBSy9ELE1BQU0sT0FBTyxrQkFBa0I7SUFHN0I7UUFGaUIsUUFBRyxHQUFHLE1BQU0sQ0FBQyxVQUFVLENBQUMsQ0FBQztJQUUxQixDQUFDO0lBRWpCLDBCQUEwQixDQUFDLFVBQWtCO1FBQzNDLE9BQU8sSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLENBQUMsR0FBRyxDQUFDLFNBQVMsQ0FBQyx3QkFBd0IsQ0FBQyxVQUFVLENBQUMsQ0FBQzthQUN0RSxJQUFJLENBQUMsR0FBRyxDQUFDLENBQUMsS0FBOEIsRUFBRSxFQUFFO1lBQzNDLEtBQUssQ0FBQyxPQUFPLENBQUMsSUFBSSxDQUFDLEVBQUU7Z0JBQ25CLGFBQWE7Z0JBQ2IsT0FBTyxJQUFJLENBQUMsTUFBTSxDQUFDLFdBQVcsQ0FBQztZQUNqQyxDQUFDLENBQUMsQ0FBQztZQUNILE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQyxDQUFDLENBQUMsQ0FBQztJQUNSLENBQUM7SUFFRCw2QkFBNkIsQ0FBQyxVQUFrQjtRQUM5QyxPQUFPLElBQUksQ0FBQyxHQUFHLENBQUMsS0FBSyxDQUFDLEdBQUcsQ0FBQyxTQUFTLENBQUMscUJBQXFCLENBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQztJQUN6RSxDQUFDO0lBRUQsUUFBUSxDQUFDLEtBQWE7UUFDcEIsSUFBSSxDQUFDLEdBQUcsQ0FBQyxLQUFLLEdBQUcsS0FBSyxDQUFDO0lBQ3pCLENBQUM7O2dIQXRCVSxrQkFBa0I7b0hBQWxCLGtCQUFrQixjQUZqQixNQUFNOzRGQUVQLGtCQUFrQjtrQkFIOUIsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBpbmplY3QsIEluamVjdCwgSW5qZWN0YWJsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgQVBJU2VydmljZSB9IGZyb20gJy4uL3NoYXJlZC9hcGkuc2VydmljZSc7XG5pbXBvcnQgeyBVY2N4U2VydmVyRW50aXR5U3RhdHMsIFdlYmV4TGljZW5zZVVzYWdlU3VtbWFyeSB9IGZyb20gJy4vdHlwZXMvY2MtcmVhZGluZXNzJztcbmltcG9ydCB7IEFQSSB9IGZyb20gJy4uL3NoYXJlZC9hcGkuZW5kcG9pbnRzJztcbmltcG9ydCB7IGNhdGNoRXJyb3IsIG1hcCwgT2JzZXJ2YWJsZSwgdGhyb3dFcnJvciB9IGZyb20gJ3J4anMnO1xuXG5ASW5qZWN0YWJsZSh7XG4gIHByb3ZpZGVkSW46ICdyb290J1xufSlcbmV4cG9ydCBjbGFzcyBDY1JlYWRpbmVzc1NlcnZpY2Uge1xuICBwcml2YXRlIHJlYWRvbmx5IGFwaSA9IGluamVjdChBUElTZXJ2aWNlKTtcblxuICBjb25zdHJ1Y3RvcigpIHsgfVxuXG4gIGZldGNoVWNjeFNlcnZlckVudGl0eVN0YXRzKGN1c3RvbWVySWQ6IG51bWJlcik6IE9ic2VydmFibGU8VWNjeFNlcnZlckVudGl0eVN0YXRzW10+IHtcbiAgICByZXR1cm4gdGhpcy5hcGkuZmV0Y2goQVBJLlJFQURJTkVTUy5VQ0NYX1NFUlZFUl9FTlRJVFlfU1RBVFMoY3VzdG9tZXJJZCkpXG4gICAgICAucGlwZShtYXAoKHN0YXRzOiBVY2N4U2VydmVyRW50aXR5U3RhdHNbXSkgPT4ge1xuICAgICAgICBzdGF0cy5mb3JFYWNoKHN0YXQgPT4ge1xuICAgICAgICAgIC8vIEB0cy1pZ25vcmVcbiAgICAgICAgICBkZWxldGUgc3RhdC5jb3VudHMucmVhc29uQ29kZXM7XG4gICAgICAgIH0pO1xuICAgICAgICByZXR1cm4gc3RhdHM7XG4gICAgICB9KSk7XG4gIH1cblxuICBmZXRjaFdlYmV4TGljZW5zZVVzYWdlU3VtbWFyeShjdXN0b21lcklkOiBudW1iZXIpOiBPYnNlcnZhYmxlPFdlYmV4TGljZW5zZVVzYWdlU3VtbWFyeVtdPiB7XG4gICAgcmV0dXJuIHRoaXMuYXBpLmZldGNoKEFQSS5SRUFESU5FU1MuTElDRU5TRV9VU0FHRV9TVU1NQVJZKGN1c3RvbWVySWQpKTtcbiAgfVxuXG4gIHNldFRva2VuKHRva2VuOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLmFwaS50b2tlbiA9IHRva2VuO1xuICB9XG59Il19
@@ -0,0 +1,13 @@
1
+ import { Component, Input } from "@angular/core";
2
+ import * as i0 from "@angular/core";
3
+ export class CardComponent {
4
+ }
5
+ CardComponent.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CardComponent, deps: [], target: i0.ɵɵFactoryTarget.Component });
6
+ CardComponent.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "15.2.10", type: CardComponent, selector: "tk-card", inputs: { title: "title" }, ngImport: i0, template: "<article class=\"card\">\r\n <header class=\"card__header\">\r\n <h2>{{ title }}</h2>\r\n <!--<div class=\"card__menu-button\">\r\n <img width=\"18\" src=\"assets/icons/menu_icon.png\" alt=\"Menu Icon\">\r\n </div>-->\r\n </header>\r\n <div class=\"card__content\">\r\n <ng-content select=\"card-content\"></ng-content>\r\n </div>\r\n</article>\r\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;height:100%;box-sizing:border-box}.card{font-family:Inter,Inter,sans-serif;background-color:#fff;border-radius:1rem;padding:2rem;border:1px solid hsla(0,0%,0%,.2);min-width:300px;max-width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;gap:2.5rem}.card__header{position:relative;display:flex;align-items:center;justify-content:space-between}.card__header h2{font-size:16px;margin:0}.card__header .card__menu-button{position:relative;display:flex;align-items:center}.card__content{display:flex;flex-direction:column;gap:2.5rem;flex-grow:1}\n"] });
7
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: CardComponent, decorators: [{
8
+ type: Component,
9
+ args: [{ selector: 'tk-card', template: "<article class=\"card\">\r\n <header class=\"card__header\">\r\n <h2>{{ title }}</h2>\r\n <!--<div class=\"card__menu-button\">\r\n <img width=\"18\" src=\"assets/icons/menu_icon.png\" alt=\"Menu Icon\">\r\n </div>-->\r\n </header>\r\n <div class=\"card__content\">\r\n <ng-content select=\"card-content\"></ng-content>\r\n </div>\r\n</article>\r\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;height:100%;box-sizing:border-box}.card{font-family:Inter,Inter,sans-serif;background-color:#fff;border-radius:1rem;padding:2rem;border:1px solid hsla(0,0%,0%,.2);min-width:300px;max-width:100%;height:100%;display:flex;flex-direction:column;box-sizing:border-box;gap:2.5rem}.card__header{position:relative;display:flex;align-items:center;justify-content:space-between}.card__header h2{font-size:16px;margin:0}.card__header .card__menu-button{position:relative;display:flex;align-items:center}.card__content{display:flex;flex-direction:column;gap:2.5rem;flex-grow:1}\n"] }]
10
+ }], propDecorators: { title: [{
11
+ type: Input
12
+ }] } });
13
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2FyZC5jb21wb25lbnQuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy90dWtpL3dpZGdldHMvY29udGFjdC1jZW50ZXIvY2MtcmVhZGluZXNzL2NvbXBvbmVudHMvY2FyZC9jYXJkLmNvbXBvbmVudC50cyIsIi4uLy4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3R1a2kvd2lkZ2V0cy9jb250YWN0LWNlbnRlci9jYy1yZWFkaW5lc3MvY29tcG9uZW50cy9jYXJkL2NhcmQuY29tcG9uZW50Lmh0bWwiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFNBQVMsRUFBRSxLQUFLLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBT2pELE1BQU0sT0FBTyxhQUFhOzsyR0FBYixhQUFhOytGQUFiLGFBQWEsMkVDUDFCLGlaQVdBOzRGREphLGFBQWE7a0JBTHpCLFNBQVM7K0JBQ0ksU0FBUzs4QkFLVixLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0IH0gZnJvbSBcIkBhbmd1bGFyL2NvcmVcIjtcclxuXHJcbkBDb21wb25lbnQoe1xyXG4gICAgc2VsZWN0b3I6ICd0ay1jYXJkJyxcclxuICAgIHRlbXBsYXRlVXJsOiAnLi9jYXJkLmNvbXBvbmVudC5odG1sJyxcclxuICAgIHN0eWxlVXJsczogWycuL2NhcmQuY29tcG9uZW50LnNjc3MnXVxyXG59KVxyXG5leHBvcnQgY2xhc3MgQ2FyZENvbXBvbmVudCB7XHJcbiAgICBASW5wdXQoKSB0aXRsZSE6IHN0cmluZztcclxufSIsIjxhcnRpY2xlIGNsYXNzPVwiY2FyZFwiPlxyXG4gICAgPGhlYWRlciBjbGFzcz1cImNhcmRfX2hlYWRlclwiPlxyXG4gICAgICAgIDxoMj57eyB0aXRsZSB9fTwvaDI+XHJcbiAgICAgICAgPCEtLTxkaXYgY2xhc3M9XCJjYXJkX19tZW51LWJ1dHRvblwiPlxyXG4gICAgICAgICAgICA8aW1nIHdpZHRoPVwiMThcIiBzcmM9XCJhc3NldHMvaWNvbnMvbWVudV9pY29uLnBuZ1wiIGFsdD1cIk1lbnUgSWNvblwiPlxyXG4gICAgICAgIDwvZGl2Pi0tPlxyXG4gICAgPC9oZWFkZXI+XHJcbiAgICA8ZGl2IGNsYXNzPVwiY2FyZF9fY29udGVudFwiPlxyXG4gICAgICAgIDxuZy1jb250ZW50IHNlbGVjdD1cImNhcmQtY29udGVudFwiPjwvbmctY29udGVudD5cclxuICAgIDwvZGl2PlxyXG48L2FydGljbGU+XHJcbiJdfQ==
@@ -0,0 +1,5 @@
1
+ ;
2
+ ;
3
+ ;
4
+ export {};
5
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2MtcmVhZGluZXNzLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvdHVraS93aWRnZXRzL2NvbnRhY3QtY2VudGVyL2NjLXJlYWRpbmVzcy90eXBlcy9jYy1yZWFkaW5lc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBa0JDLENBQUM7QUFPRCxDQUFDO0FBUUQsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImV4cG9ydCB0eXBlIEVudGl0eVR5cGUgPVxyXG4gICAgfCAnYXBwbGljYXRpb25zJ1xyXG4gICAgfCAndHJpZ2dlcnMnXHJcbiAgICB8ICdjb250YWN0U2VydmljZVF1ZXVlcydcclxuICAgIHwgJ3Jlc291cmNlcydcclxuICAgIHwgJ3Jlc291cmNlR3JvdXBzJ1xyXG4gICAgfCAnc2tpbGxzJ1xyXG4gICAgfCAneG1sJ1xyXG4gICAgfCAnc2NyaXB0cydcclxuICAgIHwgJ2F1ZGlvUHJvbXB0cydcclxuICAgIHwgJ3RlYW1zJ1xyXG4gICAgLy8gfCAncmVhc29uQ29kZXMnXHJcbiAgICB8ICdwaG9uZWJvb2tzJ1xyXG4gICAgfCAnd3JhcFVwQ29kZXMnO1xyXG5cclxuZXhwb3J0IGludGVyZmFjZSBVY2N4U2VydmVyRW50aXR5U3RhdHMge1xyXG4gICAgdWNjeElwOiBzdHJpbmc7XHJcbiAgICBjb3VudHM6IHsgW2tleSBpbiBFbnRpdHlUeXBlXTogbnVtYmVyIH1cclxufTtcclxuXHJcbmV4cG9ydCBpbnRlcmZhY2UgV2ViZXhMaWNlbnNlRmVhdHVyZVVzYWdlIHtcclxuICAgIGZlYXR1cmU6ICdQRVJTT04nIHwgJ09SUEhBTkVEX0RFVklDRScsXHJcbiAgICB0b3RhbENvdW50OiBudW1iZXIsXHJcbiAgICB1c2FnZUNvdW50OiBudW1iZXIsXHJcbiAgICB1blVzZWRDb3VudDogbnVtYmVyXHJcbn07XHJcblxyXG5leHBvcnQgaW50ZXJmYWNlIFdlYmV4TGljZW5zZVVzYWdlU3VtbWFyeSB7XHJcbiAgICB3ZWJleFJlYWRpbmVzc0xpY2Vuc2VTdGF0czogV2ViZXhMaWNlbnNlRmVhdHVyZVVzYWdlW10sXHJcbiAgICB0b3RhbFByb2Zlc3Npb25hbExpY2Vuc2VDb3VudDogbnVtYmVyLFxyXG4gICAgdG90YWxXb3Jrc3BhY2VQcm9mZXNzaW9uYWxMaWNlbnNlQ291bnQ6IG51bWJlcixcclxuICAgIHByb2Zlc3Npb25hbFdlYmV4UmVhZGluZXNzVG90YWxMaWNlbnNlU3RhdENvdmVyYWdlOiBudW1iZXIsXHJcbiAgICB3b3Jrc3BhY2VXZWJleFJlYWRpbmVzc1RvdGFsTGljZW5zZVN0YXRDb3ZlcmFnZTogbnVtYmVyXHJcbn07Il19
@@ -0,0 +1,187 @@
1
+ import { Component, Input, inject } from '@angular/core';
2
+ import { catchError, finalize, of } from 'rxjs';
3
+ import { CcReadinessService } from '../../cc-readiness.service';
4
+ import * as i0 from "@angular/core";
5
+ import * as i1 from "@angular/common";
6
+ import * as i2 from "ng-apexcharts";
7
+ import * as i3 from "@angular/material/select";
8
+ import * as i4 from "@angular/material/core";
9
+ import * as i5 from "../../components/card/card.component";
10
+ const LEGEND_COLORS = {
11
+ existing: '#16A693',
12
+ missing: '#808080'
13
+ };
14
+ export class LicencesRequirementComponent {
15
+ constructor() {
16
+ this.ccReadinessService = inject(CcReadinessService);
17
+ this.viewModel = null;
18
+ this.loading = false;
19
+ this.error = false;
20
+ this.filters = ['All', 'Professional licenses', 'Workspace licenses'];
21
+ this.selectedSummaryIndex = 0;
22
+ this.summary = null;
23
+ }
24
+ ngOnInit() {
25
+ this.loadSummaries();
26
+ }
27
+ ngOnDestroy() {
28
+ this.subscription?.unsubscribe();
29
+ }
30
+ ngOnChanges(changes) {
31
+ if ((changes['customerId'] && !changes['customerId'].firstChange) ||
32
+ (changes['token'] && !changes['token'].firstChange)) {
33
+ this.loadSummaries();
34
+ }
35
+ }
36
+ onFilterChange(index) {
37
+ if (!isNaN(index) && index >= 0 && index < this.filters.length) {
38
+ this.selectFilter(index);
39
+ }
40
+ }
41
+ loadSummaries() {
42
+ const numericCustomerId = Number(this.customerId);
43
+ if (!numericCustomerId || !this.token) {
44
+ return;
45
+ }
46
+ this.subscription?.unsubscribe();
47
+ this.loading = true;
48
+ this.error = false;
49
+ this.viewModel = null;
50
+ this.selectedSummaryIndex = 0;
51
+ this.ccReadinessService.setToken(this.token);
52
+ this.subscription = this.ccReadinessService.fetchWebexLicenseUsageSummary(numericCustomerId)
53
+ .pipe(catchError(error => {
54
+ console.error('Unable to load Webex license usage summary', error);
55
+ this.error = true;
56
+ return of([]);
57
+ }), finalize(() => this.loading = false))
58
+ .subscribe(items => {
59
+ console.log('Raw API response:', items);
60
+ // Handle both array and single object responses
61
+ if (Array.isArray(items) && items.length > 0) {
62
+ this.summary = items[0];
63
+ }
64
+ else if (items && typeof items === 'object' && !Array.isArray(items)) {
65
+ // If single object returned, use it directly
66
+ this.summary = items;
67
+ }
68
+ else {
69
+ this.summary = null;
70
+ }
71
+ console.log('Processed summary:', this.summary);
72
+ this.selectFilter(0);
73
+ });
74
+ }
75
+ selectFilter(index) {
76
+ this.selectedSummaryIndex = index;
77
+ if (!this.summary) {
78
+ this.viewModel = null;
79
+ return;
80
+ }
81
+ this.viewModel = this.mapSummaryToViewModel(this.summary, index);
82
+ console.log('Created viewModel for filter index', index, ':', this.viewModel);
83
+ }
84
+ mapSummaryToViewModel(summary, filterIndex) {
85
+ 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
+ let existingCount;
95
+ let missingCount;
96
+ let rows;
97
+ 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 }];
112
+ }
113
+ else {
114
+ // Workspace licenses only
115
+ existingCount = workspaceOwned;
116
+ missingCount = Math.max(0, workspaceNeeded - workspaceOwned);
117
+ rows = [{ label: 'Workspace licenses', value: workspaceNeeded }];
118
+ }
119
+ const totalLicenses = existingCount + missingCount;
120
+ const legendItems = [
121
+ {
122
+ label: 'Existing licenses',
123
+ count: existingCount,
124
+ percentage: totalLicenses > 0 ? (existingCount / totalLicenses) * 100 : 0,
125
+ color: LEGEND_COLORS.existing
126
+ },
127
+ {
128
+ label: 'Missing licenses',
129
+ count: missingCount,
130
+ percentage: totalLicenses > 0 ? (missingCount / totalLicenses) * 100 : 0,
131
+ color: LEGEND_COLORS.missing
132
+ }
133
+ ];
134
+ console.log('Legend items:', legendItems);
135
+ console.log('Table rows:', rows);
136
+ const totalRequired = rows.reduce((acc, row) => acc + row.value, 0);
137
+ return {
138
+ chartOptions: this.buildChartOptions(legendItems),
139
+ legendItems,
140
+ rows,
141
+ totalRequired
142
+ };
143
+ }
144
+ buildChartOptions(legendItems) {
145
+ return {
146
+ series: legendItems.map(item => item.count),
147
+ chart: {
148
+ type: 'donut',
149
+ height: 220
150
+ },
151
+ labels: legendItems.map(item => item.label),
152
+ colors: legendItems.map(item => item.color),
153
+ legend: {
154
+ show: false
155
+ },
156
+ plotOptions: {
157
+ pie: {
158
+ donut: {
159
+ size: '75%'
160
+ }
161
+ }
162
+ },
163
+ dataLabels: {
164
+ enabled: false
165
+ }
166
+ };
167
+ }
168
+ normalizePercentage(value) {
169
+ if (!isFinite(value)) {
170
+ return 0;
171
+ }
172
+ // Convert decimal to percentage (0.9487 → 94.9)
173
+ const percentage = value * 100;
174
+ return Math.round(percentage * 10) / 10;
175
+ }
176
+ }
177
+ 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=\"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}.table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.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
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "15.2.10", ngImport: i0, type: LicencesRequirementComponent, decorators: [{
180
+ 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=\"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}.table-header,.table-row{display:flex;justify-content:space-between;font-size:14px}.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
+ }], propDecorators: { customerId: [{
183
+ type: Input
184
+ }], token: [{
185
+ type: Input
186
+ }] } });
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=\"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"]}