@mtna/web-form-angular 0.0.1
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/_index.scss +25 -0
- package/bundles/mtna-web-form-angular.umd.js +4982 -0
- package/bundles/mtna-web-form-angular.umd.js.map +1 -0
- package/esm2015/lib/core/animations.js +8 -0
- package/esm2015/lib/core/api-config.js +3 -0
- package/esm2015/lib/core/index.js +7 -0
- package/esm2015/lib/core/pipes/coerce-date-item.pipe.js +23 -0
- package/esm2015/lib/core/pipes/coerce-form-group.pipe.js +35 -0
- package/esm2015/lib/core/pipes/date-quarter.pipe.js +43 -0
- package/esm2015/lib/core/pipes/get-auto-complete.pipe.js +42 -0
- package/esm2015/lib/core/pipes/get-available-quarters.pipe.js +29 -0
- package/esm2015/lib/core/pipes/get-form-control-error.pipe.js +20 -0
- package/esm2015/lib/core/pipes/get-list-icon.pipe.js +53 -0
- package/esm2015/lib/core/pipes/get-object-type.pipe.js +17 -0
- package/esm2015/lib/core/pipes/get-selected-option.pipe.js +20 -0
- package/esm2015/lib/core/pipes/index.js +71 -0
- package/esm2015/lib/core/pipes/items-with-value.pipe.js +30 -0
- package/esm2015/lib/core/pipes/sanitize-html.pipe.js +23 -0
- package/esm2015/lib/core/services/api.service.js +67 -0
- package/esm2015/lib/core/services/form-manager.service.js +465 -0
- package/esm2015/lib/core/services/index.js +5 -0
- package/esm2015/lib/core/services/ng-form.service.js +150 -0
- package/esm2015/lib/core/services/ui.service.js +333 -0
- package/esm2015/lib/core/utilities/find-form-control.js +29 -0
- package/esm2015/lib/core/utilities/find-form-item.js +25 -0
- package/esm2015/lib/core/utilities/flatten-steps.js +10 -0
- package/esm2015/lib/core/utilities/generate-form-steps.js +25 -0
- package/esm2015/lib/core/utilities/index.js +9 -0
- package/esm2015/lib/core/utilities/is-form-group.js +6 -0
- package/esm2015/lib/core/utilities/serializer-util.js +358 -0
- package/esm2015/lib/core/utilities/update-step-status.js +38 -0
- package/esm2015/lib/core/validators/index.js +3 -0
- package/esm2015/lib/core/validators/instant-validation-result-error-state-matcher.js +23 -0
- package/esm2015/lib/core/validators/validation-result-validator.js +19 -0
- package/esm2015/lib/file-upload/file-upload.component.js +66 -0
- package/esm2015/lib/file-upload/index.js +2 -0
- package/esm2015/lib/form/base-form-item.model.js +2 -0
- package/esm2015/lib/form/form-disabler.component.js +38 -0
- package/esm2015/lib/form/form-item.directive.js +22 -0
- package/esm2015/lib/form/form.component.js +250 -0
- package/esm2015/lib/form/form.module.js +214 -0
- package/esm2015/lib/form/index.js +5 -0
- package/esm2015/lib/form/white-list.model.js +11 -0
- package/esm2015/lib/form-item-controls/base-form-item-control.directive.js +58 -0
- package/esm2015/lib/form-item-controls/base-parent-item-control.directive.js +42 -0
- package/esm2015/lib/form-item-controls/base-single-selection-parent.directive.js +53 -0
- package/esm2015/lib/form-item-controls/date-range.component.js +84 -0
- package/esm2015/lib/form-item-controls/date.component.js +134 -0
- package/esm2015/lib/form-item-controls/dropdown.component.js +109 -0
- package/esm2015/lib/form-item-controls/index.js +8 -0
- package/esm2015/lib/form-item-controls/input.component.js +149 -0
- package/esm2015/lib/form-item-controls/year-quarter-range.component.js +75 -0
- package/esm2015/lib/form-item-controls/year-quarter.component.js +228 -0
- package/esm2015/lib/form-stepper/form-step-group.component.js +95 -0
- package/esm2015/lib/form-stepper/form-step-group.pipe.js +20 -0
- package/esm2015/lib/form-stepper/form-step.component.js +110 -0
- package/esm2015/lib/form-stepper/form-step.util.js +28 -0
- package/esm2015/lib/form-stepper/form-stepper.component.js +115 -0
- package/esm2015/lib/form-stepper/form-stepper.module.js +53 -0
- package/esm2015/lib/form-stepper/index.js +8 -0
- package/esm2015/lib/form-stepper/models/base-form-step.model.js +6 -0
- package/esm2015/lib/form-stepper/models/form-step-group.model.js +13 -0
- package/esm2015/lib/form-stepper/models/form-step.model.js +12 -0
- package/esm2015/lib/form-stepper/models/index.js +4 -0
- package/esm2015/lib/groups/form-item-dialog-data.model.js +14 -0
- package/esm2015/lib/groups/group-components.js +646 -0
- package/esm2015/lib/groups/index.js +3 -0
- package/esm2015/lib/groups/repeatable-item.component.js +23 -0
- package/esm2015/lib/item-header/index.js +2 -0
- package/esm2015/lib/item-header/item-header.component.js +108 -0
- package/esm2015/lib/static-form-items/index.js +3 -0
- package/esm2015/lib/static-form-items/ordered-list-item.js +50 -0
- package/esm2015/lib/static-form-items/unordered-list-item.js +52 -0
- package/esm2015/lib/step-card/index.js +2 -0
- package/esm2015/lib/step-card/step-card.component.js +53 -0
- package/esm2015/mtna-web-form-angular.js +5 -0
- package/esm2015/public-api.js +10 -0
- package/fesm2015/mtna-web-form-angular.js +4614 -0
- package/fesm2015/mtna-web-form-angular.js.map +1 -0
- package/lib/_index.scss +25 -0
- package/lib/core/_all-color.scss +25 -0
- package/lib/core/_all-theme.scss +28 -0
- package/lib/core/_all-typography.scss +25 -0
- package/lib/core/_core-theme.scss +43 -0
- package/lib/core/animations.d.ts +2 -0
- package/lib/core/api-config.d.ts +5 -0
- package/lib/core/index.d.ts +6 -0
- package/lib/core/pipes/coerce-date-item.pipe.d.ts +11 -0
- package/lib/core/pipes/coerce-form-group.pipe.d.ts +23 -0
- package/lib/core/pipes/date-quarter.pipe.d.ts +7 -0
- package/lib/core/pipes/get-auto-complete.pipe.d.ts +7 -0
- package/lib/core/pipes/get-available-quarters.pipe.d.ts +8 -0
- package/lib/core/pipes/get-form-control-error.pipe.d.ts +14 -0
- package/lib/core/pipes/get-list-icon.pipe.d.ts +7 -0
- package/lib/core/pipes/get-object-type.pipe.d.ts +7 -0
- package/lib/core/pipes/get-selected-option.pipe.d.ts +8 -0
- package/lib/core/pipes/index.d.ts +28 -0
- package/lib/core/pipes/items-with-value.pipe.d.ts +13 -0
- package/lib/core/pipes/sanitize-html.pipe.d.ts +10 -0
- package/lib/core/services/api.service.d.ts +41 -0
- package/lib/core/services/form-manager.service.d.ts +184 -0
- package/lib/core/services/index.d.ts +4 -0
- package/lib/core/services/ng-form.service.d.ts +76 -0
- package/lib/core/services/ui.service.d.ts +123 -0
- package/lib/core/utilities/find-form-control.d.ts +11 -0
- package/lib/core/utilities/find-form-item.d.ts +10 -0
- package/lib/core/utilities/flatten-steps.d.ts +7 -0
- package/lib/core/utilities/generate-form-steps.d.ts +3 -0
- package/lib/core/utilities/index.d.ts +8 -0
- package/lib/core/utilities/is-form-group.d.ts +3 -0
- package/lib/core/utilities/serializer-util.d.ts +81 -0
- package/lib/core/utilities/update-step-status.d.ts +14 -0
- package/lib/core/validators/index.d.ts +2 -0
- package/lib/core/validators/instant-validation-result-error-state-matcher.d.ts +13 -0
- package/lib/core/validators/validation-result-validator.d.ts +12 -0
- package/lib/file-upload/file-upload.component.d.ts +23 -0
- package/lib/file-upload/index.d.ts +1 -0
- package/lib/form/base-form-item.model.d.ts +15 -0
- package/lib/form/form-disabler.component.d.ts +6 -0
- package/lib/form/form-item.directive.d.ts +13 -0
- package/lib/form/form.component.d.ts +89 -0
- package/lib/form/form.module.d.ts +52 -0
- package/lib/form/index.d.ts +4 -0
- package/lib/form/white-list.model.d.ts +6 -0
- package/lib/form-item-controls/_date-item-theme.scss +29 -0
- package/lib/form-item-controls/_form-control-theme.scss +28 -0
- package/lib/form-item-controls/_year-quarter-item-theme.scss +28 -0
- package/lib/form-item-controls/base-form-item-control.directive.d.ts +26 -0
- package/lib/form-item-controls/base-parent-item-control.directive.d.ts +19 -0
- package/lib/form-item-controls/base-single-selection-parent.directive.d.ts +26 -0
- package/lib/form-item-controls/date-range.component.d.ts +12 -0
- package/lib/form-item-controls/date.component.d.ts +24 -0
- package/lib/form-item-controls/dropdown.component.d.ts +9 -0
- package/lib/form-item-controls/index.d.ts +7 -0
- package/lib/form-item-controls/input.component.d.ts +14 -0
- package/lib/form-item-controls/year-quarter-range.component.d.ts +9 -0
- package/lib/form-item-controls/year-quarter.component.d.ts +33 -0
- package/lib/form-stepper/_form-step-group-theme.scss +43 -0
- package/lib/form-stepper/_form-stepper-theme.scss +72 -0
- package/lib/form-stepper/form-step-group.component.d.ts +23 -0
- package/lib/form-stepper/form-step-group.pipe.d.ts +11 -0
- package/lib/form-stepper/form-step.component.d.ts +27 -0
- package/lib/form-stepper/form-step.util.d.ts +27 -0
- package/lib/form-stepper/form-stepper.component.d.ts +35 -0
- package/lib/form-stepper/form-stepper.module.d.ts +18 -0
- package/lib/form-stepper/index.d.ts +7 -0
- package/lib/form-stepper/models/base-form-step.model.d.ts +4 -0
- package/lib/form-stepper/models/form-step-group.model.d.ts +8 -0
- package/lib/form-stepper/models/form-step.model.d.ts +11 -0
- package/lib/form-stepper/models/index.d.ts +3 -0
- package/lib/groups/_checkbox-theme.scss +62 -0
- package/lib/groups/_form-group-theme.scss +32 -0
- package/lib/groups/_multiple-choice-theme.scss +65 -0
- package/lib/groups/_repeatable-item-theme.scss +28 -0
- package/lib/groups/form-item-dialog-data.model.d.ts +13 -0
- package/lib/groups/group-components.d.ts +221 -0
- package/lib/groups/index.d.ts +2 -0
- package/lib/groups/repeatable-item.component.d.ts +5 -0
- package/lib/item-header/_item-header-theme.scss +28 -0
- package/lib/item-header/index.d.ts +1 -0
- package/lib/item-header/item-header.component.d.ts +35 -0
- package/lib/static-form-items/index.d.ts +2 -0
- package/lib/static-form-items/ordered-list-item.d.ts +11 -0
- package/lib/static-form-items/unordered-list-item.d.ts +11 -0
- package/lib/step-card/_step-card-theme.scss +42 -0
- package/lib/step-card/index.d.ts +1 -0
- package/lib/step-card/step-card.component.d.ts +21 -0
- package/mtna-web-form-angular.d.ts +5 -0
- package/package.json +43 -0
- package/public-api.d.ts +9 -0
|
@@ -0,0 +1,150 @@
|
|
|
1
|
+
import { Injectable } from '@angular/core';
|
|
2
|
+
import { BehaviorSubject } from 'rxjs';
|
|
3
|
+
import { skipWhile, take } from 'rxjs/operators';
|
|
4
|
+
import * as i0 from "@angular/core";
|
|
5
|
+
export class MtnaWfNgFormService {
|
|
6
|
+
constructor() {
|
|
7
|
+
this.unsavedChanges$ = new BehaviorSubject(false);
|
|
8
|
+
}
|
|
9
|
+
/**
|
|
10
|
+
* Informs a caller when the MtnaForm is dirty and has unsaved changes.
|
|
11
|
+
*/
|
|
12
|
+
get unsavedChanges() {
|
|
13
|
+
return this.unsavedChanges$.asObservable();
|
|
14
|
+
}
|
|
15
|
+
/**
|
|
16
|
+
* Add a control from the ngForm
|
|
17
|
+
* @param controlId id of the control being added
|
|
18
|
+
* @param parentIds array of ids representing the path to the parent control
|
|
19
|
+
* @param newControl the new form control to be added
|
|
20
|
+
*/
|
|
21
|
+
addControl(controlId, parentIds, newControl) {
|
|
22
|
+
if (!this.ngForm) {
|
|
23
|
+
return;
|
|
24
|
+
}
|
|
25
|
+
const parentControl = this.ngForm.get(parentIds);
|
|
26
|
+
parentControl.addControl(controlId, newControl);
|
|
27
|
+
parentControl.markAsTouched();
|
|
28
|
+
parentControl.markAsDirty();
|
|
29
|
+
this.forceUnsavedChange();
|
|
30
|
+
}
|
|
31
|
+
/**
|
|
32
|
+
* Remove a control from the ngForm.
|
|
33
|
+
*
|
|
34
|
+
* @param controlId is of the control to remove
|
|
35
|
+
* @param parentIds rray of ids representing the path to the parent control
|
|
36
|
+
*/
|
|
37
|
+
removeControl(controlId, parentIds) {
|
|
38
|
+
if (!this.ngForm) {
|
|
39
|
+
return;
|
|
40
|
+
}
|
|
41
|
+
const parentControl = this.ngForm.get(parentIds);
|
|
42
|
+
parentControl.removeControl(controlId);
|
|
43
|
+
parentControl.markAsTouched();
|
|
44
|
+
parentControl.markAsDirty();
|
|
45
|
+
this.forceUnsavedChange();
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Get the form if it exists
|
|
49
|
+
* @returns the angular form or null
|
|
50
|
+
*/
|
|
51
|
+
getForm() {
|
|
52
|
+
return this.ngForm;
|
|
53
|
+
}
|
|
54
|
+
/**
|
|
55
|
+
* Returns a JSON representation of form values
|
|
56
|
+
*/
|
|
57
|
+
getFormValue() {
|
|
58
|
+
var _a;
|
|
59
|
+
// This includes values for disabled controls.
|
|
60
|
+
// We loop through MtnaForm to insert values,
|
|
61
|
+
// so we need the disabled controls to ensure no info loss.
|
|
62
|
+
return (_a = this.ngForm) === null || _a === void 0 ? void 0 : _a.getRawValue();
|
|
63
|
+
}
|
|
64
|
+
/**
|
|
65
|
+
* @returns Whether a form has been set on this service
|
|
66
|
+
*/
|
|
67
|
+
hasForm() {
|
|
68
|
+
return !!this.ngForm;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* @returns Whether the form is dirty.
|
|
72
|
+
*/
|
|
73
|
+
isFormDirty() {
|
|
74
|
+
return !!this.ngForm && this.ngForm.dirty;
|
|
75
|
+
}
|
|
76
|
+
/**
|
|
77
|
+
* Unsubscribes from value changes & sets the ngForm to null.
|
|
78
|
+
* Should be called when any component using an MtnaForm is destroyed.
|
|
79
|
+
*/
|
|
80
|
+
disposeForm() {
|
|
81
|
+
this.unsubscribeValueChanges();
|
|
82
|
+
this.ngForm = undefined;
|
|
83
|
+
}
|
|
84
|
+
/**
|
|
85
|
+
* Forces the unsavedChanges observable to emit true.
|
|
86
|
+
* Use with care, one example is when repeatable items are added/removed from the form.
|
|
87
|
+
*/
|
|
88
|
+
forceUnsavedChange() {
|
|
89
|
+
this.unsubscribeValueChanges();
|
|
90
|
+
this.unsavedChanges$.next(true);
|
|
91
|
+
}
|
|
92
|
+
/**
|
|
93
|
+
* Marks ngForm as pristine & untouched.
|
|
94
|
+
*/
|
|
95
|
+
markAsPristineAndUntouched() {
|
|
96
|
+
if (this.ngForm) {
|
|
97
|
+
this.ngForm.markAsPristine();
|
|
98
|
+
this.ngForm.markAsUntouched();
|
|
99
|
+
this.unsavedChanges$.next(this.ngForm.dirty);
|
|
100
|
+
this.observeNgFormChanges();
|
|
101
|
+
}
|
|
102
|
+
}
|
|
103
|
+
/**
|
|
104
|
+
* Resets the ngForm.
|
|
105
|
+
*/
|
|
106
|
+
resetForm() {
|
|
107
|
+
if (this.ngForm) {
|
|
108
|
+
this.ngForm.reset();
|
|
109
|
+
this.unsavedChanges$.next(this.ngForm.dirty);
|
|
110
|
+
this.observeNgFormChanges();
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
/**
|
|
114
|
+
* Sets a FormGroup on the service so an external source may observe changes.
|
|
115
|
+
* @param ngForm FormGroup to be registered.
|
|
116
|
+
*/
|
|
117
|
+
setForm(ngForm) {
|
|
118
|
+
this.ngForm = ngForm;
|
|
119
|
+
this.unsavedChanges$.next(this.ngForm.dirty);
|
|
120
|
+
if (this.ngForm.pristine) {
|
|
121
|
+
this.observeNgFormChanges();
|
|
122
|
+
}
|
|
123
|
+
}
|
|
124
|
+
/**
|
|
125
|
+
* Subscribes to ngForm status changes & updates the unsavedChanges observable
|
|
126
|
+
*/
|
|
127
|
+
observeNgFormChanges() {
|
|
128
|
+
this.unsubscribeValueChanges();
|
|
129
|
+
if (this.ngForm) {
|
|
130
|
+
this.valueChangeSubscription = this.ngForm.valueChanges
|
|
131
|
+
.pipe(skipWhile(() => (this.ngForm ? this.ngForm.pristine : true)), take(1))
|
|
132
|
+
.subscribe(() => { var _a; return this.unsavedChanges$.next((_a = this.ngForm) === null || _a === void 0 ? void 0 : _a.dirty); });
|
|
133
|
+
}
|
|
134
|
+
}
|
|
135
|
+
/**
|
|
136
|
+
* Removes the ngForm status change subscription.
|
|
137
|
+
*/
|
|
138
|
+
unsubscribeValueChanges() {
|
|
139
|
+
if (!!this.valueChangeSubscription && !this.valueChangeSubscription.closed) {
|
|
140
|
+
this.valueChangeSubscription.unsubscribe();
|
|
141
|
+
}
|
|
142
|
+
}
|
|
143
|
+
}
|
|
144
|
+
MtnaWfNgFormService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfNgFormService, deps: [], target: i0.ɵɵFactoryTarget.Injectable });
|
|
145
|
+
MtnaWfNgFormService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfNgFormService, providedIn: 'root' });
|
|
146
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfNgFormService, decorators: [{
|
|
147
|
+
type: Injectable,
|
|
148
|
+
args: [{ providedIn: 'root' }]
|
|
149
|
+
}] });
|
|
150
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ng-form.service.js","sourceRoot":"","sources":["../../../../../../../projects/web-form-angular/src/lib/core/services/ng-form.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { AbstractControl, FormGroup } from '@angular/forms';\n\nimport { BehaviorSubject, Observable, Subject, Subscription } from 'rxjs';\nimport { skipWhile, take } from 'rxjs/operators';\n\n@Injectable({ providedIn: 'root' })\nexport class MtnaWfNgFormService {\n  private ngForm: FormGroup | undefined;\n  private unsavedChanges$: Subject<boolean> = new BehaviorSubject<boolean>(false);\n  private valueChangeSubscription: Subscription | undefined;\n\n  /**\n   * Informs a caller when the MtnaForm is dirty and has unsaved changes.\n   */\n  get unsavedChanges(): Observable<boolean> {\n    return this.unsavedChanges$.asObservable();\n  }\n\n  /**\n   * Add a control from the ngForm\n   * @param controlId id of the control being added\n   * @param parentIds array of ids representing the path to the parent control\n   * @param newControl the new form control to be added\n   */\n  addControl(controlId: string, parentIds: string[], newControl: AbstractControl) {\n    if (!this.ngForm) {\n      return;\n    }\n\n    const parentControl = this.ngForm.get(parentIds) as FormGroup;\n    parentControl.addControl(controlId, newControl);\n    parentControl.markAsTouched();\n    parentControl.markAsDirty();\n    this.forceUnsavedChange();\n  }\n\n  /**\n   * Remove a control from the ngForm.\n   *\n   * @param controlId is of the control to remove\n   * @param parentIds rray of ids representing the path to the parent control\n   */\n  removeControl(controlId: string, parentIds: string[]) {\n    if (!this.ngForm) {\n      return;\n    }\n\n    const parentControl = this.ngForm.get(parentIds) as FormGroup;\n    parentControl.removeControl(controlId);\n    parentControl.markAsTouched();\n    parentControl.markAsDirty();\n    this.forceUnsavedChange();\n  }\n\n  /**\n   * Get the form if it exists\n   * @returns the angular form or null\n   */\n  getForm(): FormGroup | undefined {\n    return this.ngForm;\n  }\n\n  /**\n   * Returns a JSON representation of form values\n   */\n  getFormValue() {\n    // This includes values for disabled controls.\n    // We loop through MtnaForm to insert values,\n    // so we need the disabled controls to ensure no info loss.\n    return this.ngForm?.getRawValue();\n  }\n\n  /**\n   * @returns Whether a form has been set on this service\n   */\n  hasForm(): boolean {\n    return !!this.ngForm;\n  }\n\n  /**\n   * @returns Whether the form is dirty.\n   */\n  isFormDirty(): boolean {\n    return !!this.ngForm && this.ngForm.dirty;\n  }\n\n  /**\n   * Unsubscribes from value changes & sets the ngForm to null.\n   * Should be called when any component using an MtnaForm is destroyed.\n   */\n  disposeForm() {\n    this.unsubscribeValueChanges();\n    this.ngForm = undefined;\n  }\n\n  /**\n   * Forces the unsavedChanges observable to emit true.\n   * Use with care, one example is when repeatable items are added/removed from the form.\n   */\n  forceUnsavedChange() {\n    this.unsubscribeValueChanges();\n    this.unsavedChanges$.next(true);\n  }\n\n  /**\n   * Marks ngForm as pristine & untouched.\n   */\n  markAsPristineAndUntouched() {\n    if (this.ngForm) {\n      this.ngForm.markAsPristine();\n      this.ngForm.markAsUntouched();\n      this.unsavedChanges$.next(this.ngForm.dirty);\n      this.observeNgFormChanges();\n    }\n  }\n\n  /**\n   * Resets the ngForm.\n   */\n  resetForm() {\n    if (this.ngForm) {\n      this.ngForm.reset();\n      this.unsavedChanges$.next(this.ngForm.dirty);\n      this.observeNgFormChanges();\n    }\n  }\n\n  /**\n   * Sets a FormGroup on the service so an external source may observe changes.\n   * @param ngForm FormGroup to be registered.\n   */\n  setForm(ngForm: FormGroup) {\n    this.ngForm = ngForm;\n    this.unsavedChanges$.next(this.ngForm.dirty);\n    if (this.ngForm.pristine) {\n      this.observeNgFormChanges();\n    }\n  }\n\n  /**\n   * Subscribes to ngForm status changes & updates the unsavedChanges observable\n   */\n  private observeNgFormChanges() {\n    this.unsubscribeValueChanges();\n    if (this.ngForm) {\n      this.valueChangeSubscription = this.ngForm.valueChanges\n        .pipe(\n          skipWhile(() => (this.ngForm ? this.ngForm.pristine : true)),\n          take(1)\n        )\n        .subscribe(() => this.unsavedChanges$.next(this.ngForm?.dirty));\n    }\n  }\n\n  /**\n   * Removes the ngForm status change subscription.\n   */\n  private unsubscribeValueChanges() {\n    if (!!this.valueChangeSubscription && !this.valueChangeSubscription.closed) {\n      this.valueChangeSubscription.unsubscribe();\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,333 @@
|
|
|
1
|
+
import { Inject, Injectable, InjectionToken, Optional } from '@angular/core';
|
|
2
|
+
import { FormGroup } from '@angular/forms';
|
|
3
|
+
import { isPresent, isType } from '@mtna/core-ts';
|
|
4
|
+
import { isConditionalItem, isOptionGroupItemImpl, isOptionItemImpl } from '@mtna/web-form-ts';
|
|
5
|
+
import { BehaviorSubject } from 'rxjs';
|
|
6
|
+
import * as i0 from "@angular/core";
|
|
7
|
+
import * as i1 from "@angular/material/dialog";
|
|
8
|
+
/** List of default error messages to display */
|
|
9
|
+
export const DEFAULT_ERROR_MESSAGES = {
|
|
10
|
+
email: 'This field must include a valid email format.',
|
|
11
|
+
max: (item) => !isPresent(item) || !isPresent(item === null || item === void 0 ? void 0 : item.maxValue)
|
|
12
|
+
? `This field's value must be less.`
|
|
13
|
+
: `${isPresent(item === null || item === void 0 ? void 0 : item.minValue) ? `Min value: ${item === null || item === void 0 ? void 0 : item.minValue} - ` : ''}Max value: ${item === null || item === void 0 ? void 0 : item.minValue}`,
|
|
14
|
+
maxlength: 'This field has exceeded the allowed maximum.',
|
|
15
|
+
maxSelected: null,
|
|
16
|
+
min: (item) => !isPresent(item) || !isPresent(item === null || item === void 0 ? void 0 : item.minValue)
|
|
17
|
+
? `This field's value must be greater.`
|
|
18
|
+
: `Min value: ${item === null || item === void 0 ? void 0 : item.minValue}${!isPresent(item === null || item === void 0 ? void 0 : item.maxValue) ? '' : ` - Max value: ${item === null || item === void 0 ? void 0 : item.maxValue}`}`,
|
|
19
|
+
minlength: 'This field has not met the required minimum.',
|
|
20
|
+
minSelected: null,
|
|
21
|
+
nonDigit: `This field's value must be numeric.`,
|
|
22
|
+
pattern: `This field's input does not meet the accepted pattern.`,
|
|
23
|
+
positive: `This field's value cannot be negative.`,
|
|
24
|
+
required: 'This field is required.',
|
|
25
|
+
requiredTrue: `This field's value must be 'true'.`,
|
|
26
|
+
wholeNumber: `This field cannot contain a decimal.`,
|
|
27
|
+
matDatepickerParse: `This field's value is not an accepted format.`,
|
|
28
|
+
};
|
|
29
|
+
export const WEB_FORM_CUSTOM_ERRORS = new InjectionToken('mtnaWebFormCustomFormErrors');
|
|
30
|
+
export class MtnaWfUIService {
|
|
31
|
+
// eslint-disable-next-line @typescript-eslint/no-explicit-any
|
|
32
|
+
constructor(dialog, errors) {
|
|
33
|
+
this.dialog = dialog;
|
|
34
|
+
/** Map of annotation counts for each annotatable item */
|
|
35
|
+
this.annotations = {};
|
|
36
|
+
/** Map of components in the MtnaForm */
|
|
37
|
+
this.components = {};
|
|
38
|
+
/** Map of item.disabled values that have changed */
|
|
39
|
+
this.disabledState = {};
|
|
40
|
+
/** Map of toggler ID's for components that have multiple togglers */
|
|
41
|
+
this.multipleEnablers = {};
|
|
42
|
+
// Workaround for: Metadata collected contains an error that will be reported at runtime: Could not resolve type CustomErrors.
|
|
43
|
+
// see https://stackoverflow.com/questions/49513359/could-not-resolve-type-document-in-angular5/53715642#53715642
|
|
44
|
+
this.errors = errors;
|
|
45
|
+
}
|
|
46
|
+
getDisabledStates() {
|
|
47
|
+
return Object.assign({}, this.disabledState);
|
|
48
|
+
}
|
|
49
|
+
/**
|
|
50
|
+
* Returns an error message matching the error id
|
|
51
|
+
* @param errorId Id of the error, to find the associated error message
|
|
52
|
+
* @param item FormItem to pass to ValidationErrors that are functions. For instance,
|
|
53
|
+
* this is useful for 'min' & 'max' value limits to display more helpful messages
|
|
54
|
+
*/
|
|
55
|
+
getError(errorId, item) {
|
|
56
|
+
// default when no custom message exists
|
|
57
|
+
let error = DEFAULT_ERROR_MESSAGES[errorId];
|
|
58
|
+
if (!!this.errors && !!item) {
|
|
59
|
+
// check for instance specific, via instanceId
|
|
60
|
+
if (isPresent(this.errors[item.instanceId]) && isPresent(this.errors[item.instanceId][errorId])) {
|
|
61
|
+
error = this.errors[item.instanceId][errorId];
|
|
62
|
+
// check for item specific, via itemId
|
|
63
|
+
}
|
|
64
|
+
else if (isPresent(this.errors[item.itemId]) && isPresent(this.errors[item.itemId][errorId])) {
|
|
65
|
+
error = this.errors[item.itemId][errorId];
|
|
66
|
+
}
|
|
67
|
+
}
|
|
68
|
+
return typeof error === 'function' ? error(item) : error;
|
|
69
|
+
}
|
|
70
|
+
/**
|
|
71
|
+
* Get a registered component by intance id.
|
|
72
|
+
* @param id the instanceId
|
|
73
|
+
* @returns the {@link FormItemComponent} that represents that ID or undefined if not registered.
|
|
74
|
+
*/
|
|
75
|
+
getItem(id) {
|
|
76
|
+
return this.components[id];
|
|
77
|
+
}
|
|
78
|
+
/**
|
|
79
|
+
* Sets the initial map of annotation counts, and/or pushes the new count onto the component observable stream
|
|
80
|
+
* @param map AnnotatableNumberMap of counts mapped to their annotatableIds
|
|
81
|
+
*/
|
|
82
|
+
registerAnnotations(map) {
|
|
83
|
+
for (const id in map) {
|
|
84
|
+
if (Object.prototype.hasOwnProperty.call(map, id)) {
|
|
85
|
+
if (!this.annotations[id]) {
|
|
86
|
+
this.annotations[id] = new BehaviorSubject(map[id]);
|
|
87
|
+
}
|
|
88
|
+
else {
|
|
89
|
+
this.annotations[id].next(map[id]);
|
|
90
|
+
}
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
}
|
|
94
|
+
/**
|
|
95
|
+
* Maps a component to its items instanceId.
|
|
96
|
+
* @param component component calling this method, i.e. 'this'
|
|
97
|
+
*/
|
|
98
|
+
registerSelf(component) {
|
|
99
|
+
this.components[component.item.instanceId] = component;
|
|
100
|
+
if (!!component.item.annotatableId && component.item.annotationEnabled) {
|
|
101
|
+
if (!this.annotations[component.item.annotatableId]) {
|
|
102
|
+
this.annotations[component.item.annotatableId] = new BehaviorSubject(null);
|
|
103
|
+
}
|
|
104
|
+
component.annotations$ = this.annotations[component.item.annotatableId].asObservable();
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
/**
|
|
108
|
+
* Toggles a FormItem's visiblity and enable/disable state
|
|
109
|
+
*
|
|
110
|
+
* @param enabledItemIds array of component IDs for enabled items.
|
|
111
|
+
* @param disabledItemIds array of component IDs for disabled items.
|
|
112
|
+
* @param enabled Whether the enabledItemIds are enabled.
|
|
113
|
+
* @param callerId instanceId of the item invoking this function
|
|
114
|
+
* @param reset Whether to reset the multiple enabler map of the toggled item
|
|
115
|
+
*/
|
|
116
|
+
toggleFormItems(enabledItemIds, disabledItemIds, enabled, callerId, reset = false) {
|
|
117
|
+
if ((!enabledItemIds || !enabledItemIds.length) && (!disabledItemIds || !disabledItemIds.length)) {
|
|
118
|
+
return;
|
|
119
|
+
}
|
|
120
|
+
const allIds = [enabledItemIds, disabledItemIds];
|
|
121
|
+
for (let i = 0; i < allIds.length; i++) {
|
|
122
|
+
const itemIds = allIds[i];
|
|
123
|
+
const isEnabled = i === 0 ? enabled : !enabled;
|
|
124
|
+
for (const id of itemIds) {
|
|
125
|
+
const component = this.components[id];
|
|
126
|
+
// If component is registered, toggle it
|
|
127
|
+
if (component) {
|
|
128
|
+
const item = component.item;
|
|
129
|
+
const hasMultipleEnablers = isConditionalItem(item) && item.condition ? item.condition.itemIds.length > 1 : false;
|
|
130
|
+
// If item has more than 1 enabler, handle multiple enablers
|
|
131
|
+
if (hasMultipleEnablers) {
|
|
132
|
+
if (reset) {
|
|
133
|
+
this.multipleEnablers[item.instanceId] = null;
|
|
134
|
+
}
|
|
135
|
+
// If a map has not been registered, then build up a map to register
|
|
136
|
+
// Should only happen the first time an item is toggled.
|
|
137
|
+
if (!this.multipleEnablers[item.instanceId] && isConditionalItem(item)) {
|
|
138
|
+
this.multipleEnablers[item.instanceId] = this._registerCurrentEnablers(item);
|
|
139
|
+
}
|
|
140
|
+
// handle Multiple Enablers
|
|
141
|
+
this._handleMultipleEnablers(component, isEnabled, callerId);
|
|
142
|
+
}
|
|
143
|
+
else {
|
|
144
|
+
// Not multiple enablers
|
|
145
|
+
this._toggleDisabledStateAndControl(component, isEnabled);
|
|
146
|
+
}
|
|
147
|
+
}
|
|
148
|
+
else {
|
|
149
|
+
console.error('[toggleFormItems] component has not been registerd', id, enabled, callerId);
|
|
150
|
+
}
|
|
151
|
+
}
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
/**
|
|
155
|
+
* Generates map of all IDs currently influencing an item.
|
|
156
|
+
* @param toggledItem item being toggled
|
|
157
|
+
*/
|
|
158
|
+
_registerCurrentEnablers(toggledItem) {
|
|
159
|
+
var _a;
|
|
160
|
+
const enablers = {};
|
|
161
|
+
const allIds = ((_a = toggledItem.condition) === null || _a === void 0 ? void 0 : _a.itemIds) || [];
|
|
162
|
+
for (const id of allIds) {
|
|
163
|
+
const component = this.components[id];
|
|
164
|
+
if (component) {
|
|
165
|
+
if (this._isCurrentlyToggling(component, toggledItem.instanceId)) {
|
|
166
|
+
enablers[component.item.instanceId] = component.item.instanceId;
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
else {
|
|
170
|
+
console.error('[registerCurrentEnablers] component has not been registered.', id);
|
|
171
|
+
}
|
|
172
|
+
}
|
|
173
|
+
return enablers;
|
|
174
|
+
}
|
|
175
|
+
/**
|
|
176
|
+
* Resets the map of components. This should be called by the form when
|
|
177
|
+
* it is destroyed.
|
|
178
|
+
*/
|
|
179
|
+
resetComponentMap() {
|
|
180
|
+
this.annotations = {};
|
|
181
|
+
this.components = {};
|
|
182
|
+
this.disabledState = {};
|
|
183
|
+
this.multipleEnablers = {};
|
|
184
|
+
}
|
|
185
|
+
/**
|
|
186
|
+
* Determines if an influencer is currently toggling the specified toggledItemId.
|
|
187
|
+
* @param influencer Item to check whether it's currently influencing the toggledItemId
|
|
188
|
+
* @param toggledId InstanceId of the item being toggled
|
|
189
|
+
* @returns Whether the influencer is currently toggling the toggled item
|
|
190
|
+
*/
|
|
191
|
+
_isCurrentlyToggling(influencer, toggledId) {
|
|
192
|
+
var _a, _b, _c;
|
|
193
|
+
const item = influencer.item;
|
|
194
|
+
// Item has single option property, so check if selected, or item.value is 'true'
|
|
195
|
+
if (isOptionItemImpl(item)) {
|
|
196
|
+
return item.option.selected || !!item.value || !!((_a = influencer.control) === null || _a === void 0 ? void 0 : _a.value);
|
|
197
|
+
}
|
|
198
|
+
else if (isOptionGroupItemImpl(item)) {
|
|
199
|
+
const value = (_c = (_b = influencer.control) === null || _b === void 0 ? void 0 : _b.value) !== null && _c !== void 0 ? _c : item.value;
|
|
200
|
+
if (Array.isArray(value)) {
|
|
201
|
+
// find the index of options that contain the 'toggledId' in their enabledItems
|
|
202
|
+
// & check if at least 1 is selected, or its 'true' in the value
|
|
203
|
+
const optionIndexes = item.options.map((o, i) => (o.enabledItems.some((id) => id === toggledId) ? i : null)).filter(isPresent);
|
|
204
|
+
return optionIndexes.some((i) => !!value[i]);
|
|
205
|
+
}
|
|
206
|
+
else if (typeof value === 'string' && value.length) {
|
|
207
|
+
// if the value is a single value, get the enabling option ids and check for the value in them
|
|
208
|
+
return item.options.some((o) => o.option.id === value && o.enabledItems.some((id) => id === toggledId));
|
|
209
|
+
}
|
|
210
|
+
}
|
|
211
|
+
return false;
|
|
212
|
+
}
|
|
213
|
+
/**
|
|
214
|
+
* Handles component visibility for a FormItem with multiple enablers in its condition
|
|
215
|
+
* @param component FormItemComponent being toggled
|
|
216
|
+
* @param enabled Whether the component is being enabled
|
|
217
|
+
* @param callerId ID of the caller toggling this component
|
|
218
|
+
*/
|
|
219
|
+
_handleMultipleEnablers(component, enabled, callerId) {
|
|
220
|
+
var _a;
|
|
221
|
+
const item = component.item;
|
|
222
|
+
const enablers = Object.assign({}, this.multipleEnablers[item.instanceId]);
|
|
223
|
+
let isEnabled;
|
|
224
|
+
if (enabled) {
|
|
225
|
+
enablers[callerId] = callerId;
|
|
226
|
+
isEnabled =
|
|
227
|
+
isConditionalItem(item) &&
|
|
228
|
+
!!item.condition &&
|
|
229
|
+
(!item.condition.allRequired || item.condition.itemIds.every((id) => !!enablers[id]));
|
|
230
|
+
}
|
|
231
|
+
else {
|
|
232
|
+
delete enablers[callerId];
|
|
233
|
+
isEnabled = !((isConditionalItem(item) && ((_a = item.condition) === null || _a === void 0 ? void 0 : _a.allRequired)) || !Object.keys(enablers).length);
|
|
234
|
+
}
|
|
235
|
+
// If enabled === disabled, then the state has changed
|
|
236
|
+
if (this._isDisabled(item) === isEnabled) {
|
|
237
|
+
this._toggleDisabledStateAndControl(component, isEnabled);
|
|
238
|
+
}
|
|
239
|
+
// set the new enabler map
|
|
240
|
+
this.multipleEnablers[item.instanceId] = enablers;
|
|
241
|
+
}
|
|
242
|
+
/**
|
|
243
|
+
* Enables or disables the control on a FormItemComponent
|
|
244
|
+
* @param component FormItemComponent to enable or disable its control
|
|
245
|
+
* @param enabled whether to enable the control
|
|
246
|
+
*/
|
|
247
|
+
_toggleControlState(component, enabled) {
|
|
248
|
+
var _a, _b, _c;
|
|
249
|
+
if (enabled && !((_a = component.control) === null || _a === void 0 ? void 0 : _a.enabled)) {
|
|
250
|
+
// Enable the control if it isn't already
|
|
251
|
+
if (component.control instanceof FormGroup) {
|
|
252
|
+
this._enableIndividualControls(component);
|
|
253
|
+
}
|
|
254
|
+
else {
|
|
255
|
+
(_b = component.control) === null || _b === void 0 ? void 0 : _b.enable();
|
|
256
|
+
}
|
|
257
|
+
}
|
|
258
|
+
else if (!enabled && ((_c = component.control) === null || _c === void 0 ? void 0 : _c.enabled)) {
|
|
259
|
+
// Disable the control if it isn't already
|
|
260
|
+
component.control.disable();
|
|
261
|
+
}
|
|
262
|
+
}
|
|
263
|
+
/**
|
|
264
|
+
* Toggles both the component visiblity and the control's enable/disable status
|
|
265
|
+
* @param component FormItemComponent to show or hide and enable or disable
|
|
266
|
+
* @param enabled whether to show the component and whether to enable the component
|
|
267
|
+
*/
|
|
268
|
+
_toggleDisabledStateAndControl(component, enabled) {
|
|
269
|
+
this.disabledState[component.item.instanceId] = !enabled;
|
|
270
|
+
this._toggleControlState(component, enabled);
|
|
271
|
+
if (isConditionalItem(component.item) && component.item.hideOnDisable) {
|
|
272
|
+
component.visible = enabled;
|
|
273
|
+
}
|
|
274
|
+
}
|
|
275
|
+
/**
|
|
276
|
+
* Enables controls individually using recursion.
|
|
277
|
+
* FormGroups should only exist for items with 'items' arrays, or
|
|
278
|
+
* if the component is a BaseParentItemControl
|
|
279
|
+
* @param comp FormItemComponent
|
|
280
|
+
*/
|
|
281
|
+
_enableIndividualControls(comp) {
|
|
282
|
+
const group = comp.control;
|
|
283
|
+
if (isOptionItemImpl(comp.item) || isOptionGroupItemImpl(comp.item)) {
|
|
284
|
+
const map = comp.item.optionItemMap;
|
|
285
|
+
const self = group.get(comp.item.instanceId);
|
|
286
|
+
self === null || self === void 0 ? void 0 : self.enable();
|
|
287
|
+
for (const optionId in map) {
|
|
288
|
+
if (Object.prototype.hasOwnProperty.call(map, optionId)) {
|
|
289
|
+
const childComp = this.components[map[optionId].instanceId];
|
|
290
|
+
if (!this._isDisabled(childComp.item)) {
|
|
291
|
+
this._toggleControlState(childComp, true);
|
|
292
|
+
}
|
|
293
|
+
}
|
|
294
|
+
}
|
|
295
|
+
}
|
|
296
|
+
else if (isType(comp.item, 'items') && comp.item.items.length) {
|
|
297
|
+
// Item must have 'items' array
|
|
298
|
+
for (const item of comp.item.items) {
|
|
299
|
+
const childComp = this.components[item.instanceId];
|
|
300
|
+
if (!this._isDisabled(item)) {
|
|
301
|
+
this._toggleControlState(childComp, true);
|
|
302
|
+
}
|
|
303
|
+
}
|
|
304
|
+
}
|
|
305
|
+
group.updateValueAndValidity();
|
|
306
|
+
}
|
|
307
|
+
/**
|
|
308
|
+
* Determines if a component is disabled. Prioritizes the disabledState value,
|
|
309
|
+
* if it exists, otherwise returns the item's previous disabled value
|
|
310
|
+
* @param item item to check for its disabled state
|
|
311
|
+
*/
|
|
312
|
+
_isDisabled(item) {
|
|
313
|
+
const existsInDisabledStatMap = isPresent(this.disabledState[item.instanceId]);
|
|
314
|
+
if (!existsInDisabledStatMap) {
|
|
315
|
+
return isConditionalItem(item) ? item.disabled : false;
|
|
316
|
+
}
|
|
317
|
+
else {
|
|
318
|
+
return this.disabledState[item.instanceId];
|
|
319
|
+
}
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
MtnaWfUIService.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfUIService, deps: [{ token: i1.MatDialog }, { token: WEB_FORM_CUSTOM_ERRORS, optional: true }], target: i0.ɵɵFactoryTarget.Injectable });
|
|
323
|
+
MtnaWfUIService.ɵprov = i0.ɵɵngDeclareInjectable({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfUIService, providedIn: 'root' });
|
|
324
|
+
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "12.2.17", ngImport: i0, type: MtnaWfUIService, decorators: [{
|
|
325
|
+
type: Injectable,
|
|
326
|
+
args: [{ providedIn: 'root' }]
|
|
327
|
+
}], ctorParameters: function () { return [{ type: i1.MatDialog }, { type: undefined, decorators: [{
|
|
328
|
+
type: Optional
|
|
329
|
+
}, {
|
|
330
|
+
type: Inject,
|
|
331
|
+
args: [WEB_FORM_CUSTOM_ERRORS]
|
|
332
|
+
}] }]; } });
|
|
333
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ui.service.js","sourceRoot":"","sources":["../../../../../../../projects/web-form-angular/src/lib/core/services/ui.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Inject, Injectable, InjectionToken, Optional } from '@angular/core';\nimport { FormGroup, ValidationErrors } from '@angular/forms';\nimport { MatDialog } from '@angular/material/dialog';\n\nimport { isPresent, isType } from '@mtna/core-ts';\nimport { AnnotatableNumberMap } from '@mtna/pojo-social-angular';\nimport { ConditionalItem, FormItem, FormItemImpl, isConditionalItem, isOptionGroupItemImpl, isOptionItemImpl } from '@mtna/web-form-ts';\n\nimport { BehaviorSubject } from 'rxjs';\n\nimport type { MtnaWfFormItemComponent } from '../../groups/group-components';\n\n/** Registry of custom error messages for a specific form item. A custom error can also\n * be a function returning a string. The FormItem object is passed to any ValidationErrors\n * that are functions, so item specific information can be provided.\n * @example {\n * max: `This field's value must be less.`,\n * min: (item: FormItemImpl<any>) => instanceofWholeNumberItem(item)\n * ? `This field's value must be greater than ${item.minValue}.`\n * : `This field's value must be greater.`\n * }\n */\n// tslint:disable-next-line:interface-over-type-literal\nexport type CustomErrors = {\n  [id: string]: ValidationErrors;\n};\n\n/** List of default error messages to display */\nexport const DEFAULT_ERROR_MESSAGES: ValidationErrors = {\n  email: 'This field must include a valid email format.',\n  max: (item?: FormItemImpl<number> & { maxValue: number; minValue: number }) =>\n    !isPresent(item) || !isPresent(item?.maxValue)\n      ? `This field's value must be less.`\n      : `${isPresent(item?.minValue) ? `Min value: ${item?.minValue} - ` : ''}Max value: ${item?.minValue}`,\n  maxlength: 'This field has exceeded the allowed maximum.',\n  maxSelected: null,\n  min: (item?: FormItemImpl<number> & { maxValue: number; minValue: number }) =>\n    !isPresent(item) || !isPresent(item?.minValue)\n      ? `This field's value must be greater.`\n      : `Min value: ${item?.minValue}${!isPresent(item?.maxValue) ? '' : ` - Max value: ${item?.maxValue}`}`,\n  minlength: 'This field has not met the required minimum.',\n  minSelected: null,\n  nonDigit: `This field's value must be numeric.`,\n  pattern: `This field's input does not meet the accepted pattern.`,\n  positive: `This field's value cannot be negative.`,\n  required: 'This field is required.',\n  requiredTrue: `This field's value must be 'true'.`,\n  wholeNumber: `This field cannot contain a decimal.`,\n  matDatepickerParse: `This field's value is not an accepted format.`,\n};\n\nexport const WEB_FORM_CUSTOM_ERRORS = new InjectionToken<CustomErrors>('mtnaWebFormCustomFormErrors');\n\n@Injectable({ providedIn: 'root' })\nexport class MtnaWfUIService {\n  /** Map of annotation counts for each annotatable item */\n  private annotations: { [id: string]: BehaviorSubject<number | null> } = {};\n  /** Map of components in the MtnaForm */\n  private components: { [instanceId: string]: MtnaWfFormItemComponent } = {};\n  /** Map of item.disabled values that have changed */\n  private disabledState: { [instanceId: string]: boolean } = {};\n  /** Map of toggler ID's for components that have multiple togglers */\n  private multipleEnablers: { [instanceId: string]: { [togglerId: string]: string } | null } = {};\n  private errors: CustomErrors;\n\n  // eslint-disable-next-line @typescript-eslint/no-explicit-any\n  constructor(private dialog: MatDialog, @Optional() @Inject(WEB_FORM_CUSTOM_ERRORS) errors?: any) {\n    // Workaround for: Metadata collected contains an error that will be reported at runtime: Could not resolve type CustomErrors.\n    // see https://stackoverflow.com/questions/49513359/could-not-resolve-type-document-in-angular5/53715642#53715642\n    this.errors = errors as CustomErrors;\n  }\n\n  getDisabledStates() {\n    return { ...this.disabledState };\n  }\n\n  /**\n   * Returns an error message matching the error id\n   * @param errorId Id of the error, to find the associated error message\n   * @param item FormItem to pass to ValidationErrors that are functions. For instance,\n   * this is useful for 'min' & 'max' value limits to display more helpful messages\n   */\n  getError(errorId: string, item?: FormItemImpl<unknown>): string {\n    // default when no custom message exists\n    let error = DEFAULT_ERROR_MESSAGES[errorId];\n    if (!!this.errors && !!item) {\n      // check for instance specific, via instanceId\n      if (isPresent(this.errors[item.instanceId]) && isPresent(this.errors[item.instanceId][errorId])) {\n        error = this.errors[item.instanceId][errorId];\n        // check for item specific, via itemId\n      } else if (isPresent(this.errors[item.itemId]) && isPresent(this.errors[item.itemId][errorId])) {\n        error = this.errors[item.itemId][errorId];\n      }\n    }\n    return typeof error === 'function' ? error(item) : error;\n  }\n\n  /**\n   * Get a registered component by intance id.\n   * @param id the instanceId\n   * @returns the {@link FormItemComponent} that represents that ID or undefined if not registered.\n   */\n  getItem(id: string): MtnaWfFormItemComponent | undefined {\n    return this.components[id];\n  }\n\n  /**\n   * Sets the initial map of annotation counts, and/or pushes the new count onto the component observable stream\n   * @param map AnnotatableNumberMap of counts mapped to their annotatableIds\n   */\n  registerAnnotations(map: AnnotatableNumberMap) {\n    for (const id in map) {\n      if (Object.prototype.hasOwnProperty.call(map, id)) {\n        if (!this.annotations[id]) {\n          this.annotations[id] = new BehaviorSubject<number | null>(map[id]);\n        } else {\n          this.annotations[id].next(map[id]);\n        }\n      }\n    }\n  }\n\n  /**\n   * Maps a component to its items instanceId.\n   * @param component component calling this method, i.e. 'this'\n   */\n  registerSelf(component: MtnaWfFormItemComponent) {\n    this.components[component.item.instanceId] = component;\n    if (!!component.item.annotatableId && component.item.annotationEnabled) {\n      if (!this.annotations[component.item.annotatableId]) {\n        this.annotations[component.item.annotatableId] = new BehaviorSubject<number | null>(null);\n      }\n      component.annotations$ = this.annotations[component.item.annotatableId].asObservable();\n    }\n  }\n\n  /**\n   * Toggles a FormItem's visiblity and enable/disable state\n   *\n   * @param enabledItemIds array of component IDs for enabled items.\n   * @param disabledItemIds array of component IDs for disabled items.\n   * @param enabled Whether the enabledItemIds are enabled.\n   * @param callerId instanceId of the item invoking this function\n   * @param reset Whether to reset the multiple enabler map of the toggled item\n   */\n  toggleFormItems(enabledItemIds: Array<string>, disabledItemIds: Array<string>, enabled: boolean, callerId: string, reset = false) {\n    if ((!enabledItemIds || !enabledItemIds.length) && (!disabledItemIds || !disabledItemIds.length)) {\n      return;\n    }\n    const allIds = [enabledItemIds, disabledItemIds];\n    for (let i = 0; i < allIds.length; i++) {\n      const itemIds = allIds[i];\n      const isEnabled = i === 0 ? enabled : !enabled;\n      for (const id of itemIds) {\n        const component = this.components[id];\n        // If component is registered, toggle it\n        if (component) {\n          const item = component.item;\n          const hasMultipleEnablers = isConditionalItem(item) && item.condition ? item.condition.itemIds.length > 1 : false;\n          // If item has more than 1 enabler, handle multiple enablers\n          if (hasMultipleEnablers) {\n            if (reset) {\n              this.multipleEnablers[item.instanceId] = null;\n            }\n            // If a map has not been registered, then build up a map to register\n            // Should only happen the first time an item is toggled.\n            if (!this.multipleEnablers[item.instanceId] && isConditionalItem(item)) {\n              this.multipleEnablers[item.instanceId] = this._registerCurrentEnablers(item);\n            }\n            // handle Multiple Enablers\n            this._handleMultipleEnablers(component, isEnabled, callerId);\n          } else {\n            // Not multiple enablers\n            this._toggleDisabledStateAndControl(component, isEnabled);\n          }\n        } else {\n          console.error('[toggleFormItems] component has not been registerd', id, enabled, callerId);\n        }\n      }\n    }\n  }\n\n  /**\n   * Generates map of all IDs currently influencing an item.\n   * @param toggledItem item being toggled\n   */\n  private _registerCurrentEnablers(toggledItem: ConditionalItem): { [callerId: string]: string } {\n    const enablers: { [callerId: string]: string } = {};\n    const allIds = toggledItem.condition?.itemIds || [];\n\n    for (const id of allIds) {\n      const component = this.components[id];\n      if (component) {\n        if (this._isCurrentlyToggling(component, toggledItem.instanceId)) {\n          enablers[component.item.instanceId] = component.item.instanceId;\n        }\n      } else {\n        console.error('[registerCurrentEnablers] component has not been registered.', id);\n      }\n    }\n\n    return enablers;\n  }\n\n  /**\n   * Resets the map of components. This should be called by the form when\n   * it is destroyed.\n   */\n  resetComponentMap() {\n    this.annotations = {};\n    this.components = {};\n    this.disabledState = {};\n    this.multipleEnablers = {};\n  }\n\n  /**\n   * Determines if an influencer is currently toggling the specified toggledItemId.\n   * @param influencer Item to check whether it's currently influencing the toggledItemId\n   * @param toggledId InstanceId of the item being toggled\n   * @returns Whether the influencer is currently toggling the toggled item\n   */\n  private _isCurrentlyToggling(influencer: MtnaWfFormItemComponent, toggledId: string): boolean {\n    const item = influencer.item;\n    // Item has single option property, so check if selected, or item.value is 'true'\n    if (isOptionItemImpl(item)) {\n      return item.option.selected || !!item.value || !!influencer.control?.value;\n    } else if (isOptionGroupItemImpl(item)) {\n      const value = influencer.control?.value ?? item.value;\n      if (Array.isArray(value)) {\n        // find the index of options that contain the 'toggledId' in their enabledItems\n        // & check if at least 1 is selected, or its 'true' in the value\n        const optionIndexes = item.options.map((o, i) => (o.enabledItems.some((id) => id === toggledId) ? i : null)).filter(isPresent);\n        return optionIndexes.some((i) => !!value[i]);\n      } else if (typeof value === 'string' && value.length) {\n        // if the value is a single value, get the enabling option ids and check for the value in them\n        return item.options.some((o) => o.option.id === value && o.enabledItems.some((id) => id === toggledId));\n      }\n    }\n    return false;\n  }\n\n  /**\n   * Handles component visibility for a FormItem with multiple enablers in its condition\n   * @param component FormItemComponent being toggled\n   * @param enabled Whether the component is being enabled\n   * @param callerId ID of the caller toggling this component\n   */\n  private _handleMultipleEnablers(component: MtnaWfFormItemComponent, enabled: boolean, callerId: string) {\n    const item = component.item;\n    const enablers = { ...this.multipleEnablers[item.instanceId] };\n    let isEnabled: boolean;\n    if (enabled) {\n      enablers[callerId] = callerId;\n      isEnabled =\n        isConditionalItem(item) &&\n        !!item.condition &&\n        (!item.condition.allRequired || item.condition.itemIds.every((id) => !!enablers[id]));\n    } else {\n      delete enablers[callerId];\n      isEnabled = !((isConditionalItem(item) && item.condition?.allRequired) || !Object.keys(enablers).length);\n    }\n    // If enabled === disabled, then the state has changed\n    if (this._isDisabled(item) === isEnabled) {\n      this._toggleDisabledStateAndControl(component, isEnabled);\n    }\n    // set the new enabler map\n    this.multipleEnablers[item.instanceId] = enablers;\n  }\n\n  /**\n   * Enables or disables the control on a FormItemComponent\n   * @param component FormItemComponent to enable or disable its control\n   * @param enabled whether to enable the control\n   */\n  private _toggleControlState(component: MtnaWfFormItemComponent, enabled: boolean) {\n    if (enabled && !component.control?.enabled) {\n      // Enable the control if it isn't already\n      if (component.control instanceof FormGroup) {\n        this._enableIndividualControls(component);\n      } else {\n        component.control?.enable();\n      }\n    } else if (!enabled && component.control?.enabled) {\n      // Disable the control if it isn't already\n      component.control.disable();\n    }\n  }\n\n  /**\n   * Toggles both the component visiblity and the control's enable/disable status\n   * @param component FormItemComponent to show or hide and enable or disable\n   * @param enabled whether to show the component and whether to enable the component\n   */\n  private _toggleDisabledStateAndControl(component: MtnaWfFormItemComponent, enabled: boolean) {\n    this.disabledState[component.item.instanceId] = !enabled;\n    this._toggleControlState(component, enabled);\n    if (isConditionalItem(component.item) && component.item.hideOnDisable) {\n      component.visible = enabled;\n    }\n  }\n\n  /**\n   * Enables controls individually using recursion.\n   * FormGroups should only exist for items with 'items' arrays, or\n   * if the component is a BaseParentItemControl\n   * @param comp FormItemComponent\n   */\n  private _enableIndividualControls(comp: MtnaWfFormItemComponent) {\n    const group = comp.control as FormGroup;\n\n    if (isOptionItemImpl(comp.item) || isOptionGroupItemImpl(comp.item)) {\n      const map = comp.item.optionItemMap;\n      const self = group.get(comp.item.instanceId);\n      self?.enable();\n      for (const optionId in map) {\n        if (Object.prototype.hasOwnProperty.call(map, optionId)) {\n          const childComp = this.components[map[optionId].instanceId];\n          if (!this._isDisabled(childComp.item)) {\n            this._toggleControlState(childComp, true);\n          }\n        }\n      }\n    } else if (isType<{ items: FormItemImpl<unknown>[] }>(comp.item, 'items') && comp.item.items.length) {\n      // Item must have 'items' array\n      for (const item of comp.item.items) {\n        const childComp = this.components[item.instanceId];\n        if (!this._isDisabled(item)) {\n          this._toggleControlState(childComp, true);\n        }\n      }\n    }\n    group.updateValueAndValidity();\n  }\n\n  /**\n   * Determines if a component is disabled. Prioritizes the disabledState value,\n   * if it exists, otherwise returns the item's previous disabled value\n   * @param item item to check for its disabled state\n   */\n  private _isDisabled(item: ConditionalItem | FormItem): boolean {\n    const existsInDisabledStatMap = isPresent(this.disabledState[item.instanceId]);\n    if (!existsInDisabledStatMap) {\n      return isConditionalItem(item) ? item.disabled : false;\n    } else {\n      return this.disabledState[item.instanceId];\n    }\n  }\n}\n"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
import { isFormGroup } from './is-form-group';
|
|
2
|
+
/**
|
|
3
|
+
* Recursively find a form control by name in a form group.
|
|
4
|
+
* Depth first search down the tree of form groups to find a control.
|
|
5
|
+
*
|
|
6
|
+
* @note This will only work if the entire form has unique form control names.
|
|
7
|
+
* @param name The unique name of the form cotrol to find.
|
|
8
|
+
* @param group The {@link FormGroup} to search through.
|
|
9
|
+
* @returns the found control or null
|
|
10
|
+
*/
|
|
11
|
+
export function findFormControl(name, group) {
|
|
12
|
+
for (const key in group.controls) {
|
|
13
|
+
if (Object.prototype.hasOwnProperty.call(group.controls, key)) {
|
|
14
|
+
const control = group.controls[key];
|
|
15
|
+
if (key === name) {
|
|
16
|
+
return control;
|
|
17
|
+
}
|
|
18
|
+
// Recurse down if it's a FormGroup
|
|
19
|
+
if (isFormGroup(control)) {
|
|
20
|
+
const found = findFormControl(name, control);
|
|
21
|
+
if (found) {
|
|
22
|
+
return found;
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
}
|
|
26
|
+
}
|
|
27
|
+
return null;
|
|
28
|
+
}
|
|
29
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmluZC1mb3JtLWNvbnRyb2wuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy93ZWItZm9ybS1hbmd1bGFyL3NyYy9saWIvY29yZS91dGlsaXRpZXMvZmluZC1mb3JtLWNvbnRyb2wudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBRUEsT0FBTyxFQUFFLFdBQVcsRUFBRSxNQUFNLGlCQUFpQixDQUFDO0FBRTlDOzs7Ozs7OztHQVFHO0FBQ0gsTUFBTSxVQUFVLGVBQWUsQ0FBQyxJQUFZLEVBQUUsS0FBZ0I7SUFDNUQsS0FBSyxNQUFNLEdBQUcsSUFBSSxLQUFLLENBQUMsUUFBUSxFQUFFO1FBQ2hDLElBQUksTUFBTSxDQUFDLFNBQVMsQ0FBQyxjQUFjLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxRQUFRLEVBQUUsR0FBRyxDQUFDLEVBQUU7WUFDN0QsTUFBTSxPQUFPLEdBQUcsS0FBSyxDQUFDLFFBQVEsQ0FBQyxHQUFHLENBQUMsQ0FBQztZQUNwQyxJQUFJLEdBQUcsS0FBSyxJQUFJLEVBQUU7Z0JBQ2hCLE9BQU8sT0FBTyxDQUFDO2FBQ2hCO1lBRUQsbUNBQW1DO1lBQ25DLElBQUksV0FBVyxDQUFDLE9BQU8sQ0FBQyxFQUFFO2dCQUN4QixNQUFNLEtBQUssR0FBRyxlQUFlLENBQUMsSUFBSSxFQUFFLE9BQU8sQ0FBQyxDQUFDO2dCQUM3QyxJQUFJLEtBQUssRUFBRTtvQkFDVCxPQUFPLEtBQUssQ0FBQztpQkFDZDthQUNGO1NBQ0Y7S0FDRjtJQUNELE9BQU8sSUFBSSxDQUFDO0FBQ2QsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEFic3RyYWN0Q29udHJvbCwgRm9ybUdyb3VwIH0gZnJvbSAnQGFuZ3VsYXIvZm9ybXMnO1xuXG5pbXBvcnQgeyBpc0Zvcm1Hcm91cCB9IGZyb20gJy4vaXMtZm9ybS1ncm91cCc7XG5cbi8qKlxuICogUmVjdXJzaXZlbHkgZmluZCBhIGZvcm0gY29udHJvbCBieSBuYW1lIGluIGEgZm9ybSBncm91cC5cbiAqIERlcHRoIGZpcnN0IHNlYXJjaCBkb3duIHRoZSB0cmVlIG9mIGZvcm0gZ3JvdXBzIHRvIGZpbmQgYSBjb250cm9sLlxuICpcbiAqIEBub3RlIFRoaXMgd2lsbCBvbmx5IHdvcmsgaWYgdGhlIGVudGlyZSBmb3JtIGhhcyB1bmlxdWUgZm9ybSBjb250cm9sIG5hbWVzLlxuICogQHBhcmFtIG5hbWUgVGhlIHVuaXF1ZSBuYW1lIG9mIHRoZSBmb3JtIGNvdHJvbCB0byBmaW5kLlxuICogQHBhcmFtIGdyb3VwIFRoZSB7QGxpbmsgRm9ybUdyb3VwfSB0byBzZWFyY2ggdGhyb3VnaC5cbiAqIEByZXR1cm5zIHRoZSBmb3VuZCBjb250cm9sIG9yIG51bGxcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpbmRGb3JtQ29udHJvbChuYW1lOiBzdHJpbmcsIGdyb3VwOiBGb3JtR3JvdXApOiBBYnN0cmFjdENvbnRyb2wgfCBudWxsIHtcbiAgZm9yIChjb25zdCBrZXkgaW4gZ3JvdXAuY29udHJvbHMpIHtcbiAgICBpZiAoT2JqZWN0LnByb3RvdHlwZS5oYXNPd25Qcm9wZXJ0eS5jYWxsKGdyb3VwLmNvbnRyb2xzLCBrZXkpKSB7XG4gICAgICBjb25zdCBjb250cm9sID0gZ3JvdXAuY29udHJvbHNba2V5XTtcbiAgICAgIGlmIChrZXkgPT09IG5hbWUpIHtcbiAgICAgICAgcmV0dXJuIGNvbnRyb2w7XG4gICAgICB9XG5cbiAgICAgIC8vIFJlY3Vyc2UgZG93biBpZiBpdCdzIGEgRm9ybUdyb3VwXG4gICAgICBpZiAoaXNGb3JtR3JvdXAoY29udHJvbCkpIHtcbiAgICAgICAgY29uc3QgZm91bmQgPSBmaW5kRm9ybUNvbnRyb2wobmFtZSwgY29udHJvbCk7XG4gICAgICAgIGlmIChmb3VuZCkge1xuICAgICAgICAgIHJldHVybiBmb3VuZDtcbiAgICAgICAgfVxuICAgICAgfVxuICAgIH1cbiAgfVxuICByZXR1cm4gbnVsbDtcbn1cbiJdfQ==
|
|
@@ -0,0 +1,25 @@
|
|
|
1
|
+
import { isType } from '@mtna/core-ts';
|
|
2
|
+
/**
|
|
3
|
+
* Recursively find a {@link FormItem} by instanceId in a {@link Form}.
|
|
4
|
+
* Depth first search down the tree of {@link FormItemGroup} to find an item.
|
|
5
|
+
*
|
|
6
|
+
* @param instanceId The unique ID of the {@link FormItem} to find.
|
|
7
|
+
* @param group The {@link Form} or {@link FormItemGroup} to search through.
|
|
8
|
+
* @returns the found {@link FormItem} or `null`
|
|
9
|
+
*/
|
|
10
|
+
export function findFormItem(instanceId, group) {
|
|
11
|
+
for (const item of group.items) {
|
|
12
|
+
if (item.instanceId === instanceId) {
|
|
13
|
+
return item;
|
|
14
|
+
}
|
|
15
|
+
// If it's a FormGroup, search through it's children
|
|
16
|
+
if (isType(item, 'items')) {
|
|
17
|
+
const found = findFormItem(instanceId, item);
|
|
18
|
+
if (found) {
|
|
19
|
+
return found;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
}
|
|
23
|
+
return null;
|
|
24
|
+
}
|
|
25
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmluZC1mb3JtLWl0ZW0uanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi8uLi8uLi8uLi9wcm9qZWN0cy93ZWItZm9ybS1hbmd1bGFyL3NyYy9saWIvY29yZS91dGlsaXRpZXMvZmluZC1mb3JtLWl0ZW0udHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE1BQU0sRUFBRSxNQUFNLGVBQWUsQ0FBQztBQUd2Qzs7Ozs7OztHQU9HO0FBQ0gsTUFBTSxVQUFVLFlBQVksQ0FBQyxVQUFrQixFQUFFLEtBQTJCO0lBQzFFLEtBQUssTUFBTSxJQUFJLElBQUksS0FBSyxDQUFDLEtBQUssRUFBRTtRQUM5QixJQUFJLElBQUksQ0FBQyxVQUFVLEtBQUssVUFBVSxFQUFFO1lBQ2xDLE9BQU8sSUFBSSxDQUFDO1NBQ2I7UUFFRCxvREFBb0Q7UUFDcEQsSUFBSSxNQUFNLENBQWdCLElBQUksRUFBRSxPQUFPLENBQUMsRUFBRTtZQUN4QyxNQUFNLEtBQUssR0FBRyxZQUFZLENBQUMsVUFBVSxFQUFFLElBQUksQ0FBQyxDQUFDO1lBQzdDLElBQUksS0FBSyxFQUFFO2dCQUNULE9BQU8sS0FBSyxDQUFDO2FBQ2Q7U0FDRjtLQUNGO0lBQ0QsT0FBTyxJQUFJLENBQUM7QUFDZCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgaXNUeXBlIH0gZnJvbSAnQG10bmEvY29yZS10cyc7XG5pbXBvcnQgeyBGb3JtLCBGb3JtSXRlbSwgRm9ybUl0ZW1Hcm91cCB9IGZyb20gJ0BtdG5hL3dlYi1mb3JtLXRzJztcblxuLyoqXG4gKiBSZWN1cnNpdmVseSBmaW5kIGEge0BsaW5rIEZvcm1JdGVtfSBieSBpbnN0YW5jZUlkIGluIGEge0BsaW5rIEZvcm19LlxuICogRGVwdGggZmlyc3Qgc2VhcmNoIGRvd24gdGhlIHRyZWUgb2Yge0BsaW5rIEZvcm1JdGVtR3JvdXB9IHRvIGZpbmQgYW4gaXRlbS5cbiAqXG4gKiBAcGFyYW0gaW5zdGFuY2VJZCBUaGUgdW5pcXVlIElEIG9mIHRoZSB7QGxpbmsgRm9ybUl0ZW19IHRvIGZpbmQuXG4gKiBAcGFyYW0gZ3JvdXAgVGhlIHtAbGluayBGb3JtfSBvciB7QGxpbmsgRm9ybUl0ZW1Hcm91cH0gdG8gc2VhcmNoIHRocm91Z2guXG4gKiBAcmV0dXJucyB0aGUgZm91bmQge0BsaW5rIEZvcm1JdGVtfSBvciBgbnVsbGBcbiAqL1xuZXhwb3J0IGZ1bmN0aW9uIGZpbmRGb3JtSXRlbShpbnN0YW5jZUlkOiBzdHJpbmcsIGdyb3VwOiBGb3JtIHwgRm9ybUl0ZW1Hcm91cCk6IEZvcm1JdGVtIHwgbnVsbCB7XG4gIGZvciAoY29uc3QgaXRlbSBvZiBncm91cC5pdGVtcykge1xuICAgIGlmIChpdGVtLmluc3RhbmNlSWQgPT09IGluc3RhbmNlSWQpIHtcbiAgICAgIHJldHVybiBpdGVtO1xuICAgIH1cblxuICAgIC8vIElmIGl0J3MgYSBGb3JtR3JvdXAsIHNlYXJjaCB0aHJvdWdoIGl0J3MgY2hpbGRyZW5cbiAgICBpZiAoaXNUeXBlPEZvcm1JdGVtR3JvdXA+KGl0ZW0sICdpdGVtcycpKSB7XG4gICAgICBjb25zdCBmb3VuZCA9IGZpbmRGb3JtSXRlbShpbnN0YW5jZUlkLCBpdGVtKTtcbiAgICAgIGlmIChmb3VuZCkge1xuICAgICAgICByZXR1cm4gZm91bmQ7XG4gICAgICB9XG4gICAgfVxuICB9XG4gIHJldHVybiBudWxsO1xufVxuIl19
|
|
@@ -0,0 +1,10 @@
|
|
|
1
|
+
import { isFormStepGroup } from '../../form-stepper/models';
|
|
2
|
+
/**
|
|
3
|
+
* Flattens formStepGroup.steps and combines all steps into single array
|
|
4
|
+
* @param steps Array<FormStep | FormStepGroup>
|
|
5
|
+
* @returns FormStep[] array of the steps and children of step groups
|
|
6
|
+
*/
|
|
7
|
+
export function flattenSteps(steps) {
|
|
8
|
+
return steps.reduce((a, b) => a.concat(isFormStepGroup(b) ? b.steps : b), new Array());
|
|
9
|
+
}
|
|
10
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZmxhdHRlbi1zdGVwcy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uLy4uLy4uLy4uL3Byb2plY3RzL3dlYi1mb3JtLWFuZ3VsYXIvc3JjL2xpYi9jb3JlL3V0aWxpdGllcy9mbGF0dGVuLXN0ZXBzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBMkIsZUFBZSxFQUFFLE1BQU0sMkJBQTJCLENBQUM7QUFFckY7Ozs7R0FJRztBQUNILE1BQU0sVUFBVSxZQUFZLENBQUMsS0FBc0M7SUFDakUsT0FBTyxLQUFLLENBQUMsTUFBTSxDQUFDLENBQUMsQ0FBQyxFQUFFLENBQUMsRUFBRSxFQUFFLENBQUMsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxlQUFlLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxFQUFFLElBQUksS0FBSyxFQUFZLENBQUMsQ0FBQztBQUNuRyxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgRm9ybVN0ZXAsIEZvcm1TdGVwR3JvdXAsIGlzRm9ybVN0ZXBHcm91cCB9IGZyb20gJy4uLy4uL2Zvcm0tc3RlcHBlci9tb2RlbHMnO1xuXG4vKipcbiAqIEZsYXR0ZW5zIGZvcm1TdGVwR3JvdXAuc3RlcHMgYW5kIGNvbWJpbmVzIGFsbCBzdGVwcyBpbnRvIHNpbmdsZSBhcnJheVxuICogQHBhcmFtIHN0ZXBzIEFycmF5PEZvcm1TdGVwIHwgRm9ybVN0ZXBHcm91cD5cbiAqIEByZXR1cm5zIEZvcm1TdGVwW10gYXJyYXkgb2YgdGhlIHN0ZXBzIGFuZCBjaGlsZHJlbiBvZiBzdGVwIGdyb3Vwc1xuICovXG5leHBvcnQgZnVuY3Rpb24gZmxhdHRlblN0ZXBzKHN0ZXBzOiBBcnJheTxGb3JtU3RlcCB8IEZvcm1TdGVwR3JvdXA+KTogQXJyYXk8Rm9ybVN0ZXA+IHtcbiAgcmV0dXJuIHN0ZXBzLnJlZHVjZSgoYSwgYikgPT4gYS5jb25jYXQoaXNGb3JtU3RlcEdyb3VwKGIpID8gYi5zdGVwcyA6IGIpLCBuZXcgQXJyYXk8Rm9ybVN0ZXA+KCkpO1xufVxuIl19
|