valtech-components 2.0.397 → 2.0.399
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/multi-select-search/multi-select-search.component.mjs +373 -0
- package/esm2022/lib/components/organisms/form/form.component.mjs +75 -132
- package/fesm2022/valtech-components.mjs +434 -131
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/multi-select-search/multi-select-search.component.d.ts +48 -0
- package/lib/components/organisms/article/article.component.d.ts +1 -1
- package/lib/components/organisms/form/form.component.d.ts +0 -14
- package/package.json +1 -1
|
@@ -1,7 +1,6 @@
|
|
|
1
1
|
import { CommonModule } from '@angular/common';
|
|
2
2
|
import { Component, EventEmitter, Input, Output } from '@angular/core';
|
|
3
3
|
import { ReactiveFormsModule } from '@angular/forms';
|
|
4
|
-
import { distinctUntilChanged } from 'rxjs/operators';
|
|
5
4
|
import { isAtEnd } from '../../../shared/utils/dom';
|
|
6
5
|
import { DisplayComponent } from '../../atoms/display/display.component';
|
|
7
6
|
import { DividerComponent } from '../../atoms/divider/divider.component';
|
|
@@ -21,6 +20,7 @@ import { PinInputComponent } from '../../molecules/pin-input/pin-input.component
|
|
|
21
20
|
import { RadioInputComponent } from '../../molecules/radio-input/radio-input.component';
|
|
22
21
|
import { SearchSelectorComponent } from '../../molecules/select-input/select-input.component';
|
|
23
22
|
import { SelectSearchComponent } from '../../molecules/select-search/select-search.component';
|
|
23
|
+
import { MultiSelectSearchComponent } from '../../molecules/multi-select-search/multi-select-search.component';
|
|
24
24
|
import { TextInputComponent } from '../../molecules/text-input/text-input.component';
|
|
25
25
|
import { ComponentStates, InputType } from '../../types';
|
|
26
26
|
import * as i0 from "@angular/core";
|
|
@@ -35,16 +35,8 @@ export class FormComponent {
|
|
|
35
35
|
this.onSelectChange = new EventEmitter();
|
|
36
36
|
this.types = InputType;
|
|
37
37
|
this.subscriptions = [];
|
|
38
|
-
// Cache processed field properties
|
|
39
|
-
this.processedSections = [];
|
|
40
|
-
this.previousValues = new Map();
|
|
41
38
|
}
|
|
42
39
|
ngOnInit() {
|
|
43
|
-
this.createForm();
|
|
44
|
-
this.processAllSections();
|
|
45
|
-
this.setupChangeTracking();
|
|
46
|
-
}
|
|
47
|
-
createForm() {
|
|
48
40
|
const formControls = {};
|
|
49
41
|
this.props.sections.forEach(section => {
|
|
50
42
|
section.fields.forEach(field => {
|
|
@@ -59,27 +51,36 @@ export class FormComponent {
|
|
|
59
51
|
});
|
|
60
52
|
});
|
|
61
53
|
this.form = this.fb.group(formControls);
|
|
54
|
+
this.props.sections.forEach(section => {
|
|
55
|
+
section.fields
|
|
56
|
+
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT || x.type === this.types.SEARCH_SELECT || x.type === this.types.MULTI_SELECT)
|
|
57
|
+
.forEach(field => {
|
|
58
|
+
this.trackSelectChanges(field.name);
|
|
59
|
+
});
|
|
60
|
+
});
|
|
62
61
|
}
|
|
63
|
-
|
|
64
|
-
|
|
65
|
-
this.
|
|
66
|
-
|
|
67
|
-
|
|
68
|
-
|
|
62
|
+
ngOnDestroy() {
|
|
63
|
+
// Cleanup subscriptions to avoid memory leaks
|
|
64
|
+
this.subscriptions.forEach(sub => sub.unsubscribe());
|
|
65
|
+
}
|
|
66
|
+
trackSelectChanges(fieldName) {
|
|
67
|
+
const control = this.getControl(fieldName);
|
|
68
|
+
const subscription = control.valueChanges.subscribe(value => {
|
|
69
|
+
this.onSelectChange.emit({ field: fieldName, value });
|
|
70
|
+
});
|
|
71
|
+
this.subscriptions.push(subscription);
|
|
69
72
|
}
|
|
70
|
-
|
|
71
|
-
|
|
73
|
+
async submitHandler(token) {
|
|
74
|
+
this.onSubmit.emit({ fields: this.form.getRawValue(), token });
|
|
75
|
+
}
|
|
76
|
+
getControl(field) {
|
|
77
|
+
return this.Form.get(field);
|
|
78
|
+
}
|
|
79
|
+
getFieldProp(field) {
|
|
80
|
+
field.token;
|
|
72
81
|
if (!field.token) {
|
|
73
82
|
field.token = `input-${field.type}-${field.name}`;
|
|
74
83
|
}
|
|
75
|
-
// Debug: verificar opciones para select fields - SOLO UNA VEZ
|
|
76
|
-
if (field.type === this.types.SEARCH_SELECT || field.type === this.types.MULTI_SELECT || field.type === this.types.MULTI_SELECT_SIMPLE) {
|
|
77
|
-
console.log(`[FormComponent] Processing field ${field.name} options:`, {
|
|
78
|
-
type: field.type,
|
|
79
|
-
optionsCount: field.options?.length || 0,
|
|
80
|
-
options: field.options?.slice(0, 2) || []
|
|
81
|
-
});
|
|
82
|
-
}
|
|
83
84
|
if (field.type === this.types.NUMBER_FROM_TO) {
|
|
84
85
|
const fromControl = this.getControl(`${field.name}_from`);
|
|
85
86
|
const toControl = this.getControl(`${field.name}_to`);
|
|
@@ -91,11 +92,12 @@ export class FormComponent {
|
|
|
91
92
|
fromControl.enable();
|
|
92
93
|
toControl.enable();
|
|
93
94
|
}
|
|
94
|
-
|
|
95
|
-
|
|
96
|
-
|
|
97
|
-
|
|
98
|
-
|
|
95
|
+
return {
|
|
96
|
+
...field,
|
|
97
|
+
fromControl,
|
|
98
|
+
toControl,
|
|
99
|
+
control: undefined, // Remove control for NUMBER_FROM_TO fields
|
|
100
|
+
};
|
|
99
101
|
}
|
|
100
102
|
else {
|
|
101
103
|
const control = this.getControl(field.name);
|
|
@@ -105,65 +107,12 @@ export class FormComponent {
|
|
|
105
107
|
else {
|
|
106
108
|
control.enable();
|
|
107
109
|
}
|
|
108
|
-
|
|
109
|
-
|
|
110
|
-
|
|
110
|
+
return {
|
|
111
|
+
...field,
|
|
112
|
+
control,
|
|
113
|
+
};
|
|
111
114
|
}
|
|
112
115
|
}
|
|
113
|
-
setupChangeTracking() {
|
|
114
|
-
this.props.sections.forEach(section => {
|
|
115
|
-
section.fields
|
|
116
|
-
.filter(x => x.type === this.types.SELECT || x.type === this.types.TEXT || x.type === this.types.SEARCH_SELECT || x.type === this.types.MULTI_SELECT || x.type === this.types.MULTI_SELECT_SIMPLE)
|
|
117
|
-
.forEach(field => {
|
|
118
|
-
console.log(`[FormComponent] Tracking changes for field: ${field.name}, type: ${field.type}`);
|
|
119
|
-
this.trackSelectChanges(field.name);
|
|
120
|
-
});
|
|
121
|
-
});
|
|
122
|
-
}
|
|
123
|
-
ngOnDestroy() {
|
|
124
|
-
// Cleanup subscriptions to avoid memory leaks
|
|
125
|
-
this.subscriptions.forEach(sub => sub.unsubscribe());
|
|
126
|
-
}
|
|
127
|
-
trackSelectChanges(fieldName) {
|
|
128
|
-
const control = this.getControl(fieldName);
|
|
129
|
-
// Initialize previous value
|
|
130
|
-
this.previousValues.set(fieldName, control.value);
|
|
131
|
-
const subscription = control.valueChanges
|
|
132
|
-
.pipe(distinctUntilChanged()) // Evitar valores duplicados
|
|
133
|
-
.subscribe(value => {
|
|
134
|
-
console.log(`[FormComponent] valueChanges for ${fieldName}:`, {
|
|
135
|
-
field: fieldName,
|
|
136
|
-
newValue: value,
|
|
137
|
-
previousValue: this.previousValues.get(fieldName),
|
|
138
|
-
willEmit: this.previousValues.get(fieldName) !== value
|
|
139
|
-
});
|
|
140
|
-
// Evitar loops - solo emitir si el valor realmente cambió
|
|
141
|
-
const previousValue = this.previousValues.get(fieldName);
|
|
142
|
-
if (previousValue !== value) {
|
|
143
|
-
this.previousValues.set(fieldName, value);
|
|
144
|
-
console.log(`[FormComponent] EMITTING onSelectChange:`, { field: fieldName, value });
|
|
145
|
-
this.onSelectChange.emit({ field: fieldName, value });
|
|
146
|
-
}
|
|
147
|
-
else {
|
|
148
|
-
console.log(`[FormComponent] SKIPPING emit - same value for ${fieldName}`);
|
|
149
|
-
}
|
|
150
|
-
});
|
|
151
|
-
this.subscriptions.push(subscription);
|
|
152
|
-
}
|
|
153
|
-
async submitHandler(token) {
|
|
154
|
-
this.onSubmit.emit({ fields: this.form.getRawValue(), token });
|
|
155
|
-
}
|
|
156
|
-
getControl(field) {
|
|
157
|
-
return this.Form.get(field);
|
|
158
|
-
}
|
|
159
|
-
/**
|
|
160
|
-
* @deprecated This method is now only used internally.
|
|
161
|
-
* Use processedSections property instead to avoid multiple calls.
|
|
162
|
-
*/
|
|
163
|
-
getFieldProp(field) {
|
|
164
|
-
console.warn('[FormComponent] getFieldProp is deprecated and should not be called directly');
|
|
165
|
-
return this.processField(field);
|
|
166
|
-
}
|
|
167
116
|
get isAtEndOfForm() {
|
|
168
117
|
return isAtEnd(this.elementRef);
|
|
169
118
|
}
|
|
@@ -176,19 +125,6 @@ export class FormComponent {
|
|
|
176
125
|
data: this.props,
|
|
177
126
|
};
|
|
178
127
|
}
|
|
179
|
-
// Helper method to reset field value without triggering change event
|
|
180
|
-
resetField(fieldName, emitEvent = false) {
|
|
181
|
-
const control = this.getControl(fieldName);
|
|
182
|
-
const currentValue = control.value;
|
|
183
|
-
console.log(`[FormComponent] Resetting field ${fieldName}:`, { currentValue, emitEvent });
|
|
184
|
-
// Update our cache to avoid triggering change event
|
|
185
|
-
if (!emitEvent) {
|
|
186
|
-
this.previousValues.set(fieldName, '');
|
|
187
|
-
}
|
|
188
|
-
control.setValue('', { emitEvent });
|
|
189
|
-
control.markAsPristine();
|
|
190
|
-
control.markAsUntouched();
|
|
191
|
-
}
|
|
192
128
|
get actions() {
|
|
193
129
|
if (!this.form) {
|
|
194
130
|
return [];
|
|
@@ -219,53 +155,56 @@ export class FormComponent {
|
|
|
219
155
|
size: 'large',
|
|
220
156
|
}"
|
|
221
157
|
></val-display>
|
|
222
|
-
<div class="section" *ngFor="let s of
|
|
158
|
+
<div class="section" *ngFor="let s of props.sections">
|
|
223
159
|
<val-title [props]="{ content: s.name, size: 'large', color: '', bold: false }"></val-title>
|
|
224
160
|
<div class="input" *ngFor="let f of s.fields">
|
|
225
161
|
<val-title [props]="{ content: f.label, size: 'small', color: 'dark', bold: false }"></val-title>
|
|
226
162
|
<ng-container *ngIf="f.type === types.TEXT">
|
|
227
|
-
<val-text-input [props]="f"></val-text-input>
|
|
163
|
+
<val-text-input [props]="getFieldProp(f)"></val-text-input>
|
|
228
164
|
</ng-container>
|
|
229
165
|
<ng-container *ngIf="f.type === types.CHECK">
|
|
230
166
|
<val-check-input></val-check-input>
|
|
231
167
|
</ng-container>
|
|
232
168
|
<ng-container *ngIf="f.type === types.COMMENT">
|
|
233
|
-
<val-comment-input [props]="f"></val-comment-input>
|
|
169
|
+
<val-comment-input [props]="getFieldProp(f)"></val-comment-input>
|
|
234
170
|
</ng-container>
|
|
235
171
|
<ng-container *ngIf="f.type === types.DATE">
|
|
236
|
-
<val-date-input [props]="f"></val-date-input>
|
|
172
|
+
<val-date-input [props]="getFieldProp(f)"></val-date-input>
|
|
237
173
|
</ng-container>
|
|
238
174
|
<ng-container *ngIf="f.type === types.EMAIL">
|
|
239
|
-
<val-email-input [props]="f"></val-email-input>
|
|
175
|
+
<val-email-input [props]="getFieldProp(f)"></val-email-input>
|
|
240
176
|
</ng-container>
|
|
241
177
|
<ng-container *ngIf="f.type === types.FILE">
|
|
242
|
-
<val-file-input [props]="f"></val-file-input>
|
|
178
|
+
<val-file-input [props]="getFieldProp(f)"></val-file-input>
|
|
243
179
|
</ng-container>
|
|
244
180
|
<ng-container *ngIf="f.type === types.HOUR">
|
|
245
|
-
<val-hour-input [props]="f"></val-hour-input>
|
|
181
|
+
<val-hour-input [props]="getFieldProp(f)"></val-hour-input>
|
|
246
182
|
</ng-container>
|
|
247
183
|
<ng-container *ngIf="f.type === types.NUMBER">
|
|
248
|
-
<val-number-input [props]="f"></val-number-input>
|
|
184
|
+
<val-number-input [props]="getFieldProp(f)"></val-number-input>
|
|
249
185
|
</ng-container>
|
|
250
186
|
<ng-container *ngIf="f.type === types.NUMBER_FROM_TO">
|
|
251
|
-
<val-number-from-to [props]="f"></val-number-from-to>
|
|
187
|
+
<val-number-from-to [props]="getFieldProp(f)"></val-number-from-to>
|
|
252
188
|
</ng-container>
|
|
253
189
|
<ng-container *ngIf="f.type === types.PASSWORD">
|
|
254
|
-
<val-password-input [props]="f"></val-password-input>
|
|
190
|
+
<val-password-input [props]="getFieldProp(f)"></val-password-input>
|
|
255
191
|
</ng-container>
|
|
256
192
|
<ng-container *ngIf="f.type === types.PIN_CODE">
|
|
257
|
-
<val-pin-input [props]="f"></val-pin-input>
|
|
193
|
+
<val-pin-input [props]="getFieldProp(f)"></val-pin-input>
|
|
258
194
|
</ng-container>
|
|
259
195
|
<ng-container *ngIf="f.type === types.RADIO">
|
|
260
|
-
<val-radio-input [props]="f"></val-radio-input>
|
|
196
|
+
<val-radio-input [props]="getFieldProp(f)"></val-radio-input>
|
|
261
197
|
</ng-container>
|
|
262
198
|
<ng-container *ngIf="f.type === types.SELECT">
|
|
263
|
-
<val-select-input [props]="f"></val-select-input>
|
|
199
|
+
<val-select-input [props]="getFieldProp(f)"></val-select-input>
|
|
264
200
|
</ng-container>
|
|
265
201
|
<ng-container *ngIf="f.type === types.SEARCH_SELECT">
|
|
266
|
-
<val-select-search [props]="f"></val-select-search>
|
|
202
|
+
<val-select-search [props]="getFieldProp(f)"></val-select-search>
|
|
203
|
+
</ng-container>
|
|
204
|
+
<ng-container *ngIf="f.type === types.MULTI_SELECT">
|
|
205
|
+
<val-multi-select-search [props]="getFieldProp(f)"></val-multi-select-search>
|
|
267
206
|
</ng-container>
|
|
268
|
-
<val-hint [props]="f"></val-hint>
|
|
207
|
+
<val-hint [props]="getFieldProp(f)"></val-hint>
|
|
269
208
|
</div>
|
|
270
209
|
<val-divider [props]="{ fill: 'solid', size: 'medium', color: 'medium' }"></val-divider>
|
|
271
210
|
<ng-content></ng-content>
|
|
@@ -276,7 +215,7 @@ export class FormComponent {
|
|
|
276
215
|
></val-button-group>
|
|
277
216
|
</form>
|
|
278
217
|
</div>
|
|
279
|
-
`, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.section{margin-top:1rem}.input{margin:.5rem 0}@media (min-width: 768px){.input{margin:.75rem 0}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: DisplayComponent, selector: "val-display", inputs: ["props"] }, { kind: "component", type: TitleComponent, selector: "val-title", inputs: ["props"] }, { kind: "component", type: TextInputComponent, selector: "val-text-input", inputs: ["props"] }, { kind: "component", type: CheckInputComponent, selector: "val-check-input" }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: DividerComponent, selector: "val-divider", inputs: ["props"] }, { kind: "component", type: HintComponent, selector: "val-hint", inputs: ["props"] }, { kind: "component", type: CommentInputComponent, selector: "val-comment-input", inputs: ["props"] }, { kind: "component", type: DateInputComponent, selector: "val-date-input", inputs: ["props"] }, { kind: "component", type: FileInputComponent, selector: "val-file-input", inputs: ["props"] }, { kind: "component", type: HourInputComponent, selector: "val-hour-input", inputs: ["props"] }, { kind: "component", type: EmailInputComponent, selector: "val-email-input", inputs: ["props"] }, { kind: "component", type: NumberInputComponent, selector: "val-number-input", inputs: ["props"] }, { kind: "component", type: NumberFromToComponent, selector: "val-number-from-to", inputs: ["props"] }, { kind: "component", type: RadioInputComponent, selector: "val-radio-input", inputs: ["props"] }, { kind: "component", type: PasswordInputComponent, selector: "val-password-input", inputs: ["props"] }, { kind: "component", type: PinInputComponent, selector: "val-pin-input", inputs: ["props"] }, { kind: "component", type: SelectSearchComponent, selector: "val-select-search", inputs: ["label", "labelProperty", "valueProperty", "multiple", "placeholder", "props"] }, { kind: "component", type: SearchSelectorComponent, selector: "val-select-input", inputs: ["props"] }] }); }
|
|
218
|
+
`, isInline: true, styles: [":root{--ion-color-primary: #7026df;--ion-color-primary-rgb: 112, 38, 223;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #6321c4;--ion-color-primary-tint: #7e3ce2;--ion-color-secondary: #e2ccff;--ion-color-secondary-rgb: 226, 204, 255;--ion-color-secondary-contrast: #000000;--ion-color-secondary-contrast-rgb: 0, 0, 0;--ion-color-secondary-shade: #c7b4e0;--ion-color-secondary-tint: #e5d1ff;--ion-color-texti: #354c69;--ion-color-texti-rgb: 53, 76, 105;--ion-color-texti-contrast: #ffffff;--ion-color-texti-contrast-rgb: 255, 255, 255;--ion-color-texti-shade: #2f435c;--ion-color-texti-tint: #495e78;--ion-color-darki: #090f1b;--ion-color-darki-rgb: 9, 15, 27;--ion-color-darki-contrast: #ffffff;--ion-color-darki-contrast-rgb: 255, 255, 255;--ion-color-darki-shade: #080d18;--ion-color-darki-tint: #222732;--ion-color-medium: #9e9e9e;--ion-color-medium-rgb: 158, 158, 158;--ion-color-medium-contrast: #000000;--ion-color-medium-contrast-rgb: 0, 0, 0;--ion-color-medium-shade: #8b8b8b;--ion-color-medium-tint: #a8a8a8;--swiper-pagination-color: var(--ion-color-primary);--swiper-navigation-color: var(--ion-color-primary);--swiper-pagination-bullet-inactive-color: var(--ion-color-medium)}@media (prefers-color-scheme: dark){:root{--ion-color-texti: #8fc1ff;--ion-color-texti-rgb: 143, 193, 255;--ion-color-texti-contrast: #000000;--ion-color-texti-contrast-rgb: 0, 0, 0;--ion-color-texti-shade: #7eaae0;--ion-color-texti-tint: #9ac7ff;--ion-color-darki: #ffffff;--ion-color-darki-rgb: 255, 255, 255;--ion-color-darki-contrast: #000000;--ion-color-darki-contrast-rgb: 0, 0, 0;--ion-color-darki-shade: #e0e0e0;--ion-color-darki-tint: #ffffff;--ion-color-primary: #8f49f8;--ion-color-primary-rgb: 143, 73, 248;--ion-color-primary-contrast: #ffffff;--ion-color-primary-contrast-rgb: 255, 255, 255;--ion-color-primary-shade: #7e40da;--ion-color-primary-tint: #9a5bf9}}.ion-color-texti{--ion-color-base: var(--ion-color-texti);--ion-color-base-rgb: var(--ion-color-texti-rgb);--ion-color-contrast: var(--ion-color-texti-contrast);--ion-color-contrast-rgb: var(--ion-color-texti-contrast-rgb);--ion-color-shade: var(--ion-color-texti-shade);--ion-color-tint: var(--ion-color-texti-tint)}.ion-color-darki{--ion-color-base: var(--ion-color-darki);--ion-color-base-rgb: var(--ion-color-darki-rgb);--ion-color-contrast: var(--ion-color-darki-contrast);--ion-color-contrast-rgb: var(--ion-color-darki-contrast-rgb);--ion-color-shade: var(--ion-color-darki-shade);--ion-color-tint: var(--ion-color-darki-tint)}.section{margin-top:1rem}.input{margin:.5rem 0}@media (min-width: 768px){.input{margin:.75rem 0}}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i2.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i2.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i1.ɵNgNoValidate, selector: "form:not([ngNoForm]):not([ngNativeValidate])" }, { kind: "directive", type: i1.NgControlStatusGroup, selector: "[formGroupName],[formArrayName],[ngModelGroup],[formGroup],form:not([ngNoForm]),[ngForm]" }, { kind: "directive", type: i1.FormGroupDirective, selector: "[formGroup]", inputs: ["formGroup"], outputs: ["ngSubmit"], exportAs: ["ngForm"] }, { kind: "component", type: DisplayComponent, selector: "val-display", inputs: ["props"] }, { kind: "component", type: TitleComponent, selector: "val-title", inputs: ["props"] }, { kind: "component", type: TextInputComponent, selector: "val-text-input", inputs: ["props"] }, { kind: "component", type: CheckInputComponent, selector: "val-check-input" }, { kind: "component", type: ButtonGroupComponent, selector: "val-button-group", inputs: ["props"], outputs: ["onClick"] }, { kind: "component", type: DividerComponent, selector: "val-divider", inputs: ["props"] }, { kind: "component", type: HintComponent, selector: "val-hint", inputs: ["props"] }, { kind: "component", type: CommentInputComponent, selector: "val-comment-input", inputs: ["props"] }, { kind: "component", type: DateInputComponent, selector: "val-date-input", inputs: ["props"] }, { kind: "component", type: FileInputComponent, selector: "val-file-input", inputs: ["props"] }, { kind: "component", type: HourInputComponent, selector: "val-hour-input", inputs: ["props"] }, { kind: "component", type: EmailInputComponent, selector: "val-email-input", inputs: ["props"] }, { kind: "component", type: NumberInputComponent, selector: "val-number-input", inputs: ["props"] }, { kind: "component", type: NumberFromToComponent, selector: "val-number-from-to", inputs: ["props"] }, { kind: "component", type: RadioInputComponent, selector: "val-radio-input", inputs: ["props"] }, { kind: "component", type: PasswordInputComponent, selector: "val-password-input", inputs: ["props"] }, { kind: "component", type: PinInputComponent, selector: "val-pin-input", inputs: ["props"] }, { kind: "component", type: SelectSearchComponent, selector: "val-select-search", inputs: ["label", "labelProperty", "valueProperty", "multiple", "placeholder", "props"] }, { kind: "component", type: MultiSelectSearchComponent, selector: "val-multi-select-search", inputs: ["label", "labelProperty", "valueProperty", "placeholder", "props"] }, { kind: "component", type: SearchSelectorComponent, selector: "val-select-input", inputs: ["props"] }] }); }
|
|
280
219
|
}
|
|
281
220
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImport: i0, type: FormComponent, decorators: [{
|
|
282
221
|
type: Component,
|
|
@@ -301,6 +240,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
301
240
|
PasswordInputComponent,
|
|
302
241
|
PinInputComponent,
|
|
303
242
|
SelectSearchComponent,
|
|
243
|
+
MultiSelectSearchComponent,
|
|
304
244
|
SearchSelectorComponent,
|
|
305
245
|
], template: `
|
|
306
246
|
<div class="container">
|
|
@@ -313,53 +253,56 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
313
253
|
size: 'large',
|
|
314
254
|
}"
|
|
315
255
|
></val-display>
|
|
316
|
-
<div class="section" *ngFor="let s of
|
|
256
|
+
<div class="section" *ngFor="let s of props.sections">
|
|
317
257
|
<val-title [props]="{ content: s.name, size: 'large', color: '', bold: false }"></val-title>
|
|
318
258
|
<div class="input" *ngFor="let f of s.fields">
|
|
319
259
|
<val-title [props]="{ content: f.label, size: 'small', color: 'dark', bold: false }"></val-title>
|
|
320
260
|
<ng-container *ngIf="f.type === types.TEXT">
|
|
321
|
-
<val-text-input [props]="f"></val-text-input>
|
|
261
|
+
<val-text-input [props]="getFieldProp(f)"></val-text-input>
|
|
322
262
|
</ng-container>
|
|
323
263
|
<ng-container *ngIf="f.type === types.CHECK">
|
|
324
264
|
<val-check-input></val-check-input>
|
|
325
265
|
</ng-container>
|
|
326
266
|
<ng-container *ngIf="f.type === types.COMMENT">
|
|
327
|
-
<val-comment-input [props]="f"></val-comment-input>
|
|
267
|
+
<val-comment-input [props]="getFieldProp(f)"></val-comment-input>
|
|
328
268
|
</ng-container>
|
|
329
269
|
<ng-container *ngIf="f.type === types.DATE">
|
|
330
|
-
<val-date-input [props]="f"></val-date-input>
|
|
270
|
+
<val-date-input [props]="getFieldProp(f)"></val-date-input>
|
|
331
271
|
</ng-container>
|
|
332
272
|
<ng-container *ngIf="f.type === types.EMAIL">
|
|
333
|
-
<val-email-input [props]="f"></val-email-input>
|
|
273
|
+
<val-email-input [props]="getFieldProp(f)"></val-email-input>
|
|
334
274
|
</ng-container>
|
|
335
275
|
<ng-container *ngIf="f.type === types.FILE">
|
|
336
|
-
<val-file-input [props]="f"></val-file-input>
|
|
276
|
+
<val-file-input [props]="getFieldProp(f)"></val-file-input>
|
|
337
277
|
</ng-container>
|
|
338
278
|
<ng-container *ngIf="f.type === types.HOUR">
|
|
339
|
-
<val-hour-input [props]="f"></val-hour-input>
|
|
279
|
+
<val-hour-input [props]="getFieldProp(f)"></val-hour-input>
|
|
340
280
|
</ng-container>
|
|
341
281
|
<ng-container *ngIf="f.type === types.NUMBER">
|
|
342
|
-
<val-number-input [props]="f"></val-number-input>
|
|
282
|
+
<val-number-input [props]="getFieldProp(f)"></val-number-input>
|
|
343
283
|
</ng-container>
|
|
344
284
|
<ng-container *ngIf="f.type === types.NUMBER_FROM_TO">
|
|
345
|
-
<val-number-from-to [props]="f"></val-number-from-to>
|
|
285
|
+
<val-number-from-to [props]="getFieldProp(f)"></val-number-from-to>
|
|
346
286
|
</ng-container>
|
|
347
287
|
<ng-container *ngIf="f.type === types.PASSWORD">
|
|
348
|
-
<val-password-input [props]="f"></val-password-input>
|
|
288
|
+
<val-password-input [props]="getFieldProp(f)"></val-password-input>
|
|
349
289
|
</ng-container>
|
|
350
290
|
<ng-container *ngIf="f.type === types.PIN_CODE">
|
|
351
|
-
<val-pin-input [props]="f"></val-pin-input>
|
|
291
|
+
<val-pin-input [props]="getFieldProp(f)"></val-pin-input>
|
|
352
292
|
</ng-container>
|
|
353
293
|
<ng-container *ngIf="f.type === types.RADIO">
|
|
354
|
-
<val-radio-input [props]="f"></val-radio-input>
|
|
294
|
+
<val-radio-input [props]="getFieldProp(f)"></val-radio-input>
|
|
355
295
|
</ng-container>
|
|
356
296
|
<ng-container *ngIf="f.type === types.SELECT">
|
|
357
|
-
<val-select-input [props]="f"></val-select-input>
|
|
297
|
+
<val-select-input [props]="getFieldProp(f)"></val-select-input>
|
|
358
298
|
</ng-container>
|
|
359
299
|
<ng-container *ngIf="f.type === types.SEARCH_SELECT">
|
|
360
|
-
<val-select-search [props]="f"></val-select-search>
|
|
300
|
+
<val-select-search [props]="getFieldProp(f)"></val-select-search>
|
|
301
|
+
</ng-container>
|
|
302
|
+
<ng-container *ngIf="f.type === types.MULTI_SELECT">
|
|
303
|
+
<val-multi-select-search [props]="getFieldProp(f)"></val-multi-select-search>
|
|
361
304
|
</ng-container>
|
|
362
|
-
<val-hint [props]="f"></val-hint>
|
|
305
|
+
<val-hint [props]="getFieldProp(f)"></val-hint>
|
|
363
306
|
</div>
|
|
364
307
|
<val-divider [props]="{ fill: 'solid', size: 'medium', color: 'medium' }"></val-divider>
|
|
365
308
|
<ng-content></ng-content>
|
|
@@ -380,4 +323,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "17.3.12", ngImpo
|
|
|
380
323
|
}], onSelectChange: [{
|
|
381
324
|
type: Output
|
|
382
325
|
}] } });
|
|
383
|
-
//# 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 { distinctUntilChanged } from 'rxjs/operators';\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 { NumberFromToComponent } from '../../molecules/number-from-to/number-from-to.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    NumberFromToComponent,\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 processedSections\">\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]=\"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]=\"f\"></val-comment-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.DATE\">\n              <val-date-input [props]=\"f\"></val-date-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.EMAIL\">\n              <val-email-input [props]=\"f\"></val-email-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.FILE\">\n              <val-file-input [props]=\"f\"></val-file-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.HOUR\">\n              <val-hour-input [props]=\"f\"></val-hour-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.NUMBER\">\n              <val-number-input [props]=\"f\"></val-number-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.NUMBER_FROM_TO\">\n              <val-number-from-to [props]=\"f\"></val-number-from-to>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PASSWORD\">\n              <val-password-input [props]=\"f\"></val-password-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.PIN_CODE\">\n              <val-pin-input [props]=\"f\"></val-pin-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.RADIO\">\n              <val-radio-input [props]=\"f\"></val-radio-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SELECT\">\n              <val-select-input [props]=\"f\"></val-select-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.SEARCH_SELECT\">\n              <val-select-search [props]=\"f\"></val-select-search>\n            </ng-container>\n            <val-hint [props]=\"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  // Cache processed field properties\n  processedSections: { name: string; fields: InputMetadata[] }[] = [];\n  private previousValues = new Map<string, any>();\n\n  constructor(\n    private fb: FormBuilder,\n    private elementRef: ElementRef\n  ) {}\n\n  ngOnInit() {\n    this.createForm();\n    this.processAllSections();\n    this.setupChangeTracking();\n  }\n  \n  private createForm() {\n    const formControls = {};\n    this.props.sections.forEach(section => {\n      section.fields.forEach(field => {\n        if (field.type === this.types.NUMBER_FROM_TO) {\n          // Crear dos controles para campos NUMBER_FROM_TO\n          formControls[`${field.name}_from`] = [undefined, field.validators || []];\n          formControls[`${field.name}_to`] = [undefined, field.validators || []];\n        } else {\n          formControls[field.name] = [undefined, field.validators || []];\n        }\n      });\n    });\n    this.form = this.fb.group(formControls);\n  }\n  \n  private processAllSections() {\n    console.log('[FormComponent] Processing all sections - ONE TIME ONLY');\n    \n    this.processedSections = this.props.sections.map(section => ({\n      name: section.name,\n      fields: section.fields.map(field => this.processField(field))\n    }));\n  }\n  \n  private processField(field: InputMetadata): InputMetadata {\n    // Generate token if not provided\n    if (!field.token) {\n      field.token = `input-${field.type}-${field.name}`;\n    }\n    \n    // Debug: verificar opciones para select fields - SOLO UNA VEZ\n    if (field.type === this.types.SEARCH_SELECT || field.type === this.types.MULTI_SELECT || field.type === this.types.MULTI_SELECT_SIMPLE) {\n      console.log(`[FormComponent] Processing field ${field.name} options:`, {\n        type: field.type,\n        optionsCount: field.options?.length || 0,\n        options: field.options?.slice(0, 2) || []\n      });\n    }\n    \n    if (field.type === this.types.NUMBER_FROM_TO) {\n      const fromControl = this.getControl(`${field.name}_from`);\n      const toControl = this.getControl(`${field.name}_to`);\n      \n      if (field.state === ComponentStates.DISABLED) {\n        fromControl.disable();\n        toControl.disable();\n      } else {\n        fromControl.enable();\n        toControl.enable();\n      }\n      \n      // Modify original field object to maintain reactivity\n      field.fromControl = fromControl;\n      field.toControl = toControl;\n      field.control = undefined; // Remove control for NUMBER_FROM_TO fields\n      return field;\n    } else {\n      const control = this.getControl(field.name);\n      if (field.state === ComponentStates.DISABLED) {\n        control.disable();\n      } else {\n        control.enable();\n      }\n      \n      // Modify original field object to maintain reactivity\n      field.control = control;\n      return field;\n    }\n  }\n  \n  private setupChangeTracking() {\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 || x.type === this.types.MULTI_SELECT || x.type === this.types.MULTI_SELECT_SIMPLE)\n        .forEach(field => {\n          console.log(`[FormComponent] Tracking changes for field: ${field.name}, type: ${field.type}`);\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    \n    // Initialize previous value\n    this.previousValues.set(fieldName, control.value);\n    \n    const subscription = control.valueChanges\n      .pipe(distinctUntilChanged()) // Evitar valores duplicados\n      .subscribe(value => {\n        console.log(`[FormComponent] valueChanges for ${fieldName}:`, {\n          field: fieldName,\n          newValue: value,\n          previousValue: this.previousValues.get(fieldName),\n          willEmit: this.previousValues.get(fieldName) !== value\n        });\n        \n        // Evitar loops - solo emitir si el valor realmente cambió\n        const previousValue = this.previousValues.get(fieldName);\n        if (previousValue !== value) {\n          this.previousValues.set(fieldName, value);\n          console.log(`[FormComponent] EMITTING onSelectChange:`, { field: fieldName, value });\n          this.onSelectChange.emit({ field: fieldName, value });\n        } else {\n          console.log(`[FormComponent] SKIPPING emit - same value for ${fieldName}`);\n        }\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  /**\n   * @deprecated This method is now only used internally. \n   * Use processedSections property instead to avoid multiple calls.\n   */\n  getFieldProp(field: InputMetadata): InputMetadata {\n    console.warn('[FormComponent] getFieldProp is deprecated and should not be called directly');\n    return this.processField(field);\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  // Helper method to reset field value without triggering change event\n  resetField(fieldName: string, emitEvent: boolean = false): void {\n    const control = this.getControl(fieldName);\n    const currentValue = control.value;\n    \n    console.log(`[FormComponent] Resetting field ${fieldName}:`, { currentValue, emitEvent });\n    \n    // Update our cache to avoid triggering change event\n    if (!emitEvent) {\n      this.previousValues.set(fieldName, '');\n    }\n    \n    control.setValue('', { emitEvent });\n    control.markAsPristine();\n    control.markAsUntouched();\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"]}
|
|
326
|
+
//# 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 { NumberFromToComponent } from '../../molecules/number-from-to/number-from-to.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 { MultiSelectSearchComponent } from '../../molecules/multi-select-search/multi-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    NumberFromToComponent,\n    RadioInputComponent,\n    PasswordInputComponent,\n    PinInputComponent,\n    SelectSearchComponent,\n    MultiSelectSearchComponent,\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.NUMBER_FROM_TO\">\n              <val-number-from-to [props]=\"getFieldProp(f)\"></val-number-from-to>\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            <ng-container *ngIf=\"f.type === types.MULTI_SELECT\">\n              <val-multi-select-search [props]=\"getFieldProp(f)\"></val-multi-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        if (field.type === this.types.NUMBER_FROM_TO) {\n          // Crear dos controles para campos NUMBER_FROM_TO\n          formControls[`${field.name}_from`] = [undefined, field.validators || []];\n          formControls[`${field.name}_to`] = [undefined, field.validators || []];\n        } else {\n          formControls[field.name] = [undefined, field.validators || []];\n        }\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 || x.type === this.types.MULTI_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    \n    if (field.type === this.types.NUMBER_FROM_TO) {\n      const fromControl = this.getControl(`${field.name}_from`);\n      const toControl = this.getControl(`${field.name}_to`);\n      \n      if (field.state === ComponentStates.DISABLED) {\n        fromControl.disable();\n        toControl.disable();\n      } else {\n        fromControl.enable();\n        toControl.enable();\n      }\n      \n      return {\n        ...field,\n        fromControl,\n        toControl,\n        control: undefined, // Remove control for NUMBER_FROM_TO fields\n      };\n    } else {\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\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"]}
|