valtech-components 2.0.147 → 2.0.149
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/components/molecules/select-search/select-search.component.mjs +73 -16
- package/esm2022/lib/components/organisms/form/form.component.mjs +2 -2
- package/fesm2022/valtech-components.mjs +73 -16
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/select-search/select-search.component.d.ts +4 -0
- package/package.json +1 -1
- package/src/lib/components/styles/overrides.scss +1 -1
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
|
-
import { Component, inject, Input, ViewChild } from '@angular/core';
|
|
2
|
+
import { Component, inject, Input, ViewChild, ChangeDetectorRef } from '@angular/core';
|
|
3
3
|
import { FormsModule, ReactiveFormsModule } from '@angular/forms';
|
|
4
4
|
import { IonicModule } from '@ionic/angular';
|
|
5
5
|
import { IconService } from '../../../services/icons.service';
|
|
@@ -17,10 +17,14 @@ export class SelectSearchComponent {
|
|
|
17
17
|
this.multiple = false;
|
|
18
18
|
this.placeholder = 'Seleccione una opción';
|
|
19
19
|
this.icon = inject(IconService);
|
|
20
|
+
this.changeDetector = inject(ChangeDetectorRef);
|
|
20
21
|
this.searchTerm = '';
|
|
21
22
|
this.filteredItems = [];
|
|
22
23
|
this.selectedItems = [];
|
|
23
24
|
this.displayValue = '';
|
|
25
|
+
// Usar para evitar bucles infinitos
|
|
26
|
+
this.previousOptions = [];
|
|
27
|
+
this.isProcessingChanges = false;
|
|
24
28
|
}
|
|
25
29
|
ngOnInit() {
|
|
26
30
|
this.initializeItems();
|
|
@@ -28,17 +32,57 @@ export class SelectSearchComponent {
|
|
|
28
32
|
this.updateDisplayValue();
|
|
29
33
|
}
|
|
30
34
|
ngOnChanges(changes) {
|
|
31
|
-
//
|
|
35
|
+
// Evitar bucles infinitos
|
|
36
|
+
if (this.isProcessingChanges) {
|
|
37
|
+
return;
|
|
38
|
+
}
|
|
32
39
|
if (changes['props'] && this.props?.options) {
|
|
40
|
+
// Verificar si las opciones han cambiado realmente
|
|
41
|
+
const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
|
|
42
|
+
if (optionsChanged) {
|
|
43
|
+
try {
|
|
44
|
+
this.isProcessingChanges = true;
|
|
45
|
+
this.previousOptions = [...this.props.options];
|
|
46
|
+
this.initializeItems();
|
|
47
|
+
this.syncControlValueWithSelectedItems();
|
|
48
|
+
this.updateDisplayValue();
|
|
49
|
+
}
|
|
50
|
+
finally {
|
|
51
|
+
this.isProcessingChanges = false;
|
|
52
|
+
}
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
ionViewWillEnter() {
|
|
57
|
+
if (this.isProcessingChanges) {
|
|
58
|
+
return;
|
|
59
|
+
}
|
|
60
|
+
try {
|
|
61
|
+
this.isProcessingChanges = true;
|
|
33
62
|
this.initializeItems();
|
|
34
63
|
this.syncControlValueWithSelectedItems();
|
|
35
64
|
this.updateDisplayValue();
|
|
36
65
|
}
|
|
66
|
+
finally {
|
|
67
|
+
this.isProcessingChanges = false;
|
|
68
|
+
}
|
|
37
69
|
}
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
70
|
+
// Compara si dos arrays de opciones son iguales
|
|
71
|
+
areOptionsEqual(prevOptions, newOptions) {
|
|
72
|
+
if (!prevOptions || !newOptions) {
|
|
73
|
+
return prevOptions === newOptions;
|
|
74
|
+
}
|
|
75
|
+
if (prevOptions.length !== newOptions.length) {
|
|
76
|
+
return false;
|
|
77
|
+
}
|
|
78
|
+
// Comparación simple basada en longitud y valores de ID
|
|
79
|
+
// Esto se puede mejorar según tus necesidades específicas
|
|
80
|
+
for (let i = 0; i < prevOptions.length; i++) {
|
|
81
|
+
if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
|
|
82
|
+
return false;
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
return true;
|
|
42
86
|
}
|
|
43
87
|
initializeItems() {
|
|
44
88
|
if (this.props?.options) {
|
|
@@ -49,8 +93,11 @@ export class SelectSearchComponent {
|
|
|
49
93
|
}
|
|
50
94
|
}
|
|
51
95
|
syncControlValueWithSelectedItems() {
|
|
96
|
+
if (!this.props?.control) {
|
|
97
|
+
return;
|
|
98
|
+
}
|
|
52
99
|
// Sincronizar el valor del control con los elementos seleccionados
|
|
53
|
-
if (this.props
|
|
100
|
+
if (this.props.control.value) {
|
|
54
101
|
const controlValue = this.props.control.value;
|
|
55
102
|
const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);
|
|
56
103
|
if (selectedOption) {
|
|
@@ -60,24 +107,34 @@ export class SelectSearchComponent {
|
|
|
60
107
|
this.selectedItems = [];
|
|
61
108
|
}
|
|
62
109
|
}
|
|
110
|
+
else {
|
|
111
|
+
this.selectedItems = [];
|
|
112
|
+
}
|
|
63
113
|
}
|
|
64
114
|
onFilter(event) {
|
|
65
|
-
|
|
115
|
+
// Resetear a todas las opciones disponibles si no hay texto de búsqueda
|
|
116
|
+
if (!event || event.trim() === '') {
|
|
66
117
|
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
118
|
+
this.changeDetector.detectChanges();
|
|
67
119
|
return;
|
|
68
120
|
}
|
|
121
|
+
// Si no hay opciones disponibles, no hace falta filtrar
|
|
69
122
|
if (!this.props?.options || this.props.options.length === 0) {
|
|
123
|
+
this.filteredItems = [];
|
|
124
|
+
this.changeDetector.detectChanges();
|
|
70
125
|
return;
|
|
71
126
|
}
|
|
72
127
|
const text = replaceSpecialChars(event.toLowerCase());
|
|
73
|
-
const
|
|
74
|
-
const values = Object.values(element).map((a) =>
|
|
75
|
-
|
|
76
|
-
|
|
77
|
-
|
|
78
|
-
|
|
128
|
+
const filteredResults = this.props.options.filter(element => {
|
|
129
|
+
const values = Object.values(element).map((a) => {
|
|
130
|
+
if (a === null || a === undefined)
|
|
131
|
+
return '';
|
|
132
|
+
return replaceSpecialChars(`${a}`).toLowerCase();
|
|
133
|
+
});
|
|
134
|
+
return values.some(value => value.includes(text));
|
|
79
135
|
});
|
|
80
|
-
this.filteredItems =
|
|
136
|
+
this.filteredItems = filteredResults;
|
|
137
|
+
this.changeDetector.detectChanges();
|
|
81
138
|
}
|
|
82
139
|
onFocus() {
|
|
83
140
|
console.log('onFocus');
|
|
@@ -246,4 +303,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
246
303
|
}], props: [{
|
|
247
304
|
type: Input
|
|
248
305
|
}] } });
|
|
249
|
-
//# 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 { Component, inject, Input, OnChanges, SimpleChanges, ViewChild } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { ViewWillEnter } from '@ionic/angular/standalone';\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})\nexport class SelectSearchComponent implements ViewWillEnter, OnChanges {\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  @Input() props: InputMetadata;\n\n  icon = inject(IconService);\n\n  searchTerm: string = '';\n  filteredItems: InputOption[] = [];\n  selectedItems: InputOption[] = [];\n  displayValue: string = '';\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Detectar cuando cambian las opciones\n    if (changes['props'] && this.props?.options) {\n      this.initializeItems();\n      this.syncControlValueWithSelectedItems();\n      this.updateDisplayValue();\n    }\n  }\n\n  ionViewWillEnter(): void {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\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    // Sincronizar el valor del control con los elementos seleccionados\n    if (this.props?.control?.value) {\n      const controlValue = this.props.control.value;\n      const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);\n\n      if (selectedOption) {\n        this.selectedItems = [selectedOption];\n      } else {\n        this.selectedItems = [];\n      }\n    }\n  }\n\n  onFilter(event) {\n    if (!event) {\n      this.filteredItems = this.props?.options ? [...this.props.options] : [];\n      return;\n    }\n\n    if (!this.props?.options || this.props.options.length === 0) {\n      return;\n    }\n\n    const text = replaceSpecialChars(event.toLowerCase());\n    const filter = this.props.options.filter(element => {\n      const values = Object.values(element).map((a: string) => replaceSpecialChars(`${a}`).toLocaleLowerCase());\n      if (values.some((value: string) => value.indexOf(text) > -1)) {\n        return true;\n      }\n      return false;\n    });\n\n    this.filteredItems = filter;\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(); // Evita el comportamiento predeterminado del selector\n    event.stopPropagation(); // Detiene la propagación del evento\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    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.updateValueAndValidity();\n    this.props.control.markAsDirty();\n  }\n}\n"]}
|
|
306
|
+
//# 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 { Component, inject, Input, OnChanges, SimpleChanges, ViewChild, ChangeDetectorRef } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { IonicModule, IonModal } from '@ionic/angular';\nimport { ViewWillEnter } from '@ionic/angular/standalone';\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})\nexport class SelectSearchComponent implements ViewWillEnter, OnChanges {\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  @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  // Usar para evitar bucles infinitos\n  private previousOptions: any[] = [];\n  private isProcessingChanges = false;\n\n  ngOnInit() {\n    this.initializeItems();\n    this.syncControlValueWithSelectedItems();\n    this.updateDisplayValue();\n  }\n\n  ngOnChanges(changes: SimpleChanges): void {\n    // Evitar bucles infinitos\n    if (this.isProcessingChanges) {\n      return;\n    }\n\n    if (changes['props'] && 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        try {\n          this.isProcessingChanges = true;\n          this.previousOptions = [...this.props.options];\n          this.initializeItems();\n          this.syncControlValueWithSelectedItems();\n          this.updateDisplayValue();\n        } finally {\n          this.isProcessingChanges = false;\n        }\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    } finally {\n      this.isProcessingChanges = false;\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    // Comparación simple basada en longitud y valores de ID\n    // Esto se puede mejorar según tus necesidades específicas\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      return;\n    }\n\n    // Sincronizar el valor del control con los elementos seleccionados\n    if (this.props.control.value) {\n      const controlValue = this.props.control.value;\n      const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);\n\n      if (selectedOption) {\n        this.selectedItems = [selectedOption];\n      } else {\n        this.selectedItems = [];\n      }\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(); // Evita el comportamiento predeterminado del selector\n    event.stopPropagation(); // Detiene la propagación del evento\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    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.updateValueAndValidity();\n    this.props.control.markAsDirty();\n  }\n}\n"]}
|
|
@@ -44,7 +44,7 @@ export class FormComponent {
|
|
|
44
44
|
this.form = this.fb.group(formControls);
|
|
45
45
|
this.props.sections.forEach(section => {
|
|
46
46
|
section.fields
|
|
47
|
-
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT)
|
|
47
|
+
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT || x.type === this.types.SEARCH_SELECT)
|
|
48
48
|
.forEach(field => {
|
|
49
49
|
this.trackSelectChanges(field.name);
|
|
50
50
|
});
|
|
@@ -280,4 +280,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
280
280
|
}], onSelectChange: [{
|
|
281
281
|
type: Output
|
|
282
282
|
}] } });
|
|
283
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/form/form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, ElementRef, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { FormBuilder, FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { Subscription } from 'rxjs';\nimport { isAtEnd } from '../../../shared/utils/dom';\nimport { DisplayComponent } from '../../atoms/display/display.component';\nimport { DividerComponent } from '../../atoms/divider/divider.component';\nimport { TitleComponent } from '../../atoms/title/title.component';\nimport { ButtonGroupComponent } from '../../molecules/button-group/button-group.component';\nimport { CheckInputComponent } from '../../molecules/check-input/check-input.component';\nimport { CommentInputComponent } from '../../molecules/comment-input/comment-input.component';\nimport { DateInputComponent } from '../../molecules/date-input/date-input.component';\nimport { EmailInputComponent } from '../../molecules/email-input/email-input.component';\nimport { FileInputComponent } from '../../molecules/file-input/file-input.component';\nimport { HintComponent } from '../../molecules/hint/hint.component';\nimport { HourInputComponent } from '../../molecules/hour-input/hour-input.component';\nimport { NumberInputComponent } from '../../molecules/number-input/number-input.component';\nimport { PasswordInputComponent } from '../../molecules/password-input/password-input.component';\nimport { PinInputComponent } from '../../molecules/pin-input/pin-input.component';\nimport { RadioInputComponent } from '../../molecules/radio-input/radio-input.component';\nimport { SearchSelectorComponent } from '../../molecules/select-input/select-input.component';\nimport { SelectSearchComponent } from '../../molecules/select-search/select-search.component';\nimport { TextInputComponent } from '../../molecules/text-input/text-input.component';\nimport { ButtonMetadata, ComponentStates, FormMetadata, FormSubmit, InputMetadata, InputType } from '../../types';\n\n@Component({\n  selector: 'val-form',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    DisplayComponent,\n    TitleComponent,\n    TextInputComponent,\n    CheckInputComponent,\n    ButtonGroupComponent,\n    DividerComponent,\n    HintComponent,\n    CommentInputComponent,\n    DateInputComponent,\n    FileInputComponent,\n    HourInputComponent,\n    EmailInputComponent,\n    NumberInputComponent,\n    RadioInputComponent,\n    PasswordInputComponent,\n    PinInputComponent,\n    SelectSearchComponent,\n    SearchSelectorComponent,\n  ],\n  template: `\n    <div class=\"container\">\n      <form [formGroup]=\"form\">\n        <val-display\n          *ngIf=\"props.name\"\n          [props]=\"{\n            content: props.name,\n            color: 'dark',\n            size: 'large',\n          }\"\n        ></val-display>\n        <div class=\"section\" *ngFor=\"let s of props.sections\">\n          <val-title [props]=\"{ content: s.name, size: 'large', color: '', bold: false }\"></val-title>\n          <div class=\"input\" *ngFor=\"let f of s.fields\">\n            <val-title [props]=\"{ content: f.label, size: 'small', color: 'dark', bold: false }\"></val-title>\n            <ng-container *ngIf=\"f.type === types.TEXT\">\n              <val-text-input [props]=\"getFieldProp(f)\"></val-text-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.CHECK\">\n              <val-check-input></val-check-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.COMMENT\">\n              <val-comment-input [props]=\"getFieldProp(f)\"></val-comment-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.DATE\">\n              <val-date-input [props]=\"getFieldProp(f)\"></val-date-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.EMAIL\">\n              <val-email-input [props]=\"getFieldProp(f)\"></val-email-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.FILE\">\n              <val-file-input [props]=\"getFieldProp(f)\"></val-file-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.HOUR\">\n              <val-hour-input [props]=\"getFieldProp(f)\"></val-hour-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.NUMBER\">\n              <val-number-input [props]=\"getFieldProp(f)\"></val-number-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PASSWORD\">\n              <val-password-input [props]=\"getFieldProp(f)\"></val-password-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PIN_CODE\">\n              <val-pin-input [props]=\"getFieldProp(f)\"></val-pin-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.RADIO\">\n              <val-radio-input [props]=\"getFieldProp(f)\"></val-radio-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SELECT\">\n              <val-select-input [props]=\"getFieldProp(f)\"></val-select-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SEARCH_SELECT\">\n              <val-select-search [props]=\"getFieldProp(f)\"></val-select-search>\n            </ng-container>\n            <val-hint [props]=\"getFieldProp(f)\"></val-hint>\n          </div>\n          <val-divider [props]=\"{ fill: 'solid', size: 'medium', color: 'medium' }\"></val-divider>\n          <ng-content></ng-content>\n        </div>\n        <val-button-group\n          [props]=\"{ buttons: actions, position: 'center', columned: false }\"\n          (onClick)=\"submitHandler($event)\"\n        ></val-button-group>\n      </form>\n    </div>\n  `,\n  styleUrls: ['./form.component.scss'],\n})\nexport class FormComponent implements OnInit {\n  @Input()\n  props: FormMetadata;\n\n  @Output()\n  onSubmit = new EventEmitter<FormSubmit>();\n\n  @Output()\n  onInvalid = new EventEmitter();\n\n  @Output()\n  onSelectChange = new EventEmitter<{ field: string; value: any }>();\n\n  form: FormGroup;\n  types = InputType;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private fb: FormBuilder,\n    private elementRef: ElementRef\n  ) {}\n\n  ngOnInit() {\n    const formControls = {};\n    this.props.sections.forEach(section => {\n      section.fields.forEach(field => {\n        formControls[field.name] = [undefined, field.validators || []];\n      });\n    });\n    this.form = this.fb.group(formControls);\n    this.props.sections.forEach(section => {\n      section.fields\n        .filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT)\n        .forEach(field => {\n          this.trackSelectChanges(field.name);\n        });\n    });\n  }\n\n  ngOnDestroy() {\n    // Cleanup subscriptions to avoid memory leaks\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n  }\n\n  trackSelectChanges(fieldName: string) {\n    const control = this.getControl(fieldName);\n    const subscription = control.valueChanges.subscribe(value => {\n      this.onSelectChange.emit({ field: fieldName, value });\n    });\n    this.subscriptions.push(subscription);\n  }\n\n  async submitHandler(token?: string) {\n    this.onSubmit.emit({ fields: this.form.getRawValue(), token });\n  }\n\n  getControl(field: string): FormControl {\n    return this.Form.get(field) as FormControl;\n  }\n\n  getFieldProp(field: InputMetadata): InputMetadata {\n    field.token;\n    if (!field.token) {\n      field.token = `input-${field.type}-${field.name}`;\n    }\n    const control = this.getControl(field.name);\n    if (field.state === ComponentStates.DISABLED) {\n      control.disable();\n    } else {\n      control.enable();\n    }\n    return {\n      ...field,\n      control,\n    };\n  }\n\n  get isAtEndOfForm(): boolean {\n    return isAtEnd(this.elementRef);\n  }\n\n  get Form(): FormGroup {\n    return this.form;\n  }\n\n  get FormState(): { form: FormGroup; data: FormMetadata } {\n    return {\n      form: this.Form,\n      data: this.props,\n    };\n  }\n\n  get actions(): ButtonMetadata[] {\n    if (!this.form) {\n      return [];\n    }\n\n    if (this.form.valid) {\n      this.props.actions.state = ComponentStates.ENABLED;\n    }\n\n    if (this.props.state === ComponentStates.WORKING) {\n      this.props.actions.state = ComponentStates.WORKING;\n    }\n\n    if (this.props.state === ComponentStates.ENABLED) {\n      this.props.actions.state = ComponentStates.ENABLED;\n    }\n\n    if (this.props.state === ComponentStates.DISABLED) {\n      this.props.actions.state = ComponentStates.DISABLED;\n    }\n\n    return [this.props.actions];\n  }\n}\n"]}
|
|
283
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../../../projects/valtech-components/src/lib/components/organisms/form/form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, ElementRef, EventEmitter, Input, OnInit, Output } from '@angular/core';\nimport { FormBuilder, FormControl, FormGroup, ReactiveFormsModule } from '@angular/forms';\nimport { Subscription } from 'rxjs';\nimport { isAtEnd } from '../../../shared/utils/dom';\nimport { DisplayComponent } from '../../atoms/display/display.component';\nimport { DividerComponent } from '../../atoms/divider/divider.component';\nimport { TitleComponent } from '../../atoms/title/title.component';\nimport { ButtonGroupComponent } from '../../molecules/button-group/button-group.component';\nimport { CheckInputComponent } from '../../molecules/check-input/check-input.component';\nimport { CommentInputComponent } from '../../molecules/comment-input/comment-input.component';\nimport { DateInputComponent } from '../../molecules/date-input/date-input.component';\nimport { EmailInputComponent } from '../../molecules/email-input/email-input.component';\nimport { FileInputComponent } from '../../molecules/file-input/file-input.component';\nimport { HintComponent } from '../../molecules/hint/hint.component';\nimport { HourInputComponent } from '../../molecules/hour-input/hour-input.component';\nimport { NumberInputComponent } from '../../molecules/number-input/number-input.component';\nimport { PasswordInputComponent } from '../../molecules/password-input/password-input.component';\nimport { PinInputComponent } from '../../molecules/pin-input/pin-input.component';\nimport { RadioInputComponent } from '../../molecules/radio-input/radio-input.component';\nimport { SearchSelectorComponent } from '../../molecules/select-input/select-input.component';\nimport { SelectSearchComponent } from '../../molecules/select-search/select-search.component';\nimport { TextInputComponent } from '../../molecules/text-input/text-input.component';\nimport { ButtonMetadata, ComponentStates, FormMetadata, FormSubmit, InputMetadata, InputType } from '../../types';\n\n@Component({\n  selector: 'val-form',\n  standalone: true,\n  imports: [\n    CommonModule,\n    ReactiveFormsModule,\n    DisplayComponent,\n    TitleComponent,\n    TextInputComponent,\n    CheckInputComponent,\n    ButtonGroupComponent,\n    DividerComponent,\n    HintComponent,\n    CommentInputComponent,\n    DateInputComponent,\n    FileInputComponent,\n    HourInputComponent,\n    EmailInputComponent,\n    NumberInputComponent,\n    RadioInputComponent,\n    PasswordInputComponent,\n    PinInputComponent,\n    SelectSearchComponent,\n    SearchSelectorComponent,\n  ],\n  template: `\n    <div class=\"container\">\n      <form [formGroup]=\"form\">\n        <val-display\n          *ngIf=\"props.name\"\n          [props]=\"{\n            content: props.name,\n            color: 'dark',\n            size: 'large',\n          }\"\n        ></val-display>\n        <div class=\"section\" *ngFor=\"let s of props.sections\">\n          <val-title [props]=\"{ content: s.name, size: 'large', color: '', bold: false }\"></val-title>\n          <div class=\"input\" *ngFor=\"let f of s.fields\">\n            <val-title [props]=\"{ content: f.label, size: 'small', color: 'dark', bold: false }\"></val-title>\n            <ng-container *ngIf=\"f.type === types.TEXT\">\n              <val-text-input [props]=\"getFieldProp(f)\"></val-text-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.CHECK\">\n              <val-check-input></val-check-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.COMMENT\">\n              <val-comment-input [props]=\"getFieldProp(f)\"></val-comment-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.DATE\">\n              <val-date-input [props]=\"getFieldProp(f)\"></val-date-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.EMAIL\">\n              <val-email-input [props]=\"getFieldProp(f)\"></val-email-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.FILE\">\n              <val-file-input [props]=\"getFieldProp(f)\"></val-file-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.HOUR\">\n              <val-hour-input [props]=\"getFieldProp(f)\"></val-hour-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.NUMBER\">\n              <val-number-input [props]=\"getFieldProp(f)\"></val-number-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PASSWORD\">\n              <val-password-input [props]=\"getFieldProp(f)\"></val-password-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PIN_CODE\">\n              <val-pin-input [props]=\"getFieldProp(f)\"></val-pin-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.RADIO\">\n              <val-radio-input [props]=\"getFieldProp(f)\"></val-radio-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SELECT\">\n              <val-select-input [props]=\"getFieldProp(f)\"></val-select-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SEARCH_SELECT\">\n              <val-select-search [props]=\"getFieldProp(f)\"></val-select-search>\n            </ng-container>\n            <val-hint [props]=\"getFieldProp(f)\"></val-hint>\n          </div>\n          <val-divider [props]=\"{ fill: 'solid', size: 'medium', color: 'medium' }\"></val-divider>\n          <ng-content></ng-content>\n        </div>\n        <val-button-group\n          [props]=\"{ buttons: actions, position: 'center', columned: false }\"\n          (onClick)=\"submitHandler($event)\"\n        ></val-button-group>\n      </form>\n    </div>\n  `,\n  styleUrls: ['./form.component.scss'],\n})\nexport class FormComponent implements OnInit {\n  @Input()\n  props: FormMetadata;\n\n  @Output()\n  onSubmit = new EventEmitter<FormSubmit>();\n\n  @Output()\n  onInvalid = new EventEmitter();\n\n  @Output()\n  onSelectChange = new EventEmitter<{ field: string; value: any }>();\n\n  form: FormGroup;\n  types = InputType;\n  private subscriptions: Subscription[] = [];\n\n  constructor(\n    private fb: FormBuilder,\n    private elementRef: ElementRef\n  ) {}\n\n  ngOnInit() {\n    const formControls = {};\n    this.props.sections.forEach(section => {\n      section.fields.forEach(field => {\n        formControls[field.name] = [undefined, field.validators || []];\n      });\n    });\n    this.form = this.fb.group(formControls);\n    this.props.sections.forEach(section => {\n      section.fields\n        .filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT || x.type === this.types.SEARCH_SELECT)\n        .forEach(field => {\n          this.trackSelectChanges(field.name);\n        });\n    });\n  }\n\n  ngOnDestroy() {\n    // Cleanup subscriptions to avoid memory leaks\n    this.subscriptions.forEach(sub => sub.unsubscribe());\n  }\n\n  trackSelectChanges(fieldName: string) {\n    const control = this.getControl(fieldName);\n    const subscription = control.valueChanges.subscribe(value => {\n      this.onSelectChange.emit({ field: fieldName, value });\n    });\n    this.subscriptions.push(subscription);\n  }\n\n  async submitHandler(token?: string) {\n    this.onSubmit.emit({ fields: this.form.getRawValue(), token });\n  }\n\n  getControl(field: string): FormControl {\n    return this.Form.get(field) as FormControl;\n  }\n\n  getFieldProp(field: InputMetadata): InputMetadata {\n    field.token;\n    if (!field.token) {\n      field.token = `input-${field.type}-${field.name}`;\n    }\n    const control = this.getControl(field.name);\n    if (field.state === ComponentStates.DISABLED) {\n      control.disable();\n    } else {\n      control.enable();\n    }\n    return {\n      ...field,\n      control,\n    };\n  }\n\n  get isAtEndOfForm(): boolean {\n    return isAtEnd(this.elementRef);\n  }\n\n  get Form(): FormGroup {\n    return this.form;\n  }\n\n  get FormState(): { form: FormGroup; data: FormMetadata } {\n    return {\n      form: this.Form,\n      data: this.props,\n    };\n  }\n\n  get actions(): ButtonMetadata[] {\n    if (!this.form) {\n      return [];\n    }\n\n    if (this.form.valid) {\n      this.props.actions.state = ComponentStates.ENABLED;\n    }\n\n    if (this.props.state === ComponentStates.WORKING) {\n      this.props.actions.state = ComponentStates.WORKING;\n    }\n\n    if (this.props.state === ComponentStates.ENABLED) {\n      this.props.actions.state = ComponentStates.ENABLED;\n    }\n\n    if (this.props.state === ComponentStates.DISABLED) {\n      this.props.actions.state = ComponentStates.DISABLED;\n    }\n\n    return [this.props.actions];\n  }\n}\n"]}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import * as i0 from '@angular/core';
|
|
2
|
-
import { EventEmitter, Component, Input, Output, Injectable, ViewChild, inject, InjectionToken, Inject } from '@angular/core';
|
|
2
|
+
import { EventEmitter, Component, Input, Output, Injectable, ViewChild, inject, ChangeDetectorRef, InjectionToken, Inject } from '@angular/core';
|
|
3
3
|
import { IonAvatar, IonCard, IonIcon, IonButton, IonSpinner, IonText, IonProgressBar, IonCardContent, IonCardHeader, IonCardTitle, IonCardSubtitle, IonCheckbox, IonButtons, IonTextarea, IonDatetime, IonDatetimeButton, IonModal, IonInput, IonRadioGroup, IonRadio, IonSearchbar, IonSelect, IonSelectOption, IonToolbar, IonTitle, IonFooter, IonHeader, IonList, IonListHeader, IonLabel, IonNote, IonItem, IonContent } from '@ionic/angular/standalone';
|
|
4
4
|
import * as i1 from '@angular/common';
|
|
5
5
|
import { CommonModule, NgStyle, NgIf, NgFor, NgClass } from '@angular/common';
|
|
@@ -2132,10 +2132,14 @@ class SelectSearchComponent {
|
|
|
2132
2132
|
this.multiple = false;
|
|
2133
2133
|
this.placeholder = 'Seleccione una opción';
|
|
2134
2134
|
this.icon = inject(IconService);
|
|
2135
|
+
this.changeDetector = inject(ChangeDetectorRef);
|
|
2135
2136
|
this.searchTerm = '';
|
|
2136
2137
|
this.filteredItems = [];
|
|
2137
2138
|
this.selectedItems = [];
|
|
2138
2139
|
this.displayValue = '';
|
|
2140
|
+
// Usar para evitar bucles infinitos
|
|
2141
|
+
this.previousOptions = [];
|
|
2142
|
+
this.isProcessingChanges = false;
|
|
2139
2143
|
}
|
|
2140
2144
|
ngOnInit() {
|
|
2141
2145
|
this.initializeItems();
|
|
@@ -2143,17 +2147,57 @@ class SelectSearchComponent {
|
|
|
2143
2147
|
this.updateDisplayValue();
|
|
2144
2148
|
}
|
|
2145
2149
|
ngOnChanges(changes) {
|
|
2146
|
-
//
|
|
2150
|
+
// Evitar bucles infinitos
|
|
2151
|
+
if (this.isProcessingChanges) {
|
|
2152
|
+
return;
|
|
2153
|
+
}
|
|
2147
2154
|
if (changes['props'] && this.props?.options) {
|
|
2155
|
+
// Verificar si las opciones han cambiado realmente
|
|
2156
|
+
const optionsChanged = !this.areOptionsEqual(this.previousOptions, this.props.options);
|
|
2157
|
+
if (optionsChanged) {
|
|
2158
|
+
try {
|
|
2159
|
+
this.isProcessingChanges = true;
|
|
2160
|
+
this.previousOptions = [...this.props.options];
|
|
2161
|
+
this.initializeItems();
|
|
2162
|
+
this.syncControlValueWithSelectedItems();
|
|
2163
|
+
this.updateDisplayValue();
|
|
2164
|
+
}
|
|
2165
|
+
finally {
|
|
2166
|
+
this.isProcessingChanges = false;
|
|
2167
|
+
}
|
|
2168
|
+
}
|
|
2169
|
+
}
|
|
2170
|
+
}
|
|
2171
|
+
ionViewWillEnter() {
|
|
2172
|
+
if (this.isProcessingChanges) {
|
|
2173
|
+
return;
|
|
2174
|
+
}
|
|
2175
|
+
try {
|
|
2176
|
+
this.isProcessingChanges = true;
|
|
2148
2177
|
this.initializeItems();
|
|
2149
2178
|
this.syncControlValueWithSelectedItems();
|
|
2150
2179
|
this.updateDisplayValue();
|
|
2151
2180
|
}
|
|
2181
|
+
finally {
|
|
2182
|
+
this.isProcessingChanges = false;
|
|
2183
|
+
}
|
|
2152
2184
|
}
|
|
2153
|
-
|
|
2154
|
-
|
|
2155
|
-
|
|
2156
|
-
|
|
2185
|
+
// Compara si dos arrays de opciones son iguales
|
|
2186
|
+
areOptionsEqual(prevOptions, newOptions) {
|
|
2187
|
+
if (!prevOptions || !newOptions) {
|
|
2188
|
+
return prevOptions === newOptions;
|
|
2189
|
+
}
|
|
2190
|
+
if (prevOptions.length !== newOptions.length) {
|
|
2191
|
+
return false;
|
|
2192
|
+
}
|
|
2193
|
+
// Comparación simple basada en longitud y valores de ID
|
|
2194
|
+
// Esto se puede mejorar según tus necesidades específicas
|
|
2195
|
+
for (let i = 0; i < prevOptions.length; i++) {
|
|
2196
|
+
if (prevOptions[i][this.valueProperty] !== newOptions[i][this.valueProperty]) {
|
|
2197
|
+
return false;
|
|
2198
|
+
}
|
|
2199
|
+
}
|
|
2200
|
+
return true;
|
|
2157
2201
|
}
|
|
2158
2202
|
initializeItems() {
|
|
2159
2203
|
if (this.props?.options) {
|
|
@@ -2164,8 +2208,11 @@ class SelectSearchComponent {
|
|
|
2164
2208
|
}
|
|
2165
2209
|
}
|
|
2166
2210
|
syncControlValueWithSelectedItems() {
|
|
2211
|
+
if (!this.props?.control) {
|
|
2212
|
+
return;
|
|
2213
|
+
}
|
|
2167
2214
|
// Sincronizar el valor del control con los elementos seleccionados
|
|
2168
|
-
if (this.props
|
|
2215
|
+
if (this.props.control.value) {
|
|
2169
2216
|
const controlValue = this.props.control.value;
|
|
2170
2217
|
const selectedOption = this.props.options?.find(opt => opt[this.valueProperty] === controlValue);
|
|
2171
2218
|
if (selectedOption) {
|
|
@@ -2175,24 +2222,34 @@ class SelectSearchComponent {
|
|
|
2175
2222
|
this.selectedItems = [];
|
|
2176
2223
|
}
|
|
2177
2224
|
}
|
|
2225
|
+
else {
|
|
2226
|
+
this.selectedItems = [];
|
|
2227
|
+
}
|
|
2178
2228
|
}
|
|
2179
2229
|
onFilter(event) {
|
|
2180
|
-
|
|
2230
|
+
// Resetear a todas las opciones disponibles si no hay texto de búsqueda
|
|
2231
|
+
if (!event || event.trim() === '') {
|
|
2181
2232
|
this.filteredItems = this.props?.options ? [...this.props.options] : [];
|
|
2233
|
+
this.changeDetector.detectChanges();
|
|
2182
2234
|
return;
|
|
2183
2235
|
}
|
|
2236
|
+
// Si no hay opciones disponibles, no hace falta filtrar
|
|
2184
2237
|
if (!this.props?.options || this.props.options.length === 0) {
|
|
2238
|
+
this.filteredItems = [];
|
|
2239
|
+
this.changeDetector.detectChanges();
|
|
2185
2240
|
return;
|
|
2186
2241
|
}
|
|
2187
2242
|
const text = replaceSpecialChars(event.toLowerCase());
|
|
2188
|
-
const
|
|
2189
|
-
const values = Object.values(element).map((a) =>
|
|
2190
|
-
|
|
2191
|
-
|
|
2192
|
-
|
|
2193
|
-
|
|
2243
|
+
const filteredResults = this.props.options.filter(element => {
|
|
2244
|
+
const values = Object.values(element).map((a) => {
|
|
2245
|
+
if (a === null || a === undefined)
|
|
2246
|
+
return '';
|
|
2247
|
+
return replaceSpecialChars(`${a}`).toLowerCase();
|
|
2248
|
+
});
|
|
2249
|
+
return values.some(value => value.includes(text));
|
|
2194
2250
|
});
|
|
2195
|
-
this.filteredItems =
|
|
2251
|
+
this.filteredItems = filteredResults;
|
|
2252
|
+
this.changeDetector.detectChanges();
|
|
2196
2253
|
}
|
|
2197
2254
|
onFocus() {
|
|
2198
2255
|
console.log('onFocus');
|
|
@@ -2839,7 +2896,7 @@ class FormComponent {
|
|
|
2839
2896
|
this.form = this.fb.group(formControls);
|
|
2840
2897
|
this.props.sections.forEach(section => {
|
|
2841
2898
|
section.fields
|
|
2842
|
-
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT)
|
|
2899
|
+
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT || x.type === this.types.SEARCH_SELECT)
|
|
2843
2900
|
.forEach(field => {
|
|
2844
2901
|
this.trackSelectChanges(field.name);
|
|
2845
2902
|
});
|