@piserve-tech/form-submission 1.3.272 → 1.3.273
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.
|
@@ -77,49 +77,67 @@ export class WhenClauseService {
|
|
|
77
77
|
}
|
|
78
78
|
}
|
|
79
79
|
evaluateCondition(answer, operator, value) {
|
|
80
|
-
|
|
80
|
+
// treat null/undefined/empty-string/empty-array as "no answer"
|
|
81
|
+
if (answer === null || answer === undefined)
|
|
81
82
|
return false;
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
83
|
+
if (typeof answer === 'string' && answer === '')
|
|
84
|
+
return false;
|
|
85
|
+
if (Array.isArray(answer) && answer.length === 0)
|
|
86
|
+
return false;
|
|
87
|
+
// helpers to normalize values to arrays of lower-case strings (for string ops)
|
|
88
|
+
const toStr = (v) => v === null || v === undefined ? '' : String(v).toLowerCase();
|
|
89
|
+
const ensureArray = (v) => {
|
|
90
|
+
if (Array.isArray(v))
|
|
91
|
+
return v.map((x) => toStr(x));
|
|
92
|
+
// if comma-separated string => split, else single element array
|
|
93
|
+
if (typeof v === 'string' && v.indexOf(',') !== -1) {
|
|
94
|
+
return v
|
|
95
|
+
.split(',')
|
|
96
|
+
.map((x) => x.trim())
|
|
97
|
+
.map((x) => toStr(x));
|
|
98
|
+
}
|
|
99
|
+
return [toStr(v)];
|
|
100
|
+
};
|
|
101
|
+
const ansArr = ensureArray(answer);
|
|
102
|
+
const valArr = ensureArray(value);
|
|
103
|
+
// numeric comparisons: only when both sides are scalars (non-array) and numeric
|
|
104
|
+
const isNumericScalar = (x) => !Array.isArray(x) && x !== null && x !== undefined && !Number.isNaN(Number(x));
|
|
85
105
|
switch (operator) {
|
|
86
|
-
// Number comparisons
|
|
87
106
|
case '<':
|
|
88
|
-
|
|
107
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
108
|
+
return Number(answer) < Number(value);
|
|
109
|
+
return false;
|
|
89
110
|
case '>':
|
|
90
|
-
|
|
111
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
112
|
+
return Number(answer) > Number(value);
|
|
113
|
+
return false;
|
|
91
114
|
case '<=':
|
|
92
|
-
|
|
115
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
116
|
+
return Number(answer) <= Number(value);
|
|
117
|
+
return false;
|
|
93
118
|
case '>=':
|
|
94
|
-
|
|
119
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
120
|
+
return Number(answer) >= Number(value);
|
|
121
|
+
return false;
|
|
122
|
+
// equality: true if any answer element equals any value element
|
|
95
123
|
case '=':
|
|
96
|
-
return
|
|
124
|
+
return ansArr.some((a) => valArr.some((v) => a === v));
|
|
97
125
|
case '!=':
|
|
98
|
-
return
|
|
99
|
-
//
|
|
126
|
+
return !ansArr.some((a) => valArr.some((v) => a === v));
|
|
127
|
+
// LIKE / NOT LIKE: check substring match anywhere in any answer element
|
|
100
128
|
case 'LIKE':
|
|
101
|
-
return
|
|
129
|
+
return ansArr.some((a) => valArr.some((v) => a.includes(v)));
|
|
102
130
|
case 'NOT LIKE':
|
|
103
|
-
return !
|
|
131
|
+
return !ansArr.some((a) => valArr.some((v) => a.includes(v)));
|
|
104
132
|
case 'BEGIN WITH':
|
|
105
|
-
return
|
|
133
|
+
return ansArr.some((a) => valArr.some((v) => a.startsWith(v)));
|
|
106
134
|
case 'END WITH':
|
|
107
|
-
return
|
|
108
|
-
//
|
|
135
|
+
return ansArr.some((a) => valArr.some((v) => a.endsWith(v)));
|
|
136
|
+
// IN / NOT IN: true if any answer element exists in value list
|
|
109
137
|
case 'IN':
|
|
110
|
-
return
|
|
111
|
-
? val.includes(ans)
|
|
112
|
-
: String(val)
|
|
113
|
-
.split(',')
|
|
114
|
-
.map((v) => v.trim())
|
|
115
|
-
.includes(String(ans));
|
|
138
|
+
return ansArr.some((a) => valArr.includes(a));
|
|
116
139
|
case 'NOT IN':
|
|
117
|
-
return
|
|
118
|
-
? !val.includes(ans)
|
|
119
|
-
: !String(val)
|
|
120
|
-
.split(',')
|
|
121
|
-
.map((v) => v.trim())
|
|
122
|
-
.includes(String(ans));
|
|
140
|
+
return !ansArr.some((a) => valArr.includes(a));
|
|
123
141
|
default:
|
|
124
142
|
return false;
|
|
125
143
|
}
|
|
@@ -292,4 +310,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
292
310
|
providedIn: 'root',
|
|
293
311
|
}]
|
|
294
312
|
}], ctorParameters: function () { return []; } });
|
|
295
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"whenClause.service.js","sourceRoot":"","sources":["../../../../projects/form-submission/src/services/whenClause.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Submission } from '../models/submission.model';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class WhenClauseService {\n  private activeWhenClauseGrids: any[] = [];\n  private questionAnswerMap: { [questionId: string]: any } = {};\n  isInitialLoad: boolean = true;\n  constructor() {}\n\n  markLoadCompleted() {\n    this.isInitialLoad = false;\n  }\n\n  /**\n   * Check if a grid has an active when clause\n   */\n  isActiveWhenClause(grid: any): boolean {\n    const when = grid?.whenClause;\n    return !!(\n      when &&\n      when.activate &&\n      Array.isArray(when.whenClauseCondition) &&\n      when.whenClauseCondition.length > 0\n    );\n  }\n\n  /**\n   * Add a grid to tracking list if it has an active when clause\n   */\n  addGrid(grid: any): void {\n    if (!grid) return;\n\n    if (this.isActiveWhenClause(grid)) {\n      const exists = this.activeWhenClauseGrids.some(\n        (g) => g.element?.id === grid.element?.id\n      );\n\n      if (!exists) {\n        this.activeWhenClauseGrids.push(grid);\n      }\n    }\n  }\n\n  setQuestionAnswer(questionId: string, answer: any,initial:boolean = false): void {\n    if (!questionId) return;\n    this.questionAnswerMap[questionId] = answer;\n    this.evaluateAllWhenClauses(initial);\n  }\n\n  getAnswer(questionId: string): any {\n    return this.questionAnswerMap[questionId];\n  }\n\n  private evaluateAllWhenClauses(initial:boolean): void {\n    const visibleGrids: any[] = [];\n\n    for (const grid of this.activeWhenClauseGrids) {\n      const when = grid.whenClause;\n      const conditions = when.whenClauseCondition;\n\n      if (!conditions || !conditions.length) continue;\n\n      let overallResult = true;\n\n      for (let i = 0; i < conditions.length; i++) {\n        const clause = conditions[i];\n        const logic = (clause.logic || (i === 0 ? 'AND' : 'AND')).toUpperCase(); // Default to AND for first\n        const answer = this.getAnswer(clause.questionid);\n        const result = this.evaluateCondition(\n          answer,\n          clause.condition,\n          clause.value\n        );\n\n        // Combine results based on logic (AND / OR)\n        if (i === 0) {\n          overallResult = result; // initialize with first condition\n        } else {\n          if (logic === 'AND') {\n            overallResult = overallResult && result;\n          } else if (logic === 'OR') {\n            overallResult = overallResult || result;\n          }\n        }\n      }\n\n      // ✅ If overallResult is true, show element; else hide\n      grid.show = overallResult;\n      if (!overallResult && !initial) {\n        grid.element.answer = null;\n      }\n      if (overallResult) visibleGrids.push(grid);\n    }\n  }\n\n  evaluateCondition(answer: any, operator: string, value: any): boolean {\n    if (answer === null || answer === undefined || answer === '') return false;\n\n    // Convert to lowercase for string comparison\n    const ans = typeof answer === 'string' ? answer.toLowerCase() : answer;\n    const val = typeof value === 'string' ? value.toLowerCase() : value;\n\n    switch (operator) {\n      // Number comparisons\n      case '<':\n        return Number(ans) < Number(val);\n      case '>':\n        return Number(ans) > Number(val);\n      case '<=':\n        return Number(ans) <= Number(val);\n      case '>=':\n        return Number(ans) >= Number(val);\n      case '=':\n        return ans == val;\n      case '!=':\n        return ans != val;\n\n      // String comparisons\n      case 'LIKE':\n        return ans.includes(val);\n      case 'NOT LIKE':\n        return !ans.includes(val);\n      case 'BEGIN WITH':\n        return ans.startsWith(val);\n      case 'END WITH':\n        return ans.endsWith(val);\n\n      // Array-based\n      case 'IN':\n        return Array.isArray(val)\n          ? val.includes(ans)\n          : String(val)\n              .split(',')\n              .map((v) => v.trim())\n              .includes(String(ans));\n      case 'NOT IN':\n        return Array.isArray(val)\n          ? !val.includes(ans)\n          : !String(val)\n              .split(',')\n              .map((v) => v.trim())\n              .includes(String(ans));\n\n      default:\n        return false;\n    }\n  }\n\n  addGridsFromSubform(subform: any): void {\n    if (!subform || !Array.isArray(subform.rows)) return;\n\n    for (const row of subform.rows) {\n      if (!row.grid || !Array.isArray(row.grid)) continue;\n\n      for (const grid of row.grid) {\n        // 1️⃣ Add this grid itself if it has an active whenClause\n        this.addGrid(grid);\n\n        // 2️⃣ Then, recursively process nested childLogics inside its element\n        if (grid.element?.childLogics?.length) {\n          this.processChildLogics(grid.element.childLogics);\n        }\n      }\n    }\n  }\n\n  /**\n   * Recursively process childLogics structure\n   */\n  private processChildLogics(childLogics: any[]): void {\n    if (!Array.isArray(childLogics)) return;\n\n    for (const logic of childLogics) {\n      // 1️⃣ Each logic may have its own rows (similar to subform rows)\n      if (Array.isArray(logic.rows)) {\n        for (const row of logic.rows) {\n          if (Array.isArray(row.grid)) {\n            for (const grid of row.grid) {\n              // Check whenClause for this nested grid\n              this.addGrid(grid);\n\n              // 2️⃣ Recurse again if this nested question also has its own childLogics\n              if (grid.element?.childLogics?.length) {\n                this.processChildLogics(grid.element.childLogics);\n              }\n            }\n          }\n        }\n      }\n\n      // 3️⃣ A logic itself might directly contain nested logics\n      if (Array.isArray(logic.childLogics)) {\n        this.processChildLogics(logic.childLogics);\n      }\n    }\n  }\n\n  /**\n   * Return ALL elements with an active whenClause inside a subform,\n   * including nested childLogics (recursive)\n   */\n  getWhenClauseElements(subform: any): any[] {\n    const result: any[] = [];\n\n    if (!subform || !Array.isArray(subform.rows)) {\n      return result;\n    }\n\n    for (const row of subform.rows) {\n      if (!Array.isArray(row.grid)) continue;\n\n      for (const grid of row.grid) {\n        // 1️⃣ Add this grid if it has an active whenClause\n        if (this.isActiveWhenClause(grid)) {\n          result.push(grid);\n        }\n\n        // 2️⃣ Process nested childLogics inside this element\n        if (Array.isArray(grid.element?.childLogics)) {\n          result.push(\n            ...this.getElementsFromChildLogics(grid.element.childLogics)\n          );\n        }\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Recursively gather grids with active whenClause inside childLogics\n   */\n  private getElementsFromChildLogics(childLogics: any[]): any[] {\n    const found: any[] = [];\n\n    if (!Array.isArray(childLogics)) return found;\n\n    for (const logic of childLogics) {\n      // Each logic can have rows\n      if (Array.isArray(logic.rows)) {\n        for (const row of logic.rows) {\n          if (!Array.isArray(row.grid)) continue;\n\n          for (const grid of row.grid) {\n            if (this.isActiveWhenClause(grid)) {\n              found.push(grid);\n            }\n\n            // Recurse deeper if this grid has nested childLogics\n            if (Array.isArray(grid.element?.childLogics)) {\n              found.push(\n                ...this.getElementsFromChildLogics(grid.element.childLogics)\n              );\n            }\n          }\n        }\n      }\n\n      // Logic itself may have nested childLogics\n      if (Array.isArray(logic.childLogics)) {\n        found.push(...this.getElementsFromChildLogics(logic.childLogics));\n      }\n    }\n\n    return found;\n  }\n\n  checkElementVisibility(id: string, submission: Submission): boolean {\n    if (!id || !submission || !Array.isArray(submission.answers)) {\n      return true; // default visible if invalid input\n    }\n\n    const matchedAnswer = submission.answers.find(\n      (ans) => ans.questionId === id\n    );\n\n    // Return the 'show' flag if found, else default to true\n    return matchedAnswer ? matchedAnswer.show ?? true : true;\n  }\n\n  evaluateSubformWhenClauses(\n    changedAnswer: any,\n    submission: Submission,\n    changedQuestionId: string,\n    elementsWithWhenClause: any\n  ) {\n    if (!elementsWithWhenClause.length) return;\n\n    for (const grid of elementsWithWhenClause) {\n      const elementId = grid.element.id;\n      const when = grid.whenClause;\n      const conditions = when.whenClauseCondition;\n\n      if (!conditions?.length) continue;\n\n      let overallResult = true;\n\n      for (let i = 0; i < conditions.length; i++) {\n        const clause = conditions[i];\n\n        const logic = (clause.logic || (i === 0 ? 'AND' : 'AND')).toUpperCase();\n\n        // Determine which answer to use for this condition\n        let answerToCheck: any;\n\n        if (clause.questionid === changedQuestionId) {\n          // Use updated answer for the changed question\n          answerToCheck = changedAnswer;\n        } else {\n          // Fetch existing value from submission for other questions\n          const existing = submission.answers.find(\n            (a) => a.questionId === clause.questionid\n          );\n          answerToCheck = existing?.answer;\n        }\n\n        const result = this.evaluateCondition(\n          answerToCheck,\n          clause.condition,\n          clause.value\n        );\n\n        // Apply AND/OR logic\n        if (i === 0) {\n          overallResult = result;\n        } else {\n          overallResult =\n            logic === 'AND' ? overallResult && result : overallResult || result;\n        }\n      }\n\n      // Update show flag for the element controlled by this whenClause\n      const matchedAnswer = submission.answers.find(\n        (ans) => ans.questionId === elementId\n      );\n      if (matchedAnswer) {\n        matchedAnswer.show = overallResult;\n      }\n    }\n  }\n\n  resetService(): void {\n    // Clear tracked grids\n    this.activeWhenClauseGrids = [];\n  \n    // Clear stored question → answer map\n    this.questionAnswerMap = {};\n  \n    // Reset visibility logic to initial state\n  }\n\n  getActiveWhenClauseGrids(): any[] {\n    return this.activeWhenClauseGrids;\n  }\n}\n"]}
|
|
313
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"whenClause.service.js","sourceRoot":"","sources":["../../../../projects/form-submission/src/services/whenClause.service.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,UAAU,EAAE,MAAM,eAAe,CAAC;;AAM3C,MAAM,OAAO,iBAAiB;IAI5B;QAHQ,0BAAqB,GAAU,EAAE,CAAC;QAClC,sBAAiB,GAAkC,EAAE,CAAC;QAC9D,kBAAa,GAAY,IAAI,CAAC;IACf,CAAC;IAEhB,iBAAiB;QACf,IAAI,CAAC,aAAa,GAAG,KAAK,CAAC;IAC7B,CAAC;IAED;;OAEG;IACH,kBAAkB,CAAC,IAAS;QAC1B,MAAM,IAAI,GAAG,IAAI,EAAE,UAAU,CAAC;QAC9B,OAAO,CAAC,CAAC,CACP,IAAI;YACJ,IAAI,CAAC,QAAQ;YACb,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,mBAAmB,CAAC;YACvC,IAAI,CAAC,mBAAmB,CAAC,MAAM,GAAG,CAAC,CACpC,CAAC;IACJ,CAAC;IAED;;OAEG;IACH,OAAO,CAAC,IAAS;QACf,IAAI,CAAC,IAAI;YAAE,OAAO;QAElB,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE;YACjC,MAAM,MAAM,GAAG,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAC5C,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,OAAO,EAAE,EAAE,KAAK,IAAI,CAAC,OAAO,EAAE,EAAE,CAC1C,CAAC;YAEF,IAAI,CAAC,MAAM,EAAE;gBACX,IAAI,CAAC,qBAAqB,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;aACvC;SACF;IACH,CAAC;IAED,iBAAiB,CAAC,UAAkB,EAAE,MAAW,EAAC,UAAkB,KAAK;QACvE,IAAI,CAAC,UAAU;YAAE,OAAO;QACxB,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,GAAG,MAAM,CAAC;QAC5C,IAAI,CAAC,sBAAsB,CAAC,OAAO,CAAC,CAAC;IACvC,CAAC;IAED,SAAS,CAAC,UAAkB;QAC1B,OAAO,IAAI,CAAC,iBAAiB,CAAC,UAAU,CAAC,CAAC;IAC5C,CAAC;IAEO,sBAAsB,CAAC,OAAe;QAC5C,MAAM,YAAY,GAAU,EAAE,CAAC;QAE/B,KAAK,MAAM,IAAI,IAAI,IAAI,CAAC,qBAAqB,EAAE;YAC7C,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC;YAC7B,MAAM,UAAU,GAAG,IAAI,CAAC,mBAAmB,CAAC;YAE5C,IAAI,CAAC,UAAU,IAAI,CAAC,UAAU,CAAC,MAAM;gBAAE,SAAS;YAEhD,IAAI,aAAa,GAAG,IAAI,CAAC;YAEzB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;gBAC1C,MAAM,MAAM,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;gBAC7B,MAAM,KAAK,GAAG,CAAC,MAAM,CAAC,KAAK,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC,CAAC,2BAA2B;gBACpG,MAAM,MAAM,GAAG,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,UAAU,CAAC,CAAC;gBACjD,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CACnC,MAAM,EACN,MAAM,CAAC,SAAS,EAChB,MAAM,CAAC,KAAK,CACb,CAAC;gBAEF,4CAA4C;gBAC5C,IAAI,CAAC,KAAK,CAAC,EAAE;oBACX,aAAa,GAAG,MAAM,CAAC,CAAC,kCAAkC;iBAC3D;qBAAM;oBACL,IAAI,KAAK,KAAK,KAAK,EAAE;wBACnB,aAAa,GAAG,aAAa,IAAI,MAAM,CAAC;qBACzC;yBAAM,IAAI,KAAK,KAAK,IAAI,EAAE;wBACzB,aAAa,GAAG,aAAa,IAAI,MAAM,CAAC;qBACzC;iBACF;aACF;YAED,sDAAsD;YACtD,IAAI,CAAC,IAAI,GAAG,aAAa,CAAC;YAC1B,IAAI,CAAC,aAAa,IAAI,CAAC,OAAO,EAAE;gBAC9B,IAAI,CAAC,OAAO,CAAC,MAAM,GAAG,IAAI,CAAC;aAC5B;YACD,IAAI,aAAa;gBAAE,YAAY,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;SAC5C;IACH,CAAC;IAED,iBAAiB,CAAC,MAAW,EAAE,QAAgB,EAAE,KAAU;QAEzD,+DAA+D;QAC/D,IAAI,MAAM,KAAK,IAAI,IAAI,MAAM,KAAK,SAAS;YAAE,OAAO,KAAK,CAAC;QAC1D,IAAI,OAAO,MAAM,KAAK,QAAQ,IAAI,MAAM,KAAK,EAAE;YAAE,OAAO,KAAK,CAAC;QAC9D,IAAI,KAAK,CAAC,OAAO,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,MAAM,KAAK,CAAC;YAAE,OAAO,KAAK,CAAC;QAE/D,+EAA+E;QAC/E,MAAM,KAAK,GAAG,CAAC,CAAM,EAAE,EAAE,CACvB,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,SAAS,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;QAE/D,MAAM,WAAW,GAAG,CAAC,CAAM,EAAY,EAAE;YACvC,IAAI,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC;gBAAE,OAAO,CAAC,CAAC,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;YACpD,gEAAgE;YAChE,IAAI,OAAO,CAAC,KAAK,QAAQ,IAAI,CAAC,CAAC,OAAO,CAAC,GAAG,CAAC,KAAK,CAAC,CAAC,EAAE;gBAClD,OAAO,CAAC;qBACL,KAAK,CAAC,GAAG,CAAC;qBACV,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,EAAE,CAAC;qBACpB,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;aACzB;YACD,OAAO,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC;QACpB,CAAC,CAAC;QAEF,MAAM,MAAM,GAAG,WAAW,CAAC,MAAM,CAAC,CAAC;QACnC,MAAM,MAAM,GAAG,WAAW,CAAC,KAAK,CAAC,CAAC;QAElC,gFAAgF;QAChF,MAAM,eAAe,GAAG,CAAC,CAAM,EAAE,EAAE,CACjC,CAAC,KAAK,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAAC,KAAK,IAAI,IAAI,CAAC,KAAK,SAAS,IAAI,CAAC,MAAM,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC,CAAC,CAAC,CAAC;QAEjF,QAAQ,QAAQ,EAAE;YAChB,KAAK,GAAG;gBACN,IAAI,eAAe,CAAC,MAAM,CAAC,IAAI,eAAe,CAAC,KAAK,CAAC;oBACnD,OAAO,MAAM,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBACxC,OAAO,KAAK,CAAC;YACf,KAAK,GAAG;gBACN,IAAI,eAAe,CAAC,MAAM,CAAC,IAAI,eAAe,CAAC,KAAK,CAAC;oBACnD,OAAO,MAAM,CAAC,MAAM,CAAC,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC;gBACxC,OAAO,KAAK,CAAC;YACf,KAAK,IAAI;gBACP,IAAI,eAAe,CAAC,MAAM,CAAC,IAAI,eAAe,CAAC,KAAK,CAAC;oBACnD,OAAO,MAAM,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;gBACzC,OAAO,KAAK,CAAC;YACf,KAAK,IAAI;gBACP,IAAI,eAAe,CAAC,MAAM,CAAC,IAAI,eAAe,CAAC,KAAK,CAAC;oBACnD,OAAO,MAAM,CAAC,MAAM,CAAC,IAAI,MAAM,CAAC,KAAK,CAAC,CAAC;gBACzC,OAAO,KAAK,CAAC;YAEf,gEAAgE;YAChE,KAAK,GAAG;gBACN,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAEzD,KAAK,IAAI;gBACP,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC;YAE1D,wEAAwE;YACxE,KAAK,MAAM;gBACT,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAC/D,KAAK,UAAU;gBACb,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAEhE,KAAK,YAAY;gBACf,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YACjE,KAAK,UAAU;gBACb,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;YAE/D,+DAA+D;YAC/D,KAAK,IAAI;gBACP,OAAO,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YAChD,KAAK,QAAQ;gBACX,OAAO,CAAC,MAAM,CAAC,IAAI,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC,CAAC,CAAC,CAAC;YAEjD;gBACE,OAAO,KAAK,CAAC;SAChB;IACH,CAAC;IAED,mBAAmB,CAAC,OAAY;QAC9B,IAAI,CAAC,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC;YAAE,OAAO;QAErD,KAAK,MAAM,GAAG,IAAI,OAAO,CAAC,IAAI,EAAE;YAC9B,IAAI,CAAC,GAAG,CAAC,IAAI,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC;gBAAE,SAAS;YAEpD,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,IAAI,EAAE;gBAC3B,0DAA0D;gBAC1D,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;gBAEnB,sEAAsE;gBACtE,IAAI,IAAI,CAAC,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE;oBACrC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;iBACnD;aACF;SACF;IACH,CAAC;IAED;;OAEG;IACK,kBAAkB,CAAC,WAAkB;QAC3C,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC;YAAE,OAAO;QAExC,KAAK,MAAM,KAAK,IAAI,WAAW,EAAE;YAC/B,iEAAiE;YACjE,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE;gBAC7B,KAAK,MAAM,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE;oBAC5B,IAAI,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC,EAAE;wBAC3B,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,IAAI,EAAE;4BAC3B,wCAAwC;4BACxC,IAAI,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;4BAEnB,yEAAyE;4BACzE,IAAI,IAAI,CAAC,OAAO,EAAE,WAAW,EAAE,MAAM,EAAE;gCACrC,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;6BACnD;yBACF;qBACF;iBACF;aACF;YAED,0DAA0D;YAC1D,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,EAAE;gBACpC,IAAI,CAAC,kBAAkB,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC;aAC5C;SACF;IACH,CAAC;IAED;;;OAGG;IACH,qBAAqB,CAAC,OAAY;QAChC,MAAM,MAAM,GAAU,EAAE,CAAC;QAEzB,IAAI,CAAC,OAAO,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,OAAO,CAAC,IAAI,CAAC,EAAE;YAC5C,OAAO,MAAM,CAAC;SACf;QAED,KAAK,MAAM,GAAG,IAAI,OAAO,CAAC,IAAI,EAAE;YAC9B,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC;gBAAE,SAAS;YAEvC,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,IAAI,EAAE;gBAC3B,mDAAmD;gBACnD,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE;oBACjC,MAAM,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;iBACnB;gBAED,qDAAqD;gBACrD,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,WAAW,CAAC,EAAE;oBAC5C,MAAM,CAAC,IAAI,CACT,GAAG,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAC7D,CAAC;iBACH;aACF;SACF;QAED,OAAO,MAAM,CAAC;IAChB,CAAC;IAED;;OAEG;IACK,0BAA0B,CAAC,WAAkB;QACnD,MAAM,KAAK,GAAU,EAAE,CAAC;QAExB,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC;YAAE,OAAO,KAAK,CAAC;QAE9C,KAAK,MAAM,KAAK,IAAI,WAAW,EAAE;YAC/B,2BAA2B;YAC3B,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,EAAE;gBAC7B,KAAK,MAAM,GAAG,IAAI,KAAK,CAAC,IAAI,EAAE;oBAC5B,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,GAAG,CAAC,IAAI,CAAC;wBAAE,SAAS;oBAEvC,KAAK,MAAM,IAAI,IAAI,GAAG,CAAC,IAAI,EAAE;wBAC3B,IAAI,IAAI,CAAC,kBAAkB,CAAC,IAAI,CAAC,EAAE;4BACjC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;yBAClB;wBAED,qDAAqD;wBACrD,IAAI,KAAK,CAAC,OAAO,CAAC,IAAI,CAAC,OAAO,EAAE,WAAW,CAAC,EAAE;4BAC5C,KAAK,CAAC,IAAI,CACR,GAAG,IAAI,CAAC,0BAA0B,CAAC,IAAI,CAAC,OAAO,CAAC,WAAW,CAAC,CAC7D,CAAC;yBACH;qBACF;iBACF;aACF;YAED,2CAA2C;YAC3C,IAAI,KAAK,CAAC,OAAO,CAAC,KAAK,CAAC,WAAW,CAAC,EAAE;gBACpC,KAAK,CAAC,IAAI,CAAC,GAAG,IAAI,CAAC,0BAA0B,CAAC,KAAK,CAAC,WAAW,CAAC,CAAC,CAAC;aACnE;SACF;QAED,OAAO,KAAK,CAAC;IACf,CAAC;IAED,sBAAsB,CAAC,EAAU,EAAE,UAAsB;QACvD,IAAI,CAAC,EAAE,IAAI,CAAC,UAAU,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE;YAC5D,OAAO,IAAI,CAAC,CAAC,mCAAmC;SACjD;QAED,MAAM,aAAa,GAAG,UAAU,CAAC,OAAO,CAAC,IAAI,CAC3C,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,UAAU,KAAK,EAAE,CAC/B,CAAC;QAEF,wDAAwD;QACxD,OAAO,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC,IAAI,IAAI,IAAI,CAAC,CAAC,CAAC,IAAI,CAAC;IAC3D,CAAC;IAED,0BAA0B,CACxB,aAAkB,EAClB,UAAsB,EACtB,iBAAyB,EACzB,sBAA2B;QAE3B,IAAI,CAAC,sBAAsB,CAAC,MAAM;YAAE,OAAO;QAE3C,KAAK,MAAM,IAAI,IAAI,sBAAsB,EAAE;YACzC,MAAM,SAAS,GAAG,IAAI,CAAC,OAAO,CAAC,EAAE,CAAC;YAClC,MAAM,IAAI,GAAG,IAAI,CAAC,UAAU,CAAC;YAC7B,MAAM,UAAU,GAAG,IAAI,CAAC,mBAAmB,CAAC;YAE5C,IAAI,CAAC,UAAU,EAAE,MAAM;gBAAE,SAAS;YAElC,IAAI,aAAa,GAAG,IAAI,CAAC;YAEzB,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,UAAU,CAAC,MAAM,EAAE,CAAC,EAAE,EAAE;gBAC1C,MAAM,MAAM,GAAG,UAAU,CAAC,CAAC,CAAC,CAAC;gBAE7B,MAAM,KAAK,GAAG,CAAC,MAAM,CAAC,KAAK,IAAI,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,WAAW,EAAE,CAAC;gBAExE,mDAAmD;gBACnD,IAAI,aAAkB,CAAC;gBAEvB,IAAI,MAAM,CAAC,UAAU,KAAK,iBAAiB,EAAE;oBAC3C,8CAA8C;oBAC9C,aAAa,GAAG,aAAa,CAAC;iBAC/B;qBAAM;oBACL,2DAA2D;oBAC3D,MAAM,QAAQ,GAAG,UAAU,CAAC,OAAO,CAAC,IAAI,CACtC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,CAAC,UAAU,KAAK,MAAM,CAAC,UAAU,CAC1C,CAAC;oBACF,aAAa,GAAG,QAAQ,EAAE,MAAM,CAAC;iBAClC;gBAED,MAAM,MAAM,GAAG,IAAI,CAAC,iBAAiB,CACnC,aAAa,EACb,MAAM,CAAC,SAAS,EAChB,MAAM,CAAC,KAAK,CACb,CAAC;gBAEF,qBAAqB;gBACrB,IAAI,CAAC,KAAK,CAAC,EAAE;oBACX,aAAa,GAAG,MAAM,CAAC;iBACxB;qBAAM;oBACL,aAAa;wBACX,KAAK,KAAK,KAAK,CAAC,CAAC,CAAC,aAAa,IAAI,MAAM,CAAC,CAAC,CAAC,aAAa,IAAI,MAAM,CAAC;iBACvE;aACF;YAED,iEAAiE;YACjE,MAAM,aAAa,GAAG,UAAU,CAAC,OAAO,CAAC,IAAI,CAC3C,CAAC,GAAG,EAAE,EAAE,CAAC,GAAG,CAAC,UAAU,KAAK,SAAS,CACtC,CAAC;YACF,IAAI,aAAa,EAAE;gBACjB,aAAa,CAAC,IAAI,GAAG,aAAa,CAAC;aACpC;SACF;IACH,CAAC;IAED,YAAY;QACV,sBAAsB;QACtB,IAAI,CAAC,qBAAqB,GAAG,EAAE,CAAC;QAEhC,qCAAqC;QACrC,IAAI,CAAC,iBAAiB,GAAG,EAAE,CAAC;QAE5B,0CAA0C;IAC5C,CAAC;IAED,wBAAwB;QACtB,OAAO,IAAI,CAAC,qBAAqB,CAAC;IACpC,CAAC;+GAtXU,iBAAiB;mHAAjB,iBAAiB,cAFhB,MAAM;;4FAEP,iBAAiB;kBAH7B,UAAU;mBAAC;oBACV,UAAU,EAAE,MAAM;iBACnB","sourcesContent":["import { Injectable } from '@angular/core';\nimport { Submission } from '../models/submission.model';\n\n@Injectable({\n  providedIn: 'root',\n})\nexport class WhenClauseService {\n  private activeWhenClauseGrids: any[] = [];\n  private questionAnswerMap: { [questionId: string]: any } = {};\n  isInitialLoad: boolean = true;\n  constructor() {}\n\n  markLoadCompleted() {\n    this.isInitialLoad = false;\n  }\n\n  /**\n   * Check if a grid has an active when clause\n   */\n  isActiveWhenClause(grid: any): boolean {\n    const when = grid?.whenClause;\n    return !!(\n      when &&\n      when.activate &&\n      Array.isArray(when.whenClauseCondition) &&\n      when.whenClauseCondition.length > 0\n    );\n  }\n\n  /**\n   * Add a grid to tracking list if it has an active when clause\n   */\n  addGrid(grid: any): void {\n    if (!grid) return;\n\n    if (this.isActiveWhenClause(grid)) {\n      const exists = this.activeWhenClauseGrids.some(\n        (g) => g.element?.id === grid.element?.id\n      );\n\n      if (!exists) {\n        this.activeWhenClauseGrids.push(grid);\n      }\n    }\n  }\n\n  setQuestionAnswer(questionId: string, answer: any,initial:boolean = false): void {\n    if (!questionId) return;\n    this.questionAnswerMap[questionId] = answer;\n    this.evaluateAllWhenClauses(initial);\n  }\n\n  getAnswer(questionId: string): any {\n    return this.questionAnswerMap[questionId];\n  }\n\n  private evaluateAllWhenClauses(initial:boolean): void {\n    const visibleGrids: any[] = [];\n\n    for (const grid of this.activeWhenClauseGrids) {\n      const when = grid.whenClause;\n      const conditions = when.whenClauseCondition;\n\n      if (!conditions || !conditions.length) continue;\n\n      let overallResult = true;\n\n      for (let i = 0; i < conditions.length; i++) {\n        const clause = conditions[i];\n        const logic = (clause.logic || (i === 0 ? 'AND' : 'AND')).toUpperCase(); // Default to AND for first\n        const answer = this.getAnswer(clause.questionid);\n        const result = this.evaluateCondition(\n          answer,\n          clause.condition,\n          clause.value\n        );\n\n        // Combine results based on logic (AND / OR)\n        if (i === 0) {\n          overallResult = result; // initialize with first condition\n        } else {\n          if (logic === 'AND') {\n            overallResult = overallResult && result;\n          } else if (logic === 'OR') {\n            overallResult = overallResult || result;\n          }\n        }\n      }\n\n      // ✅ If overallResult is true, show element; else hide\n      grid.show = overallResult;\n      if (!overallResult && !initial) {\n        grid.element.answer = null;\n      }\n      if (overallResult) visibleGrids.push(grid);\n    }\n  }\n\n  evaluateCondition(answer: any, operator: string, value: any): boolean {\n  \n    // treat null/undefined/empty-string/empty-array as \"no answer\"\n    if (answer === null || answer === undefined) return false;\n    if (typeof answer === 'string' && answer === '') return false;\n    if (Array.isArray(answer) && answer.length === 0) return false;\n  \n    // helpers to normalize values to arrays of lower-case strings (for string ops)\n    const toStr = (v: any) =>\n      v === null || v === undefined ? '' : String(v).toLowerCase();\n  \n    const ensureArray = (v: any): string[] => {\n      if (Array.isArray(v)) return v.map((x) => toStr(x));\n      // if comma-separated string => split, else single element array\n      if (typeof v === 'string' && v.indexOf(',') !== -1) {\n        return v\n          .split(',')\n          .map((x) => x.trim())\n          .map((x) => toStr(x));\n      }\n      return [toStr(v)];\n    };\n  \n    const ansArr = ensureArray(answer);\n    const valArr = ensureArray(value);\n  \n    // numeric comparisons: only when both sides are scalars (non-array) and numeric\n    const isNumericScalar = (x: any) =>\n      !Array.isArray(x) && x !== null && x !== undefined && !Number.isNaN(Number(x));\n  \n    switch (operator) {\n      case '<':\n        if (isNumericScalar(answer) && isNumericScalar(value))\n          return Number(answer) < Number(value);\n        return false;\n      case '>':\n        if (isNumericScalar(answer) && isNumericScalar(value))\n          return Number(answer) > Number(value);\n        return false;\n      case '<=':\n        if (isNumericScalar(answer) && isNumericScalar(value))\n          return Number(answer) <= Number(value);\n        return false;\n      case '>=':\n        if (isNumericScalar(answer) && isNumericScalar(value))\n          return Number(answer) >= Number(value);\n        return false;\n  \n      // equality: true if any answer element equals any value element\n      case '=':\n        return ansArr.some((a) => valArr.some((v) => a === v));\n  \n      case '!=':\n        return !ansArr.some((a) => valArr.some((v) => a === v));\n  \n      // LIKE / NOT LIKE: check substring match anywhere in any answer element\n      case 'LIKE':\n        return ansArr.some((a) => valArr.some((v) => a.includes(v)));\n      case 'NOT LIKE':\n        return !ansArr.some((a) => valArr.some((v) => a.includes(v)));\n  \n      case 'BEGIN WITH':\n        return ansArr.some((a) => valArr.some((v) => a.startsWith(v)));\n      case 'END WITH':\n        return ansArr.some((a) => valArr.some((v) => a.endsWith(v)));\n  \n      // IN / NOT IN: true if any answer element exists in value list\n      case 'IN':\n        return ansArr.some((a) => valArr.includes(a));\n      case 'NOT IN':\n        return !ansArr.some((a) => valArr.includes(a));\n  \n      default:\n        return false;\n    }\n  }  \n\n  addGridsFromSubform(subform: any): void {\n    if (!subform || !Array.isArray(subform.rows)) return;\n\n    for (const row of subform.rows) {\n      if (!row.grid || !Array.isArray(row.grid)) continue;\n\n      for (const grid of row.grid) {\n        // 1️⃣ Add this grid itself if it has an active whenClause\n        this.addGrid(grid);\n\n        // 2️⃣ Then, recursively process nested childLogics inside its element\n        if (grid.element?.childLogics?.length) {\n          this.processChildLogics(grid.element.childLogics);\n        }\n      }\n    }\n  }\n\n  /**\n   * Recursively process childLogics structure\n   */\n  private processChildLogics(childLogics: any[]): void {\n    if (!Array.isArray(childLogics)) return;\n\n    for (const logic of childLogics) {\n      // 1️⃣ Each logic may have its own rows (similar to subform rows)\n      if (Array.isArray(logic.rows)) {\n        for (const row of logic.rows) {\n          if (Array.isArray(row.grid)) {\n            for (const grid of row.grid) {\n              // Check whenClause for this nested grid\n              this.addGrid(grid);\n\n              // 2️⃣ Recurse again if this nested question also has its own childLogics\n              if (grid.element?.childLogics?.length) {\n                this.processChildLogics(grid.element.childLogics);\n              }\n            }\n          }\n        }\n      }\n\n      // 3️⃣ A logic itself might directly contain nested logics\n      if (Array.isArray(logic.childLogics)) {\n        this.processChildLogics(logic.childLogics);\n      }\n    }\n  }\n\n  /**\n   * Return ALL elements with an active whenClause inside a subform,\n   * including nested childLogics (recursive)\n   */\n  getWhenClauseElements(subform: any): any[] {\n    const result: any[] = [];\n\n    if (!subform || !Array.isArray(subform.rows)) {\n      return result;\n    }\n\n    for (const row of subform.rows) {\n      if (!Array.isArray(row.grid)) continue;\n\n      for (const grid of row.grid) {\n        // 1️⃣ Add this grid if it has an active whenClause\n        if (this.isActiveWhenClause(grid)) {\n          result.push(grid);\n        }\n\n        // 2️⃣ Process nested childLogics inside this element\n        if (Array.isArray(grid.element?.childLogics)) {\n          result.push(\n            ...this.getElementsFromChildLogics(grid.element.childLogics)\n          );\n        }\n      }\n    }\n\n    return result;\n  }\n\n  /**\n   * Recursively gather grids with active whenClause inside childLogics\n   */\n  private getElementsFromChildLogics(childLogics: any[]): any[] {\n    const found: any[] = [];\n\n    if (!Array.isArray(childLogics)) return found;\n\n    for (const logic of childLogics) {\n      // Each logic can have rows\n      if (Array.isArray(logic.rows)) {\n        for (const row of logic.rows) {\n          if (!Array.isArray(row.grid)) continue;\n\n          for (const grid of row.grid) {\n            if (this.isActiveWhenClause(grid)) {\n              found.push(grid);\n            }\n\n            // Recurse deeper if this grid has nested childLogics\n            if (Array.isArray(grid.element?.childLogics)) {\n              found.push(\n                ...this.getElementsFromChildLogics(grid.element.childLogics)\n              );\n            }\n          }\n        }\n      }\n\n      // Logic itself may have nested childLogics\n      if (Array.isArray(logic.childLogics)) {\n        found.push(...this.getElementsFromChildLogics(logic.childLogics));\n      }\n    }\n\n    return found;\n  }\n\n  checkElementVisibility(id: string, submission: Submission): boolean {\n    if (!id || !submission || !Array.isArray(submission.answers)) {\n      return true; // default visible if invalid input\n    }\n\n    const matchedAnswer = submission.answers.find(\n      (ans) => ans.questionId === id\n    );\n\n    // Return the 'show' flag if found, else default to true\n    return matchedAnswer ? matchedAnswer.show ?? true : true;\n  }\n\n  evaluateSubformWhenClauses(\n    changedAnswer: any,\n    submission: Submission,\n    changedQuestionId: string,\n    elementsWithWhenClause: any\n  ) {\n    if (!elementsWithWhenClause.length) return;\n\n    for (const grid of elementsWithWhenClause) {\n      const elementId = grid.element.id;\n      const when = grid.whenClause;\n      const conditions = when.whenClauseCondition;\n\n      if (!conditions?.length) continue;\n\n      let overallResult = true;\n\n      for (let i = 0; i < conditions.length; i++) {\n        const clause = conditions[i];\n\n        const logic = (clause.logic || (i === 0 ? 'AND' : 'AND')).toUpperCase();\n\n        // Determine which answer to use for this condition\n        let answerToCheck: any;\n\n        if (clause.questionid === changedQuestionId) {\n          // Use updated answer for the changed question\n          answerToCheck = changedAnswer;\n        } else {\n          // Fetch existing value from submission for other questions\n          const existing = submission.answers.find(\n            (a) => a.questionId === clause.questionid\n          );\n          answerToCheck = existing?.answer;\n        }\n\n        const result = this.evaluateCondition(\n          answerToCheck,\n          clause.condition,\n          clause.value\n        );\n\n        // Apply AND/OR logic\n        if (i === 0) {\n          overallResult = result;\n        } else {\n          overallResult =\n            logic === 'AND' ? overallResult && result : overallResult || result;\n        }\n      }\n\n      // Update show flag for the element controlled by this whenClause\n      const matchedAnswer = submission.answers.find(\n        (ans) => ans.questionId === elementId\n      );\n      if (matchedAnswer) {\n        matchedAnswer.show = overallResult;\n      }\n    }\n  }\n\n  resetService(): void {\n    // Clear tracked grids\n    this.activeWhenClauseGrids = [];\n  \n    // Clear stored question → answer map\n    this.questionAnswerMap = {};\n  \n    // Reset visibility logic to initial state\n  }\n\n  getActiveWhenClauseGrids(): any[] {\n    return this.activeWhenClauseGrids;\n  }\n}\n"]}
|
|
@@ -786,49 +786,67 @@ class WhenClauseService {
|
|
|
786
786
|
}
|
|
787
787
|
}
|
|
788
788
|
evaluateCondition(answer, operator, value) {
|
|
789
|
-
|
|
789
|
+
// treat null/undefined/empty-string/empty-array as "no answer"
|
|
790
|
+
if (answer === null || answer === undefined)
|
|
790
791
|
return false;
|
|
791
|
-
|
|
792
|
-
|
|
793
|
-
|
|
792
|
+
if (typeof answer === 'string' && answer === '')
|
|
793
|
+
return false;
|
|
794
|
+
if (Array.isArray(answer) && answer.length === 0)
|
|
795
|
+
return false;
|
|
796
|
+
// helpers to normalize values to arrays of lower-case strings (for string ops)
|
|
797
|
+
const toStr = (v) => v === null || v === undefined ? '' : String(v).toLowerCase();
|
|
798
|
+
const ensureArray = (v) => {
|
|
799
|
+
if (Array.isArray(v))
|
|
800
|
+
return v.map((x) => toStr(x));
|
|
801
|
+
// if comma-separated string => split, else single element array
|
|
802
|
+
if (typeof v === 'string' && v.indexOf(',') !== -1) {
|
|
803
|
+
return v
|
|
804
|
+
.split(',')
|
|
805
|
+
.map((x) => x.trim())
|
|
806
|
+
.map((x) => toStr(x));
|
|
807
|
+
}
|
|
808
|
+
return [toStr(v)];
|
|
809
|
+
};
|
|
810
|
+
const ansArr = ensureArray(answer);
|
|
811
|
+
const valArr = ensureArray(value);
|
|
812
|
+
// numeric comparisons: only when both sides are scalars (non-array) and numeric
|
|
813
|
+
const isNumericScalar = (x) => !Array.isArray(x) && x !== null && x !== undefined && !Number.isNaN(Number(x));
|
|
794
814
|
switch (operator) {
|
|
795
|
-
// Number comparisons
|
|
796
815
|
case '<':
|
|
797
|
-
|
|
816
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
817
|
+
return Number(answer) < Number(value);
|
|
818
|
+
return false;
|
|
798
819
|
case '>':
|
|
799
|
-
|
|
820
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
821
|
+
return Number(answer) > Number(value);
|
|
822
|
+
return false;
|
|
800
823
|
case '<=':
|
|
801
|
-
|
|
824
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
825
|
+
return Number(answer) <= Number(value);
|
|
826
|
+
return false;
|
|
802
827
|
case '>=':
|
|
803
|
-
|
|
828
|
+
if (isNumericScalar(answer) && isNumericScalar(value))
|
|
829
|
+
return Number(answer) >= Number(value);
|
|
830
|
+
return false;
|
|
831
|
+
// equality: true if any answer element equals any value element
|
|
804
832
|
case '=':
|
|
805
|
-
return
|
|
833
|
+
return ansArr.some((a) => valArr.some((v) => a === v));
|
|
806
834
|
case '!=':
|
|
807
|
-
return
|
|
808
|
-
//
|
|
835
|
+
return !ansArr.some((a) => valArr.some((v) => a === v));
|
|
836
|
+
// LIKE / NOT LIKE: check substring match anywhere in any answer element
|
|
809
837
|
case 'LIKE':
|
|
810
|
-
return
|
|
838
|
+
return ansArr.some((a) => valArr.some((v) => a.includes(v)));
|
|
811
839
|
case 'NOT LIKE':
|
|
812
|
-
return !
|
|
840
|
+
return !ansArr.some((a) => valArr.some((v) => a.includes(v)));
|
|
813
841
|
case 'BEGIN WITH':
|
|
814
|
-
return
|
|
842
|
+
return ansArr.some((a) => valArr.some((v) => a.startsWith(v)));
|
|
815
843
|
case 'END WITH':
|
|
816
|
-
return
|
|
817
|
-
//
|
|
844
|
+
return ansArr.some((a) => valArr.some((v) => a.endsWith(v)));
|
|
845
|
+
// IN / NOT IN: true if any answer element exists in value list
|
|
818
846
|
case 'IN':
|
|
819
|
-
return
|
|
820
|
-
? val.includes(ans)
|
|
821
|
-
: String(val)
|
|
822
|
-
.split(',')
|
|
823
|
-
.map((v) => v.trim())
|
|
824
|
-
.includes(String(ans));
|
|
847
|
+
return ansArr.some((a) => valArr.includes(a));
|
|
825
848
|
case 'NOT IN':
|
|
826
|
-
return
|
|
827
|
-
? !val.includes(ans)
|
|
828
|
-
: !String(val)
|
|
829
|
-
.split(',')
|
|
830
|
-
.map((v) => v.trim())
|
|
831
|
-
.includes(String(ans));
|
|
849
|
+
return !ansArr.some((a) => valArr.includes(a));
|
|
832
850
|
default:
|
|
833
851
|
return false;
|
|
834
852
|
}
|