@muraai/mnl-form 0.0.1-alpha-db0d938 → 0.0.1-alpha-f274a13

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