@muraai/mnl-form 0.0.1-alpha-d61aa5d → 0.0.1-alpha-f368a48

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.
@@ -51,7 +51,15 @@ export class ScoringService {
51
51
  break;
52
52
  case 'exactMatch':
53
53
  if (Array.isArray(scoring.answer)) {
54
- correct = scoring.answer.includes(answer);
54
+ // Handle case where answer is also an array (multiselect)
55
+ if (Array.isArray(answer)) {
56
+ // Check if answer is not empty AND all selected answers are in the expected answers
57
+ correct = answer.length > 0 && answer.every(a => scoring.answer.includes(a));
58
+ }
59
+ else {
60
+ // Single value case
61
+ correct = scoring.answer.includes(answer);
62
+ }
55
63
  }
56
64
  break;
57
65
  case 'range':
@@ -92,4 +100,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
92
100
  type: Injectable,
93
101
  args: [{ providedIn: 'root' }]
94
102
  }] });
95
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scoring.service.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/services/scoring.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { FormlyFieldConfig } from '@ngx-formly/core';\nimport { CustomFormlyFieldConfig } from '../../public-api';\n\n@Injectable({ providedIn: 'root' })\nexport class ScoringService {\n\n  updateScore(model: any, fields: FormlyFieldConfig[]) {\n    const result = this.calculateScore(model, fields);\n    return result;\n  }\n\n private calculateScore(model: any, fields: FormlyFieldConfig[] | CustomFormlyFieldConfig[]) {\n  let totalScore = 0;\n  let categoryResults: Record<string, { score: number; weight: number }> = {};\n\n  const processCategory = (category: FormlyFieldConfig | CustomFormlyFieldConfig) => {\n    const categoryLabel = category.props?.label;\n    const categoryWeight = category.props?.weight ?? 0;\n\n    // Only treat as scoring category if it has a label + weight\n    if (!categoryLabel || !categoryWeight) {\n      // Still need to go deeper in case children have categories\n      if (category.fieldGroup?.length) {\n        category.fieldGroup.forEach(processCategory);\n      }\n      return;\n    }\n\n    const questions = (category.fieldGroup || []).filter(q => q.props?.scoring);\n    if (questions.length === 0) return;\n\n    const scoredQuestions = questions.filter(q => q.props?.hasScoreImpact !== false);\n    if (scoredQuestions.length === 0) {\n      categoryResults[categoryLabel] = {\n        score: 0,\n        weight: categoryWeight,\n      };\n      return;\n    }\n\n    const perQuestionWeight = categoryWeight / scoredQuestions.length;\n    let categoryScore = 0;\n\n    for (const q of scoredQuestions) {\n      const parentKey = category.key as any;\n      const nestedPath = Array.isArray(parentKey)\n        ? [...(parentKey as any[]), q.key as any]\n        : [parentKey as any, q.key as any];\n      const answer = this.getNestedValue(model, nestedPath as any);\n\n      const scoring = q.props?.scoring;\n      if (!scoring) continue;\n\n      let correct = false;\n\n      switch (scoring.criteria) {\n        case 'attempted':\n          correct =\n            answer !== null &&\n            answer !== undefined &&\n            (typeof answer === 'string' ? answer.trim() !== '' : true);\n          break;\n\n        case 'exactMatch':\n          if (Array.isArray(scoring.answer)) {\n            correct = scoring.answer.includes(answer);\n          }\n          break;\n\n        case 'range':\n          const num = typeof answer === 'number' ? answer : Number(answer);\n          correct =\n            Number.isFinite(num) &&\n            num >= scoring.answer.min &&\n            num <= scoring.answer.max;\n          break;\n      }\n\n      if (correct) {\n        categoryScore += perQuestionWeight;\n      }\n    }\n\n    totalScore += categoryScore;\n    categoryResults[categoryLabel] = {\n      score: categoryScore,\n      weight: categoryWeight,\n    };\n  };\n\n  // start recursive walk\n  fields.forEach(processCategory);\n\n  return {\n    totalScore,\n    categoryResults,\n  };\n}\n\n\n  private getNestedValue(obj: any, key: string | number | (string | number)[]): any {\n    if (Array.isArray(key)) {\n      return key.reduce((current, k) => current?.[k], obj);\n    }\n    return obj?.[key];\n  }\n}"]}
103
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"scoring.service.js","sourceRoot":"","sources":["../../../../../../projects/form-lib/src/lib/services/scoring.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { BehaviorSubject } from 'rxjs';\nimport { FormlyFieldConfig } from '@ngx-formly/core';\nimport { CustomFormlyFieldConfig } from '../../public-api';\n\n@Injectable({ providedIn: 'root' })\nexport class ScoringService {\n\n  updateScore(model: any, fields: FormlyFieldConfig[]) {\n    const result = this.calculateScore(model, fields);\n    return result;\n  }\n\n private calculateScore(model: any, fields: FormlyFieldConfig[] | CustomFormlyFieldConfig[]) {\n  let totalScore = 0;\n  let categoryResults: Record<string, { score: number; weight: number }> = {};\n\n  const processCategory = (category: FormlyFieldConfig | CustomFormlyFieldConfig) => {\n    const categoryLabel = category.props?.label;\n    const categoryWeight = category.props?.weight ?? 0;\n\n    // Only treat as scoring category if it has a label + weight\n    if (!categoryLabel || !categoryWeight) {\n      // Still need to go deeper in case children have categories\n      if (category.fieldGroup?.length) {\n        category.fieldGroup.forEach(processCategory);\n      }\n      return;\n    }\n\n    const questions = (category.fieldGroup || []).filter(q => q.props?.scoring);\n    if (questions.length === 0) return;\n\n    const scoredQuestions = questions.filter(q => q.props?.hasScoreImpact !== false);\n    if (scoredQuestions.length === 0) {\n      categoryResults[categoryLabel] = {\n        score: 0,\n        weight: categoryWeight,\n      };\n      return;\n    }\n\n    const perQuestionWeight = categoryWeight / scoredQuestions.length;\n    let categoryScore = 0;\n\n    for (const q of scoredQuestions) {\n      const parentKey = category.key as any;\n      const nestedPath = Array.isArray(parentKey)\n        ? [...(parentKey as any[]), q.key as any]\n        : [parentKey as any, q.key as any];\n      const answer = this.getNestedValue(model, nestedPath as any);\n\n      const scoring = q.props?.scoring;\n      if (!scoring) continue;\n\n      let correct = false;\n      \n      switch (scoring.criteria) {\n        case 'attempted':\n          correct =\n            answer !== null &&\n            answer !== undefined &&\n            (typeof answer === 'string' ? answer.trim() !== '' : true);\n          break;\n\n        case 'exactMatch':\n          if (Array.isArray(scoring.answer)) {\n            // Handle case where answer is also an array (multiselect)\n            if (Array.isArray(answer)) {\n              // Check if answer is not empty AND all selected answers are in the expected answers\n              correct = answer.length > 0 && answer.every(a => scoring.answer.includes(a));\n            } else {\n              // Single value case\n              correct = scoring.answer.includes(answer);\n            }\n          }\n          break;\n\n        case 'range':\n          const num = typeof answer === 'number' ? answer : Number(answer);\n          correct =\n            Number.isFinite(num) &&\n            num >= scoring.answer.min &&\n            num <= scoring.answer.max;\n          break;\n      }\n\n      if (correct) {\n        categoryScore += perQuestionWeight;\n      }\n    }\n\n    totalScore += categoryScore;\n    categoryResults[categoryLabel] = {\n      score: categoryScore,\n      weight: categoryWeight,\n    };\n  };\n\n  // start recursive walk\n  fields.forEach(processCategory);\n\n  return {\n    totalScore,\n    categoryResults,\n  };\n}\n\n\n  private getNestedValue(obj: any, key: string | number | (string | number)[]): any {\n    if (Array.isArray(key)) {\n      return key.reduce((current, k) => current?.[k], obj);\n    }\n    return obj?.[key];\n  }\n}"]}
@@ -1,5 +1,5 @@
1
1
  import { CommonModule } from '@angular/common';
2
- import { Component } from '@angular/core';
2
+ import { Component, ViewChild } from '@angular/core';
3
3
  import { FormControl, FormsModule, ReactiveFormsModule } from '@angular/forms';
4
4
  import { MatNativeDateModule } from '@angular/material/core';
5
5
  import { MatDatepickerModule } from '@angular/material/datepicker';
@@ -13,9 +13,10 @@ import * as i0 from "@angular/core";
13
13
  import * as i1 from "@angular/common";
14
14
  import * as i2 from "@angular/material/input";
15
15
  import * as i3 from "@angular/material/form-field";
16
- import * as i4 from "@ngx-formly/core";
17
- import * as i5 from "@angular/forms";
18
- import * as i6 from "@angular/material/slide-toggle";
16
+ import * as i4 from "@angular/cdk/text-field";
17
+ import * as i5 from "@ngx-formly/core";
18
+ import * as i6 from "@angular/forms";
19
+ import * as i7 from "@angular/material/slide-toggle";
19
20
  export class MnlFormHelperTextInputComponent extends FieldType {
20
21
  get helperLabel() {
21
22
  return this.to['helperProps']?.['helperLabel'] || 'Need help?';
@@ -74,6 +75,9 @@ export class MnlFormHelperTextInputComponent extends FieldType {
74
75
  this.checked = true;
75
76
  }
76
77
  }
78
+ ngAfterViewInit() {
79
+ setTimeout(() => this.autosize?.resizeToFitContent(true));
80
+ }
77
81
  toggleSlide() {
78
82
  this.checked = !this.checked;
79
83
  if (this.checked) {
@@ -91,10 +95,27 @@ export class MnlFormHelperTextInputComponent extends FieldType {
91
95
  setupDefaultField() {
92
96
  const fieldConfig = this.to['defaultFieldConfig'];
93
97
  if (fieldConfig) {
98
+ // Get current value from model and sync with formControl
99
+ const fieldKey = this.field.key;
100
+ const modelValue = this.field.model?.[fieldKey];
101
+ // Sync formControl value with model value if it exists
102
+ if (modelValue !== undefined && modelValue !== null) {
103
+ this.formControl.setValue(modelValue, { emitEvent: false });
104
+ }
105
+ // Create enhanced props with necessary fixes for production
106
+ const enhancedProps = {
107
+ ...fieldConfig.props,
108
+ };
109
+ // For select fields, add compareWith function to ensure proper value matching in production builds
110
+ if (fieldConfig.type === 'select' && !enhancedProps['compareWith']) {
111
+ enhancedProps['compareWith'] = (o1, o2) => o1 === o2;
112
+ }
94
113
  this.defaultFieldConfig = {
95
- ...fieldConfig,
114
+ type: fieldConfig.type,
96
115
  key: this.field.key || fieldConfig.key || 'defaultKey',
97
116
  formControl: this.formControl,
117
+ props: enhancedProps,
118
+ wrappers: fieldConfig.wrappers || [],
98
119
  modelOptions: fieldConfig.modelOptions || this.field.modelOptions || {},
99
120
  validators: fieldConfig.validators || this.field.validators || {},
100
121
  asyncValidators: fieldConfig.asyncValidators || this.field.asyncValidators || {},
@@ -116,9 +137,22 @@ export class MnlFormHelperTextInputComponent extends FieldType {
116
137
  // Auto-open helper based on scoring criteria
117
138
  switch (this.to['scoring']?.['criteria']) {
118
139
  case 'exactMatch':
119
- if (Array.isArray(this.to['scoring']?.['answer']) &&
120
- !this.to['scoring']?.['answer'].includes(this.originalValue)) {
121
- this.checked = true;
140
+ if (Array.isArray(this.to['scoring']?.['answer'])) {
141
+ // Handle multiselect case where answer is also an array
142
+ if (Array.isArray(this.originalValue)) {
143
+ // Check if not empty AND all selected answers are in expected answers
144
+ const isCorrect = this.originalValue.length > 0 &&
145
+ this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));
146
+ if (!isCorrect) {
147
+ this.checked = true;
148
+ }
149
+ }
150
+ else {
151
+ // Single value case
152
+ if (!this.to['scoring']?.['answer'].includes(this.originalValue)) {
153
+ this.checked = true;
154
+ }
155
+ }
122
156
  }
123
157
  break;
124
158
  case 'range':
@@ -146,13 +180,30 @@ export class MnlFormHelperTextInputComponent extends FieldType {
146
180
  break;
147
181
  case 'exactMatch':
148
182
  if (Array.isArray(this.to['scoring']?.['answer'])) {
149
- if (this.to['scoring']?.['answer'].includes(this.originalValue)) {
150
- // Answer is correct - set empty value in helper
151
- this.helperFormControl.setValue('');
183
+ // Handle multiselect case where answer is also an array
184
+ if (Array.isArray(this.originalValue)) {
185
+ // Check if not empty AND all selected answers are in expected answers
186
+ const isCorrect = this.originalValue.length > 0 &&
187
+ this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));
188
+ if (isCorrect) {
189
+ // Answer is correct - set empty value in helper
190
+ this.helperFormControl.setValue('');
191
+ }
192
+ else {
193
+ // Answer is incorrect - set defaultHelperText in helper
194
+ this.helperFormControl.setValue(this.defaultHelperText);
195
+ }
152
196
  }
153
197
  else {
154
- // Answer is incorrect - set defaultHelperText in helper
155
- this.helperFormControl.setValue(this.defaultHelperText);
198
+ // Single value case
199
+ if (this.to['scoring']?.['answer'].includes(this.originalValue)) {
200
+ // Answer is correct - set empty value in helper
201
+ this.helperFormControl.setValue('');
202
+ }
203
+ else {
204
+ // Answer is incorrect - set defaultHelperText in helper
205
+ this.helperFormControl.setValue(this.defaultHelperText);
206
+ }
156
207
  }
157
208
  }
158
209
  break;
@@ -176,7 +227,7 @@ export class MnlFormHelperTextInputComponent extends FieldType {
176
227
  }
177
228
  }
178
229
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MnlFormHelperTextInputComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
179
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MnlFormHelperTextInputComponent, isStandalone: true, selector: "mu-helper-text", usesInheritance: true, ngImport: i0, template: `
230
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MnlFormHelperTextInputComponent, isStandalone: true, selector: "mu-helper-text", viewQueries: [{ propertyName: "autosize", first: true, predicate: ["autosize"], descendants: true }], usesInheritance: true, ngImport: i0, template: `
180
231
  <div class="helper-text-container">
181
232
  <!-- Helper Mode - Show toggle and conditional content -->
182
233
  <div *ngIf="to['enableHelper']">
@@ -193,12 +244,14 @@ export class MnlFormHelperTextInputComponent extends FieldType {
193
244
 
194
245
  <mat-form-field class="textarea" appearance="outline" *ngIf="checked">
195
246
  <textarea
247
+ cdkTextareaAutosize
248
+ #autosize="cdkTextareaAutosize"
249
+ cdkAutosizeMinRows="2"
196
250
  matInput
197
251
  [formControl]="helperFormControl"
198
252
  [readonly]="readonly || field.form?.disabled"
199
253
  [disabled]="field.form?.disabled"
200
- >
201
- </textarea>
254
+ ></textarea>
202
255
  </mat-form-field>
203
256
  </div>
204
257
 
@@ -208,7 +261,7 @@ export class MnlFormHelperTextInputComponent extends FieldType {
208
261
  </formly-field>
209
262
  </div>
210
263
  </div>
211
- `, isInline: true, 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"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i4.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i5.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i5.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i5.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i5.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "component", type: i6.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "ngmodule", type: MatNativeDateModule }] }); }
264
+ `, isInline: true, 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"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4.CdkTextareaAutosize, selector: "textarea[cdkTextareaAutosize]", inputs: ["cdkAutosizeMinRows", "cdkAutosizeMaxRows", "cdkTextareaAutosize", "placeholder"], exportAs: ["cdkTextareaAutosize"] }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i5.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i6.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i6.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i6.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i6.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "component", type: i7.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "ngmodule", type: MatNativeDateModule }] }); }
212
265
  }
213
266
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MnlFormHelperTextInputComponent, decorators: [{
214
267
  type: Component,
@@ -240,12 +293,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
240
293
 
241
294
  <mat-form-field class="textarea" appearance="outline" *ngIf="checked">
242
295
  <textarea
296
+ cdkTextareaAutosize
297
+ #autosize="cdkTextareaAutosize"
298
+ cdkAutosizeMinRows="2"
243
299
  matInput
244
300
  [formControl]="helperFormControl"
245
301
  [readonly]="readonly || field.form?.disabled"
246
302
  [disabled]="field.form?.disabled"
247
- >
248
- </textarea>
303
+ ></textarea>
249
304
  </mat-form-field>
250
305
  </div>
251
306
 
@@ -256,5 +311,8 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
256
311
  </div>
257
312
  </div>
258
313
  `, 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"] }]
259
- }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
260
- //# 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,WAAW,EAAE,mBAAmB,EAAE,MAAM,gBAAgB,CAAC;AAC/E,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;;;;;;;;AAwEjD,MAAM,OAAO,+BACX,SAAQ,SAA0B;IAMlC,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;IAED,yCAAyC;IACzC,IAAI,aAAa;QACf,MAAM,QAAQ,GAAG,IAAI,CAAC,KAAK,CAAC,GAAa,CAAC;QAC1C,OAAO,IAAI,CAAC,KAAK,CAAC,KAAK,EAAE,CAAC,QAAQ,CAAC,CAAC;IACtC,CAAC;IAED,8CAA8C;IAC9C,IAAI,WAAW;QACb,OAAO,CACL,IAAI,CAAC,EAAE,CAAC,aAAa,CAAC;YACtB,IAAI,CAAC,KAAK,CAAC,aAAa,CAAC;YACzB,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,CAAC,aAAa,CAAC,CACzC,CAAC;IACJ,CAAC;IAED,IAAI,SAAS;QACX,OAAO,IAAI,CAAC,EAAE,CAAC,WAAW,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,GAAG,CAAC;IAChD,CAAC;IAED,IAAI,kBAAkB;QACpB,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACvC,OAAO,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAmB,CAAC,CAAC;QACpD,CAAC;QACD,OAAO,EAAE,CAAC;IACZ,CAAC;IAED,IAAI,kBAAkB,CAAC,KAAU;QAC/B,IAAI,IAAI,CAAC,WAAW,IAAI,IAAI,CAAC,SAAS,EAAE,CAAC;YACvC,IAAI,CAAC,WAAW,CAAC,IAAI,CAAC,SAAmB,CAAC,GAAG,KAAK,CAAC;QACrD,CAAC;IACH,CAAC;IAED,IAAI,QAAQ;QACV,OAAO,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,UAAU,CAAC,IAAI,KAAK,CAAC;IAChE,CAAC;IAID,YAAoB,GAAsB;QACxC,KAAK,EAAE,CAAC;QADU,QAAG,GAAH,GAAG,CAAmB;QAhD1C,YAAO,GAAG,KAAK,CAAC;IAkDhB,CAAC;IAED,QAAQ;QACN,8DAA8D;QAC9D,IAAI,CAAC,iBAAiB,GAAG,IAAI,WAAW,CAAC,IAAI,CAAC,kBAAkB,IAAI,EAAE,CAAC,CAAC;QAExE,sEAAsE;QACtE,IAAI,CAAC,iBAAiB,CAAC,YAAY,CAAC,SAAS,CAAC,CAAC,KAAK,EAAE,EAAE;YACtD,IAAI,CAAC,kBAAkB,GAAG,KAAK,CAAC;YAChC,+CAA+C;YAC/C,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,CAAC,qBAAqB,CAAC,EAAE,CAAC,IAAI,CAAC,WAAW,CAAC,CAAC;QACvE,CAAC,CAAC,CAAC;QAEH,IAAI,CAAC,iBAAiB,EAAE,CAAC;QAEzB,IAAI,IAAI,CAAC,EAAE,CAAC,cAAc,CAAC,EAAE,CAAC;YAC5B,IAAI,CAAC,eAAe,EAAE,CAAC;YACvB,IAAI,CAAC,qBAAqB,EAAE,CAAC;QAC/B,CAAC;QACD,IAAI,IAAI,CAAC,kBAAkB,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;YACrE,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;QACtB,CAAC;IACH,CAAC;IAED,WAAW;QACT,IAAI,CAAC,OAAO,GAAG,CAAC,IAAI,CAAC,OAAO,CAAC;QAC7B,IAAI,IAAI,CAAC,OAAO,EAAE,CAAC;YACjB,wDAAwD;YACxD,MAAM,aAAa,GAAG,IAAI,CAAC,iBAAiB,CAAC,KAAK,CAAC;YACnD,IAAI,CAAC,aAAa,IAAI,aAAa,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;gBAClD,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;YAC1D,CAAC;QACH,CAAC;aAAM,CAAC;YACN,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;QACtC,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,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,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,aAAa,CAAC,EAC5D,CAAC;oBACD,IAAI,CAAC,OAAO,GAAG,IAAI,CAAC;gBACtB,CAAC;gBACD,MAAM;YAER,KAAK,OAAO;gBACV,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;gBAClC,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,MAAM,kBAAkB,GAAG,IAAI,CAAC,kBAAkB,CAAC;QAEnD,8DAA8D;QAC9D,IAAI,kBAAkB,IAAI,kBAAkB,CAAC,IAAI,EAAE,KAAK,EAAE,EAAE,CAAC;YAC3D,OAAO;QACT,CAAC;QAED,QAAQ,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,UAAU,CAAC,EAAE,CAAC;YACzC,KAAK,WAAW;gBACd,uCAAuC;gBACvC,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;gBACpC,MAAM;YAER,KAAK,YAAY;gBACf,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,EAAE,CAAC;oBAClD,IAAI,IAAI,CAAC,EAAE,CAAC,SAAS,CAAC,EAAE,CAAC,QAAQ,CAAC,CAAC,QAAQ,CAAC,IAAI,CAAC,aAAa,CAAC,EAAE,CAAC;wBAChE,gDAAgD;wBAChD,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;oBACtC,CAAC;yBAAM,CAAC;wBACN,wDAAwD;wBACxD,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;oBAC1D,CAAC;gBACH,CAAC;gBACD,MAAM;YAER,KAAK,OAAO;gBACV,MAAM,MAAM,GAAG,IAAI,CAAC,aAAa,CAAC;gBAClC,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,OAAO,EAAE,CAAC;oBACZ,gDAAgD;oBAChD,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC;gBACtC,CAAC;qBAAM,CAAC;oBACN,wDAAwD;oBACxD,IAAI,CAAC,iBAAiB,CAAC,QAAQ,CAAC,IAAI,CAAC,iBAAiB,CAAC,CAAC;gBAC1D,CAAC;gBACD,MAAM;YAER;gBACE,MAAM;QACV,CAAC;IACH,CAAC;+GAlMU,+BAA+B;mGAA/B,+BAA+B,iGAvDhC;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCT,+RA3CC,YAAY,kIACZ,cAAc,snBACd,YAAY,wHACZ,mBAAmB,ykBACnB,WAAW,sPACX,kBAAkB,8BAClB,oBAAoB,yXACpB,eAAe,8BACf,mBAAmB,8BACnB,mBAAmB;;4FAyDV,+BAA+B;kBAtE3C,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;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GAgCT","sourcesContent":["import { CommonModule } from '@angular/common';\nimport { ChangeDetectorRef, Component, OnInit } from '@angular/core';\nimport { FormControl, 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            [disabled]=\"field.form?.disabled\"\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]=\"helperFormControl\"\n            [readonly]=\"readonly || field.form?.disabled\"\n            [disabled]=\"field.form?.disabled\"\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  helperFormControl!: FormControl; // Separate form control for helper model\n\n  get helperLabel() {\n    return this.to['helperProps']?.['helperLabel'] || 'Need help?';\n  }\n  get defaultHelperText() {\n    return this.to['helperProps']?.['defaultHelperText'] || '';\n  }\n\n  // Get value from primary model (answers)\n  get originalValue() {\n    const fieldKey = this.field.key as string;\n    return this.field.model?.[fieldKey];\n  }\n\n  // Get/Set value from secondary model (helper)\n  get helperModel() {\n    return (\n      this.to['helperModel'] ||\n      this.props['helperModel'] ||\n      this.options?.formState?.['helperModel']\n    );\n  }\n\n  get helperKey() {\n    return this.to['helperKey'] || this.field.key;\n  }\n\n  get currentHelperValue() {\n    if (this.helperModel && this.helperKey) {\n      return this.helperModel[this.helperKey as string];\n    }\n    return '';\n  }\n\n  set currentHelperValue(value: any) {\n    if (this.helperModel && this.helperKey) {\n      this.helperModel[this.helperKey as string] = value;\n    }\n  }\n\n  get readonly() {\n    return this.to['readonly'] || this.props['readonly'] || false;\n  }\n\n  defaultFieldConfig?: FormlyFieldConfig;\n\n  constructor(private cdr: ChangeDetectorRef) {\n    super();\n  }\n\n  ngOnInit() {\n    // Initialize helper form control with value from helper model\n    this.helperFormControl = new FormControl(this.currentHelperValue || '');\n\n    // Subscribe to changes in helper form control and update helper model\n    this.helperFormControl.valueChanges.subscribe((value) => {\n      this.currentHelperValue = value;\n      // Emit helper model change to parent component\n      this.options?.formState?.['onHelperModelChange']?.(this.helperModel);\n    });\n\n    this.setupDefaultField();\n\n    if (this.to['enableHelper']) {\n      this.setCorrectValue();\n      this.determineInitialState();\n    }\n    if (this.currentHelperValue && this.currentHelperValue.trim() !== '') {\n      this.checked = true;\n    }\n  }\n\n  toggleSlide() {\n    this.checked = !this.checked;\n    if (this.checked) {\n      // When toggle is opened, check if helper field is empty\n      const currenthelper = this.helperFormControl.value;\n      if (!currenthelper || currenthelper.trim() === '') {\n        this.helperFormControl.setValue(this.defaultHelperText);\n      }\n    } else {\n      this.helperFormControl.setValue('');\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        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        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.originalValue)\n        ) {\n          this.checked = true;\n        }\n        break;\n\n      case 'range':\n        const answer = this.originalValue;\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    const currentHelperValue = this.currentHelperValue;\n\n    // If helper already has a value (user edited), don't override\n    if (currentHelperValue && currentHelperValue.trim() !== '') {\n      return;\n    }\n\n    switch (this.to['scoring']?.['criteria']) {\n      case 'attempted':\n        // For attempted, set defaultHelperText\n        this.helperFormControl.setValue('');\n        break;\n\n      case 'exactMatch':\n        if (Array.isArray(this.to['scoring']?.['answer'])) {\n          if (this.to['scoring']?.['answer'].includes(this.originalValue)) {\n            // Answer is correct - set empty value in helper\n            this.helperFormControl.setValue('');\n          } else {\n            // Answer is incorrect - set defaultHelperText in helper\n            this.helperFormControl.setValue(this.defaultHelperText);\n          }\n        }\n        break;\n\n      case 'range':\n        const answer = this.originalValue;\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          // Answer is correct - set empty value in helper\n          this.helperFormControl.setValue('');\n        } else {\n          // Answer is incorrect - set defaultHelperText in helper\n          this.helperFormControl.setValue(this.defaultHelperText);\n        }\n        break;\n\n      default:\n        break;\n    }\n  }\n}\n"]}
314
+ }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { autosize: [{
315
+ type: ViewChild,
316
+ args: ['autosize']
317
+ }] } });
318
+ //# 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 { CdkTextareaAutosize } from '@angular/cdk/text-field';\nimport { CommonModule } from '@angular/common';\nimport { AfterViewInit, ChangeDetectorRef, Component, OnInit, ViewChild } from '@angular/core';\nimport { FormControl, 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            [disabled]=\"field.form?.disabled\"\n            >{{ helperLabel }}</mat-slide-toggle\n          >\n        </div>\n\n        <mat-form-field class=\"textarea\" appearance=\"outline\" *ngIf=\"checked\">\n          <textarea\n            cdkTextareaAutosize\n            #autosize=\"cdkTextareaAutosize\"\n            cdkAutosizeMinRows=\"2\"\n            matInput\n            [formControl]=\"helperFormControl\"\n            [readonly]=\"readonly || field.form?.disabled\"\n            [disabled]=\"field.form?.disabled\"\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, AfterViewInit\n{\n  @ViewChild('autosize') autosize!: CdkTextareaAutosize;\n  checked = false;\n  helperFormControl!: FormControl; // Separate form control for helper model\n\n  get helperLabel() {\n    return this.to['helperProps']?.['helperLabel'] || 'Need help?';\n  }\n  get defaultHelperText() {\n    return this.to['helperProps']?.['defaultHelperText'] || '';\n  }\n\n  // Get value from primary model (answers)\n  get originalValue() {\n    const fieldKey = this.field.key as string;\n    return this.field.model?.[fieldKey];\n  }\n\n  // Get/Set value from secondary model (helper)\n  get helperModel() {\n    return (\n      this.to['helperModel'] ||\n      this.props['helperModel'] ||\n      this.options?.formState?.['helperModel']\n    );\n  }\n\n  get helperKey() {\n    return this.to['helperKey'] || this.field.key;\n  }\n\n  get currentHelperValue() {\n    if (this.helperModel && this.helperKey) {\n      return this.helperModel[this.helperKey as string];\n    }\n    return '';\n  }\n\n  set currentHelperValue(value: any) {\n    if (this.helperModel && this.helperKey) {\n      this.helperModel[this.helperKey as string] = value;\n    }\n  }\n\n  get readonly() {\n    return this.to['readonly'] || this.props['readonly'] || false;\n  }\n\n  defaultFieldConfig?: FormlyFieldConfig;\n\n  constructor(private cdr: ChangeDetectorRef) {\n    super();\n  }\n\n  ngOnInit() {\n    // Initialize helper form control with value from helper model\n    this.helperFormControl = new FormControl(this.currentHelperValue || '');\n\n    // Subscribe to changes in helper form control and update helper model\n    this.helperFormControl.valueChanges.subscribe((value) => {\n      this.currentHelperValue = value;\n      // Emit helper model change to parent component\n      this.options?.formState?.['onHelperModelChange']?.(this.helperModel);\n    });\n\n    this.setupDefaultField();\n\n    if (this.to['enableHelper']) {\n      this.setCorrectValue();\n      this.determineInitialState();\n    }\n    if (this.currentHelperValue && this.currentHelperValue.trim() !== '') {\n      this.checked = true;\n    }\n  }\n\n  ngAfterViewInit() {\n    setTimeout(() => this.autosize?.resizeToFitContent(true));\n  }\n\n  toggleSlide() {\n    this.checked = !this.checked;\n    if (this.checked) {\n      // When toggle is opened, check if helper field is empty\n      const currenthelper = this.helperFormControl.value;\n      if (!currenthelper || currenthelper.trim() === '') {\n        this.helperFormControl.setValue(this.defaultHelperText);\n      }\n    } else {\n      this.helperFormControl.setValue('');\n    }\n    this.cdr.detectChanges();\n  }\n\n  setupDefaultField() {\n    const fieldConfig = this.to['defaultFieldConfig'];\n    if (fieldConfig) {\n      // Get current value from model and sync with formControl\n      const fieldKey = this.field.key as string;\n      const modelValue = this.field.model?.[fieldKey];\n      \n      // Sync formControl value with model value if it exists\n      if (modelValue !== undefined && modelValue !== null) {\n        this.formControl.setValue(modelValue, { emitEvent: false });\n      }\n      \n      // Create enhanced props with necessary fixes for production\n      const enhancedProps = {\n        ...fieldConfig.props,\n      };\n      \n      // For select fields, add compareWith function to ensure proper value matching in production builds\n      if (fieldConfig.type === 'select' && !enhancedProps['compareWith']) {\n        enhancedProps['compareWith'] = (o1: any, o2: any) => o1 === o2;\n      }\n      \n      this.defaultFieldConfig = {\n        type: fieldConfig.type,\n        key: this.field.key || fieldConfig.key || 'defaultKey',\n        formControl: this.formControl,\n        props: enhancedProps,\n        wrappers: fieldConfig.wrappers || [],\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        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 (Array.isArray(this.to['scoring']?.['answer'])) {\n          // Handle multiselect case where answer is also an array\n          if (Array.isArray(this.originalValue)) {\n            // Check if not empty AND all selected answers are in expected answers\n            const isCorrect = this.originalValue.length > 0 && \n              this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));\n            if (!isCorrect) {\n              this.checked = true;\n            }\n          } else {\n            // Single value case\n            if (!this.to['scoring']?.['answer'].includes(this.originalValue)) {\n              this.checked = true;\n            }\n          }\n        }\n        break;\n\n      case 'range':\n        const answer = this.originalValue;\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    const currentHelperValue = this.currentHelperValue;\n\n    // If helper already has a value (user edited), don't override\n    if (currentHelperValue && currentHelperValue.trim() !== '') {\n      return;\n    }\n\n    switch (this.to['scoring']?.['criteria']) {\n      case 'attempted':\n        // For attempted, set defaultHelperText\n        this.helperFormControl.setValue('');\n        break;\n\n      case 'exactMatch':\n        if (Array.isArray(this.to['scoring']?.['answer'])) {\n          // Handle multiselect case where answer is also an array\n          if (Array.isArray(this.originalValue)) {\n            // Check if not empty AND all selected answers are in expected answers\n            const isCorrect = this.originalValue.length > 0 && \n              this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));\n            if (isCorrect) {\n              // Answer is correct - set empty value in helper\n              this.helperFormControl.setValue('');\n            } else {\n              // Answer is incorrect - set defaultHelperText in helper\n              this.helperFormControl.setValue(this.defaultHelperText);\n            }\n          } else {\n            // Single value case\n            if (this.to['scoring']?.['answer'].includes(this.originalValue)) {\n              // Answer is correct - set empty value in helper\n              this.helperFormControl.setValue('');\n            } else {\n              // Answer is incorrect - set defaultHelperText in helper\n              this.helperFormControl.setValue(this.defaultHelperText);\n            }\n          }\n        }\n        break;\n\n      case 'range':\n        const answer = this.originalValue;\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          // Answer is correct - set empty value in helper\n          this.helperFormControl.setValue('');\n        } else {\n          // Answer is incorrect - set defaultHelperText in helper\n          this.helperFormControl.setValue(this.defaultHelperText);\n        }\n        break;\n\n      default:\n        break;\n    }\n  }\n}\n"]}
@@ -63,8 +63,9 @@ import { FormlyMatDatepickerModule } from '@ngx-formly/material/datepicker';
63
63
  import * as i3$4 from '@angular/material/card';
64
64
  import { MatCardModule } from '@angular/material/card';
65
65
  import { MatDatepickerModule } from '@angular/material/datepicker';
66
- import * as i6$3 from '@angular/material/slide-toggle';
66
+ import * as i7$3 from '@angular/material/slide-toggle';
67
67
  import { MatSlideToggleModule } from '@angular/material/slide-toggle';
68
+ import * as i4$2 from '@angular/cdk/text-field';
68
69
  import { FormlyMatCheckboxModule } from '@ngx-formly/material/checkbox';
69
70
  import { FormlyMatInputModule } from '@ngx-formly/material/input';
70
71
  import { FormlyMatRadioModule } from '@ngx-formly/material/radio';
@@ -4575,6 +4576,9 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4575
4576
  this.checked = true;
4576
4577
  }
4577
4578
  }
4579
+ ngAfterViewInit() {
4580
+ setTimeout(() => this.autosize?.resizeToFitContent(true));
4581
+ }
4578
4582
  toggleSlide() {
4579
4583
  this.checked = !this.checked;
4580
4584
  if (this.checked) {
@@ -4592,10 +4596,27 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4592
4596
  setupDefaultField() {
4593
4597
  const fieldConfig = this.to['defaultFieldConfig'];
4594
4598
  if (fieldConfig) {
4599
+ // Get current value from model and sync with formControl
4600
+ const fieldKey = this.field.key;
4601
+ const modelValue = this.field.model?.[fieldKey];
4602
+ // Sync formControl value with model value if it exists
4603
+ if (modelValue !== undefined && modelValue !== null) {
4604
+ this.formControl.setValue(modelValue, { emitEvent: false });
4605
+ }
4606
+ // Create enhanced props with necessary fixes for production
4607
+ const enhancedProps = {
4608
+ ...fieldConfig.props,
4609
+ };
4610
+ // For select fields, add compareWith function to ensure proper value matching in production builds
4611
+ if (fieldConfig.type === 'select' && !enhancedProps['compareWith']) {
4612
+ enhancedProps['compareWith'] = (o1, o2) => o1 === o2;
4613
+ }
4595
4614
  this.defaultFieldConfig = {
4596
- ...fieldConfig,
4615
+ type: fieldConfig.type,
4597
4616
  key: this.field.key || fieldConfig.key || 'defaultKey',
4598
4617
  formControl: this.formControl,
4618
+ props: enhancedProps,
4619
+ wrappers: fieldConfig.wrappers || [],
4599
4620
  modelOptions: fieldConfig.modelOptions || this.field.modelOptions || {},
4600
4621
  validators: fieldConfig.validators || this.field.validators || {},
4601
4622
  asyncValidators: fieldConfig.asyncValidators || this.field.asyncValidators || {},
@@ -4617,9 +4638,22 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4617
4638
  // Auto-open helper based on scoring criteria
4618
4639
  switch (this.to['scoring']?.['criteria']) {
4619
4640
  case 'exactMatch':
4620
- if (Array.isArray(this.to['scoring']?.['answer']) &&
4621
- !this.to['scoring']?.['answer'].includes(this.originalValue)) {
4622
- this.checked = true;
4641
+ if (Array.isArray(this.to['scoring']?.['answer'])) {
4642
+ // Handle multiselect case where answer is also an array
4643
+ if (Array.isArray(this.originalValue)) {
4644
+ // Check if not empty AND all selected answers are in expected answers
4645
+ const isCorrect = this.originalValue.length > 0 &&
4646
+ this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));
4647
+ if (!isCorrect) {
4648
+ this.checked = true;
4649
+ }
4650
+ }
4651
+ else {
4652
+ // Single value case
4653
+ if (!this.to['scoring']?.['answer'].includes(this.originalValue)) {
4654
+ this.checked = true;
4655
+ }
4656
+ }
4623
4657
  }
4624
4658
  break;
4625
4659
  case 'range':
@@ -4647,13 +4681,30 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4647
4681
  break;
4648
4682
  case 'exactMatch':
4649
4683
  if (Array.isArray(this.to['scoring']?.['answer'])) {
4650
- if (this.to['scoring']?.['answer'].includes(this.originalValue)) {
4651
- // Answer is correct - set empty value in helper
4652
- this.helperFormControl.setValue('');
4684
+ // Handle multiselect case where answer is also an array
4685
+ if (Array.isArray(this.originalValue)) {
4686
+ // Check if not empty AND all selected answers are in expected answers
4687
+ const isCorrect = this.originalValue.length > 0 &&
4688
+ this.originalValue.every(a => this.to['scoring']?.['answer'].includes(a));
4689
+ if (isCorrect) {
4690
+ // Answer is correct - set empty value in helper
4691
+ this.helperFormControl.setValue('');
4692
+ }
4693
+ else {
4694
+ // Answer is incorrect - set defaultHelperText in helper
4695
+ this.helperFormControl.setValue(this.defaultHelperText);
4696
+ }
4653
4697
  }
4654
4698
  else {
4655
- // Answer is incorrect - set defaultHelperText in helper
4656
- this.helperFormControl.setValue(this.defaultHelperText);
4699
+ // Single value case
4700
+ if (this.to['scoring']?.['answer'].includes(this.originalValue)) {
4701
+ // Answer is correct - set empty value in helper
4702
+ this.helperFormControl.setValue('');
4703
+ }
4704
+ else {
4705
+ // Answer is incorrect - set defaultHelperText in helper
4706
+ this.helperFormControl.setValue(this.defaultHelperText);
4707
+ }
4657
4708
  }
4658
4709
  }
4659
4710
  break;
@@ -4677,7 +4728,7 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4677
4728
  }
4678
4729
  }
4679
4730
  static { this.ɵfac = i0.ɵɵngDeclareFactory({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MnlFormHelperTextInputComponent, deps: [{ token: i0.ChangeDetectorRef }], target: i0.ɵɵFactoryTarget.Component }); }
4680
- static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MnlFormHelperTextInputComponent, isStandalone: true, selector: "mu-helper-text", usesInheritance: true, ngImport: i0, template: `
4731
+ static { this.ɵcmp = i0.ɵɵngDeclareComponent({ minVersion: "14.0.0", version: "18.2.14", type: MnlFormHelperTextInputComponent, isStandalone: true, selector: "mu-helper-text", viewQueries: [{ propertyName: "autosize", first: true, predicate: ["autosize"], descendants: true }], usesInheritance: true, ngImport: i0, template: `
4681
4732
  <div class="helper-text-container">
4682
4733
  <!-- Helper Mode - Show toggle and conditional content -->
4683
4734
  <div *ngIf="to['enableHelper']">
@@ -4694,12 +4745,14 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4694
4745
 
4695
4746
  <mat-form-field class="textarea" appearance="outline" *ngIf="checked">
4696
4747
  <textarea
4748
+ cdkTextareaAutosize
4749
+ #autosize="cdkTextareaAutosize"
4750
+ cdkAutosizeMinRows="2"
4697
4751
  matInput
4698
4752
  [formControl]="helperFormControl"
4699
4753
  [readonly]="readonly || field.form?.disabled"
4700
4754
  [disabled]="field.form?.disabled"
4701
- >
4702
- </textarea>
4755
+ ></textarea>
4703
4756
  </mat-form-field>
4704
4757
  </div>
4705
4758
 
@@ -4709,7 +4762,7 @@ class MnlFormHelperTextInputComponent extends FieldType$1 {
4709
4762
  </formly-field>
4710
4763
  </div>
4711
4764
  </div>
4712
- `, isInline: true, 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"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i3$3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3$3.MatLabel, selector: "mat-label" }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i2$1.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "component", type: i6$3.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "ngmodule", type: MatNativeDateModule }] }); }
4765
+ `, isInline: true, 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"], dependencies: [{ kind: "ngmodule", type: CommonModule }, { kind: "directive", type: i1.NgIf, selector: "[ngIf]", inputs: ["ngIf", "ngIfThen", "ngIfElse"] }, { kind: "ngmodule", type: MatInputModule }, { kind: "directive", type: i2.MatInput, selector: "input[matInput], textarea[matInput], select[matNativeControl], input[matNativeControl], textarea[matNativeControl]", inputs: ["disabled", "id", "placeholder", "name", "required", "type", "errorStateMatcher", "aria-describedby", "value", "readonly"], exportAs: ["matInput"] }, { kind: "component", type: i3$3.MatFormField, selector: "mat-form-field", inputs: ["hideRequiredMarker", "color", "floatLabel", "appearance", "subscriptSizing", "hintLabel"], exportAs: ["matFormField"] }, { kind: "directive", type: i3$3.MatLabel, selector: "mat-label" }, { kind: "directive", type: i4$2.CdkTextareaAutosize, selector: "textarea[cdkTextareaAutosize]", inputs: ["cdkAutosizeMinRows", "cdkAutosizeMaxRows", "cdkTextareaAutosize", "placeholder"], exportAs: ["cdkTextareaAutosize"] }, { kind: "ngmodule", type: FormlyModule }, { kind: "component", type: i2$1.FormlyField, selector: "formly-field", inputs: ["field"] }, { kind: "ngmodule", type: ReactiveFormsModule }, { kind: "directive", type: i3.DefaultValueAccessor, selector: "input:not([type=checkbox])[formControlName],textarea[formControlName],input:not([type=checkbox])[formControl],textarea[formControl],input:not([type=checkbox])[ngModel],textarea[ngModel],[ngDefaultControl]" }, { kind: "directive", type: i3.NgControlStatus, selector: "[formControlName],[ngModel],[formControl]" }, { kind: "directive", type: i3.FormControlDirective, selector: "[formControl]", inputs: ["formControl", "disabled", "ngModel"], outputs: ["ngModelChange"], exportAs: ["ngForm"] }, { kind: "ngmodule", type: FormsModule }, { kind: "directive", type: i3.NgModel, selector: "[ngModel]:not([formControlName]):not([formControl])", inputs: ["name", "disabled", "ngModel", "ngModelOptions"], outputs: ["ngModelChange"], exportAs: ["ngModel"] }, { kind: "ngmodule", type: MatFormFieldModule }, { kind: "ngmodule", type: MatSlideToggleModule }, { kind: "component", type: i7$3.MatSlideToggle, selector: "mat-slide-toggle", inputs: ["name", "id", "labelPosition", "aria-label", "aria-labelledby", "aria-describedby", "required", "color", "disabled", "disableRipple", "tabIndex", "checked", "hideIcon", "disabledInteractive"], outputs: ["change", "toggleChange"], exportAs: ["matSlideToggle"] }, { kind: "ngmodule", type: MatSelectModule }, { kind: "ngmodule", type: MatDatepickerModule }, { kind: "ngmodule", type: MatNativeDateModule }] }); }
4713
4766
  }
4714
4767
  i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImport: i0, type: MnlFormHelperTextInputComponent, decorators: [{
4715
4768
  type: Component,
@@ -4741,12 +4794,14 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
4741
4794
 
4742
4795
  <mat-form-field class="textarea" appearance="outline" *ngIf="checked">
4743
4796
  <textarea
4797
+ cdkTextareaAutosize
4798
+ #autosize="cdkTextareaAutosize"
4799
+ cdkAutosizeMinRows="2"
4744
4800
  matInput
4745
4801
  [formControl]="helperFormControl"
4746
4802
  [readonly]="readonly || field.form?.disabled"
4747
4803
  [disabled]="field.form?.disabled"
4748
- >
4749
- </textarea>
4804
+ ></textarea>
4750
4805
  </mat-form-field>
4751
4806
  </div>
4752
4807
 
@@ -4757,7 +4812,10 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "18.2.14", ngImpo
4757
4812
  </div>
4758
4813
  </div>
4759
4814
  `, 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"] }]
4760
- }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }] });
4815
+ }], ctorParameters: () => [{ type: i0.ChangeDetectorRef }], propDecorators: { autosize: [{
4816
+ type: ViewChild,
4817
+ args: ['autosize']
4818
+ }] } });
4761
4819
 
4762
4820
  const mnlFormProviders = [
4763
4821
  importProvidersFrom(CommonsModule.forRoot(), CommonModule, ReactiveFormsModule, FormlyModule.forRoot({
@@ -4918,7 +4976,15 @@ class ScoringService {
4918
4976
  break;
4919
4977
  case 'exactMatch':
4920
4978
  if (Array.isArray(scoring.answer)) {
4921
- correct = scoring.answer.includes(answer);
4979
+ // Handle case where answer is also an array (multiselect)
4980
+ if (Array.isArray(answer)) {
4981
+ // Check if answer is not empty AND all selected answers are in the expected answers
4982
+ correct = answer.length > 0 && answer.every(a => scoring.answer.includes(a));
4983
+ }
4984
+ else {
4985
+ // Single value case
4986
+ correct = scoring.answer.includes(answer);
4987
+ }
4922
4988
  }
4923
4989
  break;
4924
4990
  case 'range':