@rangertechnologies/ngnxt
Version:
This library was used for creating dymanic UI based on the input JSON/data
213 lines • 38.5 kB
JavaScript
import { Component, Input, Output, EventEmitter, ViewChild, } from "@angular/core";
import { ChangeWrapper } from "../../model/changeWrapper";
import * as i0 from "@angular/core";
import * as i1 from "../../services/data.service";
import * as i2 from "../../i18n.service";
import * as i3 from "@angular/common";
import * as i4 from "@angular/forms";
export class SearchBoxComponent {
dataService;
i18nService;
placeHolderText;
auto;
question;
apiMeta;
id;
readOnly = false; // VD 12Jun24 - readonly change
searchValueChange = new EventEmitter();
apiObj;
SearchItem;
filterName; // VD 20Aug default filter value as input
finalResults = [];
searchKeyWord = '';
newResult;
showResult = false;
noResult = false;
showSuggestion = true;
el;
serv = 'api';
tkn = '';
constructor(dataService, i18nService) {
this.dataService = dataService;
this.i18nService = i18nService;
}
ngOnInit() {
// VD 03May- search changes
// AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string
if (this.apiMeta) {
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
let apiObj = this.apiMeta;
this.SearchItem = apiObj.field;
}
}
////RS 03FEB2025
// Resets state when filterName or apiMeta changes to reflect updated data
// AP-26MAR25 Ensure ques.subText is always an object by parsing it if it's a string
ngOnChanges(changes) {
if (changes['apiMeta'] && this.apiMeta) {
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
this.apiObj = this.apiMeta;
this.SearchItem = this.apiObj?.field;
}
}
//RS 03FEB2025
// Clears search-related data, including results, search term, and suggestions.
resetComponentState() {
this.finalResults = [];
this.filterName = '';
this.searchKeyWord = '';
this.showSuggestion = false;
this.noResult = false;
}
clearList() {
setTimeout(() => {
this.finalResults = [];
}, 1000);
}
getSourceDataLocal(event) {
event.preventDefault();
if (event.target.value.length > 2) {
this.showSuggestion = true;
this.finalResults = [];
this.searchKeyWord = event.target.value;
this.showResult = false;
this.getSourceData(event.target.value);
}
else {
this.showSuggestion = false;
this.finalResults = [];
this.noResult = false;
}
}
// VD 03May- search changes
// VD 31NOV24 null check
// RS 29JAN25
//Multi-word search across all object values
getSourceData = (keyword) => {
// MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string
if (this.apiMeta) {
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
let apiObj = this.apiMeta;
this.dataService.apiResponse(apiObj.endpoint).subscribe((apiResponse) => {
let response;
if (apiObj.variable) {
response = this.dataService.getValue(apiResponse, apiObj.variable);
}
else {
response = apiResponse;
}
let results = [];
let searchTerms = keyword.toLowerCase()?.split(" ");
for (let i = 0; i < response.length; i++) {
let obj = response[i];
let combinedValues = this.getCombinedValuesFromColumns(obj, apiObj.field);
let match = searchTerms.every(term => combinedValues.includes(term));
if (match) {
results.push(obj);
}
}
this.noResult = results.length === 0;
this.finalResults = results;
});
}
};
// SKS27FEB Helper functions
getNestedValue(obj, path) {
//SKS27FEB Convert array indexes to dot notation (e.g., [0] -> .0)
const processedPath = path.replace(/\[(\d+)\]/g, '.$1');
const parts = processedPath.split('.');
let current = obj;
for (const part of parts) {
if (!current || typeof current !== 'object')
return '';
current = current[part];
}
return current !== null && current !== undefined
? String(current).toLowerCase().trim()
: '';
}
getCombinedValuesFromColumns(obj, columns) {
const values = [];
// SKS28FEB check if columns is an array
if (typeof columns === 'string' && !Array.isArray(columns)) {
columns = [columns];
}
for (const column of columns) {
const value = this.getNestedValue(obj, column);
if (value) {
values.push(value);
}
}
return values.join(' ');
}
// SKS27FEB get value from data specific column
getValues(element, columns) {
const result = {};
// SKS28FEB check if columns is an array
if (typeof columns === 'string' && !Array.isArray(columns)) {
columns = [columns];
}
columns.forEach((column) => {
let tempElement = element;
let flds = column?.split('.');
for (let i = 0; i < flds.length; i++) {
let splitFlds = flds[i]?.split('[');
if (splitFlds.length === 1) {
tempElement = tempElement[flds[i]] || '';
}
else {
let index = Number(splitFlds[1]?.split(']')[0]);
tempElement = tempElement[splitFlds[0]]?.[index] || '';
}
}
result[column] = tempElement;
});
return result;
}
// SKS27FEB column value get funtion
getKeys(obj) {
return Object.keys(obj);
}
clickItem(event) {
// console.log('inside clickItem of ' + JSON.stringify(event, null, 2));
// MSM-27MAR25 Ensure ques.subText is always an object by parsing it if it's a string
if (this.apiMeta) {
this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);
let apiObj = this.apiMeta;
// let apiObj: APIMeta = JSON.parse(this.apiMeta);
this.filterName = event[apiObj.defaultField || apiObj.field[0] || apiObj.field]; //SKS27FEB defaultField is used for showing a search field input if field have array of data
let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');
change.fromQuestionId = this.id;
change.valueObj = event;
change.field = apiObj.field;
this.searchValueChange.emit(change);
}
}
//RS 03FEB2025
// Resets component state when the component is destroyed
ngOnDestroy() { this.resetComponentState(); }
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SearchBoxComponent, deps: [{ token: i1.DataService }, { token: i2.I18nService }], target: i0.ɵɵFactoryTarget.Component });
static ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.13", type: SearchBoxComponent, selector: "lib-search-box", inputs: { placeHolderText: "placeHolderText", question: "question", apiMeta: "apiMeta", id: "id", readOnly: "readOnly", filterName: "filterName" }, outputs: { searchValueChange: "searchValueChange" }, viewQueries: [{ propertyName: "auto", first: true, predicate: ["auto"], descendants: true }], usesOnChanges: true, ngImport: i0, template: "<!-- // VD 12Jun24 - readonly change-->\n<div id=\"autocomplete-input\"> <!-- SKS5NOV25 search icon -->\n <input #auto id=\"searchbox-style\"\n (blur)=\"clearList()\"\n [(ngModel)]=\"filterName\"\n type=\"text\"\n name=\"name\"\n [readOnly]=\"readOnly\"\n [placeholder]=\"placeHolderText\"\n style=\"margin: 0 !important; padding-right: 30px;\"\n class=\"searchInput she-line-input form-control\"\n (focusin)=\"getSourceDataLocal($event)\"\n (input)=\"getSourceDataLocal($event)\">\n <div id=\"selectList\" style=\"position: absolute;position: absolute;background: white;z-index: 2;\">\n <div *ngIf=\"finalResults.length > 0 && showSuggestion\"\n style=\"max-height: 20vh;border: 1px solid #d2d4d6;overflow: scroll; min-width:100px\"\n class=\"suggestions-container\">\n <!-- HA 20DEC23 Uncommented the logic -->\n <!-- VD 03May- search changes -->\n <div *ngFor=\"let item of finalResults\" (click)='clickItem(item)' class=\"hoover\">\n <!-- VD 26Jun24 - id condition removed -->\n <div class=\"grid-x align-middle\" style=\"\">\n <div *ngIf=\"item.thumbnail\" class=\"cell shrink\" style=\"width: 60px; margin-right: 1.6rem;\">\n <img [src]=\"item.thumbnail\" style=\"width: 60px;\" alt=\"\">\n </div>\n <div class=\"cell auto\" style=\"text-align: left; padding:5px 8px 0 8px; display: flex; gap: 5px; \">\n <!--// VD 26JUN24 - pipe changes -->\n <!-- RS 29JAN25 -->\n <div style=\"display: flex;\" *ngFor=\"let key of getKeys(getValues(item, SearchItem))\">\n <div>\n {{ getValues(item, SearchItem)[key] }}\n </div>\n </div> \n </div>\n </div>\n </div>\n <!-- HA 20DEC23 For Commented this for future purpose -->\n <!-- <table class=\"table table-striped table-bordered\">\n <thead>\n <tr>\n <th>{{ 'firstName' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'lastName' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'division' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'numberPlate' | i18n:i18nService.currentLanguage }}</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let item of finalResults\" (click)='clickItem(item)'>\n <td>{{ item.firstName }}</td>\n <td>{{ item.lastName }}</td>\n <td>{{ item.division }}</td>\n <td>{{ item.numberPlate }}</td>\n </tr>\n </tbody>\n </table> -->\n </div>\n</div>\n\n", styles: ["#searchbox-style{background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"11\" cy=\"11\" r=\"7\" stroke=\"%23434555\" stroke-opacity=\"0.65\" stroke-width=\"2\"/><path d=\"M20 20L17 17\" stroke=\"%23434555\" stroke-opacity=\"0.65\" stroke-width=\"2\" stroke-linecap=\"round\"/></svg>');background-position:right 5px center;background-repeat:no-repeat;background-size:24px;padding-right:35px}\n"], dependencies: [{ kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i4.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i4.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i4.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }] });
}
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.13", ngImport: i0, type: SearchBoxComponent, decorators: [{
type: Component,
args: [{ selector: 'lib-search-box', template: "<!-- // VD 12Jun24 - readonly change-->\n<div id=\"autocomplete-input\"> <!-- SKS5NOV25 search icon -->\n <input #auto id=\"searchbox-style\"\n (blur)=\"clearList()\"\n [(ngModel)]=\"filterName\"\n type=\"text\"\n name=\"name\"\n [readOnly]=\"readOnly\"\n [placeholder]=\"placeHolderText\"\n style=\"margin: 0 !important; padding-right: 30px;\"\n class=\"searchInput she-line-input form-control\"\n (focusin)=\"getSourceDataLocal($event)\"\n (input)=\"getSourceDataLocal($event)\">\n <div id=\"selectList\" style=\"position: absolute;position: absolute;background: white;z-index: 2;\">\n <div *ngIf=\"finalResults.length > 0 && showSuggestion\"\n style=\"max-height: 20vh;border: 1px solid #d2d4d6;overflow: scroll; min-width:100px\"\n class=\"suggestions-container\">\n <!-- HA 20DEC23 Uncommented the logic -->\n <!-- VD 03May- search changes -->\n <div *ngFor=\"let item of finalResults\" (click)='clickItem(item)' class=\"hoover\">\n <!-- VD 26Jun24 - id condition removed -->\n <div class=\"grid-x align-middle\" style=\"\">\n <div *ngIf=\"item.thumbnail\" class=\"cell shrink\" style=\"width: 60px; margin-right: 1.6rem;\">\n <img [src]=\"item.thumbnail\" style=\"width: 60px;\" alt=\"\">\n </div>\n <div class=\"cell auto\" style=\"text-align: left; padding:5px 8px 0 8px; display: flex; gap: 5px; \">\n <!--// VD 26JUN24 - pipe changes -->\n <!-- RS 29JAN25 -->\n <div style=\"display: flex;\" *ngFor=\"let key of getKeys(getValues(item, SearchItem))\">\n <div>\n {{ getValues(item, SearchItem)[key] }}\n </div>\n </div> \n </div>\n </div>\n </div>\n <!-- HA 20DEC23 For Commented this for future purpose -->\n <!-- <table class=\"table table-striped table-bordered\">\n <thead>\n <tr>\n <th>{{ 'firstName' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'lastName' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'division' | i18n:i18nService.currentLanguage }}</th>\n <th>{{ 'numberPlate' | i18n:i18nService.currentLanguage }}</th>\n </tr>\n </thead>\n <tbody>\n <tr *ngFor=\"let item of finalResults\" (click)='clickItem(item)'>\n <td>{{ item.firstName }}</td>\n <td>{{ item.lastName }}</td>\n <td>{{ item.division }}</td>\n <td>{{ item.numberPlate }}</td>\n </tr>\n </tbody>\n </table> -->\n </div>\n</div>\n\n", styles: ["#searchbox-style{background-image:url('data:image/svg+xml;utf8,<svg width=\"24\" height=\"24\" viewBox=\"0 0 24 24\" fill=\"none\" xmlns=\"http://www.w3.org/2000/svg\"><circle cx=\"11\" cy=\"11\" r=\"7\" stroke=\"%23434555\" stroke-opacity=\"0.65\" stroke-width=\"2\"/><path d=\"M20 20L17 17\" stroke=\"%23434555\" stroke-opacity=\"0.65\" stroke-width=\"2\" stroke-linecap=\"round\"/></svg>');background-position:right 5px center;background-repeat:no-repeat;background-size:24px;padding-right:35px}\n"] }]
}], ctorParameters: () => [{ type: i1.DataService }, { type: i2.I18nService }], propDecorators: { placeHolderText: [{
type: Input
}], auto: [{
type: ViewChild,
args: ['auto']
}], question: [{
type: Input
}], apiMeta: [{
type: Input
}], id: [{
type: Input
}], readOnly: [{
type: Input
}], searchValueChange: [{
type: Output
}], filterName: [{
type: Input
}] } });
//# 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,\n} from \"@angular/core\";\nimport { DataService } from '../../services/data.service';\n// VD 23JAN24 removed httpClient import used service file for callout\nimport { APIMeta } from \"../../interfaces/apimeta\";\nimport { ChangeWrapper } from \"../../model/changeWrapper\";\n// HA 19DEC23 imported translation service\nimport { I18nService } from \"../../i18n.service\";\n\n@Component({\n  selector: 'lib-search-box',\n  templateUrl: './search-box.component.html',\n  styleUrls: ['./search-box.component.css']\n})\nexport class SearchBoxComponent implements OnInit {\n  @Input() placeHolderText: string;\n  @ViewChild('auto') auto;\n\n  @Input() question;\n  @Input() apiMeta: string;\n  @Input() id: string;\n  @Input() readOnly = false; // VD 12Jun24 - readonly change\n  @Output() searchValueChange: EventEmitter<ChangeWrapper> = new EventEmitter<ChangeWrapper>();\n   apiObj:any;\n   SearchItem: any;\n\n  @Input() filterName: string; // VD 20Aug default filter value as input\n  public finalResults: any[] = [];\n  public searchKeyWord: string = '';\n  public newResult: any;\n  public showResult = false;\n  public noResult = false;\n  public showSuggestion = true;\n  private el: HTMLElement;\n  private serv: string = 'api';\n  private tkn: string = '';\n  \n  constructor( private dataService: DataService, public i18nService: I18nService) { }\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      let apiObj: any = this.apiMeta;\n      this.SearchItem = apiObj.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['apiMeta'] && this.apiMeta) {\n    this.apiMeta = typeof this.apiMeta === 'object' ? this.apiMeta : JSON.parse(this.apiMeta);\n    this.apiObj = this.apiMeta;\n    this.SearchItem = this.apiObj?.field;\n  }\n}\n//RS 03FEB2025\n// Clears search-related data, including results, search term, and suggestions.\nprivate resetComponentState(): void {\n  this.finalResults = [];\n  this.filterName = ''; \n  this.searchKeyWord = '';  \n  this.showSuggestion = false;\n  this.noResult = false;\n}\nclearList(){\n  setTimeout(()=> {\n    this.finalResults = [];\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  event.preventDefault();\n  if(event.target.value.length > 2){\n    this.showSuggestion=true;\n    this.finalResults=[];\n    this.searchKeyWord = event.target.value;\n    this.showResult = false;\n    this.getSourceData(event.target.value);\n  } else {\n    this.showSuggestion=false;\n    this.finalResults = [];\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      let apiObj: any = this.apiMeta;\n    \n    this.dataService.apiResponse(apiObj.endpoint).subscribe((apiResponse) => {\n      let response;\n      if (apiObj.variable) {\n        response = this.dataService.getValue(apiResponse, apiObj.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, apiObj.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 = 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      let apiObj: any = this.apiMeta;\n    \n    // let apiObj: APIMeta = JSON.parse(this.apiMeta);\n    this.filterName = event[apiObj.defaultField || apiObj.field[0] || apiObj.field]; //SKS27FEB defaultField is used for showing a search field input if field have array of data\n    let change = new ChangeWrapper(); // ChangeWrapper = JSON.parse('{}');\n    change.fromQuestionId = this.id;\n    change.valueObj = event;\n    change.field = apiObj.field;\n    this.searchValueChange.emit(change);\n  }\n  }\n  //RS 03FEB2025\n  // Resets component state when the component is destroyed\n  ngOnDestroy(): void { this.resetComponentState();  }\n}\n    ","<!-- // VD 12Jun24 - readonly change-->\n<div id=\"autocomplete-input\"> <!-- SKS5NOV25 search icon -->\n  <input #auto id=\"searchbox-style\"\n          (blur)=\"clearList()\"\n          [(ngModel)]=\"filterName\"\n          type=\"text\"\n          name=\"name\"\n          [readOnly]=\"readOnly\"\n          [placeholder]=\"placeHolderText\"\n          style=\"margin: 0 !important; padding-right: 30px;\"\n          class=\"searchInput she-line-input form-control\"\n          (focusin)=\"getSourceDataLocal($event)\"\n          (input)=\"getSourceDataLocal($event)\">\n  <div id=\"selectList\" style=\"position: absolute;position: absolute;background: white;z-index: 2;\">\n  <div *ngIf=\"finalResults.length > 0 && showSuggestion\"\n    style=\"max-height: 20vh;border: 1px solid #d2d4d6;overflow: scroll; min-width:100px\"\n       class=\"suggestions-container\">\n      <!-- HA 20DEC23 Uncommented the logic -->\n       <!-- VD 03May- search changes -->\n      <div *ngFor=\"let item of finalResults\" (click)='clickItem(item)' class=\"hoover\">\n        <!-- VD 26Jun24 - id condition removed -->\n          <div class=\"grid-x align-middle\" style=\"\">\n              <div *ngIf=\"item.thumbnail\" class=\"cell shrink\" style=\"width: 60px; margin-right: 1.6rem;\">\n                  <img [src]=\"item.thumbnail\" style=\"width: 60px;\" alt=\"\">\n              </div>\n              <div class=\"cell auto\" style=\"text-align: left; padding:5px 8px 0 8px; display: flex; gap: 5px; \">\n                <!--// VD 26JUN24 - pipe changes -->\n                <!-- RS 29JAN25 -->\n                <div style=\"display: flex;\" *ngFor=\"let key of getKeys(getValues(item, SearchItem))\">\n                  <div>\n                    {{ getValues(item, SearchItem)[key] }}\n                  </div>\n                </div>                \n              </div>\n          </div>\n      </div>\n      <!-- HA 20DEC23 For Commented this for future purpose -->\n      <!-- <table class=\"table table-striped table-bordered\">\n          <thead>\n            <tr>\n              <th>{{ 'firstName' | i18n:i18nService.currentLanguage }}</th>\n              <th>{{ 'lastName' | i18n:i18nService.currentLanguage }}</th>\n              <th>{{ 'division' | i18n:i18nService.currentLanguage }}</th>\n              <th>{{ 'numberPlate' | i18n:i18nService.currentLanguage }}</th>\n            </tr>\n          </thead>\n          <tbody>\n            <tr *ngFor=\"let item of finalResults\" (click)='clickItem(item)'>\n              <td>{{ item.firstName }}</td>\n              <td>{{ item.lastName }}</td>\n              <td>{{ item.division }}</td>\n              <td>{{ item.numberPlate }}</td>\n            </tr>\n          </tbody>\n        </table> -->\n  </div>\n</div>\n\n"]}