valtech-components 2.0.392 → 2.0.394

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, forwardRef, inject, Input, ViewChild, signal, computed } from '@angular/core';
2
+ import { ChangeDetectorRef, Component, forwardRef, inject, Input, ViewChild } from '@angular/core';
3
3
  import { NG_VALUE_ACCESSOR, FormsModule, ReactiveFormsModule } from '@angular/forms';
4
4
  import { IonicModule } from '@ionic/angular';
5
5
  import { LangService } from '../../../services/lang-provider/lang-provider.service';
@@ -15,45 +15,13 @@ export class MultiSelectSimpleComponent {
15
15
  this.valueProperty = 'id';
16
16
  this.placeholder = '';
17
17
  this.langService = inject(LangService);
18
- // Signals for reactive state management
19
- this.isOpen = signal(false);
20
- this.searchTerm = signal('');
21
- this.selectedValues = signal([]);
22
- // Computed signals
23
- this.displayValue = computed(() => {
24
- const selected = this.selectedValues();
25
- if (selected.length === 0) {
26
- return '';
27
- }
28
- if (selected.length === 1) {
29
- const option = this.getOptionByValue(selected[0]);
30
- return option ? option[this.labelProperty] : '';
31
- }
32
- return `${selected.length} elementos seleccionados`;
33
- });
34
- this.filteredOptions = computed(() => {
35
- const options = this.props?.options || [];
36
- const search = this.searchTerm().toLowerCase();
37
- // Debug
38
- console.log('[MultiSelectSimple] filteredOptions computed:', {
39
- optionsCount: options.length,
40
- searchTerm: search,
41
- options: options.slice(0, 3) // Primeras 3 opciones
42
- });
43
- if (!search) {
44
- return options;
45
- }
46
- return options.filter(option => {
47
- const label = option[this.labelProperty]
48
- ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())
49
- : '';
50
- const value = option[this.valueProperty]
51
- ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())
52
- : '';
53
- const searchTerm = replaceSpecialChars(search);
54
- return label.includes(searchTerm) || value.includes(searchTerm);
55
- });
56
- });
18
+ this.changeDetector = inject(ChangeDetectorRef);
19
+ // Classic Angular properties
20
+ this.isOpen = false;
21
+ this.searchTerm = '';
22
+ this.selectedValues = [];
23
+ this.displayValue = '';
24
+ this.filteredOptions = [];
57
25
  this.searchPlaceholder = '';
58
26
  // ControlValueAccessor
59
27
  this.onChange = (_value) => { };
@@ -65,20 +33,26 @@ export class MultiSelectSimpleComponent {
65
33
  }
66
34
  ngOnInit() {
67
35
  this.applyDefaultValue();
36
+ this.initializeOptions();
68
37
  this.syncSelectedValues();
38
+ this.updateDisplayValue();
69
39
  }
70
40
  ngOnDestroy() {
71
41
  document.removeEventListener('click', this.handleClickOutside.bind(this));
72
42
  }
73
43
  ngOnChanges(changes) {
74
44
  if (changes['props'] && this.props) {
45
+ this.initializeOptions();
75
46
  this.syncSelectedValues();
47
+ this.updateDisplayValue();
48
+ this.changeDetector.detectChanges();
76
49
  }
77
50
  }
78
51
  // ControlValueAccessor implementation
79
52
  writeValue(value) {
80
53
  const valueArray = this.parseValue(value);
81
- this.selectedValues.set(valueArray);
54
+ this.selectedValues = valueArray;
55
+ this.updateDisplayValue();
82
56
  }
83
57
  registerOnChange(fn) {
84
58
  this.onChange = fn;
@@ -92,8 +66,8 @@ export class MultiSelectSimpleComponent {
92
66
  // Component methods
93
67
  toggleDropdown(event) {
94
68
  event.stopPropagation();
95
- this.isOpen.update(value => !value);
96
- if (this.isOpen()) {
69
+ this.isOpen = !this.isOpen;
70
+ if (this.isOpen) {
97
71
  this.onTouched();
98
72
  // Focus search bar when opening
99
73
  setTimeout(() => {
@@ -105,45 +79,48 @@ export class MultiSelectSimpleComponent {
105
79
  }
106
80
  }
107
81
  onSearch(event) {
108
- this.searchTerm.set(event.detail.value || '');
82
+ this.searchTerm = event.detail.value || '';
83
+ this.filterOptions();
109
84
  }
110
85
  toggleOption(option) {
111
- const value = String(option[this.valueProperty]); // Convertir a string
112
- const currentSelected = this.selectedValues();
86
+ const value = String(option[this.valueProperty]);
113
87
  if (this.isSelected(option)) {
114
88
  // Remove from selection
115
- this.selectedValues.set(currentSelected.filter(v => v !== value));
89
+ this.selectedValues = this.selectedValues.filter(v => v !== value);
116
90
  }
117
91
  else {
118
92
  // Add to selection
119
- this.selectedValues.set([...currentSelected, value]);
93
+ this.selectedValues = [...this.selectedValues, value];
120
94
  }
95
+ this.updateDisplayValue();
121
96
  this.emitValue();
122
97
  }
123
98
  selectAll() {
124
- const allValues = this.filteredOptions().map(option => String(option[this.valueProperty]));
125
- const currentSelected = this.selectedValues();
99
+ const allValues = this.filteredOptions.map(option => String(option[this.valueProperty]));
126
100
  // Add only new values to avoid duplicates
127
- const newValues = allValues.filter(value => !currentSelected.includes(value));
128
- this.selectedValues.set([...currentSelected, ...newValues]);
101
+ const newValues = allValues.filter(value => !this.selectedValues.includes(value));
102
+ this.selectedValues = [...this.selectedValues, ...newValues];
103
+ this.updateDisplayValue();
129
104
  this.emitValue();
130
105
  }
131
106
  clearAll() {
132
- this.selectedValues.set([]);
107
+ this.selectedValues = [];
108
+ this.updateDisplayValue();
133
109
  this.emitValue();
134
110
  }
135
111
  isSelected(option) {
136
- return this.selectedValues().includes(String(option[this.valueProperty]));
112
+ return this.selectedValues.includes(String(option[this.valueProperty]));
137
113
  }
138
114
  trackByFn(_index, option) {
139
115
  return option[this.valueProperty];
140
116
  }
141
117
  handleClickOutside(event) {
142
- if (this.isOpen() &&
118
+ if (this.isOpen &&
143
119
  !this.mainInputRef?.nativeElement?.contains(event.target) &&
144
120
  !this.dropdownRef?.nativeElement?.contains(event.target)) {
145
- this.isOpen.set(false);
146
- this.searchTerm.set('');
121
+ this.isOpen = false;
122
+ this.searchTerm = '';
123
+ this.initializeOptions();
147
124
  }
148
125
  }
149
126
  parseValue(value) {
@@ -157,7 +134,7 @@ export class MultiSelectSimpleComponent {
157
134
  return [String(value)];
158
135
  }
159
136
  emitValue() {
160
- const value = this.selectedValues().join(',');
137
+ const value = this.selectedValues.join(',');
161
138
  this.onChange(value);
162
139
  if (this.props?.control) {
163
140
  this.props.control.setValue(value);
@@ -166,17 +143,49 @@ export class MultiSelectSimpleComponent {
166
143
  }
167
144
  }
168
145
  getOptionByValue(value) {
169
- return this.props?.options?.find(option => String(option[this.valueProperty]) === String(value));
146
+ return this.filteredOptions.find(option => String(option[this.valueProperty]) === String(value));
170
147
  }
171
148
  applyDefaultValue() {
172
149
  if (this.props) {
173
150
  applyDefaultValueToControl(this.props);
174
151
  }
175
152
  }
153
+ initializeOptions() {
154
+ this.filteredOptions = this.props?.options || [];
155
+ }
156
+ filterOptions() {
157
+ if (!this.searchTerm) {
158
+ this.initializeOptions();
159
+ return;
160
+ }
161
+ const search = replaceSpecialChars(this.searchTerm.toLowerCase());
162
+ this.filteredOptions = (this.props?.options || []).filter(option => {
163
+ const label = option[this.labelProperty]
164
+ ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())
165
+ : '';
166
+ const value = option[this.valueProperty]
167
+ ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())
168
+ : '';
169
+ return label.includes(search) || value.includes(search);
170
+ });
171
+ }
172
+ updateDisplayValue() {
173
+ if (this.selectedValues.length === 0) {
174
+ this.displayValue = '';
175
+ return;
176
+ }
177
+ if (this.selectedValues.length === 1) {
178
+ const option = this.getOptionByValue(this.selectedValues[0]);
179
+ this.displayValue = option ? option[this.labelProperty] : '';
180
+ }
181
+ else {
182
+ this.displayValue = `${this.selectedValues.length} elementos seleccionados`;
183
+ }
184
+ }
176
185
  syncSelectedValues() {
177
186
  if (this.props?.control?.value) {
178
187
  const valueArray = this.parseValue(this.props.control.value);
179
- this.selectedValues.set(valueArray);
188
+ this.selectedValues = valueArray;
180
189
  }
181
190
  }
182
191
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiSelectSimpleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
@@ -192,18 +201,18 @@ export class MultiSelectSimpleComponent {
192
201
  <ion-input
193
202
  #mainInput
194
203
  type="text"
195
- [value]="displayValue()"
204
+ [value]="displayValue"
196
205
  [placeholder]="props?.placeholder || placeholder"
197
206
  readonly
198
207
  class="main-input"
199
- [class.is-open]="isOpen()"
208
+ [class.is-open]="isOpen"
200
209
  />
201
210
 
202
211
  <!-- Dropdown icon -->
203
212
  <ion-icon
204
213
  name="chevron-down-outline"
205
214
  class="dropdown-icon"
206
- [class.rotated]="isOpen()"
215
+ [class.rotated]="isOpen"
207
216
  ></ion-icon>
208
217
 
209
218
  <!-- Hidden input for form control -->
@@ -217,7 +226,7 @@ export class MultiSelectSimpleComponent {
217
226
  <!-- Dropdown overlay -->
218
227
  <div
219
228
  class="dropdown-overlay"
220
- [class.visible]="isOpen()"
229
+ [class.visible]="isOpen"
221
230
  #dropdown
222
231
  >
223
232
  <!-- Search bar -->
@@ -226,7 +235,7 @@ export class MultiSelectSimpleComponent {
226
235
  #searchbar
227
236
  [placeholder]="searchPlaceholder"
228
237
  (ionInput)="onSearch($event)"
229
- [value]="searchTerm()"
238
+ [value]="searchTerm"
230
239
  show-clear-button="focus"
231
240
  ></ion-searchbar>
232
241
  </div>
@@ -237,7 +246,7 @@ export class MultiSelectSimpleComponent {
237
246
  fill="clear"
238
247
  size="small"
239
248
  (click)="selectAll()"
240
- [disabled]="filteredOptions().length === 0"
249
+ [disabled]="filteredOptions.length === 0"
241
250
  >
242
251
  Seleccionar todos
243
252
  </ion-button>
@@ -246,7 +255,7 @@ export class MultiSelectSimpleComponent {
246
255
  size="small"
247
256
  color="medium"
248
257
  (click)="clearAll()"
249
- [disabled]="selectedValues().length === 0"
258
+ [disabled]="selectedValues.length === 0"
250
259
  >
251
260
  Limpiar
252
261
  </ion-button>
@@ -256,7 +265,7 @@ export class MultiSelectSimpleComponent {
256
265
  <div class="options-container">
257
266
  <ion-list class="options-list">
258
267
  <ion-item
259
- *ngFor="let option of filteredOptions(); trackBy: trackByFn"
268
+ *ngFor="let option of filteredOptions; trackBy: trackByFn"
260
269
  button
261
270
  (click)="toggleOption(option)"
262
271
  class="option-item"
@@ -269,9 +278,9 @@ export class MultiSelectSimpleComponent {
269
278
  </ion-item>
270
279
 
271
280
  <!-- No results message -->
272
- <ion-item *ngIf="filteredOptions().length === 0" class="no-results">
281
+ <ion-item *ngIf="filteredOptions.length === 0" class="no-results">
273
282
  <ion-label color="medium">
274
- {{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
283
+ {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
275
284
  </ion-label>
276
285
  </ion-item>
277
286
  </ion-list>
@@ -293,18 +302,18 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
293
302
  <ion-input
294
303
  #mainInput
295
304
  type="text"
296
- [value]="displayValue()"
305
+ [value]="displayValue"
297
306
  [placeholder]="props?.placeholder || placeholder"
298
307
  readonly
299
308
  class="main-input"
300
- [class.is-open]="isOpen()"
309
+ [class.is-open]="isOpen"
301
310
  />
302
311
 
303
312
  <!-- Dropdown icon -->
304
313
  <ion-icon
305
314
  name="chevron-down-outline"
306
315
  class="dropdown-icon"
307
- [class.rotated]="isOpen()"
316
+ [class.rotated]="isOpen"
308
317
  ></ion-icon>
309
318
 
310
319
  <!-- Hidden input for form control -->
@@ -318,7 +327,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
318
327
  <!-- Dropdown overlay -->
319
328
  <div
320
329
  class="dropdown-overlay"
321
- [class.visible]="isOpen()"
330
+ [class.visible]="isOpen"
322
331
  #dropdown
323
332
  >
324
333
  <!-- Search bar -->
@@ -327,7 +336,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
327
336
  #searchbar
328
337
  [placeholder]="searchPlaceholder"
329
338
  (ionInput)="onSearch($event)"
330
- [value]="searchTerm()"
339
+ [value]="searchTerm"
331
340
  show-clear-button="focus"
332
341
  ></ion-searchbar>
333
342
  </div>
@@ -338,7 +347,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
338
347
  fill="clear"
339
348
  size="small"
340
349
  (click)="selectAll()"
341
- [disabled]="filteredOptions().length === 0"
350
+ [disabled]="filteredOptions.length === 0"
342
351
  >
343
352
  Seleccionar todos
344
353
  </ion-button>
@@ -347,7 +356,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
347
356
  size="small"
348
357
  color="medium"
349
358
  (click)="clearAll()"
350
- [disabled]="selectedValues().length === 0"
359
+ [disabled]="selectedValues.length === 0"
351
360
  >
352
361
  Limpiar
353
362
  </ion-button>
@@ -357,7 +366,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
357
366
  <div class="options-container">
358
367
  <ion-list class="options-list">
359
368
  <ion-item
360
- *ngFor="let option of filteredOptions(); trackBy: trackByFn"
369
+ *ngFor="let option of filteredOptions; trackBy: trackByFn"
361
370
  button
362
371
  (click)="toggleOption(option)"
363
372
  class="option-item"
@@ -370,9 +379,9 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
370
379
  </ion-item>
371
380
 
372
381
  <!-- No results message -->
373
- <ion-item *ngIf="filteredOptions().length === 0" class="no-results">
382
+ <ion-item *ngIf="filteredOptions.length === 0" class="no-results">
374
383
  <ion-label color="medium">
375
- {{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
384
+ {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
376
385
  </ion-label>
377
386
  </ion-item>
378
387
  </ion-list>
@@ -394,4 +403,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
394
403
  }], placeholder: [{
395
404
  type: Input
396
405
  }] } });
397
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select-simple.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/multi-select-simple/multi-select-simple.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  ElementRef,\n  forwardRef,\n  inject,\n  Input,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  signal,\n  computed\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule } from '@ionic/angular';\nimport { LangService } from '../../../services/lang-provider/lang-provider.service';\nimport { applyDefaultValueToControl } from '../../../shared/utils/form-defaults';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\n\n@Component({\n  selector: 'val-multi-select-simple',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectSimpleComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"multi-select-container\" (click)=\"toggleDropdown($event)\">\n      <!-- Main input display -->\n      <ion-input\n        #mainInput\n        type=\"text\"\n        [value]=\"displayValue()\"\n        [placeholder]=\"props?.placeholder || placeholder\"\n        readonly\n        class=\"main-input\"\n        [class.is-open]=\"isOpen()\"\n      />\n      \n      <!-- Dropdown icon -->\n      <ion-icon \n        name=\"chevron-down-outline\" \n        class=\"dropdown-icon\" \n        [class.rotated]=\"isOpen()\"\n      ></ion-icon>\n      \n      <!-- Hidden input for form control -->\n      <ion-input \n        style=\"position: absolute; opacity: 0; pointer-events: none;\" \n        [formControl]=\"props?.control\" \n        type=\"hidden\"\n      />\n    </div>\n\n    <!-- Dropdown overlay -->\n    <div \n      class=\"dropdown-overlay\" \n      [class.visible]=\"isOpen()\"\n      #dropdown\n    >\n      <!-- Search bar -->\n      <div class=\"search-container\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"searchPlaceholder\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm()\"\n          show-clear-button=\"focus\"\n        ></ion-searchbar>\n      </div>\n      \n      <!-- Action buttons -->\n      <div class=\"actions-container\">\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          (click)=\"selectAll()\"\n          [disabled]=\"filteredOptions().length === 0\"\n        >\n          Seleccionar todos\n        </ion-button>\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          color=\"medium\" \n          (click)=\"clearAll()\"\n          [disabled]=\"selectedValues().length === 0\"\n        >\n          Limpiar\n        </ion-button>\n      </div>\n      \n      <!-- Options list -->\n      <div class=\"options-container\">\n        <ion-list class=\"options-list\">\n          <ion-item \n            *ngFor=\"let option of filteredOptions(); trackBy: trackByFn\" \n            button \n            (click)=\"toggleOption(option)\"\n            class=\"option-item\"\n          >\n            <ion-checkbox\n              slot=\"start\"\n              [checked]=\"isSelected(option)\"\n            ></ion-checkbox>\n            <ion-label>{{ option[labelProperty] }}</ion-label>\n          </ion-item>\n          \n          <!-- No results message -->\n          <ion-item *ngIf=\"filteredOptions().length === 0\" class=\"no-results\">\n            <ion-label color=\"medium\">\n              {{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}\n            </ion-label>\n          </ion-item>\n        </ion-list>\n      </div>\n    </div>\n  `,\n  styleUrls: ['./multi-select-simple.component.scss']\n})\nexport class MultiSelectSimpleComponent implements OnInit, OnDestroy, OnChanges, ControlValueAccessor {\n  @ViewChild('dropdown') dropdownRef!: ElementRef;\n  @ViewChild('mainInput') mainInputRef!: ElementRef;\n\n  @Input() props!: InputMetadata;\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() placeholder: string = '';\n\n  private langService = inject(LangService);\n  \n  // Signals for reactive state management\n  isOpen = signal(false);\n  searchTerm = signal('');\n  selectedValues = signal<string[]>([]);\n  \n  // Computed signals\n  displayValue = computed(() => {\n    const selected = this.selectedValues();\n    if (selected.length === 0) {\n      return '';\n    }\n    if (selected.length === 1) {\n      const option = this.getOptionByValue(selected[0]);\n      return option ? option[this.labelProperty] : '';\n    }\n    return `${selected.length} elementos seleccionados`;\n  });\n\n  filteredOptions = computed(() => {\n    const options = this.props?.options || [];\n    const search = this.searchTerm().toLowerCase();\n    \n    // Debug\n    console.log('[MultiSelectSimple] filteredOptions computed:', {\n      optionsCount: options.length,\n      searchTerm: search,\n      options: options.slice(0, 3) // Primeras 3 opciones\n    });\n    \n    if (!search) {\n      return options;\n    }\n    \n    return options.filter(option => {\n      const label = option[this.labelProperty]\n        ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())\n        : '';\n      const value = option[this.valueProperty]\n        ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())\n        : '';\n      const searchTerm = replaceSpecialChars(search);\n      return label.includes(searchTerm) || value.includes(searchTerm);\n    });\n  });\n\n  searchPlaceholder: string = '';\n\n  // ControlValueAccessor\n  private onChange = (_value: any) => {};\n  private onTouched = () => {};\n\n  constructor() {\n    this.placeholder = this.langService.getText('_global', 'selectOptions', 'Seleccione opciones');\n    this.searchPlaceholder = this.langService.getText('_global', 'search', 'Buscar');\n    \n    // Close dropdown when clicking outside\n    document.addEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnInit() {\n    this.applyDefaultValue();\n    this.syncSelectedValues();\n  }\n\n  ngOnDestroy() {\n    document.removeEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['props'] && this.props) {\n      this.syncSelectedValues();\n    }\n  }\n\n  // ControlValueAccessor implementation\n  writeValue(value: any): void {\n    const valueArray = this.parseValue(value);\n    this.selectedValues.set(valueArray);\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(_isDisabled: boolean): void {\n    // Handle disabled state if needed\n  }\n\n  // Component methods\n  toggleDropdown(event: Event): void {\n    event.stopPropagation();\n    this.isOpen.update(value => !value);\n    \n    if (this.isOpen()) {\n      this.onTouched();\n      // Focus search bar when opening\n      setTimeout(() => {\n        const searchbar = this.dropdownRef?.nativeElement?.querySelector('ion-searchbar');\n        if (searchbar) {\n          searchbar.setFocus();\n        }\n      }, 100);\n    }\n  }\n\n  onSearch(event: any): void {\n    this.searchTerm.set(event.detail.value || '');\n  }\n\n  toggleOption(option: InputOption): void {\n    const value = String(option[this.valueProperty]); // Convertir a string\n    const currentSelected = this.selectedValues();\n    \n    if (this.isSelected(option)) {\n      // Remove from selection\n      this.selectedValues.set(currentSelected.filter(v => v !== value));\n    } else {\n      // Add to selection\n      this.selectedValues.set([...currentSelected, value]);\n    }\n    \n    this.emitValue();\n  }\n\n  selectAll(): void {\n    const allValues = this.filteredOptions().map(option => String(option[this.valueProperty]));\n    const currentSelected = this.selectedValues();\n    \n    // Add only new values to avoid duplicates\n    const newValues = allValues.filter(value => !currentSelected.includes(value));\n    this.selectedValues.set([...currentSelected, ...newValues]);\n    \n    this.emitValue();\n  }\n\n  clearAll(): void {\n    this.selectedValues.set([]);\n    this.emitValue();\n  }\n\n  isSelected(option: InputOption): boolean {\n    return this.selectedValues().includes(String(option[this.valueProperty]));\n  }\n\n  trackByFn(_index: number, option: InputOption): any {\n    return option[this.valueProperty];\n  }\n\n  private handleClickOutside(event: Event): void {\n    if (this.isOpen() && \n        !this.mainInputRef?.nativeElement?.contains(event.target) &&\n        !this.dropdownRef?.nativeElement?.contains(event.target)) {\n      this.isOpen.set(false);\n      this.searchTerm.set('');\n    }\n  }\n\n  private parseValue(value: any): string[] {\n    if (!value) return [];\n    if (Array.isArray(value)) return value.map(v => String(v));\n    if (typeof value === 'string') {\n      return value.split(',').map(v => v.trim()).filter(v => v !== '');\n    }\n    return [String(value)];\n  }\n\n  private emitValue(): void {\n    const value = this.selectedValues().join(',');\n    this.onChange(value);\n    \n    if (this.props?.control) {\n      this.props.control.setValue(value);\n      this.props.control.markAsDirty();\n      this.props.control.markAsTouched();\n    }\n  }\n\n  private getOptionByValue(value: string): InputOption | undefined {\n    return this.props?.options?.find(option => \n      String(option[this.valueProperty]) === String(value)\n    );\n  }\n\n  private applyDefaultValue(): void {\n    if (this.props) {\n      applyDefaultValueToControl(this.props);\n    }\n  }\n\n  private syncSelectedValues(): void {\n    if (this.props?.control?.value) {\n      const valueArray = this.parseValue(this.props.control.value);\n      this.selectedValues.set(valueArray);\n    }\n  }\n}"]}
406
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select-simple.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/multi-select-simple/multi-select-simple.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,iBAAiB,EACjB,SAAS,EAET,UAAU,EACV,MAAM,EACN,KAAK,EAKL,SAAS,EACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAwB,iBAAiB,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC3G,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAE,MAAM,uDAAuD,CAAC;AACpF,OAAO,EAAE,0BAA0B,EAAE,MAAM,qCAAqC,CAAC;AACjF,OAAO,EAAE,mBAAmB,EAAE,MAAM,4BAA4B,CAAC;;;;;AA4GjE,MAAM,OAAO,0BAA0B;IAwBrC;QAnBS,kBAAa,GAAW,MAAM,CAAC;QAC/B,kBAAa,GAAW,IAAI,CAAC;QAC7B,gBAAW,GAAW,EAAE,CAAC;QAE1B,gBAAW,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAClC,mBAAc,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAEnD,6BAA6B;QAC7B,WAAM,GAAG,KAAK,CAAC;QACf,eAAU,GAAG,EAAE,CAAC;QAChB,mBAAc,GAAa,EAAE,CAAC;QAC9B,iBAAY,GAAG,EAAE,CAAC;QAClB,oBAAe,GAAkB,EAAE,CAAC;QACpC,sBAAiB,GAAW,EAAE,CAAC;QAE/B,uBAAuB;QACf,aAAQ,GAAG,CAAC,MAAW,EAAE,EAAE,GAAE,CAAC,CAAC;QAC/B,cAAS,GAAG,GAAG,EAAE,GAAE,CAAC,CAAC;QAG3B,IAAI,CAAC,WAAW,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,eAAe,EAAE,qBAAqB,CAAC,CAAC;QAC/F,IAAI,CAAC,iBAAiB,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,QAAQ,EAAE,QAAQ,CAAC,CAAC;QAEjF,uCAAuC;QACvC,QAAQ,CAAC,gBAAgB,CAAC,OAAO,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IACzE,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAED,WAAW;QACT,QAAQ,CAAC,mBAAmB,CAAC,OAAO,EAAE,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC;IAC5E,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,IAAI,OAAO,CAAC,OAAO,CAAC,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACnC,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;QACtC,CAAC;IACH,CAAC;IAED,sCAAsC;IACtC,UAAU,CAAC,KAAU;QACnB,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC;QAC1C,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC;QACjC,IAAI,CAAC,kBAAkB,EAAE,CAAC;IAC5B,CAAC;IAED,gBAAgB,CAAC,EAAO;QACtB,IAAI,CAAC,QAAQ,GAAG,EAAE,CAAC;IACrB,CAAC;IAED,iBAAiB,CAAC,EAAO;QACvB,IAAI,CAAC,SAAS,GAAG,EAAE,CAAC;IACtB,CAAC;IAED,gBAAgB,CAAC,WAAoB;QACnC,kCAAkC;IACpC,CAAC;IAED,oBAAoB;IACpB,cAAc,CAAC,KAAY;QACzB,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,MAAM,GAAG,CAAC,IAAI,CAAC,MAAM,CAAC;QAE3B,IAAI,IAAI,CAAC,MAAM,EAAE,CAAC;YAChB,IAAI,CAAC,SAAS,EAAE,CAAC;YACjB,gCAAgC;YAChC,UAAU,CAAC,GAAG,EAAE;gBACd,MAAM,SAAS,GAAG,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE,aAAa,CAAC,eAAe,CAAC,CAAC;gBAClF,IAAI,SAAS,EAAE,CAAC;oBACd,SAAS,CAAC,QAAQ,EAAE,CAAC;gBACvB,CAAC;YACH,CAAC,EAAE,GAAG,CAAC,CAAC;QACV,CAAC;IACH,CAAC;IAED,QAAQ,CAAC,KAAU;QACjB,IAAI,CAAC,UAAU,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,IAAI,EAAE,CAAC;QAC3C,IAAI,CAAC,aAAa,EAAE,CAAC;IACvB,CAAC;IAED,YAAY,CAAC,MAAmB;QAC9B,MAAM,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;QAEjD,IAAI,IAAI,CAAC,UAAU,CAAC,MAAM,CAAC,EAAE,CAAC;YAC5B,wBAAwB;YACxB,IAAI,CAAC,cAAc,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,KAAK,CAAC,CAAC;QACrE,CAAC;aAAM,CAAC;YACN,mBAAmB;YACnB,IAAI,CAAC,cAAc,GAAG,CAAC,GAAG,IAAI,CAAC,cAAc,EAAE,KAAK,CAAC,CAAC;QACxD,CAAC;QAED,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,SAAS;QACP,MAAM,SAAS,GAAG,IAAI,CAAC,eAAe,CAAC,GAAG,CAAC,MAAM,CAAC,EAAE,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;QAEzF,0CAA0C;QAC1C,MAAM,SAAS,GAAG,SAAS,CAAC,MAAM,CAAC,KAAK,CAAC,EAAE,CAAC,CAAC,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC,CAAC;QAClF,IAAI,CAAC,cAAc,GAAG,CAAC,GAAG,IAAI,CAAC,cAAc,EAAE,GAAG,SAAS,CAAC,CAAC;QAE7D,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,cAAc,GAAG,EAAE,CAAC;QACzB,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,UAAU,CAAC,MAAmB;QAC5B,OAAO,IAAI,CAAC,cAAc,CAAC,QAAQ,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;IAC1E,CAAC;IAED,SAAS,CAAC,MAAc,EAAE,MAAmB;QAC3C,OAAO,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IACpC,CAAC;IAEO,kBAAkB,CAAC,KAAY;QACrC,IAAI,IAAI,CAAC,MAAM;YACX,CAAC,IAAI,CAAC,YAAY,EAAE,aAAa,EAAE,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC;YACzD,CAAC,IAAI,CAAC,WAAW,EAAE,aAAa,EAAE,QAAQ,CAAC,KAAK,CAAC,MAAM,CAAC,EAAE,CAAC;YAC7D,IAAI,CAAC,MAAM,GAAG,KAAK,CAAC;YACpB,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;YACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAC3B,CAAC;IACH,CAAC;IAEO,UAAU,CAAC,KAAU;QAC3B,IAAI,CAAC,KAAK;YAAE,OAAO,EAAE,CAAC;QACtB,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;QAC3D,IAAI,OAAO,KAAK,KAAK,QAAQ,EAAE,CAAC;YAC9B,OAAO,KAAK,CAAC,KAAK,CAAC,GAAG,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,KAAK,EAAE,CAAC,CAAC;QACnE,CAAC;QACD,OAAO,CAAC,MAAM,CAAC,KAAK,CAAC,CAAC,CAAC;IACzB,CAAC;IAEO,SAAS;QACf,MAAM,KAAK,GAAG,IAAI,CAAC,cAAc,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC;QAC5C,IAAI,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;QAErB,IAAI,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC;YACxB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,KAAK,CAAC,CAAC;YACnC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;YACjC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,aAAa,EAAE,CAAC;QACrC,CAAC;IACH,CAAC;IAEO,gBAAgB,CAAC,KAAa;QACpC,OAAO,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,MAAM,CAAC,EAAE,CACxC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,KAAK,MAAM,CAAC,KAAK,CAAC,CACrD,CAAC;IACJ,CAAC;IAEO,iBAAiB;QACvB,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,0BAA0B,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;QACzC,CAAC;IACH,CAAC;IAEO,iBAAiB;QACvB,IAAI,CAAC,eAAe,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,EAAE,CAAC;IACnD,CAAC;IAEO,aAAa;QACnB,IAAI,CAAC,IAAI,CAAC,UAAU,EAAE,CAAC;YACrB,IAAI,CAAC,iBAAiB,EAAE,CAAC;YACzB,OAAO;QACT,CAAC;QAED,MAAM,MAAM,GAAG,mBAAmB,CAAC,IAAI,CAAC,UAAU,CAAC,WAAW,EAAE,CAAC,CAAC;QAClE,IAAI,CAAC,eAAe,GAAG,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,EAAE,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,EAAE;YACjE,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC;gBACtC,CAAC,CAAC,mBAAmB,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;gBACvE,CAAC,CAAC,EAAE,CAAC;YACP,MAAM,KAAK,GAAG,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC;gBACtC,CAAC,CAAC,mBAAmB,CAAC,MAAM,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;gBACvE,CAAC,CAAC,EAAE,CAAC;YACP,OAAO,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC1D,CAAC,CAAC,CAAC;IACL,CAAC;IAEO,kBAAkB;QACxB,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,IAAI,CAAC,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACrC,MAAM,MAAM,GAAG,IAAI,CAAC,gBAAgB,CAAC,IAAI,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,CAAC;YAC7D,IAAI,CAAC,YAAY,GAAG,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC/D,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,YAAY,GAAG,GAAG,IAAI,CAAC,cAAc,CAAC,MAAM,0BAA0B,CAAC;QAC9E,CAAC;IACH,CAAC;IAEO,kBAAkB;QACxB,IAAI,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;YAC/B,MAAM,UAAU,GAAG,IAAI,CAAC,UAAU,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC;YAC7D,IAAI,CAAC,cAAc,GAAG,UAAU,CAAC;QACnC,CAAC;IACH,CAAC;+GAxNU,0BAA0B;mGAA1B,0BAA0B,8LArG1B;YACT;gBACE,OAAO,EAAE,iBAAiB;gBAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,CAAC,0BAA0B,CAAC;gBACzD,KAAK,EAAE,IAAI;aACZ;SACF,oPACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2FT,+gFAnGS,YAAY,+PAAE,WAAW,ioEAAE,WAAW,sIAAE,mBAAmB;;4FAsG1D,0BAA0B;kBAzGtC,SAAS;+BACE,yBAAyB,cACvB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,mBAAmB,CAAC,aAC3D;wBACT;4BACE,OAAO,EAAE,iBAAiB;4BAC1B,WAAW,EAAE,UAAU,CAAC,GAAG,EAAE,2BAA2B,CAAC;4BACzD,KAAK,EAAE,IAAI;yBACZ;qBACF,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA2FT;wDAIsB,WAAW;sBAAjC,SAAS;uBAAC,UAAU;gBACG,YAAY;sBAAnC,SAAS;uBAAC,WAAW;gBAEb,KAAK;sBAAb,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,WAAW;sBAAnB,KAAK","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectorRef,\n  Component,\n  ElementRef,\n  forwardRef,\n  inject,\n  Input,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  ViewChild\n} from '@angular/core';\nimport { ControlValueAccessor, NG_VALUE_ACCESSOR, FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule } from '@ionic/angular';\nimport { LangService } from '../../../services/lang-provider/lang-provider.service';\nimport { applyDefaultValueToControl } from '../../../shared/utils/form-defaults';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\n\n@Component({\n  selector: 'val-multi-select-simple',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectSimpleComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"multi-select-container\" (click)=\"toggleDropdown($event)\">\n      <!-- Main input display -->\n      <ion-input\n        #mainInput\n        type=\"text\"\n        [value]=\"displayValue\"\n        [placeholder]=\"props?.placeholder || placeholder\"\n        readonly\n        class=\"main-input\"\n        [class.is-open]=\"isOpen\"\n      />\n      \n      <!-- Dropdown icon -->\n      <ion-icon \n        name=\"chevron-down-outline\" \n        class=\"dropdown-icon\" \n        [class.rotated]=\"isOpen\"\n      ></ion-icon>\n      \n      <!-- Hidden input for form control -->\n      <ion-input \n        style=\"position: absolute; opacity: 0; pointer-events: none;\" \n        [formControl]=\"props?.control\" \n        type=\"hidden\"\n      />\n    </div>\n\n    <!-- Dropdown overlay -->\n    <div \n      class=\"dropdown-overlay\" \n      [class.visible]=\"isOpen\"\n      #dropdown\n    >\n      <!-- Search bar -->\n      <div class=\"search-container\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"searchPlaceholder\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm\"\n          show-clear-button=\"focus\"\n        ></ion-searchbar>\n      </div>\n      \n      <!-- Action buttons -->\n      <div class=\"actions-container\">\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          (click)=\"selectAll()\"\n          [disabled]=\"filteredOptions.length === 0\"\n        >\n          Seleccionar todos\n        </ion-button>\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          color=\"medium\" \n          (click)=\"clearAll()\"\n          [disabled]=\"selectedValues.length === 0\"\n        >\n          Limpiar\n        </ion-button>\n      </div>\n      \n      <!-- Options list -->\n      <div class=\"options-container\">\n        <ion-list class=\"options-list\">\n          <ion-item \n            *ngFor=\"let option of filteredOptions; trackBy: trackByFn\" \n            button \n            (click)=\"toggleOption(option)\"\n            class=\"option-item\"\n          >\n            <ion-checkbox\n              slot=\"start\"\n              [checked]=\"isSelected(option)\"\n            ></ion-checkbox>\n            <ion-label>{{ option[labelProperty] }}</ion-label>\n          </ion-item>\n          \n          <!-- No results message -->\n          <ion-item *ngIf=\"filteredOptions.length === 0\" class=\"no-results\">\n            <ion-label color=\"medium\">\n              {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}\n            </ion-label>\n          </ion-item>\n        </ion-list>\n      </div>\n    </div>\n  `,\n  styleUrls: ['./multi-select-simple.component.scss']\n})\nexport class MultiSelectSimpleComponent implements OnInit, OnDestroy, OnChanges, ControlValueAccessor {\n  @ViewChild('dropdown') dropdownRef!: ElementRef;\n  @ViewChild('mainInput') mainInputRef!: ElementRef;\n\n  @Input() props!: InputMetadata;\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() placeholder: string = '';\n\n  private langService = inject(LangService);\n  private changeDetector = inject(ChangeDetectorRef);\n  \n  // Classic Angular properties\n  isOpen = false;\n  searchTerm = '';\n  selectedValues: string[] = [];\n  displayValue = '';\n  filteredOptions: InputOption[] = [];\n  searchPlaceholder: string = '';\n\n  // ControlValueAccessor\n  private onChange = (_value: any) => {};\n  private onTouched = () => {};\n\n  constructor() {\n    this.placeholder = this.langService.getText('_global', 'selectOptions', 'Seleccione opciones');\n    this.searchPlaceholder = this.langService.getText('_global', 'search', 'Buscar');\n    \n    // Close dropdown when clicking outside\n    document.addEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnInit() {\n    this.applyDefaultValue();\n    this.initializeOptions();\n    this.syncSelectedValues();\n    this.updateDisplayValue();\n  }\n\n  ngOnDestroy() {\n    document.removeEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['props'] && this.props) {\n      this.initializeOptions();\n      this.syncSelectedValues();\n      this.updateDisplayValue();\n      this.changeDetector.detectChanges();\n    }\n  }\n\n  // ControlValueAccessor implementation\n  writeValue(value: any): void {\n    const valueArray = this.parseValue(value);\n    this.selectedValues = valueArray;\n    this.updateDisplayValue();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(_isDisabled: boolean): void {\n    // Handle disabled state if needed\n  }\n\n  // Component methods\n  toggleDropdown(event: Event): void {\n    event.stopPropagation();\n    this.isOpen = !this.isOpen;\n    \n    if (this.isOpen) {\n      this.onTouched();\n      // Focus search bar when opening\n      setTimeout(() => {\n        const searchbar = this.dropdownRef?.nativeElement?.querySelector('ion-searchbar');\n        if (searchbar) {\n          searchbar.setFocus();\n        }\n      }, 100);\n    }\n  }\n\n  onSearch(event: any): void {\n    this.searchTerm = event.detail.value || '';\n    this.filterOptions();\n  }\n\n  toggleOption(option: InputOption): void {\n    const value = String(option[this.valueProperty]);\n    \n    if (this.isSelected(option)) {\n      // Remove from selection\n      this.selectedValues = this.selectedValues.filter(v => v !== value);\n    } else {\n      // Add to selection\n      this.selectedValues = [...this.selectedValues, value];\n    }\n    \n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  selectAll(): void {\n    const allValues = this.filteredOptions.map(option => String(option[this.valueProperty]));\n    \n    // Add only new values to avoid duplicates\n    const newValues = allValues.filter(value => !this.selectedValues.includes(value));\n    this.selectedValues = [...this.selectedValues, ...newValues];\n    \n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  clearAll(): void {\n    this.selectedValues = [];\n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  isSelected(option: InputOption): boolean {\n    return this.selectedValues.includes(String(option[this.valueProperty]));\n  }\n\n  trackByFn(_index: number, option: InputOption): any {\n    return option[this.valueProperty];\n  }\n\n  private handleClickOutside(event: Event): void {\n    if (this.isOpen && \n        !this.mainInputRef?.nativeElement?.contains(event.target) &&\n        !this.dropdownRef?.nativeElement?.contains(event.target)) {\n      this.isOpen = false;\n      this.searchTerm = '';\n      this.initializeOptions();\n    }\n  }\n\n  private parseValue(value: any): string[] {\n    if (!value) return [];\n    if (Array.isArray(value)) return value.map(v => String(v));\n    if (typeof value === 'string') {\n      return value.split(',').map(v => v.trim()).filter(v => v !== '');\n    }\n    return [String(value)];\n  }\n\n  private emitValue(): void {\n    const value = this.selectedValues.join(',');\n    this.onChange(value);\n    \n    if (this.props?.control) {\n      this.props.control.setValue(value);\n      this.props.control.markAsDirty();\n      this.props.control.markAsTouched();\n    }\n  }\n\n  private getOptionByValue(value: string): InputOption | undefined {\n    return this.filteredOptions.find(option => \n      String(option[this.valueProperty]) === String(value)\n    );\n  }\n\n  private applyDefaultValue(): void {\n    if (this.props) {\n      applyDefaultValueToControl(this.props);\n    }\n  }\n\n  private initializeOptions(): void {\n    this.filteredOptions = this.props?.options || [];\n  }\n  \n  private filterOptions(): void {\n    if (!this.searchTerm) {\n      this.initializeOptions();\n      return;\n    }\n    \n    const search = replaceSpecialChars(this.searchTerm.toLowerCase());\n    this.filteredOptions = (this.props?.options || []).filter(option => {\n      const label = option[this.labelProperty]\n        ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())\n        : '';\n      const value = option[this.valueProperty]\n        ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())\n        : '';\n      return label.includes(search) || value.includes(search);\n    });\n  }\n  \n  private updateDisplayValue(): void {\n    if (this.selectedValues.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.selectedValues.length === 1) {\n      const option = this.getOptionByValue(this.selectedValues[0]);\n      this.displayValue = option ? option[this.labelProperty] : '';\n    } else {\n      this.displayValue = `${this.selectedValues.length} elementos seleccionados`;\n    }\n  }\n\n  private syncSelectedValues(): void {\n    if (this.props?.control?.value) {\n      const valueArray = this.parseValue(this.props.control.value);\n      this.selectedValues = valueArray;\n    }\n  }\n}"]}