@rangertechnologies/ngnxt 2.1.239 → 2.1.241
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/esm2022/environments/version.mjs +2 -2
- package/esm2022/lib/components/button/nxt-button.component.mjs +4 -7
- package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +1 -1
- package/esm2022/lib/components/datatable/datatable.component.mjs +3 -3
- package/esm2022/lib/components/file-upload/file-upload.component.mjs +5 -2
- package/esm2022/lib/components/nxt-input/nxt-input.component.mjs +10 -4
- package/esm2022/lib/components/pick-location/pick-location.component.mjs +1 -1
- package/esm2022/lib/components/search-box/search-box.component.mjs +294 -158
- package/esm2022/lib/pages/booklet/booklet.component.mjs +6 -8
- package/esm2022/lib/pages/builder/properties/properties.component.mjs +1 -1
- package/esm2022/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.mjs +9 -3
- package/esm2022/lib/pages/pdfDesigner/pdf-properties/pdf-properties.component.mjs +2 -2
- package/esm2022/lib/pages/questionbook/questionbook.component.mjs +4 -4
- package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +2 -2
- package/esm2022/lib/services/translation.service.mjs +2 -1
- package/fesm2022/rangertechnologies-ngnxt.mjs +333 -186
- package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -1
- package/lib/components/nxt-input/nxt-input.component.d.ts +3 -1
- package/lib/components/search-box/search-box.component.d.ts +26 -10
- package/lib/pages/booklet/booklet.component.d.ts +1 -1
- package/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.d.ts +3 -1
- package/package.json +1 -1
- package/rangertechnologies-ngnxt-2.1.241.tgz +0 -0
- package/rangertechnologies-ngnxt-2.1.239.tgz +0 -0
|
@@ -1,9 +1,9 @@
|
|
|
1
|
-
import { Component, Input, Output, EventEmitter, signal } from "@angular/core";
|
|
2
|
-
// VD 23JAN24 removed httpClient import used service file for callout
|
|
1
|
+
import { Component, Input, Output, EventEmitter, ViewChild, signal, HostListener } from "@angular/core";
|
|
3
2
|
import { ChangeWrapper } from "../../model/changeWrapper";
|
|
4
3
|
import { CommonModule } from "@angular/common";
|
|
5
4
|
import { FormsModule } from "@angular/forms";
|
|
6
5
|
import { NxtInput } from "../nxt-input/nxt-input.component";
|
|
6
|
+
import { Subject, debounceTime, distinctUntilChanged, takeUntil } from "rxjs";
|
|
7
7
|
import * as i0 from "@angular/core";
|
|
8
8
|
import * as i1 from "../../services/data.service";
|
|
9
9
|
import * as i2 from "../../i18n.service";
|
|
@@ -13,147 +13,238 @@ export class NxtSearchBox {
|
|
|
13
13
|
dataService;
|
|
14
14
|
i18nService;
|
|
15
15
|
countryService;
|
|
16
|
-
|
|
16
|
+
elementRef;
|
|
17
|
+
placeHolderText = '';
|
|
17
18
|
question;
|
|
18
19
|
apiMeta;
|
|
19
|
-
id;
|
|
20
|
-
readOnly = false;
|
|
21
|
-
mode = 'edit';
|
|
20
|
+
id = '';
|
|
21
|
+
readOnly = false;
|
|
22
|
+
mode = 'edit';
|
|
22
23
|
searchValueChange = new EventEmitter();
|
|
24
|
+
searchWrapper;
|
|
23
25
|
SearchItem;
|
|
24
|
-
filterName; // VD 20Aug default filter value as input
|
|
25
26
|
finalResults = signal([]);
|
|
26
27
|
searchKeyWord = '';
|
|
27
28
|
newResult;
|
|
28
29
|
showResult = false;
|
|
29
30
|
noResult = false;
|
|
30
31
|
showSuggestion = false;
|
|
31
|
-
el;
|
|
32
|
-
serv = 'api';
|
|
33
|
-
tkn = '';
|
|
34
32
|
isLoading = false;
|
|
35
|
-
viewFilterName;
|
|
33
|
+
viewFilterName = '';
|
|
36
34
|
isSingleFieldChange = false;
|
|
37
35
|
valueObj;
|
|
38
|
-
|
|
36
|
+
inputFocused = false;
|
|
37
|
+
lastSearchLength = 0;
|
|
38
|
+
filterName = '';
|
|
39
|
+
destroy$ = new Subject();
|
|
40
|
+
searchSubject = new Subject();
|
|
41
|
+
// TrackBy function for performance optimization
|
|
42
|
+
trackByFn = (index, item) => {
|
|
43
|
+
return item.id || item.name || index;
|
|
44
|
+
};
|
|
45
|
+
constructor(dataService, i18nService, countryService, elementRef) {
|
|
39
46
|
this.dataService = dataService;
|
|
40
47
|
this.i18nService = i18nService;
|
|
41
48
|
this.countryService = countryService;
|
|
49
|
+
this.elementRef = elementRef;
|
|
50
|
+
if (this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)) {
|
|
51
|
+
this.clickItem(this.question?.input);
|
|
52
|
+
}
|
|
53
|
+
else {
|
|
54
|
+
this.filterName = this.question?.input;
|
|
55
|
+
}
|
|
56
|
+
// Setup debounced search
|
|
57
|
+
this.searchSubject
|
|
58
|
+
.pipe(debounceTime(300), distinctUntilChanged(), takeUntil(this.destroy$))
|
|
59
|
+
.subscribe(keyword => {
|
|
60
|
+
if (this.lastSearchLength > 2)
|
|
61
|
+
this.performSearch(keyword);
|
|
62
|
+
});
|
|
42
63
|
}
|
|
43
64
|
ngOnInit() {
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
|
|
47
|
-
|
|
48
|
-
this.
|
|
65
|
+
if (this.question?.input && typeof this.question?.input === 'object' && !Array.isArray(this.question?.input)) {
|
|
66
|
+
this.clickItem(this.question?.input);
|
|
67
|
+
}
|
|
68
|
+
else {
|
|
69
|
+
this.filterName = this.question?.input;
|
|
49
70
|
}
|
|
71
|
+
this.parseApiMeta();
|
|
50
72
|
}
|
|
51
|
-
////RS 03FEB2025
|
|
52
|
-
// Resets state when filterName or apiMeta changes to reflect updated data
|
|
53
|
-
// AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string
|
|
54
73
|
ngOnChanges(changes) {
|
|
55
74
|
if (changes['mode']) {
|
|
56
75
|
this.mode = changes['mode'].currentValue;
|
|
57
76
|
}
|
|
58
|
-
if (this.mode !== 'edit' && this.isSingleFieldChange &&
|
|
59
|
-
|
|
77
|
+
if (this.mode !== 'edit' && this.isSingleFieldChange && changes['filterName']) {
|
|
78
|
+
const previous = changes['filterName'].previousValue;
|
|
79
|
+
const current = changes['filterName'].currentValue;
|
|
80
|
+
if (previous !== current) {
|
|
81
|
+
this.filterName = previous;
|
|
82
|
+
}
|
|
60
83
|
}
|
|
61
84
|
if (changes['apiMeta'] && this.apiMeta) {
|
|
62
|
-
this.
|
|
63
|
-
|
|
85
|
+
this.parseApiMeta();
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
ngOnDestroy() {
|
|
89
|
+
this.destroy$.next();
|
|
90
|
+
this.destroy$.complete();
|
|
91
|
+
this.resetComponentState();
|
|
92
|
+
}
|
|
93
|
+
// Handle clicks outside the search component
|
|
94
|
+
onDocumentClick(event) {
|
|
95
|
+
if (!this.elementRef.nativeElement.contains(event.target)) {
|
|
96
|
+
this.hideSuggestions();
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
// Handle escape key to close suggestions
|
|
100
|
+
onEscapeKey(event) {
|
|
101
|
+
this.hideSuggestions();
|
|
102
|
+
}
|
|
103
|
+
parseApiMeta() {
|
|
104
|
+
if (this.apiMeta) {
|
|
105
|
+
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta || '{}');
|
|
106
|
+
this.SearchItem = this.apiMeta.field || [];
|
|
64
107
|
}
|
|
65
108
|
}
|
|
66
|
-
//RS 03FEB2025
|
|
67
|
-
// Clears search-related data, including results, search term, and suggestions.
|
|
68
109
|
resetComponentState() {
|
|
69
110
|
this.finalResults.set([]);
|
|
70
|
-
;
|
|
71
|
-
this.filterName = '';
|
|
72
111
|
this.searchKeyWord = '';
|
|
73
112
|
this.showSuggestion = false;
|
|
74
113
|
this.noResult = false;
|
|
114
|
+
this.isLoading = false;
|
|
115
|
+
this.inputFocused = false;
|
|
116
|
+
this.lastSearchLength = 0;
|
|
117
|
+
}
|
|
118
|
+
hideSuggestions() {
|
|
119
|
+
this.showSuggestion = false;
|
|
120
|
+
this.inputFocused = false;
|
|
121
|
+
}
|
|
122
|
+
onInputFocus() {
|
|
123
|
+
this.inputFocused = true;
|
|
124
|
+
// Trigger a search with an empty keyword to show all data
|
|
125
|
+
if (!this.searchKeyWord) {
|
|
126
|
+
this.performSearch('');
|
|
127
|
+
}
|
|
128
|
+
else if (this.finalResults().length > 0) {
|
|
129
|
+
this.showSuggestion = true;
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
onInputBlur() {
|
|
133
|
+
// Don't immediately hide suggestions to allow for clicking
|
|
134
|
+
// The document click handler will handle hiding
|
|
75
135
|
}
|
|
76
136
|
clearList() {
|
|
77
|
-
setTimeout
|
|
78
|
-
|
|
79
|
-
;
|
|
80
|
-
}, 1000);
|
|
137
|
+
// Remove the setTimeout delay that was causing issues
|
|
138
|
+
this.hideSuggestions();
|
|
81
139
|
}
|
|
82
140
|
getSourceDataLocal(event) {
|
|
83
|
-
if (event
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
|
|
141
|
+
if (!event?.value) {
|
|
142
|
+
this.resetComponentState();
|
|
143
|
+
return;
|
|
144
|
+
}
|
|
145
|
+
const searchValue = event.value.includes('|') ? event.value.split('|')[0].trim() : event.value.trim();
|
|
146
|
+
const currentLength = searchValue.length;
|
|
147
|
+
// Always update the search keyword
|
|
148
|
+
this.searchKeyWord = searchValue;
|
|
149
|
+
// Handle the case where user types 3+ characters
|
|
150
|
+
if (currentLength > 2 && !this.isSingleFieldChange) {
|
|
151
|
+
this.showSuggestion = !event.question?.singleFieldChange;
|
|
87
152
|
this.showResult = false;
|
|
88
|
-
this.
|
|
153
|
+
this.noResult = false;
|
|
154
|
+
// Always perform search when we have 3+ characters
|
|
155
|
+
this.lastSearchLength = currentLength;
|
|
156
|
+
this.searchSubject.next(searchValue);
|
|
89
157
|
}
|
|
90
|
-
else {
|
|
158
|
+
else if (currentLength <= 2) {
|
|
159
|
+
// When length is 2 or less, hide suggestions but keep the search keyword
|
|
91
160
|
this.isSingleFieldChange = false;
|
|
92
161
|
this.showSuggestion = false;
|
|
93
|
-
this.finalResults.set([]);
|
|
94
162
|
this.noResult = false;
|
|
163
|
+
this.lastSearchLength = currentLength;
|
|
164
|
+
this.searchSubject.next(searchValue);
|
|
165
|
+
this.finalResults.set([]);
|
|
95
166
|
}
|
|
96
167
|
}
|
|
97
|
-
|
|
98
|
-
|
|
99
|
-
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
|
|
106
|
-
|
|
107
|
-
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
|
|
114
|
-
|
|
115
|
-
|
|
116
|
-
|
|
117
|
-
|
|
118
|
-
|
|
119
|
-
|
|
120
|
-
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
|
|
124
|
-
|
|
125
|
-
|
|
126
|
-
|
|
127
|
-
|
|
128
|
-
|
|
129
|
-
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
else { // SKS24JUN25 Get language and country name locally from data
|
|
137
|
-
let response;
|
|
138
|
-
response = this.countryService.getCountryData(this.apiMeta.field);
|
|
139
|
-
let results = [];
|
|
140
|
-
let searchTerms = keyword.toLowerCase()?.split(" ");
|
|
141
|
-
for (let i = 0; i < response.length; i++) {
|
|
142
|
-
let obj = response[i];
|
|
143
|
-
let combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);
|
|
144
|
-
let match = searchTerms.every(term => combinedValues.includes(term));
|
|
145
|
-
if (match) {
|
|
146
|
-
results.push(obj);
|
|
147
|
-
}
|
|
148
|
-
}
|
|
149
|
-
this.noResult = results.length === 0;
|
|
150
|
-
this.finalResults.set(results);
|
|
168
|
+
performSearch(keyword) {
|
|
169
|
+
if (!this.apiMeta)
|
|
170
|
+
return;
|
|
171
|
+
this.parseApiMeta();
|
|
172
|
+
if (!this.apiMeta.isStaticData) {
|
|
173
|
+
this.searchFromApi(keyword);
|
|
174
|
+
}
|
|
175
|
+
else {
|
|
176
|
+
this.searchFromStaticData(keyword);
|
|
177
|
+
}
|
|
178
|
+
}
|
|
179
|
+
searchFromApi(keyword) {
|
|
180
|
+
this.isLoading = true;
|
|
181
|
+
this.finalResults.set([]);
|
|
182
|
+
let finalGetCall = this.apiMeta.endpoint;
|
|
183
|
+
// SKS18JUL25 query params creation
|
|
184
|
+
// const isFieldArray = Array.isArray(this.apiMeta?.field);
|
|
185
|
+
// const fields = isFieldArray ? this.apiMeta.field : [this.apiMeta.field]; // always an array now
|
|
186
|
+
// // Build query params string
|
|
187
|
+
// const queryParams = fields
|
|
188
|
+
// .map(field => `${field}=${this.searchKeyWord}`)
|
|
189
|
+
// .join('&');
|
|
190
|
+
// // Check if endpoint already has query params
|
|
191
|
+
// if (this.apiMeta.endpoint.includes('?')) {
|
|
192
|
+
// finalGetCall += `&${queryParams}`;
|
|
193
|
+
// } else {
|
|
194
|
+
// finalGetCall += `?${queryParams}`;
|
|
195
|
+
// }
|
|
196
|
+
this.dataService.apiResponse(finalGetCall)
|
|
197
|
+
.pipe(takeUntil(this.destroy$))
|
|
198
|
+
.subscribe({
|
|
199
|
+
next: (apiResponse) => {
|
|
200
|
+
this.isLoading = false;
|
|
201
|
+
this.processSearchResults(apiResponse, keyword);
|
|
202
|
+
},
|
|
203
|
+
error: (error) => {
|
|
204
|
+
this.isLoading = false;
|
|
205
|
+
this.noResult = true;
|
|
206
|
+
this.finalResults.set([]);
|
|
151
207
|
}
|
|
208
|
+
});
|
|
209
|
+
}
|
|
210
|
+
searchFromStaticData(keyword) {
|
|
211
|
+
const response = this.countryService.getCountryData(this.apiMeta.field);
|
|
212
|
+
this.processSearchResults(response, keyword);
|
|
213
|
+
}
|
|
214
|
+
processSearchResults(apiResponse, keyword) {
|
|
215
|
+
let response;
|
|
216
|
+
if (this.apiMeta.variable) {
|
|
217
|
+
response = this.dataService.getValue(apiResponse, this.apiMeta.variable);
|
|
152
218
|
}
|
|
153
|
-
|
|
154
|
-
|
|
219
|
+
else {
|
|
220
|
+
response = apiResponse;
|
|
221
|
+
}
|
|
222
|
+
if (!Array.isArray(response)) {
|
|
223
|
+
this.noResult = true;
|
|
224
|
+
this.finalResults.set([]);
|
|
225
|
+
return;
|
|
226
|
+
}
|
|
227
|
+
if (!keyword) {
|
|
228
|
+
// If no keyword, show all results
|
|
229
|
+
this.finalResults.set(response);
|
|
230
|
+
this.noResult = response.length === 0;
|
|
231
|
+
this.showSuggestion = true; // Ensure suggestions are visible
|
|
232
|
+
}
|
|
233
|
+
else {
|
|
234
|
+
// Filter results based on the keyword
|
|
235
|
+
const searchTerms = keyword.toLowerCase().split(" ").filter(term => term.length > 0);
|
|
236
|
+
const results = response.filter(obj => {
|
|
237
|
+
const combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);
|
|
238
|
+
return searchTerms.every(term => combinedValues.includes(term));
|
|
239
|
+
});
|
|
240
|
+
this.noResult = results.length === 0;
|
|
241
|
+
this.finalResults.set(results);
|
|
242
|
+
this.showSuggestion = results.length > 0; // Show suggestions only if there are results
|
|
243
|
+
}
|
|
244
|
+
}
|
|
155
245
|
getNestedValue(obj, path) {
|
|
156
|
-
|
|
246
|
+
if (!obj || !path)
|
|
247
|
+
return '';
|
|
157
248
|
const processedPath = path.replace(/\[(\d+)\]/g, '.$1');
|
|
158
249
|
const parts = processedPath.split('.');
|
|
159
250
|
let current = obj;
|
|
@@ -167,12 +258,11 @@ export class NxtSearchBox {
|
|
|
167
258
|
: '';
|
|
168
259
|
}
|
|
169
260
|
getCombinedValuesFromColumns(obj, columns) {
|
|
261
|
+
if (!obj || !columns)
|
|
262
|
+
return '';
|
|
263
|
+
const columnArray = Array.isArray(columns) ? columns : [columns];
|
|
170
264
|
const values = [];
|
|
171
|
-
|
|
172
|
-
if (typeof columns === 'string' && !Array.isArray(columns)) {
|
|
173
|
-
columns = [columns];
|
|
174
|
-
}
|
|
175
|
-
for (const column of columns) {
|
|
265
|
+
for (const column of columnArray) {
|
|
176
266
|
const value = this.getNestedValue(obj, column);
|
|
177
267
|
if (value) {
|
|
178
268
|
values.push(value);
|
|
@@ -180,77 +270,116 @@ export class NxtSearchBox {
|
|
|
180
270
|
}
|
|
181
271
|
return values.join(' ');
|
|
182
272
|
}
|
|
183
|
-
// SKS27FEB get value from data specific column
|
|
184
273
|
getValues(element, columns) {
|
|
185
274
|
const result = {};
|
|
186
|
-
|
|
187
|
-
|
|
188
|
-
|
|
189
|
-
|
|
190
|
-
|
|
275
|
+
if (!element || !columns)
|
|
276
|
+
return result;
|
|
277
|
+
const columnArray = Array.isArray(columns) ? columns : [columns];
|
|
278
|
+
columnArray.forEach((column) => {
|
|
279
|
+
if (!column)
|
|
280
|
+
return;
|
|
191
281
|
let tempElement = element;
|
|
192
|
-
|
|
193
|
-
for (
|
|
194
|
-
|
|
195
|
-
|
|
196
|
-
|
|
282
|
+
const fields = column.split('.');
|
|
283
|
+
for (const field of fields) {
|
|
284
|
+
if (!tempElement)
|
|
285
|
+
break;
|
|
286
|
+
const splitFields = field.split('[');
|
|
287
|
+
if (splitFields.length === 1) {
|
|
288
|
+
tempElement = tempElement[field];
|
|
197
289
|
}
|
|
198
290
|
else {
|
|
199
|
-
|
|
200
|
-
|
|
291
|
+
const indexMatch = splitFields[1]?.match(/(\d+)\]/);
|
|
292
|
+
if (indexMatch) {
|
|
293
|
+
const index = Number(indexMatch[1]);
|
|
294
|
+
tempElement = tempElement[splitFields[0]]?.[index];
|
|
295
|
+
}
|
|
201
296
|
}
|
|
202
297
|
}
|
|
203
|
-
result[column] = tempElement;
|
|
298
|
+
result[column] = tempElement || '';
|
|
204
299
|
});
|
|
205
300
|
return result;
|
|
206
301
|
}
|
|
207
|
-
// SKS27FEB column value get funtion
|
|
208
302
|
getKeys(obj) {
|
|
209
|
-
return Object.keys(obj);
|
|
303
|
+
return obj ? Object.keys(obj) : [];
|
|
210
304
|
}
|
|
211
305
|
clickItem(event) {
|
|
212
|
-
|
|
213
|
-
|
|
214
|
-
|
|
215
|
-
|
|
216
|
-
|
|
217
|
-
|
|
218
|
-
|
|
219
|
-
|
|
220
|
-
|
|
221
|
-
|
|
222
|
-
|
|
223
|
-
|
|
224
|
-
change.valueObj = event;
|
|
225
|
-
this.valueObj = event;
|
|
226
|
-
change.field = this.apiMeta.defaultField || Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field;
|
|
227
|
-
this.searchValueChange.emit({ question: this.question, value: change });
|
|
306
|
+
if (!this.apiMeta)
|
|
307
|
+
return;
|
|
308
|
+
this.parseApiMeta();
|
|
309
|
+
const values = this.getValues(event, this.SearchItem);
|
|
310
|
+
const displayValue = Object.values(values)
|
|
311
|
+
.filter(val => val)
|
|
312
|
+
.join(' | ');
|
|
313
|
+
if (this.mode !== 'edit') {
|
|
314
|
+
this.viewFilterName = displayValue;
|
|
315
|
+
}
|
|
316
|
+
else {
|
|
317
|
+
this.filterName = displayValue;
|
|
228
318
|
}
|
|
319
|
+
this.ngOnChanges({});
|
|
320
|
+
const defaultField = this.apiMeta.defaultField ||
|
|
321
|
+
(Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);
|
|
322
|
+
const change = new ChangeWrapper();
|
|
323
|
+
change.fromQuestionId = this.id;
|
|
324
|
+
change.valueObj = event;
|
|
325
|
+
change.field = defaultField;
|
|
326
|
+
this.valueObj = event;
|
|
327
|
+
this.hideSuggestions();
|
|
328
|
+
this.searchValueChange.emit({
|
|
329
|
+
question: this.question,
|
|
330
|
+
value: change
|
|
331
|
+
});
|
|
229
332
|
}
|
|
230
|
-
//RS 03FEB2025
|
|
231
|
-
// Resets component state when the component is destroyed
|
|
232
|
-
ngOnDestroy() { this.resetComponentState(); }
|
|
233
333
|
removeCharacters(questionText) {
|
|
234
|
-
|
|
235
|
-
|
|
334
|
+
if (!questionText)
|
|
335
|
+
return '';
|
|
336
|
+
return questionText.replace(/<[^>]*>/g, '');
|
|
236
337
|
}
|
|
237
338
|
inputValue(event, ques) {
|
|
238
|
-
|
|
239
|
-
|
|
240
|
-
|
|
241
|
-
|
|
242
|
-
|
|
243
|
-
|
|
244
|
-
|
|
245
|
-
|
|
246
|
-
|
|
247
|
-
|
|
248
|
-
|
|
249
|
-
|
|
250
|
-
|
|
339
|
+
this.filterName = event.trim();
|
|
340
|
+
if (!ques?.singleFieldChange || !this.apiMeta)
|
|
341
|
+
return;
|
|
342
|
+
this.isSingleFieldChange = true;
|
|
343
|
+
this.parseApiMeta();
|
|
344
|
+
const defaultField = this.apiMeta.defaultField ||
|
|
345
|
+
(Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);
|
|
346
|
+
const change = new ChangeWrapper();
|
|
347
|
+
change.fromQuestionId = this.id;
|
|
348
|
+
change.valueObj = this.valueObj;
|
|
349
|
+
change.field = defaultField;
|
|
350
|
+
change.referenceField = this.question.referenceField;
|
|
351
|
+
this.filterName = this.viewFilterName;
|
|
352
|
+
this.searchValueChange.emit({
|
|
353
|
+
question: ques,
|
|
354
|
+
value: change
|
|
355
|
+
});
|
|
251
356
|
}
|
|
252
|
-
|
|
253
|
-
|
|
357
|
+
highlightText(text, searchTerm) {
|
|
358
|
+
if (!text || !searchTerm)
|
|
359
|
+
return text || '';
|
|
360
|
+
// Sanitize the input text
|
|
361
|
+
const sanitizedText = this.sanitizeHtml(text);
|
|
362
|
+
const terms = searchTerm.toLowerCase().split(' ').filter(term => term.length > 0);
|
|
363
|
+
if (terms.length === 0)
|
|
364
|
+
return sanitizedText;
|
|
365
|
+
// Escape special regex characters
|
|
366
|
+
const escapedTerms = terms.map(term => term.replace(/[.*+?^${}()|[\]\\]/g, '\\$&'));
|
|
367
|
+
const regex = new RegExp(`(${escapedTerms.join('|')})`, 'gi');
|
|
368
|
+
return sanitizedText.replace(regex, '<span class="highlight">$1</span>');
|
|
369
|
+
}
|
|
370
|
+
sanitizeHtml(text) {
|
|
371
|
+
return text
|
|
372
|
+
.replace(/&/g, '&')
|
|
373
|
+
.replace(/</g, '<')
|
|
374
|
+
.replace(/>/g, '>')
|
|
375
|
+
.replace(/"/g, '"')
|
|
376
|
+
.replace(/'/g, ''');
|
|
377
|
+
}
|
|
378
|
+
removeValueEmit() {
|
|
379
|
+
this.clickItem(null);
|
|
380
|
+
}
|
|
381
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NxtSearchBox, deps: [{ token: i1.DataService }, { token: i2.I18nService }, { token: i3.CountryService }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Component });
|
|
382
|
+
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" }, 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 [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n [labelFont]=\"question.font\" \n [label]=\"removeCharacters(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=\"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 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}.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}.suggestions-container::-webkit-scrollbar{width:6px}.suggestions-container::-webkit-scrollbar-track{background:#f1f1f1}.suggestions-container::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:3px}.suggestions-container::-webkit-scrollbar-thumb:hover{background:#a8a8a8}@media (max-width: 768px){.suggestion-wrapper,.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: i4.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i4.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", "size"], outputs: ["valueChange", "inputValue", "onBlur", "onFocus", "toggleEmit", "nativeInputRef", "removeValueEmit"] }] });
|
|
254
383
|
}
|
|
255
384
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: NxtSearchBox, decorators: [{
|
|
256
385
|
type: Component,
|
|
@@ -258,8 +387,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
258
387
|
CommonModule,
|
|
259
388
|
FormsModule,
|
|
260
389
|
NxtInput
|
|
261
|
-
], template: "<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\">\n <nxt-input [type]=\"'list'\" [mode]=\"mode\" [value]=\"filterName\"\n [labelFont]=\"question.font\" [label]=\"removeCharacters(question?.questionText)\" [labelColor]=\"question.fontColor\"\n [labelSize]=\"question.fontSize\" [inputValueSize]=\"question.fontSize\" [labelWeight]=\"question.fontWeight\"\n [inputWeight]=\"question.fontWeight\" [showLabel]=\"question.style?.showLabel\" inputBorder=\"none\" svgHeight=\"20px\"\n svgWidth=\"20px\" [placeholder]=\"placeHolderText\" [required]=\"question.isOptional\" inputBgColor=\"#FAFAFA\"\n [inputId]=\"question.trackingId\" [errorMessages]=\"{ required: 'This field is required' }\"\n [inputIconLeftSrc]=\"question.iconLeftSrc\" (inputValue)=\"getSourceDataLocal($event); inputValue($event.value
|
|
262
|
-
}], ctorParameters: () => [{ type: i1.DataService }, { type: i2.I18nService }, { type: i3.CountryService }], propDecorators: { placeHolderText: [{
|
|
390
|
+
], template: "<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\" #searchWrapper>\n <nxt-input \n [type]=\"'list'\" \n [mode]=\"mode\" \n [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n [labelFont]=\"question.font\" \n [label]=\"removeCharacters(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=\"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 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}.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}.suggestions-container::-webkit-scrollbar{width:6px}.suggestions-container::-webkit-scrollbar-track{background:#f1f1f1}.suggestions-container::-webkit-scrollbar-thumb{background:#c1c1c1;border-radius:3px}.suggestions-container::-webkit-scrollbar-thumb:hover{background:#a8a8a8}@media (max-width: 768px){.suggestion-wrapper,.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"] }]
|
|
391
|
+
}], ctorParameters: () => [{ type: i1.DataService }, { type: i2.I18nService }, { type: i3.CountryService }, { type: i0.ElementRef }], propDecorators: { placeHolderText: [{
|
|
263
392
|
type: Input
|
|
264
393
|
}], question: [{
|
|
265
394
|
type: Input
|
|
@@ -273,7 +402,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImpo
|
|
|
273
402
|
type: Input
|
|
274
403
|
}], searchValueChange: [{
|
|
275
404
|
type: Output
|
|
276
|
-
}],
|
|
277
|
-
type:
|
|
405
|
+
}], searchWrapper: [{
|
|
406
|
+
type: ViewChild,
|
|
407
|
+
args: ['searchWrapper', { static: true }]
|
|
408
|
+
}], onDocumentClick: [{
|
|
409
|
+
type: HostListener,
|
|
410
|
+
args: ['document:click', ['$event']]
|
|
411
|
+
}], onEscapeKey: [{
|
|
412
|
+
type: HostListener,
|
|
413
|
+
args: ['document:keydown.escape', ['$event']]
|
|
278
414
|
}] } });
|
|
279
|
-
//# 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":"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","sourcesContent":["import {\n  Component,\n  OnInit,\n  Input,\n  Output,\n  EventEmitter,\n  ViewChild,\n  SimpleChanges,signal, WritableSignal\n} from \"@angular/core\";\nimport { DataService } from '../../services/data.service';\n// VD 23JAN24 removed httpClient import used service file for callout\nimport { ChangeWrapper } from \"../../model/changeWrapper\";\n// HA 19DEC23 imported translation service\nimport { I18nService } from \"../../i18n.service\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { NxtInput } from \"../nxt-input/nxt-input.component\";\nimport { CountryService } from \"../../services/country.service\";\n\n@Component({\n  selector: 'nxt-search-box',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    NxtInput\n  ],\n  templateUrl: './search-box.component.html',\n  styleUrls: ['./search-box.component.css']\n})\nexport class NxtSearchBox implements OnInit {\n  @Input() placeHolderText: string;\n\n  @Input() question;\n  @Input() apiMeta: any;\n  @Input() id: string;\n  @Input() readOnly = false; // VD 12Jun24 - readonly change\n  @Input() mode: 'view' | 'edit' | 'print' = 'edit'; // SKS11JUN25 New mode input\n  @Output() searchValueChange: EventEmitter<any> = new EventEmitter<any>();\n   SearchItem: any;\n\n  @Input() filterName: string; // VD 20Aug default filter value as input\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  private el: HTMLElement;\n  private serv: string = 'api';\n  private tkn: string = '';\n  isLoading = false\n  viewFilterName : string;\n  isSingleFieldChange = false;\n  valueObj : any;\n  \n  constructor( private dataService: DataService, public i18nService: I18nService,private countryService: CountryService) { }\n  ngOnInit(): void {\n    //  VD 03May- search changes\n     // AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string\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////RS 03FEB2025\n// Resets state when filterName or apiMeta changes to reflect updated data\n // AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string\nngOnChanges(changes: SimpleChanges): void {\n  if (changes['mode']) {\n    this.mode = changes['mode'].currentValue;\n  }\n  if(this.mode !== 'edit' && this.isSingleFieldChange && (changes.filterName.previousValue !==  changes.filterName.currentValue)) {\n    this.filterName = changes.filterName.previousValue\n  }\n  if (changes['apiMeta'] && this.apiMeta) {\n    this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);\n    this.SearchItem = this.apiMeta?.field;\n  }\n}\n//RS 03FEB2025\n// Clears search-related data, including results, search term, and suggestions.\nprivate resetComponentState(): void {\n  this.finalResults.set([]);;\n  this.filterName = ''; \n  this.searchKeyWord = '';  \n  this.showSuggestion = false;\n  this.noResult = false;\n}\nclearList(){\n  setTimeout(()=> {\n    this.finalResults.set([]);;\n  }, 1000);\n}\n\ngetSourceDataLocal(event: any) { //to get results list from backend API whenever key is up after the entering atleast one key\n  if (event.value.length > 2 && !this.isSingleFieldChange) {\n    event.question?.singleFieldChange ? this.showSuggestion = false : this.showSuggestion = true;\n    this.finalResults.set([]);\n    this.searchKeyWord = event.value;\n    this.showResult = false;\n    this.getSourceData(event.value);\n  } else {\n    this.isSingleFieldChange = false\n    this.showSuggestion = false;\n    this.finalResults.set([]);\n    this.noResult = false;\n  }\n}\n// VD 03May- search changes\n// VD 31NOV24 null check\n// RS 29JAN25\n//Multi-word search across all object values\npublic getSourceData = (keyword: string) => {\n   // MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string\n  if (this.apiMeta) {\n      this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);\n      if(!this.apiMeta.isStaticData){\n        this.isLoading = true;\n        this.dataService.apiResponse(this.apiMeta.endpoint).subscribe((apiResponse) => {\n          this.isLoading = false;\n          let response;\n          if (this.apiMeta.variable) {\n            response = this.dataService.getValue(apiResponse, this.apiMeta.variable);\n          } else {\n            response = apiResponse;\n          }      \n          let results = [];\n          let searchTerms = keyword.toLowerCase()?.split(\" \");\n          for (let i = 0; i < response.length; i++) {\n            let obj = response[i];\n            let combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);\n            let match = searchTerms.every(term => combinedValues.includes(term));\n            if (match) {\n              results.push(obj);\n            }\n          }\n          this.noResult = results.length === 0;\n          this.finalResults.set(results);\n        },\n        (error) => {\n          console.error('API error:', error);\n          this.isLoading = false;\n          this.noResult = true;\n          this.finalResults.set([]);;\n        });\n      } else { // SKS24JUN25 Get language and country name locally from data\n        let response;  \n        response = this.countryService.getCountryData(this.apiMeta.field)  \n        let results = [];\n        let searchTerms = keyword.toLowerCase()?.split(\" \");\n        for (let i = 0; i < response.length; i++) {\n          let obj = response[i];\n          let combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);\n          let match = searchTerms.every(term => combinedValues.includes(term));\n          if (match) {\n            results.push(obj);\n          }\n        }\n        this.noResult = results.length === 0;\n        this.finalResults.set(results);\n      }\n  }\n};\n// SKS27FEB Helper functions\nprivate getNestedValue(obj: any, path: string): string {\n  //SKS27FEB Convert array indexes to dot notation (e.g., [0] -> .0)\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\nprivate getCombinedValuesFromColumns(obj: any, columns: string[]): string {\n  const values: string[] = [];\n  // SKS28FEB check if columns is an array\n  if(typeof columns === 'string' && !Array.isArray(columns)  ) {\n    columns = [columns];\n  }\n  \n  for (const column of columns) {\n    const value = this.getNestedValue(obj, column);\n    if (value) {\n      values.push(value);\n    }\n  }\n  \n  return values.join(' ');\n}\n// SKS27FEB get value from data specific column\npublic getValues(element: any, columns: any): any {\n  const result: any = {};\n  // SKS28FEB check if columns is an array\n  if(typeof columns === 'string' && !Array.isArray(columns)  ) {\n    columns = [columns];\n  }\n  columns.forEach((column) => {\n    let tempElement = element;\n    let flds = column?.split('.');\n    for (let i = 0; i < flds.length; i++) {\n      let splitFlds = flds[i]?.split('[');\n      if (splitFlds.length === 1) {\n        tempElement = tempElement[flds[i]] || '';\n      } else {\n        let index = Number(splitFlds[1]?.split(']')[0]);\n        tempElement = tempElement[splitFlds[0]]?.[index] || '';\n      }\n    }\n    result[column] = tempElement;\n  });\n\n  return result;\n}\n// SKS27FEB column value get funtion\ngetKeys(obj: any): string[] {\n  return Object.keys(obj);\n}\n  public clickItem(event) {\n    // console.log('inside clickItem of ' + JSON.stringify(event, null, 2));\n    // MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string\n    if(this.apiMeta){\n      this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);    \n      if(this.mode !== 'edit'){\n        this.viewFilterName = this.dataService.getValue(event,this.apiMeta.defaultField ||   Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field)\n      } else {\n        this.filterName = this.dataService.getValue(event,this.apiMeta.defaultField || Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field); //SKS27FEB defaultField is used for showing a search field input if field have array of data\n      }\n    let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');\n    change.fromQuestionId = this.id;\n    change.valueObj = event;\n    this.valueObj = event;\n    change.field = this.apiMeta.defaultField ||  Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field;\n    this.searchValueChange.emit({question : this.question, value: change});\n  }\n  }\n  //RS 03FEB2025\n  // Resets component state when the component is destroyed\n  ngOnDestroy(): void { this.resetComponentState();  }\n  removeCharacters(questionText:string){\n    let updatedText = questionText?.replace(/<[^>]*>/g, '');\n    return updatedText;\n  }\n  inputValue(event,ques){\n    if(ques.singleFieldChange){\n      this.isSingleFieldChange = true;\n      if(this.apiMeta){\n        this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);      \n        let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');\n        change.fromQuestionId = this.id;\n        change.valueObj = this.valueObj;\n        change.field = this.apiMeta.defaultField ||  Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field;\n        change.referenceField = this.question.referenceField;\n        this.filterName = this.viewFilterName\n        this.searchValueChange.emit({question : ques, value: change});\n      }\n    }\n  }\n}\n    ","<!-- VD 12Jun24 - readonly change -->\n<div class=\"search-wrapper\">\n  <nxt-input [type]=\"'list'\" [mode]=\"mode\" [value]=\"filterName\"\n    [labelFont]=\"question.font\" [label]=\"removeCharacters(question?.questionText)\" [labelColor]=\"question.fontColor\"\n    [labelSize]=\"question.fontSize\" [inputValueSize]=\"question.fontSize\" [labelWeight]=\"question.fontWeight\"\n    [inputWeight]=\"question.fontWeight\" [showLabel]=\"question.style?.showLabel\" inputBorder=\"none\" svgHeight=\"20px\"\n    svgWidth=\"20px\" [placeholder]=\"placeHolderText\" [required]=\"question.isOptional\" inputBgColor=\"#FAFAFA\"\n    [inputId]=\"question.trackingId\" [errorMessages]=\"{ required: 'This field is required' }\"\n    [inputIconLeftSrc]=\"question.iconLeftSrc\" (inputValue)=\"getSourceDataLocal($event); inputValue($event.value,$event.question)\" (onBlur)=\"clearList()\"\n    [showSuggestion]=\"showSuggestion\" ariaOwns=\"selectList\" ariaHasPopup=\"listbox\" [isLoading]=\"isLoading\" [question]=\"question\">\n  </nxt-input>\n\n  <div id=\"selectList\" class=\"suggestion-wrapper\" role=\"listbox\">\n    <div *ngIf=\"finalResults().length > 0 && showSuggestion\" class=\"suggestions-container\">\n      <div *ngFor=\"let item of finalResults()\" (click)=\"clickItem(item)\" class=\"suggestion-item\" role=\"option\">\n        <div class=\"grid-x align-middle\">\n          <div *ngIf=\"item.thumbnail\" class=\"cell shrink thumbnail-wrapper\">\n            <img [src]=\"item.thumbnail\" alt=\"Thumbnail\">\n          </div>\n\n          <div class=\"cell auto item-details\">\n            <div *ngFor=\"let key of getKeys(getValues(item, SearchItem))\" class=\"item-value\">\n              {{ getValues(item, SearchItem)[key] }}\n            </div>\n          </div>\n        </div>\n      </div>\n    </div>\n  </div>\n</div>"]}
|
|
415
|
+
//# 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;AAEjM,OAAO,EAAE,aAAa,EAAE,MAAM,2BAA2B,CAAC;AAE1D,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AAE5D,OAAO,EAAE,OAAO,EAAE,YAAY,EAAE,oBAAoB,EAAE,SAAS,EAAE,MAAM,MAAM,CAAC;;;;;;AAa9E,MAAM,OAAO,YAAY;IAkCH;IAAiC;IAAkC;IAAwC;IAjCtH,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;IAExC,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;IAE9C,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,WAAwB,EAAS,WAAwB,EAAU,cAA8B,EAAU,UAAsB;QAAjI,gBAAW,GAAX,WAAW,CAAa;QAAS,gBAAW,GAAX,WAAW,CAAa;QAAU,mBAAc,GAAd,cAAc,CAAgB;QAAU,eAAU,GAAV,UAAU,CAAY;QACnJ,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,CAAC;YACN,IAAI,CAAC,UAAU,GAAG,IAAI,CAAC,QAAQ,EAAE,KAAK,CAAA;QACxC,CAAC;QACD,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,MAAM,CAAC,EAAE,CAAC;YACpB,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC,YAAY,CAAC;QAC3C,CAAC;QACD,IAAI,IAAI,CAAC,IAAI,KAAK,MAAM,IAAI,IAAI,CAAC,mBAAmB,IAAI,OAAO,CAAC,YAAY,CAAC,EAAE,CAAC;YAC9E,MAAM,QAAQ,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC,aAAa,CAAC;YACrD,MAAM,OAAO,GAAG,OAAO,CAAC,YAAY,CAAC,CAAC,YAAY,CAAC;YACnD,IAAI,QAAQ,KAAK,OAAO,EAAE,CAAC;gBACzB,IAAI,CAAC,UAAU,GAAG,QAAQ,CAAC;YAC7B,CAAC;QACH,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,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC,CAAC;QACzB,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,CAAC;YAClB,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC3B,OAAO;QACT,CAAC;QAED,MAAM,WAAW,GAAI,KAAK,CAAC,KAAK,CAAC,QAAQ,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,CAAC,KAAK,CAAC,KAAK,CAAC,IAAI,EAAE,CAAC;QACvG,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,IAAI,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC;QACzC,mCAAmC;QACnC,2DAA2D;QAC3D,kGAAkG;QAClG,+BAA+B;QAC/B,6BAA6B;QAC7B,oDAAoD;QACpD,gBAAgB;QAChB,gDAAgD;QAChD,6CAA6C;QAC7C,uCAAuC;QACvC,WAAW;QACX,uCAAuC;QACvC,IAAI;QAEJ,IAAI,CAAC,WAAW,CAAC,WAAW,CAAC,YAAY,CAAC;aACvC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,QAAQ,CAAC,CAAC;aAC9B,SAAS,CAAC;YACT,IAAI,EAAE,CAAC,WAAW,EAAE,EAAE;gBACpB,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;gBACvB,IAAI,CAAC,oBAAoB,CAAC,WAAW,EAAE,OAAO,CAAC,CAAC;YAClD,CAAC;YACD,KAAK,EAAE,CAAC,KAAK,EAAE,EAAE;gBACf,IAAI,CAAC,SAAS,GAAG,KAAK,CAAC;gBACvB,IAAI,CAAC,QAAQ,GAAG,IAAI,CAAC;gBACrB,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAC5B,CAAC;SACF,CAAC,CAAC;IACP,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,OAAO,CAAC,MAAM,GAAG,CAAC,CAAC,CAAC,6CAA6C;QACzF,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;aAAM,CAAC;YACN,IAAI,CAAC,UAAU,GAAG,YAAY,CAAC;QACjC,CAAC;QACD,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,cAAc,GAAG,IAAI,CAAC,EAAE,CAAC;QAChC,MAAM,CAAC,QAAQ,GAAG,KAAK,CAAC;QACxB,MAAM,CAAC,KAAK,GAAG,YAAY,CAAC;QAE5B,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,gBAAgB,CAAC,YAAoB;QACnC,IAAI,CAAC,YAAY;YAAE,OAAO,EAAE,CAAC;QAC7B,OAAO,YAAY,CAAC,OAAO,CAAC,UAAU,EAAE,EAAE,CAAC,CAAC;IAC9C,CAAC;IAED,UAAU,CAAC,KAAU,EAAE,IAAS;QAC9B,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,IAAI,EAAE,CAAA;QAC9B,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,cAAc,GAAG,IAAI,CAAC,EAAE,CAAC;QAChC,MAAM,CAAC,QAAQ,GAAG,IAAI,CAAC,QAAQ,CAAC;QAChC,MAAM,CAAC,KAAK,GAAG,YAAY,CAAC;QAC5B,MAAM,CAAC,cAAc,GAAG,IAAI,CAAC,QAAQ,CAAC,cAAc,CAAC;QAErD,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;wGAjaU,YAAY;4FAAZ,YAAY,khBCrBzB,moHAmFM,u/DDrEF,YAAY,+PACZ,WAAW,+BACX,QAAQ;;4FAKC,YAAY;kBAXxB,SAAS;+BACE,gBAAgB,cACd,IAAI,WACP;wBACP,YAAY;wBACZ,WAAW;wBACX,QAAQ;qBACT;gKAKQ,eAAe;sBAAvB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,OAAO;sBAAf,KAAK;gBACG,EAAE;sBAAV,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,IAAI;sBAAZ,KAAK;gBAEI,iBAAiB;sBAA1B,MAAM;gBACuC,aAAa;sBAA1D,SAAS;uBAAC,eAAe,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE;gBA4E5C,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 { DataService } from '../../services/data.service';\nimport { ChangeWrapper } from \"../../model/changeWrapper\";\nimport { I18nService } from \"../../i18n.service\";\nimport { CommonModule } from \"@angular/common\";\nimport { FormsModule } from \"@angular/forms\";\nimport { NxtInput } from \"../nxt-input/nxt-input.component\";\nimport { CountryService } from \"../../services/country.service\";\nimport { Subject, debounceTime, distinctUntilChanged, takeUntil } from \"rxjs\";\n\n@Component({\n  selector: 'nxt-search-box',\n  standalone: true,\n  imports: [\n    CommonModule,\n    FormsModule,\n    NxtInput\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  \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\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 dataService: DataService, public i18nService: I18nService, 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 {\n      this.filterName = this.question?.input\n    }\n    this.parseApiMeta();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    if (changes['mode']) {\n      this.mode = changes['mode'].currentValue;\n    }\n    if (this.mode !== 'edit' && this.isSingleFieldChange && changes['filterName']) {\n      const previous = changes['filterName'].previousValue;\n      const current = changes['filterName'].currentValue;\n      if (previous !== current) {\n        this.filterName = previous;\n      }\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) {\n      this.resetComponentState();\n      return;\n    }\n\n    const searchValue =  event.value.includes('|') ? event.value.split('|')[0].trim() : event.value.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    let finalGetCall = this.apiMeta.endpoint;\n    // SKS18JUL25 query params creation\n    // const isFieldArray = Array.isArray(this.apiMeta?.field);\n    // const fields = isFieldArray ? this.apiMeta.field : [this.apiMeta.field]; // 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 (this.apiMeta.endpoint.includes('?')) {\n    //   finalGetCall += `&${queryParams}`;\n    // } else {\n    //   finalGetCall += `?${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  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 = results.length > 0; // 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    } else {\n      this.filterName = displayValue;\n    }\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.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  removeCharacters(questionText: string): string {\n    if (!questionText) return '';\n    return questionText.replace(/<[^>]*>/g, '');\n  }\n\n  inputValue(event: any, ques: any): void {\n    this.filterName = event.trim()\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.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    [value]=\"mode === 'edit' ? filterName : viewFilterName\"\n    [labelFont]=\"question.font\" \n    [label]=\"removeCharacters(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=\"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 for \"{{searchKeyWord}}\"</p>\n      </div>\n    </div>\n  </div>\n</div>"]}
|