valtech-components 2.0.400 → 2.0.402

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.
@@ -32,15 +32,28 @@ export class MultiSelectSearchComponent {
32
32
  document.addEventListener('click', this.handleClickOutside.bind(this));
33
33
  }
34
34
  ngOnInit() {
35
+ console.log('[MultiSelectSearch] ngOnInit:', {
36
+ props: !!this.props,
37
+ options: this.props?.options?.length || 0,
38
+ propsOptions: this.props?.options
39
+ });
35
40
  this.applyDefaultValue();
36
41
  this.initializeOptions();
37
42
  this.syncSelectedValues();
38
43
  this.updateDisplayValue();
44
+ // Force change detection after initialization
45
+ this.changeDetector.detectChanges();
39
46
  }
40
47
  ngOnDestroy() {
41
48
  document.removeEventListener('click', this.handleClickOutside.bind(this));
42
49
  }
43
50
  ngOnChanges(changes) {
51
+ console.log('[MultiSelectSearch] ngOnChanges:', {
52
+ hasPropsChange: !!changes['props'],
53
+ props: !!this.props,
54
+ options: this.props?.options?.length || 0,
55
+ propsOptions: this.props?.options
56
+ });
44
57
  if (changes['props'] && this.props) {
45
58
  this.initializeOptions();
46
59
  this.syncSelectedValues();
@@ -114,6 +127,17 @@ export class MultiSelectSearchComponent {
114
127
  trackByFn(_index, option) {
115
128
  return option[this.valueProperty];
116
129
  }
130
+ debugOptions() {
131
+ console.log('[MultiSelectSearch] DEBUG CLICK:', {
132
+ props: this.props,
133
+ filteredOptions: this.filteredOptions,
134
+ searchTerm: this.searchTerm,
135
+ propsOptions: this.props?.options,
136
+ displayValue: this.displayValue,
137
+ selectedValues: this.selectedValues,
138
+ isOpen: this.isOpen
139
+ });
140
+ }
117
141
  handleClickOutside(event) {
118
142
  if (this.isOpen &&
119
143
  !this.mainInputRef?.nativeElement?.contains(event.target) &&
@@ -154,7 +178,14 @@ export class MultiSelectSearchComponent {
154
178
  }
155
179
  }
156
180
  initializeOptions() {
181
+ console.log('[MultiSelectSearch] initializeOptions:', {
182
+ propsOptions: this.props?.options,
183
+ optionsLength: this.props?.options?.length || 0
184
+ });
157
185
  this.filteredOptions = this.props?.options || [];
186
+ console.log('[MultiSelectSearch] filteredOptions set to:', this.filteredOptions);
187
+ // Force change detection
188
+ this.changeDetector.detectChanges();
158
189
  }
159
190
  filterOptions() {
160
191
  if (!this.searchTerm) {
@@ -236,8 +267,13 @@ export class MultiSelectSearchComponent {
236
267
  ></ion-searchbar>
237
268
  </div>
238
269
 
270
+ <!-- Debug button -->
271
+ <div class="search-container" style="background: red; color: white; padding: 4px;">
272
+ <button (click)="debugOptions()">DEBUG: {{ filteredOptions.length }} opciones</button>
273
+ </div>
274
+
239
275
  <!-- Action buttons -->
240
- <div class="actions-container" *ngIf="filteredOptions.length > 0">
276
+ <div class="actions-container">
241
277
  <ion-button fill="clear" size="small" (click)="selectAll()" [disabled]="filteredOptions.length === 0">
242
278
  Seleccionar todos
243
279
  </ion-button>
@@ -322,8 +358,13 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
322
358
  ></ion-searchbar>
323
359
  </div>
324
360
 
361
+ <!-- Debug button -->
362
+ <div class="search-container" style="background: red; color: white; padding: 4px;">
363
+ <button (click)="debugOptions()">DEBUG: {{ filteredOptions.length }} opciones</button>
364
+ </div>
365
+
325
366
  <!-- Action buttons -->
326
- <div class="actions-container" *ngIf="filteredOptions.length > 0">
367
+ <div class="actions-container">
327
368
  <ion-button fill="clear" size="small" (click)="selectAll()" [disabled]="filteredOptions.length === 0">
328
369
  Seleccionar todos
329
370
  </ion-button>
@@ -376,4 +417,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
376
417
  }], placeholder: [{
377
418
  type: Input
378
419
  }] } });
379
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select-search.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/multi-select-search/multi-select-search.component.ts"],"names":[],"mappings":"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","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-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectSearchComponent),\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 name=\"chevron-down-outline\" class=\"dropdown-icon\" [class.rotated]=\"isOpen\"></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 class=\"dropdown-overlay\" [class.visible]=\"isOpen\" #dropdown>\n      <!-- Search bar -->\n      <div class=\"search-container\" *ngIf=\"props?.options && props.options.length > 5\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"searchPlaceholder\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm\"\n          show-clear-button=\"focus\"\n          [debounce]=\"200\"\n        ></ion-searchbar>\n      </div>\n\n      <!-- Action buttons -->\n      <div class=\"actions-container\" *ngIf=\"filteredOptions.length > 0\">\n        <ion-button fill=\"clear\" size=\"small\" (click)=\"selectAll()\" [disabled]=\"filteredOptions.length === 0\">\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 slot=\"start\" [checked]=\"isSelected(option)\"></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-search.component.scss'],\n})\nexport class MultiSelectSearchComponent 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  // Component state\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 (\n      this.isOpen &&\n      !this.mainInputRef?.nativeElement?.contains(event.target) &&\n      !this.dropdownRef?.nativeElement?.contains(event.target)\n    ) {\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\n        .split(',')\n        .map(v => v.trim())\n        .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 => String(option[this.valueProperty]) === String(value));\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}\n"]}
420
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select-search.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/multi-select-search/multi-select-search.component.ts"],"names":[],"mappings":"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","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-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectSearchComponent),\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 name=\"chevron-down-outline\" class=\"dropdown-icon\" [class.rotated]=\"isOpen\"></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 class=\"dropdown-overlay\" [class.visible]=\"isOpen\" #dropdown>\n      <!-- Search bar -->\n      <div class=\"search-container\" *ngIf=\"props?.options && props.options.length > 5\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"searchPlaceholder\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm\"\n          show-clear-button=\"focus\"\n          [debounce]=\"200\"\n        ></ion-searchbar>\n      </div>\n\n      <!-- Debug button -->\n      <div class=\"search-container\" style=\"background: red; color: white; padding: 4px;\">\n        <button (click)=\"debugOptions()\">DEBUG: {{ filteredOptions.length }} opciones</button>\n      </div>\n      \n      <!-- Action buttons -->\n      <div class=\"actions-container\">\n        <ion-button fill=\"clear\" size=\"small\" (click)=\"selectAll()\" [disabled]=\"filteredOptions.length === 0\">\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 slot=\"start\" [checked]=\"isSelected(option)\"></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-search.component.scss'],\n})\nexport class MultiSelectSearchComponent 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  // Component state\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    console.log('[MultiSelectSearch] ngOnInit:', {\n      props: !!this.props,\n      options: this.props?.options?.length || 0,\n      propsOptions: this.props?.options\n    });\n    this.applyDefaultValue();\n    this.initializeOptions();\n    this.syncSelectedValues();\n    this.updateDisplayValue();\n    \n    // Force change detection after initialization\n    this.changeDetector.detectChanges();\n  }\n\n  ngOnDestroy() {\n    document.removeEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    console.log('[MultiSelectSearch] ngOnChanges:', {\n      hasPropsChange: !!changes['props'],\n      props: !!this.props,\n      options: this.props?.options?.length || 0,\n      propsOptions: this.props?.options\n    });\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  debugOptions(): void {\n    console.log('[MultiSelectSearch] DEBUG CLICK:', {\n      props: this.props,\n      filteredOptions: this.filteredOptions,\n      searchTerm: this.searchTerm,\n      propsOptions: this.props?.options,\n      displayValue: this.displayValue,\n      selectedValues: this.selectedValues,\n      isOpen: this.isOpen\n    });\n  }\n\n  private handleClickOutside(event: Event): void {\n    if (\n      this.isOpen &&\n      !this.mainInputRef?.nativeElement?.contains(event.target) &&\n      !this.dropdownRef?.nativeElement?.contains(event.target)\n    ) {\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\n        .split(',')\n        .map(v => v.trim())\n        .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 => String(option[this.valueProperty]) === String(value));\n  }\n\n  private applyDefaultValue(): void {\n    if (this.props) {\n      applyDefaultValueToControl(this.props);\n    }\n  }\n\n  private initializeOptions(): void {\n    console.log('[MultiSelectSearch] initializeOptions:', {\n      propsOptions: this.props?.options,\n      optionsLength: this.props?.options?.length || 0\n    });\n    this.filteredOptions = this.props?.options || [];\n    console.log('[MultiSelectSearch] filteredOptions set to:', this.filteredOptions);\n    \n    // Force change detection\n    this.changeDetector.detectChanges();\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}\n"]}
@@ -7536,15 +7536,28 @@ class MultiSelectSearchComponent {
7536
7536
  document.addEventListener('click', this.handleClickOutside.bind(this));
7537
7537
  }
7538
7538
  ngOnInit() {
7539
+ console.log('[MultiSelectSearch] ngOnInit:', {
7540
+ props: !!this.props,
7541
+ options: this.props?.options?.length || 0,
7542
+ propsOptions: this.props?.options
7543
+ });
7539
7544
  this.applyDefaultValue();
7540
7545
  this.initializeOptions();
7541
7546
  this.syncSelectedValues();
7542
7547
  this.updateDisplayValue();
7548
+ // Force change detection after initialization
7549
+ this.changeDetector.detectChanges();
7543
7550
  }
7544
7551
  ngOnDestroy() {
7545
7552
  document.removeEventListener('click', this.handleClickOutside.bind(this));
7546
7553
  }
7547
7554
  ngOnChanges(changes) {
7555
+ console.log('[MultiSelectSearch] ngOnChanges:', {
7556
+ hasPropsChange: !!changes['props'],
7557
+ props: !!this.props,
7558
+ options: this.props?.options?.length || 0,
7559
+ propsOptions: this.props?.options
7560
+ });
7548
7561
  if (changes['props'] && this.props) {
7549
7562
  this.initializeOptions();
7550
7563
  this.syncSelectedValues();
@@ -7618,6 +7631,17 @@ class MultiSelectSearchComponent {
7618
7631
  trackByFn(_index, option) {
7619
7632
  return option[this.valueProperty];
7620
7633
  }
7634
+ debugOptions() {
7635
+ console.log('[MultiSelectSearch] DEBUG CLICK:', {
7636
+ props: this.props,
7637
+ filteredOptions: this.filteredOptions,
7638
+ searchTerm: this.searchTerm,
7639
+ propsOptions: this.props?.options,
7640
+ displayValue: this.displayValue,
7641
+ selectedValues: this.selectedValues,
7642
+ isOpen: this.isOpen
7643
+ });
7644
+ }
7621
7645
  handleClickOutside(event) {
7622
7646
  if (this.isOpen &&
7623
7647
  !this.mainInputRef?.nativeElement?.contains(event.target) &&
@@ -7658,7 +7682,14 @@ class MultiSelectSearchComponent {
7658
7682
  }
7659
7683
  }
7660
7684
  initializeOptions() {
7685
+ console.log('[MultiSelectSearch] initializeOptions:', {
7686
+ propsOptions: this.props?.options,
7687
+ optionsLength: this.props?.options?.length || 0
7688
+ });
7661
7689
  this.filteredOptions = this.props?.options || [];
7690
+ console.log('[MultiSelectSearch] filteredOptions set to:', this.filteredOptions);
7691
+ // Force change detection
7692
+ this.changeDetector.detectChanges();
7662
7693
  }
7663
7694
  filterOptions() {
7664
7695
  if (!this.searchTerm) {
@@ -7740,8 +7771,13 @@ class MultiSelectSearchComponent {
7740
7771
  ></ion-searchbar>
7741
7772
  </div>
7742
7773
 
7774
+ <!-- Debug button -->
7775
+ <div class="search-container" style="background: red; color: white; padding: 4px;">
7776
+ <button (click)="debugOptions()">DEBUG: {{ filteredOptions.length }} opciones</button>
7777
+ </div>
7778
+
7743
7779
  <!-- Action buttons -->
7744
- <div class="actions-container" *ngIf="filteredOptions.length > 0">
7780
+ <div class="actions-container">
7745
7781
  <ion-button fill="clear" size="small" (click)="selectAll()" [disabled]="filteredOptions.length === 0">
7746
7782
  Seleccionar todos
7747
7783
  </ion-button>
@@ -7826,8 +7862,13 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
7826
7862
  ></ion-searchbar>
7827
7863
  </div>
7828
7864
 
7865
+ <!-- Debug button -->
7866
+ <div class="search-container" style="background: red; color: white; padding: 4px;">
7867
+ <button (click)="debugOptions()">DEBUG: {{ filteredOptions.length }} opciones</button>
7868
+ </div>
7869
+
7829
7870
  <!-- Action buttons -->
7830
- <div class="actions-container" *ngIf="filteredOptions.length > 0">
7871
+ <div class="actions-container">
7831
7872
  <ion-button fill="clear" size="small" (click)="selectAll()" [disabled]="filteredOptions.length === 0">
7832
7873
  Seleccionar todos
7833
7874
  </ion-button>