valtech-components 2.0.396 → 2.0.398

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1,406 +0,0 @@
1
- import { CommonModule } from '@angular/common';
2
- import { ChangeDetectorRef, Component, forwardRef, inject, Input, ViewChild } 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
- this.changeDetector = inject(ChangeDetectorRef);
19
- // Classic Angular properties
20
- this.isOpen = false;
21
- this.searchTerm = '';
22
- this.selectedValues = [];
23
- this.displayValue = '';
24
- this.filteredOptions = [];
25
- this.searchPlaceholder = '';
26
- // ControlValueAccessor
27
- this.onChange = (_value) => { };
28
- this.onTouched = () => { };
29
- this.placeholder = this.langService.getText('_global', 'selectOptions', 'Seleccione opciones');
30
- this.searchPlaceholder = this.langService.getText('_global', 'search', 'Buscar');
31
- // Close dropdown when clicking outside
32
- document.addEventListener('click', this.handleClickOutside.bind(this));
33
- }
34
- ngOnInit() {
35
- this.applyDefaultValue();
36
- this.initializeOptions();
37
- this.syncSelectedValues();
38
- this.updateDisplayValue();
39
- }
40
- ngOnDestroy() {
41
- document.removeEventListener('click', this.handleClickOutside.bind(this));
42
- }
43
- ngOnChanges(changes) {
44
- if (changes['props'] && this.props) {
45
- this.initializeOptions();
46
- this.syncSelectedValues();
47
- this.updateDisplayValue();
48
- this.changeDetector.detectChanges();
49
- }
50
- }
51
- // ControlValueAccessor implementation
52
- writeValue(value) {
53
- const valueArray = this.parseValue(value);
54
- this.selectedValues = valueArray;
55
- this.updateDisplayValue();
56
- }
57
- registerOnChange(fn) {
58
- this.onChange = fn;
59
- }
60
- registerOnTouched(fn) {
61
- this.onTouched = fn;
62
- }
63
- setDisabledState(_isDisabled) {
64
- // Handle disabled state if needed
65
- }
66
- // Component methods
67
- toggleDropdown(event) {
68
- event.stopPropagation();
69
- this.isOpen = !this.isOpen;
70
- if (this.isOpen) {
71
- this.onTouched();
72
- // Focus search bar when opening
73
- setTimeout(() => {
74
- const searchbar = this.dropdownRef?.nativeElement?.querySelector('ion-searchbar');
75
- if (searchbar) {
76
- searchbar.setFocus();
77
- }
78
- }, 100);
79
- }
80
- }
81
- onSearch(event) {
82
- this.searchTerm = event.detail.value || '';
83
- this.filterOptions();
84
- }
85
- toggleOption(option) {
86
- const value = String(option[this.valueProperty]);
87
- if (this.isSelected(option)) {
88
- // Remove from selection
89
- this.selectedValues = this.selectedValues.filter(v => v !== value);
90
- }
91
- else {
92
- // Add to selection
93
- this.selectedValues = [...this.selectedValues, value];
94
- }
95
- this.updateDisplayValue();
96
- this.emitValue();
97
- }
98
- selectAll() {
99
- const allValues = this.filteredOptions.map(option => String(option[this.valueProperty]));
100
- // Add only new values to avoid duplicates
101
- const newValues = allValues.filter(value => !this.selectedValues.includes(value));
102
- this.selectedValues = [...this.selectedValues, ...newValues];
103
- this.updateDisplayValue();
104
- this.emitValue();
105
- }
106
- clearAll() {
107
- this.selectedValues = [];
108
- this.updateDisplayValue();
109
- this.emitValue();
110
- }
111
- isSelected(option) {
112
- return this.selectedValues.includes(String(option[this.valueProperty]));
113
- }
114
- trackByFn(_index, option) {
115
- return option[this.valueProperty];
116
- }
117
- handleClickOutside(event) {
118
- if (this.isOpen &&
119
- !this.mainInputRef?.nativeElement?.contains(event.target) &&
120
- !this.dropdownRef?.nativeElement?.contains(event.target)) {
121
- this.isOpen = false;
122
- this.searchTerm = '';
123
- this.initializeOptions();
124
- }
125
- }
126
- parseValue(value) {
127
- if (!value)
128
- return [];
129
- if (Array.isArray(value))
130
- return value.map(v => String(v));
131
- if (typeof value === 'string') {
132
- return value.split(',').map(v => v.trim()).filter(v => v !== '');
133
- }
134
- return [String(value)];
135
- }
136
- emitValue() {
137
- const value = this.selectedValues.join(',');
138
- this.onChange(value);
139
- if (this.props?.control) {
140
- this.props.control.setValue(value);
141
- this.props.control.markAsDirty();
142
- this.props.control.markAsTouched();
143
- }
144
- }
145
- getOptionByValue(value) {
146
- return this.filteredOptions.find(option => String(option[this.valueProperty]) === String(value));
147
- }
148
- applyDefaultValue() {
149
- if (this.props) {
150
- applyDefaultValueToControl(this.props);
151
- }
152
- }
153
- initializeOptions() {
154
- this.filteredOptions = this.props?.options || [];
155
- }
156
- filterOptions() {
157
- if (!this.searchTerm) {
158
- this.initializeOptions();
159
- return;
160
- }
161
- const search = replaceSpecialChars(this.searchTerm.toLowerCase());
162
- this.filteredOptions = (this.props?.options || []).filter(option => {
163
- const label = option[this.labelProperty]
164
- ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())
165
- : '';
166
- const value = option[this.valueProperty]
167
- ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())
168
- : '';
169
- return label.includes(search) || value.includes(search);
170
- });
171
- }
172
- updateDisplayValue() {
173
- if (this.selectedValues.length === 0) {
174
- this.displayValue = '';
175
- return;
176
- }
177
- if (this.selectedValues.length === 1) {
178
- const option = this.getOptionByValue(this.selectedValues[0]);
179
- this.displayValue = option ? option[this.labelProperty] : '';
180
- }
181
- else {
182
- this.displayValue = `${this.selectedValues.length} elementos seleccionados`;
183
- }
184
- }
185
- syncSelectedValues() {
186
- if (this.props?.control?.value) {
187
- const valueArray = this.parseValue(this.props.control.value);
188
- this.selectedValues = valueArray;
189
- }
190
- }
191
- static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiSelectSimpleComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
192
- 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: [
193
- {
194
- provide: NG_VALUE_ACCESSOR,
195
- useExisting: forwardRef(() => MultiSelectSimpleComponent),
196
- multi: true
197
- }
198
- ], viewQueries: [{ propertyName: "dropdownRef", first: true, predicate: ["dropdown"], descendants: true }, { propertyName: "mainInputRef", first: true, predicate: ["mainInput"], descendants: true }], usesOnChanges: true, ngImport: i0, template: `
199
- <div class="multi-select-container" (click)="toggleDropdown($event)">
200
- <!-- Main input display -->
201
- <ion-input
202
- #mainInput
203
- type="text"
204
- [value]="displayValue"
205
- [placeholder]="props?.placeholder || placeholder"
206
- readonly
207
- class="main-input"
208
- [class.is-open]="isOpen"
209
- />
210
-
211
- <!-- Dropdown icon -->
212
- <ion-icon
213
- name="chevron-down-outline"
214
- class="dropdown-icon"
215
- [class.rotated]="isOpen"
216
- ></ion-icon>
217
-
218
- <!-- Hidden input for form control -->
219
- <ion-input
220
- style="position: absolute; opacity: 0; pointer-events: none;"
221
- [formControl]="props?.control"
222
- type="hidden"
223
- />
224
- </div>
225
-
226
- <!-- Dropdown overlay -->
227
- <div
228
- class="dropdown-overlay"
229
- [class.visible]="isOpen"
230
- #dropdown
231
- >
232
- <!-- Search bar -->
233
- <div class="search-container">
234
- <ion-searchbar
235
- #searchbar
236
- [placeholder]="searchPlaceholder"
237
- (ionInput)="onSearch($event)"
238
- [value]="searchTerm"
239
- show-clear-button="focus"
240
- ></ion-searchbar>
241
- </div>
242
-
243
- <!-- Action buttons -->
244
- <div class="actions-container">
245
- <ion-button
246
- fill="clear"
247
- size="small"
248
- (click)="selectAll()"
249
- [disabled]="filteredOptions.length === 0"
250
- >
251
- Seleccionar todos
252
- </ion-button>
253
- <ion-button
254
- fill="clear"
255
- size="small"
256
- color="medium"
257
- (click)="clearAll()"
258
- [disabled]="selectedValues.length === 0"
259
- >
260
- Limpiar
261
- </ion-button>
262
- </div>
263
-
264
- <!-- Options list -->
265
- <div class="options-container">
266
- <ion-list class="options-list">
267
- <ion-item
268
- *ngFor="let option of filteredOptions; trackBy: trackByFn"
269
- button
270
- (click)="toggleOption(option)"
271
- class="option-item"
272
- >
273
- <ion-checkbox
274
- slot="start"
275
- [checked]="isSelected(option)"
276
- ></ion-checkbox>
277
- <ion-label>{{ option[labelProperty] }}</ion-label>
278
- </ion-item>
279
-
280
- <!-- No results message -->
281
- <ion-item *ngIf="filteredOptions.length === 0" class="no-results">
282
- <ion-label color="medium">
283
- {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
284
- </ion-label>
285
- </ion-item>
286
- </ion-list>
287
- </div>
288
- </div>
289
- `, 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"] }] }); }
290
- }
291
- i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: MultiSelectSimpleComponent, decorators: [{
292
- type: Component,
293
- args: [{ selector: 'val-multi-select-simple', standalone: true, imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule], providers: [
294
- {
295
- provide: NG_VALUE_ACCESSOR,
296
- useExisting: forwardRef(() => MultiSelectSimpleComponent),
297
- multi: true
298
- }
299
- ], template: `
300
- <div class="multi-select-container" (click)="toggleDropdown($event)">
301
- <!-- Main input display -->
302
- <ion-input
303
- #mainInput
304
- type="text"
305
- [value]="displayValue"
306
- [placeholder]="props?.placeholder || placeholder"
307
- readonly
308
- class="main-input"
309
- [class.is-open]="isOpen"
310
- />
311
-
312
- <!-- Dropdown icon -->
313
- <ion-icon
314
- name="chevron-down-outline"
315
- class="dropdown-icon"
316
- [class.rotated]="isOpen"
317
- ></ion-icon>
318
-
319
- <!-- Hidden input for form control -->
320
- <ion-input
321
- style="position: absolute; opacity: 0; pointer-events: none;"
322
- [formControl]="props?.control"
323
- type="hidden"
324
- />
325
- </div>
326
-
327
- <!-- Dropdown overlay -->
328
- <div
329
- class="dropdown-overlay"
330
- [class.visible]="isOpen"
331
- #dropdown
332
- >
333
- <!-- Search bar -->
334
- <div class="search-container">
335
- <ion-searchbar
336
- #searchbar
337
- [placeholder]="searchPlaceholder"
338
- (ionInput)="onSearch($event)"
339
- [value]="searchTerm"
340
- show-clear-button="focus"
341
- ></ion-searchbar>
342
- </div>
343
-
344
- <!-- Action buttons -->
345
- <div class="actions-container">
346
- <ion-button
347
- fill="clear"
348
- size="small"
349
- (click)="selectAll()"
350
- [disabled]="filteredOptions.length === 0"
351
- >
352
- Seleccionar todos
353
- </ion-button>
354
- <ion-button
355
- fill="clear"
356
- size="small"
357
- color="medium"
358
- (click)="clearAll()"
359
- [disabled]="selectedValues.length === 0"
360
- >
361
- Limpiar
362
- </ion-button>
363
- </div>
364
-
365
- <!-- Options list -->
366
- <div class="options-container">
367
- <ion-list class="options-list">
368
- <ion-item
369
- *ngFor="let option of filteredOptions; trackBy: trackByFn"
370
- button
371
- (click)="toggleOption(option)"
372
- class="option-item"
373
- >
374
- <ion-checkbox
375
- slot="start"
376
- [checked]="isSelected(option)"
377
- ></ion-checkbox>
378
- <ion-label>{{ option[labelProperty] }}</ion-label>
379
- </ion-item>
380
-
381
- <!-- No results message -->
382
- <ion-item *ngIf="filteredOptions.length === 0" class="no-results">
383
- <ion-label color="medium">
384
- {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}
385
- </ion-label>
386
- </ion-item>
387
- </ion-list>
388
- </div>
389
- </div>
390
- `, 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"] }]
391
- }], ctorParameters: () => [], propDecorators: { dropdownRef: [{
392
- type: ViewChild,
393
- args: ['dropdown']
394
- }], mainInputRef: [{
395
- type: ViewChild,
396
- args: ['mainInput']
397
- }], props: [{
398
- type: Input
399
- }], labelProperty: [{
400
- type: Input
401
- }], valueProperty: [{
402
- type: Input
403
- }], placeholder: [{
404
- type: Input
405
- }] } });
406
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"multi-select-simple.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/molecules/multi-select-simple/multi-select-simple.component.ts"],"names":[],"mappings":"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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-simple',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, ReactiveFormsModule],\n  providers: [\n    {\n      provide: NG_VALUE_ACCESSOR,\n      useExisting: forwardRef(() => MultiSelectSimpleComponent),\n      multi: true\n    }\n  ],\n  template: `\n    <div class=\"multi-select-container\" (click)=\"toggleDropdown($event)\">\n      <!-- Main input display -->\n      <ion-input\n        #mainInput\n        type=\"text\"\n        [value]=\"displayValue\"\n        [placeholder]=\"props?.placeholder || placeholder\"\n        readonly\n        class=\"main-input\"\n        [class.is-open]=\"isOpen\"\n      />\n      \n      <!-- Dropdown icon -->\n      <ion-icon \n        name=\"chevron-down-outline\" \n        class=\"dropdown-icon\" \n        [class.rotated]=\"isOpen\"\n      ></ion-icon>\n      \n      <!-- Hidden input for form control -->\n      <ion-input \n        style=\"position: absolute; opacity: 0; pointer-events: none;\" \n        [formControl]=\"props?.control\" \n        type=\"hidden\"\n      />\n    </div>\n\n    <!-- Dropdown overlay -->\n    <div \n      class=\"dropdown-overlay\" \n      [class.visible]=\"isOpen\"\n      #dropdown\n    >\n      <!-- Search bar -->\n      <div class=\"search-container\">\n        <ion-searchbar\n          #searchbar\n          [placeholder]=\"searchPlaceholder\"\n          (ionInput)=\"onSearch($event)\"\n          [value]=\"searchTerm\"\n          show-clear-button=\"focus\"\n        ></ion-searchbar>\n      </div>\n      \n      <!-- Action buttons -->\n      <div class=\"actions-container\">\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          (click)=\"selectAll()\"\n          [disabled]=\"filteredOptions.length === 0\"\n        >\n          Seleccionar todos\n        </ion-button>\n        <ion-button \n          fill=\"clear\" \n          size=\"small\" \n          color=\"medium\" \n          (click)=\"clearAll()\"\n          [disabled]=\"selectedValues.length === 0\"\n        >\n          Limpiar\n        </ion-button>\n      </div>\n      \n      <!-- Options list -->\n      <div class=\"options-container\">\n        <ion-list class=\"options-list\">\n          <ion-item \n            *ngFor=\"let option of filteredOptions; trackBy: trackByFn\" \n            button \n            (click)=\"toggleOption(option)\"\n            class=\"option-item\"\n          >\n            <ion-checkbox\n              slot=\"start\"\n              [checked]=\"isSelected(option)\"\n            ></ion-checkbox>\n            <ion-label>{{ option[labelProperty] }}</ion-label>\n          </ion-item>\n          \n          <!-- No results message -->\n          <ion-item *ngIf=\"filteredOptions.length === 0\" class=\"no-results\">\n            <ion-label color=\"medium\">\n              {{ searchTerm ? 'No se encontraron resultados' : 'No hay opciones disponibles' }}\n            </ion-label>\n          </ion-item>\n        </ion-list>\n      </div>\n    </div>\n  `,\n  styleUrls: ['./multi-select-simple.component.scss']\n})\nexport class MultiSelectSimpleComponent implements OnInit, OnDestroy, OnChanges, ControlValueAccessor {\n  @ViewChild('dropdown') dropdownRef!: ElementRef;\n  @ViewChild('mainInput') mainInputRef!: ElementRef;\n\n  @Input() props!: InputMetadata;\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() placeholder: string = '';\n\n  private langService = inject(LangService);\n  private changeDetector = inject(ChangeDetectorRef);\n  \n  // Classic Angular properties\n  isOpen = false;\n  searchTerm = '';\n  selectedValues: string[] = [];\n  displayValue = '';\n  filteredOptions: InputOption[] = [];\n  searchPlaceholder: string = '';\n\n  // ControlValueAccessor\n  private onChange = (_value: any) => {};\n  private onTouched = () => {};\n\n  constructor() {\n    this.placeholder = this.langService.getText('_global', 'selectOptions', 'Seleccione opciones');\n    this.searchPlaceholder = this.langService.getText('_global', 'search', 'Buscar');\n    \n    // Close dropdown when clicking outside\n    document.addEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnInit() {\n    this.applyDefaultValue();\n    this.initializeOptions();\n    this.syncSelectedValues();\n    this.updateDisplayValue();\n  }\n\n  ngOnDestroy() {\n    document.removeEventListener('click', this.handleClickOutside.bind(this));\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    if (changes['props'] && this.props) {\n      this.initializeOptions();\n      this.syncSelectedValues();\n      this.updateDisplayValue();\n      this.changeDetector.detectChanges();\n    }\n  }\n\n  // ControlValueAccessor implementation\n  writeValue(value: any): void {\n    const valueArray = this.parseValue(value);\n    this.selectedValues = valueArray;\n    this.updateDisplayValue();\n  }\n\n  registerOnChange(fn: any): void {\n    this.onChange = fn;\n  }\n\n  registerOnTouched(fn: any): void {\n    this.onTouched = fn;\n  }\n\n  setDisabledState(_isDisabled: boolean): void {\n    // Handle disabled state if needed\n  }\n\n  // Component methods\n  toggleDropdown(event: Event): void {\n    event.stopPropagation();\n    this.isOpen = !this.isOpen;\n    \n    if (this.isOpen) {\n      this.onTouched();\n      // Focus search bar when opening\n      setTimeout(() => {\n        const searchbar = this.dropdownRef?.nativeElement?.querySelector('ion-searchbar');\n        if (searchbar) {\n          searchbar.setFocus();\n        }\n      }, 100);\n    }\n  }\n\n  onSearch(event: any): void {\n    this.searchTerm = event.detail.value || '';\n    this.filterOptions();\n  }\n\n  toggleOption(option: InputOption): void {\n    const value = String(option[this.valueProperty]);\n    \n    if (this.isSelected(option)) {\n      // Remove from selection\n      this.selectedValues = this.selectedValues.filter(v => v !== value);\n    } else {\n      // Add to selection\n      this.selectedValues = [...this.selectedValues, value];\n    }\n    \n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  selectAll(): void {\n    const allValues = this.filteredOptions.map(option => String(option[this.valueProperty]));\n    \n    // Add only new values to avoid duplicates\n    const newValues = allValues.filter(value => !this.selectedValues.includes(value));\n    this.selectedValues = [...this.selectedValues, ...newValues];\n    \n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  clearAll(): void {\n    this.selectedValues = [];\n    this.updateDisplayValue();\n    this.emitValue();\n  }\n\n  isSelected(option: InputOption): boolean {\n    return this.selectedValues.includes(String(option[this.valueProperty]));\n  }\n\n  trackByFn(_index: number, option: InputOption): any {\n    return option[this.valueProperty];\n  }\n\n  private handleClickOutside(event: Event): void {\n    if (this.isOpen && \n        !this.mainInputRef?.nativeElement?.contains(event.target) &&\n        !this.dropdownRef?.nativeElement?.contains(event.target)) {\n      this.isOpen = false;\n      this.searchTerm = '';\n      this.initializeOptions();\n    }\n  }\n\n  private parseValue(value: any): string[] {\n    if (!value) return [];\n    if (Array.isArray(value)) return value.map(v => String(v));\n    if (typeof value === 'string') {\n      return value.split(',').map(v => v.trim()).filter(v => v !== '');\n    }\n    return [String(value)];\n  }\n\n  private emitValue(): void {\n    const value = this.selectedValues.join(',');\n    this.onChange(value);\n    \n    if (this.props?.control) {\n      this.props.control.setValue(value);\n      this.props.control.markAsDirty();\n      this.props.control.markAsTouched();\n    }\n  }\n\n  private getOptionByValue(value: string): InputOption | undefined {\n    return this.filteredOptions.find(option => \n      String(option[this.valueProperty]) === String(value)\n    );\n  }\n\n  private applyDefaultValue(): void {\n    if (this.props) {\n      applyDefaultValueToControl(this.props);\n    }\n  }\n\n  private initializeOptions(): void {\n    this.filteredOptions = this.props?.options || [];\n  }\n  \n  private filterOptions(): void {\n    if (!this.searchTerm) {\n      this.initializeOptions();\n      return;\n    }\n    \n    const search = replaceSpecialChars(this.searchTerm.toLowerCase());\n    this.filteredOptions = (this.props?.options || []).filter(option => {\n      const label = option[this.labelProperty]\n        ? replaceSpecialChars(String(option[this.labelProperty]).toLowerCase())\n        : '';\n      const value = option[this.valueProperty]\n        ? replaceSpecialChars(String(option[this.valueProperty]).toLowerCase())\n        : '';\n      return label.includes(search) || value.includes(search);\n    });\n  }\n  \n  private updateDisplayValue(): void {\n    if (this.selectedValues.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.selectedValues.length === 1) {\n      const option = this.getOptionByValue(this.selectedValues[0]);\n      this.displayValue = option ? option[this.labelProperty] : '';\n    } else {\n      this.displayValue = `${this.selectedValues.length} elementos seleccionados`;\n    }\n  }\n\n  private syncSelectedValues(): void {\n    if (this.props?.control?.value) {\n      const valueArray = this.parseValue(this.props.control.value);\n      this.selectedValues = valueArray;\n    }\n  }\n}"]}
@@ -1,48 +0,0 @@
1
- import { ElementRef, OnInit, OnDestroy, OnChanges, SimpleChanges } from '@angular/core';
2
- import { ControlValueAccessor } from '@angular/forms';
3
- import { InputMetadata, InputOption } from '../../types';
4
- import * as i0 from "@angular/core";
5
- export declare class MultiSelectSimpleComponent implements OnInit, OnDestroy, OnChanges, ControlValueAccessor {
6
- dropdownRef: ElementRef;
7
- mainInputRef: ElementRef;
8
- props: InputMetadata;
9
- labelProperty: string;
10
- valueProperty: string;
11
- placeholder: string;
12
- private langService;
13
- private changeDetector;
14
- isOpen: boolean;
15
- searchTerm: string;
16
- selectedValues: string[];
17
- displayValue: string;
18
- filteredOptions: InputOption[];
19
- searchPlaceholder: string;
20
- private onChange;
21
- private onTouched;
22
- constructor();
23
- ngOnInit(): void;
24
- ngOnDestroy(): void;
25
- ngOnChanges(changes: SimpleChanges): void;
26
- writeValue(value: any): void;
27
- registerOnChange(fn: any): void;
28
- registerOnTouched(fn: any): void;
29
- setDisabledState(_isDisabled: boolean): void;
30
- toggleDropdown(event: Event): void;
31
- onSearch(event: any): void;
32
- toggleOption(option: InputOption): void;
33
- selectAll(): void;
34
- clearAll(): void;
35
- isSelected(option: InputOption): boolean;
36
- trackByFn(_index: number, option: InputOption): any;
37
- private handleClickOutside;
38
- private parseValue;
39
- private emitValue;
40
- private getOptionByValue;
41
- private applyDefaultValue;
42
- private initializeOptions;
43
- private filterOptions;
44
- private updateDisplayValue;
45
- private syncSelectedValues;
46
- static ɵfac: i0.ɵɵFactoryDeclaration<MultiSelectSimpleComponent, never>;
47
- static ɵcmp: i0.ɵɵComponentDeclaration<MultiSelectSimpleComponent, "val-multi-select-simple", never, { "props": { "alias": "props"; "required": false; }; "labelProperty": { "alias": "labelProperty"; "required": false; }; "valueProperty": { "alias": "valueProperty"; "required": false; }; "placeholder": { "alias": "placeholder"; "required": false; }; }, {}, never, never, true, never>;
48
- }