valtech-components 2.0.382 → 2.0.383

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.
@@ -0,0 +1,386 @@
1
+ import { CommonModule } from '@angular/common';
2
+ import { Component, forwardRef, inject, Input, ViewChild, signal, computed } from '@angular/core';
3
+ import { NG_VALUE_ACCESSOR, FormsModule, ReactiveFormsModule } from '@angular/forms';
4
+ import { IonicModule } from '@ionic/angular';
5
+ import { LangService } from '../../../services/lang-provider/lang-provider.service';
6
+ import { applyDefaultValueToControl } from '../../../shared/utils/form-defaults';
7
+ import { replaceSpecialChars } from '../../../shared/utils/text';
8
+ import * as i0 from "@angular/core";
9
+ import * as i1 from "@angular/common";
10
+ import * as i2 from "@ionic/angular";
11
+ import * as i3 from "@angular/forms";
12
+ export class MultiSelectSimpleComponent {
13
+ constructor() {
14
+ this.labelProperty = 'name';
15
+ this.valueProperty = 'id';
16
+ this.placeholder = '';
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
+ if (!search) {
38
+ return options;
39
+ }
40
+ return options.filter(option => {
41
+ const label = option[this.labelProperty]
42
+ ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())
43
+ : '';
44
+ const value = option[this.valueProperty]
45
+ ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())
46
+ : '';
47
+ const searchTerm = replaceSpecialChars(search);
48
+ return label.includes(searchTerm) || value.includes(searchTerm);
49
+ });
50
+ });
51
+ this.searchPlaceholder = '';
52
+ // ControlValueAccessor
53
+ this.onChange = (value) => { };
54
+ this.onTouched = () => { };
55
+ this.placeholder = this.langService.getText('_global', 'selectOptions', 'Seleccione opciones');
56
+ this.searchPlaceholder = this.langService.getText('_global', 'search', 'Buscar');
57
+ // Close dropdown when clicking outside
58
+ document.addEventListener('click', this.handleClickOutside.bind(this));
59
+ }
60
+ ngOnInit() {
61
+ this.applyDefaultValue();
62
+ this.syncSelectedValues();
63
+ }
64
+ ngOnDestroy() {
65
+ document.removeEventListener('click', this.handleClickOutside.bind(this));
66
+ }
67
+ // ControlValueAccessor implementation
68
+ writeValue(value) {
69
+ const valueArray = this.parseValue(value);
70
+ this.selectedValues.set(valueArray);
71
+ }
72
+ registerOnChange(fn) {
73
+ this.onChange = fn;
74
+ }
75
+ registerOnTouched(fn) {
76
+ this.onTouched = fn;
77
+ }
78
+ setDisabledState(isDisabled) {
79
+ // Handle disabled state if needed
80
+ }
81
+ // Component methods
82
+ toggleDropdown(event) {
83
+ event.stopPropagation();
84
+ this.isOpen.update(value => !value);
85
+ if (this.isOpen()) {
86
+ this.onTouched();
87
+ // Focus search bar when opening
88
+ setTimeout(() => {
89
+ const searchbar = this.dropdownRef?.nativeElement?.querySelector('ion-searchbar');
90
+ if (searchbar) {
91
+ searchbar.setFocus();
92
+ }
93
+ }, 100);
94
+ }
95
+ }
96
+ onSearch(event) {
97
+ this.searchTerm.set(event.detail.value || '');
98
+ }
99
+ toggleOption(option) {
100
+ const value = option[this.valueProperty];
101
+ const currentSelected = this.selectedValues();
102
+ if (this.isSelected(option)) {
103
+ // Remove from selection
104
+ this.selectedValues.set(currentSelected.filter(v => v !== value));
105
+ }
106
+ else {
107
+ // Add to selection
108
+ this.selectedValues.set([...currentSelected, value]);
109
+ }
110
+ this.emitValue();
111
+ }
112
+ selectAll() {
113
+ const allValues = this.filteredOptions().map(option => option[this.valueProperty]);
114
+ const currentSelected = this.selectedValues();
115
+ // Add only new values to avoid duplicates
116
+ const newValues = allValues.filter(value => !currentSelected.includes(value));
117
+ this.selectedValues.set([...currentSelected, ...newValues]);
118
+ this.emitValue();
119
+ }
120
+ clearAll() {
121
+ this.selectedValues.set([]);
122
+ this.emitValue();
123
+ }
124
+ isSelected(option) {
125
+ return this.selectedValues().includes(option[this.valueProperty]);
126
+ }
127
+ trackByFn(index, option) {
128
+ return option[this.valueProperty];
129
+ }
130
+ handleClickOutside(event) {
131
+ if (this.isOpen() &&
132
+ !this.mainInputRef?.nativeElement?.contains(event.target) &&
133
+ !this.dropdownRef?.nativeElement?.contains(event.target)) {
134
+ this.isOpen.set(false);
135
+ this.searchTerm.set('');
136
+ }
137
+ }
138
+ parseValue(value) {
139
+ if (!value)
140
+ return [];
141
+ if (Array.isArray(value))
142
+ return value.map(v => String(v));
143
+ if (typeof value === 'string') {
144
+ return value.split(',').map(v => v.trim()).filter(v => v !== '');
145
+ }
146
+ return [String(value)];
147
+ }
148
+ emitValue() {
149
+ const value = this.selectedValues().join(',');
150
+ this.onChange(value);
151
+ if (this.props?.control) {
152
+ this.props.control.setValue(value);
153
+ this.props.control.markAsDirty();
154
+ this.props.control.markAsTouched();
155
+ }
156
+ }
157
+ getOptionByValue(value) {
158
+ return this.props?.options?.find(option => option[this.valueProperty] === value);
159
+ }
160
+ applyDefaultValue() {
161
+ if (this.props) {
162
+ applyDefaultValueToControl(this.props);
163
+ }
164
+ }
165
+ syncSelectedValues() {
166
+ if (this.props?.control?.value) {
167
+ const valueArray = this.parseValue(this.props.control.value);
168
+ this.selectedValues.set(valueArray);
169
+ }
170
+ }
171
+ static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiSelectSimpleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
172
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "17.3.12", type: MultiSelectSimpleComponent, isStandalone: true, selector: "val-multi-select-simple", inputs: { props: "props", labelProperty: "labelProperty", valueProperty: "valueProperty", placeholder: "placeholder" }, providers: [
173
+ {
174
+ provide: NG_VALUE_ACCESSOR,
175
+ useExisting: forwardRef(() => MultiSelectSimpleComponent),
176
+ multi: true
177
+ }
178
+ ], viewQueries: [{ propertyName: "dropdownRef", first: true, predicate: ["dropdown"], descendants: true }, { propertyName: "mainInputRef", first: true, predicate: ["mainInput"], descendants: true }], ngImport: i0, template: `
179
+ <div class="multi-select-container" (click)="toggleDropdown($event)">
180
+ <!-- Main input display -->
181
+ <ion-input
182
+ #mainInput
183
+ type="text"
184
+ [value]="displayValue()"
185
+ [placeholder]="props?.placeholder || placeholder"
186
+ readonly
187
+ class="main-input"
188
+ [class.is-open]="isOpen()"
189
+ />
190
+
191
+ <!-- Dropdown icon -->
192
+ <ion-icon
193
+ name="chevron-down-outline"
194
+ class="dropdown-icon"
195
+ [class.rotated]="isOpen()"
196
+ ></ion-icon>
197
+
198
+ <!-- Hidden input for form control -->
199
+ <ion-input
200
+ style="position: absolute; opacity: 0; pointer-events: none;"
201
+ [formControl]="props?.control"
202
+ type="hidden"
203
+ />
204
+ </div>
205
+
206
+ <!-- Dropdown overlay -->
207
+ <div
208
+ class="dropdown-overlay"
209
+ [class.visible]="isOpen()"
210
+ #dropdown
211
+ >
212
+ <!-- Search bar -->
213
+ <div class="search-container">
214
+ <ion-searchbar
215
+ #searchbar
216
+ [placeholder]="searchPlaceholder"
217
+ (ionInput)="onSearch($event)"
218
+ [value]="searchTerm()"
219
+ show-clear-button="focus"
220
+ ></ion-searchbar>
221
+ </div>
222
+
223
+ <!-- Action buttons -->
224
+ <div class="actions-container">
225
+ <ion-button
226
+ fill="clear"
227
+ size="small"
228
+ (click)="selectAll()"
229
+ [disabled]="filteredOptions().length === 0"
230
+ >
231
+ Seleccionar todos
232
+ </ion-button>
233
+ <ion-button
234
+ fill="clear"
235
+ size="small"
236
+ color="medium"
237
+ (click)="clearAll()"
238
+ [disabled]="selectedValues().length === 0"
239
+ >
240
+ Limpiar
241
+ </ion-button>
242
+ </div>
243
+
244
+ <!-- Options list -->
245
+ <div class="options-container">
246
+ <ion-list class="options-list">
247
+ <ion-item
248
+ *ngFor="let option of filteredOptions(); trackBy: trackByFn"
249
+ button
250
+ (click)="toggleOption(option)"
251
+ class="option-item"
252
+ >
253
+ <ion-checkbox
254
+ slot="start"
255
+ [checked]="isSelected(option)"
256
+ ></ion-checkbox>
257
+ <ion-label>{{ option[labelProperty] }}</ion-label>
258
+ </ion-item>
259
+
260
+ <!-- No results message -->
261
+ <ion-item *ngIf="filteredOptions().length === 0" class="no-results">
262
+ <ion-label color="medium">
263
+ {{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
264
+ </ion-label>
265
+ </ion-item>
266
+ </ion-list>
267
+ </div>
268
+ </div>
269
+ `, isInline: true, styles: [":host{display:block;position:relative;width:100%}.multi-select-container{position:relative;display:flex;align-items:center;cursor:pointer}.multi-select-container .main-input{flex:1;cursor:pointer}.multi-select-container .main-input.is-open{--border-color: var(--ion-color-primary)}.multi-select-container .dropdown-icon{position:absolute;right:12px;font-size:16px;color:var(--ion-color-medium);transition:transform .2s ease;pointer-events:none;z-index:2}.multi-select-container .dropdown-icon.rotated{transform:rotate(180deg)}.dropdown-overlay{position:absolute;top:100%;left:0;right:0;background:var(--ion-background-color);border:1px solid var(--ion-color-light);border-radius:8px;box-shadow:0 4px 16px #0000001a;z-index:1000;max-height:400px;opacity:0;transform:translateY(-8px);pointer-events:none;transition:all .2s ease}.dropdown-overlay.visible{opacity:1;transform:translateY(4px);pointer-events:all}.search-container{padding:12px;border-bottom:1px solid var(--ion-color-light)}.search-container ion-searchbar{--background: var(--ion-color-light);--border-radius: 8px;--box-shadow: none;--padding-start: 12px;--padding-end: 12px;height:40px}.actions-container{display:flex;justify-content:space-between;padding:8px 12px;border-bottom:1px solid var(--ion-color-light)}.actions-container ion-button{--padding-start: 8px;--padding-end: 8px;height:32px;font-size:14px}.options-container{max-height:240px;overflow-y:auto}.options-container .options-list{padding:0}.options-container .option-item{--padding-start: 16px;--padding-end: 16px;--min-height: 48px;cursor:pointer}.options-container .option-item:hover{--background: var(--ion-color-light)}.options-container .option-item ion-checkbox{--size: 20px;margin-right:12px}.options-container .option-item ion-label{font-size:16px;line-height:1.4}.options-container .no-results{--padding-start: 16px;--padding-end: 16px;--min-height: 48px;text-align:center}.options-container .no-results ion-label{font-style:italic;font-size:14px}@media (max-width: 768px){.dropdown-overlay{max-height:300px}.options-container{max-height:200px}.actions-container ion-button{font-size:12px;--padding-start: 6px;--padding-end: 6px}}@media (prefers-color-scheme: dark){.dropdown-overlay{box-shadow:0 4px 16px #0000004d}}.option-item:focus-within{--background: var(--ion-color-primary-tint);outline:2px solid var(--ion-color-primary);outline-offset:-2px}.option-item{transition:background-color .15s ease}.dropdown-icon{transition:transform .2s cubic-bezier(.4,0,.2,1)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: IonicModule }, { kind: "component", type: i2.IonButton, selector: "ion-button", inputs: ["buttonType", "color", "disabled", "download", "expand", "fill", "form", "href", "mode", "rel", "routerAnimation", "routerDirection", "shape", "size", "strong", "target", "type"] }, { kind: "component", type: i2.IonCheckbox, selector: "ion-checkbox", inputs: ["alignment", "checked", "color", "disabled", "indeterminate", "justify", "labelPlacement", "mode", "name", "value"] }, { kind: "component", type: i2.IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: i2.IonInput, selector: "ion-input", inputs: ["autocapitalize", "autocomplete", "autocorrect", "autofocus", "clearInput", "clearOnEdit", "color", "counter", "counterFormatter", "debounce", "disabled", "enterkeyhint", "errorText", "fill", "helperText", "inputmode", "label", "labelPlacement", "max", "maxlength", "min", "minlength", "mode", "multiple", "name", "pattern", "placeholder", "readonly", "required", "shape", "spellcheck", "step", "type", "value"] }, { kind: "component", type: i2.IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: i2.IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: i2.IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: i2.IonSearchbar, selector: "ion-searchbar", inputs: ["animated", "autocapitalize", "autocomplete", "autocorrect", "cancelButtonIcon", "cancelButtonText", "clearIcon", "color", "debounce", "disabled", "enterkeyhint", "inputmode", "maxlength", "minlength", "mode", "name", "placeholder", "searchIcon", "showCancelButton", "showClearButton", "spellcheck", "type", "value"] }, { kind: "directive", type: i2.BooleanValueAccessor, selector: "ion-checkbox,ion-toggle" }, { kind: "directive", type: i2.TextValueAccessor, selector: "ion-input:not([type=number]),ion-textarea,ion-searchbar" }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }] }); }
270
+ }
271
+ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiSelectSimpleComponent, decorators: [{
272
+ type: Component,
273
+ args: [{ selector: 'val-multi-select-simple', standalone: true, imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule], providers: [
274
+ {
275
+ provide: NG_VALUE_ACCESSOR,
276
+ useExisting: forwardRef(() => MultiSelectSimpleComponent),
277
+ multi: true
278
+ }
279
+ ], template: `
280
+ <div class="multi-select-container" (click)="toggleDropdown($event)">
281
+ <!-- Main input display -->
282
+ <ion-input
283
+ #mainInput
284
+ type="text"
285
+ [value]="displayValue()"
286
+ [placeholder]="props?.placeholder || placeholder"
287
+ readonly
288
+ class="main-input"
289
+ [class.is-open]="isOpen()"
290
+ />
291
+
292
+ <!-- Dropdown icon -->
293
+ <ion-icon
294
+ name="chevron-down-outline"
295
+ class="dropdown-icon"
296
+ [class.rotated]="isOpen()"
297
+ ></ion-icon>
298
+
299
+ <!-- Hidden input for form control -->
300
+ <ion-input
301
+ style="position: absolute; opacity: 0; pointer-events: none;"
302
+ [formControl]="props?.control"
303
+ type="hidden"
304
+ />
305
+ </div>
306
+
307
+ <!-- Dropdown overlay -->
308
+ <div
309
+ class="dropdown-overlay"
310
+ [class.visible]="isOpen()"
311
+ #dropdown
312
+ >
313
+ <!-- Search bar -->
314
+ <div class="search-container">
315
+ <ion-searchbar
316
+ #searchbar
317
+ [placeholder]="searchPlaceholder"
318
+ (ionInput)="onSearch($event)"
319
+ [value]="searchTerm()"
320
+ show-clear-button="focus"
321
+ ></ion-searchbar>
322
+ </div>
323
+
324
+ <!-- Action buttons -->
325
+ <div class="actions-container">
326
+ <ion-button
327
+ fill="clear"
328
+ size="small"
329
+ (click)="selectAll()"
330
+ [disabled]="filteredOptions().length === 0"
331
+ >
332
+ Seleccionar todos
333
+ </ion-button>
334
+ <ion-button
335
+ fill="clear"
336
+ size="small"
337
+ color="medium"
338
+ (click)="clearAll()"
339
+ [disabled]="selectedValues().length === 0"
340
+ >
341
+ Limpiar
342
+ </ion-button>
343
+ </div>
344
+
345
+ <!-- Options list -->
346
+ <div class="options-container">
347
+ <ion-list class="options-list">
348
+ <ion-item
349
+ *ngFor="let option of filteredOptions(); trackBy: trackByFn"
350
+ button
351
+ (click)="toggleOption(option)"
352
+ class="option-item"
353
+ >
354
+ <ion-checkbox
355
+ slot="start"
356
+ [checked]="isSelected(option)"
357
+ ></ion-checkbox>
358
+ <ion-label>{{ option[labelProperty] }}</ion-label>
359
+ </ion-item>
360
+
361
+ <!-- No results message -->
362
+ <ion-item *ngIf="filteredOptions().length === 0" class="no-results">
363
+ <ion-label color="medium">
364
+ {{ searchTerm() ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
365
+ </ion-label>
366
+ </ion-item>
367
+ </ion-list>
368
+ </div>
369
+ </div>
370
+ `, styles: [":host{display:block;position:relative;width:100%}.multi-select-container{position:relative;display:flex;align-items:center;cursor:pointer}.multi-select-container .main-input{flex:1;cursor:pointer}.multi-select-container .main-input.is-open{--border-color: var(--ion-color-primary)}.multi-select-container .dropdown-icon{position:absolute;right:12px;font-size:16px;color:var(--ion-color-medium);transition:transform .2s ease;pointer-events:none;z-index:2}.multi-select-container .dropdown-icon.rotated{transform:rotate(180deg)}.dropdown-overlay{position:absolute;top:100%;left:0;right:0;background:var(--ion-background-color);border:1px solid var(--ion-color-light);border-radius:8px;box-shadow:0 4px 16px #0000001a;z-index:1000;max-height:400px;opacity:0;transform:translateY(-8px);pointer-events:none;transition:all .2s ease}.dropdown-overlay.visible{opacity:1;transform:translateY(4px);pointer-events:all}.search-container{padding:12px;border-bottom:1px solid var(--ion-color-light)}.search-container ion-searchbar{--background: var(--ion-color-light);--border-radius: 8px;--box-shadow: none;--padding-start: 12px;--padding-end: 12px;height:40px}.actions-container{display:flex;justify-content:space-between;padding:8px 12px;border-bottom:1px solid var(--ion-color-light)}.actions-container ion-button{--padding-start: 8px;--padding-end: 8px;height:32px;font-size:14px}.options-container{max-height:240px;overflow-y:auto}.options-container .options-list{padding:0}.options-container .option-item{--padding-start: 16px;--padding-end: 16px;--min-height: 48px;cursor:pointer}.options-container .option-item:hover{--background: var(--ion-color-light)}.options-container .option-item ion-checkbox{--size: 20px;margin-right:12px}.options-container .option-item ion-label{font-size:16px;line-height:1.4}.options-container .no-results{--padding-start: 16px;--padding-end: 16px;--min-height: 48px;text-align:center}.options-container .no-results ion-label{font-style:italic;font-size:14px}@media (max-width: 768px){.dropdown-overlay{max-height:300px}.options-container{max-height:200px}.actions-container ion-button{font-size:12px;--padding-start: 6px;--padding-end: 6px}}@media (prefers-color-scheme: dark){.dropdown-overlay{box-shadow:0 4px 16px #0000004d}}.option-item:focus-within{--background: var(--ion-color-primary-tint);outline:2px solid var(--ion-color-primary);outline-offset:-2px}.option-item{transition:background-color .15s ease}.dropdown-icon{transition:transform .2s cubic-bezier(.4,0,.2,1)}\n"] }]
371
+ }], ctorParameters: () => [], propDecorators: { dropdownRef: [{
372
+ type: ViewChild,
373
+ args: ['dropdown']
374
+ }], mainInputRef: [{
375
+ type: ViewChild,
376
+ args: ['mainInput']
377
+ }], props: [{
378
+ type: Input
379
+ }], labelProperty: [{
380
+ type: Input
381
+ }], valueProperty: [{
382
+ type: Input
383
+ }], placeholder: [{
384
+ type: Input
385
+ }] } });
386
+ //# 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  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, 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    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  // 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 = option[this.valueProperty];\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 => 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(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 => option[this.valueProperty] === value);\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}"]}
@@ -127,7 +127,7 @@ export class PopoverSelectorComponent {
127
127
 
128
128
  <span class="trigger-text">{{ getDisplayText() }}</span>
129
129
 
130
- <ion-icon name="chevron-down" slot="end" class="chevron-icon"> </ion-icon>
130
+ <ion-icon name="chevron-down-outline" slot="end" class="chevron-icon"> </ion-icon>
131
131
  </ion-button>
132
132
 
133
133
  <ion-select-option *ngFor="let option of props.options" [value]="option.value" [disabled]="option.disabled">
@@ -169,7 +169,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
169
169
 
170
170
  <span class="trigger-text">{{ getDisplayText() }}</span>
171
171
 
172
- <ion-icon name="chevron-down" slot="end" class="chevron-icon"> </ion-icon>
172
+ <ion-icon name="chevron-down-outline" slot="end" class="chevron-icon"> </ion-icon>
173
173
  </ion-button>
174
174
 
175
175
  <ion-select-option *ngFor="let option of props.options" [value]="option.value" [disabled]="option.disabled">
@@ -186,4 +186,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
186
186
  }], selectionChange: [{
187
187
  type: Output
188
188
  }] } });
189
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/popover-selector/popover-selector.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonButton, IonIcon, IonLabel, IonSelect, IonSelectOption } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronDown } from 'ionicons/icons';\nimport { LangService } from '../../../services/lang-provider/lang-provider.service';\nimport { PopoverSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-popover-selector',\n  standalone: true,\n  imports: [CommonModule, IonButton, IonIcon, IonSelect, IonSelectOption, IonLabel],\n  template: `\n    <div class=\"popover-selector-container\">\n      <ion-label *ngIf=\"props.label\" class=\"selector-label\">{{ props.label }}</ion-label>\n      <ion-select\n        [value]=\"props.selectedValue\"\n        [placeholder]=\"getPlaceholderText()\"\n        [interface]=\"props.interface || 'popover'\"\n        [multiple]=\"props.multiple || false\"\n        [disabled]=\"props.disabled || false\"\n        [cancelText]=\"getCancelText()\"\n        [okText]=\"getOkText()\"\n        (ionChange)=\"onSelectionChange($event)\"\n        class=\"popover-selector-select\"\n      >\n        <ion-button\n          slot=\"trigger\"\n          [fill]=\"props.fill || 'clear'\"\n          [size]=\"props.size || 'default'\"\n          [shape]=\"'round'\"\n          [color]=\"props.color || 'medium'\"\n          [disabled]=\"props.disabled || false\"\n          class=\"selector-trigger\"\n        >\n          <ion-icon *ngIf=\"props.icon\" [name]=\"props.icon\" slot=\"start\"> </ion-icon>\n\n          <span class=\"trigger-text\">{{ getDisplayText() }}</span>\n\n          <ion-icon name=\"chevron-down\" slot=\"end\" class=\"chevron-icon\"> </ion-icon>\n        </ion-button>\n\n        <ion-select-option *ngFor=\"let option of props.options\" [value]=\"option.value\" [disabled]=\"option.disabled\">\n          <div class=\"option-content\">\n            <ion-icon *ngIf=\"option.icon\" [name]=\"option.icon\" class=\"option-icon\"> </ion-icon>\n            <span>{{ option.label }}</span>\n          </div>\n        </ion-select-option>\n      </ion-select>\n    </div>\n  `,\n  styleUrls: ['./popover-selector.component.scss'],\n})\n/**\n * val-popover-selector\n *\n * A generic reusable popover selector component that can be used for any type of selection.\n * Provides a button trigger that opens a popover with selectable options.\n *\n * @example\n * // Basic usage\n * <val-popover-selector\n *   [props]=\"selectorProps\"\n *   (selectionChange)=\"onSelectionChange($event)\">\n * </val-popover-selector>\n *\n * @example\n * // With icon and custom styling\n * const selectorProps: PopoverSelectorMetadata = {\n *   options: [\n *     { value: 'option1', label: 'Option 1', icon: 'star' },\n *     { value: 'option2', label: 'Option 2', icon: 'heart' }\n *   ],\n *   selectedValue: 'option1',\n *   label: 'Choose option',\n *   icon: 'settings',\n *   color: 'primary',\n *   size: 'large',\n *   fill: 'solid'\n * };\n *\n * @input props: PopoverSelectorMetadata - Configuration for the selector\n * @output selectionChange: EventEmitter<string | string[]> - Emitted when selection changes\n */\nexport class PopoverSelectorComponent {\n  /**\n   * Popover selector configuration object.\n   * @type {PopoverSelectorMetadata}\n   */\n  @Input()\n  props: PopoverSelectorMetadata;\n\n  /**\n   * Event emitted when the selection changes.\n   * Emits the selected value(s).\n   */\n  @Output()\n  selectionChange = new EventEmitter<string | string[]>();\n\n  private langService = inject(LangService);\n\n  constructor() {\n    // Register required icons\n    addIcons({ chevronDown });\n  }\n\n  /**\n   * Get reactive placeholder text.\n   */\n  getPlaceholderText(): string {\n    return this.props.placeholder || this.langService.getText('_global', 'select', 'Seleccionar...');\n  }\n\n  /**\n   * Get reactive cancel text.\n   */\n  getCancelText(): string {\n    return this.props.cancelText || this.langService.getText('_global', 'cancel', 'Cancelar');\n  }\n\n  /**\n   * Get reactive ok text.\n   */\n  getOkText(): string {\n    return this.props.okText || this.langService.getText('_global', 'ok', 'Aceptar');\n  }\n\n  /**\n   * Handle selection change from the ion-select.\n   * @param event - The ion-select change event\n   */\n  onSelectionChange(event: any) {\n    const selectedValue = event.detail.value;\n    this.selectionChange.emit(selectedValue);\n  }\n\n  /**\n   * Get the display text for the trigger button.\n   * Shows the selected option's label or placeholder.\n   * @returns The display text\n   */\n  getDisplayText(): string {\n    if (!this.props.selectedValue) {\n      return this.getPlaceholderText();\n    }\n\n    if (Array.isArray(this.props.selectedValue)) {\n      // Multiple selection\n      if (this.props.selectedValue.length === 0) {\n        return this.getPlaceholderText();\n      }\n      if (this.props.selectedValue.length === 1) {\n        const option = this.props.options.find(opt => opt.value === this.props.selectedValue[0]);\n        return option?.label || this.props.selectedValue[0];\n      }\n      const selectedText = this.langService.getText('_global', 'selected', 'seleccionados');\n      return `${this.props.selectedValue.length} ${selectedText}`;\n    }\n\n    // Single selection\n    const selectedOption = this.props.options.find(opt => opt.value === this.props.selectedValue);\n    return selectedOption?.label || this.props.selectedValue;\n  }\n}\n"]}
189
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"popover-selector.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/popover-selector/popover-selector.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,SAAS,EAAE,OAAO,EAAE,QAAQ,EAAE,SAAS,EAAE,eAAe,EAAE,MAAM,2BAA2B,CAAC;AACrG,OAAO,EAAE,QAAQ,EAAE,MAAM,UAAU,CAAC;AACpC,OAAO,EAAE,WAAW,EAAE,MAAM,gBAAgB,CAAC;AAC7C,OAAO,EAAE,WAAW,EAAE,MAAM,uDAAuD,CAAC;;;AAgDpF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA8BG;AACH,MAAM,OAAO,wBAAwB;IAiBnC;QATA;;;WAGG;QAEH,oBAAe,GAAG,IAAI,YAAY,EAAqB,CAAC;QAEhD,gBAAW,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAGxC,0BAA0B;QAC1B,QAAQ,CAAC,EAAE,WAAW,EAAE,CAAC,CAAC;IAC5B,CAAC;IAED;;OAEG;IACH,kBAAkB;QAChB,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,QAAQ,EAAE,gBAAgB,CAAC,CAAC;IACnG,CAAC;IAED;;OAEG;IACH,aAAa;QACX,OAAO,IAAI,CAAC,KAAK,CAAC,UAAU,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,QAAQ,EAAE,UAAU,CAAC,CAAC;IAC5F,CAAC;IAED;;OAEG;IACH,SAAS;QACP,OAAO,IAAI,CAAC,KAAK,CAAC,MAAM,IAAI,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,IAAI,EAAE,SAAS,CAAC,CAAC;IACnF,CAAC;IAED;;;OAGG;IACH,iBAAiB,CAAC,KAAU;QAC1B,MAAM,aAAa,GAAG,KAAK,CAAC,MAAM,CAAC,KAAK,CAAC;QACzC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;IAC3C,CAAC;IAED;;;;OAIG;IACH,cAAc;QACZ,IAAI,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,EAAE,CAAC;YAC9B,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC;QACnC,CAAC;QAED,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,EAAE,CAAC;YAC5C,qBAAqB;YACrB,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC1C,OAAO,IAAI,CAAC,kBAAkB,EAAE,CAAC;YACnC,CAAC;YACD,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBAC1C,MAAM,MAAM,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,CAAC;gBACzF,OAAO,MAAM,EAAE,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC;YACtD,CAAC;YACD,MAAM,YAAY,GAAG,IAAI,CAAC,WAAW,CAAC,OAAO,CAAC,SAAS,EAAE,UAAU,EAAE,eAAe,CAAC,CAAC;YACtF,OAAO,GAAG,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,MAAM,IAAI,YAAY,EAAE,CAAC;QAC9D,CAAC;QAED,mBAAmB;QACnB,MAAM,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,KAAK,KAAK,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC,CAAC;QAC9F,OAAO,cAAc,EAAE,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC;IAC3D,CAAC;+GA9EU,wBAAwB;mGAAxB,wBAAwB,6JAxEzB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCT,u/MAvCS,YAAY,gQAAE,SAAS,oPAAE,OAAO,2JAAE,SAAS,uTAAE,eAAe,6FAAE,QAAQ;;4FAyErE,wBAAwB;kBA5EpC,SAAS;+BACE,sBAAsB,cACpB,IAAI,WACP,CAAC,YAAY,EAAE,SAAS,EAAE,OAAO,EAAE,SAAS,EAAE,eAAe,EAAE,QAAQ,CAAC,YACvE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsCT;wDAwCD,KAAK;sBADJ,KAAK;gBAQN,eAAe;sBADd,MAAM","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, EventEmitter, inject, Input, Output } from '@angular/core';\nimport { IonButton, IonIcon, IonLabel, IonSelect, IonSelectOption } from '@ionic/angular/standalone';\nimport { addIcons } from 'ionicons';\nimport { chevronDown } from 'ionicons/icons';\nimport { LangService } from '../../../services/lang-provider/lang-provider.service';\nimport { PopoverSelectorMetadata } from './types';\n\n@Component({\n  selector: 'val-popover-selector',\n  standalone: true,\n  imports: [CommonModule, IonButton, IonIcon, IonSelect, IonSelectOption, IonLabel],\n  template: `\n    <div class=\"popover-selector-container\">\n      <ion-label *ngIf=\"props.label\" class=\"selector-label\">{{ props.label }}</ion-label>\n      <ion-select\n        [value]=\"props.selectedValue\"\n        [placeholder]=\"getPlaceholderText()\"\n        [interface]=\"props.interface || 'popover'\"\n        [multiple]=\"props.multiple || false\"\n        [disabled]=\"props.disabled || false\"\n        [cancelText]=\"getCancelText()\"\n        [okText]=\"getOkText()\"\n        (ionChange)=\"onSelectionChange($event)\"\n        class=\"popover-selector-select\"\n      >\n        <ion-button\n          slot=\"trigger\"\n          [fill]=\"props.fill || 'clear'\"\n          [size]=\"props.size || 'default'\"\n          [shape]=\"'round'\"\n          [color]=\"props.color || 'medium'\"\n          [disabled]=\"props.disabled || false\"\n          class=\"selector-trigger\"\n        >\n          <ion-icon *ngIf=\"props.icon\" [name]=\"props.icon\" slot=\"start\"> </ion-icon>\n\n          <span class=\"trigger-text\">{{ getDisplayText() }}</span>\n\n          <ion-icon name=\"chevron-down-outline\" slot=\"end\" class=\"chevron-icon\"> </ion-icon>\n        </ion-button>\n\n        <ion-select-option *ngFor=\"let option of props.options\" [value]=\"option.value\" [disabled]=\"option.disabled\">\n          <div class=\"option-content\">\n            <ion-icon *ngIf=\"option.icon\" [name]=\"option.icon\" class=\"option-icon\"> </ion-icon>\n            <span>{{ option.label }}</span>\n          </div>\n        </ion-select-option>\n      </ion-select>\n    </div>\n  `,\n  styleUrls: ['./popover-selector.component.scss'],\n})\n/**\n * val-popover-selector\n *\n * A generic reusable popover selector component that can be used for any type of selection.\n * Provides a button trigger that opens a popover with selectable options.\n *\n * @example\n * // Basic usage\n * <val-popover-selector\n *   [props]=\"selectorProps\"\n *   (selectionChange)=\"onSelectionChange($event)\">\n * </val-popover-selector>\n *\n * @example\n * // With icon and custom styling\n * const selectorProps: PopoverSelectorMetadata = {\n *   options: [\n *     { value: 'option1', label: 'Option 1', icon: 'star' },\n *     { value: 'option2', label: 'Option 2', icon: 'heart' }\n *   ],\n *   selectedValue: 'option1',\n *   label: 'Choose option',\n *   icon: 'settings',\n *   color: 'primary',\n *   size: 'large',\n *   fill: 'solid'\n * };\n *\n * @input props: PopoverSelectorMetadata - Configuration for the selector\n * @output selectionChange: EventEmitter<string | string[]> - Emitted when selection changes\n */\nexport class PopoverSelectorComponent {\n  /**\n   * Popover selector configuration object.\n   * @type {PopoverSelectorMetadata}\n   */\n  @Input()\n  props: PopoverSelectorMetadata;\n\n  /**\n   * Event emitted when the selection changes.\n   * Emits the selected value(s).\n   */\n  @Output()\n  selectionChange = new EventEmitter<string | string[]>();\n\n  private langService = inject(LangService);\n\n  constructor() {\n    // Register required icons\n    addIcons({ chevronDown });\n  }\n\n  /**\n   * Get reactive placeholder text.\n   */\n  getPlaceholderText(): string {\n    return this.props.placeholder || this.langService.getText('_global', 'select', 'Seleccionar...');\n  }\n\n  /**\n   * Get reactive cancel text.\n   */\n  getCancelText(): string {\n    return this.props.cancelText || this.langService.getText('_global', 'cancel', 'Cancelar');\n  }\n\n  /**\n   * Get reactive ok text.\n   */\n  getOkText(): string {\n    return this.props.okText || this.langService.getText('_global', 'ok', 'Aceptar');\n  }\n\n  /**\n   * Handle selection change from the ion-select.\n   * @param event - The ion-select change event\n   */\n  onSelectionChange(event: any) {\n    const selectedValue = event.detail.value;\n    this.selectionChange.emit(selectedValue);\n  }\n\n  /**\n   * Get the display text for the trigger button.\n   * Shows the selected option's label or placeholder.\n   * @returns The display text\n   */\n  getDisplayText(): string {\n    if (!this.props.selectedValue) {\n      return this.getPlaceholderText();\n    }\n\n    if (Array.isArray(this.props.selectedValue)) {\n      // Multiple selection\n      if (this.props.selectedValue.length === 0) {\n        return this.getPlaceholderText();\n      }\n      if (this.props.selectedValue.length === 1) {\n        const option = this.props.options.find(opt => opt.value === this.props.selectedValue[0]);\n        return option?.label || this.props.selectedValue[0];\n      }\n      const selectedText = this.langService.getText('_global', 'selected', 'seleccionados');\n      return `${this.props.selectedValue.length} ${selectedText}`;\n    }\n\n    // Single selection\n    const selectedOption = this.props.options.find(opt => opt.value === this.props.selectedValue);\n    return selectedOption?.label || this.props.selectedValue;\n  }\n}\n"]}