valtech-components 2.0.388 → 2.0.390
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.
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +66 -7
- package/fesm2022/valtech-components.mjs +65 -6
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/select-search/select-search.component.d.ts +2 -0
- package/lib/components/organisms/article/article.component.d.ts +1 -1
- package/package.json +1 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component, inject, Input, ViewChild, signal, computed } from '@angular/core';
|
|
2
|
+
import { Component, inject, Input, ViewChild, signal, computed, effect } from '@angular/core';
|
|
3
3
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { IonicModule } from '@ionic/angular';
|
|
5
5
|
import { LangService } from '../../../services/lang-provider/lang-provider.service';
|
|
@@ -19,6 +19,8 @@ export class SelectSearchComponent {
|
|
|
19
19
|
this.isOpen = signal(false);
|
|
20
20
|
this.searchTerm = signal('');
|
|
21
21
|
this.selectedValue = signal(null);
|
|
22
|
+
// Force options to be reactive
|
|
23
|
+
this.optionsSignal = signal([]);
|
|
22
24
|
// Computed signals
|
|
23
25
|
this.displayValue = computed(() => {
|
|
24
26
|
const value = this.selectedValue();
|
|
@@ -28,20 +30,25 @@ export class SelectSearchComponent {
|
|
|
28
30
|
return option ? option[this.labelProperty] : '';
|
|
29
31
|
});
|
|
30
32
|
this.filteredOptions = computed(() => {
|
|
31
|
-
|
|
33
|
+
// Use signal instead of props directly
|
|
34
|
+
const options = this.optionsSignal();
|
|
32
35
|
const search = this.searchTerm().toLowerCase();
|
|
33
36
|
// Debug mejorado
|
|
34
37
|
console.log('[SelectSearch] filteredOptions computed:', {
|
|
35
38
|
optionsCount: options.length,
|
|
36
39
|
searchTerm: search,
|
|
37
|
-
options: options
|
|
40
|
+
options: options,
|
|
38
41
|
labelProperty: this.labelProperty,
|
|
39
42
|
valueProperty: this.valueProperty,
|
|
43
|
+
propsExists: !!this.props,
|
|
44
|
+
propsOptionsExists: !!this.props?.options,
|
|
45
|
+
propsOptionsLength: this.props?.options?.length || 0,
|
|
40
46
|
firstOptionStructure: options[0] ? {
|
|
41
47
|
[this.labelProperty]: options[0][this.labelProperty],
|
|
42
48
|
[this.valueProperty]: options[0][this.valueProperty],
|
|
43
|
-
typeOfId: typeof options[0][this.valueProperty]
|
|
44
|
-
|
|
49
|
+
typeOfId: typeof options[0][this.valueProperty],
|
|
50
|
+
fullOption: options[0]
|
|
51
|
+
} : 'NO_OPTIONS'
|
|
45
52
|
});
|
|
46
53
|
if (!search) {
|
|
47
54
|
return options;
|
|
@@ -60,8 +67,26 @@ export class SelectSearchComponent {
|
|
|
60
67
|
this.placeholder = this.langService.getText('_global', 'selectOption', 'Seleccione una opción');
|
|
61
68
|
// Close dropdown when clicking outside
|
|
62
69
|
document.addEventListener('click', this.handleClickOutside.bind(this));
|
|
70
|
+
// Debug effect to track when filteredOptions changes
|
|
71
|
+
effect(() => {
|
|
72
|
+
const options = this.filteredOptions();
|
|
73
|
+
console.log('[SelectSearch] EFFECT - filteredOptions changed:', {
|
|
74
|
+
optionsCount: options.length,
|
|
75
|
+
options: options,
|
|
76
|
+
timestamp: new Date().toLocaleTimeString()
|
|
77
|
+
});
|
|
78
|
+
});
|
|
63
79
|
}
|
|
64
80
|
ngOnInit() {
|
|
81
|
+
console.log('[SelectSearch] ngOnInit:', {
|
|
82
|
+
props: !!this.props,
|
|
83
|
+
options: this.props?.options?.length || 0,
|
|
84
|
+
control: !!this.props?.control
|
|
85
|
+
});
|
|
86
|
+
// Update options signal
|
|
87
|
+
if (this.props?.options) {
|
|
88
|
+
this.optionsSignal.set(this.props.options);
|
|
89
|
+
}
|
|
65
90
|
this.applyDefaultValue();
|
|
66
91
|
this.syncSelectedValue();
|
|
67
92
|
}
|
|
@@ -69,7 +94,18 @@ export class SelectSearchComponent {
|
|
|
69
94
|
document.removeEventListener('click', this.handleClickOutside.bind(this));
|
|
70
95
|
}
|
|
71
96
|
ngOnChanges(changes) {
|
|
97
|
+
console.log('[SelectSearch] ngOnChanges:', {
|
|
98
|
+
hasPropsChange: !!changes['props'],
|
|
99
|
+
props: !!this.props,
|
|
100
|
+
options: this.props?.options?.length || 0,
|
|
101
|
+
propsOptions: this.props?.options
|
|
102
|
+
});
|
|
72
103
|
if (changes['props'] && this.props) {
|
|
104
|
+
// Update options signal when props change
|
|
105
|
+
if (this.props.options) {
|
|
106
|
+
console.log('[SelectSearch] Updating optionsSignal with:', this.props.options);
|
|
107
|
+
this.optionsSignal.set(this.props.options);
|
|
108
|
+
}
|
|
73
109
|
this.syncSelectedValue();
|
|
74
110
|
}
|
|
75
111
|
}
|
|
@@ -117,13 +153,22 @@ export class SelectSearchComponent {
|
|
|
117
153
|
}
|
|
118
154
|
}
|
|
119
155
|
getOptionByValue(value) {
|
|
120
|
-
return this.
|
|
156
|
+
return this.optionsSignal().find(option => String(option[this.valueProperty]) === String(value));
|
|
121
157
|
}
|
|
122
158
|
applyDefaultValue() {
|
|
123
159
|
if (this.props) {
|
|
124
160
|
applyDefaultValueToControl(this.props);
|
|
125
161
|
}
|
|
126
162
|
}
|
|
163
|
+
debugOptions() {
|
|
164
|
+
console.log('[SelectSearch] DEBUG CLICK:', {
|
|
165
|
+
props: this.props,
|
|
166
|
+
filteredOptions: this.filteredOptions(),
|
|
167
|
+
searchTerm: this.searchTerm(),
|
|
168
|
+
propsOptions: this.props?.options,
|
|
169
|
+
computed: this.filteredOptions()
|
|
170
|
+
});
|
|
171
|
+
}
|
|
127
172
|
syncSelectedValue() {
|
|
128
173
|
if (this.props?.control?.value) {
|
|
129
174
|
this.selectedValue.set(String(this.props.control.value));
|
|
@@ -164,6 +209,11 @@ export class SelectSearchComponent {
|
|
|
164
209
|
[class.visible]="isOpen()"
|
|
165
210
|
#dropdown
|
|
166
211
|
>
|
|
212
|
+
<!-- Debug button -->
|
|
213
|
+
<div class="search-container" style="background: red; color: white; padding: 4px;">
|
|
214
|
+
<button (click)="debugOptions()">DEBUG: {{ filteredOptions().length }} opciones</button>
|
|
215
|
+
</div>
|
|
216
|
+
|
|
167
217
|
<!-- Search bar -->
|
|
168
218
|
<div class="search-container" *ngIf="props?.options && props.options.length > 5">
|
|
169
219
|
<ion-searchbar
|
|
@@ -198,6 +248,8 @@ export class SelectSearchComponent {
|
|
|
198
248
|
<ion-item *ngIf="filteredOptions().length === 0" class="no-results">
|
|
199
249
|
<ion-label color="medium">
|
|
200
250
|
{{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
|
|
251
|
+
<!-- Debug info -->
|
|
252
|
+
<br><small>Debug: {{ filteredOptions().length }} opciones | Props: {{ !!props }} | Search: "{{ searchTerm() }}"</small>
|
|
201
253
|
</ion-label>
|
|
202
254
|
</ion-item>
|
|
203
255
|
</ion-list>
|
|
@@ -241,6 +293,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
241
293
|
[class.visible]="isOpen()"
|
|
242
294
|
#dropdown
|
|
243
295
|
>
|
|
296
|
+
<!-- Debug button -->
|
|
297
|
+
<div class="search-container" style="background: red; color: white; padding: 4px;">
|
|
298
|
+
<button (click)="debugOptions()">DEBUG: {{ filteredOptions().length }} opciones</button>
|
|
299
|
+
</div>
|
|
300
|
+
|
|
244
301
|
<!-- Search bar -->
|
|
245
302
|
<div class="search-container" *ngIf="props?.options && props.options.length > 5">
|
|
246
303
|
<ion-searchbar
|
|
@@ -275,6 +332,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
275
332
|
<ion-item *ngIf="filteredOptions().length === 0" class="no-results">
|
|
276
333
|
<ion-label color="medium">
|
|
277
334
|
{{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
|
|
335
|
+
<!-- Debug info -->
|
|
336
|
+
<br><small>Debug: {{ filteredOptions().length }} opciones | Props: {{ !!props }} | Search: "{{ searchTerm() }}"</small>
|
|
278
337
|
</ion-label>
|
|
279
338
|
</ion-item>
|
|
280
339
|
</ion-list>
|
|
@@ -296,4 +355,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
296
355
|
}], placeholder: [{
|
|
297
356
|
type: Input
|
|
298
357
|
}] } });
|
|
299
|
-
//# 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  ElementRef,\n  inject,\n  Input,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  signal,\n  computed\n} from '@angular/core';\nimport { 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-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  template: `\n    <div class=\"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\" *ngIf=\"props?.options && props.options.length > 5\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"'Buscar'\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm()\"\n          show-clear-button=\"focus\"\n          [debounce]=\"200\"\n        ></ion-searchbar>\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)=\"selectOption(option)\"\n            class=\"option-item\"\n          >\n            <ion-label>{{ option[labelProperty] }}</ion-label>\n            <ion-icon \n              *ngIf=\"isSelected(option)\" \n              name=\"checkmark-outline\" \n              slot=\"end\" \n              color=\"primary\"\n            ></ion-icon>\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: ['./select-search.component.scss']\n})\nexport class SelectSearchComponent implements OnInit, OnDestroy, OnChanges {\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  selectedValue = signal<string | null>(null);\n  \n  // Computed signals\n  displayValue = computed(() => {\n    const value = this.selectedValue();\n    if (!value) return '';\n    \n    const option = this.getOptionByValue(value);\n    return option ? option[this.labelProperty] : '';\n  });\n\n  filteredOptions = computed(() => {\n    const options = this.props?.options || [];\n    const search = this.searchTerm().toLowerCase();\n    \n    // Debug mejorado\n    console.log('[SelectSearch] filteredOptions computed:', {\n      optionsCount: options.length,\n      searchTerm: search,\n      options: options.slice(0, 3),\n      labelProperty: this.labelProperty,\n      valueProperty: this.valueProperty,\n      firstOptionStructure: options[0] ? {\n        [this.labelProperty]: options[0][this.labelProperty],\n        [this.valueProperty]: options[0][this.valueProperty],\n        typeOfId: typeof options[0][this.valueProperty]\n      } : null\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  constructor() {\n    this.placeholder = this.langService.getText('_global', 'selectOption', 'Seleccione una opción');\n    \n    // Close dropdown when clicking outside\n    document.addEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnInit() {\n    this.applyDefaultValue();\n    this.syncSelectedValue();\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.syncSelectedValue();\n    }\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      // Focus search bar when opening if available\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  selectOption(option: InputOption): void {\n    const value = String(option[this.valueProperty]); // Convertir a string\n    this.selectedValue.set(value);\n    this.isOpen.set(false);\n    this.searchTerm.set('');\n    \n    // Update form control\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  isSelected(option: InputOption): boolean {\n    return String(this.selectedValue()) === 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 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 syncSelectedValue(): void {\n    if (this.props?.control?.value) {\n      this.selectedValue.set(String(this.props.control.value));\n    }\n  }\n}\n"]}
|
|
358
|
+
//# 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  ElementRef,\n  inject,\n  Input,\n  OnInit,\n  OnDestroy,\n  OnChanges,\n  SimpleChanges,\n  ViewChild,\n  signal,\n  computed,\n  effect\n} from '@angular/core';\nimport { 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-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  template: `\n    <div class=\"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      <!-- 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      <!-- Search bar -->\n      <div class=\"search-container\" *ngIf=\"props?.options && props.options.length > 5\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"'Buscar'\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm()\"\n          show-clear-button=\"focus\"\n          [debounce]=\"200\"\n        ></ion-searchbar>\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)=\"selectOption(option)\"\n            class=\"option-item\"\n          >\n            <ion-label>{{ option[labelProperty] }}</ion-label>\n            <ion-icon \n              *ngIf=\"isSelected(option)\" \n              name=\"checkmark-outline\" \n              slot=\"end\" \n              color=\"primary\"\n            ></ion-icon>\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              <!-- Debug info -->\n              <br><small>Debug: {{ filteredOptions().length }} opciones | Props: {{ !!props }} | Search: \"{{ searchTerm() }}\"</small>\n            </ion-label>\n          </ion-item>\n        </ion-list>\n      </div>\n    </div>\n  `,\n  styleUrls: ['./select-search.component.scss']\n})\nexport class SelectSearchComponent implements OnInit, OnDestroy, OnChanges {\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  selectedValue = signal<string | null>(null);\n  \n  // Force options to be reactive\n  optionsSignal = signal<InputOption[]>([]);\n  \n  // Computed signals\n  displayValue = computed(() => {\n    const value = this.selectedValue();\n    if (!value) return '';\n    \n    const option = this.getOptionByValue(value);\n    return option ? option[this.labelProperty] : '';\n  });\n\n  filteredOptions = computed(() => {\n    // Use signal instead of props directly\n    const options = this.optionsSignal();\n    const search = this.searchTerm().toLowerCase();\n    \n    // Debug mejorado\n    console.log('[SelectSearch] filteredOptions computed:', {\n      optionsCount: options.length,\n      searchTerm: search,\n      options: options,\n      labelProperty: this.labelProperty,\n      valueProperty: this.valueProperty,\n      propsExists: !!this.props,\n      propsOptionsExists: !!this.props?.options,\n      propsOptionsLength: this.props?.options?.length || 0,\n      firstOptionStructure: options[0] ? {\n        [this.labelProperty]: options[0][this.labelProperty],\n        [this.valueProperty]: options[0][this.valueProperty],\n        typeOfId: typeof options[0][this.valueProperty],\n        fullOption: options[0]\n      } : 'NO_OPTIONS'\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  constructor() {\n    this.placeholder = this.langService.getText('_global', 'selectOption', 'Seleccione una opción');\n    \n    // Close dropdown when clicking outside\n    document.addEventListener('click', this.handleClickOutside.bind(this));\n    \n    // Debug effect to track when filteredOptions changes\n    effect(() => {\n      const options = this.filteredOptions();\n      console.log('[SelectSearch] EFFECT - filteredOptions changed:', {\n        optionsCount: options.length,\n        options: options,\n        timestamp: new Date().toLocaleTimeString()\n      });\n    });\n  }\n\n  ngOnInit() {\n    console.log('[SelectSearch] ngOnInit:', {\n      props: !!this.props,\n      options: this.props?.options?.length || 0,\n      control: !!this.props?.control\n    });\n    \n    // Update options signal\n    if (this.props?.options) {\n      this.optionsSignal.set(this.props.options);\n    }\n    \n    this.applyDefaultValue();\n    this.syncSelectedValue();\n  }\n\n  ngOnDestroy() {\n    document.removeEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    console.log('[SelectSearch] ngOnChanges:', {\n      hasPropsChange: !!changes['props'],\n      props: !!this.props,\n      options: this.props?.options?.length || 0,\n      propsOptions: this.props?.options\n    });\n    \n    if (changes['props'] && this.props) {\n      // Update options signal when props change\n      if (this.props.options) {\n        console.log('[SelectSearch] Updating optionsSignal with:', this.props.options);\n        this.optionsSignal.set(this.props.options);\n      }\n      this.syncSelectedValue();\n    }\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      // Focus search bar when opening if available\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  selectOption(option: InputOption): void {\n    const value = String(option[this.valueProperty]); // Convertir a string\n    this.selectedValue.set(value);\n    this.isOpen.set(false);\n    this.searchTerm.set('');\n    \n    // Update form control\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  isSelected(option: InputOption): boolean {\n    return String(this.selectedValue()) === 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 getOptionByValue(value: string): InputOption | undefined {\n    return this.optionsSignal().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  debugOptions(): void {\n    console.log('[SelectSearch] DEBUG CLICK:', {\n      props: this.props,\n      filteredOptions: this.filteredOptions(),\n      searchTerm: this.searchTerm(),\n      propsOptions: this.props?.options,\n      computed: this.filteredOptions()\n    });\n  }\n\n  private syncSelectedValue(): void {\n    if (this.props?.control?.value) {\n      this.selectedValue.set(String(this.props.control.value));\n    }\n  }\n}\n"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import * as i0 from '@angular/core';
|
|
2
|
-
import { EventEmitter, Component, Input, Output, Injectable, inject, InjectionToken, Inject, ChangeDetectorRef, HostListener, Pipe, ChangeDetectionStrategy, ViewChild, signal, computed, forwardRef } from '@angular/core';
|
|
2
|
+
import { EventEmitter, Component, Input, Output, Injectable, inject, InjectionToken, Inject, ChangeDetectorRef, HostListener, Pipe, ChangeDetectionStrategy, ViewChild, signal, computed, forwardRef, effect } from '@angular/core';
|
|
3
3
|
import { IonAvatar, IonCard, IonIcon, IonButton, IonSpinner, IonText, IonModal, IonHeader, IonToolbar, IonContent, IonButtons, IonTitle, IonProgressBar, IonCardContent, IonCardHeader, IonCardTitle, IonCardSubtitle, IonCheckbox, IonTextarea, IonDatetime, IonDatetimeButton, IonInput, IonSelect, IonSelectOption, IonLabel, IonSearchbar, IonRadioGroup, IonRadio, IonMenuButton, IonFooter, IonList, IonListHeader, IonNote, IonItem } from '@ionic/angular/standalone';
|
|
4
4
|
import * as i1 from '@angular/common';
|
|
5
5
|
import { CommonModule, NgStyle, Location, AsyncPipe, NgFor, NgClass } from '@angular/common';
|
|
@@ -6769,6 +6769,8 @@ class SelectSearchComponent {
|
|
|
6769
6769
|
this.isOpen = signal(false);
|
|
6770
6770
|
this.searchTerm = signal('');
|
|
6771
6771
|
this.selectedValue = signal(null);
|
|
6772
|
+
// Force options to be reactive
|
|
6773
|
+
this.optionsSignal = signal([]);
|
|
6772
6774
|
// Computed signals
|
|
6773
6775
|
this.displayValue = computed(() => {
|
|
6774
6776
|
const value = this.selectedValue();
|
|
@@ -6778,20 +6780,25 @@ class SelectSearchComponent {
|
|
|
6778
6780
|
return option ? option[this.labelProperty] : '';
|
|
6779
6781
|
});
|
|
6780
6782
|
this.filteredOptions = computed(() => {
|
|
6781
|
-
|
|
6783
|
+
// Use signal instead of props directly
|
|
6784
|
+
const options = this.optionsSignal();
|
|
6782
6785
|
const search = this.searchTerm().toLowerCase();
|
|
6783
6786
|
// Debug mejorado
|
|
6784
6787
|
console.log('[SelectSearch] filteredOptions computed:', {
|
|
6785
6788
|
optionsCount: options.length,
|
|
6786
6789
|
searchTerm: search,
|
|
6787
|
-
options: options
|
|
6790
|
+
options: options,
|
|
6788
6791
|
labelProperty: this.labelProperty,
|
|
6789
6792
|
valueProperty: this.valueProperty,
|
|
6793
|
+
propsExists: !!this.props,
|
|
6794
|
+
propsOptionsExists: !!this.props?.options,
|
|
6795
|
+
propsOptionsLength: this.props?.options?.length || 0,
|
|
6790
6796
|
firstOptionStructure: options[0] ? {
|
|
6791
6797
|
[this.labelProperty]: options[0][this.labelProperty],
|
|
6792
6798
|
[this.valueProperty]: options[0][this.valueProperty],
|
|
6793
|
-
typeOfId: typeof options[0][this.valueProperty]
|
|
6794
|
-
|
|
6799
|
+
typeOfId: typeof options[0][this.valueProperty],
|
|
6800
|
+
fullOption: options[0]
|
|
6801
|
+
} : 'NO_OPTIONS'
|
|
6795
6802
|
});
|
|
6796
6803
|
if (!search) {
|
|
6797
6804
|
return options;
|
|
@@ -6810,8 +6817,26 @@ class SelectSearchComponent {
|
|
|
6810
6817
|
this.placeholder = this.langService.getText('_global', 'selectOption', 'Seleccione una opción');
|
|
6811
6818
|
// Close dropdown when clicking outside
|
|
6812
6819
|
document.addEventListener('click', this.handleClickOutside.bind(this));
|
|
6820
|
+
// Debug effect to track when filteredOptions changes
|
|
6821
|
+
effect(() => {
|
|
6822
|
+
const options = this.filteredOptions();
|
|
6823
|
+
console.log('[SelectSearch] EFFECT - filteredOptions changed:', {
|
|
6824
|
+
optionsCount: options.length,
|
|
6825
|
+
options: options,
|
|
6826
|
+
timestamp: new Date().toLocaleTimeString()
|
|
6827
|
+
});
|
|
6828
|
+
});
|
|
6813
6829
|
}
|
|
6814
6830
|
ngOnInit() {
|
|
6831
|
+
console.log('[SelectSearch] ngOnInit:', {
|
|
6832
|
+
props: !!this.props,
|
|
6833
|
+
options: this.props?.options?.length || 0,
|
|
6834
|
+
control: !!this.props?.control
|
|
6835
|
+
});
|
|
6836
|
+
// Update options signal
|
|
6837
|
+
if (this.props?.options) {
|
|
6838
|
+
this.optionsSignal.set(this.props.options);
|
|
6839
|
+
}
|
|
6815
6840
|
this.applyDefaultValue();
|
|
6816
6841
|
this.syncSelectedValue();
|
|
6817
6842
|
}
|
|
@@ -6819,7 +6844,18 @@ class SelectSearchComponent {
|
|
|
6819
6844
|
document.removeEventListener('click', this.handleClickOutside.bind(this));
|
|
6820
6845
|
}
|
|
6821
6846
|
ngOnChanges(changes) {
|
|
6847
|
+
console.log('[SelectSearch] ngOnChanges:', {
|
|
6848
|
+
hasPropsChange: !!changes['props'],
|
|
6849
|
+
props: !!this.props,
|
|
6850
|
+
options: this.props?.options?.length || 0,
|
|
6851
|
+
propsOptions: this.props?.options
|
|
6852
|
+
});
|
|
6822
6853
|
if (changes['props'] && this.props) {
|
|
6854
|
+
// Update options signal when props change
|
|
6855
|
+
if (this.props.options) {
|
|
6856
|
+
console.log('[SelectSearch] Updating optionsSignal with:', this.props.options);
|
|
6857
|
+
this.optionsSignal.set(this.props.options);
|
|
6858
|
+
}
|
|
6823
6859
|
this.syncSelectedValue();
|
|
6824
6860
|
}
|
|
6825
6861
|
}
|
|
@@ -6867,13 +6903,22 @@ class SelectSearchComponent {
|
|
|
6867
6903
|
}
|
|
6868
6904
|
}
|
|
6869
6905
|
getOptionByValue(value) {
|
|
6870
|
-
return this.
|
|
6906
|
+
return this.optionsSignal().find(option => String(option[this.valueProperty]) === String(value));
|
|
6871
6907
|
}
|
|
6872
6908
|
applyDefaultValue() {
|
|
6873
6909
|
if (this.props) {
|
|
6874
6910
|
applyDefaultValueToControl(this.props);
|
|
6875
6911
|
}
|
|
6876
6912
|
}
|
|
6913
|
+
debugOptions() {
|
|
6914
|
+
console.log('[SelectSearch] DEBUG CLICK:', {
|
|
6915
|
+
props: this.props,
|
|
6916
|
+
filteredOptions: this.filteredOptions(),
|
|
6917
|
+
searchTerm: this.searchTerm(),
|
|
6918
|
+
propsOptions: this.props?.options,
|
|
6919
|
+
computed: this.filteredOptions()
|
|
6920
|
+
});
|
|
6921
|
+
}
|
|
6877
6922
|
syncSelectedValue() {
|
|
6878
6923
|
if (this.props?.control?.value) {
|
|
6879
6924
|
this.selectedValue.set(String(this.props.control.value));
|
|
@@ -6914,6 +6959,11 @@ class SelectSearchComponent {
|
|
|
6914
6959
|
[class.visible]="isOpen()"
|
|
6915
6960
|
#dropdown
|
|
6916
6961
|
>
|
|
6962
|
+
<!-- Debug button -->
|
|
6963
|
+
<div class="search-container" style="background: red; color: white; padding: 4px;">
|
|
6964
|
+
<button (click)="debugOptions()">DEBUG: {{ filteredOptions().length }} opciones</button>
|
|
6965
|
+
</div>
|
|
6966
|
+
|
|
6917
6967
|
<!-- Search bar -->
|
|
6918
6968
|
<div class="search-container" *ngIf="props?.options && props.options.length > 5">
|
|
6919
6969
|
<ion-searchbar
|
|
@@ -6948,6 +6998,8 @@ class SelectSearchComponent {
|
|
|
6948
6998
|
<ion-item *ngIf="filteredOptions().length === 0" class="no-results">
|
|
6949
6999
|
<ion-label color="medium">
|
|
6950
7000
|
{{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
|
|
7001
|
+
<!-- Debug info -->
|
|
7002
|
+
<br><small>Debug: {{ filteredOptions().length }} opciones | Props: {{ !!props }} | Search: "{{ searchTerm() }}"</small>
|
|
6951
7003
|
</ion-label>
|
|
6952
7004
|
</ion-item>
|
|
6953
7005
|
</ion-list>
|
|
@@ -6991,6 +7043,11 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
6991
7043
|
[class.visible]="isOpen()"
|
|
6992
7044
|
#dropdown
|
|
6993
7045
|
>
|
|
7046
|
+
<!-- Debug button -->
|
|
7047
|
+
<div class="search-container" style="background: red; color: white; padding: 4px;">
|
|
7048
|
+
<button (click)="debugOptions()">DEBUG: {{ filteredOptions().length }} opciones</button>
|
|
7049
|
+
</div>
|
|
7050
|
+
|
|
6994
7051
|
<!-- Search bar -->
|
|
6995
7052
|
<div class="search-container" *ngIf="props?.options && props.options.length > 5">
|
|
6996
7053
|
<ion-searchbar
|
|
@@ -7025,6 +7082,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
7025
7082
|
<ion-item *ngIf="filteredOptions().length === 0" class="no-results">
|
|
7026
7083
|
<ion-label color="medium">
|
|
7027
7084
|
{{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
|
|
7085
|
+
<!-- Debug info -->
|
|
7086
|
+
<br><small>Debug: {{ filteredOptions().length }} opciones | Props: {{ !!props }} | Search: "{{ searchTerm() }}"</small>
|
|
7028
7087
|
</ion-label>
|
|
7029
7088
|
</ion-item>
|
|
7030
7089
|
</ion-list>
|