@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.
Files changed (24) hide show
  1. package/esm2022/environments/version.mjs +2 -2
  2. package/esm2022/lib/components/button/nxt-button.component.mjs +4 -7
  3. package/esm2022/lib/components/custom-dropdown/custom-dropdown.component.mjs +1 -1
  4. package/esm2022/lib/components/datatable/datatable.component.mjs +3 -3
  5. package/esm2022/lib/components/file-upload/file-upload.component.mjs +5 -2
  6. package/esm2022/lib/components/nxt-input/nxt-input.component.mjs +10 -4
  7. package/esm2022/lib/components/pick-location/pick-location.component.mjs +1 -1
  8. package/esm2022/lib/components/search-box/search-box.component.mjs +294 -158
  9. package/esm2022/lib/pages/booklet/booklet.component.mjs +6 -8
  10. package/esm2022/lib/pages/builder/properties/properties.component.mjs +1 -1
  11. package/esm2022/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.mjs +9 -3
  12. package/esm2022/lib/pages/pdfDesigner/pdf-properties/pdf-properties.component.mjs +2 -2
  13. package/esm2022/lib/pages/questionbook/questionbook.component.mjs +4 -4
  14. package/esm2022/lib/pages/questionnaire/questionnaire.component.mjs +2 -2
  15. package/esm2022/lib/services/translation.service.mjs +2 -1
  16. package/fesm2022/rangertechnologies-ngnxt.mjs +333 -186
  17. package/fesm2022/rangertechnologies-ngnxt.mjs.map +1 -1
  18. package/lib/components/nxt-input/nxt-input.component.d.ts +3 -1
  19. package/lib/components/search-box/search-box.component.d.ts +26 -10
  20. package/lib/pages/booklet/booklet.component.d.ts +1 -1
  21. package/lib/pages/pdfDesigner/pdf-designer/pdf-designer.component.d.ts +3 -1
  22. package/package.json +1 -1
  23. package/rangertechnologies-ngnxt-2.1.241.tgz +0 -0
  24. 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
- placeHolderText;
16
+ elementRef;
17
+ placeHolderText = '';
17
18
  question;
18
19
  apiMeta;
19
- id;
20
- readOnly = false; // VD 12Jun24 - readonly change
21
- mode = 'edit'; // SKS11JUN25 New mode input
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
- constructor(dataService, i18nService, countryService) {
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
- // VD 03May- search changes
45
- // AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string
46
- if (this.apiMeta) {
47
- this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
48
- this.SearchItem = this.apiMeta.field;
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 && (changes.filterName.previousValue !== changes.filterName.currentValue)) {
59
- this.filterName = changes.filterName.previousValue;
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.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
63
- this.SearchItem = this.apiMeta?.field;
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
- this.finalResults.set([]);
79
- ;
80
- }, 1000);
137
+ // Remove the setTimeout delay that was causing issues
138
+ this.hideSuggestions();
81
139
  }
82
140
  getSourceDataLocal(event) {
83
- if (event.value.length > 2 && !this.isSingleFieldChange) {
84
- event.question?.singleFieldChange ? this.showSuggestion = false : this.showSuggestion = true;
85
- this.finalResults.set([]);
86
- this.searchKeyWord = event.value;
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.getSourceData(event.value);
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
- // VD 03May- search changes
98
- // VD 31NOV24 null check
99
- // RS 29JAN25
100
- //Multi-word search across all object values
101
- getSourceData = (keyword) => {
102
- // MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string
103
- if (this.apiMeta) {
104
- this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
105
- if (!this.apiMeta.isStaticData) {
106
- this.isLoading = true;
107
- this.dataService.apiResponse(this.apiMeta.endpoint).subscribe((apiResponse) => {
108
- this.isLoading = false;
109
- let response;
110
- if (this.apiMeta.variable) {
111
- response = this.dataService.getValue(apiResponse, this.apiMeta.variable);
112
- }
113
- else {
114
- response = apiResponse;
115
- }
116
- let results = [];
117
- let searchTerms = keyword.toLowerCase()?.split(" ");
118
- for (let i = 0; i < response.length; i++) {
119
- let obj = response[i];
120
- let combinedValues = this.getCombinedValuesFromColumns(obj, this.apiMeta.field);
121
- let match = searchTerms.every(term => combinedValues.includes(term));
122
- if (match) {
123
- results.push(obj);
124
- }
125
- }
126
- this.noResult = results.length === 0;
127
- this.finalResults.set(results);
128
- }, (error) => {
129
- console.error('API error:', error);
130
- this.isLoading = false;
131
- this.noResult = true;
132
- this.finalResults.set([]);
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
- // SKS27FEB Helper functions
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
- //SKS27FEB Convert array indexes to dot notation (e.g., [0] -> .0)
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
- // SKS28FEB check if columns is an array
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
- // SKS28FEB check if columns is an array
187
- if (typeof columns === 'string' && !Array.isArray(columns)) {
188
- columns = [columns];
189
- }
190
- columns.forEach((column) => {
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
- let flds = column?.split('.');
193
- for (let i = 0; i < flds.length; i++) {
194
- let splitFlds = flds[i]?.split('[');
195
- if (splitFlds.length === 1) {
196
- tempElement = tempElement[flds[i]] || '';
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
- let index = Number(splitFlds[1]?.split(']')[0]);
200
- tempElement = tempElement[splitFlds[0]]?.[index] || '';
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
- // console.log('inside clickItem of ' + JSON.stringify(event, null, 2));
213
- // MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string
214
- if (this.apiMeta) {
215
- this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
216
- if (this.mode !== 'edit') {
217
- this.viewFilterName = this.dataService.getValue(event, this.apiMeta.defaultField || Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field);
218
- }
219
- else {
220
- 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
221
- }
222
- let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');
223
- change.fromQuestionId = this.id;
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
- let updatedText = questionText?.replace(/<[^>]*>/g, '');
235
- return updatedText;
334
+ if (!questionText)
335
+ return '';
336
+ return questionText.replace(/<[^>]*>/g, '');
236
337
  }
237
338
  inputValue(event, ques) {
238
- if (ques.singleFieldChange) {
239
- this.isSingleFieldChange = true;
240
- if (this.apiMeta) {
241
- this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
242
- let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');
243
- change.fromQuestionId = this.id;
244
- change.valueObj = this.valueObj;
245
- change.field = this.apiMeta.defaultField || Array.isArray(this.apiMeta?.field) ? this.apiMeta?.field?.[0] : this.apiMeta.field;
246
- change.referenceField = this.question.referenceField;
247
- this.filterName = this.viewFilterName;
248
- this.searchValueChange.emit({ question: ques, value: change });
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
- 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 }], target: i0.ɵɵFactoryTarget.Component });
253
- 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", filterName: "filterName" }, outputs: { searchValueChange: "searchValueChange" }, usesOnChanges: true, ngImport: i0, 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,$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>", styles: [".suggestion-wrapper{position:absolute;top:100%;left:0;background:#fff;z-index:2}.search-wrapper{position:relative}.suggestions-container{max-height:20vh;border:1px solid #d2d4d6;overflow:auto;min-width:100px}.suggestion-item{cursor:pointer}.thumbnail-wrapper{width:60px;margin-right:1.6rem}.thumbnail-wrapper img{width:60px}.item-details{text-align:left;padding:5px 8px 0;display:flex;gap:5px}.item-value{display:flex}\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"] }] });
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, '&amp;')
373
+ .replace(/</g, '&lt;')
374
+ .replace(/>/g, '&gt;')
375
+ .replace(/"/g, '&quot;')
376
+ .replace(/'/g, '&#39;');
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,$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>", styles: [".suggestion-wrapper{position:absolute;top:100%;left:0;background:#fff;z-index:2}.search-wrapper{position:relative}.suggestions-container{max-height:20vh;border:1px solid #d2d4d6;overflow:auto;min-width:100px}.suggestion-item{cursor:pointer}.thumbnail-wrapper{width:60px;margin-right:1.6rem}.thumbnail-wrapper img{width:60px}.item-details{text-align:left;padding:5px 8px 0;display:flex;gap:5px}.item-value{display:flex}\n"] }]
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
- }], filterName: [{
277
- type: Input
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":"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","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>"]}