@rangertechnologies/ngnxt 2.1.253 → 2.1.255-beta
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/fesm2022/rangertechnologies-ngnxt.mjs +258 -433
- package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -1
- package/lib/components/datatable/datatable.component.d.ts +0 -1
- package/lib/components/pick-location/pick-location.component.d.ts +0 -1
- package/lib/model/bookletWrapper.d.ts +0 -1
- package/lib/pages/booklet/booklet.component.d.ts +0 -2
- package/lib/pages/builder/element/element.component.d.ts +0 -3
- package/lib/pages/builder/properties/properties.component.d.ts +6 -2
- package/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.d.ts +0 -3
- package/lib/pages/questionbook/questionbook.component.d.ts +0 -1
- package/lib/pages/questionnaire/questionnaire.component.d.ts +0 -1
- package/lib/pipe/custom-translate.pipe.d.ts +1 -1
- package/lib/pipe/svg/svg.pipe.d.ts +10 -0
- package/lib/services/form-builder/form-builder.service.d.ts +1 -0
- package/lib/services/translation/translation.service.d.ts +1 -1
- package/package.json +4 -6
- package/rangertechnologies-ngnxt-2.1.255-beta.tgz +0 -0
- package/esm2022/environments/version.mjs +0 -15
- package/esm2022/lib/components/button/nxt-button.component.mjs +0 -154
- package/esm2022/lib/components/custom-calendar/custom-calendar.component.mjs +0 -360
- package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +0 -263
- package/esm2022/lib/components/custom-model/custom-model.component.mjs +0 -53
- package/esm2022/lib/components/custom-radio/custom-radio.component.mjs +0 -156
- package/esm2022/lib/components/datatable/datatable.component.mjs +0 -1757
- package/esm2022/lib/components/file-upload/file-upload.component.mjs +0 -292
- package/esm2022/lib/components/icon-selector/icon-selector.component.mjs +0 -105
- package/esm2022/lib/components/image-cropper/component/cropper.state.mjs +0 -208
- package/esm2022/lib/components/image-cropper/component/image-cropper.component.mjs +0 -562
- package/esm2022/lib/components/image-cropper/interfaces/basic-event.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/cropper-options.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/cropper-position.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/dimensions.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/exif-transform.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/image-cropped-event.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/image-transform.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/index.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/loaded-image.interface.mjs +0 -2
- package/esm2022/lib/components/image-cropper/interfaces/move-start.interface.mjs +0 -8
- package/esm2022/lib/components/image-cropper/services/crop.service.mjs +0 -139
- package/esm2022/lib/components/image-cropper/services/load-image.service.mjs +0 -194
- package/esm2022/lib/components/image-cropper/utils/cropper-position.utils.mjs +0 -239
- package/esm2022/lib/components/image-cropper/utils/exif.utils.mjs +0 -79
- package/esm2022/lib/components/image-cropper/utils/keyboard.utils.mjs +0 -40
- package/esm2022/lib/components/image-cropper/utils/percentage.utils.mjs +0 -4
- package/esm2022/lib/components/image-cropper/utils/resize.utils.mjs +0 -75
- package/esm2022/lib/components/list-view-filter/list-view-filter.component.mjs +0 -392
- package/esm2022/lib/components/nxt-input/nxt-input.component.mjs +0 -2972
- package/esm2022/lib/components/pagination/pagination.component.mjs +0 -105
- package/esm2022/lib/components/pick-location/pick-location.component.mjs +0 -220
- package/esm2022/lib/components/search-box/search-box.component.mjs +0 -470
- package/esm2022/lib/country.json +0 -43237
- package/esm2022/lib/interfaces/actionMeta.mjs +0 -2
- package/esm2022/lib/interfaces/dependencyMeta.mjs +0 -2
- package/esm2022/lib/model/bookletWrapper.mjs +0 -9
- package/esm2022/lib/model/changeWrapper.mjs +0 -10
- package/esm2022/lib/model/errorWrapper.mjs +0 -6
- package/esm2022/lib/nxt-app.component.mjs +0 -22
- package/esm2022/lib/nxt-app.module.mjs +0 -115
- package/esm2022/lib/nxt-app.service.mjs +0 -14
- package/esm2022/lib/pages/booklet/booklet.component.mjs +0 -658
- package/esm2022/lib/pages/builder/element/element.component.mjs +0 -508
- package/esm2022/lib/pages/builder/form/form.component.mjs +0 -48
- package/esm2022/lib/pages/builder/properties/common-fields.constants.mjs +0 -97
- package/esm2022/lib/pages/builder/properties/properties.component.mjs +0 -1089
- package/esm2022/lib/pages/builder/templates/templates.component.mjs +0 -35
- package/esm2022/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.mjs +0 -681
- package/esm2022/lib/pages/pdfDesigner/pdf-properties/pdf-properties.component.mjs +0 -1118
- package/esm2022/lib/pages/questionbook/questionbook.component.mjs +0 -801
- package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +0 -2219
- package/esm2022/lib/pipe/button-styles.pipe.mjs +0 -26
- package/esm2022/lib/pipe/custom-translate.pipe.mjs +0 -36
- package/esm2022/lib/pipe/get-value.pipe.mjs +0 -54
- package/esm2022/lib/pipe/question-by-row.pipe.mjs +0 -51
- package/esm2022/lib/pipe/search-filter/search-filter.pipe.mjs +0 -40
- package/esm2022/lib/sample.mjs +0 -3715
- package/esm2022/lib/services/change/change.service.mjs +0 -46
- package/esm2022/lib/services/country/country.service.mjs +0 -135
- package/esm2022/lib/services/data/data.service.mjs +0 -100
- package/esm2022/lib/services/form-builder/form-builder.service.mjs +0 -500
- package/esm2022/lib/services/pdf-designer/pdf-designer.service.mjs +0 -398
- package/esm2022/lib/services/salesforce/salesforce.service.mjs +0 -41
- package/esm2022/lib/services/shared/shared.service.mjs +0 -100
- package/esm2022/lib/services/storage/storage.service.mjs +0 -59
- package/esm2022/lib/services/template/template.service.mjs +0 -351
- package/esm2022/lib/services/translation/translation.service.mjs +0 -121
- package/esm2022/lib/wrapper.mjs +0 -175
- package/esm2022/public-api.mjs +0 -22
- package/esm2022/rangertechnologies-ngnxt.mjs +0 -5
- package/rangertechnologies-ngnxt-2.1.253.tgz +0 -0
|
@@ -1,470 +0,0 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter, ViewChild, signal, HostListener } from "@angular/core";
|
|
2
|
-
import { ChangeWrapper } from "../../model/changeWrapper";
|
|
3
|
-
import { CommonModule } from "@angular/common";
|
|
4
|
-
import { FormsModule } from "@angular/forms";
|
|
5
|
-
import { NxtInput } from "../nxt-input/nxt-input.component";
|
|
6
|
-
import { Subject, debounceTime, distinctUntilChanged, takeUntil } from "rxjs";
|
|
7
|
-
import { NxtCustomTranslatePipe } from "../../pipe/custom-translate.pipe";
|
|
8
|
-
import * as i0 from "@angular/core";
|
|
9
|
-
import * as i1 from "../../services/storage/storage.service";
|
|
10
|
-
import * as i2 from "../../services/change/change.service";
|
|
11
|
-
import * as i3 from "../../services/data/data.service";
|
|
12
|
-
import * as i4 from "../../services/country/country.service";
|
|
13
|
-
import * as i5 from "@angular/common";
|
|
14
|
-
export class NxtSearchBox {
|
|
15
|
-
storageService;
|
|
16
|
-
changeService;
|
|
17
|
-
dataService;
|
|
18
|
-
countryService;
|
|
19
|
-
elementRef;
|
|
20
|
-
placeHolderText = '';
|
|
21
|
-
question;
|
|
22
|
-
apiMeta;
|
|
23
|
-
id = '';
|
|
24
|
-
readOnly = false;
|
|
25
|
-
mode = 'edit';
|
|
26
|
-
from;
|
|
27
|
-
searchValueChange = new EventEmitter();
|
|
28
|
-
searchWrapper;
|
|
29
|
-
SearchItem;
|
|
30
|
-
finalResults = signal([]);
|
|
31
|
-
searchKeyWord = '';
|
|
32
|
-
newResult;
|
|
33
|
-
showResult = false;
|
|
34
|
-
noResult = false;
|
|
35
|
-
showSuggestion = false;
|
|
36
|
-
isLoading = false;
|
|
37
|
-
viewFilterName = '';
|
|
38
|
-
isSingleFieldChange = false;
|
|
39
|
-
valueObj;
|
|
40
|
-
inputFocused = false;
|
|
41
|
-
lastSearchLength = 0;
|
|
42
|
-
filterName = '';
|
|
43
|
-
destroy$ = new Subject();
|
|
44
|
-
searchSubject = new Subject();
|
|
45
|
-
subscription;
|
|
46
|
-
// TrackBy function for performance optimization
|
|
47
|
-
trackByFn = (index, item) => {
|
|
48
|
-
return item.id || item.name || index;
|
|
49
|
-
};
|
|
50
|
-
constructor(storageService, changeService, dataService, countryService, elementRef) {
|
|
51
|
-
this.storageService = storageService;
|
|
52
|
-
this.changeService = changeService;
|
|
53
|
-
this.dataService = dataService;
|
|
54
|
-
this.countryService = countryService;
|
|
55
|
-
this.elementRef = elementRef;
|
|
56
|
-
if (this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)) {
|
|
57
|
-
this.clickItem(this.question?.input);
|
|
58
|
-
}
|
|
59
|
-
else {
|
|
60
|
-
this.filterName = this.question?.input;
|
|
61
|
-
}
|
|
62
|
-
// Setup debounced search
|
|
63
|
-
this.searchSubject
|
|
64
|
-
.pipe(debounceTime(300), distinctUntilChanged(), takeUntil(this.destroy$))
|
|
65
|
-
.subscribe(keyword => {
|
|
66
|
-
if (this.lastSearchLength > 2)
|
|
67
|
-
this.performSearch(keyword);
|
|
68
|
-
});
|
|
69
|
-
}
|
|
70
|
-
ngOnInit() {
|
|
71
|
-
if (this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)) {
|
|
72
|
-
this.clickItem(this.question?.input);
|
|
73
|
-
}
|
|
74
|
-
else if (Array.isArray(this.question?.input)) {
|
|
75
|
-
this.filterName = '';
|
|
76
|
-
this.viewFilterName = '';
|
|
77
|
-
}
|
|
78
|
-
else {
|
|
79
|
-
this.filterName = this.question?.input;
|
|
80
|
-
this.viewFilterName = this.question?.input;
|
|
81
|
-
}
|
|
82
|
-
this.parseApiMeta();
|
|
83
|
-
if (this.question?.subText != undefined) {
|
|
84
|
-
let dependencyObj = this.question?.subText;
|
|
85
|
-
// Subscribe for the changes
|
|
86
|
-
this.subscription = this.changeService.changeAnnounced$.subscribe((changeValue) => {
|
|
87
|
-
if (changeValue != undefined) {
|
|
88
|
-
if (changeValue.fromQuestionId && changeValue.valueObj !== undefined && (Array.isArray(dependencyObj.sourceQuestionId) ? dependencyObj.sourceQuestionId.includes(changeValue.fromQuestionId) : changeValue.fromQuestionId === dependencyObj.sourceQuestionId)) {
|
|
89
|
-
if (Array.isArray(dependencyObj.queryValue) ? dependencyObj.queryValue.includes(changeValue.referenceField) : changeValue.referenceField === dependencyObj.queryValue) {
|
|
90
|
-
this.storageService.queryValueStore(this.question.id, changeValue.referenceField, changeValue.valueObj);
|
|
91
|
-
}
|
|
92
|
-
}
|
|
93
|
-
this.changeService.confirmChange(changeValue.fromQuestionId);
|
|
94
|
-
}
|
|
95
|
-
});
|
|
96
|
-
}
|
|
97
|
-
}
|
|
98
|
-
ngOnChanges(changes) {
|
|
99
|
-
if (changes['question']) {
|
|
100
|
-
// if(this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)){
|
|
101
|
-
// this.clickItem(this.question?.input)
|
|
102
|
-
// } else {
|
|
103
|
-
// this.filterName = this.question?.input
|
|
104
|
-
// }
|
|
105
|
-
}
|
|
106
|
-
if (changes['mode']) {
|
|
107
|
-
this.mode = changes['mode'].currentValue;
|
|
108
|
-
}
|
|
109
|
-
if (changes['apiMeta'] && this.apiMeta) {
|
|
110
|
-
this.parseApiMeta();
|
|
111
|
-
}
|
|
112
|
-
}
|
|
113
|
-
ngOnDestroy() {
|
|
114
|
-
this.destroy$.next();
|
|
115
|
-
this.destroy$.complete();
|
|
116
|
-
this.resetComponentState();
|
|
117
|
-
}
|
|
118
|
-
// Handle clicks outside the search component
|
|
119
|
-
onDocumentClick(event) {
|
|
120
|
-
if (!this.elementRef.nativeElement.contains(event.target)) {
|
|
121
|
-
this.hideSuggestions();
|
|
122
|
-
}
|
|
123
|
-
}
|
|
124
|
-
// Handle escape key to close suggestions
|
|
125
|
-
onEscapeKey(event) {
|
|
126
|
-
this.hideSuggestions();
|
|
127
|
-
}
|
|
128
|
-
parseApiMeta() {
|
|
129
|
-
if (this.apiMeta) {
|
|
130
|
-
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta || '{}');
|
|
131
|
-
this.SearchItem = this.apiMeta.field || [];
|
|
132
|
-
}
|
|
133
|
-
}
|
|
134
|
-
resetComponentState() {
|
|
135
|
-
this.finalResults.set([]);
|
|
136
|
-
this.searchKeyWord = '';
|
|
137
|
-
this.showSuggestion = false;
|
|
138
|
-
this.noResult = false;
|
|
139
|
-
this.isLoading = false;
|
|
140
|
-
this.inputFocused = false;
|
|
141
|
-
this.lastSearchLength = 0;
|
|
142
|
-
}
|
|
143
|
-
hideSuggestions() {
|
|
144
|
-
this.showSuggestion = false;
|
|
145
|
-
this.inputFocused = false;
|
|
146
|
-
}
|
|
147
|
-
onInputFocus() {
|
|
148
|
-
this.inputFocused = true;
|
|
149
|
-
// Trigger a search with an empty keyword to show all data
|
|
150
|
-
if (!this.searchKeyWord) {
|
|
151
|
-
// this.performSearch('');
|
|
152
|
-
}
|
|
153
|
-
else if (this.finalResults().length > 0) {
|
|
154
|
-
this.showSuggestion = true;
|
|
155
|
-
}
|
|
156
|
-
}
|
|
157
|
-
onInputBlur() {
|
|
158
|
-
// Don't immediately hide suggestions to allow for clicking
|
|
159
|
-
// The document click handler will handle hiding
|
|
160
|
-
}
|
|
161
|
-
clearList() {
|
|
162
|
-
// Remove the setTimeout delay that was causing issues
|
|
163
|
-
this.hideSuggestions();
|
|
164
|
-
}
|
|
165
|
-
getSourceDataLocal(event) {
|
|
166
|
-
if (!event?.value?.valueObj) {
|
|
167
|
-
this.resetComponentState();
|
|
168
|
-
return;
|
|
169
|
-
}
|
|
170
|
-
const searchValue = event.value?.valueObj?.includes('|') ? event.value?.valueObj?.split('|')[0].trim() : event.value?.valueObj?.trim();
|
|
171
|
-
const currentLength = searchValue.length;
|
|
172
|
-
// Always update the search keyword
|
|
173
|
-
this.searchKeyWord = searchValue;
|
|
174
|
-
// Handle the case where user types 3+ characters
|
|
175
|
-
if (currentLength > 2 && !this.isSingleFieldChange) {
|
|
176
|
-
this.showSuggestion = !event.question?.singleFieldChange;
|
|
177
|
-
this.showResult = false;
|
|
178
|
-
this.noResult = false;
|
|
179
|
-
// Always perform search when we have 3+ characters
|
|
180
|
-
this.lastSearchLength = currentLength;
|
|
181
|
-
this.searchSubject.next(searchValue);
|
|
182
|
-
}
|
|
183
|
-
else if (currentLength <= 2) {
|
|
184
|
-
// When length is 2 or less, hide suggestions but keep the search keyword
|
|
185
|
-
this.isSingleFieldChange = false;
|
|
186
|
-
this.showSuggestion = false;
|
|
187
|
-
this.noResult = false;
|
|
188
|
-
this.lastSearchLength = currentLength;
|
|
189
|
-
this.searchSubject.next(searchValue);
|
|
190
|
-
this.finalResults.set([]);
|
|
191
|
-
}
|
|
192
|
-
}
|
|
193
|
-
performSearch(keyword) {
|
|
194
|
-
if (!this.apiMeta)
|
|
195
|
-
return;
|
|
196
|
-
this.parseApiMeta();
|
|
197
|
-
if (!this.apiMeta.isStaticData) {
|
|
198
|
-
this.searchFromApi(keyword);
|
|
199
|
-
}
|
|
200
|
-
else {
|
|
201
|
-
this.searchFromStaticData(keyword);
|
|
202
|
-
}
|
|
203
|
-
}
|
|
204
|
-
searchFromApi(keyword) {
|
|
205
|
-
this.isLoading = true;
|
|
206
|
-
this.finalResults.set([]);
|
|
207
|
-
if (!this.apiMeta.endpoint && Array.isArray(this.question?.input)) {
|
|
208
|
-
this.isLoading = false;
|
|
209
|
-
this.processSearchResults(this.question?.input, keyword);
|
|
210
|
-
}
|
|
211
|
-
else {
|
|
212
|
-
let finalGetCall = this.apiMeta.endpoint;
|
|
213
|
-
// SKS18JUL25 query params creation
|
|
214
|
-
if (this.apiMeta?.queryParameter) {
|
|
215
|
-
const isFieldArray = Array.isArray(this.apiMeta?.queryParameter);
|
|
216
|
-
const fields = isFieldArray ? this.apiMeta.queryParameter : [this.apiMeta.queryParameter]; // always an array now
|
|
217
|
-
// Build query params string
|
|
218
|
-
const queryParams = fields
|
|
219
|
-
.map(field => `${field}=${this.searchKeyWord}`)
|
|
220
|
-
.join('&');
|
|
221
|
-
// Check if endpoint already has query params
|
|
222
|
-
if (queryParams && queryParams !== '') {
|
|
223
|
-
if (this.apiMeta.endpoint.includes('?')) {
|
|
224
|
-
finalGetCall += `&${queryParams}`;
|
|
225
|
-
}
|
|
226
|
-
else {
|
|
227
|
-
finalGetCall += `?${queryParams}`;
|
|
228
|
-
}
|
|
229
|
-
}
|
|
230
|
-
}
|
|
231
|
-
if (this.apiMeta?.queryField && this.apiMeta?.queryValue) {
|
|
232
|
-
const queryFields = Array.isArray(this.apiMeta.queryField)
|
|
233
|
-
? this.apiMeta.queryField
|
|
234
|
-
: [this.apiMeta.queryField];
|
|
235
|
-
const queryValues = Array.isArray(this.apiMeta.queryValue)
|
|
236
|
-
? this.apiMeta.queryValue
|
|
237
|
-
: [this.apiMeta.queryValue];
|
|
238
|
-
const stored = this.storageService.getQueryValue(this.question.id); // single object
|
|
239
|
-
const queryParamsArray = [];
|
|
240
|
-
queryFields.forEach((field, index) => {
|
|
241
|
-
const valueKey = queryValues[index] ?? queryValues[0];
|
|
242
|
-
const value = stored[valueKey];
|
|
243
|
-
if (value !== undefined && value !== null) {
|
|
244
|
-
queryParamsArray.push(`${field}=${encodeURIComponent(value)}`);
|
|
245
|
-
}
|
|
246
|
-
});
|
|
247
|
-
const queryParams = queryParamsArray.join('&');
|
|
248
|
-
if (queryParams) {
|
|
249
|
-
finalGetCall += this.apiMeta.endpoint.includes('?') ? `&${queryParams}` : `?${queryParams}`;
|
|
250
|
-
}
|
|
251
|
-
}
|
|
252
|
-
this.dataService.apiResponse(finalGetCall)
|
|
253
|
-
.pipe(takeUntil(this.destroy$))
|
|
254
|
-
.subscribe({
|
|
255
|
-
next: (apiResponse) => {
|
|
256
|
-
this.isLoading = false;
|
|
257
|
-
this.processSearchResults(apiResponse, keyword);
|
|
258
|
-
},
|
|
259
|
-
error: (error) => {
|
|
260
|
-
this.isLoading = false;
|
|
261
|
-
this.noResult = true;
|
|
262
|
-
this.finalResults.set([]);
|
|
263
|
-
}
|
|
264
|
-
});
|
|
265
|
-
}
|
|
266
|
-
}
|
|
267
|
-
searchFromStaticData(keyword) {
|
|
268
|
-
const response = this.countryService.getCountryData(this.apiMeta.field);
|
|
269
|
-
this.processSearchResults(response, keyword);
|
|
270
|
-
}
|
|
271
|
-
processSearchResults(apiResponse, keyword) {
|
|
272
|
-
let response;
|
|
273
|
-
if (this.apiMeta.variable) {
|
|
274
|
-
response = this.dataService.getValue(apiResponse, this.apiMeta.variable);
|
|
275
|
-
}
|
|
276
|
-
else {
|
|
277
|
-
response = apiResponse;
|
|
278
|
-
}
|
|
279
|
-
if (!Array.isArray(response)) {
|
|
280
|
-
this.noResult = true;
|
|
281
|
-
this.finalResults.set([]);
|
|
282
|
-
return;
|
|
283
|
-
}
|
|
284
|
-
if (!keyword) {
|
|
285
|
-
// If no keyword, show all results
|
|
286
|
-
this.finalResults.set(response);
|
|
287
|
-
this.noResult = response.length === 0;
|
|
288
|
-
this.showSuggestion = true; // Ensure suggestions are visible
|
|
289
|
-
}
|
|
290
|
-
else {
|
|
291
|
-
// Filter results based on the keyword
|
|
292
|
-
const searchTerms = keyword.toLowerCase().split(" ").filter(term => term.length > 0);
|
|
293
|
-
const results = response.filter(obj => {
|
|
294
|
-
const combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);
|
|
295
|
-
return searchTerms.every(term => combinedValues.includes(term));
|
|
296
|
-
});
|
|
297
|
-
this.noResult = results.length === 0;
|
|
298
|
-
this.finalResults.set(results);
|
|
299
|
-
this.showSuggestion = true; // Show suggestions only if there are results
|
|
300
|
-
}
|
|
301
|
-
}
|
|
302
|
-
getNestedValue(obj, path) {
|
|
303
|
-
if (!obj || !path)
|
|
304
|
-
return '';
|
|
305
|
-
const processedPath = path.replace(/\[(\d+)\]/g, '.$1');
|
|
306
|
-
const parts = processedPath.split('.');
|
|
307
|
-
let current = obj;
|
|
308
|
-
for (const part of parts) {
|
|
309
|
-
if (!current || typeof current !== 'object')
|
|
310
|
-
return '';
|
|
311
|
-
current = current[part];
|
|
312
|
-
}
|
|
313
|
-
return current !== null && current !== undefined
|
|
314
|
-
? String(current).toLowerCase().trim()
|
|
315
|
-
: '';
|
|
316
|
-
}
|
|
317
|
-
getCombinedValuesFromColumns(obj, columns) {
|
|
318
|
-
if (!obj || !columns)
|
|
319
|
-
return '';
|
|
320
|
-
const columnArray = Array.isArray(columns) ? columns : [columns];
|
|
321
|
-
const values = [];
|
|
322
|
-
for (const column of columnArray) {
|
|
323
|
-
const value = this.getNestedValue(obj, column);
|
|
324
|
-
if (value) {
|
|
325
|
-
values.push(value);
|
|
326
|
-
}
|
|
327
|
-
}
|
|
328
|
-
return values.join(' ');
|
|
329
|
-
}
|
|
330
|
-
getValues(element, columns) {
|
|
331
|
-
const result = {};
|
|
332
|
-
if (!element || !columns)
|
|
333
|
-
return result;
|
|
334
|
-
const columnArray = Array.isArray(columns) ? columns : [columns];
|
|
335
|
-
columnArray.forEach((column) => {
|
|
336
|
-
if (!column)
|
|
337
|
-
return;
|
|
338
|
-
let tempElement = element;
|
|
339
|
-
const fields = column.split('.');
|
|
340
|
-
for (const field of fields) {
|
|
341
|
-
if (!tempElement)
|
|
342
|
-
break;
|
|
343
|
-
const splitFields = field.split('[');
|
|
344
|
-
if (splitFields.length === 1) {
|
|
345
|
-
tempElement = tempElement[field];
|
|
346
|
-
}
|
|
347
|
-
else {
|
|
348
|
-
const indexMatch = splitFields[1]?.match(/(\d+)\]/);
|
|
349
|
-
if (indexMatch) {
|
|
350
|
-
const index = Number(indexMatch[1]);
|
|
351
|
-
tempElement = tempElement[splitFields[0]]?.[index];
|
|
352
|
-
}
|
|
353
|
-
}
|
|
354
|
-
}
|
|
355
|
-
result[column] = tempElement || '';
|
|
356
|
-
});
|
|
357
|
-
return result;
|
|
358
|
-
}
|
|
359
|
-
getKeys(obj) {
|
|
360
|
-
return obj ? Object.keys(obj) : [];
|
|
361
|
-
}
|
|
362
|
-
clickItem(event) {
|
|
363
|
-
if (!this.apiMeta)
|
|
364
|
-
return;
|
|
365
|
-
this.parseApiMeta();
|
|
366
|
-
const values = this.getValues(event, this.SearchItem);
|
|
367
|
-
const displayValue = Object.values(values)
|
|
368
|
-
.filter(val => val)
|
|
369
|
-
.join(' | ');
|
|
370
|
-
if (this.mode !== 'edit') {
|
|
371
|
-
this.viewFilterName = displayValue;
|
|
372
|
-
}
|
|
373
|
-
this.filterName = displayValue;
|
|
374
|
-
this.ngOnChanges({});
|
|
375
|
-
const defaultField = this.apiMeta.defaultField ||
|
|
376
|
-
(Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);
|
|
377
|
-
const change = new ChangeWrapper();
|
|
378
|
-
change['isObject'] = this.apiMeta?.isObject;
|
|
379
|
-
change['fromQuestionId'] = this.id;
|
|
380
|
-
change['valueObj'] = event;
|
|
381
|
-
change['field'] = defaultField;
|
|
382
|
-
this.valueObj = event;
|
|
383
|
-
this.hideSuggestions();
|
|
384
|
-
this.searchValueChange.emit({
|
|
385
|
-
question: this.question,
|
|
386
|
-
value: change
|
|
387
|
-
});
|
|
388
|
-
}
|
|
389
|
-
inputValue(event, ques) {
|
|
390
|
-
this.filterName = event?.valueObj ? event?.valueObj?.trim() : this.filterName;
|
|
391
|
-
if (!ques?.singleFieldChange || !this.apiMeta)
|
|
392
|
-
return;
|
|
393
|
-
this.isSingleFieldChange = true;
|
|
394
|
-
this.parseApiMeta();
|
|
395
|
-
const defaultField = this.apiMeta.defaultField ||
|
|
396
|
-
(Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);
|
|
397
|
-
const change = new ChangeWrapper();
|
|
398
|
-
change['isObject'] = this.apiMeta?.isObject;
|
|
399
|
-
change['fromQuestionId'] = this.id;
|
|
400
|
-
change['valueObj'] = this.valueObj;
|
|
401
|
-
change['field'] = defaultField;
|
|
402
|
-
change['referenceField'] = this.question.referenceField;
|
|
403
|
-
this.filterName = this.viewFilterName;
|
|
404
|
-
this.searchValueChange.emit({
|
|
405
|
-
question: ques,
|
|
406
|
-
value: change
|
|
407
|
-
});
|
|
408
|
-
}
|
|
409
|
-
highlightText(text, searchTerm) {
|
|
410
|
-
if (!text || !searchTerm)
|
|
411
|
-
return text || '';
|
|
412
|
-
// Sanitize the input text
|
|
413
|
-
const sanitizedText = this.sanitizeHtml(text);
|
|
414
|
-
const terms = searchTerm.toLowerCase().split(' ').filter(term => term.length > 0);
|
|
415
|
-
if (terms.length === 0)
|
|
416
|
-
return sanitizedText;
|
|
417
|
-
// Escape special regex characters
|
|
418
|
-
const escapedTerms = terms.map(term => term.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'));
|
|
419
|
-
const regex = new RegExp(`(${escapedTerms.join('|')})`, 'gi');
|
|
420
|
-
return sanitizedText.replace(regex, '<span class="highlight">$1</span>');
|
|
421
|
-
}
|
|
422
|
-
sanitizeHtml(text) {
|
|
423
|
-
return text
|
|
424
|
-
.replace(/&/g, '&')
|
|
425
|
-
.replace(/</g, '<')
|
|
426
|
-
.replace(/>/g, '>')
|
|
427
|
-
.replace(/"/g, '"')
|
|
428
|
-
.replace(/'/g, ''');
|
|
429
|
-
}
|
|
430
|
-
removeValueEmit() {
|
|
431
|
-
this.clickItem(null);
|
|
432
|
-
}
|
|
433
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NxtSearchBox, deps: [{ token: i1.StorageService }, { token: i2.ChangeService }, { token: i3.DataService }, { token: i4.CountryService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
434
|
-
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: NxtSearchBox, isStandalone: true, selector: "nxt-search-box", inputs: { placeHolderText: "placeHolderText", question: "question", apiMeta: "apiMeta", id: "id", readOnly: "readOnly", mode: "mode", from: "from" }, outputs: { searchValueChange: "searchValueChange" }, host: { listeners: { "document:click": "onDocumentClick($event)", "document:keydown.escape": "onEscapeKey($event)" } }, viewQueries: [{ propertyName: "searchWrapper", first: true, predicate: ["searchWrapper"], descendants: true, static: true }], usesOnChanges: true, ngImport: i0, template: "<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\" #searchWrapper>\n <nxt-input \n [type]=\"'list'\" \n [mode]=\"mode\" \n [from]=\"from\"\n [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n [labelFont]=\"question?.font\" \n [label]=\"question?.questionText ? ((question?.id+'.questionText') | nxtCustomTranslate : question?.questionText): ''\" \n [labelColor]=\"question?.fontColor\"\n [labelSize]=\"question?.fontSize\" \n [inputValueSize]=\"question?.fontSize\" \n [labelWeight]=\"question?.fontWeight\"\n [inputWeight]=\"question?.fontWeight\" \n [showLabel]=\"question?.style?.showLabel\" \n inputBorder=\"none\" \n svgHeight=\"20px\"\n svgWidth=\"20px\" \n [placeholder]=\"placeHolderText\" \n [required]=\"question?.isOptional\" \n inputBgColor=\"#FAFAFA\"\n [inputId]=\"question?.trackingId\" \n [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"question?.iconLeftSrc\" \n (inputValue)=\"getSourceDataLocal($event); inputValue($event.value, $event.question)\" \n (onFocus)=\"onInputFocus()\"\n (onBlur)=\"onInputBlur()\"\n [showSuggestion]=\"showSuggestion\" \n ariaOwns=\"selectList\" \n ariaHasPopup=\"listbox\" \n [isLoading]=\"isLoading\" \n [question]=\"question\"\n (removeValueEmit)=\"removeValueEmit()\">\n </nxt-input>\n\n <div id=\"selectList\" class=\"suggestion-wrapper\" role=\"listbox\" *ngIf=\"showSuggestion\">\n <div *ngIf=\"finalResults().length > 0\" class=\"nxt-suggestions-container\">\n <div *ngFor=\"let item of finalResults(); trackBy: trackByFn\" \n (click)=\"clickItem(item)\" \n class=\"suggestion-item\" \n role=\"option\"\n tabindex=\"0\"\n (keydown.enter)=\"clickItem(item)\"\n (keydown.space)=\"clickItem(item)\">\n <div class=\"item-content\">\n <div class=\"item-details\">\n <div *ngIf=\"item.thumbnail\" class=\"thumbnail-wrapper\">\n <img [src]=\"item.thumbnail\" [alt]=\"item.name + ' avatar'\" />\n </div>\n <div class=\"item-text\">\n <!-- Primary text (first key) -->\n <ng-container *ngIf=\"getKeys(getValues(item, SearchItem)).length > 0\">\n <div class=\"item-name primary\" \n [innerHTML]=\"highlightText(getValues(item, SearchItem)[getKeys(getValues(item, SearchItem))[0]], searchKeyWord)\">\n </div>\n </ng-container>\n \n <!-- Secondary text (remaining keys) -->\n <div class=\"item-secondary\" *ngIf=\"getKeys(getValues(item, SearchItem)).length > 1\">\n <ng-container *ngFor=\"let key of getKeys(getValues(item, SearchItem)); let i = index\">\n <ng-container *ngIf=\"i > 0\">\n <span class=\"secondary-text\" \n [innerHTML]=\"highlightText(getValues(item, SearchItem)[key], searchKeyWord)\">\n </span>\n <span class=\"separator\" *ngIf=\"i < getKeys(getValues(item, SearchItem)).length - 1\"> | </span>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"noResult && searchKeyWord\" class=\"no-results\">\n <div class=\"no-results-content\">\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1\">\n <circle cx=\"11\" cy=\"11\" r=\"8\"></circle>\n <path d=\"m21 21-4.35-4.35\"></path>\n </svg>\n <p>{{ 'NO_RESULTS_FOUND' | nxtCustomTranslate : 'No results found for' }} \"{{searchKeyWord}}\"</p>\n </div>\n </div>\n </div>\n</div>", styles: [".search-wrapper{position:relative;width:100%}.suggestion-wrapper{position:absolute;top:100%;left:0;right:0;background:#fff;border:1px solid #e0e0e0;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 2px 8px #0000001a;z-index:1000;max-height:300px;overflow:hidden}.nxt-suggestions-container{max-height:300px;overflow-y:auto;min-width:100%}.suggestion-item{cursor:pointer;border-bottom:1px solid #f0f0f0;transition:background-color .2s ease;outline:none}.suggestion-item:hover,.suggestion-item:focus{background-color:#f5f5f5}.suggestion-item:last-child{border-bottom:none}.item-content{padding:5px}.item-details{display:flex;align-items:flex-start;gap:12px;text-align:left}.thumbnail-wrapper{flex-shrink:0}.thumbnail-wrapper img{width:40px;height:40px;border-radius:50%;object-fit:cover;border:1px solid #e0e0e0}.item-text{flex:1;min-width:0}.item-name{font-size:14px;font-weight:500;color:#333;line-height:1.4;margin-bottom:2px;word-wrap:break-word;overflow-wrap:break-word}.item-secondary{font-size:12px;color:#666;line-height:1.3;display:flex}.secondary-text{display:inline}.separator{color:#999;margin:0 3px}.no-results{padding:20px;text-align:center;color:#666}.no-results-content{display:flex;align-items:center;justify-content:center;gap:8px;flex-direction:column}.no-results-content svg{color:#d1d5db;margin-bottom:8px}.no-results-content p{color:#6b7280;margin:0;font-size:14px;font-weight:500}:host ::ng-deep .highlight{font-weight:700;display:inline}.nxt-suggestions-container::-webkit-scrollbar{width:6px}.nxt-suggestions-container::-webkit-scrollbar-track{background:#f1f1f1}.nxt-suggestions-container::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:3px}.nxt-suggestions-container::-webkit-scrollbar-thumb:hover{background:#a8a8a8}@media (max-width: 768px){.suggestion-wrapper,.nxt-suggestions-container{max-height:250px}.item-content{padding:10px 12px}.item-details{gap:8px}.thumbnail-wrapper img{width:32px;height:32px}.item-name{font-size:13px}.item-secondary{font-size:11px}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i5.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i5.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: FormsModule }, { kind: "component", type: NxtInput, selector: "nxt-input", inputs: ["label", "labelFont", "labelWeight", "inputWeight", "labelSize", "inputValueSize", "labelColor", "showLabel", "svgHeight", "svgWidth", "type", "inputIconRightSrc", "inputIconLeftSrc", "required", "minLength", "pattern", "errorMessages", "maxLength", "placeholder", "inputBgColor", "inputBorder", "placeholderColor", "placeholderFont", "placeholderWeight", "placeholderSize", "inputTextColor", "inputHeight", "inputWidth", "inputId", "inputBorderSize", "inputConfig", "confPassVal", "confPass", "mode", "value", "question", "showSuggestion", "ariaOwns", "ariaHasPopup", "isLoading", "options", "minDate", "maxDate", "rows", "from", "hyperLink", "size"], outputs: ["valueChange", "inputValue", "onBlur", "onFocus", "toggleEmit", "nativeInputRef", "removeValueEmit", "hyperlinkEmit"] }, { kind: "pipe", type: NxtCustomTranslatePipe, name: "nxtCustomTranslate" }] });
|
|
435
|
-
}
|
|
436
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NxtSearchBox, decorators: [{
|
|
437
|
-
type: Component,
|
|
438
|
-
args: [{ selector: 'nxt-search-box', standalone: true, imports: [
|
|
439
|
-
CommonModule,
|
|
440
|
-
FormsModule,
|
|
441
|
-
NxtInput,
|
|
442
|
-
NxtCustomTranslatePipe
|
|
443
|
-
], template: "<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\" #searchWrapper>\n <nxt-input \n [type]=\"'list'\" \n [mode]=\"mode\" \n [from]=\"from\"\n [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n [labelFont]=\"question?.font\" \n [label]=\"question?.questionText ? ((question?.id+'.questionText') | nxtCustomTranslate : question?.questionText): ''\" \n [labelColor]=\"question?.fontColor\"\n [labelSize]=\"question?.fontSize\" \n [inputValueSize]=\"question?.fontSize\" \n [labelWeight]=\"question?.fontWeight\"\n [inputWeight]=\"question?.fontWeight\" \n [showLabel]=\"question?.style?.showLabel\" \n inputBorder=\"none\" \n svgHeight=\"20px\"\n svgWidth=\"20px\" \n [placeholder]=\"placeHolderText\" \n [required]=\"question?.isOptional\" \n inputBgColor=\"#FAFAFA\"\n [inputId]=\"question?.trackingId\" \n [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"question?.iconLeftSrc\" \n (inputValue)=\"getSourceDataLocal($event); inputValue($event.value, $event.question)\" \n (onFocus)=\"onInputFocus()\"\n (onBlur)=\"onInputBlur()\"\n [showSuggestion]=\"showSuggestion\" \n ariaOwns=\"selectList\" \n ariaHasPopup=\"listbox\" \n [isLoading]=\"isLoading\" \n [question]=\"question\"\n (removeValueEmit)=\"removeValueEmit()\">\n </nxt-input>\n\n <div id=\"selectList\" class=\"suggestion-wrapper\" role=\"listbox\" *ngIf=\"showSuggestion\">\n <div *ngIf=\"finalResults().length > 0\" class=\"nxt-suggestions-container\">\n <div *ngFor=\"let item of finalResults(); trackBy: trackByFn\" \n (click)=\"clickItem(item)\" \n class=\"suggestion-item\" \n role=\"option\"\n tabindex=\"0\"\n (keydown.enter)=\"clickItem(item)\"\n (keydown.space)=\"clickItem(item)\">\n <div class=\"item-content\">\n <div class=\"item-details\">\n <div *ngIf=\"item.thumbnail\" class=\"thumbnail-wrapper\">\n <img [src]=\"item.thumbnail\" [alt]=\"item.name + ' avatar'\" />\n </div>\n <div class=\"item-text\">\n <!-- Primary text (first key) -->\n <ng-container *ngIf=\"getKeys(getValues(item, SearchItem)).length > 0\">\n <div class=\"item-name primary\" \n [innerHTML]=\"highlightText(getValues(item, SearchItem)[getKeys(getValues(item, SearchItem))[0]], searchKeyWord)\">\n </div>\n </ng-container>\n \n <!-- Secondary text (remaining keys) -->\n <div class=\"item-secondary\" *ngIf=\"getKeys(getValues(item, SearchItem)).length > 1\">\n <ng-container *ngFor=\"let key of getKeys(getValues(item, SearchItem)); let i = index\">\n <ng-container *ngIf=\"i > 0\">\n <span class=\"secondary-text\" \n [innerHTML]=\"highlightText(getValues(item, SearchItem)[key], searchKeyWord)\">\n </span>\n <span class=\"separator\" *ngIf=\"i < getKeys(getValues(item, SearchItem)).length - 1\"> | </span>\n </ng-container>\n </ng-container>\n </div>\n </div>\n </div>\n </div>\n </div>\n </div>\n\n <div *ngIf=\"noResult && searchKeyWord\" class=\"no-results\">\n <div class=\"no-results-content\">\n <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1\">\n <circle cx=\"11\" cy=\"11\" r=\"8\"></circle>\n <path d=\"m21 21-4.35-4.35\"></path>\n </svg>\n <p>{{ 'NO_RESULTS_FOUND' | nxtCustomTranslate : 'No results found for' }} \"{{searchKeyWord}}\"</p>\n </div>\n </div>\n </div>\n</div>", styles: [".search-wrapper{position:relative;width:100%}.suggestion-wrapper{position:absolute;top:100%;left:0;right:0;background:#fff;border:1px solid #e0e0e0;border-top:none;border-radius:0 0 4px 4px;box-shadow:0 2px 8px #0000001a;z-index:1000;max-height:300px;overflow:hidden}.nxt-suggestions-container{max-height:300px;overflow-y:auto;min-width:100%}.suggestion-item{cursor:pointer;border-bottom:1px solid #f0f0f0;transition:background-color .2s ease;outline:none}.suggestion-item:hover,.suggestion-item:focus{background-color:#f5f5f5}.suggestion-item:last-child{border-bottom:none}.item-content{padding:5px}.item-details{display:flex;align-items:flex-start;gap:12px;text-align:left}.thumbnail-wrapper{flex-shrink:0}.thumbnail-wrapper img{width:40px;height:40px;border-radius:50%;object-fit:cover;border:1px solid #e0e0e0}.item-text{flex:1;min-width:0}.item-name{font-size:14px;font-weight:500;color:#333;line-height:1.4;margin-bottom:2px;word-wrap:break-word;overflow-wrap:break-word}.item-secondary{font-size:12px;color:#666;line-height:1.3;display:flex}.secondary-text{display:inline}.separator{color:#999;margin:0 3px}.no-results{padding:20px;text-align:center;color:#666}.no-results-content{display:flex;align-items:center;justify-content:center;gap:8px;flex-direction:column}.no-results-content svg{color:#d1d5db;margin-bottom:8px}.no-results-content p{color:#6b7280;margin:0;font-size:14px;font-weight:500}:host ::ng-deep .highlight{font-weight:700;display:inline}.nxt-suggestions-container::-webkit-scrollbar{width:6px}.nxt-suggestions-container::-webkit-scrollbar-track{background:#f1f1f1}.nxt-suggestions-container::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:3px}.nxt-suggestions-container::-webkit-scrollbar-thumb:hover{background:#a8a8a8}@media (max-width: 768px){.suggestion-wrapper,.nxt-suggestions-container{max-height:250px}.item-content{padding:10px 12px}.item-details{gap:8px}.thumbnail-wrapper img{width:32px;height:32px}.item-name{font-size:13px}.item-secondary{font-size:11px}}\n"] }]
|
|
444
|
-
}], ctorParameters: () => [{ type: i1.StorageService }, { type: i2.ChangeService }, { type: i3.DataService }, { type: i4.CountryService }, { type: i0.ElementRef }], propDecorators: { placeHolderText: [{
|
|
445
|
-
type: Input
|
|
446
|
-
}], question: [{
|
|
447
|
-
type: Input
|
|
448
|
-
}], apiMeta: [{
|
|
449
|
-
type: Input
|
|
450
|
-
}], id: [{
|
|
451
|
-
type: Input
|
|
452
|
-
}], readOnly: [{
|
|
453
|
-
type: Input
|
|
454
|
-
}], mode: [{
|
|
455
|
-
type: Input
|
|
456
|
-
}], from: [{
|
|
457
|
-
type: Input
|
|
458
|
-
}], searchValueChange: [{
|
|
459
|
-
type: Output
|
|
460
|
-
}], searchWrapper: [{
|
|
461
|
-
type: ViewChild,
|
|
462
|
-
args: ['searchWrapper', { static: true }]
|
|
463
|
-
}], onDocumentClick: [{
|
|
464
|
-
type: HostListener,
|
|
465
|
-
args: ['document:click', ['$event']]
|
|
466
|
-
}], onEscapeKey: [{
|
|
467
|
-
type: HostListener,
|
|
468
|
-
args: ['document:keydown.escape', ['$event']]
|
|
469
|
-
}] } });
|
|
470
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"search-box.component.js","sourceRoot":"","sources":["../../../../../../projects/nxt-app/src/lib/components/search-box/search-box.component.ts","../../../../../../projects/nxt-app/src/lib/components/search-box/search-box.component.html"],"names":[],"mappings":"AAAA,OAAO,EAAE,SAAS,EAAgC,KAAK,EAAE,MAAM,EAAE,YAAY,EAAE,SAAS,EAA6B,MAAM,EAAmC,YAAY,EAAC,MAAM,eAAe,CAAC;AACjM,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAC1D,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AAC5D,OAAO,EAAE,OAAO,EAAgB,YAAY,EAAE,oBAAoB,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;AAK5F,OAAO,EAAE,sBAAsB,EAAE,MAAM,kCAAkC,CAAC;;;;;;;AAc1E,MAAM,OAAO,YAAY;IAoCH;IAAuC;IAAqC;IAAkC;IAAwC;IAnCjK,eAAe,GAAW,EAAE,CAAC;IAC7B,QAAQ,CAAM;IACd,OAAO,CAAM;IACb,EAAE,GAAW,EAAE,CAAC;IAChB,QAAQ,GAAG,KAAK,CAAC;IACjB,IAAI,GAA8B,MAAM,CAAC;IACzC,IAAI,CAAM;IAET,iBAAiB,GAAsB,IAAI,YAAY,EAAO,CAAC;IAC3B,aAAa,CAAc;IAEzE,UAAU,CAAM;IACT,YAAY,GAA0B,MAAM,CAAC,EAAE,CAAC,CAAC;IACjD,aAAa,GAAW,EAAE,CAAC;IAC3B,SAAS,CAAM;IACf,UAAU,GAAG,KAAK,CAAC;IACnB,QAAQ,GAAG,KAAK,CAAC;IACjB,cAAc,GAAG,KAAK,CAAC;IACvB,SAAS,GAAG,KAAK,CAAC;IAClB,cAAc,GAAW,EAAE,CAAC;IAC5B,mBAAmB,GAAG,KAAK,CAAC;IAC5B,QAAQ,CAAM;IACd,YAAY,GAAG,KAAK,CAAC;IACrB,gBAAgB,GAAG,CAAC,CAAC;IACrB,UAAU,GAAE,EAAE,CAAA;IAEb,QAAQ,GAAG,IAAI,OAAO,EAAQ,CAAC;IAC/B,aAAa,GAAG,IAAI,OAAO,EAAU,CAAC;IAC9C,YAAY,CAAe;IAE3B,gDAAgD;IAChD,SAAS,GAAyB,CAAC,KAAa,EAAE,IAAS,EAAE,EAAE;QAC7D,OAAO,IAAI,CAAC,EAAE,IAAI,IAAI,CAAC,IAAI,IAAI,KAAK,CAAC;IACvC,CAAC,CAAC;IAEF,YAAoB,cAA8B,EAAS,aAA4B,EAAS,WAAwB,EAAU,cAA8B,EAAU,UAAsB;QAA5K,mBAAc,GAAd,cAAc,CAAgB;QAAS,kBAAa,GAAb,aAAa,CAAe;QAAS,gBAAW,GAAX,WAAW,CAAa;QAAU,mBAAc,GAAd,cAAc,CAAgB;QAAU,eAAU,GAAV,UAAU,CAAY;QAC9L,IAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,IAAI,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAC,CAAC;YAC3G,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAA;QACtC,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAA;QACxC,CAAC;QACD,yBAAyB;QACzB,IAAI,CAAC,aAAa;aACf,IAAI,CACH,YAAY,CAAC,GAAG,CAAC,EACjB,oBAAoB,EAAE,EACtB,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CACzB;aACA,SAAS,CAAC,OAAO,CAAC,EAAE;YACnB,IAAG,IAAI,CAAC,gBAAgB,GAAG,CAAC;gBAAE,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAC5D,CAAC,CAAC,CAAC;IACP,CAAC;IAED,QAAQ;QACN,IAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,IAAI,OAAO,IAAI,CAAC,QAAQ,EAAE,KAAK,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAC,CAAC;YAC3G,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,CAAA;QACtC,CAAC;aAAM,IAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAC,CAAC;YAC7C,IAAI,CAAC,UAAU,GAAG,EAAE,CAAA;YACpB,IAAI,CAAC,cAAc,GAAG,EAAE,CAAA;QAC1B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAA;YACtC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAA;QAC5C,CAAC;QACD,IAAI,CAAC,YAAY,EAAE,CAAC;QACpB,IAAG,IAAI,CAAC,QAAQ,EAAE,OAAO,IAAI,SAAS,EAAE,CAAC;YACvC,IAAI,aAAa,GAAQ,IAAI,CAAC,QAAQ,EAAE,OAAO,CAAC;YAChD,4BAA4B;YAC5B,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,gBAAgB,CAAC,SAAS,CAC/D,CAAC,WAAW,EAAE,EAAE;gBACd,IAAG,WAAW,IAAI,SAAS,EAAE,CAAC;oBAC5B,IAAG,WAAW,CAAC,cAAc,IAAI,WAAW,CAAC,QAAQ,KAAK,SAAS,IAAI,CAAE,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,gBAAgB,CAAC,CAAA,CAAC,CAAC,aAAa,CAAC,gBAAgB,CAAC,QAAQ,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,cAAc,KAAK,aAAa,CAAC,gBAAgB,CAAC,EAAE,CAAC;wBAC7P,IAAG,KAAK,CAAC,OAAO,CAAC,aAAa,CAAC,UAAU,CAAC,CAAA,CAAC,CAAC,aAAa,CAAC,UAAU,CAAC,QAAQ,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,cAAc,KAAK,aAAa,CAAC,UAAU,EAAC,CAAC;4BACnK,IAAI,CAAC,cAAc,CAAC,eAAe,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,EAAC,WAAW,CAAC,cAAc,EAAC,WAAW,CAAC,QAAQ,CAAC,CAAA;wBACvG,CAAC;oBACH,CAAC;oBACD,IAAI,CAAC,aAAa,CAAC,aAAa,CAAC,WAAW,CAAC,cAAc,CAAC,CAAC;gBAC/D,CAAC;YACH,CAAC,CACF,CAAC;QACJ,CAAC;IACH,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAG,OAAO,CAAC,UAAU,CAAC,EAAC,CAAC;YACtB,gHAAgH;YAChH,yCAAyC;YACzC,WAAW;YACX,2CAA2C;YAC3C,IAAI;QACN,CAAC;QACD,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YACpB,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC;QAC3C,CAAC;QACD,IAAI,OAAO,CAAC,SAAS,CAAC,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACvC,IAAI,CAAC,YAAY,EAAE,CAAC;QACtB,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,QAAQ,CAAC,IAAI,EAAE,CAAC;QACrB,IAAI,CAAC,QAAQ,CAAC,QAAQ,EAAE,CAAC;QACzB,IAAI,CAAC,mBAAmB,EAAE,CAAC;IAC7B,CAAC;IAED,6CAA6C;IAE7C,eAAe,CAAC,KAAY;QAC1B,IAAI,CAAC,IAAI,CAAC,UAAU,CAAC,aAAa,CAAC,QAAQ,CAAC,KAAK,CAAC,MAAc,CAAC,EAAE,CAAC;YAClE,IAAI,CAAC,eAAe,EAAE,CAAC;QACzB,CAAC;IACH,CAAC;IAED,yCAAyC;IAEzC,WAAW,CAAC,KAAoB;QAC9B,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAEO,YAAY;QAClB,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,IAAI,CAAC,OAAO,GAAG,OAAO,IAAI,CAAC,OAAO,KAAK,QAAQ,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,OAAO,IAAI,IAAI,CAAC,CAAC;YAClG,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,OAAO,CAAC,KAAK,IAAI,EAAE,CAAC;QAC7C,CAAC;IACH,CAAC;IAEO,mBAAmB;QACzB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC1B,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAC5B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QACtB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;QACvB,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;QAC1B,IAAI,CAAC,gBAAgB,GAAG,CAAC,CAAC;IAC5B,CAAC;IAEO,eAAe;QACrB,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;QAC5B,IAAI,CAAC,YAAY,GAAG,KAAK,CAAC;IAC5B,CAAC;IAED,YAAY;QACV,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC;QACzB,0DAA0D;QAC1D,IAAI,CAAC,IAAI,CAAC,aAAa,EAAE,CAAC;YACxB,0BAA0B;QAC5B,CAAC;aAAM,IAAI,IAAI,CAAC,YAAY,EAAE,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YAC1C,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC;QAC7B,CAAC;IACH,CAAC;IAED,WAAW;QACT,2DAA2D;QAC3D,gDAAgD;IAClD,CAAC;IAED,SAAS;QACP,sDAAsD;QACtD,IAAI,CAAC,eAAe,EAAE,CAAC;IACzB,CAAC;IAED,kBAAkB,CAAC,KAAU;QAC3B,IAAI,CAAC,KAAK,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC;YAC5B,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,OAAO;QACT,CAAC;QAED,MAAM,WAAW,GAAI,KAAK,CAAC,KAAK,EAAE,QAAQ,EAAE,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,EAAE,QAAQ,EAAE,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC;QACxI,MAAM,aAAa,GAAG,WAAW,CAAC,MAAM,CAAC;QAEzC,mCAAmC;QACnC,IAAI,CAAC,aAAa,GAAG,WAAW,CAAC;QAEjC,iDAAiD;QACjD,IAAI,aAAa,GAAG,CAAC,IAAI,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC;YACnD,IAAI,CAAC,cAAc,GAAG,CAAC,KAAK,CAAC,QAAQ,EAAE,iBAAiB,CAAC;YACzD,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC;YACxB,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,mDAAmD;YACnD,IAAI,CAAC,gBAAgB,GAAG,aAAa,CAAC;YACtC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QAEvC,CAAC;aAAM,IAAI,aAAa,IAAI,CAAC,EAAE,CAAC;YAC9B,yEAAyE;YACzE,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;YACjC,IAAI,CAAC,cAAc,GAAG,KAAK,CAAC;YAC5B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;YACtB,IAAI,CAAC,gBAAgB,GAAG,aAAa,CAAC;YACtC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;YACrC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC5B,CAAC;IACH,CAAC;IAEO,aAAa,CAAC,OAAe;QACnC,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAE1B,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,CAAC;YAC/B,IAAI,CAAC,aAAa,CAAC,OAAO,CAAC,CAAC;QAC9B,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,oBAAoB,CAAC,OAAO,CAAC,CAAC;QACrC,CAAC;IACH,CAAC;IAEO,aAAa,CAAC,OAAe;QACnC,IAAI,CAAC,SAAS,GAAG,IAAI,CAAC;QACtB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QAC1B,IAAG,CAAC,IAAI,CAAC,OAAO,CAAC,QAAQ,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAC,EAAE,CAAC;YACjE,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;YACvB,IAAI,CAAC,oBAAoB,CAAC,IAAI,CAAC,QAAQ,EAAE,KAAK,EAAE,OAAO,CAAC,CAAC;QAC3D,CAAC;aAAM,CAAC;YACN,IAAI,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC;YACzC,mCAAmC;YACnC,IAAG,IAAI,CAAC,OAAO,EAAE,cAAc,EAAC,CAAC;gBAC/B,MAAM,YAAY,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,cAAc,CAAC,CAAC;gBACjE,MAAM,MAAM,GAAG,YAAY,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,cAAc,CAAC,CAAC,CAAC,sBAAsB;gBACjH,4BAA4B;gBAC5B,MAAM,WAAW,GAAG,MAAM;qBACvB,GAAG,CAAC,KAAK,CAAC,EAAE,CAAC,GAAG,KAAK,IAAI,IAAI,CAAC,aAAa,EAAE,CAAC;qBAC9C,IAAI,CAAC,GAAG,CAAC,CAAC;gBACb,6CAA6C;gBAC7C,IAAG,WAAW,IAAI,WAAW,KAAK,EAAE,EAAC,CAAC;oBACpC,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,EAAE,CAAC;wBACxC,YAAY,IAAI,IAAI,WAAW,EAAE,CAAC;oBACpC,CAAC;yBAAM,CAAC;wBACN,YAAY,IAAI,IAAI,WAAW,EAAE,CAAC;oBACpC,CAAC;gBACH,CAAC;YACH,CAAC;YACD,IAAI,IAAI,CAAC,OAAO,EAAE,UAAU,IAAI,IAAI,CAAC,OAAO,EAAE,UAAU,EAAE,CAAC;gBACzD,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC;oBACxD,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU;oBACzB,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;gBAE9B,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC;oBACxD,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU;oBACzB,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;gBAE9B,MAAM,MAAM,GAAG,IAAI,CAAC,cAAc,CAAC,aAAa,CAAC,IAAI,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,CAAC,gBAAgB;gBAEpF,MAAM,gBAAgB,GAAa,EAAE,CAAC;gBAEtC,WAAW,CAAC,OAAO,CAAC,CAAC,KAAK,EAAE,KAAK,EAAE,EAAE;oBACnC,MAAM,QAAQ,GAAG,WAAW,CAAC,KAAK,CAAC,IAAI,WAAW,CAAC,CAAC,CAAC,CAAC;oBACtD,MAAM,KAAK,GAAG,MAAM,CAAC,QAAQ,CAAC,CAAC;oBAC/B,IAAI,KAAK,KAAK,SAAS,IAAI,KAAK,KAAK,IAAI,EAAE,CAAC;wBAC1C,gBAAgB,CAAC,IAAI,CAAC,GAAG,KAAK,IAAI,kBAAkB,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC;oBACjE,CAAC;gBACH,CAAC,CAAC,CAAC;gBAEH,MAAM,WAAW,GAAG,gBAAgB,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;gBAE/C,IAAI,WAAW,EAAE,CAAC;oBAChB,YAAY,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,IAAI,WAAW,EAAE,CAAC,CAAC,CAAC,IAAI,WAAW,EAAE,CAAC;gBAC9F,CAAC;YACH,CAAC;YACD,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,YAAY,CAAC;iBACzC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;iBAC9B,SAAS,CAAC;gBACT,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE;oBACpB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;oBACvB,IAAI,CAAC,oBAAoB,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;gBAClD,CAAC;gBACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;oBACf,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;oBACvB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;oBACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;gBAC5B,CAAC;aACF,CAAC,CAAC;QACL,CAAC;IACH,CAAC;IAEO,oBAAoB,CAAC,OAAe;QAC1C,MAAM,QAAQ,GAAG,IAAI,CAAC,cAAc,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QACxE,IAAI,CAAC,oBAAoB,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;IAC/C,CAAC;IAEO,oBAAoB,CAAC,WAAgB,EAAE,OAAe;QAC5D,IAAI,QAAQ,CAAC;QAEb,IAAI,IAAI,CAAC,OAAO,CAAC,QAAQ,EAAE,CAAC;YAC1B,QAAQ,GAAG,IAAI,CAAC,WAAW,CAAC,QAAQ,CAAC,WAAW,EAAE,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;QAC3E,CAAC;aAAM,CAAC;YACN,QAAQ,GAAG,WAAW,CAAC;QACzB,CAAC;QAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,EAAE,CAAC;YAC7B,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;YACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC1B,OAAO;QACT,CAAC;QAED,IAAI,CAAC,OAAO,EAAE,CAAC;YACb,kCAAkC;YAClC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;YAChC,IAAI,CAAC,QAAQ,GAAG,QAAQ,CAAC,MAAM,KAAK,CAAC,CAAC;YACtC,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,CAAC,iCAAiC;QAC/D,CAAC;aAAM,CAAC;YACN,sCAAsC;YACtC,MAAM,WAAW,GAAG,OAAO,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;YACrF,MAAM,OAAO,GAAG,QAAQ,CAAC,MAAM,CAAC,GAAG,CAAC,EAAE;gBACpC,MAAM,cAAc,GAAG,IAAI,CAAC,4BAA4B,CAAC,GAAG,EAAE,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;gBAClF,OAAO,WAAW,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC,CAAC;YAClE,CAAC,CAAC,CAAC;YAEH,IAAI,CAAC,QAAQ,GAAG,OAAO,CAAC,MAAM,KAAK,CAAC,CAAC;YACrC,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,OAAO,CAAC,CAAC;YAC/B,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,CAAC,6CAA6C;QAC3E,CAAC;IACH,CAAC;IAEO,cAAc,CAAC,GAAQ,EAAE,IAAY;QAC3C,IAAI,CAAC,GAAG,IAAI,CAAC,IAAI;YAAE,OAAO,EAAE,CAAC;QAE7B,MAAM,aAAa,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY,EAAE,KAAK,CAAC,CAAC;QACxD,MAAM,KAAK,GAAG,aAAa,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;QACvC,IAAI,OAAO,GAAG,GAAG,CAAC;QAElB,KAAK,MAAM,IAAI,IAAI,KAAK,EAAE,CAAC;YACzB,IAAI,CAAC,OAAO,IAAI,OAAO,OAAO,KAAK,QAAQ;gBAAE,OAAO,EAAE,CAAC;YACvD,OAAO,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;QAC1B,CAAC;QAED,OAAO,OAAO,KAAK,IAAI,IAAI,OAAO,KAAK,SAAS;YAC9C,CAAC,CAAC,MAAM,CAAC,OAAO,CAAC,CAAC,WAAW,EAAE,CAAC,IAAI,EAAE;YACtC,CAAC,CAAC,EAAE,CAAC;IACT,CAAC;IAEO,4BAA4B,CAAC,GAAQ,EAAE,OAAiB;QAC9D,IAAI,CAAC,GAAG,IAAI,CAAC,OAAO;YAAE,OAAO,EAAE,CAAC;QAEhC,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;QACjE,MAAM,MAAM,GAAa,EAAE,CAAC;QAE5B,KAAK,MAAM,MAAM,IAAI,WAAW,EAAE,CAAC;YACjC,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,GAAG,EAAE,MAAM,CAAC,CAAC;YAC/C,IAAI,KAAK,EAAE,CAAC;gBACV,MAAM,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;YACrB,CAAC;QACH,CAAC;QAED,OAAO,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;IAC1B,CAAC;IAEM,SAAS,CAAC,OAAY,EAAE,OAAY;QACzC,MAAM,MAAM,GAAQ,EAAE,CAAC;QACvB,IAAI,CAAC,OAAO,IAAI,CAAC,OAAO;YAAE,OAAO,MAAM,CAAC;QAExC,MAAM,WAAW,GAAG,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC;QAEjE,WAAW,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,EAAE;YAC7B,IAAI,CAAC,MAAM;gBAAE,OAAO;YAEpB,IAAI,WAAW,GAAG,OAAO,CAAC;YAC1B,MAAM,MAAM,GAAG,MAAM,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;YAEjC,KAAK,MAAM,KAAK,IAAI,MAAM,EAAE,CAAC;gBAC3B,IAAI,CAAC,WAAW;oBAAE,MAAM;gBAExB,MAAM,WAAW,GAAG,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC;gBACrC,IAAI,WAAW,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;oBAC7B,WAAW,GAAG,WAAW,CAAC,KAAK,CAAC,CAAC;gBACnC,CAAC;qBAAM,CAAC;oBACN,MAAM,UAAU,GAAG,WAAW,CAAC,CAAC,CAAC,EAAE,KAAK,CAAC,SAAS,CAAC,CAAC;oBACpD,IAAI,UAAU,EAAE,CAAC;wBACf,MAAM,KAAK,GAAG,MAAM,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC;wBACpC,WAAW,GAAG,WAAW,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,KAAK,CAAC,CAAC;oBACrD,CAAC;gBACH,CAAC;YACH,CAAC;YAED,MAAM,CAAC,MAAM,CAAC,GAAG,WAAW,IAAI,EAAE,CAAC;QACrC,CAAC,CAAC,CAAC;QAEH,OAAO,MAAM,CAAC;IAChB,CAAC;IAED,OAAO,CAAC,GAAQ;QACd,OAAO,GAAG,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;IACrC,CAAC;IAEM,SAAS,CAAC,KAAU;QACzB,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAE1B,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,KAAK,EAAE,IAAI,CAAC,UAAU,CAAC,CAAC;QACtD,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC,MAAM,CAAC;aACvC,MAAM,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC;aAClB,IAAI,CAAC,KAAK,CAAC,CAAC;QAEf,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,EAAE,CAAC;YACzB,IAAI,CAAC,cAAc,GAAG,YAAY,CAAC;QACrC,CAAC;QACD,IAAI,CAAC,UAAU,GAAG,YAAY,CAAC;QAC/B,IAAI,CAAC,WAAW,CAAC,EAAE,CAAC,CAAA;QAEpB,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY;YAC5C,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAEvF,MAAM,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC;QACnC,MAAM,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAA;QAC3C,MAAM,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,EAAE,CAAC;QACnC,MAAM,CAAC,UAAU,CAAC,GAAG,KAAK,CAAC;QAC3B,MAAM,CAAC,OAAO,CAAC,GAAG,YAAY,CAAC;QAE/B,IAAI,CAAC,QAAQ,GAAG,KAAK,CAAC;QAEtB,IAAI,CAAC,eAAe,EAAE,CAAC;QAEvB,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;YAC1B,QAAQ,EAAE,IAAI,CAAC,QAAQ;YACvB,KAAK,EAAE,MAAM;SACd,CAAC,CAAC;IACL,CAAC;IAED,UAAU,CAAC,KAAU,EAAE,IAAS;QAC9B,IAAI,CAAC,UAAU,GAAG,KAAK,EAAE,QAAQ,CAAC,CAAC,CAAC,KAAK,EAAE,QAAQ,EAAE,IAAI,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,UAAU,CAAA;QAC7E,IAAI,CAAC,IAAI,EAAE,iBAAiB,IAAI,CAAC,IAAI,CAAC,OAAO;YAAE,OAAO;QAEtD,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;QAChC,IAAI,CAAC,YAAY,EAAE,CAAC;QAEpB,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,YAAY;YAC5C,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;QAEvF,MAAM,MAAM,GAAG,IAAI,aAAa,EAAE,CAAC;QACnC,MAAM,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,OAAO,EAAE,QAAQ,CAAA;QAC3C,MAAM,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,EAAE,CAAC;QACnC,MAAM,CAAC,UAAU,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC;QACnC,MAAM,CAAC,OAAO,CAAC,GAAG,YAAY,CAAC;QAC/B,MAAM,CAAC,gBAAgB,CAAC,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC;QAExD,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,cAAc,CAAC;QAEtC,IAAI,CAAC,iBAAiB,CAAC,IAAI,CAAC;YAC1B,QAAQ,EAAE,IAAI;YACd,KAAK,EAAE,MAAM;SACd,CAAC,CAAC;IACL,CAAC;IAED,aAAa,CAAC,IAAY,EAAE,UAAkB;QAC5C,IAAI,CAAC,IAAI,IAAI,CAAC,UAAU;YAAE,OAAO,IAAI,IAAI,EAAE,CAAC;QAE5C,0BAA0B;QAC1B,MAAM,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC,IAAI,CAAC,CAAC;QAC9C,MAAM,KAAK,GAAG,UAAU,CAAC,WAAW,EAAE,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,EAAE,CAAC,IAAI,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC;QAElF,IAAI,KAAK,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,aAAa,CAAC;QAE7C,kCAAkC;QAClC,MAAM,YAAY,GAAG,KAAK,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE,CACpC,IAAI,CAAC,OAAO,CAAC,qBAAqB,EAAE,MAAM,CAAC,CAC5C,CAAC;QAEF,MAAM,KAAK,GAAG,IAAI,MAAM,CAAC,IAAI,YAAY,CAAC,IAAI,CAAC,GAAG,CAAC,GAAG,EAAE,IAAI,CAAC,CAAC;QAC9D,OAAO,aAAa,CAAC,OAAO,CAAC,KAAK,EAAE,mCAAmC,CAAC,CAAC;IAC3E,CAAC;IAEO,YAAY,CAAC,IAAY;QAC/B,OAAO,IAAI;aACR,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC;aACtB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;aACrB,OAAO,CAAC,IAAI,EAAE,MAAM,CAAC;aACrB,OAAO,CAAC,IAAI,EAAE,QAAQ,CAAC;aACvB,OAAO,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;IAC5B,CAAC;IACD,eAAe;QACb,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,CAAA;IACtB,CAAC;wGAtdU,YAAY;4FAAZ,YAAY,giBCxBzB,uxHAoFM,+gEDpEF,YAAY,+PACZ,WAAW,+BACX,QAAQ,w0BACR,sBAAsB;;4FAKb,YAAY;kBAZxB,SAAS;+BACE,gBAAgB,cACd,IAAI,WACP;wBACP,YAAY;wBACZ,WAAW;wBACX,QAAQ;wBACR,sBAAsB;qBACvB;+LAKQ,eAAe;sBAAvB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,EAAE;sBAAV,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBAEI,iBAAiB;sBAA1B,MAAM;gBACuC,aAAa;sBAA1D,SAAS;uBAAC,eAAe,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBAiG5C,eAAe;sBADd,YAAY;uBAAC,gBAAgB,EAAE,CAAC,QAAQ,CAAC;gBAS1C,WAAW;sBADV,YAAY;uBAAC,yBAAyB,EAAE,CAAC,QAAQ,CAAC","sourcesContent":["import { Component, OnInit, OnDestroy, OnChanges, Input, Output, EventEmitter, ViewChild, ElementRef, SimpleChanges, signal, WritableSignal, TrackByFunction, HostListener} from \"@angular/core\";\nimport { ChangeWrapper } from \"../../model/changeWrapper\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { NxtInput } from \"../nxt-input/nxt-input.component\";\nimport { Subject, Subscription, debounceTime, distinctUntilChanged, takeUntil } from \"rxjs\";\nimport { StorageService } from \"../../services/storage/storage.service\";\nimport { ChangeService } from \"../../services/change/change.service\";\nimport { DataService } from \"../../services/data/data.service\";\nimport { CountryService } from \"../../services/country/country.service\";\nimport { NxtCustomTranslatePipe } from \"../../pipe/custom-translate.pipe\";\n\n@Component({\n  selector: 'nxt-search-box',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    NxtInput,\n    NxtCustomTranslatePipe\n  ],\n  templateUrl: './search-box.component.html',\n  styleUrls: ['./search-box.component.css']\n})\nexport class NxtSearchBox implements OnInit, OnChanges, OnDestroy {\n  @Input() placeHolderText: string = '';\n  @Input() question: any;\n  @Input() apiMeta: any;\n  @Input() id: string = '';\n  @Input() readOnly = false;\n  @Input() mode: 'view' | 'edit' | 'print' = 'edit';\n  @Input() from: any;\n  \n  @Output() searchValueChange: EventEmitter<any> = new EventEmitter<any>();\n  @ViewChild('searchWrapper', { static: true }) searchWrapper!: ElementRef;\n\n  SearchItem: any;\n  public finalResults: WritableSignal<any[]> = signal([]);\n  public searchKeyWord: string = '';\n  public newResult: any;\n  public showResult = false;\n  public noResult = false;\n  public showSuggestion = false;\n  public isLoading = false;\n  public viewFilterName: string = '';\n  public isSingleFieldChange = false;\n  public valueObj: any;\n  public inputFocused = false;\n  public lastSearchLength = 0;\n  public filterName =''\n\n  private destroy$ = new Subject<void>();\n  private searchSubject = new Subject<string>();\n  subscription: Subscription;\n\n  // TrackBy function for performance optimization\n  trackByFn: TrackByFunction<any> = (index: number, item: any) => {\n    return item.id || item.name || index;\n  };\n\n  constructor(private storageService: StorageService,private changeService: ChangeService,private dataService: DataService, private countryService: CountryService, private elementRef: ElementRef) {\n    if(this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)){\n      this.clickItem(this.question?.input)\n    } else {\n      this.filterName = this.question?.input\n    }    \n    // Setup debounced search\n    this.searchSubject\n      .pipe(\n        debounceTime(300),\n        distinctUntilChanged(),\n        takeUntil(this.destroy$)\n      )\n      .subscribe(keyword => {\n        if(this.lastSearchLength > 2) this.performSearch(keyword);\n      });\n  }\n\n  ngOnInit(): void {\n    if(this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)){\n      this.clickItem(this.question?.input)\n    } else if(Array.isArray(this.question?.input)){\n      this.filterName = ''\n      this.viewFilterName = ''\n    } else {\n      this.filterName = this.question?.input\n      this.viewFilterName = this.question?.input\n    }\n    this.parseApiMeta();\n    if(this.question?.subText != undefined) {\n      let dependencyObj: any = this.question?.subText;\n      // Subscribe for the changes\n      this.subscription = this.changeService.changeAnnounced$.subscribe(\n        (changeValue) => {\n          if(changeValue != undefined) {\n            if(changeValue.fromQuestionId && changeValue.valueObj !== undefined && ( Array.isArray(dependencyObj.sourceQuestionId)? dependencyObj.sourceQuestionId.includes(changeValue.fromQuestionId) : changeValue.fromQuestionId === dependencyObj.sourceQuestionId)) {\n              if(Array.isArray(dependencyObj.queryValue)? dependencyObj.queryValue.includes(changeValue.referenceField) : changeValue.referenceField === dependencyObj.queryValue){\n                this.storageService.queryValueStore(this.question.id,changeValue.referenceField,changeValue.valueObj)\n              }\n            }\n            this.changeService.confirmChange(changeValue.fromQuestionId);\n          }\n        }\n      );\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if(changes['question']){\n      // if(this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)){\n      //   this.clickItem(this.question?.input)\n      // } else {\n      //   this.filterName = this.question?.input\n      // }\n    }\n    if (changes['mode']) {\n      this.mode = changes['mode'].currentValue;\n    }\n    if (changes['apiMeta'] && this.apiMeta) {\n      this.parseApiMeta();\n    }\n  }\n\n  ngOnDestroy(): void {\n    this.destroy$.next();\n    this.destroy$.complete();\n    this.resetComponentState();\n  }\n\n  // Handle clicks outside the search component\n  @HostListener('document:click', ['$event'])\n  onDocumentClick(event: Event): void {\n    if (!this.elementRef.nativeElement.contains(event.target as Node)) {\n      this.hideSuggestions();\n    }\n  }\n\n  // Handle escape key to close suggestions\n  @HostListener('document:keydown.escape', ['$event'])\n  onEscapeKey(event: KeyboardEvent): void {\n    this.hideSuggestions();\n  }\n\n  private parseApiMeta(): void {\n    if (this.apiMeta) {\n      this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta || '{}');\n      this.SearchItem = this.apiMeta.field || [];\n    }\n  }\n\n  private resetComponentState(): void {\n    this.finalResults.set([]);\n    this.searchKeyWord = '';\n    this.showSuggestion = false;\n    this.noResult = false;\n    this.isLoading = false;\n    this.inputFocused = false;\n    this.lastSearchLength = 0;\n  }\n\n  private hideSuggestions(): void {\n    this.showSuggestion = false;\n    this.inputFocused = false;\n  }\n\n  onInputFocus(): void {\n    this.inputFocused = true;\n    // Trigger a search with an empty keyword to show all data\n    if (!this.searchKeyWord) {\n      // this.performSearch('');\n    } else if (this.finalResults().length > 0) {\n      this.showSuggestion = true;\n    }\n  }\n\n  onInputBlur(): void {\n    // Don't immediately hide suggestions to allow for clicking\n    // The document click handler will handle hiding\n  }\n\n  clearList(): void {\n    // Remove the setTimeout delay that was causing issues\n    this.hideSuggestions();\n  }\n\n  getSourceDataLocal(event: any): void {\n    if (!event?.value?.valueObj) {\n      this.resetComponentState();\n      return;\n    }\n\n    const searchValue =  event.value?.valueObj?.includes('|') ? event.value?.valueObj?.split('|')[0].trim() : event.value?.valueObj?.trim();\n    const currentLength = searchValue.length;\n    \n    // Always update the search keyword\n    this.searchKeyWord = searchValue;\n    \n    // Handle the case where user types 3+ characters\n    if (currentLength > 2 && !this.isSingleFieldChange) {\n      this.showSuggestion = !event.question?.singleFieldChange;\n      this.showResult = false;\n      this.noResult = false;\n      // Always perform search when we have 3+ characters\n      this.lastSearchLength = currentLength;\n      this.searchSubject.next(searchValue);\n      \n    } else if (currentLength <= 2) {\n      // When length is 2 or less, hide suggestions but keep the search keyword\n      this.isSingleFieldChange = false;\n      this.showSuggestion = false;\n      this.noResult = false;\n      this.lastSearchLength = currentLength;\n      this.searchSubject.next(searchValue);\n      this.finalResults.set([]);\n    }\n  }\n\n  private performSearch(keyword: string): void {\n    if (!this.apiMeta) return;\n\n    this.parseApiMeta();\n    \n    if (!this.apiMeta.isStaticData) {\n      this.searchFromApi(keyword);\n    } else {\n      this.searchFromStaticData(keyword);\n    }\n  }\n\n  private searchFromApi(keyword: string): void {\n    this.isLoading = true;\n    this.finalResults.set([]);\n    if(!this.apiMeta.endpoint && Array.isArray(this.question?.input) ){\n      this.isLoading = false;\n      this.processSearchResults(this.question?.input, keyword);\n    } else {\n      let finalGetCall = this.apiMeta.endpoint;\n      // SKS18JUL25 query params creation\n      if(this.apiMeta?.queryParameter){\n        const isFieldArray = Array.isArray(this.apiMeta?.queryParameter);\n        const fields = isFieldArray ? this.apiMeta.queryParameter : [this.apiMeta.queryParameter]; // always an array now\n        // Build query params string\n        const queryParams = fields\n          .map(field => `${field}=${this.searchKeyWord}`)\n          .join('&');\n        // Check if endpoint already has query params\n        if(queryParams && queryParams !== ''){\n          if (this.apiMeta.endpoint.includes('?')) {\n            finalGetCall += `&${queryParams}`;\n          } else {\n            finalGetCall += `?${queryParams}`;\n          }\n        }\n      }\n      if (this.apiMeta?.queryField && this.apiMeta?.queryValue) {\n        const queryFields = Array.isArray(this.apiMeta.queryField)\n          ? this.apiMeta.queryField\n          : [this.apiMeta.queryField];\n      \n        const queryValues = Array.isArray(this.apiMeta.queryValue)\n          ? this.apiMeta.queryValue\n          : [this.apiMeta.queryValue];\n      \n        const stored = this.storageService.getQueryValue(this.question.id); // single object\n      \n        const queryParamsArray: string[] = [];\n      \n        queryFields.forEach((field, index) => {\n          const valueKey = queryValues[index] ?? queryValues[0];\n          const value = stored[valueKey];\n          if (value !== undefined && value !== null) {\n            queryParamsArray.push(`${field}=${encodeURIComponent(value)}`);\n          }\n        });\n      \n        const queryParams = queryParamsArray.join('&');\n      \n        if (queryParams) {\n          finalGetCall += this.apiMeta.endpoint.includes('?') ? `&${queryParams}` : `?${queryParams}`;\n        }\n      }      \n      this.dataService.apiResponse(finalGetCall)\n      .pipe(takeUntil(this.destroy$))\n      .subscribe({\n        next: (apiResponse) => {\n          this.isLoading = false;\n          this.processSearchResults(apiResponse, keyword);\n        },\n        error: (error) => {\n          this.isLoading = false;\n          this.noResult = true;\n          this.finalResults.set([]);\n        }\n      });\n    }\n  }\n\n  private searchFromStaticData(keyword: string): void {\n    const response = this.countryService.getCountryData(this.apiMeta.field);\n    this.processSearchResults(response, keyword);\n  }\n\n  private processSearchResults(apiResponse: any, keyword: string): void {\n    let response;\n  \n    if (this.apiMeta.variable) {\n      response = this.dataService.getValue(apiResponse, this.apiMeta.variable);\n    } else {\n      response = apiResponse;\n    }\n  \n    if (!Array.isArray(response)) {\n      this.noResult = true;\n      this.finalResults.set([]);\n      return;\n    }\n  \n    if (!keyword) {\n      // If no keyword, show all results\n      this.finalResults.set(response);\n      this.noResult = response.length === 0;\n      this.showSuggestion = true; // Ensure suggestions are visible\n    } else {\n      // Filter results based on the keyword\n      const searchTerms = keyword.toLowerCase().split(\" \").filter(term => term.length > 0);\n      const results = response.filter(obj => {\n        const combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);\n        return searchTerms.every(term => combinedValues.includes(term));\n      });\n  \n      this.noResult = results.length === 0;\n      this.finalResults.set(results);\n      this.showSuggestion = true; // Show suggestions only if there are results\n    }\n  }\n\n  private getNestedValue(obj: any, path: string): string {\n    if (!obj || !path) return '';\n    \n    const processedPath = path.replace(/\\[(\\d+)\\]/g, '.$1');\n    const parts = processedPath.split('.');\n    let current = obj;\n\n    for (const part of parts) {\n      if (!current || typeof current !== 'object') return '';\n      current = current[part];\n    }\n\n    return current !== null && current !== undefined \n      ? String(current).toLowerCase().trim() \n      : '';\n  }\n\n  private getCombinedValuesFromColumns(obj: any, columns: string[]): string {\n    if (!obj || !columns) return '';\n    \n    const columnArray = Array.isArray(columns) ? columns : [columns];\n    const values: string[] = [];\n\n    for (const column of columnArray) {\n      const value = this.getNestedValue(obj, column);\n      if (value) {\n        values.push(value);\n      }\n    }\n\n    return values.join(' ');\n  }\n\n  public getValues(element: any, columns: any): any {\n    const result: any = {};\n    if (!element || !columns) return result;\n\n    const columnArray = Array.isArray(columns) ? columns : [columns];\n    \n    columnArray.forEach((column) => {\n      if (!column) return;\n      \n      let tempElement = element;\n      const fields = column.split('.');\n      \n      for (const field of fields) {\n        if (!tempElement) break;\n        \n        const splitFields = field.split('[');\n        if (splitFields.length === 1) {\n          tempElement = tempElement[field];\n        } else {\n          const indexMatch = splitFields[1]?.match(/(\\d+)\\]/);\n          if (indexMatch) {\n            const index = Number(indexMatch[1]);\n            tempElement = tempElement[splitFields[0]]?.[index];\n          }\n        }\n      }\n      \n      result[column] = tempElement || '';\n    });\n\n    return result;\n  }\n\n  getKeys(obj: any): string[] {\n    return obj ? Object.keys(obj) : [];\n  }\n\n  public clickItem(event: any): void {\n    if (!this.apiMeta) return;\n  \n    this.parseApiMeta();\n    \n    const values = this.getValues(event, this.SearchItem);\n    const displayValue = Object.values(values)\n      .filter(val => val)\n      .join(' | ');\n    \n    if (this.mode !== 'edit') {\n      this.viewFilterName = displayValue;\n    }\n    this.filterName = displayValue;\n    this.ngOnChanges({})\n  \n    const defaultField = this.apiMeta.defaultField || \n      (Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);\n    \n    const change = new ChangeWrapper();\n    change['isObject'] = this.apiMeta?.isObject\n    change['fromQuestionId'] = this.id;\n    change['valueObj'] = event;\n    change['field'] = defaultField;\n    \n    this.valueObj = event;\n    \n    this.hideSuggestions();\n    \n    this.searchValueChange.emit({\n      question: this.question,\n      value: change\n    });\n  }\n\n  inputValue(event: any, ques: any): void {\n    this.filterName = event?.valueObj ? event?.valueObj?.trim() : this.filterName\n    if (!ques?.singleFieldChange || !this.apiMeta) return;\n\n    this.isSingleFieldChange = true;\n    this.parseApiMeta();\n\n    const defaultField = this.apiMeta.defaultField || \n      (Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);\n\n    const change = new ChangeWrapper();\n    change['isObject'] = this.apiMeta?.isObject\n    change['fromQuestionId'] = this.id;\n    change['valueObj'] = this.valueObj;\n    change['field'] = defaultField;\n    change['referenceField'] = this.question.referenceField;\n    \n    this.filterName = this.viewFilterName;\n    \n    this.searchValueChange.emit({\n      question: ques,\n      value: change\n    });\n  }\n\n  highlightText(text: string, searchTerm: string): string {\n    if (!text || !searchTerm) return text || '';\n\n    // Sanitize the input text\n    const sanitizedText = this.sanitizeHtml(text);\n    const terms = searchTerm.toLowerCase().split(' ').filter(term => term.length > 0);\n    \n    if (terms.length === 0) return sanitizedText;\n\n    // Escape special regex characters\n    const escapedTerms = terms.map(term => \n      term.replace(/[.*+?^${}()|[\\]\\\\]/g, '\\\\$&')\n    );\n\n    const regex = new RegExp(`(${escapedTerms.join('|')})`, 'gi');\n    return sanitizedText.replace(regex, '<span class=\"highlight\">$1</span>');\n  }\n\n  private sanitizeHtml(text: string): string {\n    return text\n      .replace(/&/g, '&amp;')\n      .replace(/</g, '&lt;')\n      .replace(/>/g, '&gt;')\n      .replace(/\"/g, '&quot;')\n      .replace(/'/g, '&#39;');\n  }\n  removeValueEmit(){\n    this.clickItem(null)\n  }\n}","<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\" #searchWrapper>\n  <nxt-input \n    [type]=\"'list'\" \n    [mode]=\"mode\" \n    [from]=\"from\"\n    [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n    [labelFont]=\"question?.font\" \n    [label]=\"question?.questionText ? ((question?.id+'.questionText') | nxtCustomTranslate : question?.questionText): ''\" \n    [labelColor]=\"question?.fontColor\"\n    [labelSize]=\"question?.fontSize\" \n    [inputValueSize]=\"question?.fontSize\" \n    [labelWeight]=\"question?.fontWeight\"\n    [inputWeight]=\"question?.fontWeight\" \n    [showLabel]=\"question?.style?.showLabel\" \n    inputBorder=\"none\" \n    svgHeight=\"20px\"\n    svgWidth=\"20px\" \n    [placeholder]=\"placeHolderText\" \n    [required]=\"question?.isOptional\" \n    inputBgColor=\"#FAFAFA\"\n    [inputId]=\"question?.trackingId\" \n    [errorMessages]=\"{ required: 'This field is required' }\"\n    [inputIconLeftSrc]=\"question?.iconLeftSrc\" \n    (inputValue)=\"getSourceDataLocal($event); inputValue($event.value, $event.question)\" \n    (onFocus)=\"onInputFocus()\"\n    (onBlur)=\"onInputBlur()\"\n    [showSuggestion]=\"showSuggestion\" \n    ariaOwns=\"selectList\" \n    ariaHasPopup=\"listbox\" \n    [isLoading]=\"isLoading\" \n    [question]=\"question\"\n    (removeValueEmit)=\"removeValueEmit()\">\n  </nxt-input>\n\n  <div id=\"selectList\" class=\"suggestion-wrapper\" role=\"listbox\" *ngIf=\"showSuggestion\">\n    <div *ngIf=\"finalResults().length > 0\" class=\"nxt-suggestions-container\">\n      <div *ngFor=\"let item of finalResults(); trackBy: trackByFn\" \n           (click)=\"clickItem(item)\" \n           class=\"suggestion-item\" \n           role=\"option\"\n           tabindex=\"0\"\n           (keydown.enter)=\"clickItem(item)\"\n           (keydown.space)=\"clickItem(item)\">\n        <div class=\"item-content\">\n          <div class=\"item-details\">\n            <div *ngIf=\"item.thumbnail\" class=\"thumbnail-wrapper\">\n              <img [src]=\"item.thumbnail\" [alt]=\"item.name + ' avatar'\" />\n            </div>\n            <div class=\"item-text\">\n              <!-- Primary text (first key) -->\n              <ng-container *ngIf=\"getKeys(getValues(item, SearchItem)).length > 0\">\n                <div class=\"item-name primary\" \n                     [innerHTML]=\"highlightText(getValues(item, SearchItem)[getKeys(getValues(item, SearchItem))[0]], searchKeyWord)\">\n                </div>\n              </ng-container>\n              \n              <!-- Secondary text (remaining keys) -->\n              <div class=\"item-secondary\" *ngIf=\"getKeys(getValues(item, SearchItem)).length > 1\">\n                <ng-container *ngFor=\"let key of getKeys(getValues(item, SearchItem)); let i = index\">\n                  <ng-container *ngIf=\"i > 0\">\n                    <span class=\"secondary-text\" \n                          [innerHTML]=\"highlightText(getValues(item, SearchItem)[key], searchKeyWord)\">\n                    </span>\n                    <span class=\"separator\" *ngIf=\"i < getKeys(getValues(item, SearchItem)).length - 1\"> | </span>\n                  </ng-container>\n                </ng-container>\n              </div>\n            </div>\n          </div>\n        </div>\n      </div>\n    </div>\n\n    <div *ngIf=\"noResult && searchKeyWord\" class=\"no-results\">\n      <div class=\"no-results-content\">\n        <svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" stroke=\"currentColor\" stroke-width=\"1\">\n          <circle cx=\"11\" cy=\"11\" r=\"8\"></circle>\n          <path d=\"m21 21-4.35-4.35\"></path>\n        </svg>\n        <p>{{ 'NO_RESULTS_FOUND' | nxtCustomTranslate : 'No results found for' }} \"{{searchKeyWord}}\"</p>\n      </div>\n    </div>\n  </div>\n</div>"]}
|