valtech-components 2.0.194 → 2.0.197
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/pin-input/pin-input.component.mjs +4 -4
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +25 -25
- package/esm2022/lib/components/organisms/item-list/item-list.component.mjs +3 -3
- package/fesm2022/valtech-components.mjs +29 -29
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/package.json +1 -1
- package/src/lib/components/styles/overrides.scss +16 -2
|
@@ -17,9 +17,9 @@ export class PinInputComponent {
|
|
|
17
17
|
this.codeLength = 5;
|
|
18
18
|
this.otpInputConfig = {
|
|
19
19
|
inputStyles: {
|
|
20
|
-
'font-size': '
|
|
21
|
-
width: '
|
|
22
|
-
height: '
|
|
20
|
+
'font-size': '32px',
|
|
21
|
+
width: '58px',
|
|
22
|
+
height: '60px',
|
|
23
23
|
},
|
|
24
24
|
inputClass: 'otp-input-box',
|
|
25
25
|
length: this.codeLength,
|
|
@@ -52,4 +52,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
52
52
|
}], props: [{
|
|
53
53
|
type: Input
|
|
54
54
|
}] } });
|
|
55
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGluLWlucHV0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3ZhbHRlY2gtY29tcG9uZW50cy9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3Bpbi1pbnB1dC9waW4taW5wdXQuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNwRSxPQUFPLEVBQUUsbUJBQW1CLEVBQW9CLGdCQUFnQixFQUFFLE1BQU0sY0FBYyxDQUFDOzs7QUFjdkY7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQXNCNUI7UUFaQSxlQUFVLEdBQUcsQ0FBQyxDQUFDO1FBQ2YsbUJBQWMsR0FBcUI7WUFDakMsV0FBVyxFQUFFO2dCQUNYLFdBQVcsRUFBRSxNQUFNO2dCQUNuQixLQUFLLEVBQUUsTUFBTTtnQkFDYixNQUFNLEVBQUUsTUFBTTthQUNmO1lBQ0QsVUFBVSxFQUFFLGVBQWU7WUFDM0IsTUFBTSxFQUFFLElBQUksQ0FBQyxVQUFVO1lBQ3ZCLGdCQUFnQixFQUFFLElBQUk7U0FDdkIsQ0FBQztJQUVhLENBQUM7SUFFaEIsUUFBUSxLQUFJLENBQUM7SUFFYixLQUFLO1FBQ0gsSUFBSSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDNUIsQ0FBQztJQUNILENBQUM7K0dBOUJVLGlCQUFpQjttR0FBakIsaUJBQWlCLDhJQUNqQixtQkFBbUIsZ0RBbEJwQjs7OztHQUlULGlUQUxTLGdCQUFnQjs7NEZBa0JmLGlCQUFpQjtrQkFyQjdCLFNBQVM7K0JBQ0UsZUFBZSxjQUNiLElBQUksV0FDUCxDQUFDLGdCQUFnQixDQUFDLFlBQ2pCOzs7O0dBSVQ7d0RBY2tELE9BQU87c0JBQXpELFNBQVM7dUJBQUMsbUJBQW1CLEVBQUUsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFO2dCQU94QyxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkluaXQsIFZpZXdDaGlsZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTmdPdHBJbnB1dENvbXBvbmVudCwgTmdPdHBJbnB1dENvbmZpZywgTmdPdHBJbnB1dE1vZHVsZSB9IGZyb20gJ25nLW90cC1pbnB1dCc7XG5pbXBvcnQgeyBJbnB1dE1ldGFkYXRhIH0gZnJvbSAnLi4vLi4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtcGluLWlucHV0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW05nT3RwSW5wdXRNb2R1bGVdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJvdHBcIj5cbiAgICAgIDxuZy1vdHAtaW5wdXQgW2Zvcm1DdHJsXT1cInByb3BzLmNvbnRyb2xcIiBbY29uZmlnXT1cIm90cElucHV0Q29uZmlnXCI+PC9uZy1vdHAtaW5wdXQ+XG4gICAgPC9kaXY+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL3Bpbi1pbnB1dC5jb21wb25lbnQuc2NzcyddLFxufSlcbi8qKlxuICogdmFsLXBpbi1pbnB1dFxuICpcbiAqIEEgUElOIGNvZGUgaW5wdXQgdXNpbmcgbmctb3RwLWlucHV0LCBpbnRlZ3JhdGVkIHdpdGggQW5ndWxhciBmb3Jtcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogPHZhbC1waW4taW5wdXQgW3Byb3BzXT1cInsgY29udHJvbDogbXlDb250cm9sIH1cIj48L3ZhbC1waW4taW5wdXQ+
|
|
55
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoicGluLWlucHV0LmNvbXBvbmVudC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3ZhbHRlY2gtY29tcG9uZW50cy9zcmMvbGliL2NvbXBvbmVudHMvbW9sZWN1bGVzL3Bpbi1pbnB1dC9waW4taW5wdXQuY29tcG9uZW50LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLFNBQVMsRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUNwRSxPQUFPLEVBQUUsbUJBQW1CLEVBQW9CLGdCQUFnQixFQUFFLE1BQU0sY0FBYyxDQUFDOzs7QUFjdkY7Ozs7Ozs7OztHQVNHO0FBQ0gsTUFBTSxPQUFPLGlCQUFpQjtJQXNCNUI7UUFaQSxlQUFVLEdBQUcsQ0FBQyxDQUFDO1FBQ2YsbUJBQWMsR0FBcUI7WUFDakMsV0FBVyxFQUFFO2dCQUNYLFdBQVcsRUFBRSxNQUFNO2dCQUNuQixLQUFLLEVBQUUsTUFBTTtnQkFDYixNQUFNLEVBQUUsTUFBTTthQUNmO1lBQ0QsVUFBVSxFQUFFLGVBQWU7WUFDM0IsTUFBTSxFQUFFLElBQUksQ0FBQyxVQUFVO1lBQ3ZCLGdCQUFnQixFQUFFLElBQUk7U0FDdkIsQ0FBQztJQUVhLENBQUM7SUFFaEIsUUFBUSxLQUFJLENBQUM7SUFFYixLQUFLO1FBQ0gsSUFBSSxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDakIsSUFBSSxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDNUIsQ0FBQztJQUNILENBQUM7K0dBOUJVLGlCQUFpQjttR0FBakIsaUJBQWlCLDhJQUNqQixtQkFBbUIsZ0RBbEJwQjs7OztHQUlULGlUQUxTLGdCQUFnQjs7NEZBa0JmLGlCQUFpQjtrQkFyQjdCLFNBQVM7K0JBQ0UsZUFBZSxjQUNiLElBQUksV0FDUCxDQUFDLGdCQUFnQixDQUFDLFlBQ2pCOzs7O0dBSVQ7d0RBY2tELE9BQU87c0JBQXpELFNBQVM7dUJBQUMsbUJBQW1CLEVBQUUsRUFBRSxNQUFNLEVBQUUsS0FBSyxFQUFFO2dCQU94QyxLQUFLO3NCQUFiLEtBQUsiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkluaXQsIFZpZXdDaGlsZCB9IGZyb20gJ0Bhbmd1bGFyL2NvcmUnO1xuaW1wb3J0IHsgTmdPdHBJbnB1dENvbXBvbmVudCwgTmdPdHBJbnB1dENvbmZpZywgTmdPdHBJbnB1dE1vZHVsZSB9IGZyb20gJ25nLW90cC1pbnB1dCc7XG5pbXBvcnQgeyBJbnB1dE1ldGFkYXRhIH0gZnJvbSAnLi4vLi4vdHlwZXMnO1xuXG5AQ29tcG9uZW50KHtcbiAgc2VsZWN0b3I6ICd2YWwtcGluLWlucHV0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW05nT3RwSW5wdXRNb2R1bGVdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJvdHBcIj5cbiAgICAgIDxuZy1vdHAtaW5wdXQgW2Zvcm1DdHJsXT1cInByb3BzLmNvbnRyb2xcIiBbY29uZmlnXT1cIm90cElucHV0Q29uZmlnXCI+PC9uZy1vdHAtaW5wdXQ+XG4gICAgPC9kaXY+XG4gIGAsXG4gIHN0eWxlVXJsczogWycuL3Bpbi1pbnB1dC5jb21wb25lbnQuc2NzcyddLFxufSlcbi8qKlxuICogdmFsLXBpbi1pbnB1dFxuICpcbiAqIEEgUElOIGNvZGUgaW5wdXQgdXNpbmcgbmctb3RwLWlucHV0LCBpbnRlZ3JhdGVkIHdpdGggQW5ndWxhciBmb3Jtcy5cbiAqXG4gKiBAZXhhbXBsZVxuICogPHZhbC1waW4taW5wdXQgW3Byb3BzXT1cInsgY29udHJvbDogbXlDb250cm9sIH1cIj48L3ZhbC1waW4taW5wdXQ+XG4gKlxuICogQGlucHV0IHByb3BzOiBJbnB1dE1ldGFkYXRhIC0gQ29uZmlndXJhdGlvbiBmb3IgdGhlIFBJTiBpbnB1dCAoZm9ybSBjb250cm9sLCBldGMuKVxuICovXG5leHBvcnQgY2xhc3MgUGluSW5wdXRDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICBAVmlld0NoaWxkKE5nT3RwSW5wdXRDb21wb25lbnQsIHsgc3RhdGljOiBmYWxzZSB9KSBwaW5Db2RlOiBOZ090cElucHV0Q29tcG9uZW50IHwgdW5kZWZpbmVkO1xuXG4gIC8qKlxuICAgKiBJbnB1dCBjb25maWd1cmF0aW9uIG9iamVjdC5cbiAgICogQHR5cGUge0lucHV0TWV0YWRhdGF9XG4gICAqIEBwcm9wZXJ0eSBjb250cm9sIC0gVGhlIEFuZ3VsYXIgRm9ybUNvbnRyb2wgZm9yIHRoZSBQSU4gaW5wdXQuXG4gICAqL1xuICBASW5wdXQoKSBwcm9wczogSW5wdXRNZXRhZGF0YTtcblxuICBjb2RlTGVuZ3RoID0gNTtcbiAgb3RwSW5wdXRDb25maWc6IE5nT3RwSW5wdXRDb25maWcgPSB7XG4gICAgaW5wdXRTdHlsZXM6IHtcbiAgICAgICdmb250LXNpemUnOiAnMzJweCcsXG4gICAgICB3aWR0aDogJzU4cHgnLFxuICAgICAgaGVpZ2h0OiAnNjBweCcsXG4gICAgfSxcbiAgICBpbnB1dENsYXNzOiAnb3RwLWlucHV0LWJveCcsXG4gICAgbGVuZ3RoOiB0aGlzLmNvZGVMZW5ndGgsXG4gICAgYWxsb3dOdW1iZXJzT25seTogdHJ1ZSxcbiAgfTtcblxuICBjb25zdHJ1Y3RvcigpIHt9XG5cbiAgbmdPbkluaXQoKSB7fVxuXG4gIHJlc2V0KCkge1xuICAgIGlmICh0aGlzLnBpbkNvZGUpIHtcbiAgICAgIHRoaXMucGluQ29kZS5zZXRWYWx1ZSgnJyk7XG4gICAgfVxuICB9XG59XG4iXX0=
|
|
@@ -113,12 +113,14 @@ export class SelectSearchComponent {
|
|
|
113
113
|
}
|
|
114
114
|
// Compara si dos arrays de opciones son iguales
|
|
115
115
|
areOptionsEqual(prevOptions, newOptions) {
|
|
116
|
-
|
|
116
|
+
// PERF: Use reference equality first for fast path
|
|
117
|
+
if (prevOptions === newOptions)
|
|
118
|
+
return true;
|
|
119
|
+
if (!prevOptions || !newOptions)
|
|
117
120
|
return prevOptions === newOptions;
|
|
118
|
-
|
|
119
|
-
if (prevOptions.length !== newOptions.length) {
|
|
121
|
+
if (prevOptions.length !== newOptions.length)
|
|
120
122
|
return false;
|
|
121
|
-
|
|
123
|
+
// Only compare valueProperty for equality
|
|
122
124
|
for (let i = 0; i < prevOptions.length; i++) {
|
|
123
125
|
if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
|
|
124
126
|
return false;
|
|
@@ -127,12 +129,8 @@ export class SelectSearchComponent {
|
|
|
127
129
|
return true;
|
|
128
130
|
}
|
|
129
131
|
initializeItems() {
|
|
130
|
-
|
|
131
|
-
|
|
132
|
-
}
|
|
133
|
-
else {
|
|
134
|
-
this.filteredItems = [];
|
|
135
|
-
}
|
|
132
|
+
// PERF: Avoid unnecessary array copies
|
|
133
|
+
this.filteredItems = this.props?.options || [];
|
|
136
134
|
}
|
|
137
135
|
syncControlValueWithSelectedItems() {
|
|
138
136
|
if (!this.props?.control) {
|
|
@@ -140,14 +138,14 @@ export class SelectSearchComponent {
|
|
|
140
138
|
return;
|
|
141
139
|
}
|
|
142
140
|
const controlValue = this.props.control.value;
|
|
143
|
-
// Si el control no tiene valor, limpiar seleccionados
|
|
144
141
|
if (controlValue === null || controlValue === undefined) {
|
|
145
142
|
this.selectedItems = [];
|
|
146
143
|
return;
|
|
147
144
|
}
|
|
148
|
-
//
|
|
145
|
+
// PERF: Use a Map for faster lookup if options are large
|
|
149
146
|
if (this.props.options && this.props.options.length > 0) {
|
|
150
|
-
const
|
|
147
|
+
const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));
|
|
148
|
+
const selectedOption = map.get(controlValue);
|
|
151
149
|
this.selectedItems = selectedOption ? [selectedOption] : [];
|
|
152
150
|
}
|
|
153
151
|
else {
|
|
@@ -155,28 +153,26 @@ export class SelectSearchComponent {
|
|
|
155
153
|
}
|
|
156
154
|
}
|
|
157
155
|
onFilter(event) {
|
|
158
|
-
//
|
|
156
|
+
// If no search term, show all options
|
|
159
157
|
if (!event || event.trim() === '') {
|
|
160
158
|
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
161
159
|
this.changeDetector.detectChanges();
|
|
162
160
|
return;
|
|
163
161
|
}
|
|
164
|
-
//
|
|
162
|
+
// If no options, nothing to filter
|
|
165
163
|
if (!this.props?.options || this.props.options.length === 0) {
|
|
166
164
|
this.filteredItems = [];
|
|
167
165
|
this.changeDetector.detectChanges();
|
|
168
166
|
return;
|
|
169
167
|
}
|
|
170
|
-
|
|
171
|
-
const
|
|
172
|
-
|
|
173
|
-
|
|
174
|
-
|
|
175
|
-
|
|
176
|
-
|
|
177
|
-
return values.some(value => value.includes(text));
|
|
168
|
+
// PERF: Avoid repeated replaceSpecialChars and toLowerCase for each option
|
|
169
|
+
const search = replaceSpecialChars(event.toLowerCase());
|
|
170
|
+
this.filteredItems = this.props.options.filter(element => {
|
|
171
|
+
// Only use labelProperty and valueProperty for filtering (faster)
|
|
172
|
+
const label = element[this.labelProperty] ? replaceSpecialChars(String(element[this.labelProperty]).toLowerCase()) : '';
|
|
173
|
+
const value = element[this.valueProperty] ? replaceSpecialChars(String(element[this.valueProperty]).toLowerCase()) : '';
|
|
174
|
+
return label.includes(search) || value.includes(search);
|
|
178
175
|
});
|
|
179
|
-
this.filteredItems = filteredResults;
|
|
180
176
|
this.changeDetector.detectChanges();
|
|
181
177
|
}
|
|
182
178
|
onFocus() {
|
|
@@ -213,6 +209,10 @@ export class SelectSearchComponent {
|
|
|
213
209
|
else {
|
|
214
210
|
this.selectedItems = [item];
|
|
215
211
|
this.cancelModal();
|
|
212
|
+
// Reset filter and show all options when closing modal
|
|
213
|
+
this.searchTerm = '';
|
|
214
|
+
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
215
|
+
this.changeDetector.detectChanges();
|
|
216
216
|
}
|
|
217
217
|
this.updateDisplayValue();
|
|
218
218
|
this.applyChanges();
|
|
@@ -361,4 +361,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
361
361
|
}], props: [{
|
|
362
362
|
type: Input
|
|
363
363
|
}] } });
|
|
364
|
-
//# 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  ChangeDetectorRef,\n  Component,\n  inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { Subscription } from 'rxjs';\nimport { IconService } from '../../../services/icons.service';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\n\n@Component({\n  selector: 'val-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, SearchbarComponent, ReactiveFormsModule],\n  template: `\n    <ion-input\n      type=\"text\"\n      [value]=\"displayValue\"\n      [placeholder]=\"props?.placeholder || placeholder\"\n      readonly\n      (mousedown)=\"preventDefaultBehavior($event)\"\n    />\n\n    <ion-input style=\"position: absolute;\" [formControl]=\"props.control\" type=\"hidden\"></ion-input>\n\n    <ion-modal #modal trigger=\"open-modal\" [initialBreakpoint]=\"0.75\" [breakpoints]=\"[0, 0.5, 0.75, 1]\">\n      <ng-template>\n        <ion-header>\n          <ion-toolbar>\n            <ion-title>{{ label }}</ion-title>\n            <ion-buttons slot=\"end\">\n              <ion-button (click)=\"cancelModal()\">Cancelar</ion-button>\n            </ion-buttons>\n          </ion-toolbar>\n          <ion-toolbar>\n            <val-searchbar (filterEvent)=\"onFilter($event)\" (focusEvent)=\"onFocus()\" (blurEvent)=\"onBlur()\" />\n          </ion-toolbar>\n        </ion-header>\n        <ion-content>\n          <ion-list>\n            <ion-item *ngFor=\"let item of filteredItems\" button (click)=\"selectItem(item)\" detail=\"false\">\n              <ion-label>{{ item[labelProperty] }}</ion-label>\n              <ion-icon *ngIf=\"isItemSelected(item)\" name=\"checkmark-outline\" slot=\"end\" color=\"primary\"></ion-icon>\n            </ion-item>\n            <ion-item *ngIf=\"filteredItems.length === 0\">\n              <ion-label color=\"medium\">No se encontraron resultados</ion-label>\n            </ion-item>\n          </ion-list>\n        </ion-content>\n      </ng-template>\n    </ion-modal>\n  `,\n  styles: [``],\n})\n/**\n * val-select-search\n *\n * A searchable select/dropdown input with modal and filtering, integrated with Angular forms.\n *\n * @example\n * <val-select-search [props]=\"{ control: myControl, label: 'Choose', options: [{ id: '1', name: 'Option 1' }] }\"></val-select-search>\n *\n * @input props: InputMetadata - Configuration for the select input (form control, label, options, etc.)\n */\nexport class SelectSearchComponent implements OnChanges, OnDestroy {\n  @ViewChild('modal') modal!: IonModal;\n\n  @Input() label: string = 'Seleccionar';\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() multiple: boolean = false;\n  @Input() placeholder: string = 'Seleccione una opción';\n  /**\n   * Input configuration object.\n   * @type {InputMetadata}\n   * @property control - The Angular FormControl for the select input.\n   * @property label - The label for the select.\n   * @property options - The available options for the select.\n   * @property placeholder - The placeholder text.\n   */\n  @Input() props: InputMetadata;\n\n  icon = inject(IconService);\n  private changeDetector = inject(ChangeDetectorRef);\n\n  searchTerm: string = '';\n  filteredItems: InputOption[] = [];\n  selectedItems: InputOption[] = [];\n  displayValue: string = '';\n\n  private previousOptions: any[] = [];\n  private isProcessingChanges = false;\n  private valueChangesSubscription?: Subscription;\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n    this.subscribeToValueChanges();\n  }\n\n  ngOnDestroy() {\n    // Limpiar suscripciones para evitar memory leaks\n    if (this.valueChangesSubscription) {\n      this.valueChangesSubscription.unsubscribe();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Evitar bucles infinitos\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    // Cuando cambia props o props.options\n    if (changes['props']) {\n      try {\n        this.isProcessingChanges = true;\n\n        // Desuscribirse del antiguo control si existe\n        if (this.valueChangesSubscription) {\n          this.valueChangesSubscription.unsubscribe();\n        }\n\n        if (this.props?.options) {\n          // Verificar si las opciones han cambiado realmente\n          const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);\n\n          if (optionsChanged) {\n            this.previousOptions = [...this.props.options];\n            this.initializeItems();\n          }\n        }\n\n        // Sincronizar con el nuevo control si existe\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n\n        // Suscribirse al nuevo control\n        this.subscribeToValueChanges();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    }\n  }\n\n  ionViewWillEnter(): void {\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n      this.initializeItems();\n      this.syncControlValueWithSelectedItems();\n      this.updateDisplayValue();\n      this.subscribeToValueChanges();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Suscribirse a cambios en el FormControl\n  private subscribeToValueChanges() {\n    if (!this.props?.control) return;\n\n    this.valueChangesSubscription = this.props.control.valueChanges.subscribe(value => {\n      if (this.isProcessingChanges) return;\n\n      try {\n        this.isProcessingChanges = true;\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    });\n  }\n\n  // Compara si dos arrays de opciones son iguales\n  private areOptionsEqual(prevOptions: any[], newOptions: any[]): boolean {\n    if (!prevOptions || !newOptions) {\n      return prevOptions === newOptions;\n    }\n\n    if (prevOptions.length !== newOptions.length) {\n      return false;\n    }\n\n    for (let i = 0; i < prevOptions.length; i++) {\n      if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {\n        return false;\n      }\n    }\n\n    return true;\n  }\n\n  private initializeItems(): void {\n    if (this.props?.options) {\n      this.filteredItems = [...this.props.options];\n    } else {\n      this.filteredItems = [];\n    }\n  }\n\n  private syncControlValueWithSelectedItems(): void {\n    if (!this.props?.control) {\n      this.selectedItems = [];\n      return;\n    }\n\n    const controlValue = this.props.control.value;\n\n    // Si el control no tiene valor, limpiar seleccionados\n    if (controlValue === null || controlValue === undefined) {\n      this.selectedItems = [];\n      return;\n    }\n\n    // Buscar la opción que corresponde al valor del control\n    if (this.props.options && this.props.options.length > 0) {\n      const selectedOption = this.props.options.find(opt => opt[this.valueProperty] === controlValue);\n\n      this.selectedItems = selectedOption ? [selectedOption] : [];\n    } else {\n      this.selectedItems = [];\n    }\n  }\n\n  onFilter(event) {\n    // Resetear a todas las opciones disponibles si no hay texto de búsqueda\n    if (!event || event.trim() === '') {\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    // Si no hay opciones disponibles, no hace falta filtrar\n    if (!this.props?.options || this.props.options.length === 0) {\n      this.filteredItems = [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    const text = replaceSpecialChars(event.toLowerCase());\n\n    const filteredResults = this.props.options.filter(element => {\n      const values = Object.values(element).map((a: any) => {\n        if (a === null || a === undefined) return '';\n        return replaceSpecialChars(`${a}`).toLowerCase();\n      });\n\n      return values.some(value => value.includes(text));\n    });\n\n    this.filteredItems = filteredResults;\n    this.changeDetector.detectChanges();\n  }\n\n  onFocus() {\n    console.log('onFocus');\n  }\n\n  onBlur() {\n    console.log('onBlur');\n  }\n\n  openModal() {\n    if (this.modal) {\n      this.modal.present();\n    }\n  }\n\n  preventDefaultBehavior(event: MouseEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.openModal();\n  }\n\n  cancelModal() {\n    if (this.modal) {\n      this.modal.dismiss();\n    }\n  }\n\n  selectItem(item: any) {\n    if (this.multiple) {\n      const index = this.selectedItems.findIndex(\n        selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]\n      );\n\n      if (index === -1) {\n        this.selectedItems.push(item);\n      } else {\n        this.selectedItems.splice(index, 1);\n      }\n    } else {\n      this.selectedItems = [item];\n      this.cancelModal();\n    }\n\n    this.updateDisplayValue();\n    this.applyChanges();\n  }\n\n  isItemSelected(item: any): boolean {\n    return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);\n  }\n\n  updateDisplayValue() {\n    if (this.selectedItems.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.multiple) {\n      if (this.selectedItems.length === 1) {\n        this.displayValue = this.selectedItems[0][this.labelProperty];\n      } else {\n        this.displayValue = `${this.selectedItems.length} elementos seleccionados`;\n      }\n    } else {\n      this.displayValue = this.selectedItems[0][this.labelProperty];\n    }\n  }\n\n  applyChanges() {\n    if (!this.props?.control) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n\n      if (this.selectedItems.length > 0) {\n        this.props.control.setValue(this.selectedItems[0][this.valueProperty]);\n      } else {\n        this.props.control.setValue(null);\n      }\n\n      this.props.control.markAsDirty();\n      this.props.control.updateValueAndValidity();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Método público para reiniciar el componente\n  reset() {\n    this.selectedItems = [];\n    this.displayValue = '';\n    if (this.props?.control) {\n      this.props.control.setValue(null);\n    }\n    this.changeDetector.detectChanges();\n  }\n}\n"]}
|
|
364
|
+
//# 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":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,iBAAiB,EACjB,SAAS,EACT,MAAM,EACN,KAAK,EAIL,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAClE,OAAO,EAAE,WAAW,EAAY,MAAM,gBAAgB,CAAC;AAEvD,OAAO,EAAE,WAAW,EAAE,MAAM,iCAAiC,CAAC;AAC9D,OAAO,EAAE,mBAAmB,EAAE,MAAM,4BAA4B,CAAC;AAEjE,OAAO,EAAE,kBAAkB,EAAE,MAAM,kCAAkC,CAAC;;;;;AA8CtE;;;;;;;;;GASG;AACH,MAAM,OAAO,qBAAqB;IAtDlC;QAyDW,UAAK,GAAW,aAAa,CAAC;QAC9B,kBAAa,GAAW,MAAM,CAAC;QAC/B,kBAAa,GAAW,IAAI,CAAC;QAC7B,aAAQ,GAAY,KAAK,CAAC;QAC1B,gBAAW,GAAW,uBAAuB,CAAC;QAWvD,SAAI,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QACnB,mBAAc,GAAG,MAAM,CAAC,iBAAiB,CAAC,CAAC;QAEnD,eAAU,GAAW,EAAE,CAAC;QACxB,kBAAa,GAAkB,EAAE,CAAC;QAClC,kBAAa,GAAkB,EAAE,CAAC;QAClC,iBAAY,GAAW,EAAE,CAAC;QAElB,oBAAe,GAAU,EAAE,CAAC;QAC5B,wBAAmB,GAAG,KAAK,CAAC;KA+PrC;IA5PC,QAAQ;QACN,IAAI,CAAC,eAAe,EAAE,CAAC;QACvB,IAAI,CAAC,iCAAiC,EAAE,CAAC;QACzC,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,uBAAuB,EAAE,CAAC;IACjC,CAAC;IAED,WAAW;QACT,iDAAiD;QACjD,IAAI,IAAI,CAAC,wBAAwB,EAAE,CAAC;YAClC,IAAI,CAAC,wBAAwB,CAAC,WAAW,EAAE,CAAC;QAC9C,CAAC;IACH,CAAC;IAED,WAAW,CAAC,OAAsB;QAChC,0BAA0B;QAC1B,IAAI,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7B,OAAO;QACT,CAAC;QAED,sCAAsC;QACtC,IAAI,OAAO,CAAC,OAAO,CAAC,EAAE,CAAC;YACrB,IAAI,CAAC;gBACH,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;gBAEhC,8CAA8C;gBAC9C,IAAI,IAAI,CAAC,wBAAwB,EAAE,CAAC;oBAClC,IAAI,CAAC,wBAAwB,CAAC,WAAW,EAAE,CAAC;gBAC9C,CAAC;gBAED,IAAI,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC;oBACxB,mDAAmD;oBACnD,MAAM,cAAc,GAAG,CAAC,IAAI,CAAC,eAAe,CAAC,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;oBAEvF,IAAI,cAAc,EAAE,CAAC;wBACnB,IAAI,CAAC,eAAe,GAAG,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC;wBAC/C,IAAI,CAAC,eAAe,EAAE,CAAC;oBACzB,CAAC;gBACH,CAAC;gBAED,6CAA6C;gBAC7C,IAAI,CAAC,iCAAiC,EAAE,CAAC;gBACzC,IAAI,CAAC,kBAAkB,EAAE,CAAC;gBAE1B,+BAA+B;gBAC/B,IAAI,CAAC,uBAAuB,EAAE,CAAC;YACjC,CAAC;oBAAS,CAAC;gBACT,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;YACnC,CAAC;QACH,CAAC;IACH,CAAC;IAED,gBAAgB;QACd,IAAI,IAAI,CAAC,mBAAmB,EAAE,CAAC;YAC7B,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;YAChC,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,iCAAiC,EAAE,CAAC;YACzC,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC1B,IAAI,CAAC,uBAAuB,EAAE,CAAC;QACjC,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;QACnC,CAAC;IACH,CAAC;IAED,0CAA0C;IAClC,uBAAuB;QAC7B,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO;YAAE,OAAO;QAEjC,IAAI,CAAC,wBAAwB,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,YAAY,CAAC,SAAS,CAAC,KAAK,CAAC,EAAE;YAChF,IAAI,IAAI,CAAC,mBAAmB;gBAAE,OAAO;YAErC,IAAI,CAAC;gBACH,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;gBAChC,IAAI,CAAC,iCAAiC,EAAE,CAAC;gBACzC,IAAI,CAAC,kBAAkB,EAAE,CAAC;YAC5B,CAAC;oBAAS,CAAC;gBACT,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;YACnC,CAAC;QACH,CAAC,CAAC,CAAC;IACL,CAAC;IAED,gDAAgD;IACxC,eAAe,CAAC,WAAkB,EAAE,UAAiB;QAC3D,mDAAmD;QACnD,IAAI,WAAW,KAAK,UAAU;YAAE,OAAO,IAAI,CAAC;QAC5C,IAAI,CAAC,WAAW,IAAI,CAAC,UAAU;YAAE,OAAO,WAAW,KAAK,UAAU,CAAC;QACnE,IAAI,WAAW,CAAC,MAAM,KAAK,UAAU,CAAC,MAAM;YAAE,OAAO,KAAK,CAAC;QAC3D,0CAA0C;QAC1C,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,WAAW,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE,CAAC;YAC5C,IAAI,WAAW,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,UAAU,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC;gBAC7E,OAAO,KAAK,CAAC;YACf,CAAC;QACH,CAAC;QACD,OAAO,IAAI,CAAC;IACd,CAAC;IAEO,eAAe;QACrB,uCAAuC;QACvC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,EAAE,CAAC;IACjD,CAAC;IAEO,iCAAiC;QACvC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC;YACzB,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;YACxB,OAAO;QACT,CAAC;QACD,MAAM,YAAY,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC;QAC9C,IAAI,YAAY,KAAK,IAAI,IAAI,YAAY,KAAK,SAAS,EAAE,CAAC;YACxD,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;YACxB,OAAO;QACT,CAAC;QACD,yDAAyD;QACzD,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;YACxD,MAAM,GAAG,GAAG,IAAI,GAAG,CAAC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC,GAAG,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,GAAG,CAAC,CAAC,CAAC,CAAC;YACnF,MAAM,cAAc,GAAG,GAAG,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;YAC7C,IAAI,CAAC,aAAa,GAAG,cAAc,CAAC,CAAC,CAAC,CAAC,cAAc,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;QAC9D,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QAC1B,CAAC;IACH,CAAC;IAED,QAAQ,CAAC,KAAK;QACZ,sCAAsC;QACtC,IAAI,CAAC,KAAK,IAAI,KAAK,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;YAClC,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACxE,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;YACpC,OAAO;QACT,CAAC;QAED,mCAAmC;QACnC,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YAC5D,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;YACxB,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;YACpC,OAAO;QACT,CAAC;QAED,2EAA2E;QAC3E,MAAM,MAAM,GAAG,mBAAmB,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;QACxD,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,OAAO,CAAC,EAAE;YACvD,kEAAkE;YAClE,MAAM,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACxH,MAAM,KAAK,GAAG,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,mBAAmB,CAAC,MAAM,CAAC,OAAO,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACxH,OAAO,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,IAAI,KAAK,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC;QAC1D,CAAC,CAAC,CAAC;QACH,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;IACtC,CAAC;IAED,OAAO;QACL,OAAO,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;IACzB,CAAC;IAED,MAAM;QACJ,OAAO,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;IACxB,CAAC;IAED,SAAS;QACP,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAED,sBAAsB,CAAC,KAAiB;QACtC,KAAK,CAAC,cAAc,EAAE,CAAC;QACvB,KAAK,CAAC,eAAe,EAAE,CAAC;QACxB,IAAI,CAAC,SAAS,EAAE,CAAC;IACnB,CAAC;IAED,WAAW;QACT,IAAI,IAAI,CAAC,KAAK,EAAE,CAAC;YACf,IAAI,CAAC,KAAK,CAAC,OAAO,EAAE,CAAC;QACvB,CAAC;IACH,CAAC;IAED,UAAU,CAAC,IAAS;QAClB,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClB,MAAM,KAAK,GAAG,IAAI,CAAC,aAAa,CAAC,SAAS,CACxC,YAAY,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAC9E,CAAC;YAEF,IAAI,KAAK,KAAK,CAAC,CAAC,EAAE,CAAC;gBACjB,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;YAChC,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,aAAa,CAAC,MAAM,CAAC,KAAK,EAAE,CAAC,CAAC,CAAC;YACtC,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,aAAa,GAAG,CAAC,IAAI,CAAC,CAAC;YAC5B,IAAI,CAAC,WAAW,EAAE,CAAC;YACnB,uDAAuD;YACvD,IAAI,CAAC,UAAU,GAAG,EAAE,CAAC;YACrB,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,KAAK,EAAE,OAAO,CAAC,CAAC,CAAC,CAAC,GAAG,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACxE,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;QACtC,CAAC;QAED,IAAI,CAAC,kBAAkB,EAAE,CAAC;QAC1B,IAAI,CAAC,YAAY,EAAE,CAAC;IACtB,CAAC;IAED,cAAc,CAAC,IAAS;QACtB,OAAO,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,YAAY,CAAC,EAAE,CAAC,YAAY,CAAC,IAAI,CAAC,aAAa,CAAC,KAAK,IAAI,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;IAChH,CAAC;IAED,kBAAkB;QAChB,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;YACpC,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;YACvB,OAAO;QACT,CAAC;QAED,IAAI,IAAI,CAAC,QAAQ,EAAE,CAAC;YAClB,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;gBACpC,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;YAChE,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,YAAY,GAAG,GAAG,IAAI,CAAC,aAAa,CAAC,MAAM,0BAA0B,CAAC;YAC7E,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,YAAY,GAAG,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC;QAChE,CAAC;IACH,CAAC;IAED,YAAY;QACV,IAAI,CAAC,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC;YACzB,OAAO;QACT,CAAC;QAED,IAAI,CAAC;YACH,IAAI,CAAC,mBAAmB,GAAG,IAAI,CAAC;YAEhC,IAAI,IAAI,CAAC,aAAa,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;gBAClC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC,CAAC,IAAI,CAAC,aAAa,CAAC,CAAC,CAAC;YACzE,CAAC;iBAAM,CAAC;gBACN,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;YACpC,CAAC;YAED,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC;YACjC,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,sBAAsB,EAAE,CAAC;QAC9C,CAAC;gBAAS,CAAC;YACT,IAAI,CAAC,mBAAmB,GAAG,KAAK,CAAC;QACnC,CAAC;IACH,CAAC;IAED,8CAA8C;IAC9C,KAAK;QACH,IAAI,CAAC,aAAa,GAAG,EAAE,CAAC;QACxB,IAAI,CAAC,YAAY,GAAG,EAAE,CAAC;QACvB,IAAI,IAAI,CAAC,KAAK,EAAE,OAAO,EAAE,CAAC;YACxB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;QACpC,CAAC;QACD,IAAI,CAAC,cAAc,CAAC,aAAa,EAAE,CAAC;IACtC,CAAC;+GAzRU,qBAAqB;mGAArB,qBAAqB,gWAlDtB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT,yEAtCS,YAAY,+PAAE,WAAW,wjEAAE,WAAW,uIAAE,kBAAkB,oIAAE,mBAAmB;;4FAmD9E,qBAAqB;kBAtDjC,SAAS;+BACE,mBAAmB,cACjB,IAAI,WACP,CAAC,YAAY,EAAE,WAAW,EAAE,WAAW,EAAE,kBAAkB,EAAE,mBAAmB,CAAC,YAChF;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqCT;8BAcmB,KAAK;sBAAxB,SAAS;uBAAC,OAAO;gBAET,KAAK;sBAAb,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,aAAa;sBAArB,KAAK;gBACG,QAAQ;sBAAhB,KAAK;gBACG,WAAW;sBAAnB,KAAK;gBASG,KAAK;sBAAb,KAAK","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  ChangeDetectorRef,\n  Component,\n  inject,\n  Input,\n  OnChanges,\n  OnDestroy,\n  SimpleChanges,\n  ViewChild,\n} from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { Subscription } from 'rxjs';\nimport { IconService } from '../../../services/icons.service';\nimport { replaceSpecialChars } from '../../../shared/utils/text';\nimport { InputMetadata, InputOption } from '../../types';\nimport { SearchbarComponent } from '../searchbar/searchbar.component';\n\n@Component({\n  selector: 'val-select-search',\n  standalone: true,\n  imports: [CommonModule, IonicModule, FormsModule, SearchbarComponent, ReactiveFormsModule],\n  template: `\n    <ion-input\n      type=\"text\"\n      [value]=\"displayValue\"\n      [placeholder]=\"props?.placeholder || placeholder\"\n      readonly\n      (mousedown)=\"preventDefaultBehavior($event)\"\n    />\n\n    <ion-input style=\"position: absolute;\" [formControl]=\"props.control\" type=\"hidden\"></ion-input>\n\n    <ion-modal #modal trigger=\"open-modal\" [initialBreakpoint]=\"0.75\" [breakpoints]=\"[0, 0.5, 0.75, 1]\">\n      <ng-template>\n        <ion-header>\n          <ion-toolbar>\n            <ion-title>{{ label }}</ion-title>\n            <ion-buttons slot=\"end\">\n              <ion-button (click)=\"cancelModal()\">Cancelar</ion-button>\n            </ion-buttons>\n          </ion-toolbar>\n          <ion-toolbar>\n            <val-searchbar (filterEvent)=\"onFilter($event)\" (focusEvent)=\"onFocus()\" (blurEvent)=\"onBlur()\" />\n          </ion-toolbar>\n        </ion-header>\n        <ion-content>\n          <ion-list>\n            <ion-item *ngFor=\"let item of filteredItems\" button (click)=\"selectItem(item)\" detail=\"false\">\n              <ion-label>{{ item[labelProperty] }}</ion-label>\n              <ion-icon *ngIf=\"isItemSelected(item)\" name=\"checkmark-outline\" slot=\"end\" color=\"primary\"></ion-icon>\n            </ion-item>\n            <ion-item *ngIf=\"filteredItems.length === 0\">\n              <ion-label color=\"medium\">No se encontraron resultados</ion-label>\n            </ion-item>\n          </ion-list>\n        </ion-content>\n      </ng-template>\n    </ion-modal>\n  `,\n  styles: [``],\n})\n/**\n * val-select-search\n *\n * A searchable select/dropdown input with modal and filtering, integrated with Angular forms.\n *\n * @example\n * <val-select-search [props]=\"{ control: myControl, label: 'Choose', options: [{ id: '1', name: 'Option 1' }] }\"></val-select-search>\n *\n * @input props: InputMetadata - Configuration for the select input (form control, label, options, etc.)\n */\nexport class SelectSearchComponent implements OnChanges, OnDestroy {\n  @ViewChild('modal') modal!: IonModal;\n\n  @Input() label: string = 'Seleccionar';\n  @Input() labelProperty: string = 'name';\n  @Input() valueProperty: string = 'id';\n  @Input() multiple: boolean = false;\n  @Input() placeholder: string = 'Seleccione una opción';\n  /**\n   * Input configuration object.\n   * @type {InputMetadata}\n   * @property control - The Angular FormControl for the select input.\n   * @property label - The label for the select.\n   * @property options - The available options for the select.\n   * @property placeholder - The placeholder text.\n   */\n  @Input() props: InputMetadata;\n\n  icon = inject(IconService);\n  private changeDetector = inject(ChangeDetectorRef);\n\n  searchTerm: string = '';\n  filteredItems: InputOption[] = [];\n  selectedItems: InputOption[] = [];\n  displayValue: string = '';\n\n  private previousOptions: any[] = [];\n  private isProcessingChanges = false;\n  private valueChangesSubscription?: Subscription;\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n    this.subscribeToValueChanges();\n  }\n\n  ngOnDestroy() {\n    // Limpiar suscripciones para evitar memory leaks\n    if (this.valueChangesSubscription) {\n      this.valueChangesSubscription.unsubscribe();\n    }\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Evitar bucles infinitos\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    // Cuando cambia props o props.options\n    if (changes['props']) {\n      try {\n        this.isProcessingChanges = true;\n\n        // Desuscribirse del antiguo control si existe\n        if (this.valueChangesSubscription) {\n          this.valueChangesSubscription.unsubscribe();\n        }\n\n        if (this.props?.options) {\n          // Verificar si las opciones han cambiado realmente\n          const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);\n\n          if (optionsChanged) {\n            this.previousOptions = [...this.props.options];\n            this.initializeItems();\n          }\n        }\n\n        // Sincronizar con el nuevo control si existe\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n\n        // Suscribirse al nuevo control\n        this.subscribeToValueChanges();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    }\n  }\n\n  ionViewWillEnter(): void {\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n      this.initializeItems();\n      this.syncControlValueWithSelectedItems();\n      this.updateDisplayValue();\n      this.subscribeToValueChanges();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Suscribirse a cambios en el FormControl\n  private subscribeToValueChanges() {\n    if (!this.props?.control) return;\n\n    this.valueChangesSubscription = this.props.control.valueChanges.subscribe(value => {\n      if (this.isProcessingChanges) return;\n\n      try {\n        this.isProcessingChanges = true;\n        this.syncControlValueWithSelectedItems();\n        this.updateDisplayValue();\n      } finally {\n        this.isProcessingChanges = false;\n      }\n    });\n  }\n\n  // Compara si dos arrays de opciones son iguales\n  private areOptionsEqual(prevOptions: any[], newOptions: any[]): boolean {\n    // PERF: Use reference equality first for fast path\n    if (prevOptions === newOptions) return true;\n    if (!prevOptions || !newOptions) return prevOptions === newOptions;\n    if (prevOptions.length !== newOptions.length) return false;\n    // Only compare valueProperty for equality\n    for (let i = 0; i < prevOptions.length; i++) {\n      if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {\n        return false;\n      }\n    }\n    return true;\n  }\n\n  private initializeItems(): void {\n    // PERF: Avoid unnecessary array copies\n    this.filteredItems = this.props?.options || [];\n  }\n\n  private syncControlValueWithSelectedItems(): void {\n    if (!this.props?.control) {\n      this.selectedItems = [];\n      return;\n    }\n    const controlValue = this.props.control.value;\n    if (controlValue === null || controlValue === undefined) {\n      this.selectedItems = [];\n      return;\n    }\n    // PERF: Use a Map for faster lookup if options are large\n    if (this.props.options && this.props.options.length > 0) {\n      const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));\n      const selectedOption = map.get(controlValue);\n      this.selectedItems = selectedOption ? [selectedOption] : [];\n    } else {\n      this.selectedItems = [];\n    }\n  }\n\n  onFilter(event) {\n    // If no search term, show all options\n    if (!event || event.trim() === '') {\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    // If no options, nothing to filter\n    if (!this.props?.options || this.props.options.length === 0) {\n      this.filteredItems = [];\n      this.changeDetector.detectChanges();\n      return;\n    }\n\n    // PERF: Avoid repeated replaceSpecialChars and toLowerCase for each option\n    const search = replaceSpecialChars(event.toLowerCase());\n    this.filteredItems = this.props.options.filter(element => {\n      // Only use labelProperty and valueProperty for filtering (faster)\n      const label = element[this.labelProperty] ? replaceSpecialChars(String(element[this.labelProperty]).toLowerCase()) : '';\n      const value = element[this.valueProperty] ? replaceSpecialChars(String(element[this.valueProperty]).toLowerCase()) : '';\n      return label.includes(search) || value.includes(search);\n    });\n    this.changeDetector.detectChanges();\n  }\n\n  onFocus() {\n    console.log('onFocus');\n  }\n\n  onBlur() {\n    console.log('onBlur');\n  }\n\n  openModal() {\n    if (this.modal) {\n      this.modal.present();\n    }\n  }\n\n  preventDefaultBehavior(event: MouseEvent) {\n    event.preventDefault();\n    event.stopPropagation();\n    this.openModal();\n  }\n\n  cancelModal() {\n    if (this.modal) {\n      this.modal.dismiss();\n    }\n  }\n\n  selectItem(item: any) {\n    if (this.multiple) {\n      const index = this.selectedItems.findIndex(\n        selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]\n      );\n\n      if (index === -1) {\n        this.selectedItems.push(item);\n      } else {\n        this.selectedItems.splice(index, 1);\n      }\n    } else {\n      this.selectedItems = [item];\n      this.cancelModal();\n      // Reset filter and show all options when closing modal\n      this.searchTerm = '';\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      this.changeDetector.detectChanges();\n    }\n\n    this.updateDisplayValue();\n    this.applyChanges();\n  }\n\n  isItemSelected(item: any): boolean {\n    return this.selectedItems.some(selectedItem => selectedItem[this.valueProperty] === item[this.valueProperty]);\n  }\n\n  updateDisplayValue() {\n    if (this.selectedItems.length === 0) {\n      this.displayValue = '';\n      return;\n    }\n\n    if (this.multiple) {\n      if (this.selectedItems.length === 1) {\n        this.displayValue = this.selectedItems[0][this.labelProperty];\n      } else {\n        this.displayValue = `${this.selectedItems.length} elementos seleccionados`;\n      }\n    } else {\n      this.displayValue = this.selectedItems[0][this.labelProperty];\n    }\n  }\n\n  applyChanges() {\n    if (!this.props?.control) {\n      return;\n    }\n\n    try {\n      this.isProcessingChanges = true;\n\n      if (this.selectedItems.length > 0) {\n        this.props.control.setValue(this.selectedItems[0][this.valueProperty]);\n      } else {\n        this.props.control.setValue(null);\n      }\n\n      this.props.control.markAsDirty();\n      this.props.control.updateValueAndValidity();\n    } finally {\n      this.isProcessingChanges = false;\n    }\n  }\n\n  // Método público para reiniciar el componente\n  reset() {\n    this.selectedItems = [];\n    this.displayValue = '';\n    if (this.props?.control) {\n      this.props.control.setValue(null);\n    }\n    this.changeDetector.detectChanges();\n  }\n}\n"]}
|
|
@@ -219,7 +219,7 @@ export class ItemListComponent {
|
|
|
219
219
|
}
|
|
220
220
|
}
|
|
221
221
|
</ion-list>
|
|
222
|
-
`, isInline: true, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonListHeader, selector: "ion-list-header", inputs: ["color", "lines", "mode"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonNote, selector: "ion-note", inputs: ["color", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonAvatar, selector: "ion-avatar" }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: 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: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }] }); }
|
|
222
|
+
`, isInline: true, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important;padding:.25rem 0rem}ion-list-header ion-label{margin-top:1rem}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonListHeader, selector: "ion-list-header", inputs: ["color", "lines", "mode"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonNote, selector: "ion-note", inputs: ["color", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonAvatar, selector: "ion-avatar" }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: 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: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }] }); }
|
|
223
223
|
}
|
|
224
224
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ItemListComponent, decorators: [{
|
|
225
225
|
type: Component,
|
|
@@ -401,7 +401,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
401
401
|
}
|
|
402
402
|
}
|
|
403
403
|
</ion-list>
|
|
404
|
-
`, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"] }]
|
|
404
|
+
`, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important;padding:.25rem 0rem}ion-list-header ion-label{margin-top:1rem}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"] }]
|
|
405
405
|
}], ctorParameters: () => [{ type: i1.IconService }], propDecorators: { props: [{
|
|
406
406
|
type: Input
|
|
407
407
|
}], onClick: [{
|
|
@@ -409,4 +409,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
409
409
|
}], onActionClick: [{
|
|
410
410
|
type: Output
|
|
411
411
|
}] } });
|
|
412
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"item-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/item-list/item-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EACL,SAAS,EACT,SAAS,EACT,OAAO,EACP,OAAO,EACP,QAAQ,EACR,OAAO,EACP,aAAa,EACb,OAAO,EACP,UAAU,GACX,MAAM,2BAA2B,CAAC;AAEnC,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;;;AA4L9C;;;;;;;;;;;;GAYG;AACH,MAAM,OAAO,iBAAiB;IAsB5B,YAAY,IAAiB;QArB7B,WAAM,GAAG,eAAe,CAAC;QACzB,iBAAY,GAAG,SAAS,CAAC;QAQzB;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAU,CAAC;QAErC;;WAEG;QAEH,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;IAEX,CAAC;IAEjC,QAAQ,KAAI,CAAC;IAEb;;;OAGG;IACH,YAAY,CAAC,KAAc;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,KAAc;QACjC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,KAAK;QACH,OAAO,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC;IAC7D,CAAC;+GA/CU,iBAAiB;mGAAjB,iBAAiB,sKArLlB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT,6mBAlLC,UAAU,oOAEV,OAAO,oFACP,OAAO,yFACP,aAAa,gGACb,QAAQ,6FACR,OAAO,gFACP,OAAO,0NACP,SAAS,uDACT,OAAO,2JACP,SAAS,oPACT,UAAU;;4FAuLD,iBAAiB;kBAtM7B,SAAS;+BACE,eAAe,cACb,IAAI,WACP;wBACP,UAAU;wBACV,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,aAAa;wBACb,QAAQ;wBACR,OAAO;wBACP,OAAO;wBACP,SAAS;wBACT,OAAO;wBACP,SAAS;wBACT,UAAU;qBACX,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT;gFAwBD,KAAK;sBADJ,KAAK;gBAON,OAAO;sBADN,MAAM;gBAOP,aAAa;sBADZ,MAAM","sourcesContent":["import { NgClass, NgStyle } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport {\n  IonAvatar,\n  IonButton,\n  IonIcon,\n  IonItem,\n  IonLabel,\n  IonList,\n  IonListHeader,\n  IonNote,\n  IonSpinner,\n} from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { resolveColor } from '../../../shared/utils/styles';\nimport { ComponentStates } from '../../types';\nimport { ListMetadata } from './types';\n\n@Component({\n  selector: 'val-item-list',\n  standalone: true,\n  imports: [\n    RouterLink,\n    NgStyle,\n    NgClass,\n    IonList,\n    IonListHeader,\n    IonLabel,\n    IonNote,\n    IonItem,\n    IonAvatar,\n    IonIcon,\n    IonButton,\n    IonSpinner,\n  ],\n  template: `\n    <ion-list\n      [inset]=\"props.rounded\"\n      [class.shadowed]=\"props.shadowed\"\n      [class.bordered]=\"props.bordered\"\n      [style.borderColor]=\"this.color()\"\n    >\n      @if (props.title) {\n        <ion-list-header>\n          <ion-label>{{ props.title }}</ion-label>\n        </ion-list-header>\n      }\n      @for (item of props.items; track item.token) {\n        @if (item.mode == 'justext') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n\n              @if (item.subtitle) {\n                <p>{{ item.subtitle }}</p>\n              }\n            </ion-label>\n\n            @if (item.endNote) {\n              <ion-note slot=\"end\">{{ item.endNote }}</ion-note>\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'actionable') {\n          <ion-item [lines]=\"item.lines\" [button]=\"item.clickable\" [disabled]=\"item.disabled\">\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.endNote) {\n                <h3>{{ item.endNote }}</h3>\n              }\n\n              @if (item.subtitle) {\n                <ion-note color=\"medium\"> {{ item.subtitle }} </ion-note>\n              }\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n\n            @if (item.actions) {\n              @for (action of item.actions; track action.token) {\n                <ion-button\n                  fill=\"outline\"\n                  color=\"dark\"\n                  slot=\"end\"\n                  [disabled]=\"action.state === states.WORKING\"\n                  (click)=\"onActionClickHandler(action.token)\"\n                >\n                  @if (action.state !== states.WORKING) {\n                    @if (action.icon) {\n                      <ion-icon aria-hidden=\"true\" [name]=\"action.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n                    }\n\n                    {{ action.description }}\n                  }\n\n                  @if (action.state === states.WORKING) {\n                    <ion-spinner class=\"action-spinner\" name=\"circular\"></ion-spinner>\n                  }\n                </ion-button>\n              }\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'route') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n            [routerLink]=\"item.routerLink\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'subtitule') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h1>{{ item.text }}</h1>\n              <p>{{ item.subtitle }}</p>\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'avatar') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-avatar aria-hidden=\"true\" slot=\"start\">\n              <img alt=\"\" [src]=\"item.image\" />\n            </ion-avatar>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'icon') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-icon aria-hidden=\"true\" [name]=\"item.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n      }\n    </ion-list>\n  `,\n  styleUrls: ['./item-list.component.scss'],\n})\n/**\n * ItemListComponent\n *\n * Componente de lista reutilizable para mostrar una colección de elementos con diferentes modos de visualización (texto, subtítulo, avatar, icono, ruta, acciones).\n * Permite acciones por elemento, indicadores de no leído, y personalización de estilos.\n *\n * @example\n * <val-item-list [props]=\"{ title: 'Usuarios', items: [{ text: 'Juan', mode: 'avatar', image: 'avatar.jpg' }] }\" (onClick)=\"handleClick($event)\"></val-item-list>\n *\n * @input props {ListMetadata} - Metadatos de la lista y sus elementos.\n * @output onClick - Emite el token del elemento clicado.\n * @output onActionClick - Emite el token de la acción clicada en un elemento.\n */\nexport class ItemListComponent implements OnInit {\n  states = ComponentStates;\n  defaultColor = 'primary';\n\n  /**\n   * Metadatos de la lista y sus elementos.\n   */\n  @Input()\n  props: ListMetadata;\n\n  /**\n   * Evento emitido al hacer click en un elemento de la lista.\n   */\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  /**\n   * Evento emitido al hacer click en una acción de un elemento.\n   */\n  @Output()\n  onActionClick = new EventEmitter<string>();\n\n  constructor(icon: IconService) {}\n\n  ngOnInit() {}\n\n  /**\n   * Emite el token del elemento clicado.\n   * @param token Token del elemento\n   */\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  /**\n   * Emite el token de la acción clicada.\n   * @param token Token de la acción\n   */\n  onActionClickHandler(token?: string) {\n    this.onActionClick.emit(token);\n  }\n\n  /**\n   * Devuelve el color de borde de la lista.\n   */\n  color() {\n    return resolveColor(this.props.color || this.defaultColor);\n  }\n}\n"]}
|
|
412
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"item-list.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/item-list/item-list.component.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,OAAO,EAAE,OAAO,EAAE,MAAM,iBAAiB,CAAC;AACnD,OAAO,EAAE,SAAS,EAAE,YAAY,EAAE,KAAK,EAAU,MAAM,EAAE,MAAM,eAAe,CAAC;AAC/E,OAAO,EAAE,UAAU,EAAE,MAAM,iBAAiB,CAAC;AAC7C,OAAO,EACL,SAAS,EACT,SAAS,EACT,OAAO,EACP,OAAO,EACP,QAAQ,EACR,OAAO,EACP,aAAa,EACb,OAAO,EACP,UAAU,GACX,MAAM,2BAA2B,CAAC;AAEnC,OAAO,EAAE,YAAY,EAAE,MAAM,8BAA8B,CAAC;AAC5D,OAAO,EAAE,eAAe,EAAE,MAAM,aAAa,CAAC;;;AA4L9C;;;;;;;;;;;;GAYG;AACH,MAAM,OAAO,iBAAiB;IAsB5B,YAAY,IAAiB;QArB7B,WAAM,GAAG,eAAe,CAAC;QACzB,iBAAY,GAAG,SAAS,CAAC;QAQzB;;WAEG;QAEH,YAAO,GAAG,IAAI,YAAY,EAAU,CAAC;QAErC;;WAEG;QAEH,kBAAa,GAAG,IAAI,YAAY,EAAU,CAAC;IAEX,CAAC;IAEjC,QAAQ,KAAI,CAAC;IAEb;;;OAGG;IACH,YAAY,CAAC,KAAc;QACzB,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;IAED;;;OAGG;IACH,oBAAoB,CAAC,KAAc;QACjC,IAAI,CAAC,aAAa,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IACjC,CAAC;IAED;;OAEG;IACH,KAAK;QACH,OAAO,YAAY,CAAC,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,IAAI,CAAC,YAAY,CAAC,CAAC;IAC7D,CAAC;+GA/CU,iBAAiB;mGAAjB,iBAAiB,sKArLlB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT,2qBAlLC,UAAU,oOAEV,OAAO,oFACP,OAAO,yFACP,aAAa,gGACb,QAAQ,6FACR,OAAO,gFACP,OAAO,0NACP,SAAS,uDACT,OAAO,2JACP,SAAS,oPACT,UAAU;;4FAuLD,iBAAiB;kBAtM7B,SAAS;+BACE,eAAe,cACb,IAAI,WACP;wBACP,UAAU;wBACV,OAAO;wBACP,OAAO;wBACP,OAAO;wBACP,aAAa;wBACb,QAAQ;wBACR,OAAO;wBACP,OAAO;wBACP,SAAS;wBACT,OAAO;wBACP,SAAS;wBACT,UAAU;qBACX,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAqKT;gFAwBD,KAAK;sBADJ,KAAK;gBAON,OAAO;sBADN,MAAM;gBAOP,aAAa;sBADZ,MAAM","sourcesContent":["import { NgClass, NgStyle } from '@angular/common';\nimport { Component, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { RouterLink } from '@angular/router';\nimport {\n  IonAvatar,\n  IonButton,\n  IonIcon,\n  IonItem,\n  IonLabel,\n  IonList,\n  IonListHeader,\n  IonNote,\n  IonSpinner,\n} from '@ionic/angular/standalone';\nimport { IconService } from '../../../services/icons.service';\nimport { resolveColor } from '../../../shared/utils/styles';\nimport { ComponentStates } from '../../types';\nimport { ListMetadata } from './types';\n\n@Component({\n  selector: 'val-item-list',\n  standalone: true,\n  imports: [\n    RouterLink,\n    NgStyle,\n    NgClass,\n    IonList,\n    IonListHeader,\n    IonLabel,\n    IonNote,\n    IonItem,\n    IonAvatar,\n    IonIcon,\n    IonButton,\n    IonSpinner,\n  ],\n  template: `\n    <ion-list\n      [inset]=\"props.rounded\"\n      [class.shadowed]=\"props.shadowed\"\n      [class.bordered]=\"props.bordered\"\n      [style.borderColor]=\"this.color()\"\n    >\n      @if (props.title) {\n        <ion-list-header>\n          <ion-label>{{ props.title }}</ion-label>\n        </ion-list-header>\n      }\n      @for (item of props.items; track item.token) {\n        @if (item.mode == 'justext') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n\n              @if (item.subtitle) {\n                <p>{{ item.subtitle }}</p>\n              }\n            </ion-label>\n\n            @if (item.endNote) {\n              <ion-note slot=\"end\">{{ item.endNote }}</ion-note>\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'actionable') {\n          <ion-item [lines]=\"item.lines\" [button]=\"item.clickable\" [disabled]=\"item.disabled\">\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.endNote) {\n                <h3>{{ item.endNote }}</h3>\n              }\n\n              @if (item.subtitle) {\n                <ion-note color=\"medium\"> {{ item.subtitle }} </ion-note>\n              }\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n\n            @if (item.actions) {\n              @for (action of item.actions; track action.token) {\n                <ion-button\n                  fill=\"outline\"\n                  color=\"dark\"\n                  slot=\"end\"\n                  [disabled]=\"action.state === states.WORKING\"\n                  (click)=\"onActionClickHandler(action.token)\"\n                >\n                  @if (action.state !== states.WORKING) {\n                    @if (action.icon) {\n                      <ion-icon aria-hidden=\"true\" [name]=\"action.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n                    }\n\n                    {{ action.description }}\n                  }\n\n                  @if (action.state === states.WORKING) {\n                    <ion-spinner class=\"action-spinner\" name=\"circular\"></ion-spinner>\n                  }\n                </ion-button>\n              }\n            }\n          </ion-item>\n        }\n\n        @if (item.mode == 'route') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n            [routerLink]=\"item.routerLink\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h2>{{ item.text }}</h2>\n\n              @if (item.comments) {\n                <ion-note color=\"medium\"> {{ item.comments }} </ion-note>\n              }\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'subtitule') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            @if (item.unreadIndicator) {\n              <div class=\"unread-indicator-wrapper\" slot=\"start\">\n                <div class=\"unread-indicator\"></div>\n              </div>\n            }\n\n            <ion-label [ngClass]=\"{ 'ion-text-nowrap': item.ellipsis }\">\n              <h1>{{ item.text }}</h1>\n              <p>{{ item.subtitle }}</p>\n            </ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'avatar') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-avatar aria-hidden=\"true\" slot=\"start\">\n              <img alt=\"\" [src]=\"item.image\" />\n            </ion-avatar>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n\n        @if (item.mode == 'icon') {\n          <ion-item\n            [lines]=\"item.lines\"\n            [button]=\"item.clickable\"\n            [disabled]=\"item.disabled\"\n            (click)=\"clickHandler(item.token)\"\n          >\n            <ion-icon aria-hidden=\"true\" [name]=\"item.icon\" size=\"medium\" slot=\"start\"></ion-icon>\n            <ion-label>{{ item.text }}</ion-label>\n          </ion-item>\n        }\n      }\n    </ion-list>\n  `,\n  styleUrls: ['./item-list.component.scss'],\n})\n/**\n * ItemListComponent\n *\n * Componente de lista reutilizable para mostrar una colección de elementos con diferentes modos de visualización (texto, subtítulo, avatar, icono, ruta, acciones).\n * Permite acciones por elemento, indicadores de no leído, y personalización de estilos.\n *\n * @example\n * <val-item-list [props]=\"{ title: 'Usuarios', items: [{ text: 'Juan', mode: 'avatar', image: 'avatar.jpg' }] }\" (onClick)=\"handleClick($event)\"></val-item-list>\n *\n * @input props {ListMetadata} - Metadatos de la lista y sus elementos.\n * @output onClick - Emite el token del elemento clicado.\n * @output onActionClick - Emite el token de la acción clicada en un elemento.\n */\nexport class ItemListComponent implements OnInit {\n  states = ComponentStates;\n  defaultColor = 'primary';\n\n  /**\n   * Metadatos de la lista y sus elementos.\n   */\n  @Input()\n  props: ListMetadata;\n\n  /**\n   * Evento emitido al hacer click en un elemento de la lista.\n   */\n  @Output()\n  onClick = new EventEmitter<string>();\n\n  /**\n   * Evento emitido al hacer click en una acción de un elemento.\n   */\n  @Output()\n  onActionClick = new EventEmitter<string>();\n\n  constructor(icon: IconService) {}\n\n  ngOnInit() {}\n\n  /**\n   * Emite el token del elemento clicado.\n   * @param token Token del elemento\n   */\n  clickHandler(token?: string) {\n    this.onClick.emit(token);\n  }\n\n  /**\n   * Emite el token de la acción clicada.\n   * @param token Token de la acción\n   */\n  onActionClickHandler(token?: string) {\n    this.onActionClick.emit(token);\n  }\n\n  /**\n   * Devuelve el color de borde de la lista.\n   */\n  color() {\n    return resolveColor(this.props.color || this.defaultColor);\n  }\n}\n"]}
|
|
@@ -2211,9 +2211,9 @@ class PinInputComponent {
|
|
|
2211
2211
|
this.codeLength = 5;
|
|
2212
2212
|
this.otpInputConfig = {
|
|
2213
2213
|
inputStyles: {
|
|
2214
|
-
'font-size': '
|
|
2215
|
-
width: '
|
|
2216
|
-
height: '
|
|
2214
|
+
'font-size': '32px',
|
|
2215
|
+
width: '58px',
|
|
2216
|
+
height: '60px',
|
|
2217
2217
|
},
|
|
2218
2218
|
inputClass: 'otp-input-box',
|
|
2219
2219
|
length: this.codeLength,
|
|
@@ -2762,12 +2762,14 @@ class SelectSearchComponent {
|
|
|
2762
2762
|
}
|
|
2763
2763
|
// Compara si dos arrays de opciones son iguales
|
|
2764
2764
|
areOptionsEqual(prevOptions, newOptions) {
|
|
2765
|
-
|
|
2765
|
+
// PERF: Use reference equality first for fast path
|
|
2766
|
+
if (prevOptions === newOptions)
|
|
2767
|
+
return true;
|
|
2768
|
+
if (!prevOptions || !newOptions)
|
|
2766
2769
|
return prevOptions === newOptions;
|
|
2767
|
-
|
|
2768
|
-
if (prevOptions.length !== newOptions.length) {
|
|
2770
|
+
if (prevOptions.length !== newOptions.length)
|
|
2769
2771
|
return false;
|
|
2770
|
-
|
|
2772
|
+
// Only compare valueProperty for equality
|
|
2771
2773
|
for (let i = 0; i < prevOptions.length; i++) {
|
|
2772
2774
|
if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
|
|
2773
2775
|
return false;
|
|
@@ -2776,12 +2778,8 @@ class SelectSearchComponent {
|
|
|
2776
2778
|
return true;
|
|
2777
2779
|
}
|
|
2778
2780
|
initializeItems() {
|
|
2779
|
-
|
|
2780
|
-
|
|
2781
|
-
}
|
|
2782
|
-
else {
|
|
2783
|
-
this.filteredItems = [];
|
|
2784
|
-
}
|
|
2781
|
+
// PERF: Avoid unnecessary array copies
|
|
2782
|
+
this.filteredItems = this.props?.options || [];
|
|
2785
2783
|
}
|
|
2786
2784
|
syncControlValueWithSelectedItems() {
|
|
2787
2785
|
if (!this.props?.control) {
|
|
@@ -2789,14 +2787,14 @@ class SelectSearchComponent {
|
|
|
2789
2787
|
return;
|
|
2790
2788
|
}
|
|
2791
2789
|
const controlValue = this.props.control.value;
|
|
2792
|
-
// Si el control no tiene valor, limpiar seleccionados
|
|
2793
2790
|
if (controlValue === null || controlValue === undefined) {
|
|
2794
2791
|
this.selectedItems = [];
|
|
2795
2792
|
return;
|
|
2796
2793
|
}
|
|
2797
|
-
//
|
|
2794
|
+
// PERF: Use a Map for faster lookup if options are large
|
|
2798
2795
|
if (this.props.options && this.props.options.length > 0) {
|
|
2799
|
-
const
|
|
2796
|
+
const map = new Map(this.props.options.map(opt => [opt[this.valueProperty], opt]));
|
|
2797
|
+
const selectedOption = map.get(controlValue);
|
|
2800
2798
|
this.selectedItems = selectedOption ? [selectedOption] : [];
|
|
2801
2799
|
}
|
|
2802
2800
|
else {
|
|
@@ -2804,28 +2802,26 @@ class SelectSearchComponent {
|
|
|
2804
2802
|
}
|
|
2805
2803
|
}
|
|
2806
2804
|
onFilter(event) {
|
|
2807
|
-
//
|
|
2805
|
+
// If no search term, show all options
|
|
2808
2806
|
if (!event || event.trim() === '') {
|
|
2809
2807
|
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
2810
2808
|
this.changeDetector.detectChanges();
|
|
2811
2809
|
return;
|
|
2812
2810
|
}
|
|
2813
|
-
//
|
|
2811
|
+
// If no options, nothing to filter
|
|
2814
2812
|
if (!this.props?.options || this.props.options.length === 0) {
|
|
2815
2813
|
this.filteredItems = [];
|
|
2816
2814
|
this.changeDetector.detectChanges();
|
|
2817
2815
|
return;
|
|
2818
2816
|
}
|
|
2819
|
-
|
|
2820
|
-
const
|
|
2821
|
-
|
|
2822
|
-
|
|
2823
|
-
|
|
2824
|
-
|
|
2825
|
-
|
|
2826
|
-
return values.some(value => value.includes(text));
|
|
2817
|
+
// PERF: Avoid repeated replaceSpecialChars and toLowerCase for each option
|
|
2818
|
+
const search = replaceSpecialChars(event.toLowerCase());
|
|
2819
|
+
this.filteredItems = this.props.options.filter(element => {
|
|
2820
|
+
// Only use labelProperty and valueProperty for filtering (faster)
|
|
2821
|
+
const label = element[this.labelProperty] ? replaceSpecialChars(String(element[this.labelProperty]).toLowerCase()) : '';
|
|
2822
|
+
const value = element[this.valueProperty] ? replaceSpecialChars(String(element[this.valueProperty]).toLowerCase()) : '';
|
|
2823
|
+
return label.includes(search) || value.includes(search);
|
|
2827
2824
|
});
|
|
2828
|
-
this.filteredItems = filteredResults;
|
|
2829
2825
|
this.changeDetector.detectChanges();
|
|
2830
2826
|
}
|
|
2831
2827
|
onFocus() {
|
|
@@ -2862,6 +2858,10 @@ class SelectSearchComponent {
|
|
|
2862
2858
|
else {
|
|
2863
2859
|
this.selectedItems = [item];
|
|
2864
2860
|
this.cancelModal();
|
|
2861
|
+
// Reset filter and show all options when closing modal
|
|
2862
|
+
this.searchTerm = '';
|
|
2863
|
+
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
2864
|
+
this.changeDetector.detectChanges();
|
|
2865
2865
|
}
|
|
2866
2866
|
this.updateDisplayValue();
|
|
2867
2867
|
this.applyChanges();
|
|
@@ -4082,7 +4082,7 @@ class ItemListComponent {
|
|
|
4082
4082
|
}
|
|
4083
4083
|
}
|
|
4084
4084
|
</ion-list>
|
|
4085
|
-
`, isInline: true, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonListHeader, selector: "ion-list-header", inputs: ["color", "lines", "mode"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonNote, selector: "ion-note", inputs: ["color", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonAvatar, selector: "ion-avatar" }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: 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: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }] }); }
|
|
4085
|
+
`, isInline: true, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important;padding:.25rem 0rem}ion-list-header ion-label{margin-top:1rem}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"], dependencies: [{ kind: "directive", type: RouterLink, selector: "[routerLink]", inputs: ["target", "queryParams", "fragment", "queryParamsHandling", "state", "info", "relativeTo", "preserveFragment", "skipLocationChange", "replaceUrl", "routerLink"] }, { kind: "directive", type: NgClass, selector: "[ngClass]", inputs: ["class", "ngClass"] }, { kind: "component", type: IonList, selector: "ion-list", inputs: ["inset", "lines", "mode"] }, { kind: "component", type: IonListHeader, selector: "ion-list-header", inputs: ["color", "lines", "mode"] }, { kind: "component", type: IonLabel, selector: "ion-label", inputs: ["color", "mode", "position"] }, { kind: "component", type: IonNote, selector: "ion-note", inputs: ["color", "mode"] }, { kind: "component", type: IonItem, selector: "ion-item", inputs: ["button", "color", "detail", "detailIcon", "disabled", "download", "href", "lines", "mode", "rel", "routerAnimation", "routerDirection", "target", "type"] }, { kind: "component", type: IonAvatar, selector: "ion-avatar" }, { kind: "component", type: IonIcon, selector: "ion-icon", inputs: ["color", "flipRtl", "icon", "ios", "lazy", "md", "mode", "name", "sanitize", "size", "src"] }, { kind: "component", type: 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: IonSpinner, selector: "ion-spinner", inputs: ["color", "duration", "name", "paused"] }] }); }
|
|
4086
4086
|
}
|
|
4087
4087
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: ItemListComponent, decorators: [{
|
|
4088
4088
|
type: Component,
|
|
@@ -4264,7 +4264,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
4264
4264
|
}
|
|
4265
4265
|
}
|
|
4266
4266
|
</ion-list>
|
|
4267
|
-
`, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"] }]
|
|
4267
|
+
`, styles: ["ion-list{margin:0!important;border-radius:1.5rem!important;padding:.25rem 0rem}ion-list-header ion-label{margin-top:1rem}ion-label{font-family:var(--ion-default-font),Arial,sans-serif}ion-note{font-family:var(--ion-default-font),Arial,sans-serif;font-weight:400}ion-button{font-family:var(--ion-default-font),Arial,sans-serif}.shadowed{box-shadow:.1875rem .625rem .5rem #1219541a}.bordered{border:1px solid}.unread-indicator{background:var(--ion-color-primary);width:.625rem;height:.625rem;border-radius:100%;position:absolute;inset-inline-start:.75rem;top:.75rem}.action-spinner{width:3.125rem;height:1rem}\n"] }]
|
|
4268
4268
|
}], ctorParameters: () => [{ type: IconService }], propDecorators: { props: [{
|
|
4269
4269
|
type: Input
|
|
4270
4270
|
}], onClick: [{
|