valtech-components 2.0.692 → 2.0.694
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/checkbox-radio-input/checkbox-radio-input.component.mjs +79 -0
- package/esm2022/lib/components/molecules/refresher/refresher.component.mjs +4 -2
- package/esm2022/lib/components/organisms/form/form.component.mjs +18 -2
- package/esm2022/lib/components/types.mjs +2 -1
- package/esm2022/lib/version.mjs +2 -2
- package/esm2022/public-api.mjs +2 -1
- package/fesm2022/valtech-components.mjs +94 -4
- package/fesm2022/valtech-components.mjs.map +1 -1
- package/lib/components/molecules/checkbox-radio-input/checkbox-radio-input.component.d.ts +17 -0
- package/lib/components/types.d.ts +2 -1
- package/lib/version.d.ts +1 -1
- package/package.json +1 -1
- package/public-api.d.ts +1 -0
package/esm2022/lib/components/molecules/checkbox-radio-input/checkbox-radio-input.component.mjs
ADDED
|
@@ -0,0 +1,79 @@
|
|
|
1
|
+
import { CommonModule } from '@angular/common';
|
|
2
|
+
import { Component, Input } from '@angular/core';
|
|
3
|
+
import { ReactiveFormsModule } from '@angular/forms';
|
|
4
|
+
import { IonCheckbox } from '@ionic/angular/standalone';
|
|
5
|
+
import { ComponentStates } from '../../types';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
/**
|
|
8
|
+
* val-checkbox-radio-input
|
|
9
|
+
*
|
|
10
|
+
* A group of checkboxes with radio behavior (single selection).
|
|
11
|
+
* Visually displays as checkboxes but only allows one option to be selected at a time.
|
|
12
|
+
*
|
|
13
|
+
* @example
|
|
14
|
+
* <val-checkbox-radio-input [props]="{
|
|
15
|
+
* control: myControl,
|
|
16
|
+
* options: [
|
|
17
|
+
* { id: 'yes', name: 'Sí', order: 0 },
|
|
18
|
+
* { id: 'no', name: 'No', order: 1 }
|
|
19
|
+
* ]
|
|
20
|
+
* }"></val-checkbox-radio-input>
|
|
21
|
+
*
|
|
22
|
+
* @input props: InputMetadata - Configuration including control and options
|
|
23
|
+
*/
|
|
24
|
+
export class CheckboxRadioInputComponent {
|
|
25
|
+
constructor() {
|
|
26
|
+
this.states = ComponentStates;
|
|
27
|
+
}
|
|
28
|
+
ngOnInit() { }
|
|
29
|
+
isSelected(option) {
|
|
30
|
+
return this.props.control?.value === option.id;
|
|
31
|
+
}
|
|
32
|
+
onOptionSelect(option, event) {
|
|
33
|
+
if (event.detail.checked) {
|
|
34
|
+
// Seleccionar esta opción
|
|
35
|
+
this.props.control?.setValue(option.id);
|
|
36
|
+
}
|
|
37
|
+
else {
|
|
38
|
+
// Si se deselecciona la opción actual, limpiar
|
|
39
|
+
if (this.props.control?.value === option.id) {
|
|
40
|
+
this.props.control?.setValue(null);
|
|
41
|
+
}
|
|
42
|
+
}
|
|
43
|
+
}
|
|
44
|
+
static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CheckboxRadioInputComponent, deps: [], target: i0.ɵɵFactoryTarget.Component }); }
|
|
45
|
+
static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "17.0.0", version: "18.2.14", type: CheckboxRadioInputComponent, isStandalone: true, selector: "val-checkbox-radio-input", inputs: { props: "props" }, ngImport: i0, template: `
|
|
46
|
+
<div class="checkbox-radio-group">
|
|
47
|
+
@for (option of props.options; track option.id) {
|
|
48
|
+
<ion-checkbox
|
|
49
|
+
[checked]="isSelected(option)"
|
|
50
|
+
[disabled]="props.state === states.DISABLED"
|
|
51
|
+
[labelPlacement]="'end'"
|
|
52
|
+
(ionChange)="onOptionSelect(option, $event)"
|
|
53
|
+
>
|
|
54
|
+
{{ option.name }}
|
|
55
|
+
</ion-checkbox>
|
|
56
|
+
}
|
|
57
|
+
</div>
|
|
58
|
+
`, isInline: true, styles: [".checkbox-radio-group{display:flex;flex-direction:column;gap:12px}.checkbox-radio-group ion-checkbox{--size: 24px;--checkbox-background-checked: var(--ion-color-primary);--checkmark-color: var(--ion-color-primary-contrast)}\n"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "component", type: IonCheckbox, selector: "ion-checkbox", inputs: ["checked", "color", "disabled", "errorText", "helperText", "indeterminate", "justify", "labelPlacement", "mode", "name", "value"] }] }); }
|
|
59
|
+
}
|
|
60
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: CheckboxRadioInputComponent, decorators: [{
|
|
61
|
+
type: Component,
|
|
62
|
+
args: [{ selector: 'val-checkbox-radio-input', standalone: true, imports: [CommonModule, ReactiveFormsModule, IonCheckbox], template: `
|
|
63
|
+
<div class="checkbox-radio-group">
|
|
64
|
+
@for (option of props.options; track option.id) {
|
|
65
|
+
<ion-checkbox
|
|
66
|
+
[checked]="isSelected(option)"
|
|
67
|
+
[disabled]="props.state === states.DISABLED"
|
|
68
|
+
[labelPlacement]="'end'"
|
|
69
|
+
(ionChange)="onOptionSelect(option, $event)"
|
|
70
|
+
>
|
|
71
|
+
{{ option.name }}
|
|
72
|
+
</ion-checkbox>
|
|
73
|
+
}
|
|
74
|
+
</div>
|
|
75
|
+
`, styles: [".checkbox-radio-group{display:flex;flex-direction:column;gap:12px}.checkbox-radio-group ion-checkbox{--size: 24px;--checkbox-background-checked: var(--ion-color-primary);--checkmark-color: var(--ion-color-primary-contrast)}\n"] }]
|
|
76
|
+
}], propDecorators: { props: [{
|
|
77
|
+
type: Input
|
|
78
|
+
}] } });
|
|
79
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiY2hlY2tib3gtcmFkaW8taW5wdXQuY29tcG9uZW50LmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vLi4vc3JjL2xpYi9jb21wb25lbnRzL21vbGVjdWxlcy9jaGVja2JveC1yYWRpby1pbnB1dC9jaGVja2JveC1yYWRpby1pbnB1dC5jb21wb25lbnQudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFlBQVksRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBQy9DLE9BQU8sRUFBRSxTQUFTLEVBQUUsS0FBSyxFQUFVLE1BQU0sZUFBZSxDQUFDO0FBQ3pELE9BQU8sRUFBRSxtQkFBbUIsRUFBRSxNQUFNLGdCQUFnQixDQUFDO0FBQ3JELE9BQU8sRUFBRSxXQUFXLEVBQUUsTUFBTSwyQkFBMkIsQ0FBQztBQUN4RCxPQUFPLEVBQThCLGVBQWUsRUFBRSxNQUFNLGFBQWEsQ0FBQzs7QUFzQjFFOzs7Ozs7Ozs7Ozs7Ozs7O0dBZ0JHO0FBQ0gsTUFBTSxPQUFPLDJCQUEyQjtJQXJDeEM7UUF1Q0UsV0FBTSxHQUFHLGVBQWUsQ0FBQztLQW1CMUI7SUFqQkMsUUFBUSxLQUFJLENBQUM7SUFFYixVQUFVLENBQUMsTUFBbUI7UUFDNUIsT0FBTyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxLQUFLLEtBQUssTUFBTSxDQUFDLEVBQUUsQ0FBQztJQUNqRCxDQUFDO0lBRUQsY0FBYyxDQUFDLE1BQW1CLEVBQUUsS0FBVTtRQUM1QyxJQUFJLEtBQUssQ0FBQyxNQUFNLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDekIsMEJBQTBCO1lBQzFCLElBQUksQ0FBQyxLQUFLLENBQUMsT0FBTyxFQUFFLFFBQVEsQ0FBQyxNQUFNLENBQUMsRUFBRSxDQUFDLENBQUM7UUFDMUMsQ0FBQzthQUFNLENBQUM7WUFDTiwrQ0FBK0M7WUFDL0MsSUFBSSxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxLQUFLLEtBQUssTUFBTSxDQUFDLEVBQUUsRUFBRSxDQUFDO2dCQUM1QyxJQUFJLENBQUMsS0FBSyxDQUFDLE9BQU8sRUFBRSxRQUFRLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDckMsQ0FBQztRQUNILENBQUM7SUFDSCxDQUFDOytHQXBCVSwyQkFBMkI7bUdBQTNCLDJCQUEyQixnSEFqQzVCOzs7Ozs7Ozs7Ozs7O0dBYVQsMFNBZFMsWUFBWSw4QkFBRSxtQkFBbUIsK0JBQUUsV0FBVzs7NEZBa0M3QywyQkFBMkI7a0JBckN2QyxTQUFTOytCQUNFLDBCQUEwQixjQUN4QixJQUFJLFdBQ1AsQ0FBQyxZQUFZLEVBQUUsbUJBQW1CLEVBQUUsV0FBVyxDQUFDLFlBQy9DOzs7Ozs7Ozs7Ozs7O0dBYVQ7OEJBcUJRLEtBQUs7c0JBQWIsS0FBSyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IENvbW1vbk1vZHVsZSB9IGZyb20gJ0Bhbmd1bGFyL2NvbW1vbic7XG5pbXBvcnQgeyBDb21wb25lbnQsIElucHV0LCBPbkluaXQgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IFJlYWN0aXZlRm9ybXNNb2R1bGUgfSBmcm9tICdAYW5ndWxhci9mb3Jtcyc7XG5pbXBvcnQgeyBJb25DaGVja2JveCB9IGZyb20gJ0Bpb25pYy9hbmd1bGFyL3N0YW5kYWxvbmUnO1xuaW1wb3J0IHsgSW5wdXRNZXRhZGF0YSwgSW5wdXRPcHRpb24sIENvbXBvbmVudFN0YXRlcyB9IGZyb20gJy4uLy4uL3R5cGVzJztcblxuQENvbXBvbmVudCh7XG4gIHNlbGVjdG9yOiAndmFsLWNoZWNrYm94LXJhZGlvLWlucHV0JyxcbiAgc3RhbmRhbG9uZTogdHJ1ZSxcbiAgaW1wb3J0czogW0NvbW1vbk1vZHVsZSwgUmVhY3RpdmVGb3Jtc01vZHVsZSwgSW9uQ2hlY2tib3hdLFxuICB0ZW1wbGF0ZTogYFxuICAgIDxkaXYgY2xhc3M9XCJjaGVja2JveC1yYWRpby1ncm91cFwiPlxuICAgICAgQGZvciAob3B0aW9uIG9mIHByb3BzLm9wdGlvbnM7IHRyYWNrIG9wdGlvbi5pZCkge1xuICAgICAgICA8aW9uLWNoZWNrYm94XG4gICAgICAgICAgW2NoZWNrZWRdPVwiaXNTZWxlY3RlZChvcHRpb24pXCJcbiAgICAgICAgICBbZGlzYWJsZWRdPVwicHJvcHMuc3RhdGUgPT09IHN0YXRlcy5ESVNBQkxFRFwiXG4gICAgICAgICAgW2xhYmVsUGxhY2VtZW50XT1cIidlbmQnXCJcbiAgICAgICAgICAoaW9uQ2hhbmdlKT1cIm9uT3B0aW9uU2VsZWN0KG9wdGlvbiwgJGV2ZW50KVwiXG4gICAgICAgID5cbiAgICAgICAgICB7eyBvcHRpb24ubmFtZSB9fVxuICAgICAgICA8L2lvbi1jaGVja2JveD5cbiAgICAgIH1cbiAgICA8L2Rpdj5cbiAgYCxcbiAgc3R5bGVVcmxzOiBbJy4vY2hlY2tib3gtcmFkaW8taW5wdXQuY29tcG9uZW50LnNjc3MnXSxcbn0pXG4vKipcbiAqIHZhbC1jaGVja2JveC1yYWRpby1pbnB1dFxuICpcbiAqIEEgZ3JvdXAgb2YgY2hlY2tib3hlcyB3aXRoIHJhZGlvIGJlaGF2aW9yIChzaW5nbGUgc2VsZWN0aW9uKS5cbiAqIFZpc3VhbGx5IGRpc3BsYXlzIGFzIGNoZWNrYm94ZXMgYnV0IG9ubHkgYWxsb3dzIG9uZSBvcHRpb24gdG8gYmUgc2VsZWN0ZWQgYXQgYSB0aW1lLlxuICpcbiAqIEBleGFtcGxlXG4gKiA8dmFsLWNoZWNrYm94LXJhZGlvLWlucHV0IFtwcm9wc109XCJ7XG4gKiAgIGNvbnRyb2w6IG15Q29udHJvbCxcbiAqICAgb3B0aW9uczogW1xuICogICAgIHsgaWQ6ICd5ZXMnLCBuYW1lOiAnU8OtJywgb3JkZXI6IDAgfSxcbiAqICAgICB7IGlkOiAnbm8nLCBuYW1lOiAnTm8nLCBvcmRlcjogMSB9XG4gKiAgIF1cbiAqIH1cIj48L3ZhbC1jaGVja2JveC1yYWRpby1pbnB1dD5cbiAqXG4gKiBAaW5wdXQgcHJvcHM6IElucHV0TWV0YWRhdGEgLSBDb25maWd1cmF0aW9uIGluY2x1ZGluZyBjb250cm9sIGFuZCBvcHRpb25zXG4gKi9cbmV4cG9ydCBjbGFzcyBDaGVja2JveFJhZGlvSW5wdXRDb21wb25lbnQgaW1wbGVtZW50cyBPbkluaXQge1xuICBASW5wdXQoKSBwcm9wczogSW5wdXRNZXRhZGF0YTtcbiAgc3RhdGVzID0gQ29tcG9uZW50U3RhdGVzO1xuXG4gIG5nT25Jbml0KCkge31cblxuICBpc1NlbGVjdGVkKG9wdGlvbjogSW5wdXRPcHRpb24pOiBib29sZWFuIHtcbiAgICByZXR1cm4gdGhpcy5wcm9wcy5jb250cm9sPy52YWx1ZSA9PT0gb3B0aW9uLmlkO1xuICB9XG5cbiAgb25PcHRpb25TZWxlY3Qob3B0aW9uOiBJbnB1dE9wdGlvbiwgZXZlbnQ6IGFueSk6IHZvaWQge1xuICAgIGlmIChldmVudC5kZXRhaWwuY2hlY2tlZCkge1xuICAgICAgLy8gU2VsZWNjaW9uYXIgZXN0YSBvcGNpw7NuXG4gICAgICB0aGlzLnByb3BzLmNvbnRyb2w/LnNldFZhbHVlKG9wdGlvbi5pZCk7XG4gICAgfSBlbHNlIHtcbiAgICAgIC8vIFNpIHNlIGRlc2VsZWNjaW9uYSBsYSBvcGNpw7NuIGFjdHVhbCwgbGltcGlhclxuICAgICAgaWYgKHRoaXMucHJvcHMuY29udHJvbD8udmFsdWUgPT09IG9wdGlvbi5pZCkge1xuICAgICAgICB0aGlzLnByb3BzLmNvbnRyb2w/LnNldFZhbHVlKG51bGwpO1xuICAgICAgfVxuICAgIH1cbiAgfVxufVxuIl19
|
|
@@ -110,7 +110,9 @@ export class RefresherComponent {
|
|
|
110
110
|
}
|
|
111
111
|
/** Handler para evento ionPull */
|
|
112
112
|
onIonPull(event) {
|
|
113
|
-
const detail = event
|
|
113
|
+
const detail = event?.detail;
|
|
114
|
+
if (!detail || typeof detail.progress !== 'number')
|
|
115
|
+
return;
|
|
114
116
|
const progress = Math.min(detail.progress, 1);
|
|
115
117
|
this.pullProgress.set(progress);
|
|
116
118
|
this.pullProgressChange.emit({
|
|
@@ -266,4 +268,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
266
268
|
}], stateChange: [{
|
|
267
269
|
type: Output
|
|
268
270
|
}] } });
|
|
269
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"refresher.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/refresher/refresher.component.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,SAAS,EACT,KAAK,EACL,MAAM,EACN,YAAY,EACZ,MAAM,EACN,SAAS,EAET,YAAY,EACZ,MAAM,GACP,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,YAAY,EACZ,mBAAmB,EACnB,UAAU,EACV,OAAO,GACR,MAAM,2BAA2B,CAAC;AACnC,OAAO,EAAE,WAAW,EAAE,MAAM,wBAAwB,CAAC;AACrD,OAAO,EAKL,0BAA0B,GAC3B,MAAM,SAAS,CAAC;;;AAEjB;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA4BG;AA8EH,MAAM,OAAO,kBAAkB;IA7E/B;QAqFE,kCAAkC;QACzB,UAAK,GAAsB,EAAE,CAAC;QAEvC,iDAAiD;QACvC,YAAO,GAAG,IAAI,YAAY,EAAgB,CAAC;QAErD,yCAAyC;QAC/B,uBAAkB,GAAG,IAAI,YAAY,EAAoB,CAAC;QAEpE,iCAAiC;QACvB,gBAAW,GAAG,IAAI,YAAY,EAAkB,CAAC;QAEnD,SAAI,GAAG,MAAM,CAAC,WAAW,CAAC,CAAC;QAEnC,kCAAkC;QACzB,UAAK,GAAG,MAAM,CAAiB,MAAM,CAAC,CAAC;QAEhD,qCAAqC;QAC5B,iBAAY,GAAG,MAAM,CAAS,CAAC,CAAC,CAAC;KAiG3C;IA/FC,0CAA0C;IAC1C,cAAc;QACZ,OAAO,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,eAAe,CAAC,CAAC;IAChE,CAAC;IAED,6CAA6C;IAC7C,iBAAiB;QACf,OAAO,IAAI,CAAC,KAAK,CAAC,cAAc,IAAI,IAAI,CAAC,IAAI,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC;IAC7D,CAAC;IAED,oCAAoC;IACpC,IAAI,WAAW;QACb,OAAO;YACL,GAAG,0BAA0B;YAC7B,GAAG,IAAI,CAAC,KAAK;YACb,WAAW,EAAE,IAAI,CAAC,cAAc,EAAE;YAClC,cAAc,EAAE,IAAI,CAAC,iBAAiB,EAAE;SACzC,CAAC;IACJ,CAAC;IAED,wCAAwC;IACxC,IAAI,kBAAkB;QACpB,OAAO,CAAC,CAAC,CACP,IAAI,CAAC,gBAAgB;YACrB,IAAI,CAAC,mBAAmB;YACxB,IAAI,CAAC,mBAAmB;YACxB,IAAI,CAAC,KAAK,CAAC,eAAe,CAC3B,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,cAAc;QACZ,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAC7B,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED;;OAEG;IACH,QAAQ;QACN,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAC7B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACpC,IAAI,CAAC,YAAY,EAAE,QAAQ,EAAE,CAAC;QAE9B,0CAA0C;QAC1C,UAAU,CAAC,GAAG,EAAE;YACd,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;YACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;QAChC,CAAC,EAAE,IAAI,CAAC,WAAW,CAAC,aAAa,IAAI,GAAG,CAAC,CAAC;IAC5C,CAAC;IAED;;OAEG;IACH,MAAM;QACJ,IAAI,CAAC,YAAY,EAAE,MAAM,EAAE,CAAC;QAC5B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,MAAM,CAAC,CAAC;QACvB,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,MAAM,CAAC,CAAC;IAChC,CAAC;IAED,qCAAqC;IACrC,YAAY,CAAC,KAAkB;QAC7B,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,YAAY,CAAC,CAAC;QAC7B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,YAAY,CAAC,CAAC;QACpC,IAAI,CAAC,gBAAgB,EAAE,CAAC;IAC1B,CAAC;IAED,kCAAkC;IAClC,SAAS,CAAC,KAAkB;QAC1B,MAAM,MAAM,GAAG,KAAK,CAAC,MAAM,CAAC;QAC5B,MAAM,QAAQ,GAAG,IAAI,CAAC,GAAG,CAAC,MAAM,CAAC,QAAQ,EAAE,CAAC,CAAC,CAAC;QAC9C,IAAI,CAAC,YAAY,CAAC,GAAG,CAAC,QAAQ,CAAC,CAAC;QAEhC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC;YAC3B,QAAQ;YACR,gBAAgB,EAAE,QAAQ,IAAI,CAAC;SAChC,CAAC,CAAC;IACL,CAAC;IAED,mCAAmC;IACnC,UAAU;QACR,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC,SAAS,CAAC,CAAC;QAC1B,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAS,CAAC,CAAC;IACnC,CAAC;IAEO,gBAAgB;QACtB,MAAM,KAAK,GAAiB;YAC1B,QAAQ,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,QAAQ,EAAE;YAC/B,MAAM,EAAE,GAAG,EAAE,CAAC,IAAI,CAAC,MAAM,EAAE;YAC3B,SAAS,EAAE,IAAI,IAAI,EAAE;SACtB,CAAC;QACF,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC;IAC3B,CAAC;+GA1HU,kBAAkB;mGAAlB,kBAAkB,moBAzEnB;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT,6OAvDS,YAAY,sMAAE,YAAY,iKAAE,mBAAmB,iJAAE,UAAU,yGAAE,OAAO;;4FA0EnE,kBAAkB;kBA7E9B,SAAS;+BACE,eAAe,cACb,IAAI,WACP,CAAC,YAAY,EAAE,YAAY,EAAE,mBAAmB,EAAE,UAAU,EAAE,OAAO,CAAC,YACrE;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAsDT;8BAoBuB,YAAY;sBAAnC,SAAS;uBAAC,WAAW;gBAGY,gBAAgB;sBAAjD,YAAY;uBAAC,kBAAkB;gBACK,mBAAmB;sBAAvD,YAAY;uBAAC,qBAAqB;gBACE,mBAAmB;sBAAvD,YAAY;uBAAC,qBAAqB;gBAG1B,KAAK;sBAAb,KAAK;gBAGI,OAAO;sBAAhB,MAAM;gBAGG,kBAAkB;sBAA3B,MAAM;gBAGG,WAAW;sBAApB,MAAM","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  signal,\n  ViewChild,\n  TemplateRef,\n  ContentChild,\n  inject,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport {\n  IonRefresher,\n  IonRefresherContent,\n  IonSpinner,\n  IonText,\n} from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  RefresherMetadata,\n  RefresherState,\n  RefreshEvent,\n  RefreshPullEvent,\n  DEFAULT_REFRESHER_METADATA,\n} from './types';\n\n/**\n * Componente de pull-to-refresh para movil y web.\n *\n * @example\n * <!-- Uso basico -->\n * <val-refresher\n *   [props]=\"{\n *     color: 'primary',\n *     pullingText: 'Arrastra para actualizar',\n *     refreshingText: 'Cargando...'\n *   }\"\n *   (refresh)=\"onRefresh($event)\"\n * >\n *   <ion-content>\n *     <!-- Contenido scrolleable -->\n *   </ion-content>\n * </val-refresher>\n *\n * @example\n * <!-- Con indicador personalizado -->\n * <val-refresher [props]=\"{ color: 'primary' }\" (refresh)=\"onRefresh($event)\">\n *   <ng-template #pullingIndicator let-progress=\"progress\">\n *     <div class=\"custom-indicator\">\n *       <ion-icon name=\"arrow-down\" [style.transform]=\"'rotate(' + progress * 180 + 'deg)'\"></ion-icon>\n *     </div>\n *   </ng-template>\n *   <ion-content>...</ion-content>\n * </val-refresher>\n */\n@Component({\n  selector: 'val-refresher',\n  standalone: true,\n  imports: [CommonModule, IonRefresher, IonRefresherContent, IonSpinner, IonText],\n  template: `\n    <ion-refresher\n      #refresher\n      slot=\"fixed\"\n      [disabled]=\"mergedProps.disabled\"\n      [pullMin]=\"mergedProps.pullThreshold\"\n      [pullMax]=\"mergedProps.pullMax\"\n      [pullFactor]=\"mergedProps.pullFactor\"\n      [snapbackDuration]=\"mergedProps.snapbackDuration\"\n      [closeDuration]=\"mergedProps.closeDuration\"\n      (ionRefresh)=\"onIonRefresh($event)\"\n      (ionPull)=\"onIonPull($event)\"\n      (ionStart)=\"onIonStart()\"\n    >\n      @if (hasCustomIndicator) {\n        <!-- Custom indicator templates -->\n        <div class=\"refresher-custom-content\">\n          @switch (state()) {\n            @case ('pulling') {\n              @if (pullingIndicator) {\n                <ng-container\n                  *ngTemplateOutlet=\"pullingIndicator; context: { progress: pullProgress() }\"\n                ></ng-container>\n              }\n            }\n            @case ('refreshing') {\n              @if (refreshingIndicator) {\n                <ng-container *ngTemplateOutlet=\"refreshingIndicator\"></ng-container>\n              } @else {\n                <ion-spinner [name]=\"mergedProps.spinnerType\" [color]=\"mergedProps.color\"></ion-spinner>\n                @if (mergedProps.refreshingText) {\n                  <ion-text [color]=\"mergedProps.color\">{{ mergedProps.refreshingText }}</ion-text>\n                }\n              }\n            }\n            @case ('completing') {\n              @if (completingIndicator) {\n                <ng-container *ngTemplateOutlet=\"completingIndicator\"></ng-container>\n              }\n            }\n          }\n        </div>\n      } @else {\n        <!-- Default Ionic refresher content -->\n        <ion-refresher-content\n          [pullingIcon]=\"mergedProps.spinnerType === 'crescent' ? 'chevron-down-circle-outline' : undefined\"\n          [pullingText]=\"mergedProps.pullingText\"\n          [refreshingSpinner]=\"mergedProps.spinnerType\"\n          [refreshingText]=\"mergedProps.refreshingText\"\n        ></ion-refresher-content>\n      }\n    </ion-refresher>\n\n    <ng-content></ng-content>\n  `,\n  styles: [\n    `\n      :host {\n        display: block;\n        position: relative;\n      }\n\n      .refresher-custom-content {\n        display: flex;\n        flex-direction: column;\n        align-items: center;\n        justify-content: center;\n        gap: 8px;\n        padding: 16px;\n      }\n    `,\n  ],\n})\nexport class RefresherComponent {\n  @ViewChild('refresher') ionRefresher!: IonRefresher;\n\n  /** Templates personalizados via content projection */\n  @ContentChild('pullingIndicator') pullingIndicator?: TemplateRef<{ progress: number }>;\n  @ContentChild('refreshingIndicator') refreshingIndicator?: TemplateRef<void>;\n  @ContentChild('completingIndicator') completingIndicator?: TemplateRef<void>;\n\n  /** Configuracion del refresher */\n  @Input() props: RefresherMetadata = {};\n\n  /** Evento emitido cuando se activa el refresh */\n  @Output() refresh = new EventEmitter<RefreshEvent>();\n\n  /** Evento de progreso durante el pull */\n  @Output() pullProgressChange = new EventEmitter<RefreshPullEvent>();\n\n  /** Evento de cambio de estado */\n  @Output() stateChange = new EventEmitter<RefresherState>();\n\n  private i18n = inject(I18nService);\n\n  /** Estado actual del refresher */\n  readonly state = signal<RefresherState>('idle');\n\n  /** Progreso actual del pull (0-1) */\n  readonly pullProgress = signal<number>(0);\n\n  /** Get pulling text with i18n fallback */\n  getPullingText(): string {\n    return this.props.pullingText || this.i18n.t('pullToRefresh');\n  }\n\n  /** Get refreshing text with i18n fallback */\n  getRefreshingText(): string {\n    return this.props.refreshingText || this.i18n.t('loading');\n  }\n\n  /** Props combinados con defaults */\n  get mergedProps(): RefresherMetadata {\n    return {\n      ...DEFAULT_REFRESHER_METADATA,\n      ...this.props,\n      pullingText: this.getPullingText(),\n      refreshingText: this.getRefreshingText(),\n    };\n  }\n\n  /** Si hay indicadores personalizados */\n  get hasCustomIndicator(): boolean {\n    return !!(\n      this.pullingIndicator ||\n      this.refreshingIndicator ||\n      this.completingIndicator ||\n      this.props.indicatorConfig\n    );\n  }\n\n  /**\n   * Activa programaticamente el refresh.\n   */\n  triggerRefresh(): void {\n    this.state.set('refreshing');\n    this.emitRefreshEvent();\n  }\n\n  /**\n   * Completa la operacion de refresh actual.\n   */\n  complete(): void {\n    this.state.set('completing');\n    this.stateChange.emit('completing');\n    this.ionRefresher?.complete();\n\n    // Resetear a idle despues de la animacion\n    setTimeout(() => {\n      this.state.set('idle');\n      this.stateChange.emit('idle');\n    }, this.mergedProps.closeDuration ?? 280);\n  }\n\n  /**\n   * Cancela la operacion de refresh actual.\n   */\n  cancel(): void {\n    this.ionRefresher?.cancel();\n    this.state.set('idle');\n    this.stateChange.emit('idle');\n  }\n\n  /** Handler para evento ionRefresh */\n  onIonRefresh(event: CustomEvent): void {\n    this.state.set('refreshing');\n    this.stateChange.emit('refreshing');\n    this.emitRefreshEvent();\n  }\n\n  /** Handler para evento ionPull */\n  onIonPull(event: CustomEvent): void {\n    const detail = event.detail;\n    const progress = Math.min(detail.progress, 1);\n    this.pullProgress.set(progress);\n\n    this.pullProgressChange.emit({\n      progress,\n      thresholdReached: progress >= 1,\n    });\n  }\n\n  /** Handler para evento ionStart */\n  onIonStart(): void {\n    this.state.set('pulling');\n    this.stateChange.emit('pulling');\n  }\n\n  private emitRefreshEvent(): void {\n    const event: RefreshEvent = {\n      complete: () => this.complete(),\n      cancel: () => this.cancel(),\n      timestamp: new Date(),\n    };\n    this.refresh.emit(event);\n  }\n}\n"]}
|
|
271
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"refresher.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/molecules/refresher/refresher.component.ts"],"names":[],"mappings":"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","sourcesContent":["import {\n  Component,\n  Input,\n  Output,\n  EventEmitter,\n  signal,\n  ViewChild,\n  TemplateRef,\n  ContentChild,\n  inject,\n} from '@angular/core';\nimport { CommonModule } from '@angular/common';\nimport {\n  IonRefresher,\n  IonRefresherContent,\n  IonSpinner,\n  IonText,\n} from '@ionic/angular/standalone';\nimport { I18nService } from '../../../services/i18n';\nimport {\n  RefresherMetadata,\n  RefresherState,\n  RefreshEvent,\n  RefreshPullEvent,\n  DEFAULT_REFRESHER_METADATA,\n} from './types';\n\n/**\n * Componente de pull-to-refresh para movil y web.\n *\n * @example\n * <!-- Uso basico -->\n * <val-refresher\n *   [props]=\"{\n *     color: 'primary',\n *     pullingText: 'Arrastra para actualizar',\n *     refreshingText: 'Cargando...'\n *   }\"\n *   (refresh)=\"onRefresh($event)\"\n * >\n *   <ion-content>\n *     <!-- Contenido scrolleable -->\n *   </ion-content>\n * </val-refresher>\n *\n * @example\n * <!-- Con indicador personalizado -->\n * <val-refresher [props]=\"{ color: 'primary' }\" (refresh)=\"onRefresh($event)\">\n *   <ng-template #pullingIndicator let-progress=\"progress\">\n *     <div class=\"custom-indicator\">\n *       <ion-icon name=\"arrow-down\" [style.transform]=\"'rotate(' + progress * 180 + 'deg)'\"></ion-icon>\n *     </div>\n *   </ng-template>\n *   <ion-content>...</ion-content>\n * </val-refresher>\n */\n@Component({\n  selector: 'val-refresher',\n  standalone: true,\n  imports: [CommonModule, IonRefresher, IonRefresherContent, IonSpinner, IonText],\n  template: `\n    <ion-refresher\n      #refresher\n      slot=\"fixed\"\n      [disabled]=\"mergedProps.disabled\"\n      [pullMin]=\"mergedProps.pullThreshold\"\n      [pullMax]=\"mergedProps.pullMax\"\n      [pullFactor]=\"mergedProps.pullFactor\"\n      [snapbackDuration]=\"mergedProps.snapbackDuration\"\n      [closeDuration]=\"mergedProps.closeDuration\"\n      (ionRefresh)=\"onIonRefresh($event)\"\n      (ionPull)=\"onIonPull($event)\"\n      (ionStart)=\"onIonStart()\"\n    >\n      @if (hasCustomIndicator) {\n        <!-- Custom indicator templates -->\n        <div class=\"refresher-custom-content\">\n          @switch (state()) {\n            @case ('pulling') {\n              @if (pullingIndicator) {\n                <ng-container\n                  *ngTemplateOutlet=\"pullingIndicator; context: { progress: pullProgress() }\"\n                ></ng-container>\n              }\n            }\n            @case ('refreshing') {\n              @if (refreshingIndicator) {\n                <ng-container *ngTemplateOutlet=\"refreshingIndicator\"></ng-container>\n              } @else {\n                <ion-spinner [name]=\"mergedProps.spinnerType\" [color]=\"mergedProps.color\"></ion-spinner>\n                @if (mergedProps.refreshingText) {\n                  <ion-text [color]=\"mergedProps.color\">{{ mergedProps.refreshingText }}</ion-text>\n                }\n              }\n            }\n            @case ('completing') {\n              @if (completingIndicator) {\n                <ng-container *ngTemplateOutlet=\"completingIndicator\"></ng-container>\n              }\n            }\n          }\n        </div>\n      } @else {\n        <!-- Default Ionic refresher content -->\n        <ion-refresher-content\n          [pullingIcon]=\"mergedProps.spinnerType === 'crescent' ? 'chevron-down-circle-outline' : undefined\"\n          [pullingText]=\"mergedProps.pullingText\"\n          [refreshingSpinner]=\"mergedProps.spinnerType\"\n          [refreshingText]=\"mergedProps.refreshingText\"\n        ></ion-refresher-content>\n      }\n    </ion-refresher>\n\n    <ng-content></ng-content>\n  `,\n  styles: [\n    `\n      :host {\n        display: block;\n        position: relative;\n      }\n\n      .refresher-custom-content {\n        display: flex;\n        flex-direction: column;\n        align-items: center;\n        justify-content: center;\n        gap: 8px;\n        padding: 16px;\n      }\n    `,\n  ],\n})\nexport class RefresherComponent {\n  @ViewChild('refresher') ionRefresher!: IonRefresher;\n\n  /** Templates personalizados via content projection */\n  @ContentChild('pullingIndicator') pullingIndicator?: TemplateRef<{ progress: number }>;\n  @ContentChild('refreshingIndicator') refreshingIndicator?: TemplateRef<void>;\n  @ContentChild('completingIndicator') completingIndicator?: TemplateRef<void>;\n\n  /** Configuracion del refresher */\n  @Input() props: RefresherMetadata = {};\n\n  /** Evento emitido cuando se activa el refresh */\n  @Output() refresh = new EventEmitter<RefreshEvent>();\n\n  /** Evento de progreso durante el pull */\n  @Output() pullProgressChange = new EventEmitter<RefreshPullEvent>();\n\n  /** Evento de cambio de estado */\n  @Output() stateChange = new EventEmitter<RefresherState>();\n\n  private i18n = inject(I18nService);\n\n  /** Estado actual del refresher */\n  readonly state = signal<RefresherState>('idle');\n\n  /** Progreso actual del pull (0-1) */\n  readonly pullProgress = signal<number>(0);\n\n  /** Get pulling text with i18n fallback */\n  getPullingText(): string {\n    return this.props.pullingText || this.i18n.t('pullToRefresh');\n  }\n\n  /** Get refreshing text with i18n fallback */\n  getRefreshingText(): string {\n    return this.props.refreshingText || this.i18n.t('loading');\n  }\n\n  /** Props combinados con defaults */\n  get mergedProps(): RefresherMetadata {\n    return {\n      ...DEFAULT_REFRESHER_METADATA,\n      ...this.props,\n      pullingText: this.getPullingText(),\n      refreshingText: this.getRefreshingText(),\n    };\n  }\n\n  /** Si hay indicadores personalizados */\n  get hasCustomIndicator(): boolean {\n    return !!(\n      this.pullingIndicator ||\n      this.refreshingIndicator ||\n      this.completingIndicator ||\n      this.props.indicatorConfig\n    );\n  }\n\n  /**\n   * Activa programaticamente el refresh.\n   */\n  triggerRefresh(): void {\n    this.state.set('refreshing');\n    this.emitRefreshEvent();\n  }\n\n  /**\n   * Completa la operacion de refresh actual.\n   */\n  complete(): void {\n    this.state.set('completing');\n    this.stateChange.emit('completing');\n    this.ionRefresher?.complete();\n\n    // Resetear a idle despues de la animacion\n    setTimeout(() => {\n      this.state.set('idle');\n      this.stateChange.emit('idle');\n    }, this.mergedProps.closeDuration ?? 280);\n  }\n\n  /**\n   * Cancela la operacion de refresh actual.\n   */\n  cancel(): void {\n    this.ionRefresher?.cancel();\n    this.state.set('idle');\n    this.stateChange.emit('idle');\n  }\n\n  /** Handler para evento ionRefresh */\n  onIonRefresh(event: CustomEvent): void {\n    this.state.set('refreshing');\n    this.stateChange.emit('refreshing');\n    this.emitRefreshEvent();\n  }\n\n  /** Handler para evento ionPull */\n  onIonPull(event: CustomEvent): void {\n    const detail = event?.detail;\n    if (!detail || typeof detail.progress !== 'number') return;\n\n    const progress = Math.min(detail.progress, 1);\n    this.pullProgress.set(progress);\n\n    this.pullProgressChange.emit({\n      progress,\n      thresholdReached: progress >= 1,\n    });\n  }\n\n  /** Handler para evento ionStart */\n  onIonStart(): void {\n    this.state.set('pulling');\n    this.stateChange.emit('pulling');\n  }\n\n  private emitRefreshEvent(): void {\n    const event: RefreshEvent = {\n      complete: () => this.complete(),\n      cancel: () => this.cancel(),\n      timestamp: new Date(),\n    };\n    this.refresh.emit(event);\n  }\n}\n"]}
|
|
@@ -23,6 +23,8 @@ import { SelectSearchComponent } from '../../molecules/select-search/select-sear
|
|
|
23
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 { PhoneInputComponent } from '../../molecules/phone-input/phone-input.component';
|
|
26
|
+
import { ToggleInputComponent } from '../../molecules/toggle-input/toggle-input.component';
|
|
27
|
+
import { CheckboxRadioInputComponent } from '../../molecules/checkbox-radio-input/checkbox-radio-input.component';
|
|
26
28
|
import { ComponentStates, InputType } from '../../types';
|
|
27
29
|
import * as i0 from "@angular/core";
|
|
28
30
|
import * as i1 from "@angular/forms";
|
|
@@ -290,6 +292,12 @@ export class FormComponent {
|
|
|
290
292
|
<ng-container *ngIf="f.type === types.PHONE">
|
|
291
293
|
<val-phone-input [props]="getFieldProp(f)"></val-phone-input>
|
|
292
294
|
</ng-container>
|
|
295
|
+
<ng-container *ngIf="f.type === types.TOGGLE">
|
|
296
|
+
<val-toggle-input [props]="getFieldProp(f)"></val-toggle-input>
|
|
297
|
+
</ng-container>
|
|
298
|
+
<ng-container *ngIf="f.type === types.CHECKBOX_RADIO">
|
|
299
|
+
<val-checkbox-radio-input [props]="getFieldProp(f)"></val-checkbox-radio-input>
|
|
300
|
+
</ng-container>
|
|
293
301
|
<val-hint [props]="getFieldProp(f)"></val-hint>
|
|
294
302
|
</div>
|
|
295
303
|
<val-divider [props]="{ fill: 'solid', size: 'medium', color: 'medium' }"></val-divider>
|
|
@@ -301,7 +309,7 @@ export class FormComponent {
|
|
|
301
309
|
></val-button-group>
|
|
302
310
|
</form>
|
|
303
311
|
</div>
|
|
304
|
-
`, 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}}.field-description{display:block;font-size:.75rem;color:var(--ion-color-medium);margin-bottom:.25rem;line-height:1.4}\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: ["preset", "props"] }, { kind: "component", type: CheckInputComponent, selector: "val-check-input", inputs: ["preset", "props"] }, { 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: ["preset", "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: ["preset", "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: ["preset", "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: ["preset", "props"] }, { kind: "component", type: PhoneInputComponent, selector: "val-phone-input", inputs: ["preset", "props"], outputs: ["phoneChange"] }] }); }
|
|
312
|
+
`, 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}}.field-description{display:block;font-size:.75rem;color:var(--ion-color-medium);margin-bottom:.25rem;line-height:1.4}\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: ["preset", "props"] }, { kind: "component", type: CheckInputComponent, selector: "val-check-input", inputs: ["preset", "props"] }, { 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: ["preset", "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: ["preset", "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: ["preset", "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: ["preset", "props"] }, { kind: "component", type: PhoneInputComponent, selector: "val-phone-input", inputs: ["preset", "props"], outputs: ["phoneChange"] }, { kind: "component", type: ToggleInputComponent, selector: "val-toggle-input", inputs: ["preset", "props"] }, { kind: "component", type: CheckboxRadioInputComponent, selector: "val-checkbox-radio-input", inputs: ["props"] }] }); }
|
|
305
313
|
}
|
|
306
314
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormComponent, decorators: [{
|
|
307
315
|
type: Component,
|
|
@@ -329,6 +337,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
329
337
|
MultiSelectSearchComponent,
|
|
330
338
|
SearchSelectorComponent,
|
|
331
339
|
PhoneInputComponent,
|
|
340
|
+
ToggleInputComponent,
|
|
341
|
+
CheckboxRadioInputComponent,
|
|
332
342
|
], template: `
|
|
333
343
|
<div class="container">
|
|
334
344
|
<form [formGroup]="form">
|
|
@@ -393,6 +403,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
393
403
|
<ng-container *ngIf="f.type === types.PHONE">
|
|
394
404
|
<val-phone-input [props]="getFieldProp(f)"></val-phone-input>
|
|
395
405
|
</ng-container>
|
|
406
|
+
<ng-container *ngIf="f.type === types.TOGGLE">
|
|
407
|
+
<val-toggle-input [props]="getFieldProp(f)"></val-toggle-input>
|
|
408
|
+
</ng-container>
|
|
409
|
+
<ng-container *ngIf="f.type === types.CHECKBOX_RADIO">
|
|
410
|
+
<val-checkbox-radio-input [props]="getFieldProp(f)"></val-checkbox-radio-input>
|
|
411
|
+
</ng-container>
|
|
396
412
|
<val-hint [props]="getFieldProp(f)"></val-hint>
|
|
397
413
|
</div>
|
|
398
414
|
<val-divider [props]="{ fill: 'solid', size: 'medium', color: 'medium' }"></val-divider>
|
|
@@ -414,4 +430,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
414
430
|
}], onSelectChange: [{
|
|
415
431
|
type: Output
|
|
416
432
|
}] } });
|
|
417
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/form/form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, DoCheck, ElementRef, EventEmitter, Input, OnChanges, OnInit, Output, SimpleChanges } 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 { PhoneInputComponent } from '../../molecules/phone-input/phone-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    PhoneInputComponent,\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: 'small',\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 *ngIf=\"f.type !== types.PHONE\" [props]=\"{ content: f.label, size: 'small', color: 'dark', bold: false }\"></val-title>\n            <span *ngIf=\"f.description\" class=\"field-description\">{{ f.description }}</span>\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            <ng-container *ngIf=\"f.type === types.PHONE\">\n              <val-phone-input [props]=\"getFieldProp(f)\"></val-phone-input>\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, OnChanges, DoCheck {\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 para evitar crear nuevos objetos en cada ciclo de change detection\n  private fieldPropsCache: Map<string, InputMetadata> = new Map();\n  private actionsCache: ButtonMetadata[] = [];\n\n  // Estado previo para detectar cambios en ngDoCheck\n  private previousState: string | undefined;\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(\n          x =>\n            x.type === this.types.SELECT ||\n            x.type === this.types.TEXT ||\n            x.type === this.types.SEARCH_SELECT ||\n            x.type === this.types.MULTI_SELECT\n        )\n        .forEach(field => {\n          this.trackSelectChanges(field.name);\n        });\n    });\n\n    // Construir cache inicial\n    this.rebuildFieldPropsCache();\n    this.updateActionsCache();\n    this.previousState = this.props.state;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    // Cuando props cambia, reconstruir el cache\n    if (changes['props'] && this.form) {\n      this.rebuildFieldPropsCache();\n      this.updateActionsCache();\n      this.previousState = this.props.state;\n    }\n  }\n\n  ngDoCheck() {\n    // Detectar cambios en el estado del formulario (mutación sin cambiar referencia)\n    if (this.form && this.props && this.props.state !== this.previousState) {\n      this.previousState = this.props.state;\n      this.updateActionsCache();\n    }\n  }\n\n  /**\n   * Reconstruye el cache de props para cada field.\n   * Esto evita crear nuevos objetos en cada ciclo de change detection.\n   */\n  private rebuildFieldPropsCache() {\n    this.fieldPropsCache.clear();\n    this.props.sections.forEach(section => {\n      section.fields.forEach(field => {\n        const token = field.token || `input-${field.type}-${field.name}`;\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({ emitEvent: false });\n            toControl.disable({ emitEvent: false });\n          } else {\n            fromControl.enable({ emitEvent: false });\n            toControl.enable({ emitEvent: false });\n          }\n\n          this.fieldPropsCache.set(field.name, {\n            ...field,\n            token,\n            fromControl,\n            toControl,\n            control: undefined,\n          });\n        } else {\n          const control = this.getControl(field.name);\n          if (field.state === ComponentStates.DISABLED) {\n            control.disable({ emitEvent: false });\n          } else {\n            control.enable({ emitEvent: false });\n          }\n\n          this.fieldPropsCache.set(field.name, {\n            ...field,\n            token,\n            control,\n          });\n        }\n      });\n    });\n  }\n\n  /**\n   * Actualiza el cache de actions basado en el estado actual del form.\n   */\n  private updateActionsCache() {\n    let actionState = this.props.actions.state;\n\n    if (this.props.state === ComponentStates.WORKING) {\n      actionState = ComponentStates.WORKING;\n    } else if (this.props.state === ComponentStates.DISABLED) {\n      actionState = ComponentStates.DISABLED;\n    } else if (this.props.state === ComponentStates.ENABLED || this.form?.valid) {\n      actionState = ComponentStates.ENABLED;\n    }\n\n    this.actionsCache = [{ ...this.props.actions, state: actionState }];\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    // Retornar del cache para evitar crear nuevos objetos en cada ciclo de change detection\n    const cached = this.fieldPropsCache.get(field.name);\n    if (cached) {\n      return cached;\n    }\n\n    // Fallback: generar y cachear si no existe (no debería ocurrir normalmente)\n    const token = field.token || `input-${field.type}-${field.name}`;\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({ emitEvent: false });\n        toControl.disable({ emitEvent: false });\n      } else {\n        fromControl.enable({ emitEvent: false });\n        toControl.enable({ emitEvent: false });\n      }\n\n      const props = {\n        ...field,\n        token,\n        fromControl,\n        toControl,\n        control: undefined,\n      };\n      this.fieldPropsCache.set(field.name, props);\n      return props;\n    } else {\n      const control = this.getControl(field.name);\n      if (field.state === ComponentStates.DISABLED) {\n        control.disable({ emitEvent: false });\n      } else {\n        control.enable({ emitEvent: false });\n      }\n      const props = {\n        ...field,\n        token,\n        control,\n      };\n      this.fieldPropsCache.set(field.name, props);\n      return props;\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    // Retornar del cache para evitar crear nuevos objetos en cada ciclo de change detection\n    return this.actionsCache;\n  }\n}\n"]}
|
|
433
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form.component.js","sourceRoot":"","sources":["../../../../../../../src/lib/components/organisms/form/form.component.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { Component, DoCheck, ElementRef, EventEmitter, Input, OnChanges, OnInit, Output, SimpleChanges } 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 { PhoneInputComponent } from '../../molecules/phone-input/phone-input.component';\nimport { ToggleInputComponent } from '../../molecules/toggle-input/toggle-input.component';\nimport { CheckboxRadioInputComponent } from '../../molecules/checkbox-radio-input/checkbox-radio-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    PhoneInputComponent,\n    ToggleInputComponent,\n    CheckboxRadioInputComponent,\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: 'small',\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 *ngIf=\"f.type !== types.PHONE\" [props]=\"{ content: f.label, size: 'small', color: 'dark', bold: false }\"></val-title>\n            <span *ngIf=\"f.description\" class=\"field-description\">{{ f.description }}</span>\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            <ng-container *ngIf=\"f.type === types.PHONE\">\n              <val-phone-input [props]=\"getFieldProp(f)\"></val-phone-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.TOGGLE\">\n              <val-toggle-input [props]=\"getFieldProp(f)\"></val-toggle-input>\n            </ng-container>\n            <ng-container *ngIf=\"f.type === types.CHECKBOX_RADIO\">\n              <val-checkbox-radio-input [props]=\"getFieldProp(f)\"></val-checkbox-radio-input>\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, OnChanges, DoCheck {\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 para evitar crear nuevos objetos en cada ciclo de change detection\n  private fieldPropsCache: Map<string, InputMetadata> = new Map();\n  private actionsCache: ButtonMetadata[] = [];\n\n  // Estado previo para detectar cambios en ngDoCheck\n  private previousState: string | undefined;\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(\n          x =>\n            x.type === this.types.SELECT ||\n            x.type === this.types.TEXT ||\n            x.type === this.types.SEARCH_SELECT ||\n            x.type === this.types.MULTI_SELECT\n        )\n        .forEach(field => {\n          this.trackSelectChanges(field.name);\n        });\n    });\n\n    // Construir cache inicial\n    this.rebuildFieldPropsCache();\n    this.updateActionsCache();\n    this.previousState = this.props.state;\n  }\n\n  ngOnChanges(changes: SimpleChanges) {\n    // Cuando props cambia, reconstruir el cache\n    if (changes['props'] && this.form) {\n      this.rebuildFieldPropsCache();\n      this.updateActionsCache();\n      this.previousState = this.props.state;\n    }\n  }\n\n  ngDoCheck() {\n    // Detectar cambios en el estado del formulario (mutación sin cambiar referencia)\n    if (this.form && this.props && this.props.state !== this.previousState) {\n      this.previousState = this.props.state;\n      this.updateActionsCache();\n    }\n  }\n\n  /**\n   * Reconstruye el cache de props para cada field.\n   * Esto evita crear nuevos objetos en cada ciclo de change detection.\n   */\n  private rebuildFieldPropsCache() {\n    this.fieldPropsCache.clear();\n    this.props.sections.forEach(section => {\n      section.fields.forEach(field => {\n        const token = field.token || `input-${field.type}-${field.name}`;\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({ emitEvent: false });\n            toControl.disable({ emitEvent: false });\n          } else {\n            fromControl.enable({ emitEvent: false });\n            toControl.enable({ emitEvent: false });\n          }\n\n          this.fieldPropsCache.set(field.name, {\n            ...field,\n            token,\n            fromControl,\n            toControl,\n            control: undefined,\n          });\n        } else {\n          const control = this.getControl(field.name);\n          if (field.state === ComponentStates.DISABLED) {\n            control.disable({ emitEvent: false });\n          } else {\n            control.enable({ emitEvent: false });\n          }\n\n          this.fieldPropsCache.set(field.name, {\n            ...field,\n            token,\n            control,\n          });\n        }\n      });\n    });\n  }\n\n  /**\n   * Actualiza el cache de actions basado en el estado actual del form.\n   */\n  private updateActionsCache() {\n    let actionState = this.props.actions.state;\n\n    if (this.props.state === ComponentStates.WORKING) {\n      actionState = ComponentStates.WORKING;\n    } else if (this.props.state === ComponentStates.DISABLED) {\n      actionState = ComponentStates.DISABLED;\n    } else if (this.props.state === ComponentStates.ENABLED || this.form?.valid) {\n      actionState = ComponentStates.ENABLED;\n    }\n\n    this.actionsCache = [{ ...this.props.actions, state: actionState }];\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    // Retornar del cache para evitar crear nuevos objetos en cada ciclo de change detection\n    const cached = this.fieldPropsCache.get(field.name);\n    if (cached) {\n      return cached;\n    }\n\n    // Fallback: generar y cachear si no existe (no debería ocurrir normalmente)\n    const token = field.token || `input-${field.type}-${field.name}`;\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({ emitEvent: false });\n        toControl.disable({ emitEvent: false });\n      } else {\n        fromControl.enable({ emitEvent: false });\n        toControl.enable({ emitEvent: false });\n      }\n\n      const props = {\n        ...field,\n        token,\n        fromControl,\n        toControl,\n        control: undefined,\n      };\n      this.fieldPropsCache.set(field.name, props);\n      return props;\n    } else {\n      const control = this.getControl(field.name);\n      if (field.state === ComponentStates.DISABLED) {\n        control.disable({ emitEvent: false });\n      } else {\n        control.enable({ emitEvent: false });\n      }\n      const props = {\n        ...field,\n        token,\n        control,\n      };\n      this.fieldPropsCache.set(field.name, props);\n      return props;\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    // Retornar del cache para evitar crear nuevos objetos en cada ciclo de change detection\n    return this.actionsCache;\n  }\n}\n"]}
|
|
@@ -44,6 +44,7 @@ export var InputType;
|
|
|
44
44
|
InputType[InputType["RANGE"] = 19] = "RANGE";
|
|
45
45
|
InputType[InputType["PHONE"] = 20] = "PHONE";
|
|
46
46
|
InputType[InputType["CURRENCY"] = 21] = "CURRENCY";
|
|
47
|
+
InputType[InputType["CHECKBOX_RADIO"] = 22] = "CHECKBOX_RADIO";
|
|
47
48
|
})(InputType || (InputType = {}));
|
|
48
49
|
/**
|
|
49
50
|
* Possible action types for a toolbar.
|
|
@@ -55,4 +56,4 @@ export var ToolbarActionType;
|
|
|
55
56
|
ToolbarActionType["IMAGE"] = "IMAGE";
|
|
56
57
|
ToolbarActionType["BUTTON"] = "BUTTON";
|
|
57
58
|
})(ToolbarActionType || (ToolbarActionType = {}));
|
|
58
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/components/types.ts"],"names":[],"mappings":"AAQA,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,QAAQ,GAAmB,UAAU,CAAC;AAC5C,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,KAAK,GAAmB,OAAO,CAAC;AAEtC;;GAEG;AACH,MAAM,CAAC,MAAM,eAAe,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;AAErE;;GAEG;AACH,MAAM,CAAN,IAAY,UAMX;AAND,WAAY,UAAU;IACpB,iEAAe,CAAA;IACf,mEAAgB,CAAA;IAChB,iEAAe,CAAA;IACf,+DAAc,CAAA;IACd,uEAAkB,CAAA;AACpB,CAAC,EANW,UAAU,KAAV,UAAU,QAMrB;AAcD;;GAEG;AACH,MAAM,CAAN,IAAY,SAuBX;AAvBD,WAAY,SAAS;IACnB,yCAAI,CAAA;IACJ,iDAAQ,CAAA;IACR,2CAAK,CAAA;IACL,iDAAQ,CAAA;IACR,+CAAO,CAAA;IACP,6CAAM,CAAA;IACN,6DAAc,CAAA;IACd,iDAAQ,CAAA;IACR,yCAAI,CAAA;IACJ,qDAAU,CAAA;IACV,0CAAI,CAAA;IACJ,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,8CAAM,CAAA;IACN,4DAAa,CAAA;IACb,0DAAY,CAAA;IACZ,wEAAmB,CAAA;IACnB,0CAAI,CAAA;IACJ,8CAAM,CAAA;IACN,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,kDAAQ,CAAA;AACV,CAAC,EAvBW,SAAS,KAAT,SAAS,QAuBpB;AA6ID;;GAEG;AACH,MAAM,CAAN,IAAY,iBAKX;AALD,WAAY,iBAAiB;IAC3B,sCAAiB,CAAA;IACjB,kCAAa,CAAA;IACb,oCAAe,CAAA;IACf,sCAAiB,CAAA;AACnB,CAAC,EALW,iBAAiB,KAAjB,iBAAiB,QAK5B","sourcesContent":["import { FormControl, ValidatorFn } from '@angular/forms';\nimport { Color } from '@ionic/core';\nimport { ImageMetadata } from './atoms/image/types';\n\n/**\n * Possible states for an interactive component.\n */\nexport type ComponentState = 'ENABLED' | 'DISABLED' | 'WORKING' | 'ERROR';\nconst ENABLED: ComponentState = 'ENABLED';\nconst DISABLED: ComponentState = 'DISABLED';\nconst WORKING: ComponentState = 'WORKING';\nconst ERROR: ComponentState = 'ERROR';\n\n/**\n * Object containing all possible component states.\n */\nexport const ComponentStates = { ENABLED, DISABLED, WORKING, ERROR };\n\n/**\n * Types of actions that a button or link can perform.\n */\nexport enum ActionType {\n  BROWSER_NEW_TAB, // Open in a new browser tab\n  BROWSER_DOWNLOAD, // Download via browser\n  NATIVE_DOWNLOAD, // Download using native capabilities\n  APP_NAVIGATION, // Internal app navigation\n  BROWSER_NAVIGATION, // Navigation in the browser\n}\n\n/**\n * Represents an executable action for a button or link.\n */\nexport type Action = {\n  /** Action description */\n  description: string;\n  /** Action type */\n  type: ActionType;\n  /** Action source or destination */\n  source: string;\n};\n\n/**\n * Supported input types for forms.\n */\nexport enum InputType {\n  TEXT,\n  TEXTAREA,\n  EMAIL,\n  PASSWORD,\n  COMMENT,\n  NUMBER,\n  NUMBER_FROM_TO,\n  PIN_CODE,\n  DATE,\n  DATE_RANGE,\n  HOUR,\n  CHECK,\n  RADIO,\n  SELECT,\n  SEARCH_SELECT,\n  MULTI_SELECT,\n  MULTI_SELECT_SIMPLE,\n  FILE,\n  TOGGLE,\n  RANGE,\n  PHONE,\n  CURRENCY,\n}\n\n/**\n * Option for select, radio, etc. inputs.\n */\nexport type InputOption = {\n  /** Unique option identifier */\n  id: string;\n  /** Display name */\n  name: string;\n  /** Whether the option is selected by default */\n  selected?: boolean;\n  /** Display order */\n  order: number;\n};\n\n/**\n * Metadata for a form field.\n */\nexport type InputMetadata = {\n  /** Associated form control (optional, created internally by val-form) */\n  control?: FormControl;\n  /** From control (only for NUMBER_FROM_TO type) */\n  fromControl?: FormControl;\n  /** To control (only for NUMBER_FROM_TO type) */\n  toControl?: FormControl;\n  /** Unique token for the input */\n  token: string;\n  /** Display label */\n  label: string;\n  /** Field name */\n  name: string;\n  /** Help text (shown below input, for validation hints) */\n  hint: string;\n  /** Description text (shown below label, explains the field purpose) */\n  description?: string;\n  /** Input placeholder */\n  placeholder: string;\n  /** Input type */\n  type: InputType;\n  /** Display order */\n  order: number;\n  /** Associated validators */\n  validators: ValidatorFn[];\n  /** Options (for select, radio, etc.) */\n  options?: InputOption[];\n  /** Allowed range (for number, date, etc.) */\n  range?: {\n    min: number;\n    max: number;\n  };\n  /** Custom error messages */\n  errors: {\n    [key: string]: string;\n  };\n  /** Initial value for the field */\n  value?: string;\n  /** Default value configuration - string for custom defaults, true for auto defaults */\n  withDefault?: string | boolean;\n  /** Field state */\n  state: ComponentState;\n  /** Label for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromLabel?: string;\n  /** Label for \"to\" field (only for NUMBER_FROM_TO type) */\n  toLabel?: string;\n  /** Placeholder for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromPlaceholder?: string;\n  /** Placeholder for \"to\" field (only for NUMBER_FROM_TO type) */\n  toPlaceholder?: string;\n\n  // i18n support properties\n  /** Key for content lookup */\n  contentKey?: string;\n  /** Component class name for content lookup */\n  contentClass?: string;\n  /** Fallback text if content key is not found */\n  contentFallback?: string;\n\n  // select-input specific i18n properties\n  /** Custom header text for select modal */\n  modalHeader?: string;\n  /** Custom cancel button text for select modal */\n  cancelText?: string;\n  /** Custom OK button text for select modal */\n  okText?: string;\n\n  // check-input specific properties\n  /** Position of label for checkbox ('start' | 'end') */\n  labelPlacement?: 'start' | 'end';\n\n  // PIN_CODE specific properties\n  /** Number of digits in PIN (default: 5, only for PIN_CODE type) */\n  length?: number;\n  /** Size of the input boxes: 'small' | 'medium' | 'large' (default: 'medium', only for PIN_CODE type) */\n  size?: 'small' | 'medium' | 'large';\n  /** Allow only numbers (default: true, only for PIN_CODE type) */\n  allowNumbersOnly?: boolean;\n  /** Mask input - show dots instead of characters (only for PIN_CODE type) */\n  mask?: boolean;\n  /** Auto focus first input (only for PIN_CODE type) */\n  autoFocus?: boolean;\n  /** Custom input styles (only for PIN_CODE type) */\n  inputStyles?: Record<string, string>;\n};\n\n/**\n * A section in a form, grouping multiple fields.\n */\nexport type FormSection = {\n  /** Section name */\n  name: string;\n  /** Display order */\n  order: number;\n  /** Fields included in the section */\n  fields: InputMetadata[];\n};\n\n/**\n * Data sent when submitting a form.\n */\nexport type FormSubmit = {\n  /** Form field values as key-value pairs */\n  fields: Record<string, any>;\n  /** Optional token for the operation */\n  token?: string;\n};\n\n/**\n * Metadata for a complete form.\n */\nexport type FormMetadata = {\n  /** Form name */\n  name: string;\n  /** Form sections */\n  sections: FormSection[];\n  /** Action buttons configuration */\n  actions: ButtonMetadata;\n  /** Global form state */\n  state: ComponentState;\n};\n\n/**\n * Possible action types for a toolbar.\n */\nexport enum ToolbarActionType {\n  AVATAR = 'AVATAR',\n  ICON = 'ICON',\n  IMAGE = 'IMAGE',\n  BUTTON = 'BUTTON',\n}\n\n/**\n * Toolbar action definition.\n */\nexport type ToolbarAction = {\n  /** Action type */\n  type: 'AVATAR' | 'ICON' | 'IMAGE' | 'BUTTON';\n  /** Optional token identifier */\n  token?: string;\n  /** Toolbar position */\n  position: 'left' | 'right' | 'center';\n  /** Optional description */\n  description?: string;\n  /** Associated image (if any) */\n  image?: ImageMetadata;\n};\n\n/**\n * Metadata for an icon.\n */\nexport interface IconMetada {\n  /** Icon name */\n  name: string;\n  /** Icon slot position */\n  slot: 'start' | 'end';\n}\n\n/**\n * Button configuration object.\n * Supports both static and reactive content.\n * @type {ButtonMetadata}\n * @property text - Static button label (takes precedence over textConfig).\n * @property textConfig - Reactive content configuration for button text.\n * @property color - The button color (Ionic color string).\n * @property icon - Icon to display (optional).\n * @property state - Button state (enabled, disabled, working, etc.).\n * @property expand, fill, size, shape, href, target, download, handler, etc. - See ButtonMetadata for all options.\n */\nexport interface ButtonMetadata {\n  /** Associated action type */\n  actionType?: ActionType;\n  /** Button expansion */\n  expand?: 'full' | 'block';\n  /** Associated link */\n  link?: string;\n  /** Associated href link */\n  href?: string;\n  /** Link target */\n  target?: '_blank' | '_self' | '_parent' | '_top';\n  /** Download file name */\n  download?: string;\n  /** Button color */\n  color: Color;\n  /** Button state */\n  state: ComponentState;\n  /** Static display text (takes precedence over reactive content) */\n  text?: string;\n  /** Reactive content configuration for button text */\n  contentKey?: string;\n  /** Component class name for content lookup (required with contentKey) */\n  contentClass?: string;\n  /** Fallback text if contentKey is not found */\n  contentFallback?: string;\n  /** Values to interpolate into the content string */\n  contentInterpolation?: Record<string, string | number>;\n  /** Associated icon */\n  icon?: IconMetada;\n  /** Button shape */\n  shape?: 'round';\n  /** Button size */\n  size?: 'small' | 'default' | 'large';\n  /** Button fill */\n  fill?: 'clear' | 'outline' | 'solid' | 'default';\n  /** Button type */\n  type: 'button' | 'submit' | 'reset';\n  /** Optional token identifier */\n  token?: string;\n  /** Optional reference */\n  ref?: any;\n  /** Action handler */\n  handler?: (value: any) => any | Promise<any>;\n}\n\n/**\n * Configuration for reactive content in val-button component.\n * Use this interface when you only need to specify content-related properties.\n * This follows the same pattern as TextContentConfig for consistency.\n */\nexport interface ButtonContentConfig {\n  contentKey: string;\n  contentClass: string;\n  contentFallback?: string;\n  contentInterpolation?: Record<string, string | number>;\n}\n"]}
|
|
59
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["../../../../../src/lib/components/types.ts"],"names":[],"mappings":"AAQA,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,QAAQ,GAAmB,UAAU,CAAC;AAC5C,MAAM,OAAO,GAAmB,SAAS,CAAC;AAC1C,MAAM,KAAK,GAAmB,OAAO,CAAC;AAEtC;;GAEG;AACH,MAAM,CAAC,MAAM,eAAe,GAAG,EAAE,OAAO,EAAE,QAAQ,EAAE,OAAO,EAAE,KAAK,EAAE,CAAC;AAErE;;GAEG;AACH,MAAM,CAAN,IAAY,UAMX;AAND,WAAY,UAAU;IACpB,iEAAe,CAAA;IACf,mEAAgB,CAAA;IAChB,iEAAe,CAAA;IACf,+DAAc,CAAA;IACd,uEAAkB,CAAA;AACpB,CAAC,EANW,UAAU,KAAV,UAAU,QAMrB;AAcD;;GAEG;AACH,MAAM,CAAN,IAAY,SAwBX;AAxBD,WAAY,SAAS;IACnB,yCAAI,CAAA;IACJ,iDAAQ,CAAA;IACR,2CAAK,CAAA;IACL,iDAAQ,CAAA;IACR,+CAAO,CAAA;IACP,6CAAM,CAAA;IACN,6DAAc,CAAA;IACd,iDAAQ,CAAA;IACR,yCAAI,CAAA;IACJ,qDAAU,CAAA;IACV,0CAAI,CAAA;IACJ,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,8CAAM,CAAA;IACN,4DAAa,CAAA;IACb,0DAAY,CAAA;IACZ,wEAAmB,CAAA;IACnB,0CAAI,CAAA;IACJ,8CAAM,CAAA;IACN,4CAAK,CAAA;IACL,4CAAK,CAAA;IACL,kDAAQ,CAAA;IACR,8DAAc,CAAA;AAChB,CAAC,EAxBW,SAAS,KAAT,SAAS,QAwBpB;AA6ID;;GAEG;AACH,MAAM,CAAN,IAAY,iBAKX;AALD,WAAY,iBAAiB;IAC3B,sCAAiB,CAAA;IACjB,kCAAa,CAAA;IACb,oCAAe,CAAA;IACf,sCAAiB,CAAA;AACnB,CAAC,EALW,iBAAiB,KAAjB,iBAAiB,QAK5B","sourcesContent":["import { FormControl, ValidatorFn } from '@angular/forms';\nimport { Color } from '@ionic/core';\nimport { ImageMetadata } from './atoms/image/types';\n\n/**\n * Possible states for an interactive component.\n */\nexport type ComponentState = 'ENABLED' | 'DISABLED' | 'WORKING' | 'ERROR';\nconst ENABLED: ComponentState = 'ENABLED';\nconst DISABLED: ComponentState = 'DISABLED';\nconst WORKING: ComponentState = 'WORKING';\nconst ERROR: ComponentState = 'ERROR';\n\n/**\n * Object containing all possible component states.\n */\nexport const ComponentStates = { ENABLED, DISABLED, WORKING, ERROR };\n\n/**\n * Types of actions that a button or link can perform.\n */\nexport enum ActionType {\n  BROWSER_NEW_TAB, // Open in a new browser tab\n  BROWSER_DOWNLOAD, // Download via browser\n  NATIVE_DOWNLOAD, // Download using native capabilities\n  APP_NAVIGATION, // Internal app navigation\n  BROWSER_NAVIGATION, // Navigation in the browser\n}\n\n/**\n * Represents an executable action for a button or link.\n */\nexport type Action = {\n  /** Action description */\n  description: string;\n  /** Action type */\n  type: ActionType;\n  /** Action source or destination */\n  source: string;\n};\n\n/**\n * Supported input types for forms.\n */\nexport enum InputType {\n  TEXT,\n  TEXTAREA,\n  EMAIL,\n  PASSWORD,\n  COMMENT,\n  NUMBER,\n  NUMBER_FROM_TO,\n  PIN_CODE,\n  DATE,\n  DATE_RANGE,\n  HOUR,\n  CHECK,\n  RADIO,\n  SELECT,\n  SEARCH_SELECT,\n  MULTI_SELECT,\n  MULTI_SELECT_SIMPLE,\n  FILE,\n  TOGGLE,\n  RANGE,\n  PHONE,\n  CURRENCY,\n  CHECKBOX_RADIO,\n}\n\n/**\n * Option for select, radio, etc. inputs.\n */\nexport type InputOption = {\n  /** Unique option identifier */\n  id: string;\n  /** Display name */\n  name: string;\n  /** Whether the option is selected by default */\n  selected?: boolean;\n  /** Display order */\n  order: number;\n};\n\n/**\n * Metadata for a form field.\n */\nexport type InputMetadata = {\n  /** Associated form control (optional, created internally by val-form) */\n  control?: FormControl;\n  /** From control (only for NUMBER_FROM_TO type) */\n  fromControl?: FormControl;\n  /** To control (only for NUMBER_FROM_TO type) */\n  toControl?: FormControl;\n  /** Unique token for the input */\n  token: string;\n  /** Display label */\n  label: string;\n  /** Field name */\n  name: string;\n  /** Help text (shown below input, for validation hints) */\n  hint: string;\n  /** Description text (shown below label, explains the field purpose) */\n  description?: string;\n  /** Input placeholder */\n  placeholder: string;\n  /** Input type */\n  type: InputType;\n  /** Display order */\n  order: number;\n  /** Associated validators */\n  validators: ValidatorFn[];\n  /** Options (for select, radio, etc.) */\n  options?: InputOption[];\n  /** Allowed range (for number, date, etc.) */\n  range?: {\n    min: number;\n    max: number;\n  };\n  /** Custom error messages */\n  errors: {\n    [key: string]: string;\n  };\n  /** Initial value for the field */\n  value?: string;\n  /** Default value configuration - string for custom defaults, true for auto defaults */\n  withDefault?: string | boolean;\n  /** Field state */\n  state: ComponentState;\n  /** Label for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromLabel?: string;\n  /** Label for \"to\" field (only for NUMBER_FROM_TO type) */\n  toLabel?: string;\n  /** Placeholder for \"from\" field (only for NUMBER_FROM_TO type) */\n  fromPlaceholder?: string;\n  /** Placeholder for \"to\" field (only for NUMBER_FROM_TO type) */\n  toPlaceholder?: string;\n\n  // i18n support properties\n  /** Key for content lookup */\n  contentKey?: string;\n  /** Component class name for content lookup */\n  contentClass?: string;\n  /** Fallback text if content key is not found */\n  contentFallback?: string;\n\n  // select-input specific i18n properties\n  /** Custom header text for select modal */\n  modalHeader?: string;\n  /** Custom cancel button text for select modal */\n  cancelText?: string;\n  /** Custom OK button text for select modal */\n  okText?: string;\n\n  // check-input specific properties\n  /** Position of label for checkbox ('start' | 'end') */\n  labelPlacement?: 'start' | 'end';\n\n  // PIN_CODE specific properties\n  /** Number of digits in PIN (default: 5, only for PIN_CODE type) */\n  length?: number;\n  /** Size of the input boxes: 'small' | 'medium' | 'large' (default: 'medium', only for PIN_CODE type) */\n  size?: 'small' | 'medium' | 'large';\n  /** Allow only numbers (default: true, only for PIN_CODE type) */\n  allowNumbersOnly?: boolean;\n  /** Mask input - show dots instead of characters (only for PIN_CODE type) */\n  mask?: boolean;\n  /** Auto focus first input (only for PIN_CODE type) */\n  autoFocus?: boolean;\n  /** Custom input styles (only for PIN_CODE type) */\n  inputStyles?: Record<string, string>;\n};\n\n/**\n * A section in a form, grouping multiple fields.\n */\nexport type FormSection = {\n  /** Section name */\n  name: string;\n  /** Display order */\n  order: number;\n  /** Fields included in the section */\n  fields: InputMetadata[];\n};\n\n/**\n * Data sent when submitting a form.\n */\nexport type FormSubmit = {\n  /** Form field values as key-value pairs */\n  fields: Record<string, any>;\n  /** Optional token for the operation */\n  token?: string;\n};\n\n/**\n * Metadata for a complete form.\n */\nexport type FormMetadata = {\n  /** Form name */\n  name: string;\n  /** Form sections */\n  sections: FormSection[];\n  /** Action buttons configuration */\n  actions: ButtonMetadata;\n  /** Global form state */\n  state: ComponentState;\n};\n\n/**\n * Possible action types for a toolbar.\n */\nexport enum ToolbarActionType {\n  AVATAR = 'AVATAR',\n  ICON = 'ICON',\n  IMAGE = 'IMAGE',\n  BUTTON = 'BUTTON',\n}\n\n/**\n * Toolbar action definition.\n */\nexport type ToolbarAction = {\n  /** Action type */\n  type: 'AVATAR' | 'ICON' | 'IMAGE' | 'BUTTON';\n  /** Optional token identifier */\n  token?: string;\n  /** Toolbar position */\n  position: 'left' | 'right' | 'center';\n  /** Optional description */\n  description?: string;\n  /** Associated image (if any) */\n  image?: ImageMetadata;\n};\n\n/**\n * Metadata for an icon.\n */\nexport interface IconMetada {\n  /** Icon name */\n  name: string;\n  /** Icon slot position */\n  slot: 'start' | 'end';\n}\n\n/**\n * Button configuration object.\n * Supports both static and reactive content.\n * @type {ButtonMetadata}\n * @property text - Static button label (takes precedence over textConfig).\n * @property textConfig - Reactive content configuration for button text.\n * @property color - The button color (Ionic color string).\n * @property icon - Icon to display (optional).\n * @property state - Button state (enabled, disabled, working, etc.).\n * @property expand, fill, size, shape, href, target, download, handler, etc. - See ButtonMetadata for all options.\n */\nexport interface ButtonMetadata {\n  /** Associated action type */\n  actionType?: ActionType;\n  /** Button expansion */\n  expand?: 'full' | 'block';\n  /** Associated link */\n  link?: string;\n  /** Associated href link */\n  href?: string;\n  /** Link target */\n  target?: '_blank' | '_self' | '_parent' | '_top';\n  /** Download file name */\n  download?: string;\n  /** Button color */\n  color: Color;\n  /** Button state */\n  state: ComponentState;\n  /** Static display text (takes precedence over reactive content) */\n  text?: string;\n  /** Reactive content configuration for button text */\n  contentKey?: string;\n  /** Component class name for content lookup (required with contentKey) */\n  contentClass?: string;\n  /** Fallback text if contentKey is not found */\n  contentFallback?: string;\n  /** Values to interpolate into the content string */\n  contentInterpolation?: Record<string, string | number>;\n  /** Associated icon */\n  icon?: IconMetada;\n  /** Button shape */\n  shape?: 'round';\n  /** Button size */\n  size?: 'small' | 'default' | 'large';\n  /** Button fill */\n  fill?: 'clear' | 'outline' | 'solid' | 'default';\n  /** Button type */\n  type: 'button' | 'submit' | 'reset';\n  /** Optional token identifier */\n  token?: string;\n  /** Optional reference */\n  ref?: any;\n  /** Action handler */\n  handler?: (value: any) => any | Promise<any>;\n}\n\n/**\n * Configuration for reactive content in val-button component.\n * Use this interface when you only need to specify content-related properties.\n * This follows the same pattern as TextContentConfig for consistency.\n */\nexport interface ButtonContentConfig {\n  contentKey: string;\n  contentClass: string;\n  contentFallback?: string;\n  contentInterpolation?: Record<string, string | number>;\n}\n"]}
|
package/esm2022/lib/version.mjs
CHANGED
|
@@ -2,5 +2,5 @@
|
|
|
2
2
|
* Current version of valtech-components.
|
|
3
3
|
* This is automatically updated during the publish process.
|
|
4
4
|
*/
|
|
5
|
-
export const VERSION = '2.0.
|
|
6
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
5
|
+
export const VERSION = '2.0.694';
|
|
6
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9saWIvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFDSCxNQUFNLENBQUMsTUFBTSxPQUFPLEdBQUcsU0FBUyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBDdXJyZW50IHZlcnNpb24gb2YgdmFsdGVjaC1jb21wb25lbnRzLlxuICogVGhpcyBpcyBhdXRvbWF0aWNhbGx5IHVwZGF0ZWQgZHVyaW5nIHRoZSBwdWJsaXNoIHByb2Nlc3MuXG4gKi9cbmV4cG9ydCBjb25zdCBWRVJTSU9OID0gJzIuMC42OTQnO1xuIl19
|