@ng-modular-forms/core 0.1.2 → 0.2.0
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/LICENSE +21 -21
- package/README.md +125 -125
- package/fesm2022/ng-modular-forms-core.mjs +83 -8
- package/fesm2022/ng-modular-forms-core.mjs.map +1 -1
- package/lib/form-control-base.d.ts +19 -30
- package/lib/form-control-value-assessor.d.ts +32 -0
- package/package.json +1 -1
- package/public-api.d.ts +6 -1
- package/lib/index.d.ts +0 -5
package/LICENSE
CHANGED
|
@@ -1,21 +1,21 @@
|
|
|
1
|
-
MIT License
|
|
2
|
-
|
|
3
|
-
Copyright (c) 2026 Ron Bodnar
|
|
4
|
-
|
|
5
|
-
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
-
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
-
in the Software without restriction, including without limitation the rights
|
|
8
|
-
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
-
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
-
furnished to do so, subject to the following conditions:
|
|
11
|
-
|
|
12
|
-
The above copyright notice and this permission notice shall be included in all
|
|
13
|
-
copies or substantial portions of the Software.
|
|
14
|
-
|
|
15
|
-
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
-
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
-
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
-
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
-
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
-
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
-
SOFTWARE.
|
|
1
|
+
MIT License
|
|
2
|
+
|
|
3
|
+
Copyright (c) 2026 Ron Bodnar
|
|
4
|
+
|
|
5
|
+
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
6
|
+
of this software and associated documentation files (the "Software"), to deal
|
|
7
|
+
in the Software without restriction, including without limitation the rights
|
|
8
|
+
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
9
|
+
copies of the Software, and to permit persons to whom the Software is
|
|
10
|
+
furnished to do so, subject to the following conditions:
|
|
11
|
+
|
|
12
|
+
The above copyright notice and this permission notice shall be included in all
|
|
13
|
+
copies or substantial portions of the Software.
|
|
14
|
+
|
|
15
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
16
|
+
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
17
|
+
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
18
|
+
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
19
|
+
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
20
|
+
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
21
|
+
SOFTWARE.
|
package/README.md
CHANGED
|
@@ -1,125 +1,125 @@
|
|
|
1
|
-
# @ng-modular-forms/core
|
|
2
|
-
|
|
3
|
-
Core primitives for orchestrating complex Angular reactive forms.
|
|
4
|
-
|
|
5
|
-
## What This Provides
|
|
6
|
-
|
|
7
|
-
- Form orchestration
|
|
8
|
-
- Reactive logic isolation
|
|
9
|
-
- Data mapping layer
|
|
10
|
-
|
|
11
|
-
## Installation
|
|
12
|
-
|
|
13
|
-
```bash
|
|
14
|
-
npm install @ng-modular-forms/core
|
|
15
|
-
```
|
|
16
|
-
|
|
17
|
-
## Key Concepts
|
|
18
|
-
|
|
19
|
-
### FormOrchestratorBase
|
|
20
|
-
|
|
21
|
-
Coordinates form structure and lifecycle.
|
|
22
|
-
|
|
23
|
-
```ts
|
|
24
|
-
@Component({...})
|
|
25
|
-
export class ExampleComponent extends FormOrchestratorBase {
|
|
26
|
-
form = new FormGroup({});
|
|
27
|
-
|
|
28
|
-
ngOnInit() {
|
|
29
|
-
// Handlers are not required if there is no reactive logic or value change subscriptions.
|
|
30
|
-
const mainHandler = inject(ExampleFormHandler);
|
|
31
|
-
const sectionAHandler = inject(SectionAHandler);
|
|
32
|
-
|
|
33
|
-
const formOptions = {
|
|
34
|
-
mainHandler: mainHandler,
|
|
35
|
-
subHandlers: {
|
|
36
|
-
sectionA: sectionAHandler
|
|
37
|
-
}
|
|
38
|
-
};
|
|
39
|
-
|
|
40
|
-
this.initialize(this.form, formOptions);
|
|
41
|
-
}
|
|
42
|
-
|
|
43
|
-
// Only required if forms are split across multiple components
|
|
44
|
-
onSubformReady(form: FormGroup, key: string) {
|
|
45
|
-
super.onSubformReady(form, key);
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
```
|
|
49
|
-
|
|
50
|
-
---
|
|
51
|
-
|
|
52
|
-
### FormHandlerBase
|
|
53
|
-
|
|
54
|
-
Encapsulates reactive logic.
|
|
55
|
-
|
|
56
|
-
```ts
|
|
57
|
-
const CONTROL_NAMES = ["fieldA", "dependentField"] as const;
|
|
58
|
-
|
|
59
|
-
type ControlNames = (typeof CONTROL_NAMES)[number];
|
|
60
|
-
|
|
61
|
-
@Injectable()
|
|
62
|
-
export class SectionAHandler extends FormHandlerBase<ControlNames> {
|
|
63
|
-
override getReactiveLogic(): Subscription {
|
|
64
|
-
this.registerControls(this.form, [...CONTROL_NAMES]);
|
|
65
|
-
|
|
66
|
-
return this.valueChangesOf("fieldA").subscribe((value) => {
|
|
67
|
-
if (value) {
|
|
68
|
-
this.controls.dependentField.enable();
|
|
69
|
-
} else {
|
|
70
|
-
this.controls.dependentField.disable();
|
|
71
|
-
}
|
|
72
|
-
});
|
|
73
|
-
}
|
|
74
|
-
}
|
|
75
|
-
```
|
|
76
|
-
|
|
77
|
-
---
|
|
78
|
-
|
|
79
|
-
### FormMapperBase
|
|
80
|
-
|
|
81
|
-
Handles transformations between API and form.
|
|
82
|
-
|
|
83
|
-
```ts
|
|
84
|
-
export class ExampleMapper extends FormMapperBase<ApiModel, RequestModel> {
|
|
85
|
-
buildRequest(form: FormGroup) {
|
|
86
|
-
return {
|
|
87
|
-
fieldA: form.value.fieldA,
|
|
88
|
-
fieldB: form.value.fieldB,
|
|
89
|
-
};
|
|
90
|
-
}
|
|
91
|
-
|
|
92
|
-
transformFromModel(model: ApiModel) {
|
|
93
|
-
return {
|
|
94
|
-
fieldA: model.fieldA,
|
|
95
|
-
fieldB: model.fieldB,
|
|
96
|
-
};
|
|
97
|
-
}
|
|
98
|
-
}
|
|
99
|
-
```
|
|
100
|
-
|
|
101
|
-
---
|
|
102
|
-
|
|
103
|
-
### Responsibility Boundaries
|
|
104
|
-
|
|
105
|
-
| Layer | Responsibility |
|
|
106
|
-
| :--------------- | :------------------------------------------------------- |
|
|
107
|
-
| **Orchestrator** | Manages form composition and lifecycle coordination. |
|
|
108
|
-
| **Handler** | Encapsulates all reactive logic and stream management. |
|
|
109
|
-
| **Mapper** | Handles data transformation between API and Form states. |
|
|
110
|
-
|
|
111
|
-
---
|
|
112
|
-
|
|
113
|
-
### No UI Included
|
|
114
|
-
|
|
115
|
-
This package does **not** provide UI components.
|
|
116
|
-
|
|
117
|
-
Use:
|
|
118
|
-
|
|
119
|
-
- @ng-modular-forms/control
|
|
120
|
-
- @ng-modular-forms/input
|
|
121
|
-
- @ng-modular-forms/material
|
|
122
|
-
|
|
123
|
-
## License
|
|
124
|
-
|
|
125
|
-
MIT
|
|
1
|
+
# @ng-modular-forms/core
|
|
2
|
+
|
|
3
|
+
Core primitives for orchestrating complex Angular reactive forms.
|
|
4
|
+
|
|
5
|
+
## What This Provides
|
|
6
|
+
|
|
7
|
+
- Form orchestration
|
|
8
|
+
- Reactive logic isolation
|
|
9
|
+
- Data mapping layer
|
|
10
|
+
|
|
11
|
+
## Installation
|
|
12
|
+
|
|
13
|
+
```bash
|
|
14
|
+
npm install @ng-modular-forms/core
|
|
15
|
+
```
|
|
16
|
+
|
|
17
|
+
## Key Concepts
|
|
18
|
+
|
|
19
|
+
### FormOrchestratorBase
|
|
20
|
+
|
|
21
|
+
Coordinates form structure and lifecycle.
|
|
22
|
+
|
|
23
|
+
```ts
|
|
24
|
+
@Component({...})
|
|
25
|
+
export class ExampleComponent extends FormOrchestratorBase {
|
|
26
|
+
form = new FormGroup({});
|
|
27
|
+
|
|
28
|
+
ngOnInit() {
|
|
29
|
+
// Handlers are not required if there is no reactive logic or value change subscriptions.
|
|
30
|
+
const mainHandler = inject(ExampleFormHandler);
|
|
31
|
+
const sectionAHandler = inject(SectionAHandler);
|
|
32
|
+
|
|
33
|
+
const formOptions = {
|
|
34
|
+
mainHandler: mainHandler,
|
|
35
|
+
subHandlers: {
|
|
36
|
+
sectionA: sectionAHandler
|
|
37
|
+
}
|
|
38
|
+
};
|
|
39
|
+
|
|
40
|
+
this.initialize(this.form, formOptions);
|
|
41
|
+
}
|
|
42
|
+
|
|
43
|
+
// Only required if forms are split across multiple components
|
|
44
|
+
onSubformReady(form: FormGroup, key: string) {
|
|
45
|
+
super.onSubformReady(form, key);
|
|
46
|
+
}
|
|
47
|
+
}
|
|
48
|
+
```
|
|
49
|
+
|
|
50
|
+
---
|
|
51
|
+
|
|
52
|
+
### FormHandlerBase
|
|
53
|
+
|
|
54
|
+
Encapsulates reactive logic.
|
|
55
|
+
|
|
56
|
+
```ts
|
|
57
|
+
const CONTROL_NAMES = ["fieldA", "dependentField"] as const;
|
|
58
|
+
|
|
59
|
+
type ControlNames = (typeof CONTROL_NAMES)[number];
|
|
60
|
+
|
|
61
|
+
@Injectable()
|
|
62
|
+
export class SectionAHandler extends FormHandlerBase<ControlNames> {
|
|
63
|
+
override getReactiveLogic(): Subscription {
|
|
64
|
+
this.registerControls(this.form, [...CONTROL_NAMES]);
|
|
65
|
+
|
|
66
|
+
return this.valueChangesOf("fieldA").subscribe((value) => {
|
|
67
|
+
if (value) {
|
|
68
|
+
this.controls.dependentField.enable();
|
|
69
|
+
} else {
|
|
70
|
+
this.controls.dependentField.disable();
|
|
71
|
+
}
|
|
72
|
+
});
|
|
73
|
+
}
|
|
74
|
+
}
|
|
75
|
+
```
|
|
76
|
+
|
|
77
|
+
---
|
|
78
|
+
|
|
79
|
+
### FormMapperBase
|
|
80
|
+
|
|
81
|
+
Handles transformations between API and form.
|
|
82
|
+
|
|
83
|
+
```ts
|
|
84
|
+
export class ExampleMapper extends FormMapperBase<ApiModel, RequestModel> {
|
|
85
|
+
buildRequest(form: FormGroup) {
|
|
86
|
+
return {
|
|
87
|
+
fieldA: form.value.fieldA,
|
|
88
|
+
fieldB: form.value.fieldB,
|
|
89
|
+
};
|
|
90
|
+
}
|
|
91
|
+
|
|
92
|
+
transformFromModel(model: ApiModel) {
|
|
93
|
+
return {
|
|
94
|
+
fieldA: model.fieldA,
|
|
95
|
+
fieldB: model.fieldB,
|
|
96
|
+
};
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
```
|
|
100
|
+
|
|
101
|
+
---
|
|
102
|
+
|
|
103
|
+
### Responsibility Boundaries
|
|
104
|
+
|
|
105
|
+
| Layer | Responsibility |
|
|
106
|
+
| :--------------- | :------------------------------------------------------- |
|
|
107
|
+
| **Orchestrator** | Manages form composition and lifecycle coordination. |
|
|
108
|
+
| **Handler** | Encapsulates all reactive logic and stream management. |
|
|
109
|
+
| **Mapper** | Handles data transformation between API and Form states. |
|
|
110
|
+
|
|
111
|
+
---
|
|
112
|
+
|
|
113
|
+
### No UI Included
|
|
114
|
+
|
|
115
|
+
This package does **not** provide UI components.
|
|
116
|
+
|
|
117
|
+
Use:
|
|
118
|
+
|
|
119
|
+
- @ng-modular-forms/control
|
|
120
|
+
- @ng-modular-forms/input
|
|
121
|
+
- @ng-modular-forms/material
|
|
122
|
+
|
|
123
|
+
## License
|
|
124
|
+
|
|
125
|
+
MIT
|
|
@@ -1,7 +1,8 @@
|
|
|
1
1
|
import * as i0 from '@angular/core';
|
|
2
|
-
import { inject, ChangeDetectorRef, HostBinding, Input, Optional, Self, Directive, signal } from '@angular/core';
|
|
2
|
+
import { inject, ChangeDetectorRef, HostBinding, Input, Optional, Self, Directive, signal, computed } from '@angular/core';
|
|
3
3
|
import * as i1 from '@angular/forms';
|
|
4
|
-
import { FormGroup } from '@angular/forms';
|
|
4
|
+
import { FormGroup, FormGroupDirective, Validators } from '@angular/forms';
|
|
5
|
+
import { takeUntilDestroyed } from '@angular/core/rxjs-interop';
|
|
5
6
|
import { Subscription } from 'rxjs';
|
|
6
7
|
|
|
7
8
|
/**
|
|
@@ -9,7 +10,7 @@ import { Subscription } from 'rxjs';
|
|
|
9
10
|
*
|
|
10
11
|
* NOTE: This class is UI-layer only and should not be used in form orchestration logic.
|
|
11
12
|
*/
|
|
12
|
-
class
|
|
13
|
+
class FormControlValueAccessor {
|
|
13
14
|
ngControl;
|
|
14
15
|
elementRef;
|
|
15
16
|
cdr = inject(ChangeDetectorRef);
|
|
@@ -20,7 +21,7 @@ class FormControlBase {
|
|
|
20
21
|
readonly = false;
|
|
21
22
|
formControlName;
|
|
22
23
|
static nextId = 0;
|
|
23
|
-
id = `nmf-form-control-${
|
|
24
|
+
id = `nmf-form-control-${FormControlValueAccessor.nextId++}`;
|
|
24
25
|
_value = null;
|
|
25
26
|
get value() {
|
|
26
27
|
return this._value;
|
|
@@ -47,10 +48,10 @@ class FormControlBase {
|
|
|
47
48
|
registerOnTouched(fn) {
|
|
48
49
|
this.onTouched = fn;
|
|
49
50
|
}
|
|
50
|
-
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type:
|
|
51
|
-
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.2.21", type:
|
|
51
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: FormControlValueAccessor, deps: [{ token: i1.NgControl, optional: true, self: true }, { token: i0.ElementRef }], target: i0.ɵɵFactoryTarget.Directive });
|
|
52
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.2.21", type: FormControlValueAccessor, isStandalone: true, inputs: { name: "name", placeholder: "placeholder", required: "required", disabled: "disabled", readonly: "readonly", formControlName: "formControlName" }, host: { properties: { "id": "this.id" } }, ngImport: i0 });
|
|
52
53
|
}
|
|
53
|
-
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type:
|
|
54
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: FormControlValueAccessor, decorators: [{
|
|
54
55
|
type: Directive
|
|
55
56
|
}], ctorParameters: () => [{ type: i1.NgControl, decorators: [{
|
|
56
57
|
type: Optional
|
|
@@ -72,6 +73,80 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImpo
|
|
|
72
73
|
type: HostBinding
|
|
73
74
|
}] } });
|
|
74
75
|
|
|
76
|
+
class FormControlBase extends FormControlValueAccessor {
|
|
77
|
+
label;
|
|
78
|
+
classList = [];
|
|
79
|
+
loading = false;
|
|
80
|
+
_form = signal(new FormGroup({}));
|
|
81
|
+
_controlName = signal('');
|
|
82
|
+
form = this._form.asReadonly();
|
|
83
|
+
controlName = this._controlName.asReadonly();
|
|
84
|
+
control = computed(() => this._form().get(this._controlName()));
|
|
85
|
+
parentFormGroup = inject(FormGroupDirective, {
|
|
86
|
+
optional: true,
|
|
87
|
+
host: true,
|
|
88
|
+
});
|
|
89
|
+
ngOnInit() {
|
|
90
|
+
const controlName = this.formControlName ?? 'default';
|
|
91
|
+
const form = this.parentFormGroup?.form;
|
|
92
|
+
if (!form) {
|
|
93
|
+
throw new Error(`FormGroupDirective not found. Ensure component is used inside a form group`);
|
|
94
|
+
}
|
|
95
|
+
this._form.set(form);
|
|
96
|
+
this._controlName.set(controlName);
|
|
97
|
+
const control = form.get(controlName);
|
|
98
|
+
if (!control) {
|
|
99
|
+
throw new Error(`FormControl '${controlName}' not found in parent FormGroup`);
|
|
100
|
+
}
|
|
101
|
+
control.statusChanges
|
|
102
|
+
.pipe(takeUntilDestroyed())
|
|
103
|
+
.subscribe(() => this.cdr.markForCheck());
|
|
104
|
+
}
|
|
105
|
+
isRequired() {
|
|
106
|
+
const control = this._form()?.get(this._controlName());
|
|
107
|
+
return !!control && control.hasValidator(Validators.required);
|
|
108
|
+
}
|
|
109
|
+
serverError = computed(() => this.form()?.errors?.['server']);
|
|
110
|
+
getErrorMessage() {
|
|
111
|
+
const control = this._form()?.get(this._controlName());
|
|
112
|
+
if (!control || !control.errors || !control.touched)
|
|
113
|
+
return '';
|
|
114
|
+
const firstKey = Object.keys(control.errors)[0];
|
|
115
|
+
const error = control.errors[firstKey];
|
|
116
|
+
switch (firstKey) {
|
|
117
|
+
case 'required':
|
|
118
|
+
return 'This field is required';
|
|
119
|
+
case 'minlength':
|
|
120
|
+
return `Minimum length is ${error.requiredLength}`;
|
|
121
|
+
case 'maxlength':
|
|
122
|
+
return `Maximum length is ${error.requiredLength}`;
|
|
123
|
+
case 'min':
|
|
124
|
+
return `Minimum value is ${error.min}`;
|
|
125
|
+
case 'max':
|
|
126
|
+
return `Maximum value is ${error.max}`;
|
|
127
|
+
case 'email':
|
|
128
|
+
return 'Invalid email address';
|
|
129
|
+
case 'pattern':
|
|
130
|
+
return 'Invalid format';
|
|
131
|
+
case 'server':
|
|
132
|
+
return error;
|
|
133
|
+
default:
|
|
134
|
+
return 'Invalid value';
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
static ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: FormControlBase, deps: null, target: i0.ɵɵFactoryTarget.Directive });
|
|
138
|
+
static ɵdir = i0.ɵɵngDeclareDirective({ minVersion: "14.0.0", version: "19.2.21", type: FormControlBase, isStandalone: true, inputs: { label: "label", classList: "classList", loading: "loading" }, usesInheritance: true, ngImport: i0 });
|
|
139
|
+
}
|
|
140
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImport: i0, type: FormControlBase, decorators: [{
|
|
141
|
+
type: Directive
|
|
142
|
+
}], propDecorators: { label: [{
|
|
143
|
+
type: Input
|
|
144
|
+
}], classList: [{
|
|
145
|
+
type: Input
|
|
146
|
+
}], loading: [{
|
|
147
|
+
type: Input
|
|
148
|
+
}] } });
|
|
149
|
+
|
|
75
150
|
function getControl(controlName, form) {
|
|
76
151
|
if (!form) {
|
|
77
152
|
throw new Error(`Missing form instance while getting the control of "${controlName}"`);
|
|
@@ -250,5 +325,5 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "19.2.21", ngImpo
|
|
|
250
325
|
* Generated bundle index. Do not edit.
|
|
251
326
|
*/
|
|
252
327
|
|
|
253
|
-
export { FormControlBase, FormHandlerBase, FormMapperBase, FormOrchestratorBase, getControl, getControlValue, parseCurrency };
|
|
328
|
+
export { FormControlBase, FormControlValueAccessor, FormHandlerBase, FormMapperBase, FormOrchestratorBase, getControl, getControlValue, parseCurrency };
|
|
254
329
|
//# sourceMappingURL=ng-modular-forms-core.mjs.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"ng-modular-forms-core.mjs","sources":["../../../projects/core/src/lib/form-control-base.ts","../../../projects/core/src/lib/form.util.ts","../../../projects/core/src/lib/form-handler-base.ts","../../../projects/core/src/lib/form-mapper-base.ts","../../../projects/core/src/lib/form-orchestrator-base.ts","../../../projects/core/src/public-api.ts","../../../projects/core/src/ng-modular-forms-core.ts"],"sourcesContent":["import {\n Directive,\n ElementRef,\n Input,\n HostBinding,\n Optional,\n Self,\n inject,\n ChangeDetectorRef,\n} from '@angular/core';\nimport { ControlValueAccessor, NgControl } from '@angular/forms';\n\n/**\n * Base implementation for custom form controls that integrate with Angular Reactive Forms (ControlValueAccessor)\n *\n * NOTE: This class is UI-layer only and should not be used in form orchestration logic.\n */\n\n@Directive()\nexport abstract class FormControlBase<T> implements ControlValueAccessor {\n protected readonly cdr = inject(ChangeDetectorRef);\n\n @Input() name!: string;\n @Input() placeholder: string = '';\n @Input() required = false;\n @Input() disabled = false;\n @Input() readonly = false;\n\n @Input() formControlName?: string;\n\n static nextId = 0;\n\n @HostBinding()\n id = `nmf-form-control-${FormControlBase.nextId++}`;\n\n _value: T | null = null;\n\n get value(): T | null {\n return this._value;\n }\n\n set value(val: T | null) {\n this._value = val;\n }\n\n onChange = (_value: T) => {};\n onTouched = () => {};\n\n constructor(\n @Optional() @Self() public ngControl: NgControl,\n protected elementRef: ElementRef<HTMLElement>,\n ) {\n if (this.ngControl != null) {\n this.ngControl.valueAccessor = this;\n }\n }\n\n writeValue(value: T): void {\n this._value = value;\n this.cdr.markForCheck();\n }\n\n registerOnChange(fn: (value: T) => void): void {\n this.onChange = fn;\n }\n\n registerOnTouched(fn: () => void): void {\n this.onTouched = fn;\n }\n}\n","import { FormControl, FormGroup } from '@angular/forms';\n\nexport function getControl<T = unknown>(controlName: string, form: FormGroup) {\n if (!form) {\n throw new Error(\n `Missing form instance while getting the control of \"${controlName}\"`,\n );\n }\n\n const control = form.get(controlName) as FormControl<T>;\n if (!control) {\n throw new Error(`Missing control \"${controlName}\" in form \"${form}\"`);\n }\n\n return control;\n}\n\nexport function getControlValue<T = unknown>(\n controlName: string,\n form: FormGroup,\n): T | null;\n\nexport function getControlValue(\n controlName: string,\n form: FormGroup,\n): number | null;\n\nexport function getControlValue<T = unknown>(\n controlName: string,\n form: FormGroup,\n): T | null {\n const control = getControl<T>(controlName, form);\n if (!control) {\n throw new Error(`Missing control \"${controlName}\" in form \"${form}\"`);\n }\n\n const value = control.getRawValue();\n if (value === '') {\n return null;\n }\n\n if (typeof value === 'string') {\n const cleaned = value.replace(/,/g, '');\n\n if (!Number.isNaN(Number(cleaned))) {\n return parseCurrency(value) as T;\n }\n }\n\n return value;\n}\n\nexport function parseCurrency(\n currencyString: string | number | null | undefined,\n): number {\n if (currencyString == null) {\n return 0;\n }\n\n if (typeof currencyString === 'number') {\n return currencyString;\n }\n\n return Number(currencyString.replace(/[^0-9.-]/g, ''));\n}\n","import { FormControl, FormGroup } from '@angular/forms';\nimport { Observable, Subscription } from 'rxjs';\nimport { getControl } from './form.util';\n\nexport abstract class FormHandlerBase<ControlNames extends string = string> {\n abstract getReactiveLogic(form?: FormGroup): Subscription;\n\n protected registeredControls: Record<string, FormControl> = {};\n\n /**\n * Registers form controls for later reactive access.\n */\n registerControls(form: FormGroup, controlNames: ControlNames[]): void {\n controlNames.forEach((cn) => {\n const control = getControl(cn.replace(/_/g, '.'), form);\n\n if (!control) {\n throw new Error(\n `Failed to register control with name: \"${cn}\" in form group: \"${form}\"`,\n );\n }\n\n const key = cn as ControlNames;\n this.registeredControls[key] = control;\n });\n }\n\n valueChangesOf<T>(key: ControlNames): Observable<T> {\n if (!this.registeredControls[key]) {\n throw new Error(\n `Control with name: \"${key}\" not found. Ensure it is registered in registerControls(...)`,\n );\n }\n\n return this.registeredControls[key].valueChanges as Observable<T>;\n }\n}\n","import { FormGroup } from '@angular/forms';\n\nexport abstract class FormMapperBase<\n TModelIn = unknown,\n TModelOut = unknown,\n TFormModel = unknown,\n TStore = Record<string, unknown>,\n> {\n /**\n * Maps form state + external store into an API request payload.\n */\n abstract buildRequest(form: FormGroup, store: TStore): Partial<TModelOut>;\n\n /**\n * Transforms a domain/API model into a form-compatible structure.\n */\n abstract transformFromModel(model: TModelIn): TFormModel;\n\n patchForm(\n form: FormGroup,\n data: TFormModel,\n _context: TStore,\n emitEvent: boolean = false,\n ): void {\n form.patchValue(\n {\n ...(data as Record<string, unknown>),\n },\n { emitEvent },\n );\n }\n\n patchFromModel(form: FormGroup, model: TModelIn, store: TStore): void {\n const formModel = this.transformFromModel(model);\n this.patchForm(form, formModel, store);\n }\n}\n","import { FormGroup } from '@angular/forms';\nimport { Directive, OnDestroy, signal } from '@angular/core';\nimport { Subscription } from 'rxjs';\nimport { FormHandlerBase } from './form-handler-base';\n\nexport type FormStatus = 'idle' | 'submitting' | 'error' | 'success';\n\ntype FormHandlerRegistry = Record<string, FormHandlerBase<string>>;\n\ninterface FormOrchestratorOptions {\n mainHandler?: FormHandlerBase<string> | null;\n subHandlers?: FormHandlerRegistry;\n}\n\n@Directive()\nexport abstract class FormOrchestratorBase implements OnDestroy {\n private _form = signal<FormGroup>(new FormGroup({}));\n\n private _mainHandler = signal<FormHandlerBase<string> | null>(null);\n private _subHandlers = signal<FormHandlerRegistry>({});\n\n private _status = signal<FormStatus>('idle');\n private _errorMessage = signal<string | null>(null);\n\n private _loadedHandlers = signal<number>(0);\n private _allHandlersLoaded = signal<boolean>(false);\n\n private _logicSubscription = new Subscription();\n\n public readonly form = this._form.asReadonly();\n public readonly mainHandler = this._mainHandler.asReadonly();\n public readonly subHandlers = this._subHandlers.asReadonly();\n public readonly status = this._status.asReadonly();\n public readonly errorMessage = this._errorMessage.asReadonly();\n\n /**\n * Initializes orchestration state.\n * Must be called before any subform registration or handler execution.\n */\n public initialize(\n form: FormGroup,\n formOrchestratorOptions?: FormOrchestratorOptions,\n ) {\n const { mainHandler, subHandlers = {} } = formOrchestratorOptions ?? {};\n\n this._form.set(form);\n this._mainHandler.set(mainHandler ?? null);\n this._subHandlers.set(subHandlers);\n this._loadedHandlers.set(0);\n\n const subHandlerCount = Object.keys(this.subHandlers());\n if (subHandlerCount.length === 0) {\n this.loadMainHandler();\n }\n }\n\n public setForm(form: FormGroup) {\n this._form.set(form);\n }\n\n public getHandler(key: string): FormHandlerBase<string> | undefined {\n return this.subHandlers()[key];\n }\n\n public setHandler(key: string, handler: FormHandlerBase<string>) {\n this._subHandlers.set({\n ...this._subHandlers(),\n [key]: handler,\n });\n }\n\n public addReactiveLogic(subscription: Subscription) {\n this._logicSubscription.add(subscription);\n }\n\n public setStatus(status: FormStatus) {\n this._status.set(status);\n }\n\n public setErrorMessage(message: string | null) {\n this._errorMessage.set(message);\n }\n\n ngOnDestroy(): void {\n this._logicSubscription.unsubscribe();\n }\n\n /**\n * Registers a subform into the main form tree and coordinates handler execution.\n *\n * IMPORTANT:\n * - Handler execution is gated until all registered subhandlers are ready.\n * - Calling order matters; this is lifecycle-sensitive orchestration logic.\n */\n onSubformReady(\n subform: FormGroup,\n groupName: string,\n nestGroups: boolean = false,\n ): void {\n if (nestGroups) {\n this.form().setControl(groupName, subform);\n } else {\n const keys = Object.keys(subform.controls);\n keys.forEach((key) => {\n this.form().setControl(key, subform.get(key));\n });\n }\n\n // Prevent duplicate main handler execution when all subhandlers already resolved\n if (this._allHandlersLoaded()) {\n return;\n }\n\n const subformHandler = this.subHandlers()[groupName];\n\n if (subformHandler) {\n this._loadedHandlers.set(this._loadedHandlers() + 1);\n this.addReactiveLogic(subformHandler.getReactiveLogic());\n }\n\n const totalSubHandlers = Object.keys(this.subHandlers()).length;\n const allSubHandlersLoaded = this._loadedHandlers() === totalSubHandlers;\n\n if (allSubHandlersLoaded && this.mainHandler()) {\n this.loadMainHandler();\n }\n }\n\n private loadMainHandler() {\n if (!this.mainHandler()) return;\n\n this.addReactiveLogic(this.mainHandler()!.getReactiveLogic(this.form()));\n this._allHandlersLoaded.set(true);\n }\n}\n","/*\n * Public API Surface of core\n */\n\nexport * from './lib/index';\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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"}
|
|
1
|
+
{"version":3,"file":"ng-modular-forms-core.mjs","sources":["../../../projects/core/src/lib/form-control-value-assessor.ts","../../../projects/core/src/lib/form-control-base.ts","../../../projects/core/src/lib/form.util.ts","../../../projects/core/src/lib/form-handler-base.ts","../../../projects/core/src/lib/form-mapper-base.ts","../../../projects/core/src/lib/form-orchestrator-base.ts","../../../projects/core/src/public-api.ts","../../../projects/core/src/ng-modular-forms-core.ts"],"sourcesContent":["import {\r\n Directive,\r\n ElementRef,\r\n Input,\r\n HostBinding,\r\n Optional,\r\n Self,\r\n inject,\r\n ChangeDetectorRef,\r\n} from '@angular/core';\r\nimport { ControlValueAccessor, NgControl } from '@angular/forms';\r\n\r\n/**\r\n * Base implementation for custom form controls that integrate with Angular Reactive Forms (ControlValueAccessor)\r\n *\r\n * NOTE: This class is UI-layer only and should not be used in form orchestration logic.\r\n */\r\n\r\n@Directive()\r\nexport abstract class FormControlValueAccessor<\r\n T,\r\n> implements ControlValueAccessor {\r\n protected readonly cdr = inject(ChangeDetectorRef);\r\n\r\n @Input() name!: string;\r\n @Input() placeholder: string = '';\r\n @Input() required = false;\r\n @Input() disabled = false;\r\n @Input() readonly = false;\r\n\r\n @Input() formControlName?: string;\r\n\r\n static nextId = 0;\r\n\r\n @HostBinding()\r\n id = `nmf-form-control-${FormControlValueAccessor.nextId++}`;\r\n\r\n _value: T | null = null;\r\n\r\n get value(): T | null {\r\n return this._value;\r\n }\r\n\r\n set value(val: T | null) {\r\n this._value = val;\r\n }\r\n\r\n onChange = (_value: T) => {};\r\n onTouched = () => {};\r\n\r\n constructor(\r\n @Optional() @Self() public ngControl: NgControl,\r\n protected elementRef: ElementRef<HTMLElement>,\r\n ) {\r\n if (this.ngControl != null) {\r\n this.ngControl.valueAccessor = this;\r\n }\r\n }\r\n\r\n writeValue(value: T): void {\r\n this._value = value;\r\n this.cdr.markForCheck();\r\n }\r\n\r\n registerOnChange(fn: (value: T) => void): void {\r\n this.onChange = fn;\r\n }\r\n\r\n registerOnTouched(fn: () => void): void {\r\n this.onTouched = fn;\r\n }\r\n}\r\n","import {\r\n FormControl,\r\n FormGroup,\r\n FormGroupDirective,\r\n Validators,\r\n} from '@angular/forms';\r\nimport {\r\n computed,\r\n Directive,\r\n inject,\r\n Input,\r\n OnInit,\r\n signal,\r\n} from '@angular/core';\r\nimport { takeUntilDestroyed } from '@angular/core/rxjs-interop';\r\nimport { FormControlValueAccessor } from './form-control-value-assessor';\r\n\r\n@Directive()\r\nexport abstract class FormControlBase<T>\r\n extends FormControlValueAccessor<T>\r\n implements OnInit\r\n{\r\n @Input() label!: string;\r\n\r\n @Input() classList: string[] = [];\r\n\r\n @Input() loading: boolean = false;\r\n\r\n protected _form = signal<FormGroup>(new FormGroup({}));\r\n protected _controlName = signal<string>('');\r\n\r\n readonly form = this._form.asReadonly();\r\n readonly controlName = this._controlName.asReadonly();\r\n\r\n readonly control = computed(\r\n () => this._form().get(this._controlName()) as FormControl,\r\n );\r\n\r\n private parentFormGroup = inject(FormGroupDirective, {\r\n optional: true,\r\n host: true,\r\n });\r\n\r\n ngOnInit() {\r\n const controlName = this.formControlName ?? 'default';\r\n const form = this.parentFormGroup?.form;\r\n\r\n if (!form) {\r\n throw new Error(\r\n `FormGroupDirective not found. Ensure component is used inside a form group`,\r\n );\r\n }\r\n\r\n this._form.set(form);\r\n this._controlName.set(controlName);\r\n\r\n const control = form.get(controlName);\r\n\r\n if (!control) {\r\n throw new Error(\r\n `FormControl '${controlName}' not found in parent FormGroup`,\r\n );\r\n }\r\n\r\n control.statusChanges\r\n .pipe(takeUntilDestroyed())\r\n .subscribe(() => this.cdr.markForCheck());\r\n }\r\n\r\n protected isRequired(): boolean {\r\n const control = this._form()?.get(this._controlName());\r\n return !!control && control.hasValidator(Validators.required);\r\n }\r\n\r\n protected serverError = computed(\r\n () => this.form()?.errors?.['server'] as string | undefined,\r\n );\r\n\r\n protected getErrorMessage(): string {\r\n const control = this._form()?.get(this._controlName());\r\n\r\n if (!control || !control.errors || !control.touched) return '';\r\n\r\n const firstKey = Object.keys(control.errors)[0];\r\n const error = control.errors[firstKey];\r\n\r\n switch (firstKey) {\r\n case 'required':\r\n return 'This field is required';\r\n case 'minlength':\r\n return `Minimum length is ${error.requiredLength}`;\r\n case 'maxlength':\r\n return `Maximum length is ${error.requiredLength}`;\r\n case 'min':\r\n return `Minimum value is ${error.min}`;\r\n case 'max':\r\n return `Maximum value is ${error.max}`;\r\n case 'email':\r\n return 'Invalid email address';\r\n case 'pattern':\r\n return 'Invalid format';\r\n case 'server':\r\n return error;\r\n default:\r\n return 'Invalid value';\r\n }\r\n }\r\n}\r\n","import { FormControl, FormGroup } from '@angular/forms';\r\n\r\nexport function getControl<T = unknown>(controlName: string, form: FormGroup) {\r\n if (!form) {\r\n throw new Error(\r\n `Missing form instance while getting the control of \"${controlName}\"`,\r\n );\r\n }\r\n\r\n const control = form.get(controlName) as FormControl<T>;\r\n if (!control) {\r\n throw new Error(`Missing control \"${controlName}\" in form \"${form}\"`);\r\n }\r\n\r\n return control;\r\n}\r\n\r\nexport function getControlValue<T = unknown>(\r\n controlName: string,\r\n form: FormGroup,\r\n): T | null;\r\n\r\nexport function getControlValue(\r\n controlName: string,\r\n form: FormGroup,\r\n): number | null;\r\n\r\nexport function getControlValue<T = unknown>(\r\n controlName: string,\r\n form: FormGroup,\r\n): T | null {\r\n const control = getControl<T>(controlName, form);\r\n if (!control) {\r\n throw new Error(`Missing control \"${controlName}\" in form \"${form}\"`);\r\n }\r\n\r\n const value = control.getRawValue();\r\n if (value === '') {\r\n return null;\r\n }\r\n\r\n if (typeof value === 'string') {\r\n const cleaned = value.replace(/,/g, '');\r\n\r\n if (!Number.isNaN(Number(cleaned))) {\r\n return parseCurrency(value) as T;\r\n }\r\n }\r\n\r\n return value;\r\n}\r\n\r\nexport function parseCurrency(\r\n currencyString: string | number | null | undefined,\r\n): number {\r\n if (currencyString == null) {\r\n return 0;\r\n }\r\n\r\n if (typeof currencyString === 'number') {\r\n return currencyString;\r\n }\r\n\r\n return Number(currencyString.replace(/[^0-9.-]/g, ''));\r\n}\r\n","import { FormControl, FormGroup } from '@angular/forms';\r\nimport { Observable, Subscription } from 'rxjs';\r\nimport { getControl } from './form.util';\r\n\r\nexport abstract class FormHandlerBase<ControlNames extends string = string> {\r\n abstract getReactiveLogic(form?: FormGroup): Subscription;\r\n\r\n protected registeredControls: Record<string, FormControl> = {};\r\n\r\n /**\r\n * Registers form controls for later reactive access.\r\n */\r\n registerControls(form: FormGroup, controlNames: ControlNames[]): void {\r\n controlNames.forEach((cn) => {\r\n const control = getControl(cn.replace(/_/g, '.'), form);\r\n\r\n if (!control) {\r\n throw new Error(\r\n `Failed to register control with name: \"${cn}\" in form group: \"${form}\"`,\r\n );\r\n }\r\n\r\n const key = cn as ControlNames;\r\n this.registeredControls[key] = control;\r\n });\r\n }\r\n\r\n valueChangesOf<T>(key: ControlNames): Observable<T> {\r\n if (!this.registeredControls[key]) {\r\n throw new Error(\r\n `Control with name: \"${key}\" not found. Ensure it is registered in registerControls(...)`,\r\n );\r\n }\r\n\r\n return this.registeredControls[key].valueChanges as Observable<T>;\r\n }\r\n}\r\n","import { FormGroup } from '@angular/forms';\r\n\r\nexport abstract class FormMapperBase<\r\n TModelIn = unknown,\r\n TModelOut = unknown,\r\n TFormModel = unknown,\r\n TStore = Record<string, unknown>,\r\n> {\r\n /**\r\n * Maps form state + external store into an API request payload.\r\n */\r\n abstract buildRequest(form: FormGroup, store: TStore): Partial<TModelOut>;\r\n\r\n /**\r\n * Transforms a domain/API model into a form-compatible structure.\r\n */\r\n abstract transformFromModel(model: TModelIn): TFormModel;\r\n\r\n patchForm(\r\n form: FormGroup,\r\n data: TFormModel,\r\n _context: TStore,\r\n emitEvent: boolean = false,\r\n ): void {\r\n form.patchValue(\r\n {\r\n ...(data as Record<string, unknown>),\r\n },\r\n { emitEvent },\r\n );\r\n }\r\n\r\n patchFromModel(form: FormGroup, model: TModelIn, store: TStore): void {\r\n const formModel = this.transformFromModel(model);\r\n this.patchForm(form, formModel, store);\r\n }\r\n}\r\n","import { FormGroup } from '@angular/forms';\r\nimport { Directive, OnDestroy, signal } from '@angular/core';\r\nimport { Subscription } from 'rxjs';\r\nimport { FormHandlerBase } from './form-handler-base';\r\n\r\nexport type FormStatus = 'idle' | 'submitting' | 'error' | 'success';\r\n\r\ntype FormHandlerRegistry = Record<string, FormHandlerBase<string>>;\r\n\r\ninterface FormOrchestratorOptions {\r\n mainHandler?: FormHandlerBase<string> | null;\r\n subHandlers?: FormHandlerRegistry;\r\n}\r\n\r\n@Directive()\r\nexport abstract class FormOrchestratorBase implements OnDestroy {\r\n private _form = signal<FormGroup>(new FormGroup({}));\r\n\r\n private _mainHandler = signal<FormHandlerBase<string> | null>(null);\r\n private _subHandlers = signal<FormHandlerRegistry>({});\r\n\r\n private _status = signal<FormStatus>('idle');\r\n private _errorMessage = signal<string | null>(null);\r\n\r\n private _loadedHandlers = signal<number>(0);\r\n private _allHandlersLoaded = signal<boolean>(false);\r\n\r\n private _logicSubscription = new Subscription();\r\n\r\n public readonly form = this._form.asReadonly();\r\n public readonly mainHandler = this._mainHandler.asReadonly();\r\n public readonly subHandlers = this._subHandlers.asReadonly();\r\n public readonly status = this._status.asReadonly();\r\n public readonly errorMessage = this._errorMessage.asReadonly();\r\n\r\n /**\r\n * Initializes orchestration state.\r\n * Must be called before any subform registration or handler execution.\r\n */\r\n public initialize(\r\n form: FormGroup,\r\n formOrchestratorOptions?: FormOrchestratorOptions,\r\n ) {\r\n const { mainHandler, subHandlers = {} } = formOrchestratorOptions ?? {};\r\n\r\n this._form.set(form);\r\n this._mainHandler.set(mainHandler ?? null);\r\n this._subHandlers.set(subHandlers);\r\n this._loadedHandlers.set(0);\r\n\r\n const subHandlerCount = Object.keys(this.subHandlers());\r\n if (subHandlerCount.length === 0) {\r\n this.loadMainHandler();\r\n }\r\n }\r\n\r\n public setForm(form: FormGroup) {\r\n this._form.set(form);\r\n }\r\n\r\n public getHandler(key: string): FormHandlerBase<string> | undefined {\r\n return this.subHandlers()[key];\r\n }\r\n\r\n public setHandler(key: string, handler: FormHandlerBase<string>) {\r\n this._subHandlers.set({\r\n ...this._subHandlers(),\r\n [key]: handler,\r\n });\r\n }\r\n\r\n public addReactiveLogic(subscription: Subscription) {\r\n this._logicSubscription.add(subscription);\r\n }\r\n\r\n public setStatus(status: FormStatus) {\r\n this._status.set(status);\r\n }\r\n\r\n public setErrorMessage(message: string | null) {\r\n this._errorMessage.set(message);\r\n }\r\n\r\n ngOnDestroy(): void {\r\n this._logicSubscription.unsubscribe();\r\n }\r\n\r\n /**\r\n * Registers a subform into the main form tree and coordinates handler execution.\r\n *\r\n * IMPORTANT:\r\n * - Handler execution is gated until all registered subhandlers are ready.\r\n * - Calling order matters; this is lifecycle-sensitive orchestration logic.\r\n */\r\n onSubformReady(\r\n subform: FormGroup,\r\n groupName: string,\r\n nestGroups: boolean = false,\r\n ): void {\r\n if (nestGroups) {\r\n this.form().setControl(groupName, subform);\r\n } else {\r\n const keys = Object.keys(subform.controls);\r\n keys.forEach((key) => {\r\n this.form().setControl(key, subform.get(key));\r\n });\r\n }\r\n\r\n // Prevent duplicate main handler execution when all subhandlers already resolved\r\n if (this._allHandlersLoaded()) {\r\n return;\r\n }\r\n\r\n const subformHandler = this.subHandlers()[groupName];\r\n\r\n if (subformHandler) {\r\n this._loadedHandlers.set(this._loadedHandlers() + 1);\r\n this.addReactiveLogic(subformHandler.getReactiveLogic());\r\n }\r\n\r\n const totalSubHandlers = Object.keys(this.subHandlers()).length;\r\n const allSubHandlersLoaded = this._loadedHandlers() === totalSubHandlers;\r\n\r\n if (allSubHandlersLoaded && this.mainHandler()) {\r\n this.loadMainHandler();\r\n }\r\n }\r\n\r\n private loadMainHandler() {\r\n if (!this.mainHandler()) return;\r\n\r\n this.addReactiveLogic(this.mainHandler()!.getReactiveLogic(this.form()));\r\n this._allHandlersLoaded.set(true);\r\n }\r\n}\r\n","/*\r\n * Public API Surface of core\r\n */\r\n\r\nexport * from './lib/form-control-value-assessor';\r\nexport * from './lib/form-control-base';\r\nexport * from './lib/form-handler-base';\r\nexport * from './lib/form-mapper-base';\r\nexport * from './lib/form-orchestrator-base';\r\nexport * from './lib/form.util';\r\n","/**\n * Generated bundle index. Do not edit.\n */\n\nexport * from './public-api';\n"],"names":[],"mappings":"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"}
|
|
@@ -1,32 +1,21 @@
|
|
|
1
|
-
import {
|
|
2
|
-
import {
|
|
1
|
+
import { FormControl, FormGroup } from '@angular/forms';
|
|
2
|
+
import { OnInit } from '@angular/core';
|
|
3
|
+
import { FormControlValueAccessor } from './form-control-value-assessor';
|
|
3
4
|
import * as i0 from "@angular/core";
|
|
4
|
-
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
|
|
19
|
-
static
|
|
20
|
-
id: string;
|
|
21
|
-
_value: T | null;
|
|
22
|
-
get value(): T | null;
|
|
23
|
-
set value(val: T | null);
|
|
24
|
-
onChange: (_value: T) => void;
|
|
25
|
-
onTouched: () => void;
|
|
26
|
-
constructor(ngControl: NgControl, elementRef: ElementRef<HTMLElement>);
|
|
27
|
-
writeValue(value: T): void;
|
|
28
|
-
registerOnChange(fn: (value: T) => void): void;
|
|
29
|
-
registerOnTouched(fn: () => void): void;
|
|
30
|
-
static ɵfac: i0.ɵɵFactoryDeclaration<FormControlBase<any>, [{ optional: true; self: true; }, null]>;
|
|
31
|
-
static ɵdir: i0.ɵɵDirectiveDeclaration<FormControlBase<any>, never, never, { "name": { "alias": "name"; "required": false; }; "placeholder": { "alias": "placeholder"; "required": false; }; "required": { "alias": "required"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; "readonly": { "alias": "readonly"; "required": false; }; "formControlName": { "alias": "formControlName"; "required": false; }; }, {}, never, never, true, never>;
|
|
5
|
+
export declare abstract class FormControlBase<T> extends FormControlValueAccessor<T> implements OnInit {
|
|
6
|
+
label: string;
|
|
7
|
+
classList: string[];
|
|
8
|
+
loading: boolean;
|
|
9
|
+
protected _form: import("@angular/core").WritableSignal<FormGroup<any>>;
|
|
10
|
+
protected _controlName: import("@angular/core").WritableSignal<string>;
|
|
11
|
+
readonly form: import("@angular/core").Signal<FormGroup<any>>;
|
|
12
|
+
readonly controlName: import("@angular/core").Signal<string>;
|
|
13
|
+
readonly control: import("@angular/core").Signal<FormControl<any>>;
|
|
14
|
+
private parentFormGroup;
|
|
15
|
+
ngOnInit(): void;
|
|
16
|
+
protected isRequired(): boolean;
|
|
17
|
+
protected serverError: import("@angular/core").Signal<string | undefined>;
|
|
18
|
+
protected getErrorMessage(): string;
|
|
19
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<FormControlBase<any>, never>;
|
|
20
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<FormControlBase<any>, never, never, { "label": { "alias": "label"; "required": false; }; "classList": { "alias": "classList"; "required": false; }; "loading": { "alias": "loading"; "required": false; }; }, {}, never, never, true, never>;
|
|
32
21
|
}
|
|
@@ -0,0 +1,32 @@
|
|
|
1
|
+
import { ElementRef, ChangeDetectorRef } from '@angular/core';
|
|
2
|
+
import { ControlValueAccessor, NgControl } from '@angular/forms';
|
|
3
|
+
import * as i0 from "@angular/core";
|
|
4
|
+
/**
|
|
5
|
+
* Base implementation for custom form controls that integrate with Angular Reactive Forms (ControlValueAccessor)
|
|
6
|
+
*
|
|
7
|
+
* NOTE: This class is UI-layer only and should not be used in form orchestration logic.
|
|
8
|
+
*/
|
|
9
|
+
export declare abstract class FormControlValueAccessor<T> implements ControlValueAccessor {
|
|
10
|
+
ngControl: NgControl;
|
|
11
|
+
protected elementRef: ElementRef<HTMLElement>;
|
|
12
|
+
protected readonly cdr: ChangeDetectorRef;
|
|
13
|
+
name: string;
|
|
14
|
+
placeholder: string;
|
|
15
|
+
required: boolean;
|
|
16
|
+
disabled: boolean;
|
|
17
|
+
readonly: boolean;
|
|
18
|
+
formControlName?: string;
|
|
19
|
+
static nextId: number;
|
|
20
|
+
id: string;
|
|
21
|
+
_value: T | null;
|
|
22
|
+
get value(): T | null;
|
|
23
|
+
set value(val: T | null);
|
|
24
|
+
onChange: (_value: T) => void;
|
|
25
|
+
onTouched: () => void;
|
|
26
|
+
constructor(ngControl: NgControl, elementRef: ElementRef<HTMLElement>);
|
|
27
|
+
writeValue(value: T): void;
|
|
28
|
+
registerOnChange(fn: (value: T) => void): void;
|
|
29
|
+
registerOnTouched(fn: () => void): void;
|
|
30
|
+
static ɵfac: i0.ɵɵFactoryDeclaration<FormControlValueAccessor<any>, [{ optional: true; self: true; }, null]>;
|
|
31
|
+
static ɵdir: i0.ɵɵDirectiveDeclaration<FormControlValueAccessor<any>, never, never, { "name": { "alias": "name"; "required": false; }; "placeholder": { "alias": "placeholder"; "required": false; }; "required": { "alias": "required"; "required": false; }; "disabled": { "alias": "disabled"; "required": false; }; "readonly": { "alias": "readonly"; "required": false; }; "formControlName": { "alias": "formControlName"; "required": false; }; }, {}, never, never, true, never>;
|
|
32
|
+
}
|
package/package.json
CHANGED
package/public-api.d.ts
CHANGED
|
@@ -1 +1,6 @@
|
|
|
1
|
-
export * from './lib/
|
|
1
|
+
export * from './lib/form-control-value-assessor';
|
|
2
|
+
export * from './lib/form-control-base';
|
|
3
|
+
export * from './lib/form-handler-base';
|
|
4
|
+
export * from './lib/form-mapper-base';
|
|
5
|
+
export * from './lib/form-orchestrator-base';
|
|
6
|
+
export * from './lib/form.util';
|