@sunbird-cb/cbp-ai 0.0.1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +24 -0
- package/esm2022/lib/ai-cbp-routing.module.mjs +56 -0
- package/esm2022/lib/ai-cbp.component.mjs +14 -0
- package/esm2022/lib/ai-cbp.module.mjs +263 -0
- package/esm2022/lib/ai-cbp.service.mjs +14 -0
- package/esm2022/lib/components/add-course/add-course.component.mjs +444 -0
- package/esm2022/lib/components/add-designation/add-designation.component.mjs +460 -0
- package/esm2022/lib/components/add-personalisation/add-personalisation.component.mjs +33 -0
- package/esm2022/lib/components/approval-request-form/approval-request-form.component.mjs +392 -0
- package/esm2022/lib/components/approval-requests/approval-requests.component.mjs +398 -0
- package/esm2022/lib/components/dashboard/dashboard.component.mjs +265 -0
- package/esm2022/lib/components/delete-role-mapping/delete-role-mapping.component.mjs +53 -0
- package/esm2022/lib/components/delete-role-mapping-popup/delete-role-mapping-popup.component.mjs +39 -0
- package/esm2022/lib/components/edit-cbp-plan/edit-cbp-plan.component.mjs +699 -0
- package/esm2022/lib/components/gap-analysis-recommended-course/gap-analysis-recommended-course.component.mjs +188 -0
- package/esm2022/lib/components/generate-course-recommendation/generate-course-recommendation.component.mjs +1949 -0
- package/esm2022/lib/components/list-popup/list-popup.component.mjs +32 -0
- package/esm2022/lib/components/review-request/review-request.component.mjs +138 -0
- package/esm2022/lib/components/role-mapping-generation/role-mapping-generation.component.mjs +1082 -0
- package/esm2022/lib/components/role-mapping-list/role-mapping-list.component.mjs +714 -0
- package/esm2022/lib/components/suggest-more-courses/suggest-more-courses.component.mjs +370 -0
- package/esm2022/lib/components/update-designation-hierarchy/update-designation-hierarchy.component.mjs +235 -0
- package/esm2022/lib/components/view-cbp-plan/view-cbp-plan.component.mjs +125 -0
- package/esm2022/lib/components/view-course-recommendation/view-course-recommendation.component.mjs +397 -0
- package/esm2022/lib/components/view-final-cbp-plan/view-final-cbp-plan.component.mjs +883 -0
- package/esm2022/lib/modules/initial-screen/initial-screen.component.mjs +127 -0
- package/esm2022/lib/modules/shared/constant/app.constant.mjs +2036 -0
- package/esm2022/lib/modules/shared/directives/clickoutside.directive.mjs +38 -0
- package/esm2022/lib/modules/shared/directives/directive.module.mjs +21 -0
- package/esm2022/lib/modules/shared/pipes/order-by-name.pipe.mjs +23 -0
- package/esm2022/lib/modules/shared/services/event.service.mjs +24 -0
- package/esm2022/lib/modules/shared/services/events.mjs +125 -0
- package/esm2022/lib/modules/shared/services/init.service.mjs +41 -0
- package/esm2022/lib/modules/shared/services/role-mapping.service.mjs +261 -0
- package/esm2022/lib/modules/shared/services/shared.service.mjs +869 -0
- package/esm2022/lib/modules/upload-document-page/progress-dialog/progress-dialog.component.mjs +53 -0
- package/esm2022/lib/modules/upload-document-page/upload-dialog/upload-dialog.component.mjs +186 -0
- package/esm2022/lib/modules/upload-document-page/upload-document-page.component.mjs +367 -0
- package/esm2022/lib/pipe-public-URL/pipe-public-URL.module.mjs +20 -0
- package/esm2022/lib/pipe-public-URL/pipe-public-URL.pipe.mjs +23 -0
- package/esm2022/public-api.mjs +14 -0
- package/esm2022/sunbird-cb-cbp-ai.mjs +5 -0
- package/fesm2022/sunbird-cb-cbp-ai.mjs +12952 -0
- package/fesm2022/sunbird-cb-cbp-ai.mjs.map +1 -0
- package/index.d.ts +6 -0
- package/lib/ai-cbp-routing.module.d.ts +8 -0
- package/lib/ai-cbp-routing.module.d.ts.map +1 -0
- package/lib/ai-cbp.component.d.ts +7 -0
- package/lib/ai-cbp.component.d.ts.map +1 -0
- package/lib/ai-cbp.module.d.ts +68 -0
- package/lib/ai-cbp.module.d.ts.map +1 -0
- package/lib/ai-cbp.service.d.ts +7 -0
- package/lib/ai-cbp.service.d.ts.map +1 -0
- package/lib/components/add-course/add-course.component.d.ts +67 -0
- package/lib/components/add-course/add-course.component.d.ts.map +1 -0
- package/lib/components/add-designation/add-designation.component.d.ts +52 -0
- package/lib/components/add-designation/add-designation.component.d.ts.map +1 -0
- package/lib/components/add-personalisation/add-personalisation.component.d.ts +16 -0
- package/lib/components/add-personalisation/add-personalisation.component.d.ts.map +1 -0
- package/lib/components/approval-request-form/approval-request-form.component.d.ts +48 -0
- package/lib/components/approval-request-form/approval-request-form.component.d.ts.map +1 -0
- package/lib/components/approval-requests/approval-requests.component.d.ts +80 -0
- package/lib/components/approval-requests/approval-requests.component.d.ts.map +1 -0
- package/lib/components/dashboard/dashboard.component.d.ts +55 -0
- package/lib/components/dashboard/dashboard.component.d.ts.map +1 -0
- package/lib/components/delete-role-mapping/delete-role-mapping.component.d.ts +18 -0
- package/lib/components/delete-role-mapping/delete-role-mapping.component.d.ts.map +1 -0
- package/lib/components/delete-role-mapping-popup/delete-role-mapping-popup.component.d.ts +18 -0
- package/lib/components/delete-role-mapping-popup/delete-role-mapping-popup.component.d.ts.map +1 -0
- package/lib/components/edit-cbp-plan/edit-cbp-plan.component.d.ts +94 -0
- package/lib/components/edit-cbp-plan/edit-cbp-plan.component.d.ts.map +1 -0
- package/lib/components/gap-analysis-recommended-course/gap-analysis-recommended-course.component.d.ts +35 -0
- package/lib/components/gap-analysis-recommended-course/gap-analysis-recommended-course.component.d.ts.map +1 -0
- package/lib/components/generate-course-recommendation/generate-course-recommendation.component.d.ts +198 -0
- package/lib/components/generate-course-recommendation/generate-course-recommendation.component.d.ts.map +1 -0
- package/lib/components/list-popup/list-popup.component.d.ts +13 -0
- package/lib/components/list-popup/list-popup.component.d.ts.map +1 -0
- package/lib/components/review-request/review-request.component.d.ts +39 -0
- package/lib/components/review-request/review-request.component.d.ts.map +1 -0
- package/lib/components/role-mapping-generation/role-mapping-generation.component.d.ts +99 -0
- package/lib/components/role-mapping-generation/role-mapping-generation.component.d.ts.map +1 -0
- package/lib/components/role-mapping-list/role-mapping-list.component.d.ts +83 -0
- package/lib/components/role-mapping-list/role-mapping-list.component.d.ts.map +1 -0
- package/lib/components/suggest-more-courses/suggest-more-courses.component.d.ts +48 -0
- package/lib/components/suggest-more-courses/suggest-more-courses.component.d.ts.map +1 -0
- package/lib/components/update-designation-hierarchy/update-designation-hierarchy.component.d.ts +41 -0
- package/lib/components/update-designation-hierarchy/update-designation-hierarchy.component.d.ts.map +1 -0
- package/lib/components/view-cbp-plan/view-cbp-plan.component.d.ts +29 -0
- package/lib/components/view-cbp-plan/view-cbp-plan.component.d.ts.map +1 -0
- package/lib/components/view-course-recommendation/view-course-recommendation.component.d.ts +49 -0
- package/lib/components/view-course-recommendation/view-course-recommendation.component.d.ts.map +1 -0
- package/lib/components/view-final-cbp-plan/view-final-cbp-plan.component.d.ts +73 -0
- package/lib/components/view-final-cbp-plan/view-final-cbp-plan.component.d.ts.map +1 -0
- package/lib/modules/initial-screen/initial-screen.component.d.ts +85 -0
- package/lib/modules/initial-screen/initial-screen.component.d.ts.map +1 -0
- package/lib/modules/shared/constant/app.constant.d.ts +457 -0
- package/lib/modules/shared/constant/app.constant.d.ts.map +1 -0
- package/lib/modules/shared/directives/clickoutside.directive.d.ts +13 -0
- package/lib/modules/shared/directives/clickoutside.directive.d.ts.map +1 -0
- package/lib/modules/shared/directives/directive.module.d.ts +8 -0
- package/lib/modules/shared/directives/directive.module.d.ts.map +1 -0
- package/lib/modules/shared/pipes/order-by-name.pipe.d.ts +8 -0
- package/lib/modules/shared/pipes/order-by-name.pipe.d.ts.map +1 -0
- package/lib/modules/shared/services/event.service.d.ts +12 -0
- package/lib/modules/shared/services/event.service.d.ts.map +1 -0
- package/lib/modules/shared/services/events.d.ts +200 -0
- package/lib/modules/shared/services/events.d.ts.map +1 -0
- package/lib/modules/shared/services/init.service.d.ts +13 -0
- package/lib/modules/shared/services/init.service.d.ts.map +1 -0
- package/lib/modules/shared/services/role-mapping.service.d.ts +21 -0
- package/lib/modules/shared/services/role-mapping.service.d.ts.map +1 -0
- package/lib/modules/shared/services/shared.service.d.ts +108 -0
- package/lib/modules/shared/services/shared.service.d.ts.map +1 -0
- package/lib/modules/upload-document-page/progress-dialog/progress-dialog.component.d.ts +14 -0
- package/lib/modules/upload-document-page/progress-dialog/progress-dialog.component.d.ts.map +1 -0
- package/lib/modules/upload-document-page/upload-dialog/upload-dialog.component.d.ts +29 -0
- package/lib/modules/upload-document-page/upload-dialog/upload-dialog.component.d.ts.map +1 -0
- package/lib/modules/upload-document-page/upload-document-page.component.d.ts +64 -0
- package/lib/modules/upload-document-page/upload-document-page.component.d.ts.map +1 -0
- package/lib/pipe-public-URL/pipe-public-URL.module.d.ts +9 -0
- package/lib/pipe-public-URL/pipe-public-URL.module.d.ts.map +1 -0
- package/lib/pipe-public-URL/pipe-public-URL.pipe.d.ts +11 -0
- package/lib/pipe-public-URL/pipe-public-URL.pipe.d.ts.map +1 -0
- package/package.json +28 -0
- package/public-api.d.ts +11 -0
- package/public-api.d.ts.map +1 -0
- package/sunbird-cb-cbp-ai-0.0.1.tgz +0 -0
- package/sunbird-cb-cbp-ai.d.ts.map +1 -0
|
@@ -0,0 +1,1949 @@
|
|
|
1
|
+
import { MAT_DIALOG_DATA } from '@angular/material/dialog';
|
|
2
|
+
import { Component, Inject, ViewChild } from '@angular/core';
|
|
3
|
+
import { SuggestMoreCoursesComponent } from '../suggest-more-courses/suggest-more-courses.component';
|
|
4
|
+
import { AddCourseComponent } from '../add-course/add-course.component';
|
|
5
|
+
import html2pdf from 'html2pdf.js';
|
|
6
|
+
import { interval, of } from 'rxjs';
|
|
7
|
+
import { switchMap, takeWhile, tap, catchError, finalize } from 'rxjs/operators';
|
|
8
|
+
import * as i0 from "@angular/core";
|
|
9
|
+
import * as i1 from "@angular/material/dialog";
|
|
10
|
+
import * as i2 from "../../modules/shared/services/shared.service";
|
|
11
|
+
import * as i3 from "@angular/material/snack-bar";
|
|
12
|
+
import * as i4 from "@angular/forms";
|
|
13
|
+
import * as i5 from "@angular/common";
|
|
14
|
+
import * as i6 from "@angular/material/legacy-form-field";
|
|
15
|
+
import * as i7 from "@angular/material/legacy-input";
|
|
16
|
+
import * as i8 from "@angular/material/legacy-button";
|
|
17
|
+
import * as i9 from "@angular/material/icon";
|
|
18
|
+
import * as i10 from "@angular/material/legacy-select";
|
|
19
|
+
import * as i11 from "@angular/material/legacy-core";
|
|
20
|
+
import * as i12 from "@angular/material/legacy-checkbox";
|
|
21
|
+
import * as i13 from "@angular/material/progress-spinner";
|
|
22
|
+
import * as i14 from "@angular/material/tabs";
|
|
23
|
+
import * as i15 from "../gap-analysis-recommended-course/gap-analysis-recommended-course.component";
|
|
24
|
+
export class GenerateCourseRecommendationComponent {
|
|
25
|
+
constructor(dialogRef, data, sharedService, snackBar, dialog, fb) {
|
|
26
|
+
this.dialogRef = dialogRef;
|
|
27
|
+
this.data = data;
|
|
28
|
+
this.sharedService = sharedService;
|
|
29
|
+
this.snackBar = snackBar;
|
|
30
|
+
this.dialog = dialog;
|
|
31
|
+
this.fb = fb;
|
|
32
|
+
this.loading = false;
|
|
33
|
+
this.dataLoaded = false; // Track if initial data has been loaded
|
|
34
|
+
this.isRegeneratingWithProgress = false; // Track regeneration with progress loading
|
|
35
|
+
this.recommended_course_id = '';
|
|
36
|
+
// Progressive loading for course generation
|
|
37
|
+
this.currentProcessingStage = '';
|
|
38
|
+
this.progressPercentage = 0;
|
|
39
|
+
this.processingStages = [
|
|
40
|
+
'Analyzing roles and responsibilities...',
|
|
41
|
+
'Detailing competency requirements...',
|
|
42
|
+
'Preparing iGOT course recommendations...',
|
|
43
|
+
'Generating public course suggestions...',
|
|
44
|
+
'Computing comprehensive gap analysis...',
|
|
45
|
+
'Finalizing course recommendation plan...'
|
|
46
|
+
];
|
|
47
|
+
this.stageStartTime = 0;
|
|
48
|
+
this.searchText = '';
|
|
49
|
+
this.filterdCourses = [];
|
|
50
|
+
this.originalData = [];
|
|
51
|
+
this.selectFilterCourses = [];
|
|
52
|
+
this.suggestedCourses = []; // Track iGOT suggested courses separately
|
|
53
|
+
this.userAddedCourses = []; // Track user-added courses separately
|
|
54
|
+
this.mode = 'add';
|
|
55
|
+
this.cbp_plan_id = '';
|
|
56
|
+
this.expandedCompetencies = {}; // Track expanded state for each course and competency type
|
|
57
|
+
this.outerTabActiveIndex = 0;
|
|
58
|
+
this.innerTabActiveIndex = 0;
|
|
59
|
+
this.outerTabActiveText = 'all';
|
|
60
|
+
this.innerTabActiveText = 'all';
|
|
61
|
+
this.selectedCategory = 'all';
|
|
62
|
+
this.competencyCoveredCount = 0;
|
|
63
|
+
this.overallCoverage = 0;
|
|
64
|
+
this.behavioralCompetencyCoveredCount = 0;
|
|
65
|
+
this.behavioralTotalCompetencies = 0;
|
|
66
|
+
this.behavioralCoverage = 0;
|
|
67
|
+
this.functionalCompetencyCoveredCount = 0;
|
|
68
|
+
this.functionalTotalCompetencies = 0;
|
|
69
|
+
this.functionalCoverage = 0;
|
|
70
|
+
this.domainCompetencyCoveredCount = 0;
|
|
71
|
+
this.domainTotalCompetencies = 0;
|
|
72
|
+
this.domainCoverage = 0;
|
|
73
|
+
this.competencyNotMatchedByCategory = [];
|
|
74
|
+
this.competencyMatchedByCategory = [];
|
|
75
|
+
this.menuItems = [
|
|
76
|
+
{ key: 'all', label: 'All Categories' },
|
|
77
|
+
{ key: 'behavioral', label: 'Behavioural' },
|
|
78
|
+
{ key: 'functional', label: 'Functional' },
|
|
79
|
+
{ key: 'domain', label: 'Domain' }
|
|
80
|
+
];
|
|
81
|
+
this.behaviouralNotMatched = [];
|
|
82
|
+
this.functionalNotMatched = [];
|
|
83
|
+
this.domainNotMatched = [];
|
|
84
|
+
this.selectedThemeFilter = '';
|
|
85
|
+
this.originalFilteredCourses = [];
|
|
86
|
+
this.isRegenerating = false;
|
|
87
|
+
this.isPDFDownload = false;
|
|
88
|
+
this.behaviouralMatched = [];
|
|
89
|
+
this.functionalMatched = [];
|
|
90
|
+
this.domainMatched = [];
|
|
91
|
+
this.competenciesType = ['All', 'Behavioural', 'Functional', 'Domain'];
|
|
92
|
+
this.ratings = ['4.5 and above', '4.0 and above', '3.5 and above', '2.5 and above', '1 and above'];
|
|
93
|
+
this.languages = ['English', 'Hindi', 'Tamil', 'Kannada', 'Telugu', 'Malayalam', 'Assamese', 'Bengali', 'Gujarati', 'Marathi', 'Odia', 'Punjabi', 'Konkani', 'Bodo', 'Dogri', 'Kashmiri', 'Maithili', 'Manipuri', 'Nepali', 'Sanskrit', 'Santali', 'Sindhi', 'Urdu'];
|
|
94
|
+
this.durations = ['< 1 Hour', '1-5 Hours', '5+ Hours'];
|
|
95
|
+
this.providers = [];
|
|
96
|
+
this.filteredCompetency = [...this.competenciesType];
|
|
97
|
+
this.filteredRatings = [...this.ratings];
|
|
98
|
+
this.filteredLanguages = [...this.languages];
|
|
99
|
+
this.filteredDurations = [...this.durations];
|
|
100
|
+
this.filteredProviders = [...this.providers];
|
|
101
|
+
this.fullCourseList = [];
|
|
102
|
+
// Selected filters
|
|
103
|
+
this.selectedCompetency = 'All';
|
|
104
|
+
this.selectedRating = null;
|
|
105
|
+
this.selectedLanguage = null;
|
|
106
|
+
this.selectedDuration = null;
|
|
107
|
+
this.selectedProvider = null;
|
|
108
|
+
this.planData = data;
|
|
109
|
+
}
|
|
110
|
+
selectCategory(category) {
|
|
111
|
+
this.selectedCategory = category;
|
|
112
|
+
this.gapAnalysisStats();
|
|
113
|
+
}
|
|
114
|
+
ngOnInit() {
|
|
115
|
+
this.filterForm = this.fb.group({
|
|
116
|
+
competency: [[]],
|
|
117
|
+
rating: [[]],
|
|
118
|
+
language: [[]],
|
|
119
|
+
duration: [[]],
|
|
120
|
+
provider: [[]]
|
|
121
|
+
});
|
|
122
|
+
this.loadComponentData();
|
|
123
|
+
}
|
|
124
|
+
applyFilter(value) {
|
|
125
|
+
console.log('this.searchText', value);
|
|
126
|
+
this.searchText = value;
|
|
127
|
+
this.filterdCourses = this.filterData(this.searchText);
|
|
128
|
+
}
|
|
129
|
+
searchData() {
|
|
130
|
+
// this.filterdCourses = this.filterData(this.searchText);
|
|
131
|
+
}
|
|
132
|
+
selectedFilterCourses(event, item) {
|
|
133
|
+
console.log('event', event);
|
|
134
|
+
console.log('item', item);
|
|
135
|
+
if (event.checked) {
|
|
136
|
+
if (this.selectFilterCourses.indexOf(item?.identifier) < 0) {
|
|
137
|
+
this.selectFilterCourses.push(item?.identifier);
|
|
138
|
+
}
|
|
139
|
+
}
|
|
140
|
+
else {
|
|
141
|
+
const index = this.selectFilterCourses.findIndex(control => control === item.identifier);
|
|
142
|
+
if (index !== -1) {
|
|
143
|
+
this.selectFilterCourses.splice(index, 1);
|
|
144
|
+
}
|
|
145
|
+
}
|
|
146
|
+
console.log('this.selectFilterCourses', this.selectFilterCourses);
|
|
147
|
+
}
|
|
148
|
+
closeDialog() {
|
|
149
|
+
this.dialogRef.close();
|
|
150
|
+
}
|
|
151
|
+
saveCourses() {
|
|
152
|
+
this.loading = true;
|
|
153
|
+
console.log('this.planData', this.planData);
|
|
154
|
+
let reqBody = {
|
|
155
|
+
"role_mapping_id": this.planData.id,
|
|
156
|
+
"recommended_course_id": this.recommended_course_id,
|
|
157
|
+
"course_identifiers": this.selectFilterCourses
|
|
158
|
+
};
|
|
159
|
+
console.log('reqBody, ', reqBody);
|
|
160
|
+
if (this.mode === 'add') {
|
|
161
|
+
this.sharedService.saveCourse(reqBody).subscribe({
|
|
162
|
+
next: (res) => {
|
|
163
|
+
// Success handling
|
|
164
|
+
console.log('Success:', res);
|
|
165
|
+
this.loading = false;
|
|
166
|
+
this.dialogRef.close('saved');
|
|
167
|
+
this.snackBar.open('Courses Saved Successfully', 'X', {
|
|
168
|
+
duration: 3000,
|
|
169
|
+
panelClass: ['snackbar-success']
|
|
170
|
+
});
|
|
171
|
+
//this.successRoleMapping.emit(this.roleMappingForm)
|
|
172
|
+
},
|
|
173
|
+
error: (error) => {
|
|
174
|
+
console.log('error', error);
|
|
175
|
+
this.dialogRef.close();
|
|
176
|
+
// Handle 409 Conflict here
|
|
177
|
+
// alert('Conflict detected: The resource already exists or action conflicts.');
|
|
178
|
+
//this.get
|
|
179
|
+
// Or you can set a UI error message variable
|
|
180
|
+
this.snackBar.open(error?.error?.detail, 'X', {
|
|
181
|
+
duration: 3000,
|
|
182
|
+
panelClass: ['snackbar-error']
|
|
183
|
+
});
|
|
184
|
+
this.loading = false;
|
|
185
|
+
//this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)
|
|
186
|
+
}
|
|
187
|
+
});
|
|
188
|
+
}
|
|
189
|
+
else {
|
|
190
|
+
if (this.cbp_plan_id) {
|
|
191
|
+
this.sharedService.updateCourse(reqBody, this.cbp_plan_id).subscribe({
|
|
192
|
+
next: (res) => {
|
|
193
|
+
// Success handling
|
|
194
|
+
console.log('Success:', res);
|
|
195
|
+
this.loading = false;
|
|
196
|
+
this.dialogRef.close('saved');
|
|
197
|
+
this.snackBar.open('Courses Updated Successfully', 'X', {
|
|
198
|
+
duration: 3000,
|
|
199
|
+
panelClass: ['snackbar-success']
|
|
200
|
+
});
|
|
201
|
+
//this.successRoleMapping.emit(this.roleMappingForm)
|
|
202
|
+
},
|
|
203
|
+
error: (error) => {
|
|
204
|
+
console.log('error', error);
|
|
205
|
+
this.dialogRef.close();
|
|
206
|
+
// Handle 409 Conflict here
|
|
207
|
+
// alert('Conflict detected: The resource already exists or action conflicts.');
|
|
208
|
+
//this.get
|
|
209
|
+
// Or you can set a UI error message variable
|
|
210
|
+
this.snackBar.open(error?.error?.detail, 'X', {
|
|
211
|
+
duration: 3000,
|
|
212
|
+
panelClass: ['snackbar-error']
|
|
213
|
+
});
|
|
214
|
+
this.loading = false;
|
|
215
|
+
//this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)
|
|
216
|
+
}
|
|
217
|
+
});
|
|
218
|
+
}
|
|
219
|
+
else {
|
|
220
|
+
// Handle case when cbp_plan_id is not available
|
|
221
|
+
this.loading = false;
|
|
222
|
+
console.error('cbp_plan_id is not available for update operation');
|
|
223
|
+
this.snackBar.open('Unable to update courses: missing plan ID', 'X', {
|
|
224
|
+
duration: 3000,
|
|
225
|
+
panelClass: ['snackbar-error']
|
|
226
|
+
});
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
getCourses() {
|
|
231
|
+
let role_mapping_id = this.planData.id;
|
|
232
|
+
this.loading = true;
|
|
233
|
+
this.sharedService.getCourse(role_mapping_id).subscribe({
|
|
234
|
+
next: (res) => {
|
|
235
|
+
// Success handling
|
|
236
|
+
this.loading = false;
|
|
237
|
+
console.log('res', res);
|
|
238
|
+
this.cbp_plan_id = res?.id;
|
|
239
|
+
if (res && res?.selected_courses && res?.selected_courses?.length) {
|
|
240
|
+
this.mode = 'edit';
|
|
241
|
+
for (let i = 0; i < res?.selected_courses.length; i++) {
|
|
242
|
+
this.selectFilterCourses.push(res.selected_courses[i]?.identifier);
|
|
243
|
+
}
|
|
244
|
+
}
|
|
245
|
+
//this.successRoleMapping.emit(this.roleMappingForm)
|
|
246
|
+
},
|
|
247
|
+
error: (error) => {
|
|
248
|
+
console.log('error', error);
|
|
249
|
+
this.loading = false;
|
|
250
|
+
// Handle 409 Conflict here
|
|
251
|
+
// alert('Conflict detected: The resource already exists or action conflicts.');
|
|
252
|
+
//this.get
|
|
253
|
+
// Or you can set a UI error message variable
|
|
254
|
+
this.loading = false;
|
|
255
|
+
//this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)
|
|
256
|
+
}
|
|
257
|
+
});
|
|
258
|
+
}
|
|
259
|
+
getSuggestedCourse() {
|
|
260
|
+
let role_mapping_id = this.planData.id;
|
|
261
|
+
this.loading = true;
|
|
262
|
+
this.sharedService.getSuggestedCourses(role_mapping_id).subscribe({
|
|
263
|
+
next: (res) => {
|
|
264
|
+
// Success handling
|
|
265
|
+
this.loading = false;
|
|
266
|
+
console.log('getSuggestedCourses res', res);
|
|
267
|
+
console.log('this.filterdCourses', this.filterdCourses);
|
|
268
|
+
// Store suggested courses separately
|
|
269
|
+
this.suggestedCourses = [...res];
|
|
270
|
+
// Rebuild filterdCourses to include all course types
|
|
271
|
+
this.rebuildFilteredCourses();
|
|
272
|
+
// Update gap analysis stats after suggested courses are added
|
|
273
|
+
this.updateGapAnalysisAfterCoursesUpdate();
|
|
274
|
+
//this.successRoleMapping.emit(this.roleMappingForm)
|
|
275
|
+
},
|
|
276
|
+
error: (error) => {
|
|
277
|
+
console.log('error', error);
|
|
278
|
+
this.loading = false;
|
|
279
|
+
if (error.status === 401) {
|
|
280
|
+
console.log('Unauthorized access - user will be redirected to login');
|
|
281
|
+
}
|
|
282
|
+
else {
|
|
283
|
+
// Handle other errors gracefully
|
|
284
|
+
console.error('Failed to load suggested courses');
|
|
285
|
+
}
|
|
286
|
+
// Or you can set a UI error message variable
|
|
287
|
+
this.loading = false;
|
|
288
|
+
//this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)
|
|
289
|
+
}
|
|
290
|
+
});
|
|
291
|
+
}
|
|
292
|
+
getUserCourse() {
|
|
293
|
+
let role_mapping_id = this.planData.id;
|
|
294
|
+
this.loading = true;
|
|
295
|
+
this.sharedService.getUserCourse(role_mapping_id).subscribe({
|
|
296
|
+
next: (res) => {
|
|
297
|
+
// Success handling
|
|
298
|
+
this.loading = false;
|
|
299
|
+
console.log('getUserCourse API response:', res);
|
|
300
|
+
console.log('Current filterdCourses before adding user courses:', this.filterdCourses);
|
|
301
|
+
// Store user-added courses separately (reset to avoid duplicates)
|
|
302
|
+
this.userAddedCourses = [...res];
|
|
303
|
+
// Process user-added courses to ensure proper structure
|
|
304
|
+
for (let i = 0; i < res.length; i++) {
|
|
305
|
+
const userCourse = res[i];
|
|
306
|
+
console.log(`User course ${i} competencies:`, userCourse.competencies);
|
|
307
|
+
// Ensure user-added courses have the expected structure for display
|
|
308
|
+
if (userCourse && !userCourse.course_type) {
|
|
309
|
+
userCourse.course_type = 'User Added';
|
|
310
|
+
}
|
|
311
|
+
// Add identifier if missing (needed for course selection logic)
|
|
312
|
+
if (userCourse && !userCourse.identifier) {
|
|
313
|
+
userCourse.identifier = userCourse.id || `user_course_${i}_${Date.now()}`;
|
|
314
|
+
}
|
|
315
|
+
}
|
|
316
|
+
// Rebuild filterdCourses to include all course types
|
|
317
|
+
this.rebuildFilteredCourses();
|
|
318
|
+
console.log('filterdCourses after adding user courses:', this.filterdCourses);
|
|
319
|
+
//this.successRoleMapping.emit(this.roleMappingForm)
|
|
320
|
+
},
|
|
321
|
+
error: (error) => {
|
|
322
|
+
console.log('error', error);
|
|
323
|
+
this.loading = false;
|
|
324
|
+
// Handle 409 Conflict here
|
|
325
|
+
// alert('Conflict detected: The resource already exists or action conflicts.');
|
|
326
|
+
//this.get
|
|
327
|
+
// Or you can set a UI error message variable
|
|
328
|
+
this.loading = false;
|
|
329
|
+
//this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)
|
|
330
|
+
}
|
|
331
|
+
});
|
|
332
|
+
}
|
|
333
|
+
checkIfCourseExists(item) {
|
|
334
|
+
let flag = false;
|
|
335
|
+
if (this.selectFilterCourses.indexOf(item?.identifier) > -1) {
|
|
336
|
+
flag = true;
|
|
337
|
+
}
|
|
338
|
+
return flag;
|
|
339
|
+
}
|
|
340
|
+
selectAllCourses(event) {
|
|
341
|
+
if (event.checked) {
|
|
342
|
+
for (let i = 0; i < this.filterdCourses.length; i++) {
|
|
343
|
+
this.selectFilterCourses.push(this.filterdCourses[i].identifier);
|
|
344
|
+
}
|
|
345
|
+
}
|
|
346
|
+
else {
|
|
347
|
+
this.selectFilterCourses = [];
|
|
348
|
+
}
|
|
349
|
+
if (this.cbp_plan_id) {
|
|
350
|
+
this.mode = 'update';
|
|
351
|
+
}
|
|
352
|
+
else {
|
|
353
|
+
this.mode = 'add';
|
|
354
|
+
}
|
|
355
|
+
}
|
|
356
|
+
getSectors(sector) {
|
|
357
|
+
if (sector && sector.length) {
|
|
358
|
+
return Array.isArray(sector) ? sector.join('/ ') : '';
|
|
359
|
+
}
|
|
360
|
+
}
|
|
361
|
+
filterData(searchText) {
|
|
362
|
+
const filter = searchText.trim().toLowerCase();
|
|
363
|
+
return this.originalData.filter(item => {
|
|
364
|
+
const stringified = this.flattenObjectToString(item).toLowerCase();
|
|
365
|
+
return stringified.includes(filter);
|
|
366
|
+
});
|
|
367
|
+
}
|
|
368
|
+
flattenObjectToString(obj) {
|
|
369
|
+
let result = '';
|
|
370
|
+
for (const key in obj) {
|
|
371
|
+
const value = obj[key];
|
|
372
|
+
if (typeof value === 'string') {
|
|
373
|
+
result += ' ' + value;
|
|
374
|
+
}
|
|
375
|
+
else if (Array.isArray(value)) {
|
|
376
|
+
value.forEach(val => {
|
|
377
|
+
if (typeof val === 'string') {
|
|
378
|
+
result += ' ' + val;
|
|
379
|
+
}
|
|
380
|
+
else if (typeof val === 'object') {
|
|
381
|
+
result += ' ' + this.flattenObjectToString(val);
|
|
382
|
+
}
|
|
383
|
+
});
|
|
384
|
+
}
|
|
385
|
+
else if (typeof value === 'object' && value !== null) {
|
|
386
|
+
result += ' ' + this.flattenObjectToString(value);
|
|
387
|
+
}
|
|
388
|
+
}
|
|
389
|
+
return result;
|
|
390
|
+
}
|
|
391
|
+
suggestMoreCourses() {
|
|
392
|
+
// this.dialogRef.close()
|
|
393
|
+
const dialogRefNew = this.dialog.open(SuggestMoreCoursesComponent, {
|
|
394
|
+
width: '1000px',
|
|
395
|
+
data: { recommended_course_id: this.recommended_course_id, role_mapping_id: this.planData.id },
|
|
396
|
+
panelClass: 'view-cbp-plan-popup',
|
|
397
|
+
minHeight: '400px',
|
|
398
|
+
maxHeight: '90vh',
|
|
399
|
+
disableClose: true // Optional: prevent closing with outside click
|
|
400
|
+
});
|
|
401
|
+
dialogRefNew.afterClosed().subscribe(result => {
|
|
402
|
+
if (result === 'saved') {
|
|
403
|
+
console.log('Changes saved!');
|
|
404
|
+
// Refresh data or show a toast here
|
|
405
|
+
this.getSuggestedCourse();
|
|
406
|
+
}
|
|
407
|
+
else {
|
|
408
|
+
this.getSuggestedCourse();
|
|
409
|
+
}
|
|
410
|
+
});
|
|
411
|
+
}
|
|
412
|
+
generateCourseRecommendation(element) {
|
|
413
|
+
console.log('Generate Course Recommendation clicked', element);
|
|
414
|
+
// this.activeRowElement = element
|
|
415
|
+
console.log('Edit Role Mapping clicked', element);
|
|
416
|
+
// Navigate or open modal
|
|
417
|
+
console.log('View CBP Plan clicked', element);
|
|
418
|
+
const dialogRef = this.dialog.open(GenerateCourseRecommendationComponent, {
|
|
419
|
+
width: '1000px',
|
|
420
|
+
data: element,
|
|
421
|
+
panelClass: 'view-cbp-plan-popup',
|
|
422
|
+
minHeight: '400px',
|
|
423
|
+
maxHeight: '90vh',
|
|
424
|
+
disableClose: true // Optional: prevent closing with outside click
|
|
425
|
+
});
|
|
426
|
+
dialogRef.afterClosed().subscribe(result => {
|
|
427
|
+
if (result === 'saved') {
|
|
428
|
+
console.log('Changes saved!');
|
|
429
|
+
// Refresh data or show a toast here
|
|
430
|
+
console.log(this.sharedService.cbpPlanFinalObj);
|
|
431
|
+
if (this.sharedService.cbpPlanFinalObj && this.sharedService.cbpPlanFinalObj.ministry && this.sharedService.cbpPlanFinalObj.ministry.id) {
|
|
432
|
+
this.sharedService.getRoleMappingByStateCenter(this.sharedService.cbpPlanFinalObj.ministry.id).subscribe((res) => {
|
|
433
|
+
console.log('res', res);
|
|
434
|
+
// this.dataSource.data = res
|
|
435
|
+
// this.dataSource.paginator = this.paginator;
|
|
436
|
+
this.originalData = res;
|
|
437
|
+
// console.log('this.dataSource',this.dataSource)
|
|
438
|
+
});
|
|
439
|
+
}
|
|
440
|
+
else {
|
|
441
|
+
}
|
|
442
|
+
}
|
|
443
|
+
});
|
|
444
|
+
}
|
|
445
|
+
getCompetenciesByType(type, index) {
|
|
446
|
+
const course = this.filterdCourses[index];
|
|
447
|
+
if (!course) {
|
|
448
|
+
console.log(`No course found at index ${index}`);
|
|
449
|
+
return [];
|
|
450
|
+
}
|
|
451
|
+
// Handle different competency property names
|
|
452
|
+
// AI Recommended & Public courses use 'competencies'
|
|
453
|
+
// Manually Suggested - iGOT courses use 'competencies_v6'
|
|
454
|
+
// User Added courses use 'competencies'
|
|
455
|
+
let competencies = [];
|
|
456
|
+
if (course.competencies && Array.isArray(course.competencies)) {
|
|
457
|
+
competencies = course.competencies;
|
|
458
|
+
// console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) using 'competencies' property:`, competencies);
|
|
459
|
+
}
|
|
460
|
+
else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {
|
|
461
|
+
competencies = course.competencies_v6;
|
|
462
|
+
// console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) using 'competencies_v6' property:`, competencies);
|
|
463
|
+
}
|
|
464
|
+
else {
|
|
465
|
+
// console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) has no valid competencies property:`, {
|
|
466
|
+
// hasCompetencies: !!course.competencies,
|
|
467
|
+
// competenciesType: typeof course.competencies,
|
|
468
|
+
// hasCompetenciesV6: !!course.competencies_v6,
|
|
469
|
+
// competenciesV6Type: typeof course.competencies_v6,
|
|
470
|
+
// courseKeys: Object.keys(course)
|
|
471
|
+
// });
|
|
472
|
+
}
|
|
473
|
+
if (competencies.length === 0) {
|
|
474
|
+
// console.log(`No competencies found for course ${index} and type ${type}`);
|
|
475
|
+
return [];
|
|
476
|
+
}
|
|
477
|
+
// Normalize the type for comparison (case-insensitive + handle spelling variations)
|
|
478
|
+
const normalizedType = type.toLowerCase().trim();
|
|
479
|
+
const matchedCompetencies = competencies.filter(c => {
|
|
480
|
+
if (!c || !c.competencyAreaName) {
|
|
481
|
+
console.log(`Invalid competency structure in course ${index}:`, c);
|
|
482
|
+
return false;
|
|
483
|
+
}
|
|
484
|
+
const competencyArea = c.competencyAreaName.toLowerCase().trim();
|
|
485
|
+
// Handle both "behavioral" and "behavioural" spellings
|
|
486
|
+
if (normalizedType === 'behavioural' || normalizedType === 'behavioral') {
|
|
487
|
+
return competencyArea === 'behavioral' || competencyArea === 'behavioural';
|
|
488
|
+
}
|
|
489
|
+
// For other types, do case-insensitive comparison
|
|
490
|
+
return competencyArea === normalizedType;
|
|
491
|
+
});
|
|
492
|
+
// console.log(`Found ${matchedCompetencies.length} competencies of type ${type} for course ${index}:`, matchedCompetencies);
|
|
493
|
+
return matchedCompetencies;
|
|
494
|
+
}
|
|
495
|
+
getCompetenciesByBehviouralType(index) {
|
|
496
|
+
const course = this.filterdCourses[index];
|
|
497
|
+
if (!course) {
|
|
498
|
+
return '';
|
|
499
|
+
}
|
|
500
|
+
// Handle different competency property names
|
|
501
|
+
let competencies = [];
|
|
502
|
+
if (course.competencies && Array.isArray(course.competencies)) {
|
|
503
|
+
competencies = course.competencies;
|
|
504
|
+
}
|
|
505
|
+
else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {
|
|
506
|
+
competencies = course.competencies_v6;
|
|
507
|
+
}
|
|
508
|
+
if (competencies.length === 0) {
|
|
509
|
+
return '';
|
|
510
|
+
}
|
|
511
|
+
return competencies
|
|
512
|
+
.filter(c => c && (c.competencyAreaName === 'Behavioral' || c.competencyAreaName === 'Behavioural'))
|
|
513
|
+
.map(c => `${c.competencyThemeName || ''} - ${c.competencySubThemeName || ''}`)
|
|
514
|
+
.join(', ');
|
|
515
|
+
}
|
|
516
|
+
getDisplayedCompetencies(type, index) {
|
|
517
|
+
const competencies = this.getCompetenciesByType(type, index);
|
|
518
|
+
const key = `${index}-${type}`;
|
|
519
|
+
if (this.expandedCompetencies[key]) {
|
|
520
|
+
return competencies;
|
|
521
|
+
}
|
|
522
|
+
return competencies.slice(0, 2);
|
|
523
|
+
}
|
|
524
|
+
toggleCompetencies(type, index) {
|
|
525
|
+
const key = `${index}-${type}`;
|
|
526
|
+
this.expandedCompetencies[key] = !this.expandedCompetencies[key];
|
|
527
|
+
}
|
|
528
|
+
isExpanded(type, index) {
|
|
529
|
+
const key = `${index}-${type}`;
|
|
530
|
+
return this.expandedCompetencies[key] || false;
|
|
531
|
+
}
|
|
532
|
+
hasMoreThanTwo(type, index) {
|
|
533
|
+
return this.getCompetenciesByType(type, index).length > 2;
|
|
534
|
+
}
|
|
535
|
+
getRemainingCount(type, index) {
|
|
536
|
+
const totalCount = this.getCompetenciesByType(type, index).length;
|
|
537
|
+
return totalCount - 2;
|
|
538
|
+
}
|
|
539
|
+
onOuterTabChange(event) {
|
|
540
|
+
this.outerTabActiveIndex = event.index;
|
|
541
|
+
this.outerTabActiveText = event.tab.textLabel;
|
|
542
|
+
console.log('Outer Tab Index:', event.index);
|
|
543
|
+
console.log('Outer Tab Label:', event.tab.textLabel);
|
|
544
|
+
if (event.index === 1) {
|
|
545
|
+
this.gapAnalysisStats();
|
|
546
|
+
}
|
|
547
|
+
if (event.index === 0) {
|
|
548
|
+
// this.getCourses()
|
|
549
|
+
// this.getSuggestedCourse()
|
|
550
|
+
this.getUserCourse();
|
|
551
|
+
}
|
|
552
|
+
}
|
|
553
|
+
onInnerTabChange(event) {
|
|
554
|
+
this.innerTabActiveIndex = event.index;
|
|
555
|
+
this.innerTabActiveText = event.tab.textLabel;
|
|
556
|
+
this.selectedThemeFilter = ''; // Reset theme filter when switching tabs
|
|
557
|
+
console.log('Inner Tab Index:', event.index);
|
|
558
|
+
console.log('Inner Tab Label:', event.tab.textLabel);
|
|
559
|
+
let tabIndex = event.index;
|
|
560
|
+
this.competencyMatchedByCategory = [];
|
|
561
|
+
// Get all available courses (original + suggested + user added)
|
|
562
|
+
const allAvailableCourses = this.getAllAvailableCourses();
|
|
563
|
+
switch (tabIndex) {
|
|
564
|
+
case 0: // All
|
|
565
|
+
this.filterdCourses = allAvailableCourses;
|
|
566
|
+
break;
|
|
567
|
+
case 1: // Behavioral
|
|
568
|
+
this.filterdCourses = this.behavioralFilter(allAvailableCourses);
|
|
569
|
+
this.competencyMatchedByCategory = this.behavioralCompetencyFilter(this.planData.competencies);
|
|
570
|
+
break;
|
|
571
|
+
case 2: // Functional
|
|
572
|
+
this.filterdCourses = this.functionalFilter(allAvailableCourses);
|
|
573
|
+
this.competencyMatchedByCategory = this.functionalCompetencyFilter(this.planData.competencies);
|
|
574
|
+
break;
|
|
575
|
+
case 3: // Domain
|
|
576
|
+
this.filterdCourses = this.domainFilter(allAvailableCourses);
|
|
577
|
+
this.competencyMatchedByCategory = this.domainCompetencyFilter(this.planData.competencies);
|
|
578
|
+
break;
|
|
579
|
+
}
|
|
580
|
+
console.log('this.filterdCourses', this.filterdCourses);
|
|
581
|
+
console.log('this.competencyMatchedByCategory', this.competencyMatchedByCategory);
|
|
582
|
+
}
|
|
583
|
+
getAllAvailableCourses() {
|
|
584
|
+
// Start with original data (AI recommended courses)
|
|
585
|
+
const allCourses = [...this.originalData];
|
|
586
|
+
const seenIdentifiers = new Set();
|
|
587
|
+
// Track original course identifiers
|
|
588
|
+
this.originalData.forEach(course => {
|
|
589
|
+
if (course.identifier)
|
|
590
|
+
seenIdentifiers.add(course.identifier);
|
|
591
|
+
if (course.id)
|
|
592
|
+
seenIdentifiers.add(course.id);
|
|
593
|
+
});
|
|
594
|
+
// Add suggested courses (iGOT platform courses)
|
|
595
|
+
this.suggestedCourses.forEach(course => {
|
|
596
|
+
const courseId = course.identifier || course.id;
|
|
597
|
+
if (courseId && !seenIdentifiers.has(courseId)) {
|
|
598
|
+
allCourses.push(course);
|
|
599
|
+
seenIdentifiers.add(courseId);
|
|
600
|
+
}
|
|
601
|
+
});
|
|
602
|
+
// Add user-added courses
|
|
603
|
+
this.userAddedCourses.forEach(course => {
|
|
604
|
+
const courseId = course.identifier || course.id;
|
|
605
|
+
if (courseId && !seenIdentifiers.has(courseId)) {
|
|
606
|
+
// Ensure user-added courses have proper structure
|
|
607
|
+
if (!course.course_type) {
|
|
608
|
+
course.course_type = 'User Added';
|
|
609
|
+
}
|
|
610
|
+
if (!course.identifier && course.id) {
|
|
611
|
+
course.identifier = course.id;
|
|
612
|
+
}
|
|
613
|
+
allCourses.push(course);
|
|
614
|
+
seenIdentifiers.add(courseId);
|
|
615
|
+
}
|
|
616
|
+
});
|
|
617
|
+
console.log('getAllAvailableCourses breakdown:', {
|
|
618
|
+
originalData: this.originalData.length,
|
|
619
|
+
suggestedCourses: this.suggestedCourses.length,
|
|
620
|
+
userAddedCourses: this.userAddedCourses.length,
|
|
621
|
+
totalAvailable: allCourses.length
|
|
622
|
+
});
|
|
623
|
+
console.log('allCourses--', allCourses);
|
|
624
|
+
allCourses.forEach((item) => {
|
|
625
|
+
if (item && item.organisation && item.organisation.length) {
|
|
626
|
+
if (this.filteredProviders.indexOf(item.organisation[0]) < 0) {
|
|
627
|
+
this.filteredProviders.push(item.organisation[0]);
|
|
628
|
+
}
|
|
629
|
+
}
|
|
630
|
+
});
|
|
631
|
+
this.fullCourseList = allCourses;
|
|
632
|
+
let identifiersArr = [];
|
|
633
|
+
this.fullCourseList.map((item) => {
|
|
634
|
+
identifiersArr.push(item?.identifier);
|
|
635
|
+
});
|
|
636
|
+
this.sharedService.getAdditionalParameterforSuggestedCourses(identifiersArr).subscribe((response) => {
|
|
637
|
+
if (response && response.result && response.result.content && response.result.content.length) {
|
|
638
|
+
for (let i = 0; i < response.result.content.length; i++) {
|
|
639
|
+
for (let j = 0; j < this.fullCourseList.length; j++) {
|
|
640
|
+
if (this.fullCourseList[j]['identifier'] === response.result.content[i]['identifier']) {
|
|
641
|
+
this.fullCourseList[j]['language'] = response.result.content[i]['language'];
|
|
642
|
+
this.fullCourseList[j]['avgRating'] = response.result.content[i]['avgRating'];
|
|
643
|
+
this.fullCourseList[j]['course'] = response.result.content[i]['name'];
|
|
644
|
+
break;
|
|
645
|
+
}
|
|
646
|
+
}
|
|
647
|
+
}
|
|
648
|
+
}
|
|
649
|
+
});
|
|
650
|
+
return allCourses;
|
|
651
|
+
}
|
|
652
|
+
/**
|
|
653
|
+
* Rebuild filterdCourses array with all available course types
|
|
654
|
+
* This ensures consistency across the application
|
|
655
|
+
*/
|
|
656
|
+
rebuildFilteredCourses() {
|
|
657
|
+
this.filterdCourses = this.getAllAvailableCourses();
|
|
658
|
+
console.log('Rebuilt filterdCourses with all course types:', {
|
|
659
|
+
total: this.filterdCourses.length,
|
|
660
|
+
breakdown: {
|
|
661
|
+
original: this.originalData.length,
|
|
662
|
+
suggested: this.suggestedCourses.length,
|
|
663
|
+
userAdded: this.userAddedCourses.length
|
|
664
|
+
}
|
|
665
|
+
});
|
|
666
|
+
}
|
|
667
|
+
behavioralFilter(data) {
|
|
668
|
+
return data.filter(item => {
|
|
669
|
+
if (!item)
|
|
670
|
+
return false;
|
|
671
|
+
// Handle different competency property names
|
|
672
|
+
let competencies = [];
|
|
673
|
+
if (item.competencies && Array.isArray(item.competencies)) {
|
|
674
|
+
competencies = item.competencies;
|
|
675
|
+
}
|
|
676
|
+
else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {
|
|
677
|
+
competencies = item.competencies_v6;
|
|
678
|
+
}
|
|
679
|
+
return competencies.some(c => c && ((c?.competencyAreaName?.toLowerCase() === 'behavioral') || c?.competencyAreaName?.toLowerCase() === 'behavioural'));
|
|
680
|
+
});
|
|
681
|
+
}
|
|
682
|
+
behavioralCompetencyFilter(data) {
|
|
683
|
+
console.log('data--', data);
|
|
684
|
+
const behavioralThemes = data
|
|
685
|
+
.filter(item => item?.type &&
|
|
686
|
+
['behavioral', 'behavioural'].includes(item.type.toLowerCase()))
|
|
687
|
+
.map(item => `${item.theme}-${item.sub_theme}`)
|
|
688
|
+
.filter((theme) => Boolean(theme)); // Ensures type safety
|
|
689
|
+
const uniqueBehavioralThemes = Array.from(new Set(behavioralThemes));
|
|
690
|
+
console.log(uniqueBehavioralThemes);
|
|
691
|
+
return uniqueBehavioralThemes;
|
|
692
|
+
}
|
|
693
|
+
functionalCompetencyFilter(data) {
|
|
694
|
+
const functionalThemes = data
|
|
695
|
+
.filter(item => item?.type &&
|
|
696
|
+
['functional'].includes(item.type.toLowerCase()))
|
|
697
|
+
.map(item => `${item.theme}-${item.sub_theme}`)
|
|
698
|
+
.filter((theme) => Boolean(theme)); // Ensures type safety
|
|
699
|
+
const uniqueFunctionalThemes = Array.from(new Set(functionalThemes));
|
|
700
|
+
console.log(uniqueFunctionalThemes);
|
|
701
|
+
return uniqueFunctionalThemes;
|
|
702
|
+
// console.log(uniqueCompetencyAreaNames);
|
|
703
|
+
}
|
|
704
|
+
domainCompetencyFilter(data) {
|
|
705
|
+
const domainThemes = data
|
|
706
|
+
.filter(item => item?.type &&
|
|
707
|
+
['domain'].includes(item.type.toLowerCase()))
|
|
708
|
+
.map(item => `${item.theme}-${item.sub_theme}`)
|
|
709
|
+
.filter((theme) => Boolean(theme)); // Ensures type safety
|
|
710
|
+
const uniqueDomainThemes = Array.from(new Set(domainThemes));
|
|
711
|
+
console.log(uniqueDomainThemes);
|
|
712
|
+
return uniqueDomainThemes;
|
|
713
|
+
// console.log(uniqueCompetencyAreaNames);
|
|
714
|
+
}
|
|
715
|
+
functionalFilter(data) {
|
|
716
|
+
return data.filter(item => {
|
|
717
|
+
if (!item)
|
|
718
|
+
return false;
|
|
719
|
+
// Handle different competency property names
|
|
720
|
+
let competencies = [];
|
|
721
|
+
if (item.competencies && Array.isArray(item.competencies)) {
|
|
722
|
+
competencies = item.competencies;
|
|
723
|
+
}
|
|
724
|
+
else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {
|
|
725
|
+
competencies = item.competencies_v6;
|
|
726
|
+
}
|
|
727
|
+
return competencies.some(c => c && c?.competencyAreaName?.toLowerCase() === 'functional');
|
|
728
|
+
});
|
|
729
|
+
}
|
|
730
|
+
domainFilter(data) {
|
|
731
|
+
return data.filter(item => {
|
|
732
|
+
if (!item)
|
|
733
|
+
return false;
|
|
734
|
+
// Handle different competency property names
|
|
735
|
+
let competencies = [];
|
|
736
|
+
if (item.competencies && Array.isArray(item.competencies)) {
|
|
737
|
+
competencies = item.competencies;
|
|
738
|
+
}
|
|
739
|
+
else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {
|
|
740
|
+
competencies = item.competencies_v6;
|
|
741
|
+
}
|
|
742
|
+
return competencies.some(c => c && c?.competencyAreaName?.toLowerCase() === 'domain');
|
|
743
|
+
});
|
|
744
|
+
}
|
|
745
|
+
gapAnalysisStats() {
|
|
746
|
+
console.log('this.planData', this.planData);
|
|
747
|
+
// Always use ALL competencies for calculating overall stats
|
|
748
|
+
const allCompetencies = this.planData.competencies;
|
|
749
|
+
// Filter for category-specific calculations
|
|
750
|
+
let categorySpecificCompetencies = this.planData.competencies;
|
|
751
|
+
if (this.selectedCategory === 'all') {
|
|
752
|
+
categorySpecificCompetencies = this.planData.competencies;
|
|
753
|
+
}
|
|
754
|
+
else if (this.selectedCategory === 'behavioral') {
|
|
755
|
+
categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);
|
|
756
|
+
}
|
|
757
|
+
else if (this.selectedCategory === 'functional') {
|
|
758
|
+
categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);
|
|
759
|
+
}
|
|
760
|
+
else if (this.selectedCategory === 'domain') {
|
|
761
|
+
categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);
|
|
762
|
+
}
|
|
763
|
+
// Count ALL competencies by category (not just filtered ones)
|
|
764
|
+
const masterListByCategory = { total: allCompetencies.length, behavioural: 0, functional: 0, domain: 0 };
|
|
765
|
+
// IMPORTANT: Use all available courses for consistent coverage calculation
|
|
766
|
+
// This should match what's available in the Filter By Competency section
|
|
767
|
+
const allCourseCompetencies = [];
|
|
768
|
+
const allAvailableCoursesForStats = this.getAllAvailableCourses();
|
|
769
|
+
allAvailableCoursesForStats.forEach(course => {
|
|
770
|
+
// Handle different competency property names
|
|
771
|
+
let competencies = [];
|
|
772
|
+
if (course && course.competencies && Array.isArray(course.competencies)) {
|
|
773
|
+
competencies = course.competencies;
|
|
774
|
+
}
|
|
775
|
+
else if (course && course.competencies_v6 && Array.isArray(course.competencies_v6)) {
|
|
776
|
+
competencies = course.competencies_v6;
|
|
777
|
+
}
|
|
778
|
+
competencies.forEach((list) => {
|
|
779
|
+
allCourseCompetencies.push(list);
|
|
780
|
+
});
|
|
781
|
+
});
|
|
782
|
+
console.log('allCompetencies', allCompetencies);
|
|
783
|
+
console.log('categorySpecificCompetencies', categorySpecificCompetencies);
|
|
784
|
+
console.log('allCourseCompetencies', allCourseCompetencies);
|
|
785
|
+
// Count all competencies by type for proper totals
|
|
786
|
+
for (let i = 0; i < allCompetencies.length; i++) {
|
|
787
|
+
const competencyType = allCompetencies[i]['type']?.toLowerCase();
|
|
788
|
+
console.log(`Competency ${i}: type="${competencyType}", data:`, allCompetencies[i]);
|
|
789
|
+
if (competencyType === 'behavioural' || competencyType === 'behavioral') {
|
|
790
|
+
masterListByCategory['behavioural'] = masterListByCategory['behavioural'] + 1;
|
|
791
|
+
}
|
|
792
|
+
if (competencyType === 'functional') {
|
|
793
|
+
masterListByCategory['functional'] = masterListByCategory['functional'] + 1;
|
|
794
|
+
}
|
|
795
|
+
if (competencyType === 'domain') {
|
|
796
|
+
masterListByCategory['domain'] = masterListByCategory['domain'] + 1;
|
|
797
|
+
}
|
|
798
|
+
}
|
|
799
|
+
console.log('masterListByCategory after counting:', masterListByCategory);
|
|
800
|
+
// Get matching results for ALL competencies (for overall coverage)
|
|
801
|
+
const overallResult = this.getMatchedCompetencyStats(allCompetencies, allCourseCompetencies);
|
|
802
|
+
// Get matching results for category-specific competencies (for filtered view)
|
|
803
|
+
const categoryResult = this.getMatchedCompetencyStats(categorySpecificCompetencies, allCourseCompetencies);
|
|
804
|
+
// Set competencyCoveredCount based on selected category (for the filtered numbers)
|
|
805
|
+
if (this.selectedCategory === 'all') {
|
|
806
|
+
this.competencyCoveredCount = overallResult['total'];
|
|
807
|
+
}
|
|
808
|
+
else if (this.selectedCategory === 'behavioral') {
|
|
809
|
+
this.competencyCoveredCount = categoryResult['behavioral'] || 0;
|
|
810
|
+
}
|
|
811
|
+
else if (this.selectedCategory === 'functional') {
|
|
812
|
+
this.competencyCoveredCount = categoryResult['functional'] || 0;
|
|
813
|
+
}
|
|
814
|
+
else if (this.selectedCategory === 'domain') {
|
|
815
|
+
this.competencyCoveredCount = categoryResult['domain'] || 0;
|
|
816
|
+
}
|
|
817
|
+
// IMPORTANT: Overall Coverage should ALWAYS be based on ALL competencies, not filtered
|
|
818
|
+
let totalCompetencies = this.planData.competencies.length; // Always use total count
|
|
819
|
+
let overallCoveredCount = overallResult['total']; // Always use overall results
|
|
820
|
+
let mathRound = Math.round((overallCoveredCount / totalCompetencies) * 100);
|
|
821
|
+
this.overallCoverage = `${mathRound}%`;
|
|
822
|
+
console.log('Overall Coverage Calculation:', {
|
|
823
|
+
totalCompetencies: totalCompetencies,
|
|
824
|
+
overallCoveredCount: overallCoveredCount,
|
|
825
|
+
overallCoverage: this.overallCoverage,
|
|
826
|
+
selectedCategory: this.selectedCategory,
|
|
827
|
+
competencyCoveredCount: this.competencyCoveredCount
|
|
828
|
+
});
|
|
829
|
+
// Always calculate all category-specific metrics regardless of selected category
|
|
830
|
+
// This ensures the data is available when switching between categories
|
|
831
|
+
// Use overallResult for consistent calculations across all categories
|
|
832
|
+
this.behavioralCompetencyCoveredCount = overallResult['behavioral'] || 0;
|
|
833
|
+
this.behavioralTotalCompetencies = masterListByCategory['behavioural'];
|
|
834
|
+
let behavioralMathRound = this.behavioralTotalCompetencies > 0 ? Math.round((this.behavioralCompetencyCoveredCount / this.behavioralTotalCompetencies) * 100) : 0;
|
|
835
|
+
this.behavioralCoverage = `${behavioralMathRound}%`;
|
|
836
|
+
this.functionalCompetencyCoveredCount = overallResult['functional'] || 0;
|
|
837
|
+
this.functionalTotalCompetencies = masterListByCategory['functional'];
|
|
838
|
+
let functionalMathRound = this.functionalTotalCompetencies > 0 ? Math.round((this.functionalCompetencyCoveredCount / this.functionalTotalCompetencies) * 100) : 0;
|
|
839
|
+
this.functionalCoverage = `${functionalMathRound}%`;
|
|
840
|
+
this.domainCompetencyCoveredCount = overallResult['domain'] || 0;
|
|
841
|
+
this.domainTotalCompetencies = masterListByCategory['domain'];
|
|
842
|
+
let domainMathRound = this.domainTotalCompetencies > 0 ? Math.round((this.domainCompetencyCoveredCount / this.domainTotalCompetencies) * 100) : 0;
|
|
843
|
+
this.domainCoverage = `${domainMathRound}%`;
|
|
844
|
+
console.log('All category metrics calculated:', {
|
|
845
|
+
behavioral: { covered: this.behavioralCompetencyCoveredCount, total: this.behavioralTotalCompetencies, coverage: this.behavioralCoverage },
|
|
846
|
+
functional: { covered: this.functionalCompetencyCoveredCount, total: this.functionalTotalCompetencies, coverage: this.functionalCoverage },
|
|
847
|
+
domain: { covered: this.domainCompetencyCoveredCount, total: this.domainTotalCompetencies, coverage: this.domainCoverage },
|
|
848
|
+
overall: { covered: overallCoveredCount, total: totalCompetencies, coverage: this.overallCoverage },
|
|
849
|
+
selectedCategory: this.selectedCategory,
|
|
850
|
+
competencyCoveredCountForCategory: this.competencyCoveredCount
|
|
851
|
+
});
|
|
852
|
+
console.log('Overall Result:', overallResult);
|
|
853
|
+
console.log('Category Result:', categoryResult);
|
|
854
|
+
console.log('this.competencyNotMatchedByCategory', this.competencyNotMatchedByCategory);
|
|
855
|
+
this.behaviouralNotMatched = this.getCompetencyByCategoryNotMatching('behavioral');
|
|
856
|
+
this.functionalNotMatched = this.getCompetencyByCategoryNotMatching('functional');
|
|
857
|
+
this.domainNotMatched = this.getCompetencyByCategoryNotMatching('domain');
|
|
858
|
+
this.getBehaviouralMatched();
|
|
859
|
+
this.getFunctionalMatched();
|
|
860
|
+
this.getDomainMatched();
|
|
861
|
+
}
|
|
862
|
+
getMatchedCompetencyStats(primaryArray, secondaryArray) {
|
|
863
|
+
// Convert both arrays to ensure consistent casing
|
|
864
|
+
const counts = {
|
|
865
|
+
behavioral: 0,
|
|
866
|
+
functional: 0,
|
|
867
|
+
domain: 0,
|
|
868
|
+
total: 0
|
|
869
|
+
};
|
|
870
|
+
// Clear the competencyNotMatchedByCategory array for fresh calculation
|
|
871
|
+
this.competencyNotMatchedByCategory = [];
|
|
872
|
+
// Set to keep track of unique matches
|
|
873
|
+
const seen = new Set();
|
|
874
|
+
const matchedCompetencies = new Set();
|
|
875
|
+
for (const primary of primaryArray) {
|
|
876
|
+
const typeKey = primary.type?.toLowerCase().trim();
|
|
877
|
+
const themeKey = primary.theme?.toLowerCase().trim();
|
|
878
|
+
const subThemeKey = primary.sub_theme?.toLowerCase().trim();
|
|
879
|
+
let isMatched = false;
|
|
880
|
+
for (const secondary of secondaryArray) {
|
|
881
|
+
// Normalize secondary keys
|
|
882
|
+
let secType = secondary?.competencyAreaName?.toLowerCase().trim();
|
|
883
|
+
let secTheme = secondary?.competencyThemeName?.toLowerCase().trim();
|
|
884
|
+
let secSubTheme = secondary?.competencySubThemeName?.toLowerCase().trim();
|
|
885
|
+
// Fix for data inconsistency
|
|
886
|
+
if (secType === 'behavioural')
|
|
887
|
+
secType = 'behavioral';
|
|
888
|
+
// Create unique match key for primary competency
|
|
889
|
+
const matchKey = `${typeKey}|${themeKey}|${subThemeKey}`;
|
|
890
|
+
// Enhanced cross-matching logic as requested:
|
|
891
|
+
// 1. Role mapping Theme vs Course Theme
|
|
892
|
+
// 2. Role mapping Sub Theme vs Course Sub Theme
|
|
893
|
+
// 3. Role mapping Theme vs Course Sub Theme
|
|
894
|
+
// 4. Role mapping Sub Theme vs Course Theme
|
|
895
|
+
const themeToThemeMatch = themeKey === secTheme;
|
|
896
|
+
const subThemeToSubThemeMatch = subThemeKey === secSubTheme;
|
|
897
|
+
const themeToSubThemeMatch = themeKey === secSubTheme;
|
|
898
|
+
const subThemeToThemeMatch = subThemeKey === secTheme;
|
|
899
|
+
let matchType = '';
|
|
900
|
+
let isMatch = false;
|
|
901
|
+
if (typeKey === secType && !seen.has(matchKey)) {
|
|
902
|
+
if (themeToThemeMatch) {
|
|
903
|
+
matchType = 'Theme-to-Theme';
|
|
904
|
+
isMatch = true;
|
|
905
|
+
}
|
|
906
|
+
else if (subThemeToSubThemeMatch) {
|
|
907
|
+
matchType = 'SubTheme-to-SubTheme';
|
|
908
|
+
isMatch = true;
|
|
909
|
+
}
|
|
910
|
+
else if (themeToSubThemeMatch) {
|
|
911
|
+
matchType = 'Theme-to-SubTheme';
|
|
912
|
+
isMatch = true;
|
|
913
|
+
}
|
|
914
|
+
else if (subThemeToThemeMatch) {
|
|
915
|
+
matchType = 'SubTheme-to-Theme';
|
|
916
|
+
isMatch = true;
|
|
917
|
+
}
|
|
918
|
+
}
|
|
919
|
+
if (isMatch) {
|
|
920
|
+
console.log(`✅ MATCH FOUND [${matchType}]:`, {
|
|
921
|
+
roleMapping: { type: typeKey, theme: themeKey, subTheme: subThemeKey },
|
|
922
|
+
course: { type: secType, theme: secTheme, subTheme: secSubTheme },
|
|
923
|
+
matchType: matchType
|
|
924
|
+
});
|
|
925
|
+
// Store matched competency info for later use
|
|
926
|
+
let obj = {};
|
|
927
|
+
obj[typeKey] = [themeKey]; // Store the original theme, not the course theme
|
|
928
|
+
this.competencyNotMatchedByCategory.push(obj);
|
|
929
|
+
seen.add(matchKey);
|
|
930
|
+
matchedCompetencies.add(themeKey); // Track matched themes
|
|
931
|
+
isMatched = true;
|
|
932
|
+
if (counts.hasOwnProperty(typeKey)) {
|
|
933
|
+
counts[typeKey]++;
|
|
934
|
+
}
|
|
935
|
+
else {
|
|
936
|
+
counts[typeKey] = 1;
|
|
937
|
+
}
|
|
938
|
+
counts.total++;
|
|
939
|
+
break; // Stop looking for matches for this primary competency
|
|
940
|
+
}
|
|
941
|
+
}
|
|
942
|
+
}
|
|
943
|
+
return counts;
|
|
944
|
+
}
|
|
945
|
+
getCompetencyByCategoryNotMatching(categoryType) {
|
|
946
|
+
// Get all competencies for the given category from role mapping
|
|
947
|
+
let allCategoryCompetencies = [];
|
|
948
|
+
for (let i = 0; i < this.planData.competencies.length; i++) {
|
|
949
|
+
if (this.planData.competencies[i]['type']?.toLowerCase() === categoryType?.toLowerCase()) {
|
|
950
|
+
allCategoryCompetencies.push(this.planData.competencies[i]['theme']?.toLowerCase().trim());
|
|
951
|
+
}
|
|
952
|
+
}
|
|
953
|
+
// Get matched competencies for this category from courses
|
|
954
|
+
let matchedCompetencies = [];
|
|
955
|
+
for (let i = 0; i < this.competencyNotMatchedByCategory.length; i++) {
|
|
956
|
+
if (this.competencyNotMatchedByCategory[i][categoryType?.toLowerCase()]) {
|
|
957
|
+
// Extract matched themes from course competencies
|
|
958
|
+
this.competencyNotMatchedByCategory[i][categoryType?.toLowerCase()].forEach(theme => {
|
|
959
|
+
if (theme) {
|
|
960
|
+
matchedCompetencies.push(theme.toLowerCase().trim());
|
|
961
|
+
}
|
|
962
|
+
});
|
|
963
|
+
}
|
|
964
|
+
}
|
|
965
|
+
// Enhanced matching: check both theme and subtheme from courses against FRAC competencies
|
|
966
|
+
// IMPORTANT: Use all available courses for consistent coverage calculation
|
|
967
|
+
const allCoursesForMatching = this.getAllAvailableCourses();
|
|
968
|
+
allCoursesForMatching.forEach(course => {
|
|
969
|
+
// Handle different competency property names
|
|
970
|
+
let competencies = [];
|
|
971
|
+
if (course && course.competencies && Array.isArray(course.competencies)) {
|
|
972
|
+
competencies = course.competencies;
|
|
973
|
+
}
|
|
974
|
+
else if (course && course.competencies_v6 && Array.isArray(course.competencies_v6)) {
|
|
975
|
+
competencies = course.competencies_v6;
|
|
976
|
+
}
|
|
977
|
+
competencies.forEach((comp) => {
|
|
978
|
+
let secType = comp?.competencyAreaName?.toLowerCase().trim();
|
|
979
|
+
if (secType === 'behavioural')
|
|
980
|
+
secType = 'behavioral';
|
|
981
|
+
if (secType === categoryType?.toLowerCase()) {
|
|
982
|
+
const courseTheme = comp.competencyThemeName?.toLowerCase().trim();
|
|
983
|
+
const courseSubTheme = comp.competencySubThemeName?.toLowerCase().trim();
|
|
984
|
+
// Check if any FRAC competency matches this course competency
|
|
985
|
+
for (let i = 0; i < this.planData.competencies.length; i++) {
|
|
986
|
+
const fracComp = this.planData.competencies[i];
|
|
987
|
+
if (fracComp['type']?.toLowerCase() === categoryType?.toLowerCase()) {
|
|
988
|
+
const fracTheme = fracComp['theme']?.toLowerCase().trim();
|
|
989
|
+
const fracSubTheme = fracComp['sub_theme']?.toLowerCase().trim();
|
|
990
|
+
// Enhanced cross-matching logic (same as in getMatchedCompetencyStats):
|
|
991
|
+
// 1. Course Theme vs FRAC Theme
|
|
992
|
+
// 2. Course Sub Theme vs FRAC Sub Theme
|
|
993
|
+
// 3. Course Theme vs FRAC Sub Theme
|
|
994
|
+
// 4. Course Sub Theme vs FRAC Theme
|
|
995
|
+
const courseThemeToFracTheme = courseTheme === fracTheme;
|
|
996
|
+
const courseSubThemeToFracSubTheme = courseSubTheme === fracSubTheme;
|
|
997
|
+
const courseThemeToFracSubTheme = courseTheme === fracSubTheme;
|
|
998
|
+
const courseSubThemeToFracTheme = courseSubTheme === fracTheme;
|
|
999
|
+
if (courseThemeToFracTheme || courseSubThemeToFracSubTheme ||
|
|
1000
|
+
courseThemeToFracSubTheme || courseSubThemeToFracTheme) {
|
|
1001
|
+
let matchType = '';
|
|
1002
|
+
if (courseThemeToFracTheme)
|
|
1003
|
+
matchType = 'CourseTheme-to-FRACTheme';
|
|
1004
|
+
else if (courseSubThemeToFracSubTheme)
|
|
1005
|
+
matchType = 'CourseSubTheme-to-FRACSubTheme';
|
|
1006
|
+
else if (courseThemeToFracSubTheme)
|
|
1007
|
+
matchType = 'CourseTheme-to-FRACSubTheme';
|
|
1008
|
+
else if (courseSubThemeToFracTheme)
|
|
1009
|
+
matchType = 'CourseSubTheme-to-FRACTheme';
|
|
1010
|
+
console.log(`✅ COMPETENCY MATCH FOUND [${matchType}] for ${categoryType}:`, {
|
|
1011
|
+
course: { theme: courseTheme, subTheme: courseSubTheme },
|
|
1012
|
+
frac: { theme: fracTheme, subTheme: fracSubTheme },
|
|
1013
|
+
matchType: matchType
|
|
1014
|
+
});
|
|
1015
|
+
matchedCompetencies.push(fracTheme);
|
|
1016
|
+
break; // Found a match, move to next course competency
|
|
1017
|
+
}
|
|
1018
|
+
}
|
|
1019
|
+
}
|
|
1020
|
+
}
|
|
1021
|
+
});
|
|
1022
|
+
});
|
|
1023
|
+
// Remove duplicates from matched competencies
|
|
1024
|
+
matchedCompetencies = [...new Set(matchedCompetencies)];
|
|
1025
|
+
console.log(`\n=== ${categoryType.toUpperCase()} COMPETENCY MATCHING ===`);
|
|
1026
|
+
console.log(`${categoryType} - All FRAC competencies:`, allCategoryCompetencies);
|
|
1027
|
+
console.log(`${categoryType} - Matched from courses:`, matchedCompetencies);
|
|
1028
|
+
console.log(`${categoryType} - Match rate: ${matchedCompetencies.length}/${allCategoryCompetencies.length} (${Math.round((matchedCompetencies.length / allCategoryCompetencies.length) * 100)}%)`);
|
|
1029
|
+
// Return unmatched competencies (those in role mapping but not covered by courses)
|
|
1030
|
+
return this.compareStringArrays(allCategoryCompetencies, matchedCompetencies);
|
|
1031
|
+
}
|
|
1032
|
+
compareStringArrays(arr1, arr2) {
|
|
1033
|
+
console.log('All competencies in category:', arr1);
|
|
1034
|
+
console.log('Matched competencies from courses:', arr2);
|
|
1035
|
+
// Return items in arr1 that are NOT in arr2 (unmatched competencies)
|
|
1036
|
+
const unmatched = arr1.filter(item => !arr2.includes(item));
|
|
1037
|
+
console.log('Unmatched competencies:', unmatched);
|
|
1038
|
+
return unmatched;
|
|
1039
|
+
}
|
|
1040
|
+
addCourse(missingCompetency, competencyType) {
|
|
1041
|
+
let dialogData = { ...this.planData };
|
|
1042
|
+
// If competency parameters are provided, add them to the dialog data
|
|
1043
|
+
if (missingCompetency && competencyType) {
|
|
1044
|
+
// Parse the theme from the missing competency string (assumes format "theme-subtheme" or just "theme")
|
|
1045
|
+
const theme = missingCompetency.trim();
|
|
1046
|
+
dialogData.prefillCompetency = {
|
|
1047
|
+
type: competencyType,
|
|
1048
|
+
theme: theme
|
|
1049
|
+
};
|
|
1050
|
+
}
|
|
1051
|
+
const dialogRef = this.dialog.open(AddCourseComponent, {
|
|
1052
|
+
width: '1000px',
|
|
1053
|
+
data: dialogData,
|
|
1054
|
+
panelClass: 'view-cbp-plan-popup',
|
|
1055
|
+
minHeight: '400px',
|
|
1056
|
+
maxHeight: '90vh',
|
|
1057
|
+
disableClose: true // Optional: prevent closing with outside click
|
|
1058
|
+
});
|
|
1059
|
+
dialogRef.afterClosed().subscribe(result => {
|
|
1060
|
+
if (result === 'saved') {
|
|
1061
|
+
console.log('Course added successfully! Refreshing data...');
|
|
1062
|
+
// Refresh user courses to include the newly added course
|
|
1063
|
+
this.getUserCourse();
|
|
1064
|
+
// Refresh gap analysis with the latest data
|
|
1065
|
+
setTimeout(() => {
|
|
1066
|
+
this.updateGapAnalysisAfterCoursesUpdate();
|
|
1067
|
+
console.log('Gap analysis refreshed after course addition');
|
|
1068
|
+
}, 500); // Small delay to ensure data is loaded
|
|
1069
|
+
}
|
|
1070
|
+
});
|
|
1071
|
+
}
|
|
1072
|
+
filterOnCompetencyTheme(themeItem) {
|
|
1073
|
+
this.selectedThemeFilter = themeItem;
|
|
1074
|
+
let themeName = themeItem?.split("-")[0]?.trim();
|
|
1075
|
+
let subThemeName = themeItem?.split("-")[1]?.trim();
|
|
1076
|
+
console.log('Filtering by theme:', themeName, 'subTheme:', subThemeName, 'Tab index:', this.innerTabActiveIndex);
|
|
1077
|
+
// Get the current competency type based on the active tab
|
|
1078
|
+
let competencyType = '';
|
|
1079
|
+
if (this.innerTabActiveIndex === 1) {
|
|
1080
|
+
competencyType = 'behavioral';
|
|
1081
|
+
}
|
|
1082
|
+
else if (this.innerTabActiveIndex === 2) {
|
|
1083
|
+
competencyType = 'functional';
|
|
1084
|
+
}
|
|
1085
|
+
else if (this.innerTabActiveIndex === 3) {
|
|
1086
|
+
competencyType = 'domain';
|
|
1087
|
+
}
|
|
1088
|
+
// Get all available courses using the helper method
|
|
1089
|
+
const allAvailableCourses = this.getAllAvailableCourses();
|
|
1090
|
+
console.log('Total courses to filter from:', allAvailableCourses.length);
|
|
1091
|
+
this.filterdCourses = allAvailableCourses.filter(course => {
|
|
1092
|
+
if (!course)
|
|
1093
|
+
return false;
|
|
1094
|
+
// Handle different competency property names
|
|
1095
|
+
let competencies = [];
|
|
1096
|
+
if (course.competencies && Array.isArray(course.competencies)) {
|
|
1097
|
+
competencies = course.competencies;
|
|
1098
|
+
}
|
|
1099
|
+
else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {
|
|
1100
|
+
competencies = course.competencies_v6;
|
|
1101
|
+
}
|
|
1102
|
+
if (competencies.length === 0)
|
|
1103
|
+
return false;
|
|
1104
|
+
// Filter by competency type and theme/subtheme
|
|
1105
|
+
return competencies.some(comp => {
|
|
1106
|
+
if (!comp)
|
|
1107
|
+
return false;
|
|
1108
|
+
// Check competency type matches current tab
|
|
1109
|
+
let compType = comp?.competencyAreaName?.toLowerCase();
|
|
1110
|
+
if (compType === 'behavioural')
|
|
1111
|
+
compType = 'behavioral'; // Normalize spelling
|
|
1112
|
+
if (compType !== competencyType)
|
|
1113
|
+
return false;
|
|
1114
|
+
// Check cross-matching like gap analysis: theme vs subtheme and subtheme vs theme
|
|
1115
|
+
const compTheme = comp?.competencyThemeName?.toLowerCase()?.trim();
|
|
1116
|
+
const compSubTheme = comp?.competencySubThemeName?.toLowerCase()?.trim();
|
|
1117
|
+
const searchTheme = themeName?.toLowerCase();
|
|
1118
|
+
const searchSubTheme = subThemeName?.toLowerCase();
|
|
1119
|
+
// Enhanced cross-matching logic like gap analysis:
|
|
1120
|
+
// 1. Selected Theme vs Course Theme
|
|
1121
|
+
// 2. Selected Sub Theme vs Course Sub Theme
|
|
1122
|
+
// 3. Selected Theme vs Course Sub Theme
|
|
1123
|
+
// 4. Selected Sub Theme vs Course Theme
|
|
1124
|
+
const themeToThemeMatch = searchTheme && compTheme === searchTheme;
|
|
1125
|
+
const subThemeToSubThemeMatch = searchSubTheme && compSubTheme === searchSubTheme;
|
|
1126
|
+
const themeToSubThemeMatch = searchTheme && compSubTheme === searchTheme;
|
|
1127
|
+
const subThemeToThemeMatch = searchSubTheme && compTheme === searchSubTheme;
|
|
1128
|
+
const isMatch = themeToThemeMatch || subThemeToSubThemeMatch || themeToSubThemeMatch || subThemeToThemeMatch;
|
|
1129
|
+
// Log matches for debugging (similar to gap analysis)
|
|
1130
|
+
if (isMatch) {
|
|
1131
|
+
let matchType = '';
|
|
1132
|
+
if (themeToThemeMatch)
|
|
1133
|
+
matchType = 'Theme-to-Theme';
|
|
1134
|
+
else if (subThemeToSubThemeMatch)
|
|
1135
|
+
matchType = 'SubTheme-to-SubTheme';
|
|
1136
|
+
else if (themeToSubThemeMatch)
|
|
1137
|
+
matchType = 'Theme-to-SubTheme';
|
|
1138
|
+
else if (subThemeToThemeMatch)
|
|
1139
|
+
matchType = 'SubTheme-to-Theme';
|
|
1140
|
+
console.log(`✅ FILTER MATCH FOUND [${matchType}]:`, {
|
|
1141
|
+
selected: { theme: searchTheme, subTheme: searchSubTheme },
|
|
1142
|
+
course: { type: compType, theme: compTheme, subTheme: compSubTheme },
|
|
1143
|
+
matchType: matchType
|
|
1144
|
+
});
|
|
1145
|
+
}
|
|
1146
|
+
// Match if any of the cross-matching conditions are met
|
|
1147
|
+
return isMatch;
|
|
1148
|
+
});
|
|
1149
|
+
});
|
|
1150
|
+
console.log('Filtered courses for theme:', themeName, 'Count:', this.filterdCourses.length);
|
|
1151
|
+
console.log('Filtered courses:', this.filterdCourses.map(c => c.name || c.course));
|
|
1152
|
+
}
|
|
1153
|
+
clearThemeFilter() {
|
|
1154
|
+
this.selectedThemeFilter = '';
|
|
1155
|
+
// Reset to category filtered courses by calling the tab change method
|
|
1156
|
+
this.onInnerTabChange({
|
|
1157
|
+
index: this.innerTabActiveIndex,
|
|
1158
|
+
tab: { textLabel: this.innerTabActiveText }
|
|
1159
|
+
});
|
|
1160
|
+
console.log('Theme filter cleared, showing all courses for category:', this.innerTabActiveText);
|
|
1161
|
+
}
|
|
1162
|
+
isThemeSelected(theme) {
|
|
1163
|
+
return this.selectedThemeFilter === theme;
|
|
1164
|
+
}
|
|
1165
|
+
/**
|
|
1166
|
+
* Add course for the currently selected theme filter with prefilled competency data
|
|
1167
|
+
*/
|
|
1168
|
+
addCourseForSelectedTheme() {
|
|
1169
|
+
if (!this.selectedThemeFilter) {
|
|
1170
|
+
console.warn('No theme selected for adding course');
|
|
1171
|
+
return;
|
|
1172
|
+
}
|
|
1173
|
+
// Parse theme and sub-theme from selectedThemeFilter (format: "Theme - SubTheme")
|
|
1174
|
+
const themeParts = this.selectedThemeFilter.split('-');
|
|
1175
|
+
const theme = themeParts[0]?.trim();
|
|
1176
|
+
const subTheme = themeParts[1]?.trim();
|
|
1177
|
+
// Get the current competency type based on the active tab
|
|
1178
|
+
// Map the tab labels to the format expected by AddCourseComponent
|
|
1179
|
+
let competencyType = '';
|
|
1180
|
+
if (this.innerTabActiveIndex === 1) {
|
|
1181
|
+
competencyType = 'Behavioral'; // Tab label is "behavioural"
|
|
1182
|
+
}
|
|
1183
|
+
else if (this.innerTabActiveIndex === 2) {
|
|
1184
|
+
competencyType = 'Functional'; // Tab label is "functional"
|
|
1185
|
+
}
|
|
1186
|
+
else if (this.innerTabActiveIndex === 3) {
|
|
1187
|
+
competencyType = 'Domain'; // Tab label is "domain"
|
|
1188
|
+
}
|
|
1189
|
+
console.log('Tab mapping - innerTabActiveIndex:', this.innerTabActiveIndex, 'innerTabActiveText:', this.innerTabActiveText, 'mapped competencyType:', competencyType);
|
|
1190
|
+
console.log('Adding course for missing competency:', {
|
|
1191
|
+
competencyType,
|
|
1192
|
+
theme,
|
|
1193
|
+
subTheme,
|
|
1194
|
+
selectedThemeFilter: this.selectedThemeFilter
|
|
1195
|
+
});
|
|
1196
|
+
// Create dialog data with prefilled competency information
|
|
1197
|
+
let dialogData = { ...this.planData };
|
|
1198
|
+
dialogData.prefillCompetency = {
|
|
1199
|
+
type: competencyType,
|
|
1200
|
+
theme: theme,
|
|
1201
|
+
subTheme: subTheme || '' // Include sub-theme if available
|
|
1202
|
+
};
|
|
1203
|
+
const dialogRef = this.dialog.open(AddCourseComponent, {
|
|
1204
|
+
width: '800px',
|
|
1205
|
+
data: dialogData,
|
|
1206
|
+
panelClass: 'view-cbp-plan-popup',
|
|
1207
|
+
minHeight: '400px',
|
|
1208
|
+
maxHeight: '90vh',
|
|
1209
|
+
disableClose: true
|
|
1210
|
+
});
|
|
1211
|
+
dialogRef.afterClosed().subscribe(result => {
|
|
1212
|
+
if (result === 'saved') {
|
|
1213
|
+
console.log('Course added successfully for competency! Refreshing data...');
|
|
1214
|
+
// Refresh user courses to include the newly added course
|
|
1215
|
+
this.getUserCourse();
|
|
1216
|
+
// Update gap analysis after course addition
|
|
1217
|
+
this.updateGapAnalysisAfterCoursesUpdate();
|
|
1218
|
+
}
|
|
1219
|
+
});
|
|
1220
|
+
}
|
|
1221
|
+
getDuration(time) {
|
|
1222
|
+
return this.sharedService.convert(time);
|
|
1223
|
+
}
|
|
1224
|
+
getBehaviouralMatched() {
|
|
1225
|
+
this.behaviouralMatched = [];
|
|
1226
|
+
console.log('this.planData.competencies', this.planData);
|
|
1227
|
+
console.log('this.behaviouralNotMatched', this.behaviouralNotMatched);
|
|
1228
|
+
this.planData.competencies.map((item) => {
|
|
1229
|
+
if (item && item?.type === 'Behavioral' && this.behaviouralNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {
|
|
1230
|
+
this.behaviouralMatched.push(item);
|
|
1231
|
+
}
|
|
1232
|
+
});
|
|
1233
|
+
console.log('this.behaviouralMatched--', this.behaviouralMatched);
|
|
1234
|
+
}
|
|
1235
|
+
getFunctionalMatched() {
|
|
1236
|
+
this.functionalMatched = [];
|
|
1237
|
+
this.planData.competencies.map((item) => {
|
|
1238
|
+
if (item && item?.type === 'Functional' && this.functionalNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {
|
|
1239
|
+
this.functionalMatched.push(item);
|
|
1240
|
+
}
|
|
1241
|
+
});
|
|
1242
|
+
return this.functionalMatched;
|
|
1243
|
+
}
|
|
1244
|
+
getDomainMatched() {
|
|
1245
|
+
this.domainMatched = [];
|
|
1246
|
+
this.planData.competencies.map((item) => {
|
|
1247
|
+
if (item && item?.type === 'Domain' && this.domainNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {
|
|
1248
|
+
this.domainMatched.push(item);
|
|
1249
|
+
}
|
|
1250
|
+
});
|
|
1251
|
+
console.log('this.domainMatched--', this.domainMatched);
|
|
1252
|
+
}
|
|
1253
|
+
downloadPDF() {
|
|
1254
|
+
this.isPDFDownload = true;
|
|
1255
|
+
this.loading = true;
|
|
1256
|
+
const element = this.pdfContent.nativeElement;
|
|
1257
|
+
// Wait for images to load
|
|
1258
|
+
const images = element.querySelectorAll('img');
|
|
1259
|
+
const promises = Array.from(images).map((img) => {
|
|
1260
|
+
if (img.complete)
|
|
1261
|
+
return Promise.resolve();
|
|
1262
|
+
return new Promise(resolve => img.onload = resolve);
|
|
1263
|
+
});
|
|
1264
|
+
Promise.all(promises).then(() => {
|
|
1265
|
+
const options = {
|
|
1266
|
+
margin: 0.20,
|
|
1267
|
+
filename: 'Gap Analysis.pdf',
|
|
1268
|
+
image: { type: 'jpeg', quality: 0.98 },
|
|
1269
|
+
html2canvas: {
|
|
1270
|
+
scale: 2,
|
|
1271
|
+
useCORS: true,
|
|
1272
|
+
scrollY: 0,
|
|
1273
|
+
},
|
|
1274
|
+
jsPDF: {
|
|
1275
|
+
unit: 'in',
|
|
1276
|
+
format: 'a4',
|
|
1277
|
+
orientation: 'portrait'
|
|
1278
|
+
},
|
|
1279
|
+
pagebreak: {
|
|
1280
|
+
mode: ['css', 'legacy'],
|
|
1281
|
+
before: '.page-break',
|
|
1282
|
+
avoid: ['.no-break']
|
|
1283
|
+
}
|
|
1284
|
+
};
|
|
1285
|
+
html2pdf().from(element).set(options).save();
|
|
1286
|
+
setTimeout(() => {
|
|
1287
|
+
this.isPDFDownload = false;
|
|
1288
|
+
this.loading = false;
|
|
1289
|
+
}, 3000);
|
|
1290
|
+
});
|
|
1291
|
+
}
|
|
1292
|
+
redirectToToc(item) {
|
|
1293
|
+
if (item?.public_link) {
|
|
1294
|
+
window.open(item?.public_link, '_blank');
|
|
1295
|
+
}
|
|
1296
|
+
else {
|
|
1297
|
+
let url = `https://portal.igotkarmayogi.gov.in/app/toc/${item?.identifier}/overview?`;
|
|
1298
|
+
window.open(url, '_blank');
|
|
1299
|
+
}
|
|
1300
|
+
}
|
|
1301
|
+
/**
|
|
1302
|
+
* Initialize gap analysis stats after initial data load
|
|
1303
|
+
*/
|
|
1304
|
+
initializeGapAnalysisStats() {
|
|
1305
|
+
// Set a small timeout to ensure all async operations complete
|
|
1306
|
+
setTimeout(() => {
|
|
1307
|
+
if (this.filterdCourses && this.filterdCourses.length > 0) {
|
|
1308
|
+
this.gapAnalysisStats();
|
|
1309
|
+
}
|
|
1310
|
+
}, 100);
|
|
1311
|
+
}
|
|
1312
|
+
/**
|
|
1313
|
+
* Update gap analysis stats after courses are updated (like suggested courses added)
|
|
1314
|
+
*/
|
|
1315
|
+
updateGapAnalysisAfterCoursesUpdate() {
|
|
1316
|
+
// Don't modify originalData - it should remain as AI recommended courses only
|
|
1317
|
+
// The gap analysis already properly includes all course types
|
|
1318
|
+
// Recalculate gap analysis stats
|
|
1319
|
+
this.gapAnalysisStats();
|
|
1320
|
+
}
|
|
1321
|
+
/**
|
|
1322
|
+
* Regenerate course recommendations by deleting existing recommendations and generating new ones
|
|
1323
|
+
*/
|
|
1324
|
+
async regenerateCourseRecommendations() {
|
|
1325
|
+
if (this.isRegenerating) {
|
|
1326
|
+
return; // Prevent multiple simultaneous regenerations
|
|
1327
|
+
}
|
|
1328
|
+
try {
|
|
1329
|
+
this.isRegenerating = true;
|
|
1330
|
+
// Show confirmation dialog
|
|
1331
|
+
const confirmed = await this.showConfirmationDialog();
|
|
1332
|
+
if (!confirmed) {
|
|
1333
|
+
this.isRegenerating = false;
|
|
1334
|
+
return;
|
|
1335
|
+
}
|
|
1336
|
+
console.log('Starting course recommendation regeneration...');
|
|
1337
|
+
// Set regeneration progress state
|
|
1338
|
+
this.isRegeneratingWithProgress = true;
|
|
1339
|
+
this.dataLoaded = false; // Hide main content during regeneration
|
|
1340
|
+
// Step 1: Delete existing course recommendations
|
|
1341
|
+
await this.deleteCourseRecommendations();
|
|
1342
|
+
// Step 2: Generate new course recommendations
|
|
1343
|
+
await this.generateNewCourseRecommendations();
|
|
1344
|
+
// Step 3: Refresh the current data
|
|
1345
|
+
this.refreshComponent();
|
|
1346
|
+
console.log('Course recommendation regeneration completed successfully');
|
|
1347
|
+
// Show success message
|
|
1348
|
+
this.snackBar.open('Course recommendations regenerated successfully!', 'Close', {
|
|
1349
|
+
duration: 5000,
|
|
1350
|
+
panelClass: ['success-snackbar']
|
|
1351
|
+
});
|
|
1352
|
+
}
|
|
1353
|
+
catch (error) {
|
|
1354
|
+
console.error('Error during course recommendation regeneration:', error);
|
|
1355
|
+
// Show error message
|
|
1356
|
+
this.snackBar.open('Failed to regenerate course recommendations. Please try again.', 'Close', {
|
|
1357
|
+
duration: 5000,
|
|
1358
|
+
panelClass: ['error-snackbar']
|
|
1359
|
+
});
|
|
1360
|
+
}
|
|
1361
|
+
finally {
|
|
1362
|
+
this.isRegenerating = false;
|
|
1363
|
+
this.isRegeneratingWithProgress = false;
|
|
1364
|
+
this.dataLoaded = true; // Ensure main content is shown after regeneration
|
|
1365
|
+
}
|
|
1366
|
+
}
|
|
1367
|
+
/**
|
|
1368
|
+
* Show confirmation dialog for regenerate action
|
|
1369
|
+
*/
|
|
1370
|
+
showConfirmationDialog() {
|
|
1371
|
+
return new Promise((resolve) => {
|
|
1372
|
+
// Create a custom confirmation dialog
|
|
1373
|
+
const dialogRef = this.dialog.open(RegenerateConfirmationDialog, {
|
|
1374
|
+
width: '450px',
|
|
1375
|
+
panelClass: 'regenerate-confirmation-dialog',
|
|
1376
|
+
disableClose: true
|
|
1377
|
+
});
|
|
1378
|
+
dialogRef.afterClosed().subscribe(result => {
|
|
1379
|
+
resolve(result === true);
|
|
1380
|
+
});
|
|
1381
|
+
});
|
|
1382
|
+
}
|
|
1383
|
+
/**
|
|
1384
|
+
* Delete existing course recommendations for the current role mapping
|
|
1385
|
+
*/
|
|
1386
|
+
deleteCourseRecommendations() {
|
|
1387
|
+
return new Promise((resolve, reject) => {
|
|
1388
|
+
const roleMapId = this.planData?.id;
|
|
1389
|
+
if (!roleMapId) {
|
|
1390
|
+
reject(new Error('Role mapping ID not found'));
|
|
1391
|
+
return;
|
|
1392
|
+
}
|
|
1393
|
+
console.log('Deleting course recommendations for role mapping:', roleMapId);
|
|
1394
|
+
this.sharedService.deleteCourseRecommendations(roleMapId).subscribe({
|
|
1395
|
+
next: (response) => {
|
|
1396
|
+
console.log('Course recommendations deleted successfully:', response);
|
|
1397
|
+
resolve(response);
|
|
1398
|
+
},
|
|
1399
|
+
error: (error) => {
|
|
1400
|
+
console.error('Error deleting course recommendations:', error);
|
|
1401
|
+
reject(error);
|
|
1402
|
+
}
|
|
1403
|
+
});
|
|
1404
|
+
});
|
|
1405
|
+
}
|
|
1406
|
+
/**
|
|
1407
|
+
* Generate new course recommendations during regeneration
|
|
1408
|
+
*/
|
|
1409
|
+
generateNewCourseRecommendations() {
|
|
1410
|
+
console.log('Generating new course recommendations...');
|
|
1411
|
+
return this.getRecommendedCourseForRegeneration(this.planData.id);
|
|
1412
|
+
}
|
|
1413
|
+
/**
|
|
1414
|
+
* Enhanced course generation specifically for regeneration with progressive loading
|
|
1415
|
+
*/
|
|
1416
|
+
getRecommendedCourseForRegeneration(role_mapping_id) {
|
|
1417
|
+
return new Promise((resolve, reject) => {
|
|
1418
|
+
// Start the progressive loading simulation for regeneration
|
|
1419
|
+
this.startRegenerativeProgressiveLoading();
|
|
1420
|
+
// Make the actual API call
|
|
1421
|
+
// this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({
|
|
1422
|
+
// next: (response) => {
|
|
1423
|
+
// // Stop progressive loading and show completion
|
|
1424
|
+
// this.isRegeneratingWithProgress = false; // This will stop the interval
|
|
1425
|
+
// this.progressPercentage = 100;
|
|
1426
|
+
// this.currentProcessingStage = 'Course recommendations regenerated successfully!';
|
|
1427
|
+
// // Clear the loading state after showing completion
|
|
1428
|
+
// setTimeout(() => {
|
|
1429
|
+
// this.currentProcessingStage = '';
|
|
1430
|
+
// this.progressPercentage = 0;
|
|
1431
|
+
// resolve(response);
|
|
1432
|
+
// }, 2000);
|
|
1433
|
+
// },
|
|
1434
|
+
// error: (error) => {
|
|
1435
|
+
// // Stop progressive loading
|
|
1436
|
+
// this.isRegeneratingWithProgress = false;
|
|
1437
|
+
// this.currentProcessingStage = '';
|
|
1438
|
+
// this.progressPercentage = 0;
|
|
1439
|
+
// // Handle specific error cases
|
|
1440
|
+
// if (error.status === 409 || error.error?.detail?.includes('already exists')) {
|
|
1441
|
+
// // Course recommendations already exist - this is normal
|
|
1442
|
+
// console.log('Course recommendations already exist, loading existing data...');
|
|
1443
|
+
// this.currentProcessingStage = 'Loading existing course recommendations...';
|
|
1444
|
+
// // Try to get existing data
|
|
1445
|
+
// this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({
|
|
1446
|
+
// next: (existingResponse) => {
|
|
1447
|
+
// console.log('Successfully loaded existing course recommendations');
|
|
1448
|
+
// this.progressPercentage = 100;
|
|
1449
|
+
// this.currentProcessingStage = 'Loaded existing course recommendations successfully!';
|
|
1450
|
+
// setTimeout(() => {
|
|
1451
|
+
// this.currentProcessingStage = '';
|
|
1452
|
+
// this.progressPercentage = 0;
|
|
1453
|
+
// resolve(existingResponse);
|
|
1454
|
+
// }, 2000);
|
|
1455
|
+
// },
|
|
1456
|
+
// error: (loadError) => {
|
|
1457
|
+
// console.error('Failed to load existing course recommendations:', loadError);
|
|
1458
|
+
// this.currentProcessingStage = '';
|
|
1459
|
+
// this.progressPercentage = 0;
|
|
1460
|
+
// reject(loadError);
|
|
1461
|
+
// }
|
|
1462
|
+
// });
|
|
1463
|
+
// } else {
|
|
1464
|
+
// console.error('Error during course recommendation regeneration:', error);
|
|
1465
|
+
// reject(error);
|
|
1466
|
+
// }
|
|
1467
|
+
// }
|
|
1468
|
+
// });
|
|
1469
|
+
this.isRegeneratingWithProgress = true;
|
|
1470
|
+
this.progressPercentage = 0;
|
|
1471
|
+
this.currentProcessingStage = 'Generating course recommendations...';
|
|
1472
|
+
interval(5000)
|
|
1473
|
+
.pipe(switchMap(() => this.sharedService.getRecommendedCourse(role_mapping_id).pipe(catchError(error => {
|
|
1474
|
+
// Pass error through observable so polling logic can inspect it
|
|
1475
|
+
return of({ error });
|
|
1476
|
+
}))), tap(res => console.log("Polling result:", res)),
|
|
1477
|
+
/**
|
|
1478
|
+
* ⬇️ STOP POLLING only when:
|
|
1479
|
+
* - SUCCESS → res.status === COMPLETED
|
|
1480
|
+
* - ERROR → conflict / already exists
|
|
1481
|
+
*/
|
|
1482
|
+
takeWhile(res => {
|
|
1483
|
+
if (res?.status === 'COMPLETED')
|
|
1484
|
+
return false; // Stop
|
|
1485
|
+
if (res?.error?.status === 409)
|
|
1486
|
+
return false; // Stop
|
|
1487
|
+
if (res?.error?.error?.detail?.includes('already exists'))
|
|
1488
|
+
return false; // Stop
|
|
1489
|
+
return true; // Keep polling
|
|
1490
|
+
}, true) // include final emission
|
|
1491
|
+
)
|
|
1492
|
+
.subscribe({
|
|
1493
|
+
next: (res) => {
|
|
1494
|
+
/** -------------------------------
|
|
1495
|
+
* SUCCESS CASE (status COMPLETED)
|
|
1496
|
+
-------------------------------- */
|
|
1497
|
+
if (res?.status === 'COMPLETED') {
|
|
1498
|
+
this.isRegeneratingWithProgress = false;
|
|
1499
|
+
this.progressPercentage = 100;
|
|
1500
|
+
this.currentProcessingStage = 'Course recommendations regenerated successfully!';
|
|
1501
|
+
setTimeout(() => {
|
|
1502
|
+
this.currentProcessingStage = '';
|
|
1503
|
+
this.progressPercentage = 0;
|
|
1504
|
+
resolve(res);
|
|
1505
|
+
}, 2000);
|
|
1506
|
+
return;
|
|
1507
|
+
}
|
|
1508
|
+
/** -----------------------------------
|
|
1509
|
+
* ERROR CASE (409 / already exists)
|
|
1510
|
+
------------------------------------ */
|
|
1511
|
+
if (res?.error) {
|
|
1512
|
+
const error = res.error;
|
|
1513
|
+
if (error.status === 409 || error?.error?.detail?.includes('already exists')) {
|
|
1514
|
+
this.currentProcessingStage = 'Loading existing course recommendations...';
|
|
1515
|
+
this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({
|
|
1516
|
+
next: (existingData) => {
|
|
1517
|
+
this.progressPercentage = 100;
|
|
1518
|
+
this.currentProcessingStage = 'Loaded existing course recommendations successfully!';
|
|
1519
|
+
setTimeout(() => {
|
|
1520
|
+
this.currentProcessingStage = '';
|
|
1521
|
+
this.progressPercentage = 0;
|
|
1522
|
+
resolve(existingData);
|
|
1523
|
+
}, 2000);
|
|
1524
|
+
},
|
|
1525
|
+
error: loadError => reject(loadError)
|
|
1526
|
+
});
|
|
1527
|
+
return;
|
|
1528
|
+
}
|
|
1529
|
+
/** Any other error → stop & reject */
|
|
1530
|
+
this.isRegeneratingWithProgress = false;
|
|
1531
|
+
this.currentProcessingStage = '';
|
|
1532
|
+
this.progressPercentage = 0;
|
|
1533
|
+
reject(error);
|
|
1534
|
+
return;
|
|
1535
|
+
}
|
|
1536
|
+
/** ------------------------------
|
|
1537
|
+
* STILL PROCESSING (keep polling)
|
|
1538
|
+
------------------------------- */
|
|
1539
|
+
this.progressPercentage = Math.min(this.progressPercentage + 10, 95);
|
|
1540
|
+
this.currentProcessingStage = 'Still processing course recommendations...';
|
|
1541
|
+
},
|
|
1542
|
+
error: err => reject(err)
|
|
1543
|
+
});
|
|
1544
|
+
});
|
|
1545
|
+
}
|
|
1546
|
+
// Progressive loading methods for regeneration
|
|
1547
|
+
startRegenerativeProgressiveLoading() {
|
|
1548
|
+
this.currentProcessingStage = this.processingStages[0];
|
|
1549
|
+
this.progressPercentage = 0;
|
|
1550
|
+
this.stageStartTime = Date.now();
|
|
1551
|
+
this.simulateRegenerativeProgressiveStages();
|
|
1552
|
+
}
|
|
1553
|
+
simulateRegenerativeProgressiveStages() {
|
|
1554
|
+
let currentStageIndex = 0;
|
|
1555
|
+
const totalDuration = 60000; // 60 seconds estimated duration
|
|
1556
|
+
const stageInterval = totalDuration / this.processingStages.length;
|
|
1557
|
+
const progressInterval = setInterval(() => {
|
|
1558
|
+
// Stop if regeneration is complete
|
|
1559
|
+
if (!this.isRegeneratingWithProgress) {
|
|
1560
|
+
clearInterval(progressInterval);
|
|
1561
|
+
return;
|
|
1562
|
+
}
|
|
1563
|
+
const elapsed = Date.now() - this.stageStartTime;
|
|
1564
|
+
const currentStageProgress = (elapsed % stageInterval) / stageInterval;
|
|
1565
|
+
const stageIndex = Math.min(Math.floor(elapsed / stageInterval), this.processingStages.length - 1);
|
|
1566
|
+
// Update stage if needed
|
|
1567
|
+
if (stageIndex !== currentStageIndex && stageIndex < this.processingStages.length) {
|
|
1568
|
+
currentStageIndex = stageIndex;
|
|
1569
|
+
this.currentProcessingStage = this.processingStages[stageIndex];
|
|
1570
|
+
}
|
|
1571
|
+
// Calculate overall progress (each stage is worth ~16.67%)
|
|
1572
|
+
const baseProgress = (stageIndex / this.processingStages.length) * 100;
|
|
1573
|
+
const stageProgress = (currentStageProgress / this.processingStages.length) * 100;
|
|
1574
|
+
this.progressPercentage = Math.min(Math.round(baseProgress + stageProgress), 95);
|
|
1575
|
+
}, 1000);
|
|
1576
|
+
}
|
|
1577
|
+
/**
|
|
1578
|
+
* Refresh the component data after regeneration
|
|
1579
|
+
*/
|
|
1580
|
+
refreshComponent() {
|
|
1581
|
+
console.log('Refreshing component data...');
|
|
1582
|
+
// Reset all filters and selections
|
|
1583
|
+
this.selectedThemeFilter = '';
|
|
1584
|
+
this.selectFilterCourses = [];
|
|
1585
|
+
this.filterdCourses = [];
|
|
1586
|
+
this.originalFilteredCourses = [];
|
|
1587
|
+
this.suggestedCourses = [];
|
|
1588
|
+
this.userAddedCourses = [];
|
|
1589
|
+
// Reset tab states
|
|
1590
|
+
this.innerTabActiveIndex = 0;
|
|
1591
|
+
this.outerTabActiveIndex = 0;
|
|
1592
|
+
// Call the data loading methods directly to avoid resetting dataLoaded
|
|
1593
|
+
this.loadComponentData();
|
|
1594
|
+
}
|
|
1595
|
+
/**
|
|
1596
|
+
* Load component data (separated from ngOnInit for reuse)
|
|
1597
|
+
*/
|
|
1598
|
+
loadComponentData() {
|
|
1599
|
+
this.getRecommendedCourseWithProgress(this.planData.id).then((res) => {
|
|
1600
|
+
console.log('res', res);
|
|
1601
|
+
this.recommended_course_id = res.id;
|
|
1602
|
+
let allCoures = [];
|
|
1603
|
+
if (res && res.filtered_courses && res.filtered_courses.length) {
|
|
1604
|
+
res?.filtered_courses.forEach((item) => {
|
|
1605
|
+
if (item?.relevancy >= 85) {
|
|
1606
|
+
allCoures.push(item);
|
|
1607
|
+
}
|
|
1608
|
+
});
|
|
1609
|
+
this.originalData = allCoures;
|
|
1610
|
+
this.filterdCourses = allCoures;
|
|
1611
|
+
}
|
|
1612
|
+
console.log('this.filterdCourses', this.filterdCourses);
|
|
1613
|
+
this.getCourses();
|
|
1614
|
+
this.getSuggestedCourse();
|
|
1615
|
+
// Initialize gap analysis stats after courses are loaded
|
|
1616
|
+
this.initializeGapAnalysisStats();
|
|
1617
|
+
this.getUserCourse();
|
|
1618
|
+
// Mark data as loaded after all initial operations
|
|
1619
|
+
this.dataLoaded = true;
|
|
1620
|
+
}).catch((error) => {
|
|
1621
|
+
console.error('Error getting recommended courses:', error);
|
|
1622
|
+
if (error.status === 401) {
|
|
1623
|
+
console.log('Unauthorized access - user will be redirected to login');
|
|
1624
|
+
}
|
|
1625
|
+
else if (error.status !== 409) {
|
|
1626
|
+
// Don't log error for 409 (already exists) as it's handled in the method
|
|
1627
|
+
console.error('Failed to load course recommendations');
|
|
1628
|
+
}
|
|
1629
|
+
// Mark data as loaded even if there's an error to show the dialog content
|
|
1630
|
+
this.dataLoaded = true;
|
|
1631
|
+
});
|
|
1632
|
+
}
|
|
1633
|
+
// Progressive loading methods for course generation
|
|
1634
|
+
startProgressiveLoading() {
|
|
1635
|
+
this.currentProcessingStage = this.processingStages[0];
|
|
1636
|
+
this.progressPercentage = 0;
|
|
1637
|
+
this.stageStartTime = Date.now();
|
|
1638
|
+
this.simulateProgressiveStages();
|
|
1639
|
+
}
|
|
1640
|
+
simulateProgressiveStages() {
|
|
1641
|
+
let currentStageIndex = 0;
|
|
1642
|
+
const totalDuration = 60000; // 60 seconds estimated duration
|
|
1643
|
+
const stageInterval = totalDuration / this.processingStages.length;
|
|
1644
|
+
const progressInterval = setInterval(() => {
|
|
1645
|
+
const elapsed = Date.now() - this.stageStartTime;
|
|
1646
|
+
const currentStageProgress = (elapsed % stageInterval) / stageInterval;
|
|
1647
|
+
const stageIndex = Math.min(Math.floor(elapsed / stageInterval), this.processingStages.length - 1);
|
|
1648
|
+
// Update stage if needed
|
|
1649
|
+
if (stageIndex !== currentStageIndex && stageIndex < this.processingStages.length) {
|
|
1650
|
+
currentStageIndex = stageIndex;
|
|
1651
|
+
this.currentProcessingStage = this.processingStages[stageIndex];
|
|
1652
|
+
}
|
|
1653
|
+
// Calculate overall progress (each stage is worth ~16.67%)
|
|
1654
|
+
const baseProgress = (stageIndex / this.processingStages.length) * 100;
|
|
1655
|
+
const stageProgress = (currentStageProgress / this.processingStages.length) * 100;
|
|
1656
|
+
this.progressPercentage = Math.min(Math.round(baseProgress + stageProgress), 95);
|
|
1657
|
+
// Clear interval when loading stops
|
|
1658
|
+
if (!this.loading) {
|
|
1659
|
+
clearInterval(progressInterval);
|
|
1660
|
+
this.progressPercentage = 100;
|
|
1661
|
+
this.currentProcessingStage = 'Course recommendations generated successfully!';
|
|
1662
|
+
}
|
|
1663
|
+
}, 1000);
|
|
1664
|
+
}
|
|
1665
|
+
// Enhanced course generation with progressive loading
|
|
1666
|
+
getRecommendedCourseWithProgress(role_mapping_id) {
|
|
1667
|
+
return new Promise((resolve, reject) => {
|
|
1668
|
+
this.loading = true;
|
|
1669
|
+
this.currentProcessingStage = 'Generating course recommendations...';
|
|
1670
|
+
this.progressPercentage = 30;
|
|
1671
|
+
this.startProgressiveLoading();
|
|
1672
|
+
interval(5000) // ⏱ poll every 5 seconds
|
|
1673
|
+
.pipe(tap(() => {
|
|
1674
|
+
this.currentProcessingStage = 'Processing...';
|
|
1675
|
+
}), switchMap(() => this.sharedService.getRecommendedCourse(role_mapping_id)),
|
|
1676
|
+
// Continue polling while status !== COMPLETED
|
|
1677
|
+
takeWhile((response) => response?.status !== 'COMPLETED', true), finalize(() => {
|
|
1678
|
+
this.loading = false;
|
|
1679
|
+
}))
|
|
1680
|
+
.subscribe({
|
|
1681
|
+
next: (response) => {
|
|
1682
|
+
if (response?.status === 'COMPLETED') {
|
|
1683
|
+
this.progressPercentage = 100;
|
|
1684
|
+
this.currentProcessingStage = 'Course recommendations generated successfully!';
|
|
1685
|
+
setTimeout(() => {
|
|
1686
|
+
this.currentProcessingStage = '';
|
|
1687
|
+
this.progressPercentage = 0;
|
|
1688
|
+
}, 2000);
|
|
1689
|
+
this.snackBar.open('Recommendations ready!', 'X', {
|
|
1690
|
+
duration: 3000,
|
|
1691
|
+
panelClass: ['snackbar-success']
|
|
1692
|
+
});
|
|
1693
|
+
resolve(response);
|
|
1694
|
+
}
|
|
1695
|
+
},
|
|
1696
|
+
error: (error) => {
|
|
1697
|
+
this.loading = false;
|
|
1698
|
+
this.currentProcessingStage = '';
|
|
1699
|
+
this.progressPercentage = 0;
|
|
1700
|
+
if (error.status === 409 || error.error?.detail?.includes('already exists')) {
|
|
1701
|
+
this.snackBar.open('Course recommendations already exist.', 'X', {
|
|
1702
|
+
duration: 3000,
|
|
1703
|
+
panelClass: ['snackbar-success']
|
|
1704
|
+
});
|
|
1705
|
+
resolve(null); // not a real error
|
|
1706
|
+
}
|
|
1707
|
+
else {
|
|
1708
|
+
this.snackBar.open('Failed to generate course recommendations.', 'X', {
|
|
1709
|
+
duration: 3000,
|
|
1710
|
+
panelClass: ['snackbar-error']
|
|
1711
|
+
});
|
|
1712
|
+
reject(error);
|
|
1713
|
+
}
|
|
1714
|
+
}
|
|
1715
|
+
});
|
|
1716
|
+
});
|
|
1717
|
+
}
|
|
1718
|
+
resetFilters() {
|
|
1719
|
+
this.filterForm.reset({
|
|
1720
|
+
competency: [],
|
|
1721
|
+
rating: [],
|
|
1722
|
+
language: [],
|
|
1723
|
+
duration: [],
|
|
1724
|
+
provider: []
|
|
1725
|
+
});
|
|
1726
|
+
// Reset filtered dropdown lists
|
|
1727
|
+
this.filteredCompetency = [...this.competenciesType];
|
|
1728
|
+
this.filteredRatings = [...this.ratings];
|
|
1729
|
+
this.filteredLanguages = [...this.languages];
|
|
1730
|
+
this.filteredDurations = [...this.durations];
|
|
1731
|
+
this.filteredProviders = [...this.providers];
|
|
1732
|
+
// Reset course list
|
|
1733
|
+
this.filterdCourses = [...this.fullCourseList];
|
|
1734
|
+
// this.fullCourseList.forEach((item)=>{
|
|
1735
|
+
// if(item && item.organisation && item.organisation.length) {
|
|
1736
|
+
// if(this.filteredProviders.indexOf(item.organisation[0]) < 0) {
|
|
1737
|
+
// this.filteredProviders.push(item.organisation[0])
|
|
1738
|
+
// }
|
|
1739
|
+
// }
|
|
1740
|
+
// })
|
|
1741
|
+
this.getAllAvailableCourses();
|
|
1742
|
+
}
|
|
1743
|
+
filterList(value, type) {
|
|
1744
|
+
const search = value.toLowerCase();
|
|
1745
|
+
switch (type) {
|
|
1746
|
+
case 'competency':
|
|
1747
|
+
this.filteredCompetency = this.competenciesType.filter(v => v.toLowerCase().includes(search));
|
|
1748
|
+
break;
|
|
1749
|
+
case 'rating':
|
|
1750
|
+
this.filteredRatings = this.ratings.filter(v => v.toLowerCase().includes(search));
|
|
1751
|
+
break;
|
|
1752
|
+
case 'language':
|
|
1753
|
+
this.filteredLanguages = this.languages.filter(v => v.toLowerCase().includes(search));
|
|
1754
|
+
break;
|
|
1755
|
+
case 'duration':
|
|
1756
|
+
this.filteredDurations = this.durations.filter(v => v.toLowerCase().includes(search));
|
|
1757
|
+
break;
|
|
1758
|
+
case 'provider':
|
|
1759
|
+
this.filteredProviders = this.providers.filter(v => v.toLowerCase().includes(search));
|
|
1760
|
+
break;
|
|
1761
|
+
}
|
|
1762
|
+
}
|
|
1763
|
+
normalizeCompetency(value) {
|
|
1764
|
+
if (!value)
|
|
1765
|
+
return '';
|
|
1766
|
+
const v = value.toLowerCase();
|
|
1767
|
+
if (v === 'all')
|
|
1768
|
+
return 'all';
|
|
1769
|
+
if (v.includes('behavioural') || v.includes('behavioral'))
|
|
1770
|
+
return 'behavioral';
|
|
1771
|
+
if (v.includes('functional'))
|
|
1772
|
+
return 'functional';
|
|
1773
|
+
if (v.includes('domain'))
|
|
1774
|
+
return 'domain';
|
|
1775
|
+
return '';
|
|
1776
|
+
}
|
|
1777
|
+
applyFilters() {
|
|
1778
|
+
const { competency = [], rating = [], language = [], duration = [], provider = [] } = this.filterForm.value;
|
|
1779
|
+
console.log('this.fullCourseList--', this.fullCourseList);
|
|
1780
|
+
this.filterdCourses = this.fullCourseList.filter(course => {
|
|
1781
|
+
/* ---------------- Competency ---------------- */
|
|
1782
|
+
const courseCompetencies = [
|
|
1783
|
+
...(course.competencies || []),
|
|
1784
|
+
...(course.competencies_v6 || [])
|
|
1785
|
+
];
|
|
1786
|
+
const selectedCompetencies = competency
|
|
1787
|
+
.map((c) => this.normalizeCompetency(c))
|
|
1788
|
+
.filter(c => c !== '');
|
|
1789
|
+
const competencyMatch = selectedCompetencies.length === 0 ||
|
|
1790
|
+
selectedCompetencies.includes('all') ||
|
|
1791
|
+
courseCompetencies.some((c) => selectedCompetencies.includes(this.normalizeCompetency(c.competencyAreaName)));
|
|
1792
|
+
/* ---------------- Rating ---------------- */
|
|
1793
|
+
const ratingMatch = rating.length === 0 ||
|
|
1794
|
+
this.matchRating(course, rating);
|
|
1795
|
+
/* ---------------- Language ---------------- */
|
|
1796
|
+
const languageMatch = language.length === 0 ||
|
|
1797
|
+
language.some((l) => course.language === l ||
|
|
1798
|
+
course.language?.includes?.(l) ||
|
|
1799
|
+
course.language?.[0]?.includes?.(l));
|
|
1800
|
+
/* ---------------- Duration ---------------- */
|
|
1801
|
+
const durationMatch = duration.length === 0 ||
|
|
1802
|
+
this.matchDuration(course.duration, duration);
|
|
1803
|
+
/* ---------------- Provider ---------------- */
|
|
1804
|
+
const providerMatch = provider.length === 0 ||
|
|
1805
|
+
provider.some((p) => course.organisation?.includes(p));
|
|
1806
|
+
return (competencyMatch &&
|
|
1807
|
+
ratingMatch &&
|
|
1808
|
+
languageMatch &&
|
|
1809
|
+
durationMatch &&
|
|
1810
|
+
providerMatch);
|
|
1811
|
+
});
|
|
1812
|
+
}
|
|
1813
|
+
matchRating(course, selectedRatings) {
|
|
1814
|
+
if (!selectedRatings || selectedRatings.length === 0) {
|
|
1815
|
+
return true; // no filter selected → match everything
|
|
1816
|
+
}
|
|
1817
|
+
let rating = null;
|
|
1818
|
+
// 1️⃣ Use avgRating if available
|
|
1819
|
+
if (course.avgRating != null && !isNaN(course.avgRating)) {
|
|
1820
|
+
rating = Number(course.avgRating);
|
|
1821
|
+
}
|
|
1822
|
+
// 2️⃣ Map relevancy if avgRating missing
|
|
1823
|
+
else if (course.relevancy != null && !isNaN(course.relevancy)) {
|
|
1824
|
+
const rel = Number(course.relevancy);
|
|
1825
|
+
if (rel >= 90)
|
|
1826
|
+
rating = 4.5;
|
|
1827
|
+
else if (rel >= 80)
|
|
1828
|
+
rating = 4.0;
|
|
1829
|
+
else if (rel >= 70)
|
|
1830
|
+
rating = 3.5;
|
|
1831
|
+
else if (rel >= 50)
|
|
1832
|
+
rating = 2.5;
|
|
1833
|
+
else
|
|
1834
|
+
rating = 1.0;
|
|
1835
|
+
}
|
|
1836
|
+
// 3️⃣ No usable rating → exclude
|
|
1837
|
+
if (rating === null)
|
|
1838
|
+
return false;
|
|
1839
|
+
// 4️⃣ Match selected rating ranges
|
|
1840
|
+
return selectedRatings.some(r => {
|
|
1841
|
+
switch (r) {
|
|
1842
|
+
case '4.5 and above': return rating >= 4.5;
|
|
1843
|
+
case '4.0 and above': return rating >= 4.0;
|
|
1844
|
+
case '3.5 and above': return rating >= 3.5;
|
|
1845
|
+
case '2.5 and above': return rating >= 2.5;
|
|
1846
|
+
case '1 and above': return rating >= 1.0;
|
|
1847
|
+
default: return false;
|
|
1848
|
+
}
|
|
1849
|
+
});
|
|
1850
|
+
}
|
|
1851
|
+
matchDuration(durationInSeconds, selectedRanges) {
|
|
1852
|
+
const hours = Number(durationInSeconds) / 3600;
|
|
1853
|
+
return selectedRanges.some(range => {
|
|
1854
|
+
switch (range) {
|
|
1855
|
+
case '< 1 Hour':
|
|
1856
|
+
return hours < 1;
|
|
1857
|
+
case '1-5 Hours':
|
|
1858
|
+
return hours >= 1 && hours <= 5;
|
|
1859
|
+
case '5+ Hours':
|
|
1860
|
+
return hours > 5;
|
|
1861
|
+
default:
|
|
1862
|
+
return true;
|
|
1863
|
+
}
|
|
1864
|
+
});
|
|
1865
|
+
}
|
|
1866
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: GenerateCourseRecommendationComponent, deps: [{ token: i1.MatDialogRef }, { token: MAT_DIALOG_DATA }, { token: i2.SharedService }, { token: i3.MatSnackBar }, { token: i1.MatDialog }, { token: i4.FormBuilder }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
1867
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: GenerateCourseRecommendationComponent, selector: "app-generate-course-recommendation", viewQueries: [{ propertyName: "pdfContent", first: true, predicate: ["pdfContent"], descendants: true }], ngImport: i0, template: "<div class=\"view-cbp-plan\">\n <div class=\"popup-container\">\n \n <!-- Initial Loading Screen (for initial load and regeneration) -->\n <div class=\"initial-loading-container\" *ngIf=\"!dataLoaded || isRegeneratingWithProgress\">\n <div class=\"loading-content\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <div class=\"loading-text mt-3\">\n <p class=\"main-message\">{{ currentProcessingStage || (isRegeneratingWithProgress ? 'Regenerating course recommendations...' : 'Loading course recommendations...') }}</p>\n <div class=\"progress-details\" *ngIf=\"progressPercentage > 0\">\n <div class=\"progress-indicator\">\n <div class=\"progress-bar-container\">\n <div class=\"progress-bar\" \n [style.width.%]=\"progressPercentage\">\n </div>\n </div>\n <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n </div>\n <small class=\"time-estimate\">\n Real-time course generation in progress. This may take 1-2 minutes.\n </small>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Main Content -->\n <div class=\"main-content\" *ngIf=\"dataLoaded && !isRegeneratingWithProgress\">\n <div class=\"popup-header\">\n <div class=\"header-left\">\n <div class=\"heading\">Save Courses (<span>{{sharedService?.cbpPlanFinalObj?.ministry?.orgName}} <span\n *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span></span>)</div>\n </div>\n <div class=\"header-actions\">\n <button class=\"regenerate-btn\" (click)=\"regenerateCourseRecommendations()\"\n [disabled]=\"isRegenerating\"\n mat-raised-button color=\"primary\">\n <mat-icon *ngIf=\"!isRegenerating\">refresh</mat-icon>\n <mat-spinner *ngIf=\"isRegenerating\" diameter=\"20\"></mat-spinner>\n <span>{{isRegenerating ? 'Regenerating...' : 'Regenerate Recommendations'}}</span>\n </button>\n <div class=\"cursor-pointer close-btn\" (click)=\"closeDialog()\">\n <mat-icon>close</mat-icon>\n </div>\n </div>\n </div>\n <div class=\"section-container mt-4\">\n <mat-tab-group (selectedTabChange)=\"onOuterTabChange($event)\">\n <!-- First Main Tab -->\n <mat-tab label=\"course recommendation\">\n <!-- Nested Tabs -->\n <!-- <mat-tab-group (selectedTabChange)=\"onInnerTabChange($event)\">\n <mat-tab label=\"all\">\n\n </mat-tab>\n <mat-tab label=\"behavioural\">\n\n </mat-tab>\n <mat-tab label=\"functional\">\n\n </mat-tab>\n <mat-tab label=\"domain\">\n\n </mat-tab>\n </mat-tab-group> -->\n </mat-tab>\n\n <!-- Second Main Tab -->\n <mat-tab label=\"gap analysis\">\n <!-- <p>Gap analysis content goes here</p> -->\n </mat-tab>\n </mat-tab-group>\n <form [formGroup]=\"filterForm\" class=\"filters-container\">\n <div class=\"role-mapping-container\"\n *ngIf=\"outerTabActiveIndex === 0 && (innerTabActiveIndex === 0 || innerTabActiveIndex === 1 || innerTabActiveIndex === 2 || innerTabActiveIndex === 3 )\">\n <div class=\"container\">\n <div class=\"section-header-improved mt-3\">\n <div class=\"search-and-buttons-container\">\n <div>\n <label>Search Courses</label>\n </div> \n <div class=\"search-container-compact\">\n <div class=\"rsearch\">\n <mat-icon class=\"search-icon\">search</mat-icon>\n <input class=\"sinput-compact\" [(ngModel)]=\"searchText\"\n (input)=\"applyFilter($event.target.value);\"\n placeholder=\"Search Courses\" type=\"text\">\n </div>\n </div>\n \n </div>\n \n\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Competency</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.competency?.length ? '' : 'Competency' }}\n </mat-label>\n \n <mat-select\n formControlName=\"competency\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Competency\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'competency')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredCompetency\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div> \n \n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Rating</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.rating?.length ? '' : 'Rating' }}\n </mat-label>\n \n <mat-select\n formControlName=\"rating\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Reatings\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'rating')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredRatings\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n \n \n \n \n </div>\n <div class=\"section-header-improved-sec-2 mt-4\">\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Language</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.language?.length ? '' : 'Language' }}\n </mat-label>\n \n <mat-select\n formControlName=\"language\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Languages\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'language')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredLanguages\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Duration</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.duration?.length ? '' : 'Duration' }}\n </mat-label>\n \n <mat-select\n formControlName=\"duration\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Duration\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'duration')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredDurations\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Provider</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.provider?.length ? '' : 'Provider' }}\n </mat-label>\n \n <mat-select\n formControlName=\"provider\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Provider\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'provider')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredProviders\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div style=\"margin-top: 45px;margin-left: 20px;\">\n <input class=\"btn-active\" type=\"button\" value=\"Reset All Filters\"\n (click)=\"resetFilters()\" />\n </div>\n </div>\n <div class=\"mt-4\">\n <div style=\"font-size:16px;font-weight:bold;color:#000\">\n Filtered Course Count : {{filterdCourses?.length}}\n </div>\n </div>\n <div class=\"mt-5 sub-heading select-all-container\" *ngIf=\"filterdCourses?.length\">\n <div style=\"margin-left: 16px;\">\n <mat-checkbox [checked]=\"false\" (change)=\"selectAllCourses($event)\">Select All\n Courses</mat-checkbox>\n </div>\n <div class=\"btn-group-compact\">\n <div>\n <input class=\"btn-active\" type=\"button\" value=\"Suggest More Course From iGOT\"\n (click)=\"suggestMoreCourses()\" />\n </div>\n <div class=\"pl-2\">\n <input [disabled]=\"selectFilterCourses?.length > 0 ? false : true\"\n [ngClass]=\"selectFilterCourses?.length > 0 ? 'btn-active' : 'btn-disable'\"\n type=\"button\" [value]=\"mode === 'add' ? 'Save Courses' : 'Update Courses'\"\n (click)=\"saveCourses()\" />\n </div>\n </div>\n </div>\n <div class=\"disclaimer-container mt-2\" *ngIf=\"filterdCourses?.length\">\n <div class=\"disclaimer-message mt-4\">\n <strong>Disclaimer:</strong> Please verify the public course URL before adding it to the plan.\n </div>\n </div>\n <div class=\"mt-2\">\n\n <div class=\"course-list-container\">\n <div\n [ngClass]=\"innerTabActiveIndex > 0 ? 'course-list-container-flex':'course-list-container-right-flex' \">\n <div *ngIf=\"innerTabActiveIndex > 0\"\n [ngClass]=\"innerTabActiveIndex > 0 ? 'left-panel':''\">\n <ng-container *ngIf=\"innerTabActiveIndex === 1\">\n <div class=\"sub-heading\">\n <p>Competencies in Behavioural</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n <ng-container *ngIf=\"innerTabActiveIndex === 2\">\n <div class=\"sub-heading\">\n <p>Competencies in Functional</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n <ng-container *ngIf=\"innerTabActiveIndex === 3\">\n <div class=\"sub-heading\">\n <p>Competencies in Domain</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n </div>\n <div [ngClass]=\"innerTabActiveIndex > 0 ? 'right-panel':'right-panel-flex'\">\n <!-- No Course Found Message in Right Panel -->\n <div class=\"no-course-found-container\"\n *ngIf=\"filterdCourses?.length === 0 && selectedThemeFilter\">\n <div class=\"no-course-card\">\n <div class=\"no-course-icon\">\n <mat-icon>search_off</mat-icon>\n </div>\n <div class=\"no-course-content\">\n <h4>No Courses Found</h4>\n <p>No courses found for the selected theme:\n <strong>{{selectedThemeFilter}}</strong></p>\n <p>Would you like to add a course for this competency?</p>\n <button class=\"suggest-course-btn\"\n (click)=\"addCourseForSelectedTheme()\">\n <mat-icon>add_circle</mat-icon>\n Add Course\n </button>\n </div>\n </div>\n </div>\n\n <!-- Courses List -->\n <div class=\"course-list-item\" *ngFor=\"let item of filterdCourses;let i=index\">\n\n <div class=\"course-content\">\n <div class=\"course-header\">\n <div class=\"checked-course-container\">\n <div>\n <mat-checkbox [checked]=\"checkIfCourseExists(item)\"\n (change)=\"selectedFilterCourses($event, item)\"></mat-checkbox>\n </div>\n <div class=\"ml-2\">\n <div class=\"course-pill-container\">\n <div class=\"course-pill\">\n <div class=\"course-pill-text\">\n <span><img\n src=\"assets/icons/course.svg\"></span> <span>Course</span>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n\n <!-- <div>\n <div class=\"ai-recommened-pill\">\n <div class=\"ai-recommened-pill-text\">\n <span class=\"ai-loader-icon\"><img src=\"assets/icons/ai-loader.gif\"></span>\n\n <span *ngIf=\"item?.rationale\">AI Recommended\n <span *ngIf=\"item?.is_public\"> - Public</span>\n <span *ngIf=\"!item?.is_public\"> - iGOT</span>\n </span>\n <span *ngIf=\"!item?.rationale\">Manually Suggested - iGOT</span>\n\n </div>\n </div>\n </div> -->\n <div *ngIf=\"item?.rationale && !item?.is_public\">\n <div class=\"ai-recommened-pill-green\">\n <div class=\"ai-recommened-pill-green-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>AI Recommended - iGOT </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n <div *ngIf=\"item?.rationale && item?.is_public\">\n <div class=\"ai-recommened-pill-public\">\n <div class=\"ai-recommened-pill-public-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>AI Recommended - Public </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n <div *ngIf=\"!item?.rationale\">\n <div class=\"ai-recommened-pill-gray\">\n <div class=\"ai-recommened-pill-gray-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>Manually Suggested - iGOT </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n\n </div>\n <div class=\"course-title mt-3 cursor-pointer\" (click)=\"redirectToToc(item)\">\n <span *ngIf=\"item?.course\">{{item?.course}}</span>\n <!-- <span *ngIf=\"item?.name\">{{item?.name}}****</span> -->\n <span style=\"margin-left:10px;\" (click)=\"redirectToToc(item)\"><img\n height=\"18px\" width=\"18px\"\n src=\"assets/icons/redirect.png\" /></span>\n </div>\n <div class=\"course-desc mt-2\">\n <!-- <span *ngIf=\"item?.rationale\">{{item?.rationale}}</span> -->\n <span *ngIf=\"item?.description\">{{item?.description}}</span>\n </div>\n <div class=\"mt-3\" *ngIf=\"getCompetenciesByType('Behavioural',i)?.length\">\n <div class=\"competency-sub-heading\">Behavioural Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"behavioural-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Behavioural',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Behavioural',i)\"\n class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Behavioural',i)\">\n {{ isExpanded('Behavioural',i) ? 'Show Less' : '+' +\n getRemainingCount('Behavioural',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Functional',i)?.length\">\n <div class=\"competency-sub-heading\">Functional Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"functional-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Functional',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Functional',i)\"\n class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Functional',i)\">\n {{ isExpanded('Functional',i) ? 'Show Less' : '+' +\n getRemainingCount('Functional',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Domain',i)?.length\">\n <div class=\"competency-sub-heading\">Domain Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"domain-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Domain',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Domain',i)\" class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Domain',i)\">\n {{ isExpanded('Domain',i) ? 'Show Less' : '+' +\n getRemainingCount('Domain',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n\n\n </div>\n\n <!-- Relevancy and Duration section -->\n <div class=\"course-footer\" *ngIf=\"item?.relevancy\">\n <div class=\"relevancy-pill-green\">\n <div class=\"relevancy-container\">\n <div class=\"relevancy\">Relevancy</div>\n <div class=\"dash\">-</div>\n <div class=\"percentage\">{{item?.relevancy}}%</div>\n </div>\n </div>\n <div class=\"relevancy-pill-green\">\n <div class=\"relevancy-container\">\n <div class=\"relevancy\">Duration</div>\n <div class=\"dash\">-</div>\n <div class=\"percentage\">\n <span\n *ngIf=\"item?.duration\">{{getDuration(item?.duration)}}</span>\n <span *ngIf=\"!(item?.duration)\">N/A</span>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- General No Data Message (only when no theme filter is active) -->\n <div class=\"mt-2 sub-heading\" *ngIf=\"filterdCourses?.length === 0 && !selectedThemeFilter\">\n <p>No Data Found</p>\n </div>\n </div>\n </div>\n </form>\n <div class=\"role-mapping-container\" *ngIf=\"outerTabActiveIndex === 1\">\n <div id=\"gap-analysis-content\">\n\n <div class=\"download-pdf mt-4 mb-2\">\n <div>\n <input class=\"btn-active\" type=\"button\" value=\"Download\" (click)=\"downloadPDF()\" />\n </div>\n </div>\n <div>\n <!-- <div *ngIf=\"isPDFDownload\">\n <div class=\"pdf-heading\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}} <span\n *ngIf=\"planData?.designation_name \">_{{planData?.designation_name}}_Gap_Analysis</span>\n </div>\n </div> -->\n <div class=\"d-flex vh-100\" [ngClass]=\"isPDFDownload ? 'mt-4':''\">\n <!-- Sidebar -->\n <div class=\"d-flex flex-column p-3 bg-light border-end\" style=\"width: 250px;\">\n <h5 class=\"mb-4\">Analyze by Category</h5>\n <button *ngFor=\"let item of menuItems\"\n class=\"btn btn-outline-primary mb-2 text-start menu-list-item\"\n [class.active]=\"selectedCategory === item.key\" (click)=\"selectCategory(item.key)\">\n {{ item.label }}\n </button>\n </div>\n\n <!-- Main Content -->\n <div class=\"flex-grow-1 p-4\">\n <div class=\"d-flex justify-content-between gap-3 \">\n <div class=\"p-3 gray-bg\">\n <div class=\"div-label sub-heading\">\n <p>Competencies Covered</p>\n </div>\n <div class=\"div-count\">{{competencyCoveredCount}}</div>\n </div>\n <div class=\"p-3 gray-bg\">\n <div class=\"div-label sub-heading\">\n <p *ngIf=\"selectedCategory === 'all'\">Total Competencies</p>\n <p *ngIf=\"selectedCategory === 'behavioral'\">Total Behavioural Competencies\n </p>\n <p *ngIf=\"selectedCategory === 'functional'\">Total Functional Competencies\n </p>\n <p *ngIf=\"selectedCategory === 'domain'\">Total Domain Competencies</p>\n </div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'all'\">\n {{this.planData?.competencies?.length}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n {{behavioralTotalCompetencies}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'functional'\">\n {{functionalTotalCompetencies}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'domain'\">\n {{domainTotalCompetencies}}</div>\n </div>\n <div class=\"p-3 green-bg\">\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'all'\">Overall Coverage\n </div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'behavioral'\">\n Behavioural Coverage</div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'functional'\">\n Functional Coverage</div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'domain'\">Domain\n Coverage</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'all'\">\n {{overallCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n {{behavioralCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'functional'\">\n {{functionalCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'domain'\">\n {{domainCoverage}}</div>\n </div>\n </div>\n <div class=\"mt-4 all-category-list\" [ngSwitch]=\"selectedCategory\">\n <div *ngSwitchCase=\"'all'\">\n <h4>All Categories</h4>\n\n <!-- Show Behavioural section only if there are unmatched behavioural competencies -->\n <div class=\"mt-4\"\n *ngIf=\"behaviouralNotMatched && behaviouralNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Behavioural</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show Functional section only if there are unmatched functional competencies -->\n <div class=\"mt-4\"\n *ngIf=\"functionalNotMatched && functionalNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Functional</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show Domain section only if there are unmatched domain competencies -->\n <div class=\"mt-4\" *ngIf=\"domainNotMatched && domainNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Domain</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show message when all competencies are covered -->\n <div class=\"mt-4 text-center\"\n *ngIf=\"(!behaviouralNotMatched || behaviouralNotMatched.length === 0) && \n (!functionalNotMatched || functionalNotMatched.length === 0) && \n (!domainNotMatched || domainNotMatched.length === 0)\">\n <div class=\"no-gaps-message\">\n <mat-icon class=\"success-icon\">check_circle</mat-icon>\n <h4>Excellent! All competencies are covered</h4>\n <p>All required competencies have been addressed by the selected\n courses.</p>\n </div>\n </div>\n\n </div>\n\n <div *ngSwitchCase=\"'behavioral'\">\n <h4>Behavioural</h4>\n <div class=\"mt-4\" *ngIf=\"behavioralCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Behavioural</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"behavioralCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All behavioral competencies are covered by the\n selected courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchCase=\"'functional'\">\n <h4>Functional</h4>\n <div class=\"mt-4\" *ngIf=\"functionalCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Functional</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course</span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"functionalCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All functional competencies are covered by the\n selected courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchCase=\"'domain'\">\n <h4>Domain</h4>\n <div class=\"mt-4\" *ngIf=\"domainCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Domain</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"domainCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All domain competencies are covered by the selected\n courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchDefault>\n <p>Please select a category.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n </div>\n </div> <!-- Close main-content div -->\n\n </div>\n</div>\n\n\n<div [hidden]=\"!isPDFDownload\" class=\"gap-analysis-pdf-container\" #pdfContent>\n <div class=\"gap-analysis-pdf-header\">\n <img src=\"assets/icons/karmayogiLogo-min.png\" alt=\"Karmayogi Bharat Logo\" class=\"logo\" />\n <h1><span>\n <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'state'\">{{sharedService?.cbpPlanFinalObj?.department_name?.name}}</span> \n <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'ministry'\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}}</span> \n <span *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span>\n </span></h1>\n <p class=\"subtitle\">A detailed breakdown of competencies Gap Analysis</p>\n \n </div>\n \n <div class=\"competency-snapshot\">\n <h2>Competency Snapshot</h2>\n <div class=\"snapshot-content\">\n <div class=\"circle-wrapper\">\n <svg class=\"progress-ring\" width=\"120\" height=\"120\">\n <circle class=\"progress-ring-bg\" cx=\"60\" cy=\"60\" r=\"52\" />\n <circle class=\"progress-ring-circle\" cx=\"60\" cy=\"60\" r=\"52\" />\n </svg>\n <div class=\"percentage-text\">{{overallCoverage}}<br /><span>Overall Coverage</span></div>\n </div>\n \n <div class=\"competency-container\">\n <!-- Total Competencies Box -->\n <div class=\"total-box\">\n <div class=\"title\">Total Competencies</div>\n <div class=\"count\">{{this.planData?.competencies?.length}}</div>\n </div>\n \n <!-- Progress Bars -->\n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text behavioral\">Behavioural</span>\n <span class=\"covered-text\">{{behavioralCompetencyCoveredCount}} / {{behavioralTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill behavioral\" [style.width.%]=\"(behavioralCompetencyCoveredCount / behavioralTotalCompetencies) * 100\"></div>\n </div>\n </div>\n \n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text functional\">Functional</span>\n <span class=\"covered-text\">{{functionalCompetencyCoveredCount}} / {{functionalTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill functional\" [style.width.%]=\"(functionalCompetencyCoveredCount / functionalTotalCompetencies) * 100\"></div>\n </div>\n </div>\n \n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text domain\">Domain</span>\n <span class=\"covered-text\">{{domainCompetencyCoveredCount}} / {{domainTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill domain\" [style.width.%]=\"(domainCompetencyCoveredCount / domainTotalCompetencies) * 100\"></div>\n </div>\n </div>\n </div>\n </div>\n </div>\n \n <div class=\"gap-section mb-4\">\n <h2>Detailed Competency Analysis</h2>\n <div class=\"gap-box\">\n Bridge Your Competencies Gap\n <p class=\"subtitle\">Here are {{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}} key areas for development.</p>\n </div>\n </div>\n <div class=\"sub-heading mb-4\">\n <p>Gap to Address <span *ngIf=\"(this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)) > 0\">({{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}})</span></p>\n \n </div>\n <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n \u2022 \n <span class=\"tag behavioral\">Behavioural</span>\n </div>\n </div>\n \n <div class=\"competency-card functional\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n \u2022 \n <span class=\"tag functional\">Functional</span>\n </div>\n </div>\n \n <div class=\"competency-card domain\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n <!-- Project Management -->\n \u2022 \n <span class=\"tag domain\">Domain</span>\n </div>\n </div>\n <div class=\"sub-heading mb-4\">\n <p>Covered Competencies <span>({{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}})</span></p> \n </div>\n <div class=\"gap-section mb-4\">\n \n <div class=\"green-bg\">\n <div class=\"div-green-label mb-2\">\n Congratulations!\n </div> \n <p>You have successfully covered </p>\n <p class=\"subtitle\">{{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}} key competencies, forming a strong professional foundation.</p>\n </div>\n </div>\n <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n {{item?.sub_theme}} \u2022 \n <span class=\"tag behavioral\">Behavioural</span>\n </div>\n </div>\n \n <div class=\"competency-card functional\" *ngFor=\"let item of functionalMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n {{item?.sub_theme}} \u2022 \n <span class=\"tag functional\">Functional</span>\n </div>\n </div>\n \n <div class=\"competency-card domain\" *ngFor=\"let item of domainMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n <!-- Project Management -->\n {{item?.sub_theme}} \u2022 \n <span class=\"tag domain\">Domain</span>\n </div>\n </div>\n <div class=\"gap-section mb-4\">\n <h2>Recommended Courses</h2>\n </div>\n <app-gap-analysis-recommended-course [planData]=\"planData\"></app-gap-analysis-recommended-course>\n</div>\n \n\n\n<div class=\"overlay-loader\" *ngIf=\"loading && dataLoaded && !isRegeneratingWithProgress\">\n <div class=\"loading-content\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <div class=\"loading-text mt-3\">\n <p class=\"main-message\">{{ currentProcessingStage || 'Preparing course recommendation generation...' }}</p>\n <div class=\"progress-details\">\n <div class=\"progress-indicator\" *ngIf=\"progressPercentage > 0\">\n <div class=\"progress-bar-container\">\n <div class=\"progress-bar\" \n [style.width.%]=\"progressPercentage\">\n </div>\n </div>\n <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n </div>\n <small class=\"time-estimate\">\n Real-time course generation in progress. This may take 1-2 minutes.\n </small>\n </div>\n </div>\n </div>\n</div>", styles: ["@charset \"UTF-8\";.container{margin:10px auto}.sub-heading p{color:#000;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.radio-label{color:#000;font-family:Lato;font-size:16px;font-style:normal;font-weight:400;line-height:normal}.radio-btn-group{display:flex;width:32px;height:32px;padding:2px;gap:10px}.label{color:var(--Body-Text-Body-Color, #212529);font-family:Lato;font-size:16px;font-style:normal;font-weight:700;line-height:150%}::ng-deep mat-form-field{background-color:#fff!important;border-radius:25px!important;padding-left:12px;padding-right:12px;height:42px;width:100%}::ng-deep mat-select{border-radius:25px!important;background-color:#fff!important}::ng-deep .mat-select-panel{background-color:#fff!important}::ng-deep .mat-select-panel .mat-option{border-radius:0!important}::ng-deep mat-form-field .mat-form-field-underline{display:none!important}::ng-deep mat-select .mat-select-placeholder{color:#0006;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}::ng-deep mat-select .mat-select-value-text{color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;line-height:normal!important;padding-top:0!important;padding-bottom:0!important}::ng-deep mat-select .mat-select-value{display:flex!important;align-items:center!important;height:100%!important}.additional-details textarea{display:flex;padding:16px 16px 0;flex-direction:column;align-items:flex-start;gap:10px;flex:1 0 0;align-self:stretch;width:100%;color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;resize:none}.section-header{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.btn-disable{border-radius:var(--Radius-Full-Round, 9999999px);opacity:.4;background:var(--Primary-KB-Primary-Light, #1B4CA1);color:var(--white-kb-white-100, #FFF);text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:700;line-height:normal;display:flex;height:32px;padding:0 16px;justify-content:center;align-items:center;gap:8px;border:none}.btn-group{display:flex;flex-direction:row;justify-content:space-between}.pl-2{padding-left:5px}.search-filter-section{display:flex;flex-direction:row;justify-content:space-between}.search .rsearch{position:relative}.search .search-icon{position:absolute;top:15px;font-size:20px;left:10px}.search .sinput{border-radius:32px;border:1px solid #d5d0d0;padding:15px 36px;font:400 14px Lato;width:800px}.select-map-route-container{display:flex;flex-direction:row}.selected-mapping-route p{color:#1b4ca1;font-family:Lato;font-size:20px;font-style:normal;font-weight:400;line-height:normal;text-decoration-line:underline;text-decoration-style:solid;-webkit-text-decoration-skip:ink;text-decoration-skip-ink:auto;text-decoration-thickness:auto;text-underline-offset:auto;text-underline-position:from-font}.edit-plan{display:flex;padding:4px;justify-content:center;align-items:center;gap:8px;border-radius:4px;background:#1b4ca129;color:#1b4ca1;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;margin-left:10px}.search-container{display:flex;flex-direction:row}.course-container{display:flex;width:100%;padding:16px;flex-direction:column;align-items:flex-start;gap:24px;grid-row:2/span 1;grid-column:1/span 1;border-radius:12px;border:1px solid var(--borders-black-border-16-black, rgba(0, 0, 0, .16));background:#fff}.course-header{display:flex;flex-direction:row;justify-content:space-between;align-items:flex-start;width:100%;gap:8px}.course-pill{display:flex;height:24px;padding:8px;align-items:center;gap:4px;border-radius:16px;border:1px solid var(--borders-border-fill-focussed-4-light-100, #EF951E);background:#fefaf4}.ai-recommened-pill-green{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#0080001c}.ai-recommened-pill-green-text{color:#006400;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-public{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#1b4ca129}.ai-recommened-pill-public-text{color:#1b4ca1;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-gray{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#d3d3d3}.ai-recommened-pill-gray-text{color:#000;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-orange{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#ffa50052}.ai-recommened-pill-orange-text{color:#5a3f0d;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-orange-text a{color:#5a3f0d!important;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.igot-platform-pill{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);border:1px solid #EF951E;background:#ef951e29}.course-title{color:var(--Primary-KB-Primary-Light, #1B4CA1);font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:22px}.course-desc{color:#0009;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:1.4;display:-webkit-box;-webkit-line-clamp:3;-webkit-box-orient:vertical;overflow:hidden;text-overflow:ellipsis}.relevancy,.competencies-matched{color:#6c757d;font-family:Lato;font-size:12px;font-style:normal;font-weight:500;line-height:normal;align-items:center}.percentage{color:#1b4ca1;font-family:Lato;font-size:14px;font-style:normal;font-weight:600;line-height:normal}.relevancy-container,.competencies-matched-container{display:flex;flex-direction:row;align-items:center}.dash{padding:0 5px}::ng-deep .mat-radio-checked .mat-radio-inner-circle{background:#1b4ca1!important}.course-list-container{display:flex;flex-wrap:wrap;gap:16px;width:100%;padding:1rem}.course-list-item{flex:1 1 calc(50% - 10px);min-width:350px;max-width:500px;box-sizing:border-box;border:1px solid #e0e0e0;padding:16px;background-color:#fff;border-radius:12px;min-height:auto;box-shadow:0 2px 4px #0000001a;transition:box-shadow .2s ease;display:flex;flex-direction:column}.course-list-item:hover{box-shadow:0 4px 8px #00000026}.course-list-item:only-child{flex:1 1 auto;max-width:800px}.checked-course-container{display:flex;width:60%}.ml-2{margin-left:8px}::ng-deep .mat-checkbox-checked .mat-checkbox-background{background:#1b4ca1!important}.igot-platform-pill-text{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:700;line-height:normal}.course-pill-text{color:var(--accessbility-shades-without-opacity-kb-greys-black-87, #212121);text-align:center;font-family:Poppins;font-size:12px;font-style:normal;font-weight:400;line-height:normal}.ai-loader-icon img,.igot-platform-icon img{width:15.714px;height:15.714px;flex-shrink:0}.competency-container{display:flex;justify-content:space-between;flex-direction:row}.outside-layer-total{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #1B4CA1;background:#edf1f8;width:132px}.outside-layer-functional{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #E24577;background:#f8d2de;width:132px}.outside-layer-domain{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #7B47A4;background:#dfd3e9;width:132px}.outside-layer-behavioral{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #F8B861;background:#fde8cc;width:132px}.inside-layer{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:400;line-height:normal}.count{color:#1b4ca1;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.suggest-courses-header{display:flex;flex-direction:row;justify-content:space-between}.suggest-course-control{display:flex;flex-direction:row}.course-card-container{width:172px;border-radius:3.307px;border:.827px solid var(--borders-border-8-black, rgba(0, 0, 0, .08));background-color:#fff}.highlight-ai{font-family:Montserrat;font-size:24px;font-style:normal;font-weight:600;line-height:normal;background:linear-gradient(90deg,#3b27ff,#2de3d6);background-clip:text;-webkit-background-clip:text;color:transparent;-webkit-text-fill-color:transparent;display:inline-block}.ai-loader-icon-medium img{width:40px;height:40px;flex-shrink:0}.heading{color:#000;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal;margin:10px 0}.pdf-heading{color:#000;font-family:Montserrat;font-size:18px;font-style:normal;font-weight:600;line-height:normal;margin:10px 0;text-align:center}.popup-header{display:flex;justify-content:space-between;align-items:center;padding:16px;border-bottom:1px solid #e0e0e0}.header-left{flex:1}.header-actions{display:flex;align-items:center;gap:12px}.regenerate-btn{display:flex;align-items:center;gap:8px;padding:8px 16px;border-radius:6px;font-size:14px;font-weight:500;min-width:auto}.regenerate-btn .mat-icon{margin:0;font-size:18px;width:18px;height:18px}.regenerate-btn .mat-spinner{margin:0}.regenerate-btn:disabled{opacity:.7;cursor:not-allowed}.close-btn{padding:8px;border-radius:4px;transition:background-color .2s ease}.close-btn:hover{background-color:#f5f5f5}.close-btn .mat-icon{font-size:20px;width:20px;height:20px}.progress-popup{display:flex;flex-direction:column;justify-content:center;align-items:center}.progress-popup .desc{color:#000000de;text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}.behavioural-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #F8B861;background:linear-gradient(0deg,#f8b86152 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.functional-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #E24577;background:linear-gradient(0deg,#e245773d 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.domain-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #7B47A4;background:linear-gradient(0deg,#7b47a43d 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.popup-header{display:flex;flex-direction:row;justify-content:space-between}.popup-footer{display:flex;flex-direction:row;justify-content:end}.view-cbp-plan{margin:10px;padding:10px}.view-cbp-plan-popup{padding:24px;max-height:70vh;overflow-y:auto}.section{border-radius:8px;background:#1b4ca114;padding:16px}.popup-header{display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid #ddd;font-weight:700;flex-shrink:0}.popup-footer{display:flex;justify-content:flex-end;gap:16px;padding:16px 24px;border-top:1px solid #ddd;background-color:#fff;flex-shrink:0;position:sticky;bottom:0}.section-container{overflow-y:auto;flex:1 1 auto}.competency-list{display:flex;flex-direction:row;gap:0px;flex-wrap:wrap;margin:0;padding:0;list-style:none}.competency-text{color:#000;font-family:Lato;font-size:12px;font-style:normal;font-weight:400;line-height:normal;padding:3px 6px}.competency-text-behavioral{color:#654321;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-text-functional{color:#b01669;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-text-domain{color:#301934;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-sub-heading{color:#1b4ca1;font-family:Montserrat;font-size:14px;font-style:normal;font-weight:600;line-height:normal}.cursor-pointer{cursor:pointer}.btn-active{border-radius:var(--Radius-Full-Round, 9999999px);opacity:1;background:var(--Primary-KB-Primary-Light, #1B4CA1);color:var(--white-kb-white-100, #FFF);text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:700;line-height:normal;display:flex;height:32px;padding:0 16px;justify-content:center;align-items:center;gap:8px;border:none}.custom-textarea{color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}.initial-loading-container{display:flex;justify-content:center;align-items:center;min-height:400px;width:100%}.initial-loading-container .loading-content{display:flex;flex-direction:column;align-items:center;text-align:center;padding:3rem 2rem}.initial-loading-container .loading-content .loading-text{color:#1b4ca1;font-family:Lato,sans-serif;min-width:320px}.initial-loading-container .loading-content .loading-text .main-message{font-size:16px;font-weight:600;margin-bottom:1rem;color:#1b4ca1}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator{margin-bottom:.5rem}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container{width:300px;height:8px;background-color:#e0e0e0;border-radius:4px;overflow:hidden;margin-bottom:.5rem}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container .progress-bar{height:100%;background:linear-gradient(90deg,#1b4ca1,#4caf50);border-radius:4px;transition:width .3s ease-in-out}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-text{font-size:12px;color:#666;font-weight:500}.initial-loading-container .loading-content .loading-text .progress-details .time-estimate{font-size:12px;color:#888;font-style:italic}.overlay-loader{position:fixed;top:0;left:0;width:100vw;height:100vh;background:#ffffffbf;z-index:9999;display:flex;justify-content:center;align-items:center}.overlay-loader .loading-content{display:flex;flex-direction:column;align-items:center;text-align:center;background:#fff;padding:2rem;border-radius:8px;box-shadow:0 4px 12px #00000026}.overlay-loader .loading-content .loading-text{color:#1b4ca1;font-family:Lato,sans-serif;min-width:320px}.overlay-loader .loading-content .loading-text .main-message{margin:0 0 16px;font-size:18px;font-weight:600;text-align:center}.overlay-loader .loading-content .loading-text .progress-details{margin-top:16px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator{margin-bottom:12px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container{width:100%;height:8px;background-color:#e0e0e0;border-radius:4px;overflow:hidden;margin-bottom:8px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container .progress-bar{height:100%;background:linear-gradient(90deg,#1b4ca1,#2563eb);border-radius:4px;transition:width .5s ease-in-out}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-text{font-size:14px;font-weight:500;color:#1b4ca1;display:block;text-align:center}.overlay-loader .loading-content .loading-text .progress-details .time-estimate{color:#666;font-size:12px;font-style:italic;display:block;text-align:center;margin-top:8px}.competencis-grid{display:flex;flex-direction:row;justify-content:space-between;gap:16px}.grid-list-item-behavioral div{border-left:2px solid darkorange}.grid-list-item-functional div{border-left:2px solid #ff1493}.grid-list-item-domain div{border-left:2px solid #6b21a8}.competencies-grid-3{width:33%}.bg-blue-50{background-color:#fde8cc;border-radius:4px}.bg-green-50{background-color:#f8d2de;border-radius:4px}.bg-purple-50{background-color:#dfd3e9;border-radius:4px}.grid-list-item div{padding:5px 10px;margin:10px 0}.course-pill-container{display:flex;gap:8px}.relevancy-pill-green{display:flex;padding:4px 8px;align-items:center;gap:4px;border-radius:16px;border:1px solid #e0e0e0;background:#f8f9fa}.mt-1{margin-top:.25rem}.mt-2{margin-top:.5rem}.mt-3{margin-top:.75rem}.show-more-competency{display:inline-flex;align-items:center}.show-more-link{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:500;line-height:normal;margin:2px 4px 2px 0;text-decoration:none}.show-more-link:hover{text-decoration:underline}.course-content{flex:1;display:flex;flex-direction:column}.course-footer{margin-top:auto;padding-top:12px;display:flex;justify-content:space-between}::ng-deep .mdc-tab__text-label{font-weight:700;font-family:Montserrat!important;font-size:16px;color:#000!important;text-transform:capitalize}.competency-tag{display:inline-block;padding:.25rem .75rem;border-radius:9999px;font-size:.75rem;font-weight:500;margin-right:.5rem;margin-bottom:.5rem}.menu-list-item.active{background-color:#1b4ca1;color:#fff}.menu-list-item:hover{background-color:#1b4ca1;color:#fff;border-color:#1b4ca1}.gray-bg{background-color:rgb(248,250,252,var(--tw-bg-opacity, 1));border-radius:8px;border:1px solid #ccc}.div-count{font-family:Lato;font-size:20px;font-weight:700}.green-bg{background-color:#0080001c;border:1px solid #006400;border-radius:8px}.div-green-count{font-family:Lato;font-size:20px;font-weight:700;color:#006400}.div-green-label{color:#006400;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.gray-bg-item{background-color:rgb(248,250,252,var(--tw-bg-opacity, 1));border-radius:8px;border:1px solid #ccc;padding:4px 8px;margin:10px}.add-btn{color:#1b4ca1;font-size:14px;font-family:Lato}.list-flex{display:flex;flex-direction:row;justify-content:space-between}.all-category-list .mat-icon{vertical-align:middle}.course-list-container-flex{display:flex;align-items:flex-start;gap:2rem}.left-panel{flex:0 0 250px;font-weight:700;text-align:left}.right-panel{flex:1;display:flex;flex-direction:row;flex-wrap:wrap;gap:20px;align-content:flex-start;justify-content:flex-start}.category-list-item{font-weight:400;font-size:16px;font-family:Lato;padding:5px}.right-panel-flex{display:flex;flex:0 0 100%;width:100%;flex-direction:row;flex-wrap:wrap;gap:20px;justify-content:flex-start}.course-list-container-right-flex{display:flex}.no-gaps-message{display:flex;flex-direction:column;align-items:center;padding:24px;background:linear-gradient(135deg,#4caf501a,#81c7841a);border:2px solid #4CAF50;border-radius:16px;margin:16px 0}.no-gaps-message .success-icon{font-size:25px;color:#4caf50;margin-bottom:16px}.no-gaps-message h4{color:#2e7d32;font-family:Montserrat;font-size:20px;font-weight:600;margin:0 0 8px;text-align:center}.no-gaps-message p{color:#388e3c;font-family:Lato;font-size:14px;font-weight:400;margin:0;text-align:center;line-height:1.5}.section-header-improved,.section-header-improved-sec-2{display:flex;flex-direction:row;gap:0px;width:100%}.search-and-buttons-container{display:flex;justify-content:space-between;gap:5px;flex-direction:column;flex-wrap:wrap;width:37%;margin:0 5px}.search-container-compact{flex:0 0 auto;min-width:180px}.search-container-compact .rsearch{position:relative;display:flex;align-items:center}.search-container-compact .search-icon{position:absolute;left:12px;color:#666;font-size:20px;z-index:1}.search-container-compact .sinput-compact{border-radius:25px;border:2px solid #ccc;padding:20px 12px 20px 40px;height:32px;width:100%;font:400 14px Lato;box-sizing:border-box;transition:border-color .3s ease}.search-container-compact .sinput-compact:focus{outline:none;border-color:#1b4ca1;box-shadow:0 0 0 2px #1b4ca11a}.search-container-compact .sinput-compact::placeholder{color:#999}.btn-group-compact{display:flex;gap:8px;flex:0 0 auto}.category-list-item{padding:8px 12px!important;border-radius:6px;border:1px solid transparent;transition:all .3s ease;background:#f8f9fa;color:#333;font-family:Lato;font-size:14px!important;font-weight:400!important}.category-list-item:hover{background:#1b4ca10d;border-color:#1b4ca133}.category-list-item.selected-theme{background:#1b4ca11a;border-color:#1b4ca1;color:#1b4ca1;font-weight:600!important}.category-list-item.selected-theme:before{content:\"\\2713 \";color:#1b4ca1;font-weight:700}.theme-filter-controls{margin-bottom:8px}.clear-filter-btn{background:#f8f9fa;border:1px solid #ddd;border-radius:20px;padding:4px 12px;font-size:12px;color:#666;cursor:pointer;transition:all .3s ease;display:flex;align-items:center;gap:4px}.clear-filter-btn:hover{background:#e9ecef;border-color:#adb5bd}.clear-filter-btn mat-icon{font-size:16px;width:16px;height:16px}.no-course-found-container{display:flex;justify-content:center;align-items:center;min-height:300px;padding:20px}.no-course-card{background:linear-gradient(135deg,#f8f9fa,#e9ecef);border:2px dashed #ced4da;border-radius:16px;padding:40px;text-align:center;max-width:500px;box-shadow:0 4px 12px #0000000d}.no-course-card .no-course-icon{margin-bottom:20px}.no-course-card .no-course-icon mat-icon{font-size:48px;width:48px;height:48px;color:#6c757d}.no-course-card .no-course-content h4{color:#495057;font-family:Montserrat;font-size:24px;font-weight:600;margin-bottom:12px}.no-course-card .no-course-content p{color:#6c757d;font-family:Lato;font-size:16px;line-height:1.5;margin-bottom:12px}.no-course-card .no-course-content p strong{color:#1b4ca1;font-weight:600}.no-course-card .suggest-course-btn{background:linear-gradient(135deg,#1b4ca1,#164080);color:#fff;border:none;border-radius:25px;padding:12px 24px;font-family:Lato;font-size:14px;font-weight:600;cursor:pointer;display:flex;align-items:center;gap:8px;margin:20px auto 0;transition:all .3s ease;box-shadow:0 4px 12px #1b4ca14d}.no-course-card .suggest-course-btn:hover{background:linear-gradient(135deg,#164080,#0f2f5f);transform:translateY(-2px);box-shadow:0 6px 16px #1b4ca166}.no-course-card .suggest-course-btn:active{transform:translateY(0)}.no-course-card .suggest-course-btn mat-icon{font-size:18px;width:18px;height:18px}.left-panel{flex:0 0 280px!important;background:#f8f9fa;border-radius:12px;padding:16px;border:1px solid #e9ecef;max-height:600px;overflow-y:auto;font-weight:400!important}@media (max-width: 1200px){.search-and-buttons-container{flex-direction:column;align-items:stretch}.search-container-compact{min-width:auto;width:100%}.btn-group-compact{justify-content:center;flex-wrap:wrap}}@media (max-width: 1200px) and (min-width: 769px){.course-list-item{flex:1 1 calc(50% - 10px);min-width:300px;max-width:none}}@media (max-width: 768px){.course-list-container-flex{flex-direction:column}.left-panel{flex:none!important;max-height:300px}.course-list-item{flex:0 0 100%;max-width:100%}.no-course-card{padding:20px}.no-course-card .no-course-content h4{font-size:20px}.no-course-card .no-course-content p{font-size:14px}}::ng-deep .error-snackbar{background-color:#f44336!important;color:#fff!important;font-weight:500}::ng-deep .error-snackbar .mat-simple-snackbar-action{color:#fff!important}::ng-deep .session-expired-snackbar{background-color:#ff9800!important;color:#fff!important;font-weight:600}::ng-deep .session-expired-snackbar .mat-simple-snackbar-action{color:#fff!important;font-weight:700}.download-pdf{justify-content:end;display:flex;flex-direction:row}.gap-analysis-pdf-container{font-family:Lato;max-width:900px;margin:auto;color:#212121}.gap-analysis-pdf-header{text-align:center;margin-bottom:2rem}.gap-analysis-pdf-header .logo{height:60px;margin-bottom:.5rem}.gap-analysis-pdf-header h1{font-size:1.8rem;font-weight:600;margin:.5rem 0}.gap-analysis-pdf-header .subtitle{color:#555;font-size:.95rem}.gap-analysis-pdf-header .download-btn{background:#5e00ff;color:#fff;border:none;padding:10px 20px;border-radius:6px;margin-top:1rem;font-weight:500;cursor:pointer}.competency-snapshot{background:#fafafa;border-radius:12px;padding:2rem;margin-bottom:2rem}.competency-snapshot h2{font-size:1.4rem;font-weight:600;margin-bottom:1.5rem}.competency-snapshot .snapshot-content{display:flex;gap:2rem;align-items:center;justify-content:space-between}.competency-snapshot .snapshot-content .circle-wrapper{position:relative;width:120px;height:120px}.competency-snapshot .snapshot-content .circle-wrapper .progress-ring-bg{fill:none;stroke:#eee;stroke-width:10}.competency-snapshot .snapshot-content .circle-wrapper .progress-ring-circle{fill:none;stroke:#6c63ff;stroke-width:10;stroke-linecap:round;stroke-dasharray:326.72;stroke-dashoffset:117.6192;transform:rotate(-90deg);transform-origin:center;transition:stroke-dashoffset .35s}.competency-snapshot .snapshot-content .circle-wrapper .percentage-text{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);font-weight:600;font-size:1.4rem;text-align:center}.competency-snapshot .snapshot-content .circle-wrapper .percentage-text span{display:block;font-size:.75rem;font-weight:400;color:#555}.competency-snapshot .snapshot-content .details{flex-grow:1}.competency-snapshot .snapshot-content .details .total-competencies{font-size:1.2rem;margin-bottom:1rem}.competency-snapshot .snapshot-content .details .bar{margin:.5rem 0;padding:.6rem 1rem;border-radius:6px;font-weight:500;display:flex;justify-content:space-between;color:#fff}.competency-snapshot .snapshot-content .details .bar.behavioral{background:#7b1fa2}.competency-snapshot .snapshot-content .details .bar.functional{background:#0288d1}.competency-snapshot .snapshot-content .details .bar.domain{background:#f57c00}.gap-section{margin-top:2rem}.gap-section h2{font-size:1.3rem;margin-bottom:1rem}.gap-section .gap-box{background:#eef1ff;border-left:5px solid #6c63ff;padding:1rem;border-radius:6px;font-weight:500;color:#333}.gap-section .green-bg{background-color:#0080001c;border-left:5px solid #2d9b2d;padding:1rem;border-radius:6px;font-weight:500;color:#006400}.competency-container{width:100%;margin:0 auto;font-family:Segoe UI,sans-serif;color:#212121;display:flex;flex-direction:column}.total-box{background:#f8f9fc;border-radius:10px;padding:1.5rem;text-align:center;border:1px solid #e0e0e0;margin-bottom:1.5rem}.total-box .title{font-size:1rem;color:#666}.total-box .count{font-size:2rem;font-weight:700;color:#111;margin-top:.5rem}.progress-group{margin-bottom:1.5rem}.progress-group .progress-label{display:flex;justify-content:space-between;margin-bottom:6px}.progress-group .progress-label .label-text{font-weight:600}.progress-group .progress-label .covered-text{font-size:.9rem;color:#444}.progress-group .progress-label .behavioral{color:#f8b861}.progress-group .progress-label .functional{color:#e24577}.progress-group .progress-label .domain{color:#7b47a4}.progress-group .progress-bar{height:10px;background-color:#e0e0e0;border-radius:5px;overflow:hidden}.progress-group .progress-bar .progress-fill{height:100%;border-radius:5px;transition:width .5s ease-in-out}.progress-group .progress-bar .progress-fill.behavioral{background:#f8b861}.progress-group .progress-bar .progress-fill.functional{background:#e24577}.progress-group .progress-bar .progress-fill.domain{background:#7b47a4}.competency-card{border-radius:8px;padding:1rem 1.5rem;margin-bottom:1rem;border:1px solid transparent}.competency-card .title{font-weight:600;font-size:1.1rem;color:#222;margin-bottom:.4rem}.competency-card .subtitle{font-size:.95rem;color:#555}.competency-card .subtitle .tag{font-weight:500}.competency-card .subtitle .behavioral,.competency-card .subtitle .domain,.competency-card .subtitle .functional{color:#000}.competency-card.behavioral{background:linear-gradient(0deg,#f8b86152 0% 100%),#fff;border-color:#f8b861}.competency-card.functional{background:linear-gradient(0deg,#e245773d 0% 100%),#fff;border-color:#e24577}.competency-card.domain{background:linear-gradient(0deg,#7b47a43d 0% 100%),#fff;border-color:#7b47a4}.disclaimer-container{margin:8px 16px 0}.disclaimer-message{background-color:#fff3cd;border:1px solid #ffeaa7;border-radius:20px;padding:8px 16px;font-family:Lato;font-size:14px;color:#333;line-height:1.2}.disclaimer-message strong{color:#000;font-weight:600}.select-all-container{display:flex;flex-direction:row;justify-content:space-between}:host ::ng-deep .search-and-buttons-container .mat-form-field-infix{padding:10px!important}:host ::ng-deep .search-and-buttons-container .mat-form-field-wrapper{width:100%!important}:host ::ng-deep .select-search-panel{border:1px solid #ccc!important;border-radius:12px!important;box-shadow:0 8px 20px #0000001f}:host ::ng-deep .mat-select-arrow-wrapper{margin-top:5px!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline{color:#ccc}:host ::ng-deep .mat-form-field-appearance-outline.mat-focused .mat-form-field-outline-thick{color:#1b4ca1;border-color:#1b4ca1!important}:host ::ng-deep .select-search-panel .mat-option.mat-selected:not(.mat-option-disabled){background-color:#1b4ca1!important;color:#fff!important}:host ::ng-deep .mat-form-field-appearance-outline{border-radius:25px;border:1px solid #ccc}:host ::ng-deep .mat-form-field-focused .mat-form-field-appearance-outline{border-color:#1b4ca1!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-start{border-radius:25px 0 0 25px;border:none!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-end{border-right:none!important;border:none!important;border-radius:0 25px 25px 0}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-gap{border:none!important;border-radius:25px}:host::ng-deep .mat-form-field-appearance-outline .mat-form-field-infix{background-color:transparent!important}:host ::ng-deep .search-and-buttons-container .mat-option-text{color:#fff!important}\n"], dependencies: [{ kind: "directive", type: i5.NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i5.NgSwitch, selector: "[ngSwitch]", inputs: ["ngSwitch"] }, { kind: "directive", type: i5.NgSwitchCase, selector: "[ngSwitchCase]", inputs: ["ngSwitchCase"] }, { kind: "directive", type: i5.NgSwitchDefault, selector: "[ngSwitchDefault]" }, { kind: "directive", type: i4.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "component", type: i6.MatLegacyFormField, selector: "mat-form-field", inputs: ["color", "appearance", "hideRequiredMarker", "hintLabel", "floatLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i6.MatLegacyLabel, selector: "mat-label" }, { kind: "directive", type: i7.MatLegacyInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", exportAs: ["matInput"] }, { kind: "component", type: i8.MatLegacyButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "component", type: i10.MatLegacySelect, selector: "mat-select", inputs: ["disabled", "disableRipple", "tabIndex"], exportAs: ["matSelect"] }, { kind: "component", type: i11.MatLegacyOption, selector: "mat-option", exportAs: ["matOption"] }, { kind: "component", type: i12.MatLegacyCheckbox, selector: "mat-checkbox", inputs: ["disableRipple", "color", "tabIndex"], exportAs: ["matCheckbox"] }, { kind: "directive", type: i4.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "directive", type: i4.FormControlName, selector: "[formControlName]", inputs: ["formControlName", "disabled", "ngModel"], outputs: ["ngModelChange"] }, { kind: "component", type: i13.MatProgressSpinner, selector: "mat-progress-spinner, mat-spinner", inputs: ["color", "mode", "value", "diameter", "strokeWidth"], exportAs: ["matProgressSpinner"] }, { kind: "component", type: i14.MatTab, selector: "mat-tab", inputs: ["disabled"], exportAs: ["matTab"] }, { kind: "component", type: i14.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "disableRipple", "fitInkBarToContent", "mat-stretch-tabs"], exportAs: ["matTabGroup"] }, { kind: "component", type: i15.GapAnalysisRecommendedCourseComponent, selector: "app-gap-analysis-recommended-course", inputs: ["planData"] }, { kind: "pipe", type: i5.TitleCasePipe, name: "titlecase" }] }); }
|
|
1868
|
+
}
|
|
1869
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: GenerateCourseRecommendationComponent, decorators: [{
|
|
1870
|
+
type: Component,
|
|
1871
|
+
args: [{ selector: 'app-generate-course-recommendation', template: "<div class=\"view-cbp-plan\">\n <div class=\"popup-container\">\n \n <!-- Initial Loading Screen (for initial load and regeneration) -->\n <div class=\"initial-loading-container\" *ngIf=\"!dataLoaded || isRegeneratingWithProgress\">\n <div class=\"loading-content\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <div class=\"loading-text mt-3\">\n <p class=\"main-message\">{{ currentProcessingStage || (isRegeneratingWithProgress ? 'Regenerating course recommendations...' : 'Loading course recommendations...') }}</p>\n <div class=\"progress-details\" *ngIf=\"progressPercentage > 0\">\n <div class=\"progress-indicator\">\n <div class=\"progress-bar-container\">\n <div class=\"progress-bar\" \n [style.width.%]=\"progressPercentage\">\n </div>\n </div>\n <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n </div>\n <small class=\"time-estimate\">\n Real-time course generation in progress. This may take 1-2 minutes.\n </small>\n </div>\n </div>\n </div>\n </div>\n\n <!-- Main Content -->\n <div class=\"main-content\" *ngIf=\"dataLoaded && !isRegeneratingWithProgress\">\n <div class=\"popup-header\">\n <div class=\"header-left\">\n <div class=\"heading\">Save Courses (<span>{{sharedService?.cbpPlanFinalObj?.ministry?.orgName}} <span\n *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span></span>)</div>\n </div>\n <div class=\"header-actions\">\n <button class=\"regenerate-btn\" (click)=\"regenerateCourseRecommendations()\"\n [disabled]=\"isRegenerating\"\n mat-raised-button color=\"primary\">\n <mat-icon *ngIf=\"!isRegenerating\">refresh</mat-icon>\n <mat-spinner *ngIf=\"isRegenerating\" diameter=\"20\"></mat-spinner>\n <span>{{isRegenerating ? 'Regenerating...' : 'Regenerate Recommendations'}}</span>\n </button>\n <div class=\"cursor-pointer close-btn\" (click)=\"closeDialog()\">\n <mat-icon>close</mat-icon>\n </div>\n </div>\n </div>\n <div class=\"section-container mt-4\">\n <mat-tab-group (selectedTabChange)=\"onOuterTabChange($event)\">\n <!-- First Main Tab -->\n <mat-tab label=\"course recommendation\">\n <!-- Nested Tabs -->\n <!-- <mat-tab-group (selectedTabChange)=\"onInnerTabChange($event)\">\n <mat-tab label=\"all\">\n\n </mat-tab>\n <mat-tab label=\"behavioural\">\n\n </mat-tab>\n <mat-tab label=\"functional\">\n\n </mat-tab>\n <mat-tab label=\"domain\">\n\n </mat-tab>\n </mat-tab-group> -->\n </mat-tab>\n\n <!-- Second Main Tab -->\n <mat-tab label=\"gap analysis\">\n <!-- <p>Gap analysis content goes here</p> -->\n </mat-tab>\n </mat-tab-group>\n <form [formGroup]=\"filterForm\" class=\"filters-container\">\n <div class=\"role-mapping-container\"\n *ngIf=\"outerTabActiveIndex === 0 && (innerTabActiveIndex === 0 || innerTabActiveIndex === 1 || innerTabActiveIndex === 2 || innerTabActiveIndex === 3 )\">\n <div class=\"container\">\n <div class=\"section-header-improved mt-3\">\n <div class=\"search-and-buttons-container\">\n <div>\n <label>Search Courses</label>\n </div> \n <div class=\"search-container-compact\">\n <div class=\"rsearch\">\n <mat-icon class=\"search-icon\">search</mat-icon>\n <input class=\"sinput-compact\" [(ngModel)]=\"searchText\"\n (input)=\"applyFilter($event.target.value);\"\n placeholder=\"Search Courses\" type=\"text\">\n </div>\n </div>\n \n </div>\n \n\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Competency</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.competency?.length ? '' : 'Competency' }}\n </mat-label>\n \n <mat-select\n formControlName=\"competency\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Competency\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'competency')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredCompetency\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div> \n \n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Rating</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.rating?.length ? '' : 'Rating' }}\n </mat-label>\n \n <mat-select\n formControlName=\"rating\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Reatings\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'rating')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredRatings\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n \n \n \n \n </div>\n <div class=\"section-header-improved-sec-2 mt-4\">\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Language</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.language?.length ? '' : 'Language' }}\n </mat-label>\n \n <mat-select\n formControlName=\"language\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Languages\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'language')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredLanguages\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Duration</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.duration?.length ? '' : 'Duration' }}\n </mat-label>\n \n <mat-select\n formControlName=\"duration\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Duration\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'duration')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredDurations\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div class=\"search-and-buttons-container\">\n <div style=\"margin-left: 10px;\">\n <label>Select Provider</label>\n </div> \n <div>\n <mat-form-field appearance=\"outline\">\n <mat-label>\n {{ filterForm.value.provider?.length ? '' : 'Provider' }}\n </mat-label>\n \n <mat-select\n formControlName=\"provider\"\n multiple\n panelClass=\"select-search-panel\"\n (selectionChange)=\"applyFilters()\">\n \n <!-- Search box -->\n <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n matInput\n placeholder=\"Search Provider\"\n (click)=\"$event.stopPropagation()\"\n (keydown)=\"$event.stopPropagation()\"\n (keyup)=\"filterList($any($event.target).value, 'provider')\" />\n \n <!-- Values -->\n <mat-option\n *ngFor=\"let r of filteredProviders\"\n [value]=\"r\">\n {{ r }}\n </mat-option>\n \n </mat-select>\n </mat-form-field>\n </div>\n </div>\n <div style=\"margin-top: 45px;margin-left: 20px;\">\n <input class=\"btn-active\" type=\"button\" value=\"Reset All Filters\"\n (click)=\"resetFilters()\" />\n </div>\n </div>\n <div class=\"mt-4\">\n <div style=\"font-size:16px;font-weight:bold;color:#000\">\n Filtered Course Count : {{filterdCourses?.length}}\n </div>\n </div>\n <div class=\"mt-5 sub-heading select-all-container\" *ngIf=\"filterdCourses?.length\">\n <div style=\"margin-left: 16px;\">\n <mat-checkbox [checked]=\"false\" (change)=\"selectAllCourses($event)\">Select All\n Courses</mat-checkbox>\n </div>\n <div class=\"btn-group-compact\">\n <div>\n <input class=\"btn-active\" type=\"button\" value=\"Suggest More Course From iGOT\"\n (click)=\"suggestMoreCourses()\" />\n </div>\n <div class=\"pl-2\">\n <input [disabled]=\"selectFilterCourses?.length > 0 ? false : true\"\n [ngClass]=\"selectFilterCourses?.length > 0 ? 'btn-active' : 'btn-disable'\"\n type=\"button\" [value]=\"mode === 'add' ? 'Save Courses' : 'Update Courses'\"\n (click)=\"saveCourses()\" />\n </div>\n </div>\n </div>\n <div class=\"disclaimer-container mt-2\" *ngIf=\"filterdCourses?.length\">\n <div class=\"disclaimer-message mt-4\">\n <strong>Disclaimer:</strong> Please verify the public course URL before adding it to the plan.\n </div>\n </div>\n <div class=\"mt-2\">\n\n <div class=\"course-list-container\">\n <div\n [ngClass]=\"innerTabActiveIndex > 0 ? 'course-list-container-flex':'course-list-container-right-flex' \">\n <div *ngIf=\"innerTabActiveIndex > 0\"\n [ngClass]=\"innerTabActiveIndex > 0 ? 'left-panel':''\">\n <ng-container *ngIf=\"innerTabActiveIndex === 1\">\n <div class=\"sub-heading\">\n <p>Competencies in Behavioural</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n <ng-container *ngIf=\"innerTabActiveIndex === 2\">\n <div class=\"sub-heading\">\n <p>Competencies in Functional</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n <ng-container *ngIf=\"innerTabActiveIndex === 3\">\n <div class=\"sub-heading\">\n <p>Competencies in Domain</p>\n </div>\n <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n <mat-icon>clear</mat-icon> Clear Filter\n </button>\n </div>\n <div (click)=\"filterOnCompetencyTheme(item)\"\n class=\"category-list-item cursor-pointer mt-2\"\n [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n *ngFor=\"let item of competencyMatchedByCategory\">\n {{item}}\n </div>\n </ng-container>\n </div>\n <div [ngClass]=\"innerTabActiveIndex > 0 ? 'right-panel':'right-panel-flex'\">\n <!-- No Course Found Message in Right Panel -->\n <div class=\"no-course-found-container\"\n *ngIf=\"filterdCourses?.length === 0 && selectedThemeFilter\">\n <div class=\"no-course-card\">\n <div class=\"no-course-icon\">\n <mat-icon>search_off</mat-icon>\n </div>\n <div class=\"no-course-content\">\n <h4>No Courses Found</h4>\n <p>No courses found for the selected theme:\n <strong>{{selectedThemeFilter}}</strong></p>\n <p>Would you like to add a course for this competency?</p>\n <button class=\"suggest-course-btn\"\n (click)=\"addCourseForSelectedTheme()\">\n <mat-icon>add_circle</mat-icon>\n Add Course\n </button>\n </div>\n </div>\n </div>\n\n <!-- Courses List -->\n <div class=\"course-list-item\" *ngFor=\"let item of filterdCourses;let i=index\">\n\n <div class=\"course-content\">\n <div class=\"course-header\">\n <div class=\"checked-course-container\">\n <div>\n <mat-checkbox [checked]=\"checkIfCourseExists(item)\"\n (change)=\"selectedFilterCourses($event, item)\"></mat-checkbox>\n </div>\n <div class=\"ml-2\">\n <div class=\"course-pill-container\">\n <div class=\"course-pill\">\n <div class=\"course-pill-text\">\n <span><img\n src=\"assets/icons/course.svg\"></span> <span>Course</span>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n\n <!-- <div>\n <div class=\"ai-recommened-pill\">\n <div class=\"ai-recommened-pill-text\">\n <span class=\"ai-loader-icon\"><img src=\"assets/icons/ai-loader.gif\"></span>\n\n <span *ngIf=\"item?.rationale\">AI Recommended\n <span *ngIf=\"item?.is_public\"> - Public</span>\n <span *ngIf=\"!item?.is_public\"> - iGOT</span>\n </span>\n <span *ngIf=\"!item?.rationale\">Manually Suggested - iGOT</span>\n\n </div>\n </div>\n </div> -->\n <div *ngIf=\"item?.rationale && !item?.is_public\">\n <div class=\"ai-recommened-pill-green\">\n <div class=\"ai-recommened-pill-green-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>AI Recommended - iGOT </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n <div *ngIf=\"item?.rationale && item?.is_public\">\n <div class=\"ai-recommened-pill-public\">\n <div class=\"ai-recommened-pill-public-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>AI Recommended - Public </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n <div *ngIf=\"!item?.rationale\">\n <div class=\"ai-recommened-pill-gray\">\n <div class=\"ai-recommened-pill-gray-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n\n <span>Manually Suggested - iGOT </span>\n\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n target=\"_blank\"><span>Platform -\n {{item?.platform}}</span></a>\n </div>\n </div>\n <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n <div class=\"ai-recommened-pill-orange-text\">\n <span class=\"ai-loader-icon\"><img\n src=\"assets/icons/ai-loader.gif\"></span>\n <span>Provider - {{item?.organisation[0]}}</span>\n </div>\n </div>\n </div>\n\n </div>\n <div class=\"course-title mt-3 cursor-pointer\" (click)=\"redirectToToc(item)\">\n <span *ngIf=\"item?.course\">{{item?.course}}</span>\n <!-- <span *ngIf=\"item?.name\">{{item?.name}}****</span> -->\n <span style=\"margin-left:10px;\" (click)=\"redirectToToc(item)\"><img\n height=\"18px\" width=\"18px\"\n src=\"assets/icons/redirect.png\" /></span>\n </div>\n <div class=\"course-desc mt-2\">\n <!-- <span *ngIf=\"item?.rationale\">{{item?.rationale}}</span> -->\n <span *ngIf=\"item?.description\">{{item?.description}}</span>\n </div>\n <div class=\"mt-3\" *ngIf=\"getCompetenciesByType('Behavioural',i)?.length\">\n <div class=\"competency-sub-heading\">Behavioural Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"behavioural-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Behavioural',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Behavioural',i)\"\n class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Behavioural',i)\">\n {{ isExpanded('Behavioural',i) ? 'Show Less' : '+' +\n getRemainingCount('Behavioural',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Functional',i)?.length\">\n <div class=\"competency-sub-heading\">Functional Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"functional-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Functional',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Functional',i)\"\n class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Functional',i)\">\n {{ isExpanded('Functional',i) ? 'Show Less' : '+' +\n getRemainingCount('Functional',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Domain',i)?.length\">\n <div class=\"competency-sub-heading\">Domain Competencies</div>\n <ul class=\"competency-list mt-1\">\n <li class=\"domain-pill\"\n *ngFor=\"let comp of getDisplayedCompetencies('Domain',i)\">\n <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n {{\n comp.competencySubThemeName }} </span>\n </li>\n <li *ngIf=\"hasMoreThanTwo('Domain',i)\" class=\"show-more-competency\">\n <span class=\"show-more-link cursor-pointer\"\n (click)=\"toggleCompetencies('Domain',i)\">\n {{ isExpanded('Domain',i) ? 'Show Less' : '+' +\n getRemainingCount('Domain',i) + ' More' }}\n </span>\n </li>\n </ul>\n </div>\n\n\n </div>\n\n <!-- Relevancy and Duration section -->\n <div class=\"course-footer\" *ngIf=\"item?.relevancy\">\n <div class=\"relevancy-pill-green\">\n <div class=\"relevancy-container\">\n <div class=\"relevancy\">Relevancy</div>\n <div class=\"dash\">-</div>\n <div class=\"percentage\">{{item?.relevancy}}%</div>\n </div>\n </div>\n <div class=\"relevancy-pill-green\">\n <div class=\"relevancy-container\">\n <div class=\"relevancy\">Duration</div>\n <div class=\"dash\">-</div>\n <div class=\"percentage\">\n <span\n *ngIf=\"item?.duration\">{{getDuration(item?.duration)}}</span>\n <span *ngIf=\"!(item?.duration)\">N/A</span>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <!-- General No Data Message (only when no theme filter is active) -->\n <div class=\"mt-2 sub-heading\" *ngIf=\"filterdCourses?.length === 0 && !selectedThemeFilter\">\n <p>No Data Found</p>\n </div>\n </div>\n </div>\n </form>\n <div class=\"role-mapping-container\" *ngIf=\"outerTabActiveIndex === 1\">\n <div id=\"gap-analysis-content\">\n\n <div class=\"download-pdf mt-4 mb-2\">\n <div>\n <input class=\"btn-active\" type=\"button\" value=\"Download\" (click)=\"downloadPDF()\" />\n </div>\n </div>\n <div>\n <!-- <div *ngIf=\"isPDFDownload\">\n <div class=\"pdf-heading\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}} <span\n *ngIf=\"planData?.designation_name \">_{{planData?.designation_name}}_Gap_Analysis</span>\n </div>\n </div> -->\n <div class=\"d-flex vh-100\" [ngClass]=\"isPDFDownload ? 'mt-4':''\">\n <!-- Sidebar -->\n <div class=\"d-flex flex-column p-3 bg-light border-end\" style=\"width: 250px;\">\n <h5 class=\"mb-4\">Analyze by Category</h5>\n <button *ngFor=\"let item of menuItems\"\n class=\"btn btn-outline-primary mb-2 text-start menu-list-item\"\n [class.active]=\"selectedCategory === item.key\" (click)=\"selectCategory(item.key)\">\n {{ item.label }}\n </button>\n </div>\n\n <!-- Main Content -->\n <div class=\"flex-grow-1 p-4\">\n <div class=\"d-flex justify-content-between gap-3 \">\n <div class=\"p-3 gray-bg\">\n <div class=\"div-label sub-heading\">\n <p>Competencies Covered</p>\n </div>\n <div class=\"div-count\">{{competencyCoveredCount}}</div>\n </div>\n <div class=\"p-3 gray-bg\">\n <div class=\"div-label sub-heading\">\n <p *ngIf=\"selectedCategory === 'all'\">Total Competencies</p>\n <p *ngIf=\"selectedCategory === 'behavioral'\">Total Behavioural Competencies\n </p>\n <p *ngIf=\"selectedCategory === 'functional'\">Total Functional Competencies\n </p>\n <p *ngIf=\"selectedCategory === 'domain'\">Total Domain Competencies</p>\n </div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'all'\">\n {{this.planData?.competencies?.length}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n {{behavioralTotalCompetencies}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'functional'\">\n {{functionalTotalCompetencies}}</div>\n <div class=\"div-count\" *ngIf=\"selectedCategory === 'domain'\">\n {{domainTotalCompetencies}}</div>\n </div>\n <div class=\"p-3 green-bg\">\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'all'\">Overall Coverage\n </div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'behavioral'\">\n Behavioural Coverage</div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'functional'\">\n Functional Coverage</div>\n <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'domain'\">Domain\n Coverage</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'all'\">\n {{overallCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n {{behavioralCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'functional'\">\n {{functionalCoverage}}</div>\n <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'domain'\">\n {{domainCoverage}}</div>\n </div>\n </div>\n <div class=\"mt-4 all-category-list\" [ngSwitch]=\"selectedCategory\">\n <div *ngSwitchCase=\"'all'\">\n <h4>All Categories</h4>\n\n <!-- Show Behavioural section only if there are unmatched behavioural competencies -->\n <div class=\"mt-4\"\n *ngIf=\"behaviouralNotMatched && behaviouralNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Behavioural</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show Functional section only if there are unmatched functional competencies -->\n <div class=\"mt-4\"\n *ngIf=\"functionalNotMatched && functionalNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Functional</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show Domain section only if there are unmatched domain competencies -->\n <div class=\"mt-4\" *ngIf=\"domainNotMatched && domainNotMatched.length > 0\">\n <div class=\"sub-heading\">\n <p>Domain</p>\n </div>\n <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item | titlecase}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n Course</span></div>\n </div>\n </div>\n </div>\n\n <!-- Show message when all competencies are covered -->\n <div class=\"mt-4 text-center\"\n *ngIf=\"(!behaviouralNotMatched || behaviouralNotMatched.length === 0) && \n (!functionalNotMatched || functionalNotMatched.length === 0) && \n (!domainNotMatched || domainNotMatched.length === 0)\">\n <div class=\"no-gaps-message\">\n <mat-icon class=\"success-icon\">check_circle</mat-icon>\n <h4>Excellent! All competencies are covered</h4>\n <p>All required competencies have been addressed by the selected\n courses.</p>\n </div>\n </div>\n\n </div>\n\n <div *ngSwitchCase=\"'behavioral'\">\n <h4>Behavioural</h4>\n <div class=\"mt-4\" *ngIf=\"behavioralCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Behavioural</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"behavioralCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All behavioral competencies are covered by the\n selected courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchCase=\"'functional'\">\n <h4>Functional</h4>\n <div class=\"mt-4\" *ngIf=\"functionalCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Functional</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course</span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"functionalCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All functional competencies are covered by the\n selected courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchCase=\"'domain'\">\n <h4>Domain</h4>\n <div class=\"mt-4\" *ngIf=\"domainCoverage !== '100%'\">\n <!-- <div class=\"sub-heading\">\n <p>Domain</p>\n </div> -->\n <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"list-flex\">\n <div><span>{{item}}</span></div>\n <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n </div>\n </div>\n </div>\n <div class=\"mt-4\" *ngIf=\"domainCoverage === '100%'\">\n <div class=\"p-3\"\n style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n \uD83C\uDF89 Excellent! All domain competencies are covered by the selected\n courses.\n </div>\n </div>\n </div>\n </div>\n\n <div *ngSwitchDefault>\n <p>Please select a category.</p>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n </div>\n </div>\n </div> <!-- Close main-content div -->\n\n </div>\n</div>\n\n\n<div [hidden]=\"!isPDFDownload\" class=\"gap-analysis-pdf-container\" #pdfContent>\n <div class=\"gap-analysis-pdf-header\">\n <img src=\"assets/icons/karmayogiLogo-min.png\" alt=\"Karmayogi Bharat Logo\" class=\"logo\" />\n <h1><span>\n <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'state'\">{{sharedService?.cbpPlanFinalObj?.department_name?.name}}</span> \n <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'ministry'\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}}</span> \n <span *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span>\n </span></h1>\n <p class=\"subtitle\">A detailed breakdown of competencies Gap Analysis</p>\n \n </div>\n \n <div class=\"competency-snapshot\">\n <h2>Competency Snapshot</h2>\n <div class=\"snapshot-content\">\n <div class=\"circle-wrapper\">\n <svg class=\"progress-ring\" width=\"120\" height=\"120\">\n <circle class=\"progress-ring-bg\" cx=\"60\" cy=\"60\" r=\"52\" />\n <circle class=\"progress-ring-circle\" cx=\"60\" cy=\"60\" r=\"52\" />\n </svg>\n <div class=\"percentage-text\">{{overallCoverage}}<br /><span>Overall Coverage</span></div>\n </div>\n \n <div class=\"competency-container\">\n <!-- Total Competencies Box -->\n <div class=\"total-box\">\n <div class=\"title\">Total Competencies</div>\n <div class=\"count\">{{this.planData?.competencies?.length}}</div>\n </div>\n \n <!-- Progress Bars -->\n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text behavioral\">Behavioural</span>\n <span class=\"covered-text\">{{behavioralCompetencyCoveredCount}} / {{behavioralTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill behavioral\" [style.width.%]=\"(behavioralCompetencyCoveredCount / behavioralTotalCompetencies) * 100\"></div>\n </div>\n </div>\n \n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text functional\">Functional</span>\n <span class=\"covered-text\">{{functionalCompetencyCoveredCount}} / {{functionalTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill functional\" [style.width.%]=\"(functionalCompetencyCoveredCount / functionalTotalCompetencies) * 100\"></div>\n </div>\n </div>\n \n <div class=\"progress-group\">\n <div class=\"progress-label\">\n <span class=\"label-text domain\">Domain</span>\n <span class=\"covered-text\">{{domainCompetencyCoveredCount}} / {{domainTotalCompetencies}} Covered</span>\n </div>\n <div class=\"progress-bar\">\n <div class=\"progress-fill domain\" [style.width.%]=\"(domainCompetencyCoveredCount / domainTotalCompetencies) * 100\"></div>\n </div>\n </div>\n </div>\n </div>\n </div>\n \n <div class=\"gap-section mb-4\">\n <h2>Detailed Competency Analysis</h2>\n <div class=\"gap-box\">\n Bridge Your Competencies Gap\n <p class=\"subtitle\">Here are {{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}} key areas for development.</p>\n </div>\n </div>\n <div class=\"sub-heading mb-4\">\n <p>Gap to Address <span *ngIf=\"(this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)) > 0\">({{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}})</span></p>\n \n </div>\n <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n \u2022 \n <span class=\"tag behavioral\">Behavioural</span>\n </div>\n </div>\n \n <div class=\"competency-card functional\" *ngFor=\"let item of functionalNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n \u2022 \n <span class=\"tag functional\">Functional</span>\n </div>\n </div>\n \n <div class=\"competency-card domain\" *ngFor=\"let item of domainNotMatched\">\n <div class=\"title\">{{item | titlecase}}</div>\n <div class=\"subtitle\">\n <!-- Project Management -->\n \u2022 \n <span class=\"tag domain\">Domain</span>\n </div>\n </div>\n <div class=\"sub-heading mb-4\">\n <p>Covered Competencies <span>({{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}})</span></p> \n </div>\n <div class=\"gap-section mb-4\">\n \n <div class=\"green-bg\">\n <div class=\"div-green-label mb-2\">\n Congratulations!\n </div> \n <p>You have successfully covered </p>\n <p class=\"subtitle\">{{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}} key competencies, forming a strong professional foundation.</p>\n </div>\n </div>\n <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n {{item?.sub_theme}} \u2022 \n <span class=\"tag behavioral\">Behavioural</span>\n </div>\n </div>\n \n <div class=\"competency-card functional\" *ngFor=\"let item of functionalMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n {{item?.sub_theme}} \u2022 \n <span class=\"tag functional\">Functional</span>\n </div>\n </div>\n \n <div class=\"competency-card domain\" *ngFor=\"let item of domainMatched\">\n <div class=\"title\">{{item?.theme | titlecase}}</div>\n <div class=\"subtitle\">\n <!-- Project Management -->\n {{item?.sub_theme}} \u2022 \n <span class=\"tag domain\">Domain</span>\n </div>\n </div>\n <div class=\"gap-section mb-4\">\n <h2>Recommended Courses</h2>\n </div>\n <app-gap-analysis-recommended-course [planData]=\"planData\"></app-gap-analysis-recommended-course>\n</div>\n \n\n\n<div class=\"overlay-loader\" *ngIf=\"loading && dataLoaded && !isRegeneratingWithProgress\">\n <div class=\"loading-content\">\n <mat-spinner diameter=\"50\"></mat-spinner>\n <div class=\"loading-text mt-3\">\n <p class=\"main-message\">{{ currentProcessingStage || 'Preparing course recommendation generation...' }}</p>\n <div class=\"progress-details\">\n <div class=\"progress-indicator\" *ngIf=\"progressPercentage > 0\">\n <div class=\"progress-bar-container\">\n <div class=\"progress-bar\" \n [style.width.%]=\"progressPercentage\">\n </div>\n </div>\n <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n </div>\n <small class=\"time-estimate\">\n Real-time course generation in progress. This may take 1-2 minutes.\n </small>\n </div>\n </div>\n </div>\n</div>", styles: ["@charset \"UTF-8\";.container{margin:10px auto}.sub-heading p{color:#000;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.radio-label{color:#000;font-family:Lato;font-size:16px;font-style:normal;font-weight:400;line-height:normal}.radio-btn-group{display:flex;width:32px;height:32px;padding:2px;gap:10px}.label{color:var(--Body-Text-Body-Color, #212529);font-family:Lato;font-size:16px;font-style:normal;font-weight:700;line-height:150%}::ng-deep mat-form-field{background-color:#fff!important;border-radius:25px!important;padding-left:12px;padding-right:12px;height:42px;width:100%}::ng-deep mat-select{border-radius:25px!important;background-color:#fff!important}::ng-deep .mat-select-panel{background-color:#fff!important}::ng-deep .mat-select-panel .mat-option{border-radius:0!important}::ng-deep mat-form-field .mat-form-field-underline{display:none!important}::ng-deep mat-select .mat-select-placeholder{color:#0006;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}::ng-deep mat-select .mat-select-value-text{color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;line-height:normal!important;padding-top:0!important;padding-bottom:0!important}::ng-deep mat-select .mat-select-value{display:flex!important;align-items:center!important;height:100%!important}.additional-details textarea{display:flex;padding:16px 16px 0;flex-direction:column;align-items:flex-start;gap:10px;flex:1 0 0;align-self:stretch;width:100%;color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;resize:none}.section-header{display:flex;flex-direction:row;align-items:center;justify-content:space-between}.btn-disable{border-radius:var(--Radius-Full-Round, 9999999px);opacity:.4;background:var(--Primary-KB-Primary-Light, #1B4CA1);color:var(--white-kb-white-100, #FFF);text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:700;line-height:normal;display:flex;height:32px;padding:0 16px;justify-content:center;align-items:center;gap:8px;border:none}.btn-group{display:flex;flex-direction:row;justify-content:space-between}.pl-2{padding-left:5px}.search-filter-section{display:flex;flex-direction:row;justify-content:space-between}.search .rsearch{position:relative}.search .search-icon{position:absolute;top:15px;font-size:20px;left:10px}.search .sinput{border-radius:32px;border:1px solid #d5d0d0;padding:15px 36px;font:400 14px Lato;width:800px}.select-map-route-container{display:flex;flex-direction:row}.selected-mapping-route p{color:#1b4ca1;font-family:Lato;font-size:20px;font-style:normal;font-weight:400;line-height:normal;text-decoration-line:underline;text-decoration-style:solid;-webkit-text-decoration-skip:ink;text-decoration-skip-ink:auto;text-decoration-thickness:auto;text-underline-offset:auto;text-underline-position:from-font}.edit-plan{display:flex;padding:4px;justify-content:center;align-items:center;gap:8px;border-radius:4px;background:#1b4ca129;color:#1b4ca1;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal;margin-left:10px}.search-container{display:flex;flex-direction:row}.course-container{display:flex;width:100%;padding:16px;flex-direction:column;align-items:flex-start;gap:24px;grid-row:2/span 1;grid-column:1/span 1;border-radius:12px;border:1px solid var(--borders-black-border-16-black, rgba(0, 0, 0, .16));background:#fff}.course-header{display:flex;flex-direction:row;justify-content:space-between;align-items:flex-start;width:100%;gap:8px}.course-pill{display:flex;height:24px;padding:8px;align-items:center;gap:4px;border-radius:16px;border:1px solid var(--borders-border-fill-focussed-4-light-100, #EF951E);background:#fefaf4}.ai-recommened-pill-green{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#0080001c}.ai-recommened-pill-green-text{color:#006400;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-public{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#1b4ca129}.ai-recommened-pill-public-text{color:#1b4ca1;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-gray{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#d3d3d3}.ai-recommened-pill-gray-text{color:#000;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-orange{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);background:#ffa50052}.ai-recommened-pill-orange-text{color:#5a3f0d;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.ai-recommened-pill-orange-text a{color:#5a3f0d!important;font-family:Lato;font-size:11px;font-style:normal;font-weight:600;line-height:normal}.igot-platform-pill{display:flex;padding:4px 8px;align-items:center;gap:8px;border-radius:var(--Radius-4, 4px);border:1px solid #EF951E;background:#ef951e29}.course-title{color:var(--Primary-KB-Primary-Light, #1B4CA1);font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:22px}.course-desc{color:#0009;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:1.4;display:-webkit-box;-webkit-line-clamp:3;-webkit-box-orient:vertical;overflow:hidden;text-overflow:ellipsis}.relevancy,.competencies-matched{color:#6c757d;font-family:Lato;font-size:12px;font-style:normal;font-weight:500;line-height:normal;align-items:center}.percentage{color:#1b4ca1;font-family:Lato;font-size:14px;font-style:normal;font-weight:600;line-height:normal}.relevancy-container,.competencies-matched-container{display:flex;flex-direction:row;align-items:center}.dash{padding:0 5px}::ng-deep .mat-radio-checked .mat-radio-inner-circle{background:#1b4ca1!important}.course-list-container{display:flex;flex-wrap:wrap;gap:16px;width:100%;padding:1rem}.course-list-item{flex:1 1 calc(50% - 10px);min-width:350px;max-width:500px;box-sizing:border-box;border:1px solid #e0e0e0;padding:16px;background-color:#fff;border-radius:12px;min-height:auto;box-shadow:0 2px 4px #0000001a;transition:box-shadow .2s ease;display:flex;flex-direction:column}.course-list-item:hover{box-shadow:0 4px 8px #00000026}.course-list-item:only-child{flex:1 1 auto;max-width:800px}.checked-course-container{display:flex;width:60%}.ml-2{margin-left:8px}::ng-deep .mat-checkbox-checked .mat-checkbox-background{background:#1b4ca1!important}.igot-platform-pill-text{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:700;line-height:normal}.course-pill-text{color:var(--accessbility-shades-without-opacity-kb-greys-black-87, #212121);text-align:center;font-family:Poppins;font-size:12px;font-style:normal;font-weight:400;line-height:normal}.ai-loader-icon img,.igot-platform-icon img{width:15.714px;height:15.714px;flex-shrink:0}.competency-container{display:flex;justify-content:space-between;flex-direction:row}.outside-layer-total{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #1B4CA1;background:#edf1f8;width:132px}.outside-layer-functional{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #E24577;background:#f8d2de;width:132px}.outside-layer-domain{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #7B47A4;background:#dfd3e9;width:132px}.outside-layer-behavioral{display:flex;padding:4px 12px;justify-content:center;align-items:center;gap:16px;border-radius:12px 4px 4px 12px;border-left:2px solid #F8B861;background:#fde8cc;width:132px}.inside-layer{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:400;line-height:normal}.count{color:#1b4ca1;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.suggest-courses-header{display:flex;flex-direction:row;justify-content:space-between}.suggest-course-control{display:flex;flex-direction:row}.course-card-container{width:172px;border-radius:3.307px;border:.827px solid var(--borders-border-8-black, rgba(0, 0, 0, .08));background-color:#fff}.highlight-ai{font-family:Montserrat;font-size:24px;font-style:normal;font-weight:600;line-height:normal;background:linear-gradient(90deg,#3b27ff,#2de3d6);background-clip:text;-webkit-background-clip:text;color:transparent;-webkit-text-fill-color:transparent;display:inline-block}.ai-loader-icon-medium img{width:40px;height:40px;flex-shrink:0}.heading{color:#000;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal;margin:10px 0}.pdf-heading{color:#000;font-family:Montserrat;font-size:18px;font-style:normal;font-weight:600;line-height:normal;margin:10px 0;text-align:center}.popup-header{display:flex;justify-content:space-between;align-items:center;padding:16px;border-bottom:1px solid #e0e0e0}.header-left{flex:1}.header-actions{display:flex;align-items:center;gap:12px}.regenerate-btn{display:flex;align-items:center;gap:8px;padding:8px 16px;border-radius:6px;font-size:14px;font-weight:500;min-width:auto}.regenerate-btn .mat-icon{margin:0;font-size:18px;width:18px;height:18px}.regenerate-btn .mat-spinner{margin:0}.regenerate-btn:disabled{opacity:.7;cursor:not-allowed}.close-btn{padding:8px;border-radius:4px;transition:background-color .2s ease}.close-btn:hover{background-color:#f5f5f5}.close-btn .mat-icon{font-size:20px;width:20px;height:20px}.progress-popup{display:flex;flex-direction:column;justify-content:center;align-items:center}.progress-popup .desc{color:#000000de;text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}.behavioural-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #F8B861;background:linear-gradient(0deg,#f8b86152 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.functional-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #E24577;background:linear-gradient(0deg,#e245773d 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.domain-pill{display:flex;padding:2px 6px;justify-content:center;align-items:center;gap:4px;border-radius:16px;border:1px solid #7B47A4;background:linear-gradient(0deg,#7b47a43d 0% 100%),#fff;margin:2px 4px 2px 0;font-size:12px}.popup-header{display:flex;flex-direction:row;justify-content:space-between}.popup-footer{display:flex;flex-direction:row;justify-content:end}.view-cbp-plan{margin:10px;padding:10px}.view-cbp-plan-popup{padding:24px;max-height:70vh;overflow-y:auto}.section{border-radius:8px;background:#1b4ca114;padding:16px}.popup-header{display:flex;justify-content:space-between;align-items:center;border-bottom:1px solid #ddd;font-weight:700;flex-shrink:0}.popup-footer{display:flex;justify-content:flex-end;gap:16px;padding:16px 24px;border-top:1px solid #ddd;background-color:#fff;flex-shrink:0;position:sticky;bottom:0}.section-container{overflow-y:auto;flex:1 1 auto}.competency-list{display:flex;flex-direction:row;gap:0px;flex-wrap:wrap;margin:0;padding:0;list-style:none}.competency-text{color:#000;font-family:Lato;font-size:12px;font-style:normal;font-weight:400;line-height:normal;padding:3px 6px}.competency-text-behavioral{color:#654321;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-text-functional{color:#b01669;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-text-domain{color:#301934;font-family:Lato;font-size:14px;font-style:normal;line-height:normal;font-weight:550}.competency-sub-heading{color:#1b4ca1;font-family:Montserrat;font-size:14px;font-style:normal;font-weight:600;line-height:normal}.cursor-pointer{cursor:pointer}.btn-active{border-radius:var(--Radius-Full-Round, 9999999px);opacity:1;background:var(--Primary-KB-Primary-Light, #1B4CA1);color:var(--white-kb-white-100, #FFF);text-align:center;font-family:Lato;font-size:14px;font-style:normal;font-weight:700;line-height:normal;display:flex;height:32px;padding:0 16px;justify-content:center;align-items:center;gap:8px;border:none}.custom-textarea{color:#000;font-family:Lato;font-size:14px;font-style:normal;font-weight:400;line-height:normal}.initial-loading-container{display:flex;justify-content:center;align-items:center;min-height:400px;width:100%}.initial-loading-container .loading-content{display:flex;flex-direction:column;align-items:center;text-align:center;padding:3rem 2rem}.initial-loading-container .loading-content .loading-text{color:#1b4ca1;font-family:Lato,sans-serif;min-width:320px}.initial-loading-container .loading-content .loading-text .main-message{font-size:16px;font-weight:600;margin-bottom:1rem;color:#1b4ca1}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator{margin-bottom:.5rem}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container{width:300px;height:8px;background-color:#e0e0e0;border-radius:4px;overflow:hidden;margin-bottom:.5rem}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container .progress-bar{height:100%;background:linear-gradient(90deg,#1b4ca1,#4caf50);border-radius:4px;transition:width .3s ease-in-out}.initial-loading-container .loading-content .loading-text .progress-details .progress-indicator .progress-text{font-size:12px;color:#666;font-weight:500}.initial-loading-container .loading-content .loading-text .progress-details .time-estimate{font-size:12px;color:#888;font-style:italic}.overlay-loader{position:fixed;top:0;left:0;width:100vw;height:100vh;background:#ffffffbf;z-index:9999;display:flex;justify-content:center;align-items:center}.overlay-loader .loading-content{display:flex;flex-direction:column;align-items:center;text-align:center;background:#fff;padding:2rem;border-radius:8px;box-shadow:0 4px 12px #00000026}.overlay-loader .loading-content .loading-text{color:#1b4ca1;font-family:Lato,sans-serif;min-width:320px}.overlay-loader .loading-content .loading-text .main-message{margin:0 0 16px;font-size:18px;font-weight:600;text-align:center}.overlay-loader .loading-content .loading-text .progress-details{margin-top:16px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator{margin-bottom:12px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container{width:100%;height:8px;background-color:#e0e0e0;border-radius:4px;overflow:hidden;margin-bottom:8px}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-bar-container .progress-bar{height:100%;background:linear-gradient(90deg,#1b4ca1,#2563eb);border-radius:4px;transition:width .5s ease-in-out}.overlay-loader .loading-content .loading-text .progress-details .progress-indicator .progress-text{font-size:14px;font-weight:500;color:#1b4ca1;display:block;text-align:center}.overlay-loader .loading-content .loading-text .progress-details .time-estimate{color:#666;font-size:12px;font-style:italic;display:block;text-align:center;margin-top:8px}.competencis-grid{display:flex;flex-direction:row;justify-content:space-between;gap:16px}.grid-list-item-behavioral div{border-left:2px solid darkorange}.grid-list-item-functional div{border-left:2px solid #ff1493}.grid-list-item-domain div{border-left:2px solid #6b21a8}.competencies-grid-3{width:33%}.bg-blue-50{background-color:#fde8cc;border-radius:4px}.bg-green-50{background-color:#f8d2de;border-radius:4px}.bg-purple-50{background-color:#dfd3e9;border-radius:4px}.grid-list-item div{padding:5px 10px;margin:10px 0}.course-pill-container{display:flex;gap:8px}.relevancy-pill-green{display:flex;padding:4px 8px;align-items:center;gap:4px;border-radius:16px;border:1px solid #e0e0e0;background:#f8f9fa}.mt-1{margin-top:.25rem}.mt-2{margin-top:.5rem}.mt-3{margin-top:.75rem}.show-more-competency{display:inline-flex;align-items:center}.show-more-link{color:#1b4ca1;font-family:Lato;font-size:12px;font-style:normal;font-weight:500;line-height:normal;margin:2px 4px 2px 0;text-decoration:none}.show-more-link:hover{text-decoration:underline}.course-content{flex:1;display:flex;flex-direction:column}.course-footer{margin-top:auto;padding-top:12px;display:flex;justify-content:space-between}::ng-deep .mdc-tab__text-label{font-weight:700;font-family:Montserrat!important;font-size:16px;color:#000!important;text-transform:capitalize}.competency-tag{display:inline-block;padding:.25rem .75rem;border-radius:9999px;font-size:.75rem;font-weight:500;margin-right:.5rem;margin-bottom:.5rem}.menu-list-item.active{background-color:#1b4ca1;color:#fff}.menu-list-item:hover{background-color:#1b4ca1;color:#fff;border-color:#1b4ca1}.gray-bg{background-color:rgb(248,250,252,var(--tw-bg-opacity, 1));border-radius:8px;border:1px solid #ccc}.div-count{font-family:Lato;font-size:20px;font-weight:700}.green-bg{background-color:#0080001c;border:1px solid #006400;border-radius:8px}.div-green-count{font-family:Lato;font-size:20px;font-weight:700;color:#006400}.div-green-label{color:#006400;font-family:Montserrat;font-size:16px;font-style:normal;font-weight:600;line-height:normal}.gray-bg-item{background-color:rgb(248,250,252,var(--tw-bg-opacity, 1));border-radius:8px;border:1px solid #ccc;padding:4px 8px;margin:10px}.add-btn{color:#1b4ca1;font-size:14px;font-family:Lato}.list-flex{display:flex;flex-direction:row;justify-content:space-between}.all-category-list .mat-icon{vertical-align:middle}.course-list-container-flex{display:flex;align-items:flex-start;gap:2rem}.left-panel{flex:0 0 250px;font-weight:700;text-align:left}.right-panel{flex:1;display:flex;flex-direction:row;flex-wrap:wrap;gap:20px;align-content:flex-start;justify-content:flex-start}.category-list-item{font-weight:400;font-size:16px;font-family:Lato;padding:5px}.right-panel-flex{display:flex;flex:0 0 100%;width:100%;flex-direction:row;flex-wrap:wrap;gap:20px;justify-content:flex-start}.course-list-container-right-flex{display:flex}.no-gaps-message{display:flex;flex-direction:column;align-items:center;padding:24px;background:linear-gradient(135deg,#4caf501a,#81c7841a);border:2px solid #4CAF50;border-radius:16px;margin:16px 0}.no-gaps-message .success-icon{font-size:25px;color:#4caf50;margin-bottom:16px}.no-gaps-message h4{color:#2e7d32;font-family:Montserrat;font-size:20px;font-weight:600;margin:0 0 8px;text-align:center}.no-gaps-message p{color:#388e3c;font-family:Lato;font-size:14px;font-weight:400;margin:0;text-align:center;line-height:1.5}.section-header-improved,.section-header-improved-sec-2{display:flex;flex-direction:row;gap:0px;width:100%}.search-and-buttons-container{display:flex;justify-content:space-between;gap:5px;flex-direction:column;flex-wrap:wrap;width:37%;margin:0 5px}.search-container-compact{flex:0 0 auto;min-width:180px}.search-container-compact .rsearch{position:relative;display:flex;align-items:center}.search-container-compact .search-icon{position:absolute;left:12px;color:#666;font-size:20px;z-index:1}.search-container-compact .sinput-compact{border-radius:25px;border:2px solid #ccc;padding:20px 12px 20px 40px;height:32px;width:100%;font:400 14px Lato;box-sizing:border-box;transition:border-color .3s ease}.search-container-compact .sinput-compact:focus{outline:none;border-color:#1b4ca1;box-shadow:0 0 0 2px #1b4ca11a}.search-container-compact .sinput-compact::placeholder{color:#999}.btn-group-compact{display:flex;gap:8px;flex:0 0 auto}.category-list-item{padding:8px 12px!important;border-radius:6px;border:1px solid transparent;transition:all .3s ease;background:#f8f9fa;color:#333;font-family:Lato;font-size:14px!important;font-weight:400!important}.category-list-item:hover{background:#1b4ca10d;border-color:#1b4ca133}.category-list-item.selected-theme{background:#1b4ca11a;border-color:#1b4ca1;color:#1b4ca1;font-weight:600!important}.category-list-item.selected-theme:before{content:\"\\2713 \";color:#1b4ca1;font-weight:700}.theme-filter-controls{margin-bottom:8px}.clear-filter-btn{background:#f8f9fa;border:1px solid #ddd;border-radius:20px;padding:4px 12px;font-size:12px;color:#666;cursor:pointer;transition:all .3s ease;display:flex;align-items:center;gap:4px}.clear-filter-btn:hover{background:#e9ecef;border-color:#adb5bd}.clear-filter-btn mat-icon{font-size:16px;width:16px;height:16px}.no-course-found-container{display:flex;justify-content:center;align-items:center;min-height:300px;padding:20px}.no-course-card{background:linear-gradient(135deg,#f8f9fa,#e9ecef);border:2px dashed #ced4da;border-radius:16px;padding:40px;text-align:center;max-width:500px;box-shadow:0 4px 12px #0000000d}.no-course-card .no-course-icon{margin-bottom:20px}.no-course-card .no-course-icon mat-icon{font-size:48px;width:48px;height:48px;color:#6c757d}.no-course-card .no-course-content h4{color:#495057;font-family:Montserrat;font-size:24px;font-weight:600;margin-bottom:12px}.no-course-card .no-course-content p{color:#6c757d;font-family:Lato;font-size:16px;line-height:1.5;margin-bottom:12px}.no-course-card .no-course-content p strong{color:#1b4ca1;font-weight:600}.no-course-card .suggest-course-btn{background:linear-gradient(135deg,#1b4ca1,#164080);color:#fff;border:none;border-radius:25px;padding:12px 24px;font-family:Lato;font-size:14px;font-weight:600;cursor:pointer;display:flex;align-items:center;gap:8px;margin:20px auto 0;transition:all .3s ease;box-shadow:0 4px 12px #1b4ca14d}.no-course-card .suggest-course-btn:hover{background:linear-gradient(135deg,#164080,#0f2f5f);transform:translateY(-2px);box-shadow:0 6px 16px #1b4ca166}.no-course-card .suggest-course-btn:active{transform:translateY(0)}.no-course-card .suggest-course-btn mat-icon{font-size:18px;width:18px;height:18px}.left-panel{flex:0 0 280px!important;background:#f8f9fa;border-radius:12px;padding:16px;border:1px solid #e9ecef;max-height:600px;overflow-y:auto;font-weight:400!important}@media (max-width: 1200px){.search-and-buttons-container{flex-direction:column;align-items:stretch}.search-container-compact{min-width:auto;width:100%}.btn-group-compact{justify-content:center;flex-wrap:wrap}}@media (max-width: 1200px) and (min-width: 769px){.course-list-item{flex:1 1 calc(50% - 10px);min-width:300px;max-width:none}}@media (max-width: 768px){.course-list-container-flex{flex-direction:column}.left-panel{flex:none!important;max-height:300px}.course-list-item{flex:0 0 100%;max-width:100%}.no-course-card{padding:20px}.no-course-card .no-course-content h4{font-size:20px}.no-course-card .no-course-content p{font-size:14px}}::ng-deep .error-snackbar{background-color:#f44336!important;color:#fff!important;font-weight:500}::ng-deep .error-snackbar .mat-simple-snackbar-action{color:#fff!important}::ng-deep .session-expired-snackbar{background-color:#ff9800!important;color:#fff!important;font-weight:600}::ng-deep .session-expired-snackbar .mat-simple-snackbar-action{color:#fff!important;font-weight:700}.download-pdf{justify-content:end;display:flex;flex-direction:row}.gap-analysis-pdf-container{font-family:Lato;max-width:900px;margin:auto;color:#212121}.gap-analysis-pdf-header{text-align:center;margin-bottom:2rem}.gap-analysis-pdf-header .logo{height:60px;margin-bottom:.5rem}.gap-analysis-pdf-header h1{font-size:1.8rem;font-weight:600;margin:.5rem 0}.gap-analysis-pdf-header .subtitle{color:#555;font-size:.95rem}.gap-analysis-pdf-header .download-btn{background:#5e00ff;color:#fff;border:none;padding:10px 20px;border-radius:6px;margin-top:1rem;font-weight:500;cursor:pointer}.competency-snapshot{background:#fafafa;border-radius:12px;padding:2rem;margin-bottom:2rem}.competency-snapshot h2{font-size:1.4rem;font-weight:600;margin-bottom:1.5rem}.competency-snapshot .snapshot-content{display:flex;gap:2rem;align-items:center;justify-content:space-between}.competency-snapshot .snapshot-content .circle-wrapper{position:relative;width:120px;height:120px}.competency-snapshot .snapshot-content .circle-wrapper .progress-ring-bg{fill:none;stroke:#eee;stroke-width:10}.competency-snapshot .snapshot-content .circle-wrapper .progress-ring-circle{fill:none;stroke:#6c63ff;stroke-width:10;stroke-linecap:round;stroke-dasharray:326.72;stroke-dashoffset:117.6192;transform:rotate(-90deg);transform-origin:center;transition:stroke-dashoffset .35s}.competency-snapshot .snapshot-content .circle-wrapper .percentage-text{position:absolute;top:50%;left:50%;transform:translate(-50%,-50%);font-weight:600;font-size:1.4rem;text-align:center}.competency-snapshot .snapshot-content .circle-wrapper .percentage-text span{display:block;font-size:.75rem;font-weight:400;color:#555}.competency-snapshot .snapshot-content .details{flex-grow:1}.competency-snapshot .snapshot-content .details .total-competencies{font-size:1.2rem;margin-bottom:1rem}.competency-snapshot .snapshot-content .details .bar{margin:.5rem 0;padding:.6rem 1rem;border-radius:6px;font-weight:500;display:flex;justify-content:space-between;color:#fff}.competency-snapshot .snapshot-content .details .bar.behavioral{background:#7b1fa2}.competency-snapshot .snapshot-content .details .bar.functional{background:#0288d1}.competency-snapshot .snapshot-content .details .bar.domain{background:#f57c00}.gap-section{margin-top:2rem}.gap-section h2{font-size:1.3rem;margin-bottom:1rem}.gap-section .gap-box{background:#eef1ff;border-left:5px solid #6c63ff;padding:1rem;border-radius:6px;font-weight:500;color:#333}.gap-section .green-bg{background-color:#0080001c;border-left:5px solid #2d9b2d;padding:1rem;border-radius:6px;font-weight:500;color:#006400}.competency-container{width:100%;margin:0 auto;font-family:Segoe UI,sans-serif;color:#212121;display:flex;flex-direction:column}.total-box{background:#f8f9fc;border-radius:10px;padding:1.5rem;text-align:center;border:1px solid #e0e0e0;margin-bottom:1.5rem}.total-box .title{font-size:1rem;color:#666}.total-box .count{font-size:2rem;font-weight:700;color:#111;margin-top:.5rem}.progress-group{margin-bottom:1.5rem}.progress-group .progress-label{display:flex;justify-content:space-between;margin-bottom:6px}.progress-group .progress-label .label-text{font-weight:600}.progress-group .progress-label .covered-text{font-size:.9rem;color:#444}.progress-group .progress-label .behavioral{color:#f8b861}.progress-group .progress-label .functional{color:#e24577}.progress-group .progress-label .domain{color:#7b47a4}.progress-group .progress-bar{height:10px;background-color:#e0e0e0;border-radius:5px;overflow:hidden}.progress-group .progress-bar .progress-fill{height:100%;border-radius:5px;transition:width .5s ease-in-out}.progress-group .progress-bar .progress-fill.behavioral{background:#f8b861}.progress-group .progress-bar .progress-fill.functional{background:#e24577}.progress-group .progress-bar .progress-fill.domain{background:#7b47a4}.competency-card{border-radius:8px;padding:1rem 1.5rem;margin-bottom:1rem;border:1px solid transparent}.competency-card .title{font-weight:600;font-size:1.1rem;color:#222;margin-bottom:.4rem}.competency-card .subtitle{font-size:.95rem;color:#555}.competency-card .subtitle .tag{font-weight:500}.competency-card .subtitle .behavioral,.competency-card .subtitle .domain,.competency-card .subtitle .functional{color:#000}.competency-card.behavioral{background:linear-gradient(0deg,#f8b86152 0% 100%),#fff;border-color:#f8b861}.competency-card.functional{background:linear-gradient(0deg,#e245773d 0% 100%),#fff;border-color:#e24577}.competency-card.domain{background:linear-gradient(0deg,#7b47a43d 0% 100%),#fff;border-color:#7b47a4}.disclaimer-container{margin:8px 16px 0}.disclaimer-message{background-color:#fff3cd;border:1px solid #ffeaa7;border-radius:20px;padding:8px 16px;font-family:Lato;font-size:14px;color:#333;line-height:1.2}.disclaimer-message strong{color:#000;font-weight:600}.select-all-container{display:flex;flex-direction:row;justify-content:space-between}:host ::ng-deep .search-and-buttons-container .mat-form-field-infix{padding:10px!important}:host ::ng-deep .search-and-buttons-container .mat-form-field-wrapper{width:100%!important}:host ::ng-deep .select-search-panel{border:1px solid #ccc!important;border-radius:12px!important;box-shadow:0 8px 20px #0000001f}:host ::ng-deep .mat-select-arrow-wrapper{margin-top:5px!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline{color:#ccc}:host ::ng-deep .mat-form-field-appearance-outline.mat-focused .mat-form-field-outline-thick{color:#1b4ca1;border-color:#1b4ca1!important}:host ::ng-deep .select-search-panel .mat-option.mat-selected:not(.mat-option-disabled){background-color:#1b4ca1!important;color:#fff!important}:host ::ng-deep .mat-form-field-appearance-outline{border-radius:25px;border:1px solid #ccc}:host ::ng-deep .mat-form-field-focused .mat-form-field-appearance-outline{border-color:#1b4ca1!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-start{border-radius:25px 0 0 25px;border:none!important}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-end{border-right:none!important;border:none!important;border-radius:0 25px 25px 0}:host ::ng-deep .mat-form-field-appearance-outline .mat-form-field-outline-gap{border:none!important;border-radius:25px}:host::ng-deep .mat-form-field-appearance-outline .mat-form-field-infix{background-color:transparent!important}:host ::ng-deep .search-and-buttons-container .mat-option-text{color:#fff!important}\n"] }]
|
|
1872
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialogRef }, { type: undefined, decorators: [{
|
|
1873
|
+
type: Inject,
|
|
1874
|
+
args: [MAT_DIALOG_DATA]
|
|
1875
|
+
}] }, { type: i2.SharedService }, { type: i3.MatSnackBar }, { type: i1.MatDialog }, { type: i4.FormBuilder }]; }, propDecorators: { pdfContent: [{
|
|
1876
|
+
type: ViewChild,
|
|
1877
|
+
args: ['pdfContent', { static: false }]
|
|
1878
|
+
}] } });
|
|
1879
|
+
// Confirmation Dialog Component
|
|
1880
|
+
export class RegenerateConfirmationDialog {
|
|
1881
|
+
constructor(dialogRef) {
|
|
1882
|
+
this.dialogRef = dialogRef;
|
|
1883
|
+
}
|
|
1884
|
+
onCancel() {
|
|
1885
|
+
this.dialogRef.close(false);
|
|
1886
|
+
}
|
|
1887
|
+
onConfirm() {
|
|
1888
|
+
this.dialogRef.close(true);
|
|
1889
|
+
}
|
|
1890
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RegenerateConfirmationDialog, deps: [{ token: i1.MatDialogRef }], target: i0.ɵɵFactoryTarget.Component }); }
|
|
1891
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "16.2.12", type: RegenerateConfirmationDialog, selector: "regenerate-confirmation-dialog", ngImport: i0, template: `
|
|
1892
|
+
<div class="confirmation-dialog-container">
|
|
1893
|
+
<div class="dialog-header">
|
|
1894
|
+
<mat-icon class="warning-icon">refresh</mat-icon>
|
|
1895
|
+
<h2 mat-dialog-title>Regenerate Course Recommendations</h2>
|
|
1896
|
+
</div>
|
|
1897
|
+
|
|
1898
|
+
<mat-dialog-content class="dialog-content">
|
|
1899
|
+
<p class="main-message">
|
|
1900
|
+
This will delete all current course recommendations and generate new ones based on the latest role mapping.
|
|
1901
|
+
</p>
|
|
1902
|
+
<p class="sub-message">
|
|
1903
|
+
Are you sure you want to continue?
|
|
1904
|
+
</p>
|
|
1905
|
+
</mat-dialog-content>
|
|
1906
|
+
|
|
1907
|
+
<mat-dialog-actions class="dialog-actions">
|
|
1908
|
+
<button mat-button (click)="onCancel()" class="cancel-btn">
|
|
1909
|
+
Cancel
|
|
1910
|
+
</button>
|
|
1911
|
+
<button mat-raised-button color="primary" (click)="onConfirm()" class="confirm-btn">
|
|
1912
|
+
<mat-icon>refresh</mat-icon>
|
|
1913
|
+
Yes, Regenerate
|
|
1914
|
+
</button>
|
|
1915
|
+
</mat-dialog-actions>
|
|
1916
|
+
</div>
|
|
1917
|
+
`, isInline: true, styles: [".confirmation-dialog-container{padding:0}.dialog-header{display:flex;align-items:center;gap:12px;padding:24px 24px 16px;border-bottom:1px solid #e0e0e0}.warning-icon{font-size:24px;width:24px;height:24px;color:#ff9800}h2{margin:0;font-size:18px;font-weight:600;color:#333}.dialog-content{padding:24px}.main-message{font-size:16px;color:#333;margin:0 0 12px;line-height:1.5}.sub-message{font-size:14px;color:#666;margin:0;font-weight:500}.dialog-actions{padding:16px 24px 24px;gap:12px;justify-content:flex-end}.cancel-btn{color:#666}.confirm-btn{display:flex;align-items:center;gap:8px;.mat-icon{font-size:18px;width:18px;height:18px;margin:0}}\n"], dependencies: [{ kind: "component", type: i8.MatLegacyButton, selector: "button[mat-button], button[mat-raised-button], button[mat-icon-button], button[mat-fab], button[mat-mini-fab], button[mat-stroked-button], button[mat-flat-button]", inputs: ["disabled", "disableRipple", "color"], exportAs: ["matButton"] }, { kind: "component", type: i9.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "directive", type: i1.MatDialogTitle, selector: "[mat-dialog-title], [matDialogTitle]", inputs: ["id"], exportAs: ["matDialogTitle"] }, { kind: "directive", type: i1.MatDialogContent, selector: "[mat-dialog-content], mat-dialog-content, [matDialogContent]" }, { kind: "directive", type: i1.MatDialogActions, selector: "[mat-dialog-actions], mat-dialog-actions, [matDialogActions]", inputs: ["align"] }] }); }
|
|
1918
|
+
}
|
|
1919
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImport: i0, type: RegenerateConfirmationDialog, decorators: [{
|
|
1920
|
+
type: Component,
|
|
1921
|
+
args: [{ selector: 'regenerate-confirmation-dialog', template: `
|
|
1922
|
+
<div class="confirmation-dialog-container">
|
|
1923
|
+
<div class="dialog-header">
|
|
1924
|
+
<mat-icon class="warning-icon">refresh</mat-icon>
|
|
1925
|
+
<h2 mat-dialog-title>Regenerate Course Recommendations</h2>
|
|
1926
|
+
</div>
|
|
1927
|
+
|
|
1928
|
+
<mat-dialog-content class="dialog-content">
|
|
1929
|
+
<p class="main-message">
|
|
1930
|
+
This will delete all current course recommendations and generate new ones based on the latest role mapping.
|
|
1931
|
+
</p>
|
|
1932
|
+
<p class="sub-message">
|
|
1933
|
+
Are you sure you want to continue?
|
|
1934
|
+
</p>
|
|
1935
|
+
</mat-dialog-content>
|
|
1936
|
+
|
|
1937
|
+
<mat-dialog-actions class="dialog-actions">
|
|
1938
|
+
<button mat-button (click)="onCancel()" class="cancel-btn">
|
|
1939
|
+
Cancel
|
|
1940
|
+
</button>
|
|
1941
|
+
<button mat-raised-button color="primary" (click)="onConfirm()" class="confirm-btn">
|
|
1942
|
+
<mat-icon>refresh</mat-icon>
|
|
1943
|
+
Yes, Regenerate
|
|
1944
|
+
</button>
|
|
1945
|
+
</mat-dialog-actions>
|
|
1946
|
+
</div>
|
|
1947
|
+
`, styles: [".confirmation-dialog-container{padding:0}.dialog-header{display:flex;align-items:center;gap:12px;padding:24px 24px 16px;border-bottom:1px solid #e0e0e0}.warning-icon{font-size:24px;width:24px;height:24px;color:#ff9800}h2{margin:0;font-size:18px;font-weight:600;color:#333}.dialog-content{padding:24px}.main-message{font-size:16px;color:#333;margin:0 0 12px;line-height:1.5}.sub-message{font-size:14px;color:#666;margin:0;font-weight:500}.dialog-actions{padding:16px 24px 24px;gap:12px;justify-content:flex-end}.cancel-btn{color:#666}.confirm-btn{display:flex;align-items:center;gap:8px;.mat-icon{font-size:18px;width:18px;height:18px;margin:0}}\n"] }]
|
|
1948
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialogRef }]; } });
|
|
1949
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"generate-course-recommendation.component.js","sourceRoot":"","sources":["../../../../../../../library/sunbird-cb/cbp-ai/src/lib/components/generate-course-recommendation/generate-course-recommendation.component.ts","../../../../../../../library/sunbird-cb/cbp-ai/src/lib/components/generate-course-recommendation/generate-course-recommendation.component.html"],"names":[],"mappings":"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","sourcesContent":["import { MAT_DIALOG_DATA, MatDialog, MatDialogRef } from '@angular/material/dialog';\nimport { ChangeDetectorRef, Component, ElementRef, Inject, OnInit, ViewChild } from '@angular/core';\nimport { FormBuilder, FormGroup, FormArray, Validators, FormControl } from '@angular/forms';\nimport { MatSnackBar } from '@angular/material/snack-bar';\nimport { SuggestMoreCoursesComponent } from '../suggest-more-courses/suggest-more-courses.component';\nimport { MatTabChangeEvent } from '@angular/material/tabs';\nimport { AddCourseComponent } from '../add-course/add-course.component';\nimport html2pdf from 'html2pdf.js';\nimport { interval, of } from 'rxjs';\nimport { switchMap, takeWhile, tap, catchError, finalize } from 'rxjs/operators';\nimport { SharedService } from '../../modules/shared/services/shared.service';\n@Component({\n  selector: 'app-generate-course-recommendation',\n  templateUrl: './generate-course-recommendation.component.html',\n  styleUrls: ['./generate-course-recommendation.component.scss']\n})\nexport class GenerateCourseRecommendationComponent {\n  @ViewChild('pdfContent', { static: false }) pdfContent!: ElementRef;\n  planData: any\n  loading = false\n  dataLoaded = false // Track if initial data has been loaded\n  isRegeneratingWithProgress = false // Track regeneration with progress loading\n  recommended_course_id = ''\n  \n  // Progressive loading for course generation\n  currentProcessingStage: string = '';\n  progressPercentage: number = 0;\n  processingStages = [\n    'Analyzing roles and responsibilities...',\n    'Detailing competency requirements...',\n    'Preparing iGOT course recommendations...',\n    'Generating public course suggestions...',\n    'Computing comprehensive gap analysis...',\n    'Finalizing course recommendation plan...'\n  ];\n  stageStartTime: number = 0;\n  constructor(public dialogRef: MatDialogRef<GenerateCourseRecommendationComponent>,\n    @Inject(MAT_DIALOG_DATA) public data: any, public sharedService: SharedService,\n    private snackBar: MatSnackBar, public dialog: MatDialog,\n    private fb: FormBuilder) {\n    this.planData = data\n  }\n  searchText = ''\n  filterdCourses: any = []\n  originalData: any = []\n  selectFilterCourses: any = []\n  suggestedCourses: any = [] // Track iGOT suggested courses separately\n  userAddedCourses: any = [] // Track user-added courses separately\n  mode = 'add'\n  cbp_plan_id = ''\n  expandedCompetencies: any = {}; // Track expanded state for each course and competency type\n  outerTabActiveIndex = 0\n  innerTabActiveIndex = 0\n  outerTabActiveText = 'all'\n  innerTabActiveText = 'all'\n  selectedCategory = 'all';\n  competencyCoveredCount = 0\n  overallCoverage:any = 0\n  behavioralCompetencyCoveredCount = 0\n  behavioralTotalCompetencies = 0\n  behavioralCoverage: any = 0\n  functionalCompetencyCoveredCount = 0\n  functionalTotalCompetencies = 0\n  functionalCoverage: any = 0\n  domainCompetencyCoveredCount = 0\n  domainTotalCompetencies = 0\n  domainCoverage: any = 0\n  competencyNotMatchedByCategory = []\n  competencyMatchedByCategory = []\n  menuItems = [\n    { key: 'all', label: 'All Categories' },\n    { key: 'behavioral', label: 'Behavioural' },\n    { key: 'functional', label: 'Functional' },\n    { key: 'domain', label: 'Domain' }\n  ];\n  behaviouralNotMatched = []\n  functionalNotMatched = []\n  domainNotMatched = []\n  selectedThemeFilter = ''\n  originalFilteredCourses = []\n  isRegenerating = false\n  isPDFDownload = false\n  behaviouralMatched = []\n  functionalMatched = []\n  domainMatched = []\n  filterForm!: FormGroup;\n  competenciesType= ['All','Behavioural','Functional', 'Domain' ]\n  ratings = ['4.5 and above', '4.0 and above', '3.5 and above','2.5 and above','1 and above'];\n  languages = ['English', 'Hindi', 'Tamil','Kannada', 'Telugu', 'Malayalam', 'Assamese', 'Bengali', 'Gujarati', 'Marathi','Odia', 'Punjabi', 'Konkani', 'Bodo', 'Dogri', 'Kashmiri', 'Maithili','Manipuri', 'Nepali','Sanskrit','Santali','Sindhi','Urdu'];\n  durations = ['< 1 Hour', '1-5 Hours', '5+ Hours'];\n  providers = [];\n  filteredCompetency = [...this.competenciesType];\n  filteredRatings = [...this.ratings];\n  filteredLanguages = [...this.languages];\n  filteredDurations = [...this.durations];\n  filteredProviders = [...this.providers];\n  fullCourseList = []\n   // Selected filters\n  selectedCompetency = 'All';\n  selectedRating: string | null = null;\n  selectedLanguage: string | null = null;\n  selectedDuration: string | null = null;\n  selectedProvider: string | null = null;\n\n  selectCategory(category: string) {\n    this.selectedCategory = category;\n    this.gapAnalysisStats()\n  }\n  ngOnInit() {\n    this.filterForm = this.fb.group({\n      competency: [[]],\n      rating: [[]],\n      language: [[]],\n      duration: [[]],\n      provider: [[]]\n    });\n    this.loadComponentData();\n  }\n\n  applyFilter(value:any) {\n    console.log('this.searchText', value)\n    this.searchText = value\n    this.filterdCourses = this.filterData(this.searchText);\n  }\n\n  searchData() {\n    //  this.filterdCourses = this.filterData(this.searchText);\n  }\n\n  selectedFilterCourses(event, item) {\n    console.log('event', event)\n    console.log('item', item)\n    if (event.checked) {\n      if(this.selectFilterCourses.indexOf(item?.identifier) < 0) {\n        this.selectFilterCourses.push(item?.identifier)\n      }  \n    } else {\n      const index = this.selectFilterCourses.findIndex(\n        control => control === item.identifier\n      );\n\n      if (index !== -1) {\n        this.selectFilterCourses.splice(index, 1);\n      }\n    }\n    console.log('this.selectFilterCourses', this.selectFilterCourses)\n  }\n\n  closeDialog() {\n    this.dialogRef.close()\n  }\n\n  saveCourses() {\n    this.loading = true\n    console.log('this.planData', this.planData)\n    let reqBody = {\n      \"role_mapping_id\": this.planData.id,\n      \"recommended_course_id\": this.recommended_course_id,\n      \"course_identifiers\": this.selectFilterCourses\n    }\n    console.log('reqBody, ', reqBody)\n    if (this.mode === 'add') {\n      this.sharedService.saveCourse(reqBody).subscribe({\n        next: (res) => {\n          // Success handling\n          console.log('Success:', res);\n          this.loading = false\n          this.dialogRef.close('saved')\n          this.snackBar.open('Courses Saved Successfully', 'X', {\n            duration: 3000,\n            panelClass: ['snackbar-success']\n          });\n          //this.successRoleMapping.emit(this.roleMappingForm)\n        },\n        error: (error) => {\n          console.log('error', error)\n          this.dialogRef.close()\n          // Handle 409 Conflict here\n          // alert('Conflict detected: The resource already exists or action conflicts.');\n          //this.get\n          // Or you can set a UI error message variable\n          this.snackBar.open(error?.error?.detail, 'X', {\n            duration: 3000,\n            panelClass: ['snackbar-error']\n          });\n          this.loading = false\n          //this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)\n        }\n      });\n    } else {\n      if (this.cbp_plan_id) {\n        this.sharedService.updateCourse(reqBody, this.cbp_plan_id).subscribe({\n          next: (res) => {\n            // Success handling\n            console.log('Success:', res);\n            this.loading = false\n            this.dialogRef.close('saved')\n            this.snackBar.open('Courses Updated Successfully', 'X', {\n              duration: 3000,\n              panelClass: ['snackbar-success']\n            });\n            //this.successRoleMapping.emit(this.roleMappingForm)\n          },\n          error: (error) => {\n            console.log('error', error)\n            this.dialogRef.close()\n            // Handle 409 Conflict here\n            // alert('Conflict detected: The resource already exists or action conflicts.');\n            //this.get\n            // Or you can set a UI error message variable\n            this.snackBar.open(error?.error?.detail, 'X', {\n              duration: 3000,\n              panelClass: ['snackbar-error']\n            });\n            this.loading = false\n            //this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)\n          }\n        });\n      } else {\n        // Handle case when cbp_plan_id is not available\n        this.loading = false\n        console.error('cbp_plan_id is not available for update operation');\n        this.snackBar.open('Unable to update courses: missing plan ID', 'X', {\n          duration: 3000,\n          panelClass: ['snackbar-error']\n        });\n      }\n\n    }\n\n  }\n\n  getCourses() {\n    let role_mapping_id = this.planData.id\n    this.loading = true\n    this.sharedService.getCourse(role_mapping_id).subscribe({\n      next: (res) => {\n        // Success handling\n        this.loading = false\n        console.log('res', res)\n        this.cbp_plan_id = res?.id\n        if (res && res?.selected_courses && res?.selected_courses?.length) {\n          this.mode = 'edit'\n          for (let i = 0; i < res?.selected_courses.length; i++) {\n            this.selectFilterCourses.push(res.selected_courses[i]?.identifier)\n          }\n        }\n        //this.successRoleMapping.emit(this.roleMappingForm)\n      },\n      error: (error) => {\n        console.log('error', error)\n        this.loading = false\n        // Handle 409 Conflict here\n        // alert('Conflict detected: The resource already exists or action conflicts.');\n        //this.get\n        // Or you can set a UI error message variable\n\n        this.loading = false\n        //this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)\n      }\n\n\n    })\n  }\n  getSuggestedCourse() {\n    let role_mapping_id = this.planData.id\n    this.loading = true\n    this.sharedService.getSuggestedCourses(role_mapping_id).subscribe({\n      next: (res) => {\n        // Success handling\n        this.loading = false\n        console.log('getSuggestedCourses res', res)\n        console.log('this.filterdCourses', this.filterdCourses)\n        // Store suggested courses separately\n        this.suggestedCourses = [...res];\n        \n        // Rebuild filterdCourses to include all course types\n        this.rebuildFilteredCourses();\n        \n        // Update gap analysis stats after suggested courses are added\n        this.updateGapAnalysisAfterCoursesUpdate()\n        //this.successRoleMapping.emit(this.roleMappingForm)\n      },\n      error: (error) => {\n        console.log('error', error)\n        this.loading = false\n        if (error.status === 401) {\n          console.log('Unauthorized access - user will be redirected to login');\n        } else {\n          // Handle other errors gracefully\n          console.error('Failed to load suggested courses');\n        }\n        // Or you can set a UI error message variable\n\n        this.loading = false\n        //this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)\n      }\n\n\n    })\n  }\n\n  getUserCourse() {\n    let role_mapping_id = this.planData.id\n    this.loading = true\n    this.sharedService.getUserCourse(role_mapping_id).subscribe({\n      next: (res) => {\n        // Success handling\n        this.loading = false\n        console.log('getUserCourse API response:', res)\n        console.log('Current filterdCourses before adding user courses:', this.filterdCourses)\n        \n        // Store user-added courses separately (reset to avoid duplicates)\n        this.userAddedCourses = [...res];\n        \n        // Process user-added courses to ensure proper structure\n        for (let i = 0; i < res.length; i++) {\n          const userCourse = res[i];\n          console.log(`User course ${i} competencies:`, userCourse.competencies);\n          \n          // Ensure user-added courses have the expected structure for display\n          if (userCourse && !userCourse.course_type) {\n            userCourse.course_type = 'User Added';\n          }\n          \n          // Add identifier if missing (needed for course selection logic)\n          if (userCourse && !userCourse.identifier) {\n            userCourse.identifier = userCourse.id || `user_course_${i}_${Date.now()}`;\n          }\n        }\n        \n        // Rebuild filterdCourses to include all course types\n        this.rebuildFilteredCourses();\n        \n        console.log('filterdCourses after adding user courses:', this.filterdCourses);\n        //this.successRoleMapping.emit(this.roleMappingForm)\n      },\n      error: (error) => {\n        console.log('error', error)\n        this.loading = false\n        // Handle 409 Conflict here\n        // alert('Conflict detected: The resource already exists or action conflicts.');\n        //this.get\n        // Or you can set a UI error message variable\n\n        this.loading = false\n        //this.alreadyAvailableRoleMapping.emit(this.roleMappingForm)\n      }\n\n\n    })\n  }\n\n  checkIfCourseExists(item) {\n    let flag = false\n    if (this.selectFilterCourses.indexOf(item?.identifier) > -1) {\n      flag = true\n    }\n    return flag\n  }\n\n  selectAllCourses(event) {\n    if (event.checked) {\n      for (let i = 0; i < this.filterdCourses.length; i++) {\n        this.selectFilterCourses.push(this.filterdCourses[i].identifier)\n      }\n    } else {\n      this.selectFilterCourses = []\n      \n    }\n    if(this.cbp_plan_id) {\n      this.mode = 'update'\n    } else {\n      this.mode = 'add'\n    }\n\n\n  }\n  getSectors(sector) {\n    if (sector && sector.length) {\n      return Array.isArray(sector) ? sector.join('/ ') : ''\n    }\n\n  }\n\n  filterData(searchText: string): any[] {\n    const filter = searchText.trim().toLowerCase();\n\n    return this.originalData.filter(item => {\n      const stringified = this.flattenObjectToString(item).toLowerCase();\n      return stringified.includes(filter);\n    });\n  }\n\n  flattenObjectToString(obj: any): string {\n    let result = '';\n\n    for (const key in obj) {\n      const value = obj[key];\n\n      if (typeof value === 'string') {\n        result += ' ' + value;\n      } else if (Array.isArray(value)) {\n        value.forEach(val => {\n          if (typeof val === 'string') {\n            result += ' ' + val;\n          } else if (typeof val === 'object') {\n            result += ' ' + this.flattenObjectToString(val);\n          }\n        });\n      } else if (typeof value === 'object' && value !== null) {\n        result += ' ' + this.flattenObjectToString(value);\n      }\n    }\n\n    return result;\n  }\n\n  suggestMoreCourses() {\n    // this.dialogRef.close()\n    const dialogRefNew = this.dialog.open(SuggestMoreCoursesComponent, {\n      width: '1000px',\n      data: { recommended_course_id: this.recommended_course_id, role_mapping_id: this.planData.id },\n      panelClass: 'view-cbp-plan-popup',\n      minHeight: '400px',          // Set minimum height\n      maxHeight: '90vh',           // Prevent it from going beyond viewport\n      disableClose: true // Optional: prevent closing with outside click\n    });\n\n    dialogRefNew.afterClosed().subscribe(result => {\n      if (result === 'saved') {\n        console.log('Changes saved!');\n        // Refresh data or show a toast here\n        this.getSuggestedCourse()\n      }  else {\n        this.getSuggestedCourse()\n      }\n    });\n  }\n\n  generateCourseRecommendation(element: any) {\n    console.log('Generate Course Recommendation clicked', element);\n    // this.activeRowElement = element\n    console.log('Edit Role Mapping clicked', element);\n    // Navigate or open modal\n    console.log('View CBP Plan clicked', element);\n    const dialogRef = this.dialog.open(GenerateCourseRecommendationComponent, {\n      width: '1000px',\n      data: element,\n       panelClass: 'view-cbp-plan-popup',\n      minHeight: '400px',          // Set minimum height\n      maxHeight: '90vh',           // Prevent it from going beyond viewport\n      disableClose: true // Optional: prevent closing with outside click\n    });\n  \n    dialogRef.afterClosed().subscribe(result => {\n      if (result === 'saved') {\n        console.log('Changes saved!');\n        // Refresh data or show a toast here\n        console.log(this.sharedService.cbpPlanFinalObj)\n        if(this.sharedService.cbpPlanFinalObj && this.sharedService.cbpPlanFinalObj.ministry && this.sharedService.cbpPlanFinalObj.ministry.id) {\n          this.sharedService.getRoleMappingByStateCenter(this.sharedService.cbpPlanFinalObj.ministry.id).subscribe((res)=>{\n            console.log('res', res)\n            // this.dataSource.data = res\n            // this.dataSource.paginator = this.paginator;\n            this.originalData = res;\n          //  console.log('this.dataSource',this.dataSource)\n            })\n        } else {\n          \n        }\n        \n      }\n    });\n  }\n\n  getCompetenciesByType(type: string, index): any[] {\n    const course = this.filterdCourses[index];\n    if (!course) {\n      console.log(`No course found at index ${index}`);\n      return [];\n    }\n    \n    // Handle different competency property names\n    // AI Recommended & Public courses use 'competencies'\n    // Manually Suggested - iGOT courses use 'competencies_v6'\n    // User Added courses use 'competencies'\n    let competencies = [];\n    if (course.competencies && Array.isArray(course.competencies)) {\n      competencies = course.competencies;\n    //  console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) using 'competencies' property:`, competencies);\n    } else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {\n      competencies = course.competencies_v6;\n     // console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) using 'competencies_v6' property:`, competencies);\n    } else {\n      // console.log(`Course ${index} (${course.course_type || course.name || 'Unknown'}) has no valid competencies property:`, {\n      //   hasCompetencies: !!course.competencies,\n      //   competenciesType: typeof course.competencies,\n      //   hasCompetenciesV6: !!course.competencies_v6,\n      //   competenciesV6Type: typeof course.competencies_v6,\n      //   courseKeys: Object.keys(course)\n      // });\n    }\n    \n    if (competencies.length === 0) {\n    //  console.log(`No competencies found for course ${index} and type ${type}`);\n      return [];\n    }\n    \n    // Normalize the type for comparison (case-insensitive + handle spelling variations)\n    const normalizedType = type.toLowerCase().trim();\n    \n    const matchedCompetencies = competencies.filter(c => {\n      if (!c || !c.competencyAreaName) {\n        console.log(`Invalid competency structure in course ${index}:`, c);\n        return false;\n      }\n      \n      const competencyArea = c.competencyAreaName.toLowerCase().trim();\n      \n      // Handle both \"behavioral\" and \"behavioural\" spellings\n      if (normalizedType === 'behavioural' || normalizedType === 'behavioral') {\n        return competencyArea === 'behavioral' || competencyArea === 'behavioural';\n      }\n      \n      // For other types, do case-insensitive comparison\n      return competencyArea === normalizedType;\n    });\n    \n   // console.log(`Found ${matchedCompetencies.length} competencies of type ${type} for course ${index}:`, matchedCompetencies);\n    return matchedCompetencies;\n  }\n\n  getCompetenciesByBehviouralType(index): string {\n    const course = this.filterdCourses[index];\n    if (!course) {\n      return '';\n    }\n    \n    // Handle different competency property names\n    let competencies = [];\n    if (course.competencies && Array.isArray(course.competencies)) {\n      competencies = course.competencies;\n    } else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {\n      competencies = course.competencies_v6;\n    }\n    \n    if (competencies.length === 0) {\n      return '';\n    }\n    \n    return competencies\n      .filter(c => c && (c.competencyAreaName === 'Behavioral' || c.competencyAreaName === 'Behavioural'))\n      .map(c => `${c.competencyThemeName || ''} - ${c.competencySubThemeName || ''}`)\n      .join(', ');\n  }\n\n  getDisplayedCompetencies(type: string, index: number): any[] {\n    const competencies = this.getCompetenciesByType(type, index);\n    const key = `${index}-${type}`;\n\n    if (this.expandedCompetencies[key]) {\n      return competencies;\n    }\n\n    return competencies.slice(0, 2);\n  }\n\n  toggleCompetencies(type: string, index: number): void {\n    const key = `${index}-${type}`;\n    this.expandedCompetencies[key] = !this.expandedCompetencies[key];\n  }\n\n  isExpanded(type: string, index: number): boolean {\n    const key = `${index}-${type}`;\n    return this.expandedCompetencies[key] || false;\n  }\n\n  hasMoreThanTwo(type: string, index: number): boolean {\n    return this.getCompetenciesByType(type, index).length > 2;\n  }\n\n  getRemainingCount(type: string, index: number): number {\n    const totalCount = this.getCompetenciesByType(type, index).length;\n    return totalCount - 2;\n  }\n\n  onOuterTabChange(event: MatTabChangeEvent): void {\n    this.outerTabActiveIndex = event.index\n    this.outerTabActiveText = event.tab.textLabel\n    console.log('Outer Tab Index:', event.index);\n    console.log('Outer Tab Label:', event.tab.textLabel);\n    if(event.index === 1) {\n      this.gapAnalysisStats()\n    }\n    if(event.index === 0) {\n      // this.getCourses()\n      // this.getSuggestedCourse()\n      this.getUserCourse()\n    }\n  }\n\n  onInnerTabChange(event: MatTabChangeEvent): void {\n    this.innerTabActiveIndex = event.index\n    this.innerTabActiveText = event.tab.textLabel\n    this.selectedThemeFilter = ''; // Reset theme filter when switching tabs\n    console.log('Inner Tab Index:', event.index);\n    console.log('Inner Tab Label:', event.tab.textLabel);\n    let tabIndex = event.index\n    this.competencyMatchedByCategory = []\n    \n    // Get all available courses (original + suggested + user added)\n    const allAvailableCourses = this.getAllAvailableCourses();\n\n\n    \n    switch (tabIndex) {\n      case 0: // All\n        this.filterdCourses = allAvailableCourses;\n        break;\n      case 1: // Behavioral\n        this.filterdCourses = this.behavioralFilter(allAvailableCourses);\n        this.competencyMatchedByCategory = this.behavioralCompetencyFilter(this.planData.competencies);\n        break;\n      case 2: // Functional\n        this.filterdCourses = this.functionalFilter(allAvailableCourses);\n        this.competencyMatchedByCategory = this.functionalCompetencyFilter(this.planData.competencies);\n        break;\n      case 3: // Domain\n        this.filterdCourses = this.domainFilter(allAvailableCourses);\n        this.competencyMatchedByCategory = this.domainCompetencyFilter(this.planData.competencies);\n        break;\n    }\n    console.log('this.filterdCourses',this.filterdCourses)\n    console.log('this.competencyMatchedByCategory',this.competencyMatchedByCategory)\n  }\n  \n  getAllAvailableCourses() {\n    // Start with original data (AI recommended courses)\n    const allCourses = [...this.originalData];\n    const seenIdentifiers = new Set();\n    \n    // Track original course identifiers\n    this.originalData.forEach(course => {\n      if (course.identifier) seenIdentifiers.add(course.identifier);\n      if (course.id) seenIdentifiers.add(course.id);\n    });\n    \n    // Add suggested courses (iGOT platform courses)\n    this.suggestedCourses.forEach(course => {\n      const courseId = course.identifier || course.id;\n      if (courseId && !seenIdentifiers.has(courseId)) {\n        allCourses.push(course);\n        seenIdentifiers.add(courseId);\n      }\n    });\n    \n    // Add user-added courses\n    this.userAddedCourses.forEach(course => {\n      const courseId = course.identifier || course.id;\n      if (courseId && !seenIdentifiers.has(courseId)) {\n        // Ensure user-added courses have proper structure\n        if (!course.course_type) {\n          course.course_type = 'User Added';\n        }\n        if (!course.identifier && course.id) {\n          course.identifier = course.id;\n        }\n        allCourses.push(course);\n        seenIdentifiers.add(courseId);\n      }\n    });\n    \n    console.log('getAllAvailableCourses breakdown:', {\n      originalData: this.originalData.length,\n      suggestedCourses: this.suggestedCourses.length,\n      userAddedCourses: this.userAddedCourses.length,\n      totalAvailable: allCourses.length\n    });\n\n    console.log('allCourses--', allCourses)\n    allCourses.forEach((item)=>{\n      if(item && item.organisation && item.organisation.length) {\n        if(this.filteredProviders.indexOf(item.organisation[0]) < 0) {\n          this.filteredProviders.push(item.organisation[0])\n        }\n        \n      }\n    })\n    this.fullCourseList = allCourses\n    let identifiersArr = []\n    this.fullCourseList.map((item)=>{\n      identifiersArr.push(item?.identifier)\n    })\n    this.sharedService.getAdditionalParameterforSuggestedCourses(identifiersArr).subscribe((response)=>{\n      if(response && response.result && response.result.content && response.result.content.length) {\n        for(let i=0; i < response.result.content.length;i++) {\n          for(let j=0; j<this.fullCourseList.length;j++) {\n            if(this.fullCourseList[j]['identifier'] === response.result.content[i]['identifier'] ) {\n              this.fullCourseList[j]['language'] = response.result.content[i]['language']\n              this.fullCourseList[j]['avgRating'] = response.result.content[i]['avgRating']\n              this.fullCourseList[j]['course'] = response.result.content[i]['name']\n              break;\n            }\n          }\n          \n        }\n      }\n      \n    })\n    return allCourses;\n  }\n\n  \n  /**\n   * Rebuild filterdCourses array with all available course types\n   * This ensures consistency across the application\n   */\n  rebuildFilteredCourses() {\n    this.filterdCourses = this.getAllAvailableCourses();\n    \n    console.log('Rebuilt filterdCourses with all course types:', {\n      total: this.filterdCourses.length,\n      breakdown: {\n        original: this.originalData.length,\n        suggested: this.suggestedCourses.length,\n        userAdded: this.userAddedCourses.length\n      }\n    });\n  }\n\n  behavioralFilter(data: any[]): any[] {\n    return data.filter(item => {\n      if (!item) return false;\n      \n      // Handle different competency property names\n      let competencies = [];\n      if (item.competencies && Array.isArray(item.competencies)) {\n        competencies = item.competencies;\n      } else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {\n        competencies = item.competencies_v6;\n      }\n      \n      return competencies.some(c => c && ((c?.competencyAreaName?.toLowerCase() === 'behavioral') || c?.competencyAreaName?.toLowerCase() === 'behavioural'));\n    });\n  }\n\n  behavioralCompetencyFilter(data: any[]): string[] {\n    console.log('data--', data)\n    const behavioralThemes = data\n      .filter(item =>\n        item?.type &&\n        ['behavioral', 'behavioural'].includes(item.type.toLowerCase())\n      )\n      .map(item => `${item.theme}-${item.sub_theme}`)\n      .filter((theme): theme is string => Boolean(theme)); // Ensures type safety\n  \n    const uniqueBehavioralThemes = Array.from(new Set(behavioralThemes));\n    \n    console.log(uniqueBehavioralThemes);\n    return uniqueBehavioralThemes;\n  }\n\n  functionalCompetencyFilter(data:any[]):any {\n    const functionalThemes = data\n      .filter(item =>\n        item?.type &&\n        ['functional'].includes(item.type.toLowerCase())\n      )\n      .map(item => `${item.theme}-${item.sub_theme}`)\n      .filter((theme): theme is string => Boolean(theme)); // Ensures type safety\n  \n    const uniqueFunctionalThemes = Array.from(new Set(functionalThemes));\n    \n    console.log(uniqueFunctionalThemes);\n    return uniqueFunctionalThemes;\n    \n    // console.log(uniqueCompetencyAreaNames);\n  }\n\n  domainCompetencyFilter(data:any[]):any {\n    const domainThemes = data\n    .filter(item =>\n      item?.type &&\n      ['domain'].includes(item.type.toLowerCase())\n    )\n    .map(item => `${item.theme}-${item.sub_theme}`)\n    .filter((theme): theme is string => Boolean(theme)); // Ensures type safety\n\n  const uniqueDomainThemes = Array.from(new Set(domainThemes));\n  \n  console.log(uniqueDomainThemes);\n  return uniqueDomainThemes;\n    \n    // console.log(uniqueCompetencyAreaNames);\n  }\n\n\n  functionalFilter(data: any[]): any[] {\n    return data.filter(item => {\n      if (!item) return false;\n      \n      // Handle different competency property names\n      let competencies = [];\n      if (item.competencies && Array.isArray(item.competencies)) {\n        competencies = item.competencies;\n      } else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {\n        competencies = item.competencies_v6;\n      }\n      \n      return competencies.some(c => c && c?.competencyAreaName?.toLowerCase() === 'functional');\n    });\n  }\n\n  domainFilter(data: any[]): any[] {\n    return data.filter(item => {\n      if (!item) return false;\n      \n      // Handle different competency property names\n      let competencies = [];\n      if (item.competencies && Array.isArray(item.competencies)) {\n        competencies = item.competencies;\n      } else if (item.competencies_v6 && Array.isArray(item.competencies_v6)) {\n        competencies = item.competencies_v6;\n      }\n      \n      return competencies.some(c => c && c?.competencyAreaName?.toLowerCase() === 'domain');\n    });\n  }\n\n  gapAnalysisStats () {\n    console.log('this.planData',this.planData)\n    \n    // Always use ALL competencies for calculating overall stats\n    const allCompetencies = this.planData.competencies;\n    \n    // Filter for category-specific calculations\n    let categorySpecificCompetencies:any = this.planData.competencies;\n    if(this.selectedCategory === 'all') {\n      categorySpecificCompetencies = this.planData.competencies\n    } else if(this.selectedCategory === 'behavioral') {\n      categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);\n    } else if(this.selectedCategory === 'functional') {\n      categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);\n    } \n    else if(this.selectedCategory === 'domain') {\n      categorySpecificCompetencies = this.planData.competencies.filter(item => item.type?.toLowerCase() === this.selectedCategory);\n    } \n\n    // Count ALL competencies by category (not just filtered ones)\n    const masterListByCategory = {total: allCompetencies.length, behavioural: 0, functional:0, domain:0}\n    \n    // IMPORTANT: Use all available courses for consistent coverage calculation\n    // This should match what's available in the Filter By Competency section\n    const allCourseCompetencies = []\n    const allAvailableCoursesForStats = this.getAllAvailableCourses();\n    \n    allAvailableCoursesForStats.forEach(course => {\n      // Handle different competency property names\n      let competencies = [];\n      if (course && course.competencies && Array.isArray(course.competencies)) {\n        competencies = course.competencies;\n      } else if (course && course.competencies_v6 && Array.isArray(course.competencies_v6)) {\n        competencies = course.competencies_v6;\n      }\n      \n      competencies.forEach((list:any) => {\n         allCourseCompetencies.push(list)\n      });\n    });\n    \n    console.log('allCompetencies', allCompetencies)\n    console.log('categorySpecificCompetencies', categorySpecificCompetencies)\n    console.log('allCourseCompetencies', allCourseCompetencies)\n    \n    // Count all competencies by type for proper totals\n    for(let i=0; i<allCompetencies.length;i++) {\n      const competencyType = allCompetencies[i]['type']?.toLowerCase();\n      console.log(`Competency ${i}: type=\"${competencyType}\", data:`, allCompetencies[i]);\n      if(competencyType === 'behavioural' || competencyType === 'behavioral') {\n        masterListByCategory['behavioural'] = masterListByCategory['behavioural'] + 1\n      }\n      if(competencyType === 'functional') {\n        masterListByCategory['functional'] = masterListByCategory['functional'] + 1\n      }\n      if(competencyType === 'domain') {\n        masterListByCategory['domain'] = masterListByCategory['domain'] + 1\n      }\n    }\n    console.log('masterListByCategory after counting:', masterListByCategory)\n    \n    // Get matching results for ALL competencies (for overall coverage)\n    const overallResult = this.getMatchedCompetencyStats(allCompetencies, allCourseCompetencies);\n    \n    // Get matching results for category-specific competencies (for filtered view)\n    const categoryResult = this.getMatchedCompetencyStats(categorySpecificCompetencies, allCourseCompetencies);\n    \n    // Set competencyCoveredCount based on selected category (for the filtered numbers)\n    if(this.selectedCategory === 'all') {\n      this.competencyCoveredCount = overallResult['total'];\n    } else if(this.selectedCategory === 'behavioral') {\n      this.competencyCoveredCount = categoryResult['behavioral'] || 0;\n    } else if(this.selectedCategory === 'functional') {\n      this.competencyCoveredCount = categoryResult['functional'] || 0;\n    } else if(this.selectedCategory === 'domain') {\n      this.competencyCoveredCount = categoryResult['domain'] || 0;\n    }\n    \n    // IMPORTANT: Overall Coverage should ALWAYS be based on ALL competencies, not filtered\n    let totalCompetencies = this.planData.competencies.length; // Always use total count\n    let overallCoveredCount = overallResult['total']; // Always use overall results\n    let mathRound = Math.round((overallCoveredCount/totalCompetencies)*100)\n    this.overallCoverage = `${mathRound}%`\n    \n    console.log('Overall Coverage Calculation:', {\n      totalCompetencies: totalCompetencies,\n      overallCoveredCount: overallCoveredCount,\n      overallCoverage: this.overallCoverage,\n      selectedCategory: this.selectedCategory,\n      competencyCoveredCount: this.competencyCoveredCount\n    });\n    \n    // Always calculate all category-specific metrics regardless of selected category\n    // This ensures the data is available when switching between categories\n    // Use overallResult for consistent calculations across all categories\n    this.behavioralCompetencyCoveredCount = overallResult['behavioral'] || 0;\n    this.behavioralTotalCompetencies = masterListByCategory['behavioural'];\n    let behavioralMathRound = this.behavioralTotalCompetencies > 0 ? Math.round((this.behavioralCompetencyCoveredCount/this.behavioralTotalCompetencies)*100) : 0;\n    this.behavioralCoverage = `${behavioralMathRound}%`;\n    \n    this.functionalCompetencyCoveredCount = overallResult['functional'] || 0;\n    this.functionalTotalCompetencies = masterListByCategory['functional'];\n    let functionalMathRound = this.functionalTotalCompetencies > 0 ? Math.round((this.functionalCompetencyCoveredCount/this.functionalTotalCompetencies)*100) : 0;\n    this.functionalCoverage = `${functionalMathRound}%`;\n    \n    this.domainCompetencyCoveredCount = overallResult['domain'] || 0;\n    this.domainTotalCompetencies = masterListByCategory['domain'];\n    let domainMathRound = this.domainTotalCompetencies > 0 ? Math.round((this.domainCompetencyCoveredCount/this.domainTotalCompetencies)*100) : 0;\n    this.domainCoverage = `${domainMathRound}%`;\n    \n    console.log('All category metrics calculated:', {\n      behavioral: { covered: this.behavioralCompetencyCoveredCount, total: this.behavioralTotalCompetencies, coverage: this.behavioralCoverage },\n      functional: { covered: this.functionalCompetencyCoveredCount, total: this.functionalTotalCompetencies, coverage: this.functionalCoverage },\n      domain: { covered: this.domainCompetencyCoveredCount, total: this.domainTotalCompetencies, coverage: this.domainCoverage },\n      overall: { covered: overallCoveredCount, total: totalCompetencies, coverage: this.overallCoverage },\n      selectedCategory: this.selectedCategory,\n      competencyCoveredCountForCategory: this.competencyCoveredCount\n    });\n    console.log('Overall Result:', overallResult);\n    console.log('Category Result:', categoryResult);\n    console.log('this.competencyNotMatchedByCategory',this.competencyNotMatchedByCategory)\n    this.behaviouralNotMatched = this.getCompetencyByCategoryNotMatching('behavioral')\n    this.functionalNotMatched = this.getCompetencyByCategoryNotMatching('functional')\n    this.domainNotMatched = this.getCompetencyByCategoryNotMatching('domain')\n\n    this.getBehaviouralMatched()\n    this.getFunctionalMatched()\n    this.getDomainMatched()\n  }\n\n  getMatchedCompetencyStats(primaryArray: any[], secondaryArray: any[]) {\n    // Convert both arrays to ensure consistent casing\n    const counts: any = {\n      behavioral: 0,\n      functional: 0,\n      domain: 0,\n      total: 0\n    };\n    \n    // Clear the competencyNotMatchedByCategory array for fresh calculation\n    this.competencyNotMatchedByCategory = [];\n    \n    // Set to keep track of unique matches\n    const seen = new Set<string>();\n    const matchedCompetencies = new Set<string>();\n    \n    for (const primary of primaryArray) {\n      const typeKey = primary.type?.toLowerCase().trim();\n      const themeKey = primary.theme?.toLowerCase().trim();\n      const subThemeKey = primary.sub_theme?.toLowerCase().trim();\n    \n      let isMatched = false;\n      \n      for (const secondary of secondaryArray) {\n        // Normalize secondary keys\n        let secType = secondary?.competencyAreaName?.toLowerCase().trim();\n        let secTheme = secondary?.competencyThemeName?.toLowerCase().trim();\n        let secSubTheme = secondary?.competencySubThemeName?.toLowerCase().trim();\n    \n        // Fix for data inconsistency\n        if (secType === 'behavioural') secType = 'behavioral';\n    \n        // Create unique match key for primary competency\n        const matchKey = `${typeKey}|${themeKey}|${subThemeKey}`;\n    \n        // Enhanced cross-matching logic as requested:\n        // 1. Role mapping Theme vs Course Theme\n        // 2. Role mapping Sub Theme vs Course Sub Theme  \n        // 3. Role mapping Theme vs Course Sub Theme\n        // 4. Role mapping Sub Theme vs Course Theme\n        const themeToThemeMatch = themeKey === secTheme;\n        const subThemeToSubThemeMatch = subThemeKey === secSubTheme;\n        const themeToSubThemeMatch = themeKey === secSubTheme;\n        const subThemeToThemeMatch = subThemeKey === secTheme;\n        \n        let matchType = '';\n        let isMatch = false;\n        \n        if (typeKey === secType && !seen.has(matchKey)) {\n          if (themeToThemeMatch) {\n            matchType = 'Theme-to-Theme';\n            isMatch = true;\n          } else if (subThemeToSubThemeMatch) {\n            matchType = 'SubTheme-to-SubTheme';\n            isMatch = true;\n          } else if (themeToSubThemeMatch) {\n            matchType = 'Theme-to-SubTheme';\n            isMatch = true;\n          } else if (subThemeToThemeMatch) {\n            matchType = 'SubTheme-to-Theme';\n            isMatch = true;\n          }\n        }\n\n        if (isMatch) {\n          console.log(`✅ MATCH FOUND [${matchType}]:`, {\n            roleMapping: { type: typeKey, theme: themeKey, subTheme: subThemeKey },\n            course: { type: secType, theme: secTheme, subTheme: secSubTheme },\n            matchType: matchType\n          });\n          \n          // Store matched competency info for later use\n          let obj = {};\n          obj[typeKey] = [themeKey]; // Store the original theme, not the course theme\n          this.competencyNotMatchedByCategory.push(obj);\n          seen.add(matchKey);\n          matchedCompetencies.add(themeKey); // Track matched themes\n          isMatched = true;\n          \n          if (counts.hasOwnProperty(typeKey)) {\n            counts[typeKey]++;\n          } else {\n            counts[typeKey] = 1;\n          }\n    \n          counts.total++;\n          break; // Stop looking for matches for this primary competency\n        }\n      }\n    }\n    \n    return counts;\n  }\n  getCompetencyByCategoryNotMatching(categoryType) {\n    // Get all competencies for the given category from role mapping\n    let allCategoryCompetencies = [];\n    for(let i = 0; i < this.planData.competencies.length; i++) {\n      if(this.planData.competencies[i]['type']?.toLowerCase() === categoryType?.toLowerCase()) {\n        allCategoryCompetencies.push(this.planData.competencies[i]['theme']?.toLowerCase().trim());\n      }\n    }\n    \n    // Get matched competencies for this category from courses\n    let matchedCompetencies = [];\n    for(let i = 0; i < this.competencyNotMatchedByCategory.length; i++) {\n      if(this.competencyNotMatchedByCategory[i][categoryType?.toLowerCase()]) {\n        // Extract matched themes from course competencies\n        this.competencyNotMatchedByCategory[i][categoryType?.toLowerCase()].forEach(theme => {\n          if(theme) {\n            matchedCompetencies.push(theme.toLowerCase().trim());\n          }\n        });\n      }\n    }\n    \n    // Enhanced matching: check both theme and subtheme from courses against FRAC competencies\n    // IMPORTANT: Use all available courses for consistent coverage calculation\n    const allCoursesForMatching = this.getAllAvailableCourses();\n    \n    allCoursesForMatching.forEach(course => {\n      // Handle different competency property names\n      let competencies = [];\n      if (course && course.competencies && Array.isArray(course.competencies)) {\n        competencies = course.competencies;\n      } else if (course && course.competencies_v6 && Array.isArray(course.competencies_v6)) {\n        competencies = course.competencies_v6;\n      }\n      \n      competencies.forEach((comp: any) => {\n        let secType = comp?.competencyAreaName?.toLowerCase().trim();\n        if(secType === 'behavioural') secType = 'behavioral';\n        \n        if(secType === categoryType?.toLowerCase()) {\n          const courseTheme = comp.competencyThemeName?.toLowerCase().trim();\n          const courseSubTheme = comp.competencySubThemeName?.toLowerCase().trim();\n          \n          // Check if any FRAC competency matches this course competency\n          for(let i = 0; i < this.planData.competencies.length; i++) {\n            const fracComp = this.planData.competencies[i];\n            if(fracComp['type']?.toLowerCase() === categoryType?.toLowerCase()) {\n              const fracTheme = fracComp['theme']?.toLowerCase().trim();\n              const fracSubTheme = fracComp['sub_theme']?.toLowerCase().trim();\n              \n              // Enhanced cross-matching logic (same as in getMatchedCompetencyStats):\n              // 1. Course Theme vs FRAC Theme\n              // 2. Course Sub Theme vs FRAC Sub Theme  \n              // 3. Course Theme vs FRAC Sub Theme\n              // 4. Course Sub Theme vs FRAC Theme\n              const courseThemeToFracTheme = courseTheme === fracTheme;\n              const courseSubThemeToFracSubTheme = courseSubTheme === fracSubTheme;\n              const courseThemeToFracSubTheme = courseTheme === fracSubTheme;\n              const courseSubThemeToFracTheme = courseSubTheme === fracTheme;\n              \n              if(courseThemeToFracTheme || courseSubThemeToFracSubTheme || \n                 courseThemeToFracSubTheme || courseSubThemeToFracTheme) {\n                \n                let matchType = '';\n                if (courseThemeToFracTheme) matchType = 'CourseTheme-to-FRACTheme';\n                else if (courseSubThemeToFracSubTheme) matchType = 'CourseSubTheme-to-FRACSubTheme';\n                else if (courseThemeToFracSubTheme) matchType = 'CourseTheme-to-FRACSubTheme';\n                else if (courseSubThemeToFracTheme) matchType = 'CourseSubTheme-to-FRACTheme';\n                \n                console.log(`✅ COMPETENCY MATCH FOUND [${matchType}] for ${categoryType}:`, {\n                  course: { theme: courseTheme, subTheme: courseSubTheme },\n                  frac: { theme: fracTheme, subTheme: fracSubTheme },\n                  matchType: matchType\n                });\n                \n                matchedCompetencies.push(fracTheme);\n                break; // Found a match, move to next course competency\n              }\n            }\n          }\n        }\n      });\n    });\n    \n    // Remove duplicates from matched competencies\n    matchedCompetencies = [...new Set(matchedCompetencies)];\n    \n    console.log(`\\n=== ${categoryType.toUpperCase()} COMPETENCY MATCHING ===`);\n    console.log(`${categoryType} - All FRAC competencies:`, allCategoryCompetencies);\n    console.log(`${categoryType} - Matched from courses:`, matchedCompetencies);\n    console.log(`${categoryType} - Match rate: ${matchedCompetencies.length}/${allCategoryCompetencies.length} (${Math.round((matchedCompetencies.length/allCategoryCompetencies.length)*100)}%)`);\n    \n    // Return unmatched competencies (those in role mapping but not covered by courses)\n    return this.compareStringArrays(allCategoryCompetencies, matchedCompetencies);\n  }\n\n  compareStringArrays(arr1: string[], arr2: string[]) {\n    console.log('All competencies in category:', arr1);\n    console.log('Matched competencies from courses:', arr2);\n    \n    // Return items in arr1 that are NOT in arr2 (unmatched competencies)\n    const unmatched = arr1.filter(item => !arr2.includes(item));\n    console.log('Unmatched competencies:', unmatched);\n    \n    return unmatched;\n  }\n  \n  addCourse(missingCompetency?: string, competencyType?: string) {\n    let dialogData = { ...this.planData };\n    \n    // If competency parameters are provided, add them to the dialog data\n    if (missingCompetency && competencyType) {\n      // Parse the theme from the missing competency string (assumes format \"theme-subtheme\" or just \"theme\")\n      const theme = missingCompetency.trim();\n      \n      dialogData.prefillCompetency = {\n        type: competencyType,\n        theme: theme\n      };\n    }\n    \n    const dialogRef = this.dialog.open(AddCourseComponent, {\n      width: '1000px',\n      data: dialogData,\n       panelClass: 'view-cbp-plan-popup',\n      minHeight: '400px',          // Set minimum height\n      maxHeight: '90vh',           // Prevent it from going beyond viewport\n      disableClose: true // Optional: prevent closing with outside click\n    });\n  \n    dialogRef.afterClosed().subscribe(result => {\n      if (result === 'saved') {\n        console.log('Course added successfully! Refreshing data...');\n        \n        // Refresh user courses to include the newly added course\n        this.getUserCourse();\n        \n        // Refresh gap analysis with the latest data\n        setTimeout(() => {\n          this.updateGapAnalysisAfterCoursesUpdate();\n          console.log('Gap analysis refreshed after course addition');\n        }, 500); // Small delay to ensure data is loaded\n      }\n    });\n  }\n\n  filterOnCompetencyTheme(themeItem) {\n    this.selectedThemeFilter = themeItem;\n    let themeName = themeItem?.split(\"-\")[0]?.trim();\n    let subThemeName = themeItem?.split(\"-\")[1]?.trim();\n    \n    console.log('Filtering by theme:', themeName, 'subTheme:', subThemeName, 'Tab index:', this.innerTabActiveIndex);\n    \n    // Get the current competency type based on the active tab\n    let competencyType = '';\n    if(this.innerTabActiveIndex === 1) {\n      competencyType = 'behavioral';\n    } else if(this.innerTabActiveIndex === 2) {\n      competencyType = 'functional';\n    } else if(this.innerTabActiveIndex === 3) {\n      competencyType = 'domain';\n    }\n    \n    // Get all available courses using the helper method\n    const allAvailableCourses = this.getAllAvailableCourses();\n    console.log('Total courses to filter from:', allAvailableCourses.length);\n    \n    this.filterdCourses = allAvailableCourses.filter(course => {\n      if (!course) return false;\n      \n      // Handle different competency property names\n      let competencies = [];\n      if (course.competencies && Array.isArray(course.competencies)) {\n        competencies = course.competencies;\n      } else if (course.competencies_v6 && Array.isArray(course.competencies_v6)) {\n        competencies = course.competencies_v6;\n      }\n      \n      if (competencies.length === 0) return false;\n      \n      // Filter by competency type and theme/subtheme\n      return competencies.some(comp => {\n        if (!comp) return false;\n        \n        // Check competency type matches current tab\n        let compType = comp?.competencyAreaName?.toLowerCase();\n        if (compType === 'behavioural') compType = 'behavioral'; // Normalize spelling\n        \n        if (compType !== competencyType) return false;\n        \n        // Check cross-matching like gap analysis: theme vs subtheme and subtheme vs theme\n        const compTheme = comp?.competencyThemeName?.toLowerCase()?.trim();\n        const compSubTheme = comp?.competencySubThemeName?.toLowerCase()?.trim();\n        const searchTheme = themeName?.toLowerCase();\n        const searchSubTheme = subThemeName?.toLowerCase();\n        \n        // Enhanced cross-matching logic like gap analysis:\n        // 1. Selected Theme vs Course Theme\n        // 2. Selected Sub Theme vs Course Sub Theme  \n        // 3. Selected Theme vs Course Sub Theme\n        // 4. Selected Sub Theme vs Course Theme\n        const themeToThemeMatch = searchTheme && compTheme === searchTheme;\n        const subThemeToSubThemeMatch = searchSubTheme && compSubTheme === searchSubTheme;\n        const themeToSubThemeMatch = searchTheme && compSubTheme === searchTheme;\n        const subThemeToThemeMatch = searchSubTheme && compTheme === searchSubTheme;\n        \n        const isMatch = themeToThemeMatch || subThemeToSubThemeMatch || themeToSubThemeMatch || subThemeToThemeMatch;\n        \n        // Log matches for debugging (similar to gap analysis)\n        if (isMatch) {\n          let matchType = '';\n          if (themeToThemeMatch) matchType = 'Theme-to-Theme';\n          else if (subThemeToSubThemeMatch) matchType = 'SubTheme-to-SubTheme';\n          else if (themeToSubThemeMatch) matchType = 'Theme-to-SubTheme';\n          else if (subThemeToThemeMatch) matchType = 'SubTheme-to-Theme';\n          \n          console.log(`✅ FILTER MATCH FOUND [${matchType}]:`, {\n            selected: { theme: searchTheme, subTheme: searchSubTheme },\n            course: { type: compType, theme: compTheme, subTheme: compSubTheme },\n            matchType: matchType\n          });\n        }\n        \n        // Match if any of the cross-matching conditions are met\n        return isMatch;\n      });\n    });\n    \n    console.log('Filtered courses for theme:', themeName, 'Count:', this.filterdCourses.length);\n    console.log('Filtered courses:', this.filterdCourses.map(c => c.name || c.course));\n  }\n  \n  clearThemeFilter() {\n    this.selectedThemeFilter = '';\n    // Reset to category filtered courses by calling the tab change method\n    this.onInnerTabChange({ \n      index: this.innerTabActiveIndex, \n      tab: { textLabel: this.innerTabActiveText } \n    } as any);\n    console.log('Theme filter cleared, showing all courses for category:', this.innerTabActiveText);\n  }\n  \n  isThemeSelected(theme: string): boolean {\n    return this.selectedThemeFilter === theme;\n  }\n\n  /**\n   * Add course for the currently selected theme filter with prefilled competency data\n   */\n  addCourseForSelectedTheme() {\n    if (!this.selectedThemeFilter) {\n      console.warn('No theme selected for adding course');\n      return;\n    }\n\n    // Parse theme and sub-theme from selectedThemeFilter (format: \"Theme - SubTheme\")\n    const themeParts = this.selectedThemeFilter.split('-');\n    const theme = themeParts[0]?.trim();\n    const subTheme = themeParts[1]?.trim();\n\n    // Get the current competency type based on the active tab\n    // Map the tab labels to the format expected by AddCourseComponent\n    let competencyType = '';\n    if (this.innerTabActiveIndex === 1) {\n      competencyType = 'Behavioral'; // Tab label is \"behavioural\"\n    } else if (this.innerTabActiveIndex === 2) {\n      competencyType = 'Functional'; // Tab label is \"functional\"\n    } else if (this.innerTabActiveIndex === 3) {\n      competencyType = 'Domain'; // Tab label is \"domain\"\n    }\n\n    console.log('Tab mapping - innerTabActiveIndex:', this.innerTabActiveIndex, \n                'innerTabActiveText:', this.innerTabActiveText, \n                'mapped competencyType:', competencyType);\n\n    console.log('Adding course for missing competency:', {\n      competencyType,\n      theme,\n      subTheme,\n      selectedThemeFilter: this.selectedThemeFilter\n    });\n\n    // Create dialog data with prefilled competency information\n    let dialogData = { ...this.planData };\n    dialogData.prefillCompetency = {\n      type: competencyType,\n      theme: theme,\n      subTheme: subTheme || '' // Include sub-theme if available\n    };\n\n    const dialogRef = this.dialog.open(AddCourseComponent, {\n      width: '800px',\n      data: dialogData,\n      panelClass: 'view-cbp-plan-popup',\n      minHeight: '400px',\n      maxHeight: '90vh',\n      disableClose: true\n    });\n\n    dialogRef.afterClosed().subscribe(result => {\n      if (result === 'saved') {\n        console.log('Course added successfully for competency! Refreshing data...');\n        \n        // Refresh user courses to include the newly added course\n        this.getUserCourse();\n        \n        // Update gap analysis after course addition\n        this.updateGapAnalysisAfterCoursesUpdate();\n      }\n    });\n  }\n\n  getDuration(time:any) {\n    return this.sharedService.convert(time);\n  }\n\n\n  getBehaviouralMatched() {\n    this.behaviouralMatched = []\n    console.log('this.planData.competencies', this.planData)\n    console.log('this.behaviouralNotMatched', this.behaviouralNotMatched)\n    this.planData.competencies.map((item)=>{\n      if(item && item?.type === 'Behavioral' && this.behaviouralNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {\n        this.behaviouralMatched.push(item)\n      }\n    })\n\n    console.log('this.behaviouralMatched--', this.behaviouralMatched)\n  }\n\n  getFunctionalMatched() {\n    this.functionalMatched = []\n    this.planData.competencies.map((item)=>{\n      if(item && item?.type === 'Functional' && this.functionalNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {\n        this.functionalMatched.push(item)\n      }\n    })\n    return this.functionalMatched\n  }\n\n  getDomainMatched() {\n    this.domainMatched = []\n    this.planData.competencies.map((item)=>{\n      if(item && item?.type === 'Domain' && this.domainNotMatched.indexOf(item?.theme?.toLowerCase()) < 0) {\n        this.domainMatched.push(item)\n      }\n    })\n    console.log('this.domainMatched--', this.domainMatched)\n  }\n\n  downloadPDF() {\n    this.isPDFDownload = true\n    this.loading = true\n    const element = this.pdfContent.nativeElement;\n\n  // Wait for images to load\n  const images = element.querySelectorAll('img');\n  const promises = Array.from(images).map((img: HTMLImageElement) => {\n    if (img.complete) return Promise.resolve();\n    return new Promise(resolve => img.onload = resolve);\n  });\n\n  Promise.all(promises).then(() => {\n    const options = {\n      margin: 0.20,\n      filename: 'Gap Analysis.pdf',\n      image: { type: 'jpeg', quality: 0.98 },\n      html2canvas: {\n        scale: 2,\n        useCORS: true,\n        scrollY: 0,\n      },\n      jsPDF: {\n        unit: 'in',\n        format: 'a4',\n        orientation: 'portrait'\n      },\n      pagebreak: {\n        mode: ['css', 'legacy'],\n        before: '.page-break',\n        avoid: ['.no-break']\n      }\n    };\n\n    html2pdf().from(element).set(options).save()\n    setTimeout(() => {\n      this.isPDFDownload = false\n      this.loading = false;\n    }, 3000); \n  });\n  }\n\n\n  redirectToToc(item) {\n    if(item?.public_link) {\n      window.open(item?.public_link, '_blank')\n    } else {\n      let url = `https://portal.igotkarmayogi.gov.in/app/toc/${item?.identifier}/overview?`\n    window.open(url, '_blank')\n    }\n    \n  }\n\n  /**\n   * Initialize gap analysis stats after initial data load\n   */\n  initializeGapAnalysisStats() {\n    // Set a small timeout to ensure all async operations complete\n    setTimeout(() => {\n      if (this.filterdCourses && this.filterdCourses.length > 0) {\n        this.gapAnalysisStats();\n      }\n    }, 100);\n  }\n\n  /**\n   * Update gap analysis stats after courses are updated (like suggested courses added)\n   */\n  updateGapAnalysisAfterCoursesUpdate() {\n    // Don't modify originalData - it should remain as AI recommended courses only\n    // The gap analysis already properly includes all course types\n    \n    // Recalculate gap analysis stats\n    this.gapAnalysisStats();\n  }\n\n  /**\n   * Regenerate course recommendations by deleting existing recommendations and generating new ones\n   */\n  async regenerateCourseRecommendations() {\n    if (this.isRegenerating) {\n      return; // Prevent multiple simultaneous regenerations\n    }\n\n    try {\n      this.isRegenerating = true;\n      \n      // Show confirmation dialog\n      const confirmed = await this.showConfirmationDialog();\n      \n      if (!confirmed) {\n        this.isRegenerating = false;\n        return;\n      }\n\n      console.log('Starting course recommendation regeneration...');\n      \n      // Set regeneration progress state\n      this.isRegeneratingWithProgress = true;\n      this.dataLoaded = false; // Hide main content during regeneration\n      \n      // Step 1: Delete existing course recommendations\n      await this.deleteCourseRecommendations();\n      \n      // Step 2: Generate new course recommendations\n      await this.generateNewCourseRecommendations();\n      \n      // Step 3: Refresh the current data\n      this.refreshComponent();\n      \n      console.log('Course recommendation regeneration completed successfully');\n      \n      // Show success message\n      this.snackBar.open('Course recommendations regenerated successfully!', 'Close', {\n        duration: 5000,\n        panelClass: ['success-snackbar']\n      });\n      \n    } catch (error) {\n      console.error('Error during course recommendation regeneration:', error);\n      \n      // Show error message\n      this.snackBar.open('Failed to regenerate course recommendations. Please try again.', 'Close', {\n        duration: 5000,\n        panelClass: ['error-snackbar']\n      });\n    } finally {\n      this.isRegenerating = false;\n      this.isRegeneratingWithProgress = false;\n      this.dataLoaded = true; // Ensure main content is shown after regeneration\n    }\n  }\n\n  /**\n   * Show confirmation dialog for regenerate action\n   */\n  private showConfirmationDialog(): Promise<boolean> {\n    return new Promise((resolve) => {\n      // Create a custom confirmation dialog\n      const dialogRef = this.dialog.open(RegenerateConfirmationDialog, {\n        width: '450px',\n        panelClass: 'regenerate-confirmation-dialog',\n        disableClose: true\n      });\n\n      dialogRef.afterClosed().subscribe(result => {\n        resolve(result === true);\n      });\n    });\n  }\n\n  /**\n   * Delete existing course recommendations for the current role mapping\n   */\n  private deleteCourseRecommendations(): Promise<any> {\n    return new Promise((resolve, reject) => {\n      const roleMapId = this.planData?.id;\n      \n      if (!roleMapId) {\n        reject(new Error('Role mapping ID not found'));\n        return;\n      }\n\n      console.log('Deleting course recommendations for role mapping:', roleMapId);\n      \n      this.sharedService.deleteCourseRecommendations(roleMapId).subscribe({\n        next: (response) => {\n          console.log('Course recommendations deleted successfully:', response);\n          resolve(response);\n        },\n        error: (error) => {\n          console.error('Error deleting course recommendations:', error);\n          reject(error);\n        }\n      });\n    });\n  }\n\n  /**\n   * Generate new course recommendations during regeneration\n   */\n  private generateNewCourseRecommendations(): Promise<any> {\n    console.log('Generating new course recommendations...');\n    return this.getRecommendedCourseForRegeneration(this.planData.id);\n  }\n\n  /**\n   * Enhanced course generation specifically for regeneration with progressive loading\n   */\n  private getRecommendedCourseForRegeneration(role_mapping_id: string): Promise<any> {\n    return new Promise((resolve, reject) => {\n      // Start the progressive loading simulation for regeneration\n      this.startRegenerativeProgressiveLoading();\n      \n      // Make the actual API call\n      // this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({\n      //   next: (response) => {\n      //     // Stop progressive loading and show completion\n      //     this.isRegeneratingWithProgress = false; // This will stop the interval\n      //     this.progressPercentage = 100;\n      //     this.currentProcessingStage = 'Course recommendations regenerated successfully!';\n          \n      //     // Clear the loading state after showing completion\n      //     setTimeout(() => {\n      //       this.currentProcessingStage = '';\n      //       this.progressPercentage = 0;\n      //       resolve(response);\n      //     }, 2000);\n      //   },\n      //   error: (error) => {\n      //     // Stop progressive loading\n      //     this.isRegeneratingWithProgress = false;\n      //     this.currentProcessingStage = '';\n      //     this.progressPercentage = 0;\n          \n      //     // Handle specific error cases\n      //     if (error.status === 409 || error.error?.detail?.includes('already exists')) {\n      //       // Course recommendations already exist - this is normal\n      //       console.log('Course recommendations already exist, loading existing data...');\n            \n      //       this.currentProcessingStage = 'Loading existing course recommendations...';\n            \n      //       // Try to get existing data\n      //       this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({\n      //         next: (existingResponse) => {\n      //           console.log('Successfully loaded existing course recommendations');\n      //           this.progressPercentage = 100;\n      //           this.currentProcessingStage = 'Loaded existing course recommendations successfully!';\n                \n      //           setTimeout(() => {\n      //             this.currentProcessingStage = '';\n      //             this.progressPercentage = 0;\n      //             resolve(existingResponse);\n      //           }, 2000);\n      //         },\n      //         error: (loadError) => {\n      //           console.error('Failed to load existing course recommendations:', loadError);\n      //           this.currentProcessingStage = '';\n      //           this.progressPercentage = 0;\n      //           reject(loadError);\n      //         }\n      //       });\n      //     } else {\n      //       console.error('Error during course recommendation regeneration:', error);\n      //       reject(error);\n      //     }\n      //   }\n      // });\n      this.isRegeneratingWithProgress = true;\n      this.progressPercentage = 0;\n      this.currentProcessingStage = 'Generating course recommendations...';\n  \n      interval(5000)\n        .pipe(\n          switchMap(() =>\n            this.sharedService.getRecommendedCourse(role_mapping_id).pipe(\n              catchError(error => {\n                // Pass error through observable so polling logic can inspect it\n                return of({ error });\n              })\n            )\n          ),\n  \n          tap(res => console.log(\"Polling result:\", res)),\n  \n          /**\n           * ⬇️ STOP POLLING only when:\n           * - SUCCESS → res.status === COMPLETED\n           * - ERROR → conflict / already exists\n           */\n          takeWhile(res => {\n            if (res?.status === 'COMPLETED') return false;       // Stop\n            if (res?.error?.status === 409) return false;        // Stop\n            if (res?.error?.error?.detail?.includes('already exists')) return false; // Stop\n            return true; // Keep polling\n          }, true) // include final emission\n        )\n        .subscribe({\n          next: (res) => {\n            /** -------------------------------\n             *  SUCCESS CASE (status COMPLETED)\n             -------------------------------- */\n            if (res?.status === 'COMPLETED') {\n              this.isRegeneratingWithProgress = false;\n              this.progressPercentage = 100;\n              this.currentProcessingStage = 'Course recommendations regenerated successfully!';\n  \n              setTimeout(() => {\n                this.currentProcessingStage = '';\n                this.progressPercentage = 0;\n                resolve(res);\n              }, 2000);\n  \n              return;\n            }\n  \n            /** -----------------------------------\n             *  ERROR CASE (409 / already exists)\n             ------------------------------------ */\n            if (res?.error) {\n              const error = res.error;\n  \n              if (error.status === 409 || error?.error?.detail?.includes('already exists')) {\n  \n                this.currentProcessingStage = 'Loading existing course recommendations...';\n  \n                this.sharedService.getRecommendedCourse(role_mapping_id).subscribe({\n                  next: (existingData) => {\n                    this.progressPercentage = 100;\n                    this.currentProcessingStage = 'Loaded existing course recommendations successfully!';\n  \n                    setTimeout(() => {\n                      this.currentProcessingStage = '';\n                      this.progressPercentage = 0;\n                      resolve(existingData);\n                    }, 2000);\n                  },\n                  error: loadError => reject(loadError)\n                });\n  \n                return;\n              }\n  \n              /** Any other error → stop & reject */\n              this.isRegeneratingWithProgress = false;\n              this.currentProcessingStage = '';\n              this.progressPercentage = 0;\n              reject(error);\n              return;\n            }\n  \n            /** ------------------------------\n             *  STILL PROCESSING (keep polling)\n             ------------------------------- */\n            this.progressPercentage = Math.min(this.progressPercentage + 10, 95);\n            this.currentProcessingStage = 'Still processing course recommendations...';\n          },\n  \n          error: err => reject(err)\n        });\n    });\n  }\n\n \n\n  // Progressive loading methods for regeneration\n  startRegenerativeProgressiveLoading() {\n    this.currentProcessingStage = this.processingStages[0];\n    this.progressPercentage = 0;\n    this.stageStartTime = Date.now();\n    this.simulateRegenerativeProgressiveStages();\n  }\n\n  simulateRegenerativeProgressiveStages() {\n    let currentStageIndex = 0;\n    const totalDuration = 60000; // 60 seconds estimated duration\n    const stageInterval = totalDuration / this.processingStages.length;\n    \n    const progressInterval = setInterval(() => {\n      // Stop if regeneration is complete\n      if (!this.isRegeneratingWithProgress) {\n        clearInterval(progressInterval);\n        return;\n      }\n      \n      const elapsed = Date.now() - this.stageStartTime;\n      const currentStageProgress = (elapsed % stageInterval) / stageInterval;\n      const stageIndex = Math.min(Math.floor(elapsed / stageInterval), this.processingStages.length - 1);\n      \n      // Update stage if needed\n      if (stageIndex !== currentStageIndex && stageIndex < this.processingStages.length) {\n        currentStageIndex = stageIndex;\n        this.currentProcessingStage = this.processingStages[stageIndex];\n      }\n      \n      // Calculate overall progress (each stage is worth ~16.67%)\n      const baseProgress = (stageIndex / this.processingStages.length) * 100;\n      const stageProgress = (currentStageProgress / this.processingStages.length) * 100;\n      this.progressPercentage = Math.min(Math.round(baseProgress + stageProgress), 95);\n    }, 1000);\n  }\n\n  /**\n   * Refresh the component data after regeneration\n   */\n  private refreshComponent() {\n    console.log('Refreshing component data...');\n    \n    // Reset all filters and selections\n    this.selectedThemeFilter = '';\n    this.selectFilterCourses = [];\n    this.filterdCourses = [];\n    this.originalFilteredCourses = [];\n    this.suggestedCourses = [];\n    this.userAddedCourses = [];\n    \n    // Reset tab states\n    this.innerTabActiveIndex = 0;\n    this.outerTabActiveIndex = 0;\n    \n    // Call the data loading methods directly to avoid resetting dataLoaded\n    this.loadComponentData();\n  }\n\n  /**\n   * Load component data (separated from ngOnInit for reuse)\n   */\n  private loadComponentData() {\n    this.getRecommendedCourseWithProgress(this.planData.id).then((res) => {\n      console.log('res', res)\n      this.recommended_course_id = res.id\n      let allCoures = []\n      if (res && res.filtered_courses && res.filtered_courses.length) {\n        res?.filtered_courses.forEach((item) => {\n          if (item?.relevancy >= 85) {\n            allCoures.push(item)\n          }\n        })\n        this.originalData = allCoures\n        this.filterdCourses = allCoures\n      }\n\n      console.log('this.filterdCourses', this.filterdCourses)\n      this.getCourses()\n      this.getSuggestedCourse()\n      \n      // Initialize gap analysis stats after courses are loaded\n      this.initializeGapAnalysisStats()\n      this.getUserCourse()\n      \n      // Mark data as loaded after all initial operations\n      this.dataLoaded = true;\n    }).catch((error) => {\n      console.error('Error getting recommended courses:', error);\n      if (error.status === 401) {\n        console.log('Unauthorized access - user will be redirected to login');\n      } else if (error.status !== 409) {\n        // Don't log error for 409 (already exists) as it's handled in the method\n        console.error('Failed to load course recommendations');\n      }\n      // Mark data as loaded even if there's an error to show the dialog content\n      this.dataLoaded = true;\n    });\n  }\n\n  // Progressive loading methods for course generation\n  startProgressiveLoading() {\n    this.currentProcessingStage = this.processingStages[0];\n    this.progressPercentage = 0;\n    this.stageStartTime = Date.now();\n    this.simulateProgressiveStages();\n  }\n\n  simulateProgressiveStages() {\n    let currentStageIndex = 0;\n    const totalDuration = 60000; // 60 seconds estimated duration\n    const stageInterval = totalDuration / this.processingStages.length;\n    \n    const progressInterval = setInterval(() => {\n      const elapsed = Date.now() - this.stageStartTime;\n      const currentStageProgress = (elapsed % stageInterval) / stageInterval;\n      const stageIndex = Math.min(Math.floor(elapsed / stageInterval), this.processingStages.length - 1);\n      \n      // Update stage if needed\n      if (stageIndex !== currentStageIndex && stageIndex < this.processingStages.length) {\n        currentStageIndex = stageIndex;\n        this.currentProcessingStage = this.processingStages[stageIndex];\n      }\n      \n      // Calculate overall progress (each stage is worth ~16.67%)\n      const baseProgress = (stageIndex / this.processingStages.length) * 100;\n      const stageProgress = (currentStageProgress / this.processingStages.length) * 100;\n      this.progressPercentage = Math.min(Math.round(baseProgress + stageProgress), 95);\n      \n      // Clear interval when loading stops\n      if (!this.loading) {\n        clearInterval(progressInterval);\n        this.progressPercentage = 100;\n        this.currentProcessingStage = 'Course recommendations generated successfully!';\n      }\n    }, 1000);\n  }\n\n  // Enhanced course generation with progressive loading\n  getRecommendedCourseWithProgress(role_mapping_id: string): Promise<any> {\n    return new Promise((resolve, reject) => {\n      this.loading = true;\n      this.currentProcessingStage = 'Generating course recommendations...';\n      this.progressPercentage = 30;\n      this.startProgressiveLoading();\n      \n      interval(5000) // ⏱ poll every 5 seconds\n      .pipe(\n        tap(() => {\n          this.currentProcessingStage = 'Processing...';\n        }),\n        switchMap(() => this.sharedService.getRecommendedCourse(role_mapping_id)),\n\n        // Continue polling while status !== COMPLETED\n        takeWhile((response: any) => response?.status !== 'COMPLETED', true),\n\n        finalize(() => {\n          this.loading = false;\n        })\n      )\n      .subscribe({\n        next: (response: any) => {\n          if (response?.status === 'COMPLETED') {\n\n            this.progressPercentage = 100;\n            this.currentProcessingStage = 'Course recommendations generated successfully!';\n\n            setTimeout(() => {\n              this.currentProcessingStage = '';\n              this.progressPercentage = 0;\n            }, 2000);\n\n            this.snackBar.open('Recommendations ready!', 'X', {\n              duration: 3000,\n              panelClass: ['snackbar-success']\n            });\n\n            resolve(response);\n          }\n        },\n\n        error: (error) => {\n          this.loading = false;\n          this.currentProcessingStage = '';\n          this.progressPercentage = 0;\n\n          if (error.status === 409 || error.error?.detail?.includes('already exists')) {\n            this.snackBar.open('Course recommendations already exist.', 'X', {\n              duration: 3000,\n              panelClass: ['snackbar-success']\n            });\n\n            resolve(null); // not a real error\n          } else {\n            this.snackBar.open('Failed to generate course recommendations.', 'X', {\n              duration: 3000,\n              panelClass: ['snackbar-error']\n            });\n            reject(error);\n          }\n        }\n      });\n    });\n  }\n\n  resetFilters(): void {\n    this.filterForm.reset({\n      competency: [],\n      rating: [],\n      language: [],\n      duration: [],\n      provider: []\n    });\n  \n    // Reset filtered dropdown lists\n    this.filteredCompetency = [...this.competenciesType];\n    this.filteredRatings = [...this.ratings];\n    this.filteredLanguages = [...this.languages];\n    this.filteredDurations = [...this.durations];\n    this.filteredProviders = [...this.providers];\n  \n    // Reset course list\n    this.filterdCourses = [...this.fullCourseList];\n    // this.fullCourseList.forEach((item)=>{\n    //   if(item && item.organisation && item.organisation.length) {\n    //     if(this.filteredProviders.indexOf(item.organisation[0]) < 0) {\n    //       this.filteredProviders.push(item.organisation[0])\n    //     }\n        \n    //   }\n    // })\n    this.getAllAvailableCourses()\n  }\n\n  filterList(value: string, type: string) {\n    const search = value.toLowerCase();\n\n    switch (type) {\n      case 'competency':\n        this.filteredCompetency = this.competenciesType.filter(v => v.toLowerCase().includes(search));\n        break;\n      case 'rating':\n        this.filteredRatings = this.ratings.filter(v => v.toLowerCase().includes(search));\n        break;\n      case 'language':\n        this.filteredLanguages = this.languages.filter(v => v.toLowerCase().includes(search));\n        break;\n      case 'duration':\n        this.filteredDurations = this.durations.filter(v => v.toLowerCase().includes(search));\n        break;\n      case 'provider':\n        this.filteredProviders = this.providers.filter(v => v.toLowerCase().includes(search));\n        break;\n    }\n  }\n\n  normalizeCompetency(value: string): string {\n    if (!value) return '';\n    const v = value.toLowerCase();\n  \n    if (v === 'all') return 'all';\n    if (v.includes('behavioural') || v.includes('behavioral')) return 'behavioral';\n    if (v.includes('functional')) return 'functional';\n    if (v.includes('domain')) return 'domain';\n  \n    return '';\n  }\n\n  applyFilters(): void {\n    const {\n      competency = [],\n      rating = [],\n      language = [],\n      duration = [],\n      provider = []\n    } = this.filterForm.value;\n\n    console.log('this.fullCourseList--',this.fullCourseList)\n  \n    this.filterdCourses = this.fullCourseList.filter(course => {\n  \n      /* ---------------- Competency ---------------- */\n      const courseCompetencies = [\n        ...(course.competencies || []),\n        ...(course.competencies_v6 || [])\n      ];\n  \n      const selectedCompetencies = competency\n        .map((c: string) => this.normalizeCompetency(c))\n        .filter(c => c !== '');\n\n      const competencyMatch =\n        selectedCompetencies.length === 0 ||\n        selectedCompetencies.includes('all') ||\n        courseCompetencies.some((c: any) =>\n          selectedCompetencies.includes(\n            this.normalizeCompetency(c.competencyAreaName)\n          )\n        );\n  \n      /* ---------------- Rating ---------------- */\n      const ratingMatch =\n        rating.length === 0 ||\n        this.matchRating(course, rating);\n  \n      /* ---------------- Language ---------------- */\n      const languageMatch =\n        language.length === 0 ||\n        language.some((l: string) =>\n          course.language === l ||\n          course.language?.includes?.(l) ||\n          course.language?.[0]?.includes?.(l)\n        );\n  \n      /* ---------------- Duration ---------------- */\n      const durationMatch =\n        duration.length === 0 ||\n        this.matchDuration(course.duration, duration);\n  \n      /* ---------------- Provider ---------------- */\n      const providerMatch =\n        provider.length === 0 ||\n        provider.some((p: string) =>\n          course.organisation?.includes(p)\n        );\n  \n      return (\n        competencyMatch &&\n        ratingMatch &&\n        languageMatch &&\n        durationMatch &&\n        providerMatch\n      );\n    });\n  }\n\n  matchRating(course: any, selectedRatings: string[]): boolean {\n    if (!selectedRatings || selectedRatings.length === 0) {\n      return true; // no filter selected → match everything\n    }\n  \n    let rating = null;\n  \n    // 1️⃣ Use avgRating if available\n    if (course.avgRating != null && !isNaN(course.avgRating)) {\n      rating = Number(course.avgRating);\n    } \n    // 2️⃣ Map relevancy if avgRating missing\n    else if (course.relevancy != null && !isNaN(course.relevancy)) {\n      const rel = Number(course.relevancy);\n      if (rel >= 90) rating = 4.5;\n      else if (rel >= 80) rating = 4.0;\n      else if (rel >= 70) rating = 3.5;\n      else if (rel >= 50) rating = 2.5;\n      else rating = 1.0;\n    }\n  \n    // 3️⃣ No usable rating → exclude\n    if (rating === null) return false;\n  \n    // 4️⃣ Match selected rating ranges\n    return selectedRatings.some(r => {\n      switch (r) {\n        case '4.5 and above': return rating >= 4.5;\n        case '4.0 and above': return rating >= 4.0;\n        case '3.5 and above': return rating >= 3.5;\n        case '2.5 and above': return rating >= 2.5;\n        case '1 and above': return rating >= 1.0;\n        default: return false;\n      }\n    });\n  }\n  \n\n  matchDuration(durationInSeconds: number | string, selectedRanges: string[]): boolean {\n    const hours = Number(durationInSeconds) / 3600;\n  \n    return selectedRanges.some(range => {\n      switch (range) {\n        case '< 1 Hour':\n          return hours < 1;\n        case '1-5 Hours':\n          return hours >= 1 && hours <= 5;\n        case '5+ Hours':\n          return hours > 5;\n        default:\n          return true;\n      }\n    });\n  }\n}\n\n// Confirmation Dialog Component\n@Component({\n  selector: 'regenerate-confirmation-dialog',\n  template: `\n    <div class=\"confirmation-dialog-container\">\n      <div class=\"dialog-header\">\n        <mat-icon class=\"warning-icon\">refresh</mat-icon>\n        <h2 mat-dialog-title>Regenerate Course Recommendations</h2>\n      </div>\n      \n      <mat-dialog-content class=\"dialog-content\">\n        <p class=\"main-message\">\n          This will delete all current course recommendations and generate new ones based on the latest role mapping.\n        </p>\n        <p class=\"sub-message\">\n          Are you sure you want to continue?\n        </p>\n      </mat-dialog-content>\n      \n      <mat-dialog-actions class=\"dialog-actions\">\n        <button mat-button (click)=\"onCancel()\" class=\"cancel-btn\">\n          Cancel\n        </button>\n        <button mat-raised-button color=\"primary\" (click)=\"onConfirm()\" class=\"confirm-btn\">\n          <mat-icon>refresh</mat-icon>\n          Yes, Regenerate\n        </button>\n      </mat-dialog-actions>\n    </div>\n  `,\n  styles: [`\n    .confirmation-dialog-container {\n      padding: 0;\n    }\n    \n    .dialog-header {\n      display: flex;\n      align-items: center;\n      gap: 12px;\n      padding: 24px 24px 16px 24px;\n      border-bottom: 1px solid #e0e0e0;\n    }\n    \n    .warning-icon {\n      font-size: 24px;\n      width: 24px;\n      height: 24px;\n      color: #ff9800;\n    }\n    \n    h2 {\n      margin: 0;\n      font-size: 18px;\n      font-weight: 600;\n      color: #333;\n    }\n    \n    .dialog-content {\n      padding: 24px;\n    }\n    \n    .main-message {\n      font-size: 16px;\n      color: #333;\n      margin: 0 0 12px 0;\n      line-height: 1.5;\n    }\n    \n    .sub-message {\n      font-size: 14px;\n      color: #666;\n      margin: 0;\n      font-weight: 500;\n    }\n    \n    .dialog-actions {\n      padding: 16px 24px 24px 24px;\n      gap: 12px;\n      justify-content: flex-end;\n    }\n    \n    .cancel-btn {\n      color: #666;\n    }\n    \n    .confirm-btn {\n      display: flex;\n      align-items: center;\n      gap: 8px;\n      \n      .mat-icon {\n        font-size: 18px;\n        width: 18px;\n        height: 18px;\n        margin: 0;\n      }\n    }\n  `]\n})\nexport class RegenerateConfirmationDialog {\n  constructor(\n    public dialogRef: MatDialogRef<RegenerateConfirmationDialog>\n  ) {}\n\n  onCancel(): void {\n    this.dialogRef.close(false);\n  }\n\n  onConfirm(): void {\n    this.dialogRef.close(true);\n  }\n\n \n}\n","<div class=\"view-cbp-plan\">\n    <div class=\"popup-container\">\n        \n        <!-- Initial Loading Screen (for initial load and regeneration) -->\n        <div class=\"initial-loading-container\" *ngIf=\"!dataLoaded || isRegeneratingWithProgress\">\n            <div class=\"loading-content\">\n                <mat-spinner diameter=\"50\"></mat-spinner>\n                <div class=\"loading-text mt-3\">\n                    <p class=\"main-message\">{{ currentProcessingStage || (isRegeneratingWithProgress ? 'Regenerating course recommendations...' : 'Loading course recommendations...') }}</p>\n                    <div class=\"progress-details\" *ngIf=\"progressPercentage > 0\">\n                        <div class=\"progress-indicator\">\n                            <div class=\"progress-bar-container\">\n                                <div class=\"progress-bar\" \n                                     [style.width.%]=\"progressPercentage\">\n                                </div>\n                            </div>\n                            <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n                        </div>\n                        <small class=\"time-estimate\">\n                            Real-time course generation in progress. This may take 1-2 minutes.\n                        </small>\n                    </div>\n                </div>\n            </div>\n        </div>\n\n        <!-- Main Content -->\n        <div class=\"main-content\" *ngIf=\"dataLoaded && !isRegeneratingWithProgress\">\n        <div class=\"popup-header\">\n            <div class=\"header-left\">\n                <div class=\"heading\">Save Courses (<span>{{sharedService?.cbpPlanFinalObj?.ministry?.orgName}} <span\n                            *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span></span>)</div>\n            </div>\n            <div class=\"header-actions\">\n                <button class=\"regenerate-btn\" (click)=\"regenerateCourseRecommendations()\"\n                        [disabled]=\"isRegenerating\"\n                        mat-raised-button color=\"primary\">\n                    <mat-icon *ngIf=\"!isRegenerating\">refresh</mat-icon>\n                    <mat-spinner *ngIf=\"isRegenerating\" diameter=\"20\"></mat-spinner>\n                    <span>{{isRegenerating ? 'Regenerating...' : 'Regenerate Recommendations'}}</span>\n                </button>\n                <div class=\"cursor-pointer close-btn\" (click)=\"closeDialog()\">\n                    <mat-icon>close</mat-icon>\n                </div>\n            </div>\n        </div>\n        <div class=\"section-container mt-4\">\n            <mat-tab-group (selectedTabChange)=\"onOuterTabChange($event)\">\n                <!-- First Main Tab -->\n                <mat-tab label=\"course recommendation\">\n                    <!-- Nested Tabs -->\n                    <!-- <mat-tab-group (selectedTabChange)=\"onInnerTabChange($event)\">\n                        <mat-tab label=\"all\">\n\n                        </mat-tab>\n                        <mat-tab label=\"behavioural\">\n\n                        </mat-tab>\n                        <mat-tab label=\"functional\">\n\n                        </mat-tab>\n                        <mat-tab label=\"domain\">\n\n                        </mat-tab>\n                    </mat-tab-group> -->\n                </mat-tab>\n\n                <!-- Second Main Tab -->\n                <mat-tab label=\"gap analysis\">\n                    <!-- <p>Gap analysis content goes here</p> -->\n                </mat-tab>\n            </mat-tab-group>\n            <form [formGroup]=\"filterForm\" class=\"filters-container\">\n            <div class=\"role-mapping-container\"\n                *ngIf=\"outerTabActiveIndex === 0 && (innerTabActiveIndex === 0 || innerTabActiveIndex === 1 || innerTabActiveIndex === 2 || innerTabActiveIndex === 3 )\">\n                <div class=\"container\">\n                    <div class=\"section-header-improved mt-3\">\n                        <div class=\"search-and-buttons-container\">\n                            <div>\n                                <label>Search Courses</label>\n                            </div>                            \n                            <div class=\"search-container-compact\">\n                                <div class=\"rsearch\">\n                                    <mat-icon class=\"search-icon\">search</mat-icon>\n                                    <input class=\"sinput-compact\" [(ngModel)]=\"searchText\"\n                                           (input)=\"applyFilter($event.target.value);\"\n                                           placeholder=\"Search Courses\" type=\"text\">\n                                </div>\n                            </div>\n                            \n                        </div>\n                        \n\n                        <div class=\"search-and-buttons-container\">\n                            <div style=\"margin-left: 10px;\">\n                                <label>Select Competency</label>\n                            </div>   \n                            <div>\n                                <mat-form-field appearance=\"outline\">\n                                    <mat-label>\n                                        {{ filterForm.value.competency?.length ? '' : 'Competency' }}\n                                      </mat-label>\n                                  \n                                    <mat-select\n                                      formControlName=\"competency\"\n                                      multiple\n                                      panelClass=\"select-search-panel\"\n                                      (selectionChange)=\"applyFilters()\">\n                                        \n                                      <!-- Search box -->\n                                        <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n                                          matInput\n                                          placeholder=\"Search Competency\"\n                                          (click)=\"$event.stopPropagation()\"\n                                          (keydown)=\"$event.stopPropagation()\"\n                                          (keyup)=\"filterList($any($event.target).value, 'competency')\" />\n                                  \n                                      <!-- Values -->\n                                      <mat-option\n                                        *ngFor=\"let r of filteredCompetency\"\n                                        [value]=\"r\">\n                                        {{ r }}\n                                      </mat-option>\n                                  \n                                    </mat-select>\n                                  </mat-form-field>\n                            </div>                       \n                            \n                        </div>\n                        <div class=\"search-and-buttons-container\">\n                            <div style=\"margin-left: 10px;\">\n                                <label>Select Rating</label>\n                            </div>                            \n                            <div>\n                                <mat-form-field appearance=\"outline\">\n                                    <mat-label>\n                                        {{ filterForm.value.rating?.length ? '' : 'Rating' }}\n                                      </mat-label>\n                                  \n                                    <mat-select\n                                      formControlName=\"rating\"\n                                      multiple\n                                      panelClass=\"select-search-panel\"\n                                      (selectionChange)=\"applyFilters()\">\n                                  \n                                      <!-- Search box -->\n                                        <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n                                          matInput\n                                          placeholder=\"Search Reatings\"\n                                          (click)=\"$event.stopPropagation()\"\n                                          (keydown)=\"$event.stopPropagation()\"\n                                          (keyup)=\"filterList($any($event.target).value, 'rating')\" />\n                                  \n                                      <!-- Values -->\n                                      <mat-option\n                                        *ngFor=\"let r of filteredRatings\"\n                                        [value]=\"r\">\n                                        {{ r }}\n                                      </mat-option>\n                                  \n                                    </mat-select>\n                                  </mat-form-field>\n                            </div>\n                        </div>\n                       \n                      \n                        \n                        \n                    </div>\n                    <div class=\"section-header-improved-sec-2 mt-4\">\n                        <div class=\"search-and-buttons-container\">\n                            <div style=\"margin-left: 10px;\">\n                                <label>Select Language</label>\n                            </div>                            \n                            <div>\n                                <mat-form-field appearance=\"outline\">\n                                    <mat-label>\n                                        {{ filterForm.value.language?.length ? '' : 'Language' }}\n                                      </mat-label>\n                                  \n                                    <mat-select\n                                      formControlName=\"language\"\n                                      multiple\n                                      panelClass=\"select-search-panel\"\n                                      (selectionChange)=\"applyFilters()\">\n                                  \n                                      <!-- Search box -->\n                                        <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n                                          matInput\n                                          placeholder=\"Search Languages\"\n                                          (click)=\"$event.stopPropagation()\"\n                                          (keydown)=\"$event.stopPropagation()\"\n                                          (keyup)=\"filterList($any($event.target).value, 'language')\" />\n                                  \n                                      <!-- Values -->\n                                      <mat-option\n                                        *ngFor=\"let r of filteredLanguages\"\n                                        [value]=\"r\">\n                                        {{ r }}\n                                      </mat-option>\n                                  \n                                    </mat-select>\n                                  </mat-form-field>\n                            </div>\n                        </div>\n                        <div class=\"search-and-buttons-container\">\n                            <div style=\"margin-left: 10px;\">\n                                <label>Select Duration</label>\n                            </div>                            \n                            <div>\n                                <mat-form-field appearance=\"outline\">\n                                    <mat-label>\n                                        {{ filterForm.value.duration?.length ? '' : 'Duration' }}\n                                      </mat-label>\n                                  \n                                    <mat-select\n                                      formControlName=\"duration\"\n                                      multiple\n                                      panelClass=\"select-search-panel\"\n                                      (selectionChange)=\"applyFilters()\">\n                                  \n                                      <!-- Search box -->\n                                        <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n                                          matInput\n                                          placeholder=\"Search Duration\"\n                                          (click)=\"$event.stopPropagation()\"\n                                          (keydown)=\"$event.stopPropagation()\"\n                                          (keyup)=\"filterList($any($event.target).value, 'duration')\" />\n                                  \n                                      <!-- Values -->\n                                      <mat-option\n                                        *ngFor=\"let r of filteredDurations\"\n                                        [value]=\"r\">\n                                        {{ r }}\n                                      </mat-option>\n                                  \n                                    </mat-select>\n                                  </mat-form-field>\n                            </div>\n                        </div>\n                        <div class=\"search-and-buttons-container\">\n                            <div style=\"margin-left: 10px;\">\n                                <label>Select Provider</label>\n                            </div>                            \n                            <div>\n                                <mat-form-field appearance=\"outline\">\n                                    <mat-label>\n                                        {{ filterForm.value.provider?.length ? '' : 'Provider' }}\n                                      </mat-label>\n                                  \n                                    <mat-select\n                                      formControlName=\"provider\"\n                                      multiple\n                                      panelClass=\"select-search-panel\"\n                                      (selectionChange)=\"applyFilters()\">\n                                  \n                                      <!-- Search box -->\n                                        <input style=\"border:1px solid #ccc; padding: 5px;width: 92%;margin: 5px;\"\n                                          matInput\n                                          placeholder=\"Search Provider\"\n                                          (click)=\"$event.stopPropagation()\"\n                                          (keydown)=\"$event.stopPropagation()\"\n                                          (keyup)=\"filterList($any($event.target).value, 'provider')\" />\n                                  \n                                      <!-- Values -->\n                                      <mat-option\n                                        *ngFor=\"let r of filteredProviders\"\n                                        [value]=\"r\">\n                                        {{ r }}\n                                      </mat-option>\n                                  \n                                    </mat-select>\n                                  </mat-form-field>\n                            </div>\n                        </div>\n                        <div style=\"margin-top: 45px;margin-left: 20px;\">\n                            <input class=\"btn-active\" type=\"button\" value=\"Reset All Filters\"\n                                (click)=\"resetFilters()\" />\n                        </div>\n                    </div>\n                    <div class=\"mt-4\">\n                        <div style=\"font-size:16px;font-weight:bold;color:#000\">\n                            Filtered Course Count : {{filterdCourses?.length}}\n                        </div>\n                    </div>\n                    <div class=\"mt-5 sub-heading select-all-container\" *ngIf=\"filterdCourses?.length\">\n                        <div style=\"margin-left: 16px;\">\n                            <mat-checkbox [checked]=\"false\" (change)=\"selectAllCourses($event)\">Select All\n                                Courses</mat-checkbox>\n                        </div>\n                        <div class=\"btn-group-compact\">\n                            <div>\n                                <input class=\"btn-active\" type=\"button\" value=\"Suggest More Course From iGOT\"\n                                    (click)=\"suggestMoreCourses()\" />\n                            </div>\n                            <div class=\"pl-2\">\n                                <input [disabled]=\"selectFilterCourses?.length > 0 ? false : true\"\n                                    [ngClass]=\"selectFilterCourses?.length > 0 ? 'btn-active' : 'btn-disable'\"\n                                    type=\"button\" [value]=\"mode === 'add' ? 'Save Courses' : 'Update Courses'\"\n                                    (click)=\"saveCourses()\" />\n                            </div>\n                        </div>\n                    </div>\n                    <div class=\"disclaimer-container mt-2\" *ngIf=\"filterdCourses?.length\">\n                        <div class=\"disclaimer-message mt-4\">\n                            <strong>Disclaimer:</strong> Please verify the public course URL before adding it to the plan.\n                        </div>\n                    </div>\n                    <div class=\"mt-2\">\n\n                        <div class=\"course-list-container\">\n                            <div\n                                [ngClass]=\"innerTabActiveIndex > 0 ? 'course-list-container-flex':'course-list-container-right-flex' \">\n                                <div *ngIf=\"innerTabActiveIndex > 0\"\n                                    [ngClass]=\"innerTabActiveIndex > 0 ? 'left-panel':''\">\n                                    <ng-container *ngIf=\"innerTabActiveIndex === 1\">\n                                        <div class=\"sub-heading\">\n                                            <p>Competencies in Behavioural</p>\n                                        </div>\n                                        <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n                                            <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n                                                <mat-icon>clear</mat-icon> Clear Filter\n                                            </button>\n                                        </div>\n                                        <div (click)=\"filterOnCompetencyTheme(item)\"\n                                             class=\"category-list-item cursor-pointer mt-2\"\n                                             [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n                                             *ngFor=\"let item of competencyMatchedByCategory\">\n                                            {{item}}\n                                        </div>\n                                    </ng-container>\n                                    <ng-container *ngIf=\"innerTabActiveIndex === 2\">\n                                        <div class=\"sub-heading\">\n                                            <p>Competencies in Functional</p>\n                                        </div>\n                                        <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n                                            <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n                                                <mat-icon>clear</mat-icon> Clear Filter\n                                            </button>\n                                        </div>\n                                        <div (click)=\"filterOnCompetencyTheme(item)\"\n                                             class=\"category-list-item cursor-pointer mt-2\"\n                                             [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n                                             *ngFor=\"let item of competencyMatchedByCategory\">\n                                            {{item}}\n                                        </div>\n                                    </ng-container>\n                                    <ng-container *ngIf=\"innerTabActiveIndex === 3\">\n                                        <div class=\"sub-heading\">\n                                            <p>Competencies in Domain</p>\n                                        </div>\n                                        <div class=\"theme-filter-controls mb-2\" *ngIf=\"selectedThemeFilter\">\n                                            <button class=\"clear-filter-btn\" (click)=\"clearThemeFilter()\">\n                                                <mat-icon>clear</mat-icon> Clear Filter\n                                            </button>\n                                        </div>\n                                        <div (click)=\"filterOnCompetencyTheme(item)\"\n                                            class=\"category-list-item cursor-pointer mt-2\"\n                                            [ngClass]=\"{'selected-theme': isThemeSelected(item)}\"\n                                            *ngFor=\"let item of competencyMatchedByCategory\">\n                                            {{item}}\n                                        </div>\n                                    </ng-container>\n                                </div>\n                                <div [ngClass]=\"innerTabActiveIndex > 0 ? 'right-panel':'right-panel-flex'\">\n                                    <!-- No Course Found Message in Right Panel -->\n                                    <div class=\"no-course-found-container\"\n                                        *ngIf=\"filterdCourses?.length === 0 && selectedThemeFilter\">\n                                        <div class=\"no-course-card\">\n                                            <div class=\"no-course-icon\">\n                                                <mat-icon>search_off</mat-icon>\n                                            </div>\n                                            <div class=\"no-course-content\">\n                                                <h4>No Courses Found</h4>\n                                                <p>No courses found for the selected theme:\n                                                    <strong>{{selectedThemeFilter}}</strong></p>\n                                                <p>Would you like to add a course for this competency?</p>\n                                                <button class=\"suggest-course-btn\"\n                                                    (click)=\"addCourseForSelectedTheme()\">\n                                                    <mat-icon>add_circle</mat-icon>\n                                                    Add Course\n                                                </button>\n                                            </div>\n                                        </div>\n                                    </div>\n\n                                    <!-- Courses List -->\n                                    <div class=\"course-list-item\" *ngFor=\"let item of filterdCourses;let i=index\">\n\n                                        <div class=\"course-content\">\n                                            <div class=\"course-header\">\n                                                <div class=\"checked-course-container\">\n                                                    <div>\n                                                        <mat-checkbox [checked]=\"checkIfCourseExists(item)\"\n                                                            (change)=\"selectedFilterCourses($event, item)\"></mat-checkbox>\n                                                    </div>\n                                                    <div class=\"ml-2\">\n                                                        <div class=\"course-pill-container\">\n                                                            <div class=\"course-pill\">\n                                                                <div class=\"course-pill-text\">\n                                                                    <span><img\n                                                                            src=\"assets/icons/course.svg\"></span>&nbsp;<span>Course</span>\n                                                                </div>\n                                                            </div>\n                                                        </div>\n                                                    </div>\n\n                                                </div>\n\n                                                <!-- <div>\n                                        <div class=\"ai-recommened-pill\">\n                                            <div class=\"ai-recommened-pill-text\">\n                                                <span class=\"ai-loader-icon\"><img src=\"assets/icons/ai-loader.gif\"></span>\n\n                                                <span *ngIf=\"item?.rationale\">AI Recommended\n                                                    <span *ngIf=\"item?.is_public\"> - Public</span>\n                                                    <span *ngIf=\"!item?.is_public\"> - iGOT</span>\n                                                </span>\n                                                <span *ngIf=\"!item?.rationale\">Manually Suggested - iGOT</span>\n\n                                            </div>\n                                        </div>\n                                    </div> -->\n                                                <div *ngIf=\"item?.rationale && !item?.is_public\">\n                                                    <div class=\"ai-recommened-pill-green\">\n                                                        <div class=\"ai-recommened-pill-green-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n\n                                                            <span>AI Recommended - iGOT </span>\n\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n                                                                target=\"_blank\"><span>Platform -\n                                                                    {{item?.platform}}</span></a>\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <span>Provider - {{item?.organisation[0]}}</span>\n                                                        </div>\n                                                    </div>\n                                                </div>\n                                                <div *ngIf=\"item?.rationale && item?.is_public\">\n                                                    <div class=\"ai-recommened-pill-public\">\n                                                        <div class=\"ai-recommened-pill-public-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n\n                                                            <span>AI Recommended - Public </span>\n\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n                                                                target=\"_blank\"><span>Platform -\n                                                                    {{item?.platform}}</span></a>\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <span>Provider - {{item?.organisation[0]}}</span>\n                                                        </div>\n                                                    </div>\n                                                </div>\n                                                <div *ngIf=\"!item?.rationale\">\n                                                    <div class=\"ai-recommened-pill-gray\">\n                                                        <div class=\"ai-recommened-pill-gray-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n\n                                                            <span>Manually Suggested - iGOT </span>\n\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.platform\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <a class=\"cursor-pointer\" [href]=\"item?.public_link\"\n                                                                target=\"_blank\"><span>Platform -\n                                                                    {{item?.platform}}</span></a>\n                                                        </div>\n                                                    </div>\n                                                    <div class=\"ai-recommened-pill-orange mt-2\" *ngIf=\"item?.organisation?.length\">\n                                                        <div class=\"ai-recommened-pill-orange-text\">\n                                                            <span class=\"ai-loader-icon\"><img\n                                                                    src=\"assets/icons/ai-loader.gif\"></span>\n                                                            <span>Provider - {{item?.organisation[0]}}</span>\n                                                        </div>\n                                                    </div>\n                                                </div>\n\n                                            </div>\n                                            <div class=\"course-title mt-3 cursor-pointer\" (click)=\"redirectToToc(item)\">\n                                                <span *ngIf=\"item?.course\">{{item?.course}}</span>\n                                                <!-- <span *ngIf=\"item?.name\">{{item?.name}}****</span> -->\n                                                <span style=\"margin-left:10px;\" (click)=\"redirectToToc(item)\"><img\n                                                        height=\"18px\" width=\"18px\"\n                                                        src=\"assets/icons/redirect.png\" /></span>\n                                            </div>\n                                            <div class=\"course-desc mt-2\">\n                                                <!-- <span *ngIf=\"item?.rationale\">{{item?.rationale}}</span> -->\n                                                <span *ngIf=\"item?.description\">{{item?.description}}</span>\n                                            </div>\n                                            <div class=\"mt-3\" *ngIf=\"getCompetenciesByType('Behavioural',i)?.length\">\n                                                <div class=\"competency-sub-heading\">Behavioural Competencies</div>\n                                                <ul class=\"competency-list mt-1\">\n                                                    <li class=\"behavioural-pill\"\n                                                        *ngFor=\"let comp of getDisplayedCompetencies('Behavioural',i)\">\n                                                        <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n                                                            {{\n                                                            comp.competencySubThemeName }} </span>\n                                                    </li>\n                                                    <li *ngIf=\"hasMoreThanTwo('Behavioural',i)\"\n                                                        class=\"show-more-competency\">\n                                                        <span class=\"show-more-link cursor-pointer\"\n                                                            (click)=\"toggleCompetencies('Behavioural',i)\">\n                                                            {{ isExpanded('Behavioural',i) ? 'Show Less' : '+' +\n                                                            getRemainingCount('Behavioural',i) + ' More' }}\n                                                        </span>\n                                                    </li>\n                                                </ul>\n                                            </div>\n                                            <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Functional',i)?.length\">\n                                                <div class=\"competency-sub-heading\">Functional Competencies</div>\n                                                <ul class=\"competency-list mt-1\">\n                                                    <li class=\"functional-pill\"\n                                                        *ngFor=\"let comp of getDisplayedCompetencies('Functional',i)\">\n                                                        <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n                                                            {{\n                                                            comp.competencySubThemeName }} </span>\n                                                    </li>\n                                                    <li *ngIf=\"hasMoreThanTwo('Functional',i)\"\n                                                        class=\"show-more-competency\">\n                                                        <span class=\"show-more-link cursor-pointer\"\n                                                            (click)=\"toggleCompetencies('Functional',i)\">\n                                                            {{ isExpanded('Functional',i) ? 'Show Less' : '+' +\n                                                            getRemainingCount('Functional',i) + ' More' }}\n                                                        </span>\n                                                    </li>\n                                                </ul>\n                                            </div>\n                                            <div class=\"mt-2\" *ngIf=\"getCompetenciesByType('Domain',i)?.length\">\n                                                <div class=\"competency-sub-heading\">Domain Competencies</div>\n                                                <ul class=\"competency-list mt-1\">\n                                                    <li class=\"domain-pill\"\n                                                        *ngFor=\"let comp of getDisplayedCompetencies('Domain',i)\">\n                                                        <span class=\"competency-text\"> {{ comp.competencyThemeName }} -\n                                                            {{\n                                                            comp.competencySubThemeName }} </span>\n                                                    </li>\n                                                    <li *ngIf=\"hasMoreThanTwo('Domain',i)\" class=\"show-more-competency\">\n                                                        <span class=\"show-more-link cursor-pointer\"\n                                                            (click)=\"toggleCompetencies('Domain',i)\">\n                                                            {{ isExpanded('Domain',i) ? 'Show Less' : '+' +\n                                                            getRemainingCount('Domain',i) + ' More' }}\n                                                        </span>\n                                                    </li>\n                                                </ul>\n                                            </div>\n\n\n                                        </div>\n\n                                        <!-- Relevancy and Duration section -->\n                                        <div class=\"course-footer\" *ngIf=\"item?.relevancy\">\n                                            <div class=\"relevancy-pill-green\">\n                                                <div class=\"relevancy-container\">\n                                                    <div class=\"relevancy\">Relevancy</div>\n                                                    <div class=\"dash\">-</div>\n                                                    <div class=\"percentage\">{{item?.relevancy}}%</div>\n                                                </div>\n                                            </div>\n                                            <div class=\"relevancy-pill-green\">\n                                                <div class=\"relevancy-container\">\n                                                    <div class=\"relevancy\">Duration</div>\n                                                    <div class=\"dash\">-</div>\n                                                    <div class=\"percentage\">\n                                                        <span\n                                                            *ngIf=\"item?.duration\">{{getDuration(item?.duration)}}</span>\n                                                        <span *ngIf=\"!(item?.duration)\">N/A</span>\n                                                    </div>\n                                                </div>\n                                            </div>\n                                        </div>\n\n                                    </div>\n                                </div>\n                            </div>\n                        </div>\n                    </div>\n\n                    <!-- General No Data Message (only when no theme filter is active) -->\n                    <div class=\"mt-2 sub-heading\" *ngIf=\"filterdCourses?.length === 0 && !selectedThemeFilter\">\n                        <p>No Data Found</p>\n                    </div>\n                </div>\n            </div>\n        </form>\n            <div class=\"role-mapping-container\" *ngIf=\"outerTabActiveIndex === 1\">\n                <div id=\"gap-analysis-content\">\n\n                    <div class=\"download-pdf mt-4 mb-2\">\n                        <div>\n                            <input class=\"btn-active\" type=\"button\" value=\"Download\" (click)=\"downloadPDF()\" />\n                        </div>\n                    </div>\n                    <div>\n                        <!-- <div *ngIf=\"isPDFDownload\">\n                            <div class=\"pdf-heading\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}} <span\n                                    *ngIf=\"planData?.designation_name \">_{{planData?.designation_name}}_Gap_Analysis</span>\n                            </div>\n                        </div> -->\n                        <div class=\"d-flex vh-100\" [ngClass]=\"isPDFDownload ? 'mt-4':''\">\n                            <!-- Sidebar -->\n                            <div class=\"d-flex flex-column p-3 bg-light border-end\" style=\"width: 250px;\">\n                                <h5 class=\"mb-4\">Analyze by Category</h5>\n                                <button *ngFor=\"let item of menuItems\"\n                                    class=\"btn btn-outline-primary mb-2 text-start menu-list-item\"\n                                    [class.active]=\"selectedCategory === item.key\" (click)=\"selectCategory(item.key)\">\n                                    {{ item.label }}\n                                </button>\n                            </div>\n\n                            <!-- Main Content -->\n                            <div class=\"flex-grow-1 p-4\">\n                                <div class=\"d-flex justify-content-between gap-3 \">\n                                    <div class=\"p-3 gray-bg\">\n                                        <div class=\"div-label sub-heading\">\n                                            <p>Competencies Covered</p>\n                                        </div>\n                                        <div class=\"div-count\">{{competencyCoveredCount}}</div>\n                                    </div>\n                                    <div class=\"p-3 gray-bg\">\n                                        <div class=\"div-label sub-heading\">\n                                            <p *ngIf=\"selectedCategory === 'all'\">Total Competencies</p>\n                                            <p *ngIf=\"selectedCategory === 'behavioral'\">Total Behavioural Competencies\n                                            </p>\n                                            <p *ngIf=\"selectedCategory === 'functional'\">Total Functional Competencies\n                                            </p>\n                                            <p *ngIf=\"selectedCategory === 'domain'\">Total Domain Competencies</p>\n                                        </div>\n                                        <div class=\"div-count\" *ngIf=\"selectedCategory === 'all'\">\n                                            {{this.planData?.competencies?.length}}</div>\n                                        <div class=\"div-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n                                            {{behavioralTotalCompetencies}}</div>\n                                        <div class=\"div-count\" *ngIf=\"selectedCategory === 'functional'\">\n                                            {{functionalTotalCompetencies}}</div>\n                                        <div class=\"div-count\" *ngIf=\"selectedCategory === 'domain'\">\n                                            {{domainTotalCompetencies}}</div>\n                                    </div>\n                                    <div class=\"p-3 green-bg\">\n                                        <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'all'\">Overall Coverage\n                                        </div>\n                                        <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'behavioral'\">\n                                            Behavioural Coverage</div>\n                                        <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'functional'\">\n                                            Functional Coverage</div>\n                                        <div class=\"div-green-label\" *ngIf=\"selectedCategory === 'domain'\">Domain\n                                            Coverage</div>\n                                        <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'all'\">\n                                            {{overallCoverage}}</div>\n                                        <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'behavioral'\">\n                                            {{behavioralCoverage}}</div>\n                                        <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'functional'\">\n                                            {{functionalCoverage}}</div>\n                                        <div class=\"div-green-count\" *ngIf=\"selectedCategory === 'domain'\">\n                                            {{domainCoverage}}</div>\n                                    </div>\n                                </div>\n                                <div class=\"mt-4 all-category-list\" [ngSwitch]=\"selectedCategory\">\n                                    <div *ngSwitchCase=\"'all'\">\n                                        <h4>All Categories</h4>\n\n                                        <!-- Show Behavioural section only if there are unmatched behavioural competencies -->\n                                        <div class=\"mt-4\"\n                                            *ngIf=\"behaviouralNotMatched && behaviouralNotMatched.length > 0\">\n                                            <div class=\"sub-heading\">\n                                                <p>Behavioural</p>\n                                            </div>\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item | titlecase}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n                                                            Course</span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n\n                                        <!-- Show Functional section only if there are unmatched functional competencies -->\n                                        <div class=\"mt-4\"\n                                            *ngIf=\"functionalNotMatched && functionalNotMatched.length > 0\">\n                                            <div class=\"sub-heading\">\n                                                <p>Functional</p>\n                                            </div>\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item | titlecase}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n                                                            Course</span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n\n                                        <!-- Show Domain section only if there are unmatched domain competencies -->\n                                        <div class=\"mt-4\" *ngIf=\"domainNotMatched && domainNotMatched.length > 0\">\n                                            <div class=\"sub-heading\">\n                                                <p>Domain</p>\n                                            </div>\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item | titlecase}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New\n                                                            Course</span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n\n                                        <!-- Show message when all competencies are covered -->\n                                        <div class=\"mt-4 text-center\"\n                                            *ngIf=\"(!behaviouralNotMatched || behaviouralNotMatched.length === 0) && \n                                                                           (!functionalNotMatched || functionalNotMatched.length === 0) && \n                                                                           (!domainNotMatched || domainNotMatched.length === 0)\">\n                                            <div class=\"no-gaps-message\">\n                                                <mat-icon class=\"success-icon\">check_circle</mat-icon>\n                                                <h4>Excellent! All competencies are covered</h4>\n                                                <p>All required competencies have been addressed by the selected\n                                                    courses.</p>\n                                            </div>\n                                        </div>\n\n                                    </div>\n\n                                    <div *ngSwitchCase=\"'behavioral'\">\n                                        <h4>Behavioural</h4>\n                                        <div class=\"mt-4\" *ngIf=\"behavioralCoverage !== '100%'\">\n                                            <!-- <div class=\"sub-heading\">\n                                            <p>Behavioural</p>\n                                        </div> -->\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of behaviouralNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Behavioral')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n                                        <div class=\"mt-4\" *ngIf=\"behavioralCoverage === '100%'\">\n                                            <div class=\"p-3\"\n                                                style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n                                                <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n                                                    🎉 Excellent! All behavioral competencies are covered by the\n                                                    selected courses.\n                                                </div>\n                                            </div>\n                                        </div>\n                                    </div>\n\n                                    <div *ngSwitchCase=\"'functional'\">\n                                        <h4>Functional</h4>\n                                        <div class=\"mt-4\" *ngIf=\"functionalCoverage !== '100%'\">\n                                            <!-- <div class=\"sub-heading\">\n                                            <p>Functional</p>\n                                        </div> -->\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of functionalNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Functional')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course</span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n                                        <div class=\"mt-4\" *ngIf=\"functionalCoverage === '100%'\">\n                                            <div class=\"p-3\"\n                                                style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n                                                <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n                                                    🎉 Excellent! All functional competencies are covered by the\n                                                    selected courses.\n                                                </div>\n                                            </div>\n                                        </div>\n                                    </div>\n\n                                    <div *ngSwitchCase=\"'domain'\">\n                                        <h4>Domain</h4>\n                                        <div class=\"mt-4\" *ngIf=\"domainCoverage !== '100%'\">\n                                            <!-- <div class=\"sub-heading\">\n                                            <p>Domain</p>\n                                        </div> -->\n                                            <div class=\"gray-bg-item\" *ngFor=\"let item of domainNotMatched\">\n                                                <div class=\"list-flex\">\n                                                    <div><span>{{item}}</span></div>\n                                                    <div class=\"cursor-pointer\" (click)=\"addCourse(item, 'Domain')\">\n                                                        <span class=\"add-btn\"><mat-icon>add</mat-icon> Add New Course </span></div>\n                                                </div>\n                                            </div>\n                                        </div>\n                                        <div class=\"mt-4\" *ngIf=\"domainCoverage === '100%'\">\n                                            <div class=\"p-3\"\n                                                style=\"background-color: #e8f5e8; border: 1px solid #4caf50; border-radius: 8px; text-align: center;\">\n                                                <div style=\"color: #2e7d32; font-weight: 600; font-size: 16px;\">\n                                                    🎉 Excellent! All domain competencies are covered by the selected\n                                                    courses.\n                                                </div>\n                                            </div>\n                                        </div>\n                                    </div>\n\n                                    <div *ngSwitchDefault>\n                                        <p>Please select a category.</p>\n                                    </div>\n                                </div>\n                            </div>\n                        </div>\n                    </div>\n                </div>\n\n            </div>\n        </div>\n        </div> <!-- Close main-content div -->\n\n    </div>\n</div>\n\n\n<div [hidden]=\"!isPDFDownload\" class=\"gap-analysis-pdf-container\" #pdfContent>\n    <div class=\"gap-analysis-pdf-header\">\n      <img src=\"assets/icons/karmayogiLogo-min.png\" alt=\"Karmayogi Bharat Logo\" class=\"logo\" />\n      <h1><span>\n        <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'state'\">{{sharedService?.cbpPlanFinalObj?.department_name?.name}}</span> \n        <span *ngIf=\"sharedService?.cbpPlanFinalObj?.ministry?.sbOrgType === 'ministry'\">{{sharedService?.cbpPlanFinalObj?.ministry?.name}}</span>   \n        <span *ngIf=\"planData?.designation_name \">/ {{planData?.designation_name}}</span>\n        </span></h1>\n      <p class=\"subtitle\">A detailed breakdown of competencies Gap Analysis</p>\n     \n    </div>\n  \n    <div class=\"competency-snapshot\">\n      <h2>Competency Snapshot</h2>\n      <div class=\"snapshot-content\">\n        <div class=\"circle-wrapper\">\n          <svg class=\"progress-ring\" width=\"120\" height=\"120\">\n            <circle class=\"progress-ring-bg\" cx=\"60\" cy=\"60\" r=\"52\" />\n            <circle class=\"progress-ring-circle\" cx=\"60\" cy=\"60\" r=\"52\" />\n          </svg>\n          <div class=\"percentage-text\">{{overallCoverage}}<br /><span>Overall Coverage</span></div>\n        </div>\n  \n        <div class=\"competency-container\">\n            <!-- Total Competencies Box -->\n            <div class=\"total-box\">\n              <div class=\"title\">Total Competencies</div>\n              <div class=\"count\">{{this.planData?.competencies?.length}}</div>\n            </div>\n          \n            <!-- Progress Bars -->\n            <div class=\"progress-group\">\n              <div class=\"progress-label\">\n                <span class=\"label-text behavioral\">Behavioural</span>\n                <span class=\"covered-text\">{{behavioralCompetencyCoveredCount}} / {{behavioralTotalCompetencies}} Covered</span>\n              </div>\n              <div class=\"progress-bar\">\n                <div class=\"progress-fill behavioral\" [style.width.%]=\"(behavioralCompetencyCoveredCount / behavioralTotalCompetencies) * 100\"></div>\n              </div>\n            </div>\n          \n            <div class=\"progress-group\">\n              <div class=\"progress-label\">\n                <span class=\"label-text functional\">Functional</span>\n                <span class=\"covered-text\">{{functionalCompetencyCoveredCount}} / {{functionalTotalCompetencies}} Covered</span>\n              </div>\n              <div class=\"progress-bar\">\n                <div class=\"progress-fill functional\" [style.width.%]=\"(functionalCompetencyCoveredCount / functionalTotalCompetencies) * 100\"></div>\n              </div>\n            </div>\n          \n            <div class=\"progress-group\">\n              <div class=\"progress-label\">\n                <span class=\"label-text domain\">Domain</span>\n                <span class=\"covered-text\">{{domainCompetencyCoveredCount}} / {{domainTotalCompetencies}} Covered</span>\n              </div>\n              <div class=\"progress-bar\">\n                <div class=\"progress-fill domain\" [style.width.%]=\"(domainCompetencyCoveredCount / domainTotalCompetencies) * 100\"></div>\n              </div>\n            </div>\n          </div>\n      </div>\n    </div>\n  \n    <div class=\"gap-section mb-4\">\n      <h2>Detailed Competency Analysis</h2>\n      <div class=\"gap-box\">\n        Bridge Your Competencies Gap\n        <p class=\"subtitle\">Here are {{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}} key areas for development.</p>\n    </div>\n    </div>\n    <div class=\"sub-heading mb-4\">\n        <p>Gap to Address <span *ngIf=\"(this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)) > 0\">({{this.planData?.competencies?.length - (behavioralCompetencyCoveredCount + functionalCompetencyCoveredCount + domainCompetencyCoveredCount)}})</span></p>\n        \n    </div>\n    <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralNotMatched\">\n        <div class=\"title\">{{item | titlecase}}</div>\n        <div class=\"subtitle\">\n          &nbsp; • &nbsp;\n          <span class=\"tag behavioral\">Behavioural</span>\n        </div>\n    </div>\n      \n      <div class=\"competency-card functional\" *ngFor=\"let item of functionalNotMatched\">\n        <div class=\"title\">{{item | titlecase}}</div>\n        <div class=\"subtitle\">\n           &nbsp; • &nbsp;\n          <span class=\"tag functional\">Functional</span>\n        </div>\n      </div>\n      \n      <div class=\"competency-card domain\" *ngFor=\"let item of domainNotMatched\">\n        <div class=\"title\">{{item | titlecase}}</div>\n        <div class=\"subtitle\">\n          <!-- Project Management  -->\n          &nbsp; • &nbsp;\n          <span class=\"tag domain\">Domain</span>\n        </div>\n      </div>\n      <div class=\"sub-heading mb-4\">\n        <p>Covered Competencies <span>({{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}})</span></p>        \n     </div>\n     <div class=\"gap-section mb-4\">\n        \n        <div class=\"green-bg\">\n            <div class=\"div-green-label mb-2\">\n                Congratulations!\n            </div>            \n            <p>You have successfully covered </p>\n          <p class=\"subtitle\">{{this.planData?.competencies?.length - (behaviouralNotMatched?.length + functionalNotMatched?.length + domainNotMatched?.length)}} key competencies, forming a strong professional foundation.</p>\n      </div>\n      </div>\n      <div class=\"competency-card behavioral\" *ngFor=\"let item of behaviouralMatched\">\n        <div class=\"title\">{{item?.theme | titlecase}}</div>\n        <div class=\"subtitle\">\n          {{item?.sub_theme}}&nbsp; • &nbsp;\n          <span class=\"tag behavioral\">Behavioural</span>\n        </div>\n    </div>\n      \n      <div class=\"competency-card functional\" *ngFor=\"let item of functionalMatched\">\n        <div class=\"title\">{{item?.theme | titlecase}}</div>\n        <div class=\"subtitle\">\n            {{item?.sub_theme}}&nbsp; • &nbsp;\n          <span class=\"tag functional\">Functional</span>\n        </div>\n      </div>\n      \n      <div class=\"competency-card domain\" *ngFor=\"let item of domainMatched\">\n        <div class=\"title\">{{item?.theme | titlecase}}</div>\n        <div class=\"subtitle\">\n          <!-- Project Management  -->\n          {{item?.sub_theme}}&nbsp; • &nbsp;\n          <span class=\"tag domain\">Domain</span>\n        </div>\n      </div>\n      <div class=\"gap-section mb-4\">\n        <h2>Recommended Courses</h2>\n      </div>\n      <app-gap-analysis-recommended-course [planData]=\"planData\"></app-gap-analysis-recommended-course>\n</div>\n  \n\n\n<div class=\"overlay-loader\" *ngIf=\"loading && dataLoaded && !isRegeneratingWithProgress\">\n    <div class=\"loading-content\">\n        <mat-spinner diameter=\"50\"></mat-spinner>\n        <div class=\"loading-text mt-3\">\n            <p class=\"main-message\">{{ currentProcessingStage || 'Preparing course recommendation generation...' }}</p>\n            <div class=\"progress-details\">\n                <div class=\"progress-indicator\" *ngIf=\"progressPercentage > 0\">\n                    <div class=\"progress-bar-container\">\n                        <div class=\"progress-bar\" \n                             [style.width.%]=\"progressPercentage\">\n                        </div>\n                    </div>\n                    <small class=\"progress-text\">{{ progressPercentage }}% Complete</small>\n                </div>\n                <small class=\"time-estimate\">\n                    Real-time course generation in progress. This may take 1-2 minutes.\n                </small>\n            </div>\n        </div>\n    </div>\n</div>"]}
|