@muraai/mnl-form 0.0.1-alpha-2edbad0 → 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.
|
@@ -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=
|
|
@@ -23,14 +23,18 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
23
23
|
get defaultHelperText() {
|
|
24
24
|
return this.to['helperProps']?.['defaultHelperText'] || '';
|
|
25
25
|
}
|
|
26
|
+
get currentValue() {
|
|
27
|
+
const fieldKey = this.field.key;
|
|
28
|
+
return this.field.model?.[fieldKey];
|
|
29
|
+
}
|
|
26
30
|
constructor(cdr) {
|
|
27
31
|
super();
|
|
28
32
|
this.cdr = cdr;
|
|
29
33
|
this.checked = false;
|
|
30
34
|
}
|
|
31
35
|
ngOnInit() {
|
|
36
|
+
this.originalValue = this.currentValue;
|
|
32
37
|
this.setupDefaultField();
|
|
33
|
-
// If mode is helper, set up the initial state based on scoring
|
|
34
38
|
if (this.to['enableHelper']) {
|
|
35
39
|
this.setCorrectValue();
|
|
36
40
|
this.determineInitialState();
|
|
@@ -38,6 +42,9 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
38
42
|
}
|
|
39
43
|
toggleSlide() {
|
|
40
44
|
this.checked = !this.checked;
|
|
45
|
+
if (this.checked) {
|
|
46
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
47
|
+
}
|
|
41
48
|
this.cdr.detectChanges();
|
|
42
49
|
}
|
|
43
50
|
setupDefaultField() {
|
|
@@ -71,12 +78,12 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
71
78
|
switch (this.to['scoring']?.['criteria']) {
|
|
72
79
|
case 'exactMatch':
|
|
73
80
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
74
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
81
|
+
!this.to['scoring']?.['answer'].includes(this.currentValue)) {
|
|
75
82
|
this.checked = true;
|
|
76
83
|
}
|
|
77
84
|
break;
|
|
78
85
|
case 'range':
|
|
79
|
-
const answer = this.
|
|
86
|
+
const answer = this.currentValue;
|
|
80
87
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
81
88
|
const correct = Number.isFinite(num) &&
|
|
82
89
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -94,12 +101,12 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
94
101
|
break;
|
|
95
102
|
case 'exactMatch':
|
|
96
103
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
97
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
104
|
+
!this.to['scoring']?.['answer'].includes(this.originalValue)) {
|
|
98
105
|
this.formControl?.setValue(this.defaultHelperText);
|
|
99
106
|
}
|
|
100
107
|
break;
|
|
101
108
|
case 'range':
|
|
102
|
-
const answer = this.
|
|
109
|
+
const answer = this.currentValue;
|
|
103
110
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
104
111
|
const correct = Number.isFinite(num) &&
|
|
105
112
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -109,7 +116,6 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
109
116
|
}
|
|
110
117
|
break;
|
|
111
118
|
default:
|
|
112
|
-
this.formControl?.setValue(this.defaultHelperText);
|
|
113
119
|
break;
|
|
114
120
|
}
|
|
115
121
|
}
|
|
@@ -119,12 +125,12 @@ export class MnlFormHelperTextInputComponent extends FieldType {
|
|
|
119
125
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
120
126
|
<div *ngIf="to['enableHelper']">
|
|
121
127
|
<mat-label>{{ to.label }}</mat-label>
|
|
122
|
-
<div *ngIf="
|
|
123
|
-
<span class="answer">{{
|
|
128
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
129
|
+
<span class="answer">{{ originalValue }}</span>
|
|
124
130
|
<mat-slide-toggle
|
|
125
131
|
[(ngModel)]="checked"
|
|
126
132
|
(toggleChange)="toggleSlide()"
|
|
127
|
-
>{{helperLabel}}</mat-slide-toggle
|
|
133
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
128
134
|
>
|
|
129
135
|
</div>
|
|
130
136
|
|
|
@@ -165,12 +171,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
165
171
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
166
172
|
<div *ngIf="to['enableHelper']">
|
|
167
173
|
<mat-label>{{ to.label }}</mat-label>
|
|
168
|
-
<div *ngIf="
|
|
169
|
-
<span class="answer">{{
|
|
174
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
175
|
+
<span class="answer">{{ originalValue }}</span>
|
|
170
176
|
<mat-slide-toggle
|
|
171
177
|
[(ngModel)]="checked"
|
|
172
178
|
(toggleChange)="toggleSlide()"
|
|
173
|
-
>{{helperLabel}}</mat-slide-toggle
|
|
179
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
174
180
|
>
|
|
175
181
|
</div>
|
|
176
182
|
|
|
@@ -193,4 +199,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
193
199
|
</div>
|
|
194
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"] }]
|
|
195
201
|
}], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
|
|
196
|
-
//# 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\">{{to['value'] }}</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  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.defaultHelperText);\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.defaultHelperText);\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(this.defaultHelperText);\n        }\n        break;\n\n        default:\n            this.formControl?.setValue(this.defaultHelperText);\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":"AAAA,OAAO,EAAE,YAAY,EAAE,MAAM,iBAAiB,CAAC;AAC/C,OAAO,EAAqB,SAAS,EAAU,MAAM,eAAe,CAAC;AACrE,OAAO,EAAE,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAClE,OAAO,EAAE,mBAAmB,EAAE,MAAM,wBAAwB,CAAC;AAC7D,OAAO,EAAE,mBAAmB,EAAE,MAAM,8BAA8B,CAAC;AACnE,OAAO,EAAE,kBAAkB,EAAE,MAAM,8BAA8B,CAAC;AAClE,OAAO,EAAE,cAAc,EAAE,MAAM,yBAAyB,CAAC;AACzD,OAAO,EAAE,eAAe,EAAE,MAAM,0BAA0B,CAAC;AAC3D,OAAO,EAAE,oBAAoB,EAAE,MAAM,gCAAgC,CAAC;AACtE,OAAO,EAGL,YAAY,GACb,MAAM,kBAAkB,CAAC;AAC1B,OAAO,EAAE,SAAS,EAAE,MAAM,sBAAsB,CAAC;;;;;;;;AAuEjD,MAAM,OAAO,+BACX,SAAQ,SAA0B;IAIlC,IAAI,WAAW;QACb,OAAO,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,EAAE,CAAC,aAAa,CAAC,IAAI,YAAY,CAAC;IACjE,CAAC;IACD,IAAI,iBAAiB;QACnB,OAAO,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC,EAAE,CAAC,mBAAmB,CAAC,IAAI,EAAE,CAAC;IAC7D,CAAC;IACD,IAAI,YAAY;QACd,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAa,CAAC;QAC1C,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,QAAQ,CAAC,CAAC;IACtC,CAAC;IAID,YAAoB,GAAsB;QACxC,KAAK,EAAE,CAAC;QADU,QAAG,GAAH,GAAG,CAAmB;QAd1C,YAAO,GAAG,KAAK,CAAC;IAgBhB,CAAC;IAED,QAAQ;QACN,IAAI,CAAC,aAAa,GAAG,IAAI,CAAC,YAAY,CAAC;QACvC,IAAI,CAAC,iBAAiB,EAAE,CAAC;QACzB,IAAI,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC,EAAE,CAAC;YAC5B,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/B,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,OAAO,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC;QAC7B,IAAG,IAAI,CAAC,OAAO,EAAE,CAAC;YAChB,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;QACrD,CAAC;QACD,IAAI,CAAC,GAAG,CAAC,aAAa,EAAE,CAAC;IAC3B,CAAC;IAED,iBAAiB;QACf,MAAM,WAAW,GAAG,IAAI,CAAC,EAAE,CAAC,oBAAoB,CAAC,CAAC;QAClD,IAAI,WAAW,EAAE,CAAC;YAChB,IAAI,CAAC,kBAAkB,GAAG;gBACxB,GAAG,WAAW;gBACd,GAAG,EAAE,IAAI,CAAC,KAAK,CAAC,GAAG,IAAI,WAAW,CAAC,GAAG,IAAI,YAAY;gBACtD,WAAW,EAAE,IAAI,CAAC,WAAW;gBAC7B,8CAA8C;gBAC9C,YAAY,EAAE,WAAW,CAAC,YAAY,IAAI,IAAI,CAAC,KAAK,CAAC,YAAY,IAAI,EAAE;gBACvE,UAAU,EAAE,WAAW,CAAC,UAAU,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,IAAI,EAAE;gBACjE,eAAe,EACb,WAAW,CAAC,eAAe,IAAI,IAAI,CAAC,KAAK,CAAC,eAAe,IAAI,EAAE;gBACjE,KAAK,EAAE,WAAW,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,CAAC,KAAK,IAAI,EAAE;gBAClD,WAAW,EAAE,WAAW,CAAC,WAAW,IAAI,IAAI,CAAC,KAAK,CAAC,WAAW,IAAI,EAAE;gBACpE,IAAI,EAAE,WAAW,CAAC,IAAI,KAAK,SAAS,CAAC,CAAC,CAAC,WAAW,CAAC,IAAI,CAAC,CAAC,CAAC,KAAK;gBAC/D,SAAS,EAAE,WAAW,CAAC,SAAS,IAAI,IAAI,CAAC,KAAK,CAAC,SAAS,IAAI,EAAE;gBAC9D,mBAAmB,EACjB,WAAW,CAAC,mBAAmB;oBAC/B,IAAI,CAAC,KAAK,CAAC,mBAAmB;oBAC9B,EAAE;gBACJ,wCAAwC;gBACxC,MAAM,EAAE,IAAI,CAAC,KAAK,CAAC,MAAM;gBACzB,OAAO,EAAE,IAAI,CAAC,KAAK,CAAC,OAAO;gBAC3B,KAAK,EAAE,IAAI,CAAC,KAAK,CAAC,KAAK;gBACvB,IAAI,EAAE,IAAI,CAAC,KAAK,CAAC,IAAI;aACtB,CAAC;QACJ,CAAC;IACH,CAAC;IAED,qBAAqB;QACnB,6CAA6C;QAC7C,QAAQ,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC;YACzC,KAAK,YAAY;gBACf,IACE,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;oBAC7C,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,YAAY,CAAC,EAC3D,CAAC;oBACD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;gBACtB,CAAC;gBACD,MAAM;YAER,KAAK,OAAO;gBACV,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC;gBACjC,MAAM,GAAG,GAAG,OAAO,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;gBAEjE,MAAM,OAAO,GACX,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC;oBACpB,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,GAAG;oBACzC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC;gBAE5C,IAAI,CAAC,OAAO,EAAE,CAAC;oBACb,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;gBACtB,CAAC;gBACD,MAAM;QACV,CAAC;IACH,CAAC;IAED,eAAe;QACb,QAAQ,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC;YACzC,KAAK,WAAW;gBACd,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACnD,MAAM;YAER,KAAK,YAAY;gBACf,IACE,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC;oBAC7C,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,EAC5D,CAAC;oBACD,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACrD,CAAC;gBACD,MAAM;YAER,KAAK,OAAO;gBACV,MAAM,MAAM,GAAG,IAAI,CAAC,YAAY,CAAC;gBACjC,MAAM,GAAG,GAAG,OAAO,MAAM,KAAK,QAAQ,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,MAAM,CAAC,MAAM,CAAC,CAAC;gBAEjE,MAAM,OAAO,GACX,MAAM,CAAC,QAAQ,CAAC,GAAG,CAAC;oBACpB,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,GAAG;oBACzC,GAAG,IAAI,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC;gBAE5C,IAAI,CAAC,OAAO,EAAE,CAAC;oBACb,IAAI,CAAC,WAAW,EAAE,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBACrD,CAAC;gBACD,MAAM;YAER;gBACE,MAAM;QACV,CAAC;IACH,CAAC;+GAhIU,+BAA+B;mGAA/B,+BAA+B,iGAtDhC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT,+RA1CC,YAAY,kIACZ,cAAc,snBACd,YAAY,+OACZ,mBAAmB,ykBACnB,WAAW,sPACX,kBAAkB,8BAClB,oBAAoB,yXACpB,eAAe,8BACf,mBAAmB,8BACnB,mBAAmB;;4FAwDV,+BAA+B;kBArE3C,SAAS;+BACE,gBAAgB,cACd,IAAI,WACP;wBACP,YAAY;wBACZ,cAAc;wBACd,YAAY;wBACZ,mBAAmB;wBACnB,WAAW;wBACX,kBAAkB;wBAClB,oBAAoB;wBACpB,eAAe;wBACf,mBAAmB;wBACnB,mBAAmB;qBACpB,YACS;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BT","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"]}
|
|
@@ -4524,14 +4524,18 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4524
4524
|
get defaultHelperText() {
|
|
4525
4525
|
return this.to['helperProps']?.['defaultHelperText'] || '';
|
|
4526
4526
|
}
|
|
4527
|
+
get currentValue() {
|
|
4528
|
+
const fieldKey = this.field.key;
|
|
4529
|
+
return this.field.model?.[fieldKey];
|
|
4530
|
+
}
|
|
4527
4531
|
constructor(cdr) {
|
|
4528
4532
|
super();
|
|
4529
4533
|
this.cdr = cdr;
|
|
4530
4534
|
this.checked = false;
|
|
4531
4535
|
}
|
|
4532
4536
|
ngOnInit() {
|
|
4537
|
+
this.originalValue = this.currentValue;
|
|
4533
4538
|
this.setupDefaultField();
|
|
4534
|
-
// If mode is helper, set up the initial state based on scoring
|
|
4535
4539
|
if (this.to['enableHelper']) {
|
|
4536
4540
|
this.setCorrectValue();
|
|
4537
4541
|
this.determineInitialState();
|
|
@@ -4539,6 +4543,9 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4539
4543
|
}
|
|
4540
4544
|
toggleSlide() {
|
|
4541
4545
|
this.checked = !this.checked;
|
|
4546
|
+
if (this.checked) {
|
|
4547
|
+
this.formControl?.setValue(this.defaultHelperText);
|
|
4548
|
+
}
|
|
4542
4549
|
this.cdr.detectChanges();
|
|
4543
4550
|
}
|
|
4544
4551
|
setupDefaultField() {
|
|
@@ -4572,12 +4579,12 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4572
4579
|
switch (this.to['scoring']?.['criteria']) {
|
|
4573
4580
|
case 'exactMatch':
|
|
4574
4581
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
4575
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
4582
|
+
!this.to['scoring']?.['answer'].includes(this.currentValue)) {
|
|
4576
4583
|
this.checked = true;
|
|
4577
4584
|
}
|
|
4578
4585
|
break;
|
|
4579
4586
|
case 'range':
|
|
4580
|
-
const answer = this.
|
|
4587
|
+
const answer = this.currentValue;
|
|
4581
4588
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
4582
4589
|
const correct = Number.isFinite(num) &&
|
|
4583
4590
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -4595,12 +4602,12 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4595
4602
|
break;
|
|
4596
4603
|
case 'exactMatch':
|
|
4597
4604
|
if (Array.isArray(this.to['scoring']?.['answer']) &&
|
|
4598
|
-
!this.to['scoring']?.['answer'].includes(this.
|
|
4605
|
+
!this.to['scoring']?.['answer'].includes(this.originalValue)) {
|
|
4599
4606
|
this.formControl?.setValue(this.defaultHelperText);
|
|
4600
4607
|
}
|
|
4601
4608
|
break;
|
|
4602
4609
|
case 'range':
|
|
4603
|
-
const answer = this.
|
|
4610
|
+
const answer = this.currentValue;
|
|
4604
4611
|
const num = typeof answer === 'number' ? answer : Number(answer);
|
|
4605
4612
|
const correct = Number.isFinite(num) &&
|
|
4606
4613
|
num >= this.to['scoring']?.['answer'].min &&
|
|
@@ -4610,7 +4617,6 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4610
4617
|
}
|
|
4611
4618
|
break;
|
|
4612
4619
|
default:
|
|
4613
|
-
this.formControl?.setValue(this.defaultHelperText);
|
|
4614
4620
|
break;
|
|
4615
4621
|
}
|
|
4616
4622
|
}
|
|
@@ -4620,12 +4626,12 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
|
|
|
4620
4626
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
4621
4627
|
<div *ngIf="to['enableHelper']">
|
|
4622
4628
|
<mat-label>{{ to.label }}</mat-label>
|
|
4623
|
-
<div *ngIf="
|
|
4624
|
-
<span class="answer">{{
|
|
4629
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
4630
|
+
<span class="answer">{{ originalValue }}</span>
|
|
4625
4631
|
<mat-slide-toggle
|
|
4626
4632
|
[(ngModel)]="checked"
|
|
4627
4633
|
(toggleChange)="toggleSlide()"
|
|
4628
|
-
>{{helperLabel}}</mat-slide-toggle
|
|
4634
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
4629
4635
|
>
|
|
4630
4636
|
</div>
|
|
4631
4637
|
|
|
@@ -4666,12 +4672,12 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
|
|
|
4666
4672
|
<!-- Helper Mode - Show toggle and conditional content -->
|
|
4667
4673
|
<div *ngIf="to['enableHelper']">
|
|
4668
4674
|
<mat-label>{{ to.label }}</mat-label>
|
|
4669
|
-
<div *ngIf="
|
|
4670
|
-
<span class="answer">{{
|
|
4675
|
+
<div *ngIf="originalValue" class="flex-row">
|
|
4676
|
+
<span class="answer">{{ originalValue }}</span>
|
|
4671
4677
|
<mat-slide-toggle
|
|
4672
4678
|
[(ngModel)]="checked"
|
|
4673
4679
|
(toggleChange)="toggleSlide()"
|
|
4674
|
-
>{{helperLabel}}</mat-slide-toggle
|
|
4680
|
+
>{{ helperLabel }}</mat-slide-toggle
|
|
4675
4681
|
>
|
|
4676
4682
|
</div>
|
|
4677
4683
|
|
|
@@ -4811,11 +4817,21 @@ class ScoringService {
|
|
|
4811
4817
|
calculateScore(model, fields) {
|
|
4812
4818
|
let totalScore = 0;
|
|
4813
4819
|
let categoryResults = {};
|
|
4814
|
-
|
|
4815
|
-
const categoryLabel = category.props?.label
|
|
4816
|
-
const categoryWeight = category.props?.weight
|
|
4817
|
-
|
|
4818
|
-
|
|
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;
|
|
4819
4835
|
let categoryScore = 0;
|
|
4820
4836
|
for (const q of questions) {
|
|
4821
4837
|
const parentKey = category.key;
|
|
@@ -4856,7 +4872,9 @@ class ScoringService {
|
|
|
4856
4872
|
score: categoryScore,
|
|
4857
4873
|
weight: categoryWeight,
|
|
4858
4874
|
};
|
|
4859
|
-
}
|
|
4875
|
+
};
|
|
4876
|
+
// start recursive walk
|
|
4877
|
+
fields.forEach(processCategory);
|
|
4860
4878
|
return {
|
|
4861
4879
|
totalScore,
|
|
4862
4880
|
categoryResults,
|