@muraai/mnl-form 0.0.1-alpha-db0d938 → 0.0.1-alpha-f274a13
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/esm2022/lib/services/scoring.service.mjs +19 -7
- package/esm2022/lib/types/mu-helper-text.type.mjs +28 -16
- package/esm2022/lib/types/scrollable-tabs.types.mjs +3 -3
- package/fesm2022/muraai-mnl-form.mjs +47 -23
- package/fesm2022/muraai-mnl-form.mjs.map +1 -1
- package/lib/types/mu-helper-text.type.d.ts +4 -0
- package/package.json +1 -1
|
@@ -8,11 +8,21 @@ export class ScoringService {
|
|
|
8
8
|
calculateScore(model, fields) {
|
|
9
9
|
let totalScore = 0;
|
|
10
10
|
let categoryResults = {};
|
|
11
|
-
|
|
12
|
-
const categoryLabel = category.props?.label
|
|
13
|
-
const categoryWeight = category.props?.weight
|
|
14
|
-
|
|
15
|
-
|
|
11
|
+
const processCategory = (category) => {
|
|
12
|
+
const categoryLabel = category.props?.label;
|
|
13
|
+
const categoryWeight = category.props?.weight ?? 0;
|
|
14
|
+
// Only treat as scoring category if it has a label + weight
|
|
15
|
+
if (!categoryLabel || !categoryWeight) {
|
|
16
|
+
// Still need to go deeper in case children have categories
|
|
17
|
+
if (category.fieldGroup?.length) {
|
|
18
|
+
category.fieldGroup.forEach(processCategory);
|
|
19
|
+
}
|
|
20
|
+
return;
|
|
21
|
+
}
|
|
22
|
+
const questions = (category.fieldGroup || []).filter(q => q.props?.scoring);
|
|
23
|
+
if (questions.length === 0)
|
|
24
|
+
return;
|
|
25
|
+
const perQuestionWeight = categoryWeight / questions.length;
|
|
16
26
|
let categoryScore = 0;
|
|
17
27
|
for (const q of questions) {
|
|
18
28
|
const parentKey = category.key;
|
|
@@ -53,7 +63,9 @@ export class ScoringService {
|
|
|
53
63
|
score: categoryScore,
|
|
54
64
|
weight: categoryWeight,
|
|
55
65
|
};
|
|
56
|
-
}
|
|
66
|
+
};
|
|
67
|
+
// start recursive walk
|
|
68
|
+
fields.forEach(processCategory);
|
|
57
69
|
return {
|
|
58
70
|
totalScore,
|
|
59
71
|
categoryResults,
|
|
@@ -72,4 +84,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
72
84
|
type: Injectable,
|
|
73
85
|
args: [{ providedIn: 'root' }]
|
|
74
86
|
}] });
|
|
75
|
-
//# sourceMappingURL=data:application/json;base64,
|
|
87
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoic2NvcmluZy5zZXJ2aWNlLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vLi4vLi4vcHJvamVjdHMvZm9ybS1saWIvc3JjL2xpYi9zZXJ2aWNlcy9zY29yaW5nLnNlcnZpY2UudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLFVBQVUsRUFBRSxNQUFNLGVBQWUsQ0FBQzs7QUFNM0MsTUFBTSxPQUFPLGNBQWM7SUFFekIsV0FBVyxDQUFDLEtBQVUsRUFBRSxNQUEyQjtRQUNqRCxNQUFNLE1BQU0sR0FBRyxJQUFJLENBQUMsY0FBYyxDQUFDLEtBQUssRUFBRSxNQUFNLENBQUMsQ0FBQztRQUNsRCxPQUFPLE1BQU0sQ0FBQztJQUNoQixDQUFDO0lBRU0sY0FBYyxDQUFDLEtBQVUsRUFBRSxNQUF1RDtRQUN6RixJQUFJLFVBQVUsR0FBRyxDQUFDLENBQUM7UUFDbkIsSUFBSSxlQUFlLEdBQXNELEVBQUUsQ0FBQztRQUU1RSxNQUFNLGVBQWUsR0FBRyxDQUFDLFFBQXFELEVBQUUsRUFBRTtZQUNoRixNQUFNLGFBQWEsR0FBRyxRQUFRLENBQUMsS0FBSyxFQUFFLEtBQUssQ0FBQztZQUM1QyxNQUFNLGNBQWMsR0FBRyxRQUFRLENBQUMsS0FBSyxFQUFFLE1BQU0sSUFBSSxDQUFDLENBQUM7WUFFbkQsNERBQTREO1lBQzVELElBQUksQ0FBQyxhQUFhLElBQUksQ0FBQyxjQUFjLEVBQUUsQ0FBQztnQkFDdEMsMkRBQTJEO2dCQUMzRCxJQUFJLFFBQVEsQ0FBQyxVQUFVLEVBQUUsTUFBTSxFQUFFLENBQUM7b0JBQ2hDLFFBQVEsQ0FBQyxVQUFVLENBQUMsT0FBTyxDQUFDLGVBQWUsQ0FBQyxDQUFDO2dCQUMvQyxDQUFDO2dCQUNELE9BQU87WUFDVCxDQUFDO1lBRUQsTUFBTSxTQUFTLEdBQUcsQ0FBQyxRQUFRLENBQUMsVUFBVSxJQUFJLEVBQUUsQ0FBQyxDQUFDLE1BQU0sQ0FBQyxDQUFDLENBQUMsRUFBRSxDQUFD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QUFDLEdBQUcsQ0FBQyxDQUFDO0lBQ3BCLENBQUM7K0dBNUZVLGNBQWM7bUhBQWQsY0FBYyxjQURELE1BQU07OzRGQUNuQixjQUFjO2tCQUQxQixVQUFVO21CQUFDLEVBQUUsVUFBVSxFQUFFLE1BQU0sRUFBRSIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCB7IEluamVjdGFibGUgfSBmcm9tICdAYW5ndWxhci9jb3JlJztcbmltcG9ydCB7IEJlaGF2aW9yU3ViamVjdCB9IGZyb20gJ3J4anMnO1xuaW1wb3J0IHsgRm9ybWx5RmllbGRDb25maWcgfSBmcm9tICdAbmd4LWZvcm1seS9jb3JlJztcbmltcG9ydCB7IEN1c3RvbUZvcm1seUZpZWxkQ29uZmlnIH0gZnJvbSAnLi4vLi4vcHVibGljLWFwaSc7XG5cbkBJbmplY3RhYmxlKHsgcHJvdmlkZWRJbjogJ3Jvb3QnIH0pXG5leHBvcnQgY2xhc3MgU2NvcmluZ1NlcnZpY2Uge1xuXG4gIHVwZGF0ZVNjb3JlKG1vZGVsOiBhbnksIGZpZWxkczogRm9ybWx5RmllbGRDb25maWdbXSkge1xuICAgIGNvbnN0IHJlc3VsdCA9IHRoaXMuY2FsY3VsYXRlU2NvcmUobW9kZWwsIGZpZWxkcyk7XG4gICAgcmV0dXJuIHJlc3VsdDtcbiAgfVxuXG4gcHJpdmF0ZSBjYWxjdWxhdGVTY29yZShtb2RlbDogYW55LCBmaWVsZHM6IEZvcm1seUZpZWxkQ29uZmlnW10gfCBDdXN0b21Gb3JtbHlGaWVsZENvbmZpZ1tdKSB7XG4gIGxldCB0b3RhbFNjb3JlID0gMDtcbiAgbGV0IGNhdGVnb3J5UmVzdWx0czogUmVjb3JkPHN0cmluZywgeyBzY29yZTogbnVtYmVyOyB3ZWlnaHQ6IG51bWJlciB9PiA9IHt9O1xuXG4gIGNvbnN0IHByb2Nlc3NDYXRlZ29yeSA9IChjYXRlZ29yeTogRm9ybWx5RmllbGRDb25maWcgfCBDdXN0b21Gb3JtbHlGaWVsZENvbmZpZykgPT4ge1xuICAgIGNvbnN0IGNhdGVnb3J5TGFiZWwgPSBjYXRlZ29yeS5wcm9wcz8ubGFiZWw7XG4gICAgY29uc3QgY2F0ZWdvcnlXZWlnaHQgPSBjYXRlZ29yeS5wcm9wcz8ud2VpZ2h0ID8/IDA7XG5cbiAgICAvLyBPbmx5IHRyZWF0IGFzIHNjb3JpbmcgY2F0ZWdvcnkgaWYgaXQgaGFzIGEgbGFiZWwgKyB3ZWlnaHRcbiAgICBpZiAoIWNhdGVnb3J5TGFiZWwgfHwgIWNhdGVnb3J5V2VpZ2h0KSB7XG4gICAgICAvLyBTdGlsbCBuZWVkIHRvIGdvIGRlZXBlciBpbiBjYXNlIGNoaWxkcmVuIGhhdmUgY2F0ZWdvcmllc1xuICAgICAgaWYgKGNhdGVnb3J5LmZpZWxkR3JvdXA/Lmxlbmd0aCkge1xuICAgICAgICBjYXRlZ29yeS5maWVsZEdyb3VwLmZvckVhY2gocHJvY2Vzc0NhdGVnb3J5KTtcbiAgICAgIH1cbiAgICAgIHJldHVybjtcbiAgICB9XG5cbiAgICBjb25zdCBxdWVzdGlvbnMgPSAoY2F0ZWdvcnkuZmllbGRHcm91cCB8fCBbXSkuZmlsdGVyKHEgPT4gcS5wcm9wcz8uc2NvcmluZyk7XG4gICAgaWYgKHF1ZXN0aW9ucy5sZW5ndGggPT09IDApIHJldHVybjtcblxuICAgIGNvbnN0IHBlclF1ZXN0aW9uV2VpZ2h0ID0gY2F0ZWdvcnlXZWlnaHQgLyBxdWVzdGlvbnMubGVuZ3RoO1xuICAgIGxldCBjYXRlZ29yeVNjb3JlID0gMDtcblxuICAgIGZvciAoY29uc3QgcSBvZiBxdWVzdGlvbnMpIHtcbiAgICAgIGNvbnN0IHBhcmVudEtleSA9IGNhdGVnb3J5LmtleSBhcyBhbnk7XG4gICAgICBjb25zdCBuZXN0ZWRQYXRoID0gQXJyYXkuaXNBcnJheShwYXJlbnRLZXkpXG4gICAgICAgID8gWy4uLihwYXJlbnRLZXkgYXMgYW55W10pLCBxLmtleSBhcyBhbnldXG4gICAgICAgIDogW3BhcmVudEtleSBhcyBhbnksIHEua2V5IGFzIGFueV07XG4gICAgICBjb25zdCBhbnN3ZXIgPSB0aGlzLmdldE5lc3RlZFZhbHVlKG1vZGVsLCBuZXN0ZWRQYXRoIGFzIGFueSk7XG5cbiAgICAgIGNvbnN0IHNjb3JpbmcgPSBxLnByb3BzPy5zY29yaW5nO1xuICAgICAgaWYgKCFzY29yaW5nKSBjb250aW51ZTtcblxuICAgICAgbGV0IGNvcnJlY3QgPSBmYWxzZTtcblxuICAgICAgc3dpdGNoIChzY29yaW5nLmNyaXRlcmlhKSB7XG4gICAgICAgIGNhc2UgJ2F0dGVtcHRlZCc6XG4gICAgICAgICAgY29ycmVjdCA9XG4gICAgICAgICAgICBhbnN3ZXIgIT09IG51bGwgJiZcbiAgICAgICAgICAgIGFuc3dlciAhPT0gdW5kZWZpbmVkICYmXG4gICAgICAgICAgICAodHlwZW9mIGFuc3dlciA9PT0gJ3N0cmluZycgPyBhbnN3ZXIudHJpbSgpICE9PSAnJyA6IHRydWUpO1xuICAgICAgICAgIGJyZWFrO1xuXG4gICAgICAgIGNhc2UgJ2V4YWN0TWF0Y2gnOlxuICAgICAgICAgIGlmIChBcnJheS5pc0FycmF5KHNjb3JpbmcuYW5zd2VyKSkge1xuICAgICAgICAgICAgY29ycmVjdCA9IHNjb3JpbmcuYW5zd2VyLmluY2x1ZGVzKGFuc3dlcik7XG4gICAgICAgICAgfVxuICAgICAgICAgIGJyZWFrO1xuXG4gICAgICAgIGNhc2UgJ3JhbmdlJzpcbiAgICAgICAgICBjb25zdCBudW0gPSB0eXBlb2YgYW5zd2VyID09PSAnbnVtYmVyJyA/IGFuc3dlciA6IE51bWJlcihhbnN3ZXIpO1xuICAgICAgICAgIGNvcnJlY3QgPVxuICAgICAgICAgICAgTnVtYmVyLmlzRmluaXRlKG51bSkgJiZcbiAgICAgICAgICAgIG51bSA+PSBzY29yaW5nLmFuc3dlci5taW4gJiZcbiAgICAgICAgICAgIG51bSA8PSBzY29yaW5nLmFuc3dlci5tYXg7XG4gICAgICAgICAgYnJlYWs7XG4gICAgICB9XG5cbiAgICAgIGlmIChjb3JyZWN0KSB7XG4gICAgICAgIGNhdGVnb3J5U2NvcmUgKz0gcGVyUXVlc3Rpb25XZWlnaHQ7XG4gICAgICB9XG4gICAgfVxuXG4gICAgdG90YWxTY29yZSArPSBjYXRlZ29yeVNjb3JlO1xuICAgIGNhdGVnb3J5UmVzdWx0c1tjYXRlZ29yeUxhYmVsXSA9IHtcbiAgICAgIHNjb3JlOiBjYXRlZ29yeVNjb3JlLFxuICAgICAgd2VpZ2h0OiBjYXRlZ29yeVdlaWdodCxcbiAgICB9O1xuICB9O1xuXG4gIC8vIHN0YXJ0IHJlY3Vyc2l2ZSB3YWxrXG4gIGZpZWxkcy5mb3JFYWNoKHByb2Nlc3NDYXRlZ29yeSk7XG5cbiAgcmV0dXJuIHtcbiAgICB0b3RhbFNjb3JlLFxuICAgIGNhdGVnb3J5UmVzdWx0cyxcbiAgfTtcbn1cblxuXG4gIHByaXZhdGUgZ2V0TmVzdGVkVmFsdWUob2JqOiBhbnksIGtleTogc3RyaW5nIHwgbnVtYmVyIHwgKHN0cmluZyB8IG51bWJlcilbXSk6IGFueSB7XG4gICAgaWYgKEFycmF5LmlzQXJyYXkoa2V5KSkge1xuICAgICAgcmV0dXJuIGtleS5yZWR1Y2UoKGN1cnJlbnQsIGspID0+IGN1cnJlbnQ/LltrXSwgb2JqKTtcbiAgICB9XG4gICAgcmV0dXJuIG9iaj8uW2tleV07XG4gIH1cbn0iXX0=
|
|
@@ -17,14 +17,24 @@ import * as i4 from "@ngx-formly/core";
|
|
|
17
17
|
import * as i5 from "@angular/forms";
|
|
18
18
|
import * as i6 from "@angular/material/slide-toggle";
|
|
19
19
|
export class MnlFormHelperTextInputComponent extends FieldType {
|
|
20
|
+
get helperLabel() {
|
|
21
|
+
return this.to['helperProps']?.['helperLabel'] || 'Need help?';
|
|
22
|
+
}
|
|
23
|
+
get defaultHelperText() {
|
|
24
|
+
return this.to['helperProps']?.['defaultHelperText'] || '';
|
|
25
|
+
}
|
|
26
|
+
get currentValue() {
|
|
27
|
+
const fieldKey = this.field.key;
|
|
28
|
+
return this.field.model?.[fieldKey];
|
|
29
|
+
}
|
|
20
30
|
constructor(cdr) {
|
|
21
31
|
super();
|
|
22
32
|
this.cdr = cdr;
|
|
23
33
|
this.checked = false;
|
|
24
34
|
}
|
|
25
35
|
ngOnInit() {
|
|
36
|
+
this.originalValue = this.currentValue;
|
|
26
37
|
this.setupDefaultField();
|
|
27
|
-
// If mode is helper, set up the initial state based on scoring
|
|
28
38
|
if (this.to['enableHelper']) {
|
|
29
39
|
this.setCorrectValue();
|
|
30
40
|
this.determineInitialState();
|
|
@@ -32,6 +42,9 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
32
42
|
}
|
|
33
43
|
toggleSlide() {
|
|
34
44
|
this.checked = !this.checked;
|
|
45
|
+
if (this.checked) {
|
|
46
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
47
|
+
}
|
|
35
48
|
this.cdr.detectChanges();
|
|
36
49
|
}
|
|
37
50
|
setupDefaultField() {
|
|
@@ -65,12 +78,12 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
65
78
|
switch (this.to['scoring']?.['criteria']) {
|
|
66
79
|
case 'exactMatch':
|
|
67
80
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
68
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
81
|
+
!this.to['scoring']?.['answer'].includes(this.currentValue)) {
|
|
69
82
|
this.checked = true;
|
|
70
83
|
}
|
|
71
84
|
break;
|
|
72
85
|
case 'range':
|
|
73
|
-
const answer = this.
|
|
86
|
+
const answer = this.currentValue;
|
|
74
87
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
75
88
|
const correct = Number.isFinite(num) &&
|
|
76
89
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -84,26 +97,25 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
84
97
|
setCorrectValue() {
|
|
85
98
|
switch (this.to['scoring']?.['criteria']) {
|
|
86
99
|
case 'attempted':
|
|
87
|
-
this.formControl?.setValue(this.
|
|
100
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
88
101
|
break;
|
|
89
102
|
case 'exactMatch':
|
|
90
103
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
91
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
92
|
-
this.formControl?.setValue(this.
|
|
104
|
+
!this.to['scoring']?.['answer'].includes(this.originalValue)) {
|
|
105
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
93
106
|
}
|
|
94
107
|
break;
|
|
95
108
|
case 'range':
|
|
96
|
-
const answer = this.
|
|
109
|
+
const answer = this.currentValue;
|
|
97
110
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
98
111
|
const correct = Number.isFinite(num) &&
|
|
99
112
|
num >= this.to['scoring']?.['answer'].min &&
|
|
100
113
|
num <= this.to['scoring']?.['answer'].max;
|
|
101
114
|
if (!correct) {
|
|
102
|
-
this.formControl?.setValue(
|
|
115
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
103
116
|
}
|
|
104
117
|
break;
|
|
105
118
|
default:
|
|
106
|
-
this.formControl?.setValue(this.to['value']);
|
|
107
119
|
break;
|
|
108
120
|
}
|
|
109
121
|
}
|
|
@@ -113,12 +125,12 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
113
125
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
114
126
|
<div *ngIf="to['enableHelper']">
|
|
115
127
|
<mat-label>{{ to.label }}</mat-label>
|
|
116
|
-
<div *ngIf="
|
|
117
|
-
<span class="answer">
|
|
128
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
129
|
+
<span class="answer">{{ originalValue }}</span>
|
|
118
130
|
<mat-slide-toggle
|
|
119
131
|
[(ngModel)]="checked"
|
|
120
132
|
(toggleChange)="toggleSlide()"
|
|
121
|
-
>{{
|
|
133
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
122
134
|
>
|
|
123
135
|
</div>
|
|
124
136
|
|
|
@@ -159,12 +171,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
159
171
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
160
172
|
<div *ngIf="to['enableHelper']">
|
|
161
173
|
<mat-label>{{ to.label }}</mat-label>
|
|
162
|
-
<div *ngIf="
|
|
163
|
-
<span class="answer">
|
|
174
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
175
|
+
<span class="answer">{{ originalValue }}</span>
|
|
164
176
|
<mat-slide-toggle
|
|
165
177
|
[(ngModel)]="checked"
|
|
166
178
|
(toggleChange)="toggleSlide()"
|
|
167
|
-
>{{
|
|
179
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
168
180
|
>
|
|
169
181
|
</div>
|
|
170
182
|
|
|
@@ -187,4 +199,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
187
199
|
</div>
|
|
188
200
|
`, styles: [".helper-text-container{display:flex;flex-direction:column}.answer{font-size:12px}.textarea{margin:0!important;width:100%}.flex-row{display:flex;flex-direction:row;justify-content:space-between}.w-full{width:100%}\n"] }]
|
|
189
201
|
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
|
|
190
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mu-helper-text.type.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/types/mu-helper-text.type.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectorRef, Component, OnInit } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatNativeDateModule } from '@angular/material/core';\nimport { MatDatepickerModule } from '@angular/material/datepicker';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\nimport {\n  FieldTypeConfig,\n  FormlyFieldConfig,\n  FormlyModule,\n} from '@ngx-formly/core';\nimport { FieldType } from '@ngx-formly/material';\n\n@Component({\n  selector: 'mu-helper-text',\n  standalone: true,\n  imports: [\n    CommonModule,\n    MatInputModule,\n    FormlyModule,\n    ReactiveFormsModule,\n    FormsModule,\n    MatFormFieldModule,\n    MatSlideToggleModule,\n    MatSelectModule,\n    MatDatepickerModule,\n    MatNativeDateModule,\n  ],\n  template: `\n    <div class=\"helper-text-container\">\n      <!-- Helper Mode - Show toggle and conditional content -->\n      <div *ngIf=\"to['enableHelper']\">\n        <mat-label>{{ to.label }}</mat-label>\n        <div *ngIf=\"to['value']\" class=\"flex-row\">\n          <span class=\"answer\">Answer: {{ to['value'] }}</span>\n          <mat-slide-toggle\n            [(ngModel)]=\"checked\"\n            (toggleChange)=\"toggleSlide()\"\n            >{{ to['helperLabel'] }}</mat-slide-toggle\n          >\n        </div>\n\n        <mat-form-field class=\"textarea\" appearance=\"outline\" *ngIf=\"checked\">\n          <textarea\n            matInput\n            [formControl]=\"formControl\"\n            [formlyAttributes]=\"field\"\n            [readonly]=\"to['readonly']\"\n          >\n          </textarea>\n        </mat-form-field>\n      </div>\n\n      <!-- Default Mode - Show field directly -->\n      <div *ngIf=\"!to['enableHelper']\">\n        <formly-field *ngIf=\"defaultFieldConfig\" [field]=\"defaultFieldConfig\">\n        </formly-field>\n      </div>\n    </div>\n  `,\n  styles: `\n     .helper-text-container{\n      display: flex;\n      flex-direction: column;\n     }\n     .answer{\n      font-size: 12px;\n     }\n      .textarea{\n      margin: 0px !important;\n      width: 100%;\n     }\n     .flex-row{\n      display: flex;\n      flex-direction: row;\n      justify-content: space-between;\n     }\n     .w-full{\n      width: 100%;\n     }\n  `,\n})\nexport class MnlFormHelperTextInputComponent\n  extends FieldType<FieldTypeConfig>\n  implements OnInit\n{\n  checked = false;\n  defaultFieldConfig?: FormlyFieldConfig;\n\n  constructor(private cdr: ChangeDetectorRef) {\n    super();\n  }\n\n  ngOnInit() {\n    this.setupDefaultField();\n\n    // If mode is helper, set up the initial state based on scoring\n    if (this.to['enableHelper']) {\n        this.setCorrectValue();\n      this.determineInitialState();\n    }\n  }\n\n  toggleSlide() {\n    this.checked = !this.checked;\n    this.cdr.detectChanges();\n  }\n\n  setupDefaultField() {\n    const fieldConfig = this.to['defaultFieldConfig'];\n    if (fieldConfig) {\n      this.defaultFieldConfig = {\n        ...fieldConfig,\n        key: this.field.key || fieldConfig.key || 'defaultKey',\n        formControl: this.formControl,\n        // Add required properties that formly expects\n        modelOptions: fieldConfig.modelOptions || this.field.modelOptions || {},\n        validators: fieldConfig.validators || this.field.validators || {},\n        asyncValidators:\n          fieldConfig.asyncValidators || this.field.asyncValidators || {},\n        hooks: fieldConfig.hooks || this.field.hooks || {},\n        expressions: fieldConfig.expressions || this.field.expressions || {},\n        hide: fieldConfig.hide !== undefined ? fieldConfig.hide : false,\n        className: fieldConfig.className || this.field.className || '',\n        fieldGroupClassName:\n          fieldConfig.fieldGroupClassName ||\n          this.field.fieldGroupClassName ||\n          '',\n        // Inherit parent field's form and model\n        parent: this.field.parent,\n        options: this.field.options,\n        model: this.field.model,\n        form: this.field.form,\n      };\n    }\n  }\n\n  determineInitialState() {\n    // Auto-open helper based on scoring criteria\n    switch (this.to['scoring']?.['criteria']) {\n      case 'exactMatch':\n        if (\n          Array.isArray(this.to['scoring']?.['answer']) &&\n          !this.to['scoring']?.['answer'].includes(this.to['value'])\n        ) {\n          this.checked = true;\n        }\n        break;\n\n      case 'range':\n        const answer = this.to['value'];\n        const num = typeof answer === 'number' ? answer : Number(answer);\n\n        const correct =\n          Number.isFinite(num) &&\n          num >= this.to['scoring']?.['answer'].min &&\n          num <= this.to['scoring']?.['answer'].max;\n\n        if (!correct) {\n          this.checked = true;\n        }\n        break;\n    }\n  }\n\n  setCorrectValue() {\n    switch (this.to['scoring']?.['criteria']) {\n      case 'attempted':\n        this.formControl?.setValue(this.to['value']);\n        break;\n\n      case 'exactMatch':\n        if (\n          Array.isArray(this.to['scoring']?.['answer']) &&\n          !this.to['scoring']?.['answer'].includes(this.to['value'])\n        ) {\n          this.formControl?.setValue(this.to['scoring']['answer'].join(', '));\n        }\n        break;\n\n      case 'range':\n        const answer = this.to['value'];\n        const num = typeof answer === 'number' ? answer : Number(answer);\n\n        const correct =\n          Number.isFinite(num) &&\n          num >= this.to['scoring']?.['answer'].min &&\n          num <= this.to['scoring']?.['answer'].max;\n\n        if (!correct) {\n          this.formControl?.setValue(\n            `Value must be in range of ${this.to['scoring']?.['answer'].min} & ${this.to['scoring']?.['answer'].max}`\n          );\n        }\n        break;\n\n        default:\n            this.formControl?.setValue(this.to['value']);\n        break;\n    }\n  }\n}\n"]}
|
|
202
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"mu-helper-text.type.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/types/mu-helper-text.type.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectorRef, Component, OnInit } from '@angular/core';\nimport { FormsModule, ReactiveFormsModule } from '@angular/forms';\nimport { MatNativeDateModule } from '@angular/material/core';\nimport { MatDatepickerModule } from '@angular/material/datepicker';\nimport { MatFormFieldModule } from '@angular/material/form-field';\nimport { MatInputModule } from '@angular/material/input';\nimport { MatSelectModule } from '@angular/material/select';\nimport { MatSlideToggleModule } from '@angular/material/slide-toggle';\nimport {\n  FieldTypeConfig,\n  FormlyFieldConfig,\n  FormlyModule,\n} from '@ngx-formly/core';\nimport { FieldType } from '@ngx-formly/material';\n\n@Component({\n  selector: 'mu-helper-text',\n  standalone: true,\n  imports: [\n    CommonModule,\n    MatInputModule,\n    FormlyModule,\n    ReactiveFormsModule,\n    FormsModule,\n    MatFormFieldModule,\n    MatSlideToggleModule,\n    MatSelectModule,\n    MatDatepickerModule,\n    MatNativeDateModule,\n  ],\n  template: `\n    <div class=\"helper-text-container\">\n      <!-- Helper Mode - Show toggle and conditional content -->\n      <div *ngIf=\"to['enableHelper']\">\n        <mat-label>{{ to.label }}</mat-label>\n        <div *ngIf=\"originalValue\" class=\"flex-row\">\n          <span class=\"answer\">{{ originalValue }}</span>\n          <mat-slide-toggle\n            [(ngModel)]=\"checked\"\n            (toggleChange)=\"toggleSlide()\"\n            >{{ helperLabel }}</mat-slide-toggle\n          >\n        </div>\n\n        <mat-form-field class=\"textarea\" appearance=\"outline\" *ngIf=\"checked\">\n          <textarea\n            matInput\n            [formControl]=\"formControl\"\n            [formlyAttributes]=\"field\"\n            [readonly]=\"to['readonly']\"\n          >\n          </textarea>\n        </mat-form-field>\n      </div>\n\n      <!-- Default Mode - Show field directly -->\n      <div *ngIf=\"!to['enableHelper']\">\n        <formly-field *ngIf=\"defaultFieldConfig\" [field]=\"defaultFieldConfig\">\n        </formly-field>\n      </div>\n    </div>\n  `,\n  styles: `\n     .helper-text-container{\n      display: flex;\n      flex-direction: column;\n     }\n     .answer{\n      font-size: 12px;\n     }\n      .textarea{\n      margin: 0px !important;\n      width: 100%;\n     }\n     .flex-row{\n      display: flex;\n      flex-direction: row;\n      justify-content: space-between;\n     }\n     .w-full{\n      width: 100%;\n     }\n  `,\n})\nexport class MnlFormHelperTextInputComponent\n  extends FieldType<FieldTypeConfig>\n  implements OnInit\n{\n  checked = false;\n  get helperLabel() {\n    return this.to['helperProps']?.['helperLabel'] || 'Need help?';\n  }\n  get defaultHelperText() {\n    return this.to['helperProps']?.['defaultHelperText'] || '';\n  }\n  get currentValue() {\n    const fieldKey = this.field.key as string;\n    return this.field.model?.[fieldKey];\n  }\n  originalValue: any;\n  defaultFieldConfig?: FormlyFieldConfig;\n\n  constructor(private cdr: ChangeDetectorRef) {\n    super();\n  }\n\n  ngOnInit() {\n    this.originalValue = this.currentValue;    \n    this.setupDefaultField();\n    if (this.to['enableHelper']) {\n      this.setCorrectValue();\n      this.determineInitialState();\n    }\n  }\n\n  toggleSlide() {\n    this.checked = !this.checked;\n    if(this.checked) {\n      this.formControl?.setValue(this.defaultHelperText);\n    }\n    this.cdr.detectChanges();\n  }\n\n  setupDefaultField() {\n    const fieldConfig = this.to['defaultFieldConfig'];\n    if (fieldConfig) {\n      this.defaultFieldConfig = {\n        ...fieldConfig,\n        key: this.field.key || fieldConfig.key || 'defaultKey',\n        formControl: this.formControl,\n        // Add required properties that formly expects\n        modelOptions: fieldConfig.modelOptions || this.field.modelOptions || {},\n        validators: fieldConfig.validators || this.field.validators || {},\n        asyncValidators:\n          fieldConfig.asyncValidators || this.field.asyncValidators || {},\n        hooks: fieldConfig.hooks || this.field.hooks || {},\n        expressions: fieldConfig.expressions || this.field.expressions || {},\n        hide: fieldConfig.hide !== undefined ? fieldConfig.hide : false,\n        className: fieldConfig.className || this.field.className || '',\n        fieldGroupClassName:\n          fieldConfig.fieldGroupClassName ||\n          this.field.fieldGroupClassName ||\n          '',\n        // Inherit parent field's form and model\n        parent: this.field.parent,\n        options: this.field.options,\n        model: this.field.model,\n        form: this.field.form,\n      };\n    }\n  }\n\n  determineInitialState() {\n    // Auto-open helper based on scoring criteria\n    switch (this.to['scoring']?.['criteria']) {\n      case 'exactMatch':\n        if (\n          Array.isArray(this.to['scoring']?.['answer']) &&\n          !this.to['scoring']?.['answer'].includes(this.currentValue)\n        ) {\n          this.checked = true;\n        }\n        break;\n\n      case 'range':\n        const answer = this.currentValue;\n        const num = typeof answer === 'number' ? answer : Number(answer);\n\n        const correct =\n          Number.isFinite(num) &&\n          num >= this.to['scoring']?.['answer'].min &&\n          num <= this.to['scoring']?.['answer'].max;\n\n        if (!correct) {\n          this.checked = true;\n        }\n        break;\n    }\n  }\n\n  setCorrectValue() {\n    switch (this.to['scoring']?.['criteria']) {\n      case 'attempted':\n        this.formControl?.setValue(this.defaultHelperText);\n        break;\n\n      case 'exactMatch':\n        if (\n          Array.isArray(this.to['scoring']?.['answer']) &&\n          !this.to['scoring']?.['answer'].includes(this.originalValue)\n        ) {\n          this.formControl?.setValue(this.defaultHelperText);\n        }\n        break;\n\n      case 'range':\n        const answer = this.currentValue;\n        const num = typeof answer === 'number' ? answer : Number(answer);\n\n        const correct =\n          Number.isFinite(num) &&\n          num >= this.to['scoring']?.['answer'].min &&\n          num <= this.to['scoring']?.['answer'].max;\n\n        if (!correct) {\n          this.formControl?.setValue(this.defaultHelperText);\n        }\n        break;\n\n      default:\n        break;\n    }\n  }\n}\n"]}
|
|
@@ -146,7 +146,7 @@ export class FormlyFieldScrollableTabsComponent extends FieldType {
|
|
|
146
146
|
</div>
|
|
147
147
|
</div>
|
|
148
148
|
</div>
|
|
149
|
-
`, isInline: true, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100
|
|
149
|
+
`, isInline: true, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100%;.sticky-tabs{position:sticky;z-index:1}.tab-content{flex:1;overflow-y:auto;padding:16px;.tab-section{margin-bottom:45px;border-bottom:2px solid #00ae4d;h2{margin:0 0 8px}.spacer{height:400px}}}}\n"], dependencies: [{ kind: "ngmodule", type: MatTabsModule }, { kind: "directive", type: i1.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { kind: "component", type: i1.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i1.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor", "aria-label", "aria-labelledby"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i2.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i3.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i3.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i3.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i4.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i5.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i6.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] }); }
|
|
150
150
|
}
|
|
151
151
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyFieldScrollableTabsComponent, decorators: [{
|
|
152
152
|
type: Component,
|
|
@@ -209,7 +209,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
209
209
|
</div>
|
|
210
210
|
</div>
|
|
211
211
|
</div>
|
|
212
|
-
`, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100
|
|
212
|
+
`, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100%;.sticky-tabs{position:sticky;z-index:1}.tab-content{flex:1;overflow-y:auto;padding:16px;.tab-section{margin-bottom:45px;border-bottom:2px solid #00ae4d;h2{margin:0 0 8px}.spacer{height:400px}}}}\n"] }]
|
|
213
213
|
}], ctorParameters: () => [{ type: i0.NgZone }], propDecorators: { tabGroup: [{
|
|
214
214
|
type: ViewChild,
|
|
215
215
|
args: ['tabGroup']
|
|
@@ -217,4 +217,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
217
217
|
type: ViewChild,
|
|
218
218
|
args: ['tabContent', { static: false }]
|
|
219
219
|
}] } });
|
|
220
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable-tabs.types.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/types/scrollable-tabs.types.ts"],"names":[],"mappings":"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","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  ElementRef,\n  inject,\n  NgZone,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport {\n  MatTabChangeEvent,\n  MatTabGroup,\n  MatTabsModule,\n} from '@angular/material/tabs';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { Router } from '@angular/router';\nimport {\n  FieldType,\n  FieldTypeConfig,\n} from '@ngx-formly/core';\nimport { FormlyModule } from '@ngx-formly/core';\nimport { TranslateModule } from '@ngx-translate/core';\n\n@Component({\n  selector: 'mu-formly-field-tabs1',\n  standalone: true,\n  imports: [\n    MatTabsModule,\n    ReactiveFormsModule,\n    FormlyModule,\n    MatButtonModule,\n    CommonModule,\n    TranslateModule,\n    MatIconModule,\n    MatTooltipModule,\n  ],\n  template: `\n    <div class=\"tabs-container\">\n      <mat-tab-group\n        class=\"sticky-tabs\"\n        [(selectedIndex)]=\"selectedIndex\"\n        (selectedIndexChange)=\"onTabIndexChange($event)\"\n        (selectedTabChange)=\"onTabChange($event)\"\n      >\n        @for(tab of field.fieldGroup; track tab; let i = $index; let last =\n        $last){\n        <mat-tab>\n          <ng-template mat-tab-label>\n            <span\n              [ngStyle]=\"{\n                color:\n                  !isValid(field.fieldGroup ? field.fieldGroup[i] : {}) &&\n                  options.formState?.showValidationError\n                    ? 'red'\n                    : 'black'\n              }\"\n            >\n              {{ tab?.props?.label || '' }}\n            </span>\n            <mat-icon\n              matTooltip=\"{{ 'MANDATORY_FIELDS_TOOLTIP' | translate }}\"\n              *ngIf=\"\n                !isValid(field.fieldGroup ? field.fieldGroup[i] : {}) &&\n                options.formState?.showValidationError\n              \"\n              class=\"error-step\"\n              >error</mat-icon\n            >\n          </ng-template>\n        </mat-tab>\n        }\n      </mat-tab-group>\n\n      <div class=\"tab-content\" (scroll)=\"onScroll($event)\">\n        <div\n          *ngFor=\"let tab of field.fieldGroup; let i = index\"\n          [id]=\"'tab-' + i\"\n          class=\"tab-section\"\n        >\n          <h2 class=\"tab-title\" *ngIf=\"tab?.props?.label\">\n            {{ tab?.props?.label || '' }}\n          </h2>\n          <formly-field [field]=\"tab\"></formly-field>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .tab-title {\n        color: #00ae4d;\n      }\n      .error-step {\n        color: #d32f2f !important;\n      }\n      .tabs-container {\n        display: flex;\n        flex-direction: column;\n        height: 100%;\n\n        .sticky-tabs {\n          position: sticky;\n          z-index: 1;\n        }\n\n        .tab-content {\n          flex: 1;\n          overflow-y: auto;\n          padding: 16px;\n\n          .tab-section {\n            margin-bottom: 45px;\n            border-bottom: 2px solid #00ae4d;\n\n            h2 {\n              margin: 0 0 8px;\n            }\n\n            .spacer {\n              height: 400px;\n            }\n          }\n        }\n      }\n    `,\n  ],\n})\nexport class FormlyFieldScrollableTabsComponent\n  extends FieldType<FieldTypeConfig>\n  implements OnInit\n{\n  submittedData!: any[];\n\n  router = inject(Router);\n  user: any;\n  @ViewChild('tabGroup') tabGroup!: MatTabGroup;\n\n  selectedIndex = 0;\n  @ViewChild('tabContent', { static: false }) tabContent!: ElementRef;\n\n  private debounceTimeout: any;\n\n  constructor(private ngZone: NgZone) {\n    super();\n  }\n\n  nextTab(tabGroup: MatTabGroup) {\n    if (tabGroup && tabGroup.selectedIndex != null) {\n      const currentIndex = tabGroup.selectedIndex;\n      if (currentIndex < (tabGroup._tabs?.length || 0) - 1) {\n        tabGroup.selectedIndex = currentIndex + 1;\n      }\n    }\n  }\n\n  previousTab(tabGroup: MatTabGroup) {\n    if (tabGroup && tabGroup.selectedIndex != null) {\n      const currentIndex = tabGroup.selectedIndex;\n      if (currentIndex > 0) {\n        tabGroup.selectedIndex = currentIndex - 1;\n      }\n    }\n  }\n\n  isFirstTab(tabGroup: MatTabGroup): boolean {\n    return tabGroup && tabGroup.selectedIndex === 0;\n  }\n\n  isLastTab(tabGroup: MatTabGroup): boolean {\n    return (\n      tabGroup && tabGroup.selectedIndex === (tabGroup._tabs?.length || 0) - 1\n    );\n  }\n\n  ngOnInit(): void {\n    this.user = JSON.parse(localStorage.getItem('loggedInUser') || '{}');\n  }\n\n  isValid(field: any): boolean {\n    if (field.key) {\n      if (field.props?.excludeValidation) return true;\n      return field?.formControl?.valid || false;\n    }\n\n    return field.fieldGroup\n      ? field.fieldGroup.every((f: any) => this.isValid(f))\n      : true;\n  }\n\n  onTabChange(event: MatTabChangeEvent): void {\n    this.selectedIndex = event.index;\n    console.log('Selected tab index:', this.selectedIndex);\n    if (this.field && this.field.props?.['tabChange']) {\n      this.field.props?.['tabChange'](event, this.field);\n    }\n  }\n\n  onTabIndexChange(index: number) {\n    const targetElement = document.getElementById(`tab-${index}`);\n    if (targetElement) {\n      targetElement.scrollIntoView({ behavior: 'smooth', block: 'start' });\n    }\n  }\n\n  onScroll(event: Event): void {\n    clearTimeout(this.debounceTimeout);\n\n    this.debounceTimeout = setTimeout(() => {\n      const scrollContainer = event.target as HTMLElement;\n      const scrollTop = scrollContainer.scrollTop;\n\n      this.ngZone.run(() => {\n        if (this.field.fieldGroup) {\n          for (let index = 0; index < this.field.fieldGroup?.length; index++) {\n            const tabElement = document.getElementById(`tab-${index}`);\n            if (tabElement) {\n              const tabTop = tabElement.offsetTop - scrollContainer.offsetTop;\n\n              const tabHeight = tabElement.offsetHeight;\n\n              if (scrollTop >= tabTop && scrollTop < tabTop + tabHeight) {\n                if (this.selectedIndex !== index) {\n                  this.selectedIndex = index;\n                }\n                break;\n              }\n            }\n          }\n        }\n      });\n    }, 100);\n  }\n}\n"]}
|
|
220
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scrollable-tabs.types.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/types/scrollable-tabs.types.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EACL,SAAS,EAET,MAAM,EAGN,SAAS,GACV,MAAM,eAAe,CAAC;AACvB,OAAO,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AACrD,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,aAAa,EAAE,MAAM,wBAAwB,CAAC;AACvD,OAAO,EAGL,aAAa,GACd,MAAM,wBAAwB,CAAC;AAChC,OAAO,EAAE,gBAAgB,EAAE,MAAM,2BAA2B,CAAC;AAC7D,OAAO,EAAE,MAAM,EAAE,MAAM,iBAAiB,CAAC;AACzC,OAAO,EACL,SAAS,GAEV,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,YAAY,EAAE,MAAM,kBAAkB,CAAC;AAChD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;;;;;;;;AA0GtD,MAAM,OAAO,kCACX,SAAQ,SAA0B;IAclC,YAAoB,MAAc;QAChC,KAAK,EAAE,CAAC;QADU,WAAM,GAAN,MAAM,CAAQ;QATlC,WAAM,GAAG,MAAM,CAAC,MAAM,CAAC,CAAC;QAIxB,kBAAa,GAAG,CAAC,CAAC;IAOlB,CAAC;IAED,OAAO,CAAC,QAAqB;QAC3B,IAAI,QAAQ,IAAI,QAAQ,CAAC,aAAa,IAAI,IAAI,EAAE,CAAC;YAC/C,MAAM,YAAY,GAAG,QAAQ,CAAC,aAAa,CAAC;YAC5C,IAAI,YAAY,GAAG,CAAC,QAAQ,CAAC,KAAK,EAAE,MAAM,IAAI,CAAC,CAAC,GAAG,CAAC,EAAE,CAAC;gBACrD,QAAQ,CAAC,aAAa,GAAG,YAAY,GAAG,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC;IACH,CAAC;IAED,WAAW,CAAC,QAAqB;QAC/B,IAAI,QAAQ,IAAI,QAAQ,CAAC,aAAa,IAAI,IAAI,EAAE,CAAC;YAC/C,MAAM,YAAY,GAAG,QAAQ,CAAC,aAAa,CAAC;YAC5C,IAAI,YAAY,GAAG,CAAC,EAAE,CAAC;gBACrB,QAAQ,CAAC,aAAa,GAAG,YAAY,GAAG,CAAC,CAAC;YAC5C,CAAC;QACH,CAAC;IACH,CAAC;IAED,UAAU,CAAC,QAAqB;QAC9B,OAAO,QAAQ,IAAI,QAAQ,CAAC,aAAa,KAAK,CAAC,CAAC;IAClD,CAAC;IAED,SAAS,CAAC,QAAqB;QAC7B,OAAO,CACL,QAAQ,IAAI,QAAQ,CAAC,aAAa,KAAK,CAAC,QAAQ,CAAC,KAAK,EAAE,MAAM,IAAI,CAAC,CAAC,GAAG,CAAC,CACzE,CAAC;IACJ,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,IAAI,GAAG,IAAI,CAAC,KAAK,CAAC,YAAY,CAAC,OAAO,CAAC,cAAc,CAAC,IAAI,IAAI,CAAC,CAAC;IACvE,CAAC;IAED,OAAO,CAAC,KAAU;QAChB,IAAI,KAAK,CAAC,GAAG,EAAE,CAAC;YACd,IAAI,KAAK,CAAC,KAAK,EAAE,iBAAiB;gBAAE,OAAO,IAAI,CAAC;YAChD,OAAO,KAAK,EAAE,WAAW,EAAE,KAAK,IAAI,KAAK,CAAC;QAC5C,CAAC;QAED,OAAO,KAAK,CAAC,UAAU;YACrB,CAAC,CAAC,KAAK,CAAC,UAAU,CAAC,KAAK,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;YACrD,CAAC,CAAC,IAAI,CAAC;IACX,CAAC;IAED,WAAW,CAAC,KAAwB;QAClC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC,KAAK,CAAC;QACjC,OAAO,CAAC,GAAG,CAAC,qBAAqB,EAAE,IAAI,CAAC,aAAa,CAAC,CAAC;QACvD,IAAI,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,WAAW,CAAC,EAAE,CAAC;YAClD,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,WAAW,CAAC,CAAC,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,CAAC;QACrD,CAAC;IACH,CAAC;IAED,gBAAgB,CAAC,KAAa;QAC5B,MAAM,aAAa,GAAG,QAAQ,CAAC,cAAc,CAAC,OAAO,KAAK,EAAE,CAAC,CAAC;QAC9D,IAAI,aAAa,EAAE,CAAC;YAClB,aAAa,CAAC,cAAc,CAAC,EAAE,QAAQ,EAAE,QAAQ,EAAE,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;QACvE,CAAC;IACH,CAAC;IAED,QAAQ,CAAC,KAAY;QACnB,YAAY,CAAC,IAAI,CAAC,eAAe,CAAC,CAAC;QAEnC,IAAI,CAAC,eAAe,GAAG,UAAU,CAAC,GAAG,EAAE;YACrC,MAAM,eAAe,GAAG,KAAK,CAAC,MAAqB,CAAC;YACpD,MAAM,SAAS,GAAG,eAAe,CAAC,SAAS,CAAC;YAE5C,IAAI,CAAC,MAAM,CAAC,GAAG,CAAC,GAAG,EAAE;gBACnB,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE,CAAC;oBAC1B,KAAK,IAAI,KAAK,GAAG,CAAC,EAAE,KAAK,GAAG,IAAI,CAAC,KAAK,CAAC,UAAU,EAAE,MAAM,EAAE,KAAK,EAAE,EAAE,CAAC;wBACnE,MAAM,UAAU,GAAG,QAAQ,CAAC,cAAc,CAAC,OAAO,KAAK,EAAE,CAAC,CAAC;wBAC3D,IAAI,UAAU,EAAE,CAAC;4BACf,MAAM,MAAM,GAAG,UAAU,CAAC,SAAS,GAAG,eAAe,CAAC,SAAS,CAAC;4BAEhE,MAAM,SAAS,GAAG,UAAU,CAAC,YAAY,CAAC;4BAE1C,IAAI,SAAS,IAAI,MAAM,IAAI,SAAS,GAAG,MAAM,GAAG,SAAS,EAAE,CAAC;gCAC1D,IAAI,IAAI,CAAC,aAAa,KAAK,KAAK,EAAE,CAAC;oCACjC,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;gCAC7B,CAAC;gCACD,MAAM;4BACR,CAAC;wBACH,CAAC;oBACH,CAAC;gBACH,CAAC;YACH,CAAC,CAAC,CAAC;QACL,CAAC,EAAE,GAAG,CAAC,CAAC;IACV,CAAC;+GAxGU,kCAAkC;mGAAlC,kCAAkC,ySA3FnC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkDT,yYA3DC,aAAa,quBACb,mBAAmB,8BACnB,YAAY,wHACZ,eAAe,8BACf,YAAY,oVACZ,eAAe,2FACf,aAAa,mLACb,gBAAgB;;4FA6FP,kCAAkC;kBAxG9C,SAAS;+BACE,uBAAuB,cACrB,IAAI,WACP;wBACP,aAAa;wBACb,mBAAmB;wBACnB,YAAY;wBACZ,eAAe;wBACf,YAAY;wBACZ,eAAe;wBACf,aAAa;wBACb,gBAAgB;qBACjB,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAkDT;2EAiDsB,QAAQ;sBAA9B,SAAS;uBAAC,UAAU;gBAGuB,UAAU;sBAArD,SAAS;uBAAC,YAAY,EAAE,EAAE,MAAM,EAAE,KAAK,EAAE","sourcesContent":["import { CommonModule } from '@angular/common';\nimport {\n  Component,\n  ElementRef,\n  inject,\n  NgZone,\n  OnInit,\n  ViewChild,\n} from '@angular/core';\nimport { ReactiveFormsModule } from '@angular/forms';\nimport { MatButtonModule } from '@angular/material/button';\nimport { MatIconModule } from '@angular/material/icon';\nimport {\n  MatTabChangeEvent,\n  MatTabGroup,\n  MatTabsModule,\n} from '@angular/material/tabs';\nimport { MatTooltipModule } from '@angular/material/tooltip';\nimport { Router } from '@angular/router';\nimport {\n  FieldType,\n  FieldTypeConfig,\n} from '@ngx-formly/core';\nimport { FormlyModule } from '@ngx-formly/core';\nimport { TranslateModule } from '@ngx-translate/core';\n\n@Component({\n  selector: 'mu-formly-field-tabs1',\n  standalone: true,\n  imports: [\n    MatTabsModule,\n    ReactiveFormsModule,\n    FormlyModule,\n    MatButtonModule,\n    CommonModule,\n    TranslateModule,\n    MatIconModule,\n    MatTooltipModule,\n  ],\n  template: `\n    <div class=\"tabs-container\">\n      <mat-tab-group\n        class=\"sticky-tabs\"\n        [(selectedIndex)]=\"selectedIndex\"\n        (selectedIndexChange)=\"onTabIndexChange($event)\"\n        (selectedTabChange)=\"onTabChange($event)\"\n      >\n        @for(tab of field.fieldGroup; track tab; let i = $index; let last =\n        $last){\n        <mat-tab>\n          <ng-template mat-tab-label>\n            <span\n              [ngStyle]=\"{\n                color:\n                  !isValid(field.fieldGroup ? field.fieldGroup[i] : {}) &&\n                  options.formState?.showValidationError\n                    ? 'red'\n                    : 'black'\n              }\"\n            >\n              {{ tab?.props?.label || '' }}\n            </span>\n            <mat-icon\n              matTooltip=\"{{ 'MANDATORY_FIELDS_TOOLTIP' | translate }}\"\n              *ngIf=\"\n                !isValid(field.fieldGroup ? field.fieldGroup[i] : {}) &&\n                options.formState?.showValidationError\n              \"\n              class=\"error-step\"\n              >error</mat-icon\n            >\n          </ng-template>\n        </mat-tab>\n        }\n      </mat-tab-group>\n\n      <div class=\"tab-content\" (scroll)=\"onScroll($event)\">\n        <div\n          *ngFor=\"let tab of field.fieldGroup; let i = index\"\n          [id]=\"'tab-' + i\"\n          class=\"tab-section\"\n        >\n          <h2 class=\"tab-title\" *ngIf=\"tab?.props?.label\">\n            {{ tab?.props?.label || '' }}\n          </h2>\n          <formly-field [field]=\"tab\"></formly-field>\n        </div>\n      </div>\n    </div>\n  `,\n  styles: [\n    `\n      .tab-title {\n        color: #00ae4d;\n      }\n      .error-step {\n        color: #d32f2f !important;\n      }\n      .tabs-container {\n        display: flex;\n        flex-direction: column;\n        height: 100%;\n\n        .sticky-tabs {\n          position: sticky;\n          z-index: 1;\n        }\n\n        .tab-content {\n          flex: 1;\n          overflow-y: auto;\n          padding: 16px;\n\n          .tab-section {\n            margin-bottom: 45px;\n            border-bottom: 2px solid #00ae4d;\n\n            h2 {\n              margin: 0 0 8px;\n            }\n\n            .spacer {\n              height: 400px;\n            }\n          }\n        }\n      }\n    `,\n  ],\n})\nexport class FormlyFieldScrollableTabsComponent\n  extends FieldType<FieldTypeConfig>\n  implements OnInit\n{\n  submittedData!: any[];\n\n  router = inject(Router);\n  user: any;\n  @ViewChild('tabGroup') tabGroup!: MatTabGroup;\n\n  selectedIndex = 0;\n  @ViewChild('tabContent', { static: false }) tabContent!: ElementRef;\n\n  private debounceTimeout: any;\n\n  constructor(private ngZone: NgZone) {\n    super();\n  }\n\n  nextTab(tabGroup: MatTabGroup) {\n    if (tabGroup && tabGroup.selectedIndex != null) {\n      const currentIndex = tabGroup.selectedIndex;\n      if (currentIndex < (tabGroup._tabs?.length || 0) - 1) {\n        tabGroup.selectedIndex = currentIndex + 1;\n      }\n    }\n  }\n\n  previousTab(tabGroup: MatTabGroup) {\n    if (tabGroup && tabGroup.selectedIndex != null) {\n      const currentIndex = tabGroup.selectedIndex;\n      if (currentIndex > 0) {\n        tabGroup.selectedIndex = currentIndex - 1;\n      }\n    }\n  }\n\n  isFirstTab(tabGroup: MatTabGroup): boolean {\n    return tabGroup && tabGroup.selectedIndex === 0;\n  }\n\n  isLastTab(tabGroup: MatTabGroup): boolean {\n    return (\n      tabGroup && tabGroup.selectedIndex === (tabGroup._tabs?.length || 0) - 1\n    );\n  }\n\n  ngOnInit(): void {\n    this.user = JSON.parse(localStorage.getItem('loggedInUser') || '{}');\n  }\n\n  isValid(field: any): boolean {\n    if (field.key) {\n      if (field.props?.excludeValidation) return true;\n      return field?.formControl?.valid || false;\n    }\n\n    return field.fieldGroup\n      ? field.fieldGroup.every((f: any) => this.isValid(f))\n      : true;\n  }\n\n  onTabChange(event: MatTabChangeEvent): void {\n    this.selectedIndex = event.index;\n    console.log('Selected tab index:', this.selectedIndex);\n    if (this.field && this.field.props?.['tabChange']) {\n      this.field.props?.['tabChange'](event, this.field);\n    }\n  }\n\n  onTabIndexChange(index: number) {\n    const targetElement = document.getElementById(`tab-${index}`);\n    if (targetElement) {\n      targetElement.scrollIntoView({ behavior: 'smooth', block: 'start' });\n    }\n  }\n\n  onScroll(event: Event): void {\n    clearTimeout(this.debounceTimeout);\n\n    this.debounceTimeout = setTimeout(() => {\n      const scrollContainer = event.target as HTMLElement;\n      const scrollTop = scrollContainer.scrollTop;\n\n      this.ngZone.run(() => {\n        if (this.field.fieldGroup) {\n          for (let index = 0; index < this.field.fieldGroup?.length; index++) {\n            const tabElement = document.getElementById(`tab-${index}`);\n            if (tabElement) {\n              const tabTop = tabElement.offsetTop - scrollContainer.offsetTop;\n\n              const tabHeight = tabElement.offsetHeight;\n\n              if (scrollTop >= tabTop && scrollTop < tabTop + tabHeight) {\n                if (this.selectedIndex !== index) {\n                  this.selectedIndex = index;\n                }\n                break;\n              }\n            }\n          }\n        }\n      });\n    }, 100);\n  }\n}\n"]}
|
|
@@ -4193,7 +4193,7 @@ class FormlyFieldScrollableTabsComponent extends FieldType {
|
|
|
4193
4193
|
</div>
|
|
4194
4194
|
</div>
|
|
4195
4195
|
</div>
|
|
4196
|
-
`, isInline: true, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100
|
|
4196
|
+
`, isInline: true, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100%;.sticky-tabs{position:sticky;z-index:1}.tab-content{flex:1;overflow-y:auto;padding:16px;.tab-section{margin-bottom:45px;border-bottom:2px solid #00ae4d;h2{margin:0 0 8px}.spacer{height:400px}}}}\n"], dependencies: [{ kind: "ngmodule", type: MatTabsModule }, { kind: "directive", type: i1$1.MatTabLabel, selector: "[mat-tab-label], [matTabLabel]" }, { kind: "component", type: i1$1.MatTab, selector: "mat-tab", inputs: ["disabled", "label", "aria-label", "aria-labelledby", "labelClass", "bodyClass"], exportAs: ["matTab"] }, { kind: "component", type: i1$1.MatTabGroup, selector: "mat-tab-group", inputs: ["color", "fitInkBarToContent", "mat-stretch-tabs", "dynamicHeight", "selectedIndex", "headerPosition", "animationDuration", "contentTabIndex", "disablePagination", "disableRipple", "preserveContent", "backgroundColor", "aria-label", "aria-labelledby"], outputs: ["selectedIndexChange", "focusChange", "animationDone", "selectedTabChange"], exportAs: ["matTabGroup"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i2$1.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: MatButtonModule }, { kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgForOf, selector: "[ngFor][ngForOf]", inputs: ["ngForOf", "ngForTrackBy", "ngForTemplate"] }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "directive", type: i1.NgStyle, selector: "[ngStyle]", inputs: ["ngStyle"] }, { kind: "ngmodule", type: TranslateModule }, { kind: "pipe", type: i1$2.TranslatePipe, name: "translate" }, { kind: "ngmodule", type: MatIconModule }, { kind: "component", type: i2$2.MatIcon, selector: "mat-icon", inputs: ["color", "inline", "svgIcon", "fontSet", "fontIcon"], exportAs: ["matIcon"] }, { kind: "ngmodule", type: MatTooltipModule }, { kind: "directive", type: i6$1.MatTooltip, selector: "[matTooltip]", inputs: ["matTooltipPosition", "matTooltipPositionAtOrigin", "matTooltipDisabled", "matTooltipShowDelay", "matTooltipHideDelay", "matTooltipTouchGestures", "matTooltip", "matTooltipClass"], exportAs: ["matTooltip"] }] }); }
|
|
4197
4197
|
}
|
|
4198
4198
|
i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: FormlyFieldScrollableTabsComponent, decorators: [{
|
|
4199
4199
|
type: Component,
|
|
@@ -4256,7 +4256,7 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
4256
4256
|
</div>
|
|
4257
4257
|
</div>
|
|
4258
4258
|
</div>
|
|
4259
|
-
`, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100
|
|
4259
|
+
`, styles: [".tab-title{color:#00ae4d}.error-step{color:#d32f2f!important}.tabs-container{display:flex;flex-direction:column;height:100%;.sticky-tabs{position:sticky;z-index:1}.tab-content{flex:1;overflow-y:auto;padding:16px;.tab-section{margin-bottom:45px;border-bottom:2px solid #00ae4d;h2{margin:0 0 8px}.spacer{height:400px}}}}\n"] }]
|
|
4260
4260
|
}], ctorParameters: () => [{ type: i0.NgZone }], propDecorators: { tabGroup: [{
|
|
4261
4261
|
type: ViewChild,
|
|
4262
4262
|
args: ['tabGroup']
|
|
@@ -4518,14 +4518,24 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
4518
4518
|
}] });
|
|
4519
4519
|
|
|
4520
4520
|
class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
4521
|
+
get helperLabel() {
|
|
4522
|
+
return this.to['helperProps']?.['helperLabel'] || 'Need help?';
|
|
4523
|
+
}
|
|
4524
|
+
get defaultHelperText() {
|
|
4525
|
+
return this.to['helperProps']?.['defaultHelperText'] || '';
|
|
4526
|
+
}
|
|
4527
|
+
get currentValue() {
|
|
4528
|
+
const fieldKey = this.field.key;
|
|
4529
|
+
return this.field.model?.[fieldKey];
|
|
4530
|
+
}
|
|
4521
4531
|
constructor(cdr) {
|
|
4522
4532
|
super();
|
|
4523
4533
|
this.cdr = cdr;
|
|
4524
4534
|
this.checked = false;
|
|
4525
4535
|
}
|
|
4526
4536
|
ngOnInit() {
|
|
4537
|
+
this.originalValue = this.currentValue;
|
|
4527
4538
|
this.setupDefaultField();
|
|
4528
|
-
// If mode is helper, set up the initial state based on scoring
|
|
4529
4539
|
if (this.to['enableHelper']) {
|
|
4530
4540
|
this.setCorrectValue();
|
|
4531
4541
|
this.determineInitialState();
|
|
@@ -4533,6 +4543,9 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4533
4543
|
}
|
|
4534
4544
|
toggleSlide() {
|
|
4535
4545
|
this.checked = !this.checked;
|
|
4546
|
+
if (this.checked) {
|
|
4547
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
4548
|
+
}
|
|
4536
4549
|
this.cdr.detectChanges();
|
|
4537
4550
|
}
|
|
4538
4551
|
setupDefaultField() {
|
|
@@ -4566,12 +4579,12 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4566
4579
|
switch (this.to['scoring']?.['criteria']) {
|
|
4567
4580
|
case 'exactMatch':
|
|
4568
4581
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
4569
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
4582
|
+
!this.to['scoring']?.['answer'].includes(this.currentValue)) {
|
|
4570
4583
|
this.checked = true;
|
|
4571
4584
|
}
|
|
4572
4585
|
break;
|
|
4573
4586
|
case 'range':
|
|
4574
|
-
const answer = this.
|
|
4587
|
+
const answer = this.currentValue;
|
|
4575
4588
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
4576
4589
|
const correct = Number.isFinite(num) &&
|
|
4577
4590
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -4585,26 +4598,25 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4585
4598
|
setCorrectValue() {
|
|
4586
4599
|
switch (this.to['scoring']?.['criteria']) {
|
|
4587
4600
|
case 'attempted':
|
|
4588
|
-
this.formControl?.setValue(this.
|
|
4601
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
4589
4602
|
break;
|
|
4590
4603
|
case 'exactMatch':
|
|
4591
4604
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
4592
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
4593
|
-
this.formControl?.setValue(this.
|
|
4605
|
+
!this.to['scoring']?.['answer'].includes(this.originalValue)) {
|
|
4606
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
4594
4607
|
}
|
|
4595
4608
|
break;
|
|
4596
4609
|
case 'range':
|
|
4597
|
-
const answer = this.
|
|
4610
|
+
const answer = this.currentValue;
|
|
4598
4611
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
4599
4612
|
const correct = Number.isFinite(num) &&
|
|
4600
4613
|
num >= this.to['scoring']?.['answer'].min &&
|
|
4601
4614
|
num <= this.to['scoring']?.['answer'].max;
|
|
4602
4615
|
if (!correct) {
|
|
4603
|
-
this.formControl?.setValue(
|
|
4616
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
4604
4617
|
}
|
|
4605
4618
|
break;
|
|
4606
4619
|
default:
|
|
4607
|
-
this.formControl?.setValue(this.to['value']);
|
|
4608
4620
|
break;
|
|
4609
4621
|
}
|
|
4610
4622
|
}
|
|
@@ -4614,12 +4626,12 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4614
4626
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
4615
4627
|
<div *ngIf="to['enableHelper']">
|
|
4616
4628
|
<mat-label>{{ to.label }}</mat-label>
|
|
4617
|
-
<div *ngIf="
|
|
4618
|
-
<span class="answer">
|
|
4629
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
4630
|
+
<span class="answer">{{ originalValue }}</span>
|
|
4619
4631
|
<mat-slide-toggle
|
|
4620
4632
|
[(ngModel)]="checked"
|
|
4621
4633
|
(toggleChange)="toggleSlide()"
|
|
4622
|
-
>{{
|
|
4634
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
4623
4635
|
>
|
|
4624
4636
|
</div>
|
|
4625
4637
|
|
|
@@ -4660,12 +4672,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
4660
4672
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
4661
4673
|
<div *ngIf="to['enableHelper']">
|
|
4662
4674
|
<mat-label>{{ to.label }}</mat-label>
|
|
4663
|
-
<div *ngIf="
|
|
4664
|
-
<span class="answer">
|
|
4675
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
4676
|
+
<span class="answer">{{ originalValue }}</span>
|
|
4665
4677
|
<mat-slide-toggle
|
|
4666
4678
|
[(ngModel)]="checked"
|
|
4667
4679
|
(toggleChange)="toggleSlide()"
|
|
4668
|
-
>{{
|
|
4680
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
4669
4681
|
>
|
|
4670
4682
|
</div>
|
|
4671
4683
|
|
|
@@ -4805,11 +4817,21 @@ class ScoringService {
|
|
|
4805
4817
|
calculateScore(model, fields) {
|
|
4806
4818
|
let totalScore = 0;
|
|
4807
4819
|
let categoryResults = {};
|
|
4808
|
-
|
|
4809
|
-
const categoryLabel = category.props?.label
|
|
4810
|
-
const categoryWeight = category.props?.weight
|
|
4811
|
-
|
|
4812
|
-
|
|
4820
|
+
const processCategory = (category) => {
|
|
4821
|
+
const categoryLabel = category.props?.label;
|
|
4822
|
+
const categoryWeight = category.props?.weight ?? 0;
|
|
4823
|
+
// Only treat as scoring category if it has a label + weight
|
|
4824
|
+
if (!categoryLabel || !categoryWeight) {
|
|
4825
|
+
// Still need to go deeper in case children have categories
|
|
4826
|
+
if (category.fieldGroup?.length) {
|
|
4827
|
+
category.fieldGroup.forEach(processCategory);
|
|
4828
|
+
}
|
|
4829
|
+
return;
|
|
4830
|
+
}
|
|
4831
|
+
const questions = (category.fieldGroup || []).filter(q => q.props?.scoring);
|
|
4832
|
+
if (questions.length === 0)
|
|
4833
|
+
return;
|
|
4834
|
+
const perQuestionWeight = categoryWeight / questions.length;
|
|
4813
4835
|
let categoryScore = 0;
|
|
4814
4836
|
for (const q of questions) {
|
|
4815
4837
|
const parentKey = category.key;
|
|
@@ -4850,7 +4872,9 @@ class ScoringService {
|
|
|
4850
4872
|
score: categoryScore,
|
|
4851
4873
|
weight: categoryWeight,
|
|
4852
4874
|
};
|
|
4853
|
-
}
|
|
4875
|
+
};
|
|
4876
|
+
// start recursive walk
|
|
4877
|
+
fields.forEach(processCategory);
|
|
4854
4878
|
return {
|
|
4855
4879
|
totalScore,
|
|
4856
4880
|
categoryResults,
|