valtech-components 2.0.150 → 2.0.152

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.
@@ -1,5 +1,5 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component, inject, Input, ViewChild, ChangeDetectorRef } from '@angular/core';
2
+ import { Component, inject, Input, ViewChild, ChangeDetectorRef, } from '@angular/core';
3
3
  import { FormsModule, ReactiveFormsModule } from '@angular/forms';
4
4
  import { IonicModule } from '@ionic/angular';
5
5
  import { IconService } from '../../../services/icons.service';
@@ -22,7 +22,6 @@ export class SelectSearchComponent {
22
22
  this.filteredItems = [];
23
23
  this.selectedItems = [];
24
24
  this.displayValue = '';
25
- // Usar para evitar bucles infinitos
26
25
  this.previousOptions = [];
27
26
  this.isProcessingChanges = false;
28
27
  }
@@ -30,26 +29,43 @@ export class SelectSearchComponent {
30
29
  this.initializeItems();
31
30
  this.syncControlValueWithSelectedItems();
32
31
  this.updateDisplayValue();
32
+ this.subscribeToValueChanges();
33
+ }
34
+ ngOnDestroy() {
35
+ // Limpiar suscripciones para evitar memory leaks
36
+ if (this.valueChangesSubscription) {
37
+ this.valueChangesSubscription.unsubscribe();
38
+ }
33
39
  }
34
40
  ngOnChanges(changes) {
35
41
  // Evitar bucles infinitos
36
42
  if (this.isProcessingChanges) {
37
43
  return;
38
44
  }
39
- if (changes['props'] && this.props?.options) {
40
- // Verificar si las opciones han cambiado realmente
41
- const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
42
- if (optionsChanged) {
43
- try {
44
- this.isProcessingChanges = true;
45
- this.previousOptions = [...this.props.options];
46
- this.initializeItems();
47
- this.syncControlValueWithSelectedItems();
48
- this.updateDisplayValue();
45
+ // Cuando cambia props o props.options
46
+ if (changes['props']) {
47
+ try {
48
+ this.isProcessingChanges = true;
49
+ // Desuscribirse del antiguo control si existe
50
+ if (this.valueChangesSubscription) {
51
+ this.valueChangesSubscription.unsubscribe();
49
52
  }
50
- finally {
51
- this.isProcessingChanges = false;
53
+ if (this.props?.options) {
54
+ // Verificar si las opciones han cambiado realmente
55
+ const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
56
+ if (optionsChanged) {
57
+ this.previousOptions = [...this.props.options];
58
+ this.initializeItems();
59
+ }
52
60
  }
61
+ // Sincronizar con el nuevo control si existe
62
+ this.syncControlValueWithSelectedItems();
63
+ this.updateDisplayValue();
64
+ // Suscribirse al nuevo control
65
+ this.subscribeToValueChanges();
66
+ }
67
+ finally {
68
+ this.isProcessingChanges = false;
53
69
  }
54
70
  }
55
71
  }
@@ -62,11 +78,29 @@ export class SelectSearchComponent {
62
78
  this.initializeItems();
63
79
  this.syncControlValueWithSelectedItems();
64
80
  this.updateDisplayValue();
81
+ this.subscribeToValueChanges();
65
82
  }
66
83
  finally {
67
84
  this.isProcessingChanges = false;
68
85
  }
69
86
  }
87
+ // Suscribirse a cambios en el FormControl
88
+ subscribeToValueChanges() {
89
+ if (!this.props?.control)
90
+ return;
91
+ this.valueChangesSubscription = this.props.control.valueChanges.subscribe(value => {
92
+ if (this.isProcessingChanges)
93
+ return;
94
+ try {
95
+ this.isProcessingChanges = true;
96
+ this.syncControlValueWithSelectedItems();
97
+ this.updateDisplayValue();
98
+ }
99
+ finally {
100
+ this.isProcessingChanges = false;
101
+ }
102
+ });
103
+ }
70
104
  // Compara si dos arrays de opciones son iguales
71
105
  areOptionsEqual(prevOptions, newOptions) {
72
106
  if (!prevOptions || !newOptions) {
@@ -75,8 +109,6 @@ export class SelectSearchComponent {
75
109
  if (prevOptions.length !== newOptions.length) {
76
110
  return false;
77
111
  }
78
- // Comparación simple basada en longitud y valores de ID
79
- // Esto se puede mejorar según tus necesidades específicas
80
112
  for (let i = 0; i < prevOptions.length; i++) {
81
113
  if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
82
114
  return false;
@@ -94,18 +126,19 @@ export class SelectSearchComponent {
94
126
  }
95
127
  syncControlValueWithSelectedItems() {
96
128
  if (!this.props?.control) {
129
+ this.selectedItems = [];
97
130
  return;
98
131
  }
99
- // Sincronizar el valor del control con los elementos seleccionados
100
- if (this.props.control.value) {
101
- const controlValue = this.props.control.value;
102
- const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);
103
- if (selectedOption) {
104
- this.selectedItems = [selectedOption];
105
- }
106
- else {
107
- this.selectedItems = [];
108
- }
132
+ const controlValue = this.props.control.value;
133
+ // Si el control no tiene valor, limpiar seleccionados
134
+ if (controlValue === null || controlValue === undefined) {
135
+ this.selectedItems = [];
136
+ return;
137
+ }
138
+ // Buscar la opción que corresponde al valor del control
139
+ if (this.props.options && this.props.options.length > 0) {
140
+ const selectedOption = this.props.options.find(opt => opt[this.valueProperty] === controlValue);
141
+ this.selectedItems = selectedOption ? [selectedOption] : [];
109
142
  }
110
143
  else {
111
144
  this.selectedItems = [];
@@ -148,8 +181,8 @@ export class SelectSearchComponent {
148
181
  }
149
182
  }
150
183
  preventDefaultBehavior(event) {
151
- event.preventDefault(); // Evita el comportamiento predeterminado del selector
152
- event.stopPropagation(); // Detiene la propagación del evento
184
+ event.preventDefault();
185
+ event.stopPropagation();
153
186
  this.openModal();
154
187
  }
155
188
  cancelModal() {
@@ -198,14 +231,29 @@ export class SelectSearchComponent {
198
231
  if (!this.props?.control) {
199
232
  return;
200
233
  }
201
- if (this.selectedItems.length > 0) {
202
- this.props.control.setValue(this.selectedItems[0][this.valueProperty]);
234
+ try {
235
+ this.isProcessingChanges = true;
236
+ if (this.selectedItems.length > 0) {
237
+ this.props.control.setValue(this.selectedItems[0][this.valueProperty]);
238
+ }
239
+ else {
240
+ this.props.control.setValue(null);
241
+ }
242
+ this.props.control.markAsDirty();
243
+ this.props.control.updateValueAndValidity();
203
244
  }
204
- else {
245
+ finally {
246
+ this.isProcessingChanges = false;
247
+ }
248
+ }
249
+ // Método público para reiniciar el componente
250
+ reset() {
251
+ this.selectedItems = [];
252
+ this.displayValue = '';
253
+ if (this.props?.control) {
205
254
  this.props.control.setValue(null);
206
255
  }
207
- this.props.control.updateValueAndValidity();
208
- this.props.control.markAsDirty();
256
+ this.changeDetector.detectChanges();
209
257
  }
210
258
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SelectSearchComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
211
259
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SelectSearchComponent, isStandalone: true, selector: "val-select-search", inputs: { label: "label", labelProperty: "labelProperty", valueProperty: "valueProperty", multiple: "multiple", placeholder: "placeholder", props: "props" }, viewQueries: [{ propertyName: "modal", first: true, predicate: ["modal"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
@@ -303,4 +351,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
303
351
  }], props: [{
304
352
  type: Input
305
353
  }] } });
306
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select-search.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/select-search/select-search.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, inject, Input, OnChanges, SimpleChanges, ViewChild, ChangeDetectorRef } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { ViewWillEnter } from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\n\n@Component({\n  selector: 'val-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, SearchbarComponent, ReactiveFormsModule],\n  template: `\n    <ion-input\n      type=\"text\"\n      [value]=\"displayValue\"\n      [placeholder]=\"props?.placeholder || placeholder\"\n      readonly\n      (mousedown)=\"preventDefaultBehavior($event)\"\n    />\n\n    <ion-input style=\"position: absolute;\" [formControl]=\"props.control\" type=\"hidden\"></ion-input>\n\n    <ion-modal #modal trigger=\"open-modal\" [initialBreakpoint]=\"0.75\" [breakpoints]=\"[0, 0.5, 0.75, 1]\">\n      <ng-template>\n        <ion-header>\n          <ion-toolbar>\n            <ion-title>{{ label }}</ion-title>\n            <ion-buttons slot=\"end\">\n              <ion-button (click)=\"cancelModal()\">Cancelar</ion-button>\n            </ion-buttons>\n          </ion-toolbar>\n          <ion-toolbar>\n            <val-searchbar (filterEvent)=\"onFilter($event)\" (focusEvent)=\"onFocus()\" (blurEvent)=\"onBlur()\" />\n          </ion-toolbar>\n        </ion-header>\n        <ion-content>\n          <ion-list>\n            <ion-item *ngFor=\"let item of filteredItems\" button (click)=\"selectItem(item)\" detail=\"false\">\n              <ion-label>{{ item[labelProperty] }}</ion-label>\n              <ion-icon *ngIf=\"isItemSelected(item)\" name=\"checkmark-outline\" slot=\"end\" color=\"primary\"></ion-icon>\n            </ion-item>\n            <ion-item *ngIf=\"filteredItems.length === 0\">\n              <ion-label color=\"medium\">No se encontraron resultados</ion-label>\n            </ion-item>\n          </ion-list>\n        </ion-content>\n      </ng-template>\n    </ion-modal>\n  `,\n  styles: [``],\n})\nexport class SelectSearchComponent implements ViewWillEnter, OnChanges {\n  @ViewChild('modal') modal!: IonModal;\n\n  @Input() label: string = 'Seleccionar';\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() multiple: boolean = false;\n  @Input() placeholder: string = 'Seleccione una opción';\n  @Input() props: InputMetadata;\n\n  icon = inject(IconService);\n  private changeDetector = inject(ChangeDetectorRef);\n\n  searchTerm: string = '';\n  filteredItems: InputOption[] = [];\n  selectedItems: InputOption[] = [];\n  displayValue: string = '';\n\n  // Usar para evitar bucles infinitos\n  private previousOptions: any[] = [];\n  private isProcessingChanges = false;\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Evitar bucles infinitos\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    if (changes['props'] && this.props?.options) {\n      // Verificar si las opciones han cambiado realmente\n      const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);\n\n      if (optionsChanged) {\n        try {\n          this.isProcessingChanges = true;\n          this.previousOptions = [...this.props.options];\n          this.initializeItems();\n          this.syncControlValueWithSelectedItems();\n          this.updateDisplayValue();\n        } finally {\n          this.isProcessingChanges = false;\n        }\n      }\n    }\n  }\n\n  ionViewWillEnter(): void {\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n      this.initializeItems();\n      this.syncControlValueWithSelectedItems();\n      this.updateDisplayValue();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Compara si dos arrays de opciones son iguales\n  private areOptionsEqual(prevOptions: any[], newOptions: any[]): boolean {\n    if (!prevOptions || !newOptions) {\n      return prevOptions === newOptions;\n    }\n\n    if (prevOptions.length !== newOptions.length) {\n      return false;\n    }\n\n    // Comparación simple basada en longitud y valores de ID\n    // Esto se puede mejorar según tus necesidades específicas\n    for (let i = 0; i < prevOptions.length; i++) {\n      if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private initializeItems(): void {\n    if (this.props?.options) {\n      this.filteredItems = [...this.props.options];\n    } else {\n      this.filteredItems = [];\n    }\n  }\n\n  private syncControlValueWithSelectedItems(): void {\n    if (!this.props?.control) {\n      return;\n    }\n\n    // Sincronizar el valor del control con los elementos seleccionados\n    if (this.props.control.value) {\n      const controlValue = this.props.control.value;\n      const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);\n\n      if (selectedOption) {\n        this.selectedItems = [selectedOption];\n      } else {\n        this.selectedItems = [];\n      }\n    } else {\n      this.selectedItems = [];\n    }\n  }\n\n  onFilter(event) {\n    // Resetear a todas las opciones disponibles si no hay texto de búsqueda\n    if (!event || event.trim() === '') {\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    // Si no hay opciones disponibles, no hace falta filtrar\n    if (!this.props?.options || this.props.options.length === 0) {\n      this.filteredItems = [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    const text = replaceSpecialChars(event.toLowerCase());\n\n    const filteredResults = this.props.options.filter(element => {\n      const values = Object.values(element).map((a: any) => {\n        if (a === null || a === undefined) return '';\n        return replaceSpecialChars(`${a}`).toLowerCase();\n      });\n\n      return values.some(value => value.includes(text));\n    });\n\n    this.filteredItems = filteredResults;\n    this.changeDetector.detectChanges();\n  }\n\n  onFocus() {\n    console.log('onFocus');\n  }\n\n  onBlur() {\n    console.log('onBlur');\n  }\n\n  openModal() {\n    if (this.modal) {\n      this.modal.present();\n    }\n  }\n\n  preventDefaultBehavior(event: MouseEvent) {\n    event.preventDefault(); // Evita el comportamiento predeterminado del selector\n    event.stopPropagation(); // Detiene la propagación del evento\n    this.openModal();\n  }\n\n  cancelModal() {\n    if (this.modal) {\n      this.modal.dismiss();\n    }\n  }\n\n  selectItem(item: any) {\n    if (this.multiple) {\n      const index = this.selectedItems.findIndex(\n        selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]\n      );\n\n      if (index === -1) {\n        this.selectedItems.push(item);\n      } else {\n        this.selectedItems.splice(index, 1);\n      }\n    } else {\n      this.selectedItems = [item];\n      this.cancelModal();\n    }\n\n    this.updateDisplayValue();\n    this.applyChanges();\n  }\n\n  isItemSelected(item: any): boolean {\n    return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);\n  }\n\n  updateDisplayValue() {\n    if (this.selectedItems.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.multiple) {\n      if (this.selectedItems.length === 1) {\n        this.displayValue = this.selectedItems[0][this.labelProperty];\n      } else {\n        this.displayValue = `${this.selectedItems.length} elementos seleccionados`;\n      }\n    } else {\n      this.displayValue = this.selectedItems[0][this.labelProperty];\n    }\n  }\n\n  applyChanges() {\n    if (!this.props?.control) {\n      return;\n    }\n\n    if (this.selectedItems.length > 0) {\n      this.props.control.setValue(this.selectedItems[0][this.valueProperty]);\n    } else {\n      this.props.control.setValue(null);\n    }\n\n    this.props.control.updateValueAndValidity();\n    this.props.control.markAsDirty();\n  }\n}\n"]}
354
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"select-search.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/select-search/select-search.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ViewChild,\n  ChangeDetectorRef,\n} from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { ViewWillEnter } from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\nimport { Subscription } from 'rxjs';\n\n@Component({\n  selector: 'val-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, SearchbarComponent, ReactiveFormsModule],\n  template: `\n    <ion-input\n      type=\"text\"\n      [value]=\"displayValue\"\n      [placeholder]=\"props?.placeholder || placeholder\"\n      readonly\n      (mousedown)=\"preventDefaultBehavior($event)\"\n    />\n\n    <ion-input style=\"position: absolute;\" [formControl]=\"props.control\" type=\"hidden\"></ion-input>\n\n    <ion-modal #modal trigger=\"open-modal\" [initialBreakpoint]=\"0.75\" [breakpoints]=\"[0, 0.5, 0.75, 1]\">\n      <ng-template>\n        <ion-header>\n          <ion-toolbar>\n            <ion-title>{{ label }}</ion-title>\n            <ion-buttons slot=\"end\">\n              <ion-button (click)=\"cancelModal()\">Cancelar</ion-button>\n            </ion-buttons>\n          </ion-toolbar>\n          <ion-toolbar>\n            <val-searchbar (filterEvent)=\"onFilter($event)\" (focusEvent)=\"onFocus()\" (blurEvent)=\"onBlur()\" />\n          </ion-toolbar>\n        </ion-header>\n        <ion-content>\n          <ion-list>\n            <ion-item *ngFor=\"let item of filteredItems\" button (click)=\"selectItem(item)\" detail=\"false\">\n              <ion-label>{{ item[labelProperty] }}</ion-label>\n              <ion-icon *ngIf=\"isItemSelected(item)\" name=\"checkmark-outline\" slot=\"end\" color=\"primary\"></ion-icon>\n            </ion-item>\n            <ion-item *ngIf=\"filteredItems.length === 0\">\n              <ion-label color=\"medium\">No se encontraron resultados</ion-label>\n            </ion-item>\n          </ion-list>\n        </ion-content>\n      </ng-template>\n    </ion-modal>\n  `,\n  styles: [``],\n})\nexport class SelectSearchComponent implements ViewWillEnter, OnChanges, OnDestroy {\n  @ViewChild('modal') modal!: IonModal;\n\n  @Input() label: string = 'Seleccionar';\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() multiple: boolean = false;\n  @Input() placeholder: string = 'Seleccione una opción';\n  @Input() props: InputMetadata;\n\n  icon = inject(IconService);\n  private changeDetector = inject(ChangeDetectorRef);\n\n  searchTerm: string = '';\n  filteredItems: InputOption[] = [];\n  selectedItems: InputOption[] = [];\n  displayValue: string = '';\n\n  private previousOptions: any[] = [];\n  private isProcessingChanges = false;\n  private valueChangesSubscription?: Subscription;\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n    this.subscribeToValueChanges();\n  }\n\n  ngOnDestroy() {\n    // Limpiar suscripciones para evitar memory leaks\n    if (this.valueChangesSubscription) {\n      this.valueChangesSubscription.unsubscribe();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Evitar bucles infinitos\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    // Cuando cambia props o props.options\n    if (changes['props']) {\n      try {\n        this.isProcessingChanges = true;\n\n        // Desuscribirse del antiguo control si existe\n        if (this.valueChangesSubscription) {\n          this.valueChangesSubscription.unsubscribe();\n        }\n\n        if (this.props?.options) {\n          // Verificar si las opciones han cambiado realmente\n          const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);\n\n          if (optionsChanged) {\n            this.previousOptions = [...this.props.options];\n            this.initializeItems();\n          }\n        }\n\n        // Sincronizar con el nuevo control si existe\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n\n        // Suscribirse al nuevo control\n        this.subscribeToValueChanges();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    }\n  }\n\n  ionViewWillEnter(): void {\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n      this.initializeItems();\n      this.syncControlValueWithSelectedItems();\n      this.updateDisplayValue();\n      this.subscribeToValueChanges();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Suscribirse a cambios en el FormControl\n  private subscribeToValueChanges() {\n    if (!this.props?.control) return;\n\n    this.valueChangesSubscription = this.props.control.valueChanges.subscribe(value => {\n      if (this.isProcessingChanges) return;\n\n      try {\n        this.isProcessingChanges = true;\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    });\n  }\n\n  // Compara si dos arrays de opciones son iguales\n  private areOptionsEqual(prevOptions: any[], newOptions: any[]): boolean {\n    if (!prevOptions || !newOptions) {\n      return prevOptions === newOptions;\n    }\n\n    if (prevOptions.length !== newOptions.length) {\n      return false;\n    }\n\n    for (let i = 0; i < prevOptions.length; i++) {\n      if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private initializeItems(): void {\n    if (this.props?.options) {\n      this.filteredItems = [...this.props.options];\n    } else {\n      this.filteredItems = [];\n    }\n  }\n\n  private syncControlValueWithSelectedItems(): void {\n    if (!this.props?.control) {\n      this.selectedItems = [];\n      return;\n    }\n\n    const controlValue = this.props.control.value;\n\n    // Si el control no tiene valor, limpiar seleccionados\n    if (controlValue === null || controlValue === undefined) {\n      this.selectedItems = [];\n      return;\n    }\n\n    // Buscar la opción que corresponde al valor del control\n    if (this.props.options && this.props.options.length > 0) {\n      const selectedOption = this.props.options.find(opt => opt[this.valueProperty] === controlValue);\n\n      this.selectedItems = selectedOption ? [selectedOption] : [];\n    } else {\n      this.selectedItems = [];\n    }\n  }\n\n  onFilter(event) {\n    // Resetear a todas las opciones disponibles si no hay texto de búsqueda\n    if (!event || event.trim() === '') {\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    // Si no hay opciones disponibles, no hace falta filtrar\n    if (!this.props?.options || this.props.options.length === 0) {\n      this.filteredItems = [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    const text = replaceSpecialChars(event.toLowerCase());\n\n    const filteredResults = this.props.options.filter(element => {\n      const values = Object.values(element).map((a: any) => {\n        if (a === null || a === undefined) return '';\n        return replaceSpecialChars(`${a}`).toLowerCase();\n      });\n\n      return values.some(value => value.includes(text));\n    });\n\n    this.filteredItems = filteredResults;\n    this.changeDetector.detectChanges();\n  }\n\n  onFocus() {\n    console.log('onFocus');\n  }\n\n  onBlur() {\n    console.log('onBlur');\n  }\n\n  openModal() {\n    if (this.modal) {\n      this.modal.present();\n    }\n  }\n\n  preventDefaultBehavior(event: MouseEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.openModal();\n  }\n\n  cancelModal() {\n    if (this.modal) {\n      this.modal.dismiss();\n    }\n  }\n\n  selectItem(item: any) {\n    if (this.multiple) {\n      const index = this.selectedItems.findIndex(\n        selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]\n      );\n\n      if (index === -1) {\n        this.selectedItems.push(item);\n      } else {\n        this.selectedItems.splice(index, 1);\n      }\n    } else {\n      this.selectedItems = [item];\n      this.cancelModal();\n    }\n\n    this.updateDisplayValue();\n    this.applyChanges();\n  }\n\n  isItemSelected(item: any): boolean {\n    return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);\n  }\n\n  updateDisplayValue() {\n    if (this.selectedItems.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.multiple) {\n      if (this.selectedItems.length === 1) {\n        this.displayValue = this.selectedItems[0][this.labelProperty];\n      } else {\n        this.displayValue = `${this.selectedItems.length} elementos seleccionados`;\n      }\n    } else {\n      this.displayValue = this.selectedItems[0][this.labelProperty];\n    }\n  }\n\n  applyChanges() {\n    if (!this.props?.control) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n\n      if (this.selectedItems.length > 0) {\n        this.props.control.setValue(this.selectedItems[0][this.valueProperty]);\n      } else {\n        this.props.control.setValue(null);\n      }\n\n      this.props.control.markAsDirty();\n      this.props.control.updateValueAndValidity();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Método público para reiniciar el componente\n  reset() {\n    this.selectedItems = [];\n    this.displayValue = '';\n    if (this.props?.control) {\n      this.props.control.setValue(null);\n    }\n    this.changeDetector.detectChanges();\n  }\n}\n"]}
@@ -82,6 +82,14 @@ export class ItemListComponent {
82
82
  @if (item.comments) {
83
83
  <ion-note color="medium"> {{ item.comments }} </ion-note>
84
84
  }
85
+
86
+ @if (item.subtitle) {
87
+ <p>{{ item.subtitle }}</p>
88
+ }
89
+
90
+ @if (item.endNote) {
91
+ <ion-note slot="end">{{ item.endNote }}</ion-note>
92
+ }
85
93
  </ion-label>
86
94
 
87
95
  @if (item.actions) {
@@ -256,6 +264,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
256
264
  @if (item.comments) {
257
265
  <ion-note color="medium"> {{ item.comments }} </ion-note>
258
266
  }
267
+
268
+ @if (item.subtitle) {
269
+ <p>{{ item.subtitle }}</p>
270
+ }
271
+
272
+ @if (item.endNote) {
273
+ <ion-note slot="end">{{ item.endNote }}</ion-note>
274
+ }
259
275
  </ion-label>
260
276
 
261
277
  @if (item.actions) {
@@ -363,4 +379,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
363
379
  }], onActionClick: [{
364
380
  type: Output
365
381
  }] } });
366
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"item-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/item-list/item-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EACL,SAAS,EACT,SAAS,EACT,OAAO,EACP,OAAO,EACP,QAAQ,EACR,OAAO,EACP,aAAa,EACb,OAAO,EACP,UAAU,GACX,MAAM,2BAA2B,CAAC;AAEnC,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;;;AAoL9C,MAAM,OAAO,iBAAiB;IAa5B,YAAY,IAAiB;QAZ7B,WAAM,GAAG,eAAe,CAAC;QACzB,iBAAY,GAAG,SAAS,CAAC;QAMzB,YAAO,GAAG,IAAI,YAAY,EAAU,CAAC;QAGrC,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;IAEX,CAAC;IAEjC,QAAQ,KAAI,CAAC;IAEb,YAAY,CAAC,KAAc;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED,oBAAoB,CAAC,KAAc;QACjC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,KAAK;QACH,OAAO,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC;IAC7D,CAAC;+GA3BU,iBAAiB;mGAAjB,iBAAiB,sKAhKlB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6JT,kjBA1KC,UAAU,oOAEV,OAAO,oFACP,OAAO,yFACP,aAAa,gGACb,QAAQ,6FACR,OAAO,gFACP,OAAO,0NACP,SAAS,uDACT,OAAO,2JACP,SAAS,oPACT,UAAU;;4FAkKD,iBAAiB;kBAjL7B,SAAS;+BACE,eAAe,cACb,IAAI,WACP;wBACP,UAAU;wBACV,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,aAAa;wBACb,QAAQ;wBACR,OAAO;wBACP,OAAO;wBACP,SAAS;wBACT,OAAO;wBACP,SAAS;wBACT,UAAU;qBACX,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA6JT;gFAQD,KAAK;sBADJ,KAAK;gBAIN,OAAO;sBADN,MAAM;gBAIP,aAAa;sBADZ,MAAM","sourcesContent":["import { NgClass, NgStyle } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport {\n  IonAvatar,\n  IonButton,\n  IonIcon,\n  IonItem,\n  IonLabel,\n  IonList,\n  IonListHeader,\n  IonNote,\n  IonSpinner,\n} from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { resolveColor } from '../../../shared/utils/styles';\nimport { ComponentStates } from '../../types';\nimport { ListMetadata } from './types';\n\n@Component({\n  selector: 'val-item-list',\n  standalone: true,\n  imports: [\n    RouterLink,\n    NgStyle,\n    NgClass,\n    IonList,\n    IonListHeader,\n    IonLabel,\n    IonNote,\n    IonItem,\n    IonAvatar,\n    IonIcon,\n    IonButton,\n    IonSpinner,\n  ],\n  template: `\n    <ion-list\n      [inset]=\"props.rounded\"\n      [class.shadowed]=\"props.shadowed\"\n      [class.bordered]=\"props.bordered\"\n      [style.borderColor]=\"this.color()\"\n    >\n      @if (props.title) {\n        <ion-list-header>\n          <ion-label>{{ props.title }}</ion-label>\n        </ion-list-header>\n      }\n      @for (item of props.items; track item.token) {\n        @if (item.mode == 'justext') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n\n              @if (item.subtitle) {\n                <p>{{ item.subtitle }}</p>\n              }\n            </ion-label>\n\n            @if (item.endNote) {\n              <ion-note slot=\"end\">{{ item.endNote }}</ion-note>\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'actionable') {\n          <ion-item [lines]=\"item.lines\" [button]=\"item.clickable\" [disabled]=\"item.disabled\">\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n\n            @if (item.actions) {\n              @for (action of item.actions; track action.token) {\n                <ion-button\n                  fill=\"outline\"\n                  color=\"dark\"\n                  slot=\"end\"\n                  [disabled]=\"action.state === states.WORKING\"\n                  (click)=\"onActionClickHandler(action.token)\"\n                >\n                  @if (action.state !== states.WORKING) {\n                    @if (action.icon) {\n                      <ion-icon aria-hidden=\"true\" [name]=\"action.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n                    }\n\n                    {{ action.description }}\n                  }\n\n                  @if (action.state === states.WORKING) {\n                    <ion-spinner class=\"action-spinner\" name=\"circular\"></ion-spinner>\n                  }\n                </ion-button>\n              }\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'route') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n            [routerLink]=\"item.routerLink\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'subtitule') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h1>{{ item.text }}</h1>\n              <p>{{ item.subtitle }}</p>\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'avatar') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-avatar aria-hidden=\"true\" slot=\"start\">\n              <img alt=\"\" [src]=\"item.image\" />\n            </ion-avatar>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'icon') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-icon aria-hidden=\"true\" [name]=\"item.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n      }\n    </ion-list>\n  `,\n  styleUrls: ['./item-list.component.scss'],\n})\nexport class ItemListComponent implements OnInit {\n  states = ComponentStates;\n  defaultColor = 'primary';\n\n  @Input()\n  props: ListMetadata;\n\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  @Output()\n  onActionClick = new EventEmitter<string>();\n\n  constructor(icon: IconService) {}\n\n  ngOnInit() {}\n\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  onActionClickHandler(token?: string) {\n    this.onActionClick.emit(token);\n  }\n\n  color() {\n    return resolveColor(this.props.color || this.defaultColor);\n  }\n}\n"]}
382
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"item-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/item-list/item-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EACL,SAAS,EACT,SAAS,EACT,OAAO,EACP,OAAO,EACP,QAAQ,EACR,OAAO,EACP,aAAa,EACb,OAAO,EACP,UAAU,GACX,MAAM,2BAA2B,CAAC;AAEnC,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;;;AA4L9C,MAAM,OAAO,iBAAiB;IAa5B,YAAY,IAAiB;QAZ7B,WAAM,GAAG,eAAe,CAAC;QACzB,iBAAY,GAAG,SAAS,CAAC;QAMzB,YAAO,GAAG,IAAI,YAAY,EAAU,CAAC;QAGrC,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;IAEX,CAAC;IAEjC,QAAQ,KAAI,CAAC;IAEb,YAAY,CAAC,KAAc;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED,oBAAoB,CAAC,KAAc;QACjC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED,KAAK;QACH,OAAO,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC;IAC7D,CAAC;+GA3BU,iBAAiB;mGAAjB,iBAAiB,sKAxKlB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT,kjBAlLC,UAAU,oOAEV,OAAO,oFACP,OAAO,yFACP,aAAa,gGACb,QAAQ,6FACR,OAAO,gFACP,OAAO,0NACP,SAAS,uDACT,OAAO,2JACP,SAAS,oPACT,UAAU;;4FA0KD,iBAAiB;kBAzL7B,SAAS;+BACE,eAAe,cACb,IAAI,WACP;wBACP,UAAU;wBACV,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,aAAa;wBACb,QAAQ;wBACR,OAAO;wBACP,OAAO;wBACP,SAAS;wBACT,OAAO;wBACP,SAAS;wBACT,UAAU;qBACX,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT;gFAQD,KAAK;sBADJ,KAAK;gBAIN,OAAO;sBADN,MAAM;gBAIP,aAAa;sBADZ,MAAM","sourcesContent":["import { NgClass, NgStyle } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport {\n  IonAvatar,\n  IonButton,\n  IonIcon,\n  IonItem,\n  IonLabel,\n  IonList,\n  IonListHeader,\n  IonNote,\n  IonSpinner,\n} from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { resolveColor } from '../../../shared/utils/styles';\nimport { ComponentStates } from '../../types';\nimport { ListMetadata } from './types';\n\n@Component({\n  selector: 'val-item-list',\n  standalone: true,\n  imports: [\n    RouterLink,\n    NgStyle,\n    NgClass,\n    IonList,\n    IonListHeader,\n    IonLabel,\n    IonNote,\n    IonItem,\n    IonAvatar,\n    IonIcon,\n    IonButton,\n    IonSpinner,\n  ],\n  template: `\n    <ion-list\n      [inset]=\"props.rounded\"\n      [class.shadowed]=\"props.shadowed\"\n      [class.bordered]=\"props.bordered\"\n      [style.borderColor]=\"this.color()\"\n    >\n      @if (props.title) {\n        <ion-list-header>\n          <ion-label>{{ props.title }}</ion-label>\n        </ion-list-header>\n      }\n      @for (item of props.items; track item.token) {\n        @if (item.mode == 'justext') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n\n              @if (item.subtitle) {\n                <p>{{ item.subtitle }}</p>\n              }\n            </ion-label>\n\n            @if (item.endNote) {\n              <ion-note slot=\"end\">{{ item.endNote }}</ion-note>\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'actionable') {\n          <ion-item [lines]=\"item.lines\" [button]=\"item.clickable\" [disabled]=\"item.disabled\">\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n\n              @if (item.subtitle) {\n                <p>{{ item.subtitle }}</p>\n              }\n\n              @if (item.endNote) {\n                <ion-note slot=\"end\">{{ item.endNote }}</ion-note>\n              }\n            </ion-label>\n\n            @if (item.actions) {\n              @for (action of item.actions; track action.token) {\n                <ion-button\n                  fill=\"outline\"\n                  color=\"dark\"\n                  slot=\"end\"\n                  [disabled]=\"action.state === states.WORKING\"\n                  (click)=\"onActionClickHandler(action.token)\"\n                >\n                  @if (action.state !== states.WORKING) {\n                    @if (action.icon) {\n                      <ion-icon aria-hidden=\"true\" [name]=\"action.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n                    }\n\n                    {{ action.description }}\n                  }\n\n                  @if (action.state === states.WORKING) {\n                    <ion-spinner class=\"action-spinner\" name=\"circular\"></ion-spinner>\n                  }\n                </ion-button>\n              }\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'route') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n            [routerLink]=\"item.routerLink\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'subtitule') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h1>{{ item.text }}</h1>\n              <p>{{ item.subtitle }}</p>\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'avatar') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-avatar aria-hidden=\"true\" slot=\"start\">\n              <img alt=\"\" [src]=\"item.image\" />\n            </ion-avatar>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'icon') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-icon aria-hidden=\"true\" [name]=\"item.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n      }\n    </ion-list>\n  `,\n  styleUrls: ['./item-list.component.scss'],\n})\nexport class ItemListComponent implements OnInit {\n  states = ComponentStates;\n  defaultColor = 'primary';\n\n  @Input()\n  props: ListMetadata;\n\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  @Output()\n  onActionClick = new EventEmitter<string>();\n\n  constructor(icon: IconService) {}\n\n  ngOnInit() {}\n\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  onActionClickHandler(token?: string) {\n    this.onActionClick.emit(token);\n  }\n\n  color() {\n    return resolveColor(this.props.color || this.defaultColor);\n  }\n}\n"]}
@@ -2137,7 +2137,6 @@ class SelectSearchComponent {
2137
2137
  this.filteredItems = [];
2138
2138
  this.selectedItems = [];
2139
2139
  this.displayValue = '';
2140
- // Usar para evitar bucles infinitos
2141
2140
  this.previousOptions = [];
2142
2141
  this.isProcessingChanges = false;
2143
2142
  }
@@ -2145,26 +2144,43 @@ class SelectSearchComponent {
2145
2144
  this.initializeItems();
2146
2145
  this.syncControlValueWithSelectedItems();
2147
2146
  this.updateDisplayValue();
2147
+ this.subscribeToValueChanges();
2148
+ }
2149
+ ngOnDestroy() {
2150
+ // Limpiar suscripciones para evitar memory leaks
2151
+ if (this.valueChangesSubscription) {
2152
+ this.valueChangesSubscription.unsubscribe();
2153
+ }
2148
2154
  }
2149
2155
  ngOnChanges(changes) {
2150
2156
  // Evitar bucles infinitos
2151
2157
  if (this.isProcessingChanges) {
2152
2158
  return;
2153
2159
  }
2154
- if (changes['props'] && this.props?.options) {
2155
- // Verificar si las opciones han cambiado realmente
2156
- const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
2157
- if (optionsChanged) {
2158
- try {
2159
- this.isProcessingChanges = true;
2160
- this.previousOptions = [...this.props.options];
2161
- this.initializeItems();
2162
- this.syncControlValueWithSelectedItems();
2163
- this.updateDisplayValue();
2160
+ // Cuando cambia props o props.options
2161
+ if (changes['props']) {
2162
+ try {
2163
+ this.isProcessingChanges = true;
2164
+ // Desuscribirse del antiguo control si existe
2165
+ if (this.valueChangesSubscription) {
2166
+ this.valueChangesSubscription.unsubscribe();
2164
2167
  }
2165
- finally {
2166
- this.isProcessingChanges = false;
2168
+ if (this.props?.options) {
2169
+ // Verificar si las opciones han cambiado realmente
2170
+ const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
2171
+ if (optionsChanged) {
2172
+ this.previousOptions = [...this.props.options];
2173
+ this.initializeItems();
2174
+ }
2167
2175
  }
2176
+ // Sincronizar con el nuevo control si existe
2177
+ this.syncControlValueWithSelectedItems();
2178
+ this.updateDisplayValue();
2179
+ // Suscribirse al nuevo control
2180
+ this.subscribeToValueChanges();
2181
+ }
2182
+ finally {
2183
+ this.isProcessingChanges = false;
2168
2184
  }
2169
2185
  }
2170
2186
  }
@@ -2177,11 +2193,29 @@ class SelectSearchComponent {
2177
2193
  this.initializeItems();
2178
2194
  this.syncControlValueWithSelectedItems();
2179
2195
  this.updateDisplayValue();
2196
+ this.subscribeToValueChanges();
2180
2197
  }
2181
2198
  finally {
2182
2199
  this.isProcessingChanges = false;
2183
2200
  }
2184
2201
  }
2202
+ // Suscribirse a cambios en el FormControl
2203
+ subscribeToValueChanges() {
2204
+ if (!this.props?.control)
2205
+ return;
2206
+ this.valueChangesSubscription = this.props.control.valueChanges.subscribe(value => {
2207
+ if (this.isProcessingChanges)
2208
+ return;
2209
+ try {
2210
+ this.isProcessingChanges = true;
2211
+ this.syncControlValueWithSelectedItems();
2212
+ this.updateDisplayValue();
2213
+ }
2214
+ finally {
2215
+ this.isProcessingChanges = false;
2216
+ }
2217
+ });
2218
+ }
2185
2219
  // Compara si dos arrays de opciones son iguales
2186
2220
  areOptionsEqual(prevOptions, newOptions) {
2187
2221
  if (!prevOptions || !newOptions) {
@@ -2190,8 +2224,6 @@ class SelectSearchComponent {
2190
2224
  if (prevOptions.length !== newOptions.length) {
2191
2225
  return false;
2192
2226
  }
2193
- // Comparación simple basada en longitud y valores de ID
2194
- // Esto se puede mejorar según tus necesidades específicas
2195
2227
  for (let i = 0; i < prevOptions.length; i++) {
2196
2228
  if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
2197
2229
  return false;
@@ -2209,18 +2241,19 @@ class SelectSearchComponent {
2209
2241
  }
2210
2242
  syncControlValueWithSelectedItems() {
2211
2243
  if (!this.props?.control) {
2244
+ this.selectedItems = [];
2212
2245
  return;
2213
2246
  }
2214
- // Sincronizar el valor del control con los elementos seleccionados
2215
- if (this.props.control.value) {
2216
- const controlValue = this.props.control.value;
2217
- const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);
2218
- if (selectedOption) {
2219
- this.selectedItems = [selectedOption];
2220
- }
2221
- else {
2222
- this.selectedItems = [];
2223
- }
2247
+ const controlValue = this.props.control.value;
2248
+ // Si el control no tiene valor, limpiar seleccionados
2249
+ if (controlValue === null || controlValue === undefined) {
2250
+ this.selectedItems = [];
2251
+ return;
2252
+ }
2253
+ // Buscar la opción que corresponde al valor del control
2254
+ if (this.props.options && this.props.options.length > 0) {
2255
+ const selectedOption = this.props.options.find(opt => opt[this.valueProperty] === controlValue);
2256
+ this.selectedItems = selectedOption ? [selectedOption] : [];
2224
2257
  }
2225
2258
  else {
2226
2259
  this.selectedItems = [];
@@ -2263,8 +2296,8 @@ class SelectSearchComponent {
2263
2296
  }
2264
2297
  }
2265
2298
  preventDefaultBehavior(event) {
2266
- event.preventDefault(); // Evita el comportamiento predeterminado del selector
2267
- event.stopPropagation(); // Detiene la propagación del evento
2299
+ event.preventDefault();
2300
+ event.stopPropagation();
2268
2301
  this.openModal();
2269
2302
  }
2270
2303
  cancelModal() {
@@ -2313,14 +2346,29 @@ class SelectSearchComponent {
2313
2346
  if (!this.props?.control) {
2314
2347
  return;
2315
2348
  }
2316
- if (this.selectedItems.length > 0) {
2317
- this.props.control.setValue(this.selectedItems[0][this.valueProperty]);
2349
+ try {
2350
+ this.isProcessingChanges = true;
2351
+ if (this.selectedItems.length > 0) {
2352
+ this.props.control.setValue(this.selectedItems[0][this.valueProperty]);
2353
+ }
2354
+ else {
2355
+ this.props.control.setValue(null);
2356
+ }
2357
+ this.props.control.markAsDirty();
2358
+ this.props.control.updateValueAndValidity();
2318
2359
  }
2319
- else {
2360
+ finally {
2361
+ this.isProcessingChanges = false;
2362
+ }
2363
+ }
2364
+ // Método público para reiniciar el componente
2365
+ reset() {
2366
+ this.selectedItems = [];
2367
+ this.displayValue = '';
2368
+ if (this.props?.control) {
2320
2369
  this.props.control.setValue(null);
2321
2370
  }
2322
- this.props.control.updateValueAndValidity();
2323
- this.props.control.markAsDirty();
2371
+ this.changeDetector.detectChanges();
2324
2372
  }
2325
2373
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: SelectSearchComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
2326
2374
  static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: SelectSearchComponent, isStandalone: true, selector: "val-select-search", inputs: { label: "label", labelProperty: "labelProperty", valueProperty: "valueProperty", multiple: "multiple", placeholder: "placeholder", props: "props" }, viewQueries: [{ propertyName: "modal", first: true, predicate: ["modal"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
@@ -3237,6 +3285,14 @@ class ItemListComponent {
3237
3285
  @if (item.comments) {
3238
3286
  <ion-note color="medium"> {{ item.comments }} </ion-note>
3239
3287
  }
3288
+
3289
+ @if (item.subtitle) {
3290
+ <p>{{ item.subtitle }}</p>
3291
+ }
3292
+
3293
+ @if (item.endNote) {
3294
+ <ion-note slot="end">{{ item.endNote }}</ion-note>
3295
+ }
3240
3296
  </ion-label>
3241
3297
 
3242
3298
  @if (item.actions) {
@@ -3411,6 +3467,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
3411
3467
  @if (item.comments) {
3412
3468
  <ion-note color="medium"> {{ item.comments }} </ion-note>
3413
3469
  }
3470
+
3471
+ @if (item.subtitle) {
3472
+ <p>{{ item.subtitle }}</p>
3473
+ }
3474
+
3475
+ @if (item.endNote) {
3476
+ <ion-note slot="end">{{ item.endNote }}</ion-note>
3477
+ }
3414
3478
  </ion-label>
3415
3479
 
3416
3480
  @if (item.actions) {