@piserve-tech/form-submission 1.3.149 → 1.3.151
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.
|
@@ -3,32 +3,49 @@ import * as i0 from "@angular/core";
|
|
|
3
3
|
export class FormValidationService {
|
|
4
4
|
constructor() { }
|
|
5
5
|
validateRequiredQuestions(page) {
|
|
6
|
-
const validateQuestions = (formElements) => {
|
|
6
|
+
const validateQuestions = (formElements, submissionContext = null) => {
|
|
7
7
|
for (let element of formElements) {
|
|
8
8
|
if (element.entityType === 'QUESTION') {
|
|
9
|
-
const
|
|
10
|
-
const
|
|
11
|
-
|
|
12
|
-
|
|
9
|
+
const question = element.element;
|
|
10
|
+
const required = question.required;
|
|
11
|
+
const questionId = question.id;
|
|
12
|
+
const elementType = question.formElement?.elementType;
|
|
13
|
+
// If inside a subform submission context
|
|
14
|
+
if (submissionContext?.isInMultipleSubform) {
|
|
15
|
+
const { submission, questionMap } = submissionContext;
|
|
16
|
+
const submittedAnswer = submission.answers.find((ans) => ans.questionId === questionId);
|
|
17
|
+
const answer = submittedAnswer?.answer;
|
|
18
|
+
const isEmpty = answer === null ||
|
|
19
|
+
answer === undefined ||
|
|
20
|
+
(typeof answer === 'string' && answer.trim() === '') ||
|
|
21
|
+
(typeof answer === 'object' && Object.keys(answer).length === 0);
|
|
22
|
+
if (required && isEmpty) {
|
|
13
23
|
return {
|
|
14
24
|
isValid: false,
|
|
15
|
-
message: `Required
|
|
25
|
+
message: `Required question ${question.questionNumber} is not answered in a subform.`,
|
|
16
26
|
};
|
|
17
27
|
}
|
|
18
28
|
}
|
|
19
29
|
else {
|
|
20
|
-
|
|
30
|
+
// Normal non-subform validation
|
|
31
|
+
const answer = question.answer;
|
|
32
|
+
const isEmpty = answer === null ||
|
|
33
|
+
answer === undefined ||
|
|
34
|
+
(typeof answer === 'string' && answer.trim() === '') ||
|
|
35
|
+
(typeof answer === 'object' && Object.keys(answer).length === 0);
|
|
36
|
+
if (required && isEmpty) {
|
|
21
37
|
return {
|
|
22
38
|
isValid: false,
|
|
23
|
-
message: `Required question ${
|
|
39
|
+
message: `Required question ${question.questionNumber} is not answered.`,
|
|
24
40
|
};
|
|
25
41
|
}
|
|
26
42
|
}
|
|
27
|
-
|
|
28
|
-
|
|
43
|
+
// Recursively validate child logics
|
|
44
|
+
if (question.childLogics) {
|
|
45
|
+
for (let logic of question.childLogics) {
|
|
29
46
|
if (logic.showLogic) {
|
|
30
47
|
for (let row of logic.rows) {
|
|
31
|
-
const result = validateQuestions(row.grid);
|
|
48
|
+
const result = validateQuestions(row.grid, submissionContext);
|
|
32
49
|
if (!result.isValid)
|
|
33
50
|
return result;
|
|
34
51
|
}
|
|
@@ -37,8 +54,50 @@ export class FormValidationService {
|
|
|
37
54
|
}
|
|
38
55
|
}
|
|
39
56
|
else if (element.entityType === 'SUBFORM') {
|
|
40
|
-
|
|
41
|
-
|
|
57
|
+
const subform = element.element;
|
|
58
|
+
if (subform.property?.subFormStructure === 'multiple') {
|
|
59
|
+
const submissions = subform.submissions || [];
|
|
60
|
+
// Check if subform has any required questions
|
|
61
|
+
let hasRequiredQuestion = false;
|
|
62
|
+
for (const row of subform.rows) {
|
|
63
|
+
for (const field of row.grid) {
|
|
64
|
+
if (field.entityType === 'QUESTION' && field.element.required) {
|
|
65
|
+
hasRequiredQuestion = true;
|
|
66
|
+
break;
|
|
67
|
+
}
|
|
68
|
+
}
|
|
69
|
+
if (hasRequiredQuestion)
|
|
70
|
+
break;
|
|
71
|
+
}
|
|
72
|
+
if (hasRequiredQuestion && submissions.length === 0) {
|
|
73
|
+
return {
|
|
74
|
+
isValid: false,
|
|
75
|
+
message: `Subform "${subform.title || 'Untitled'}" is missing required submissions.`,
|
|
76
|
+
};
|
|
77
|
+
}
|
|
78
|
+
const questionMap = new Map();
|
|
79
|
+
for (const row of subform.rows) {
|
|
80
|
+
for (const field of row.grid) {
|
|
81
|
+
if (field.entityType === 'QUESTION') {
|
|
82
|
+
questionMap.set(field.element.id, field.element);
|
|
83
|
+
}
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
for (const submission of submissions) {
|
|
87
|
+
for (const row of subform.rows) {
|
|
88
|
+
const result = validateQuestions(row.grid, {
|
|
89
|
+
isInMultipleSubform: true,
|
|
90
|
+
submission,
|
|
91
|
+
questionMap
|
|
92
|
+
});
|
|
93
|
+
if (!result.isValid)
|
|
94
|
+
return result;
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
}
|
|
98
|
+
else {
|
|
99
|
+
// Single-entry subform
|
|
100
|
+
for (const row of subform.rows || []) {
|
|
42
101
|
const result = validateQuestions(row.grid);
|
|
43
102
|
if (!result.isValid)
|
|
44
103
|
return result;
|
|
@@ -48,7 +107,7 @@ export class FormValidationService {
|
|
|
48
107
|
else if ((element.entityType === 'QUESTION_GROUP' || element.entityType === 'MULTIFIELD') &&
|
|
49
108
|
element.element.rows) {
|
|
50
109
|
for (const row of element.element.rows) {
|
|
51
|
-
const result = validateQuestions(row.grid);
|
|
110
|
+
const result = validateQuestions(row.grid, submissionContext);
|
|
52
111
|
if (!result.isValid)
|
|
53
112
|
return result;
|
|
54
113
|
}
|
|
@@ -56,11 +115,11 @@ export class FormValidationService {
|
|
|
56
115
|
}
|
|
57
116
|
return { isValid: true, message: 'All required questions are answered.' };
|
|
58
117
|
};
|
|
59
|
-
|
|
118
|
+
// Start validation from top-level rows
|
|
119
|
+
for (const row of page.rows) {
|
|
60
120
|
const result = validateQuestions(row.grid);
|
|
61
|
-
if (!result.isValid)
|
|
121
|
+
if (!result.isValid)
|
|
62
122
|
return result;
|
|
63
|
-
}
|
|
64
123
|
}
|
|
65
124
|
return { isValid: true, message: 'All required questions are answered.' };
|
|
66
125
|
}
|
|
@@ -73,4 +132,4 @@ i0.ɵɵngDeclareClassMetadata({ minVersion: "12.0.0", version: "16.2.12", ngImpo
|
|
|
73
132
|
providedIn: 'root'
|
|
74
133
|
}]
|
|
75
134
|
}], ctorParameters: function () { return []; } });
|
|
76
|
-
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZm9ybS12YWxpZGF0aW9uLnNlcnZpY2UuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi8uLi9wcm9qZWN0cy9mb3JtLXN1Ym1pc3Npb24vc3JjL3NlcnZpY2VzL2Zvcm0tdmFsaWRhdGlvbi5zZXJ2aWNlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiJBQUFBLE9BQU8sRUFBRSxVQUFVLEVBQUUsTUFBTSxlQUFlLENBQUM7O0FBSzNDLE1BQU0sT0FBTyxxQkFBcUI7SUFFaEMsZ0JBQWUsQ0FBQztJQUVoQix5QkFBeUIsQ0FBQyxJQUFTO1FBQ2pDLE1BQU0saUJBQWlCLEdBQUcsQ0FDeEIsWUFBbUIsRUFDb0IsRUFBRTtZQUN6QyxLQUFLLElBQUksT0FBTyxJQUFJLFlBQVksRUFBRTtnQkFDaEMsSUFBSSxPQUFPLENBQUMsVUFBVSxLQUFLLFVBQVUsRUFBRTtvQkFDckMsTUFBTSxRQUFRLEdBQUcsT0FBTyxDQUFDLE9BQU8sQ0FBQyxRQUFRLENBQUM7b0JBQzFDLE1BQU0sTUFBTSxHQUFHLE9BQU8sQ0FBQyxPQUFPLENBQUMsTUFBTSxDQUFDO29CQUN0QyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsV0FBVyxDQUFDLFdBQVcsS0FBSyxhQUFhLEVBQUU7d0JBQzdELElBQUksUUFBUSxJQUFJLENBQUMsQ0FBQyxNQUFNLElBQUksTUFBTSxDQUFDLE1BQU0sS0FBSyxDQUFDLENBQUMsRUFBRTs0QkFDaEQsT0FBTztnQ0FDTCxPQUFPLEVBQUUsS0FBSztnQ0FDZCxPQUFPLEVBQUUsMEJBQTBCLE9BQU8sQ0FBQyxPQUFPLENBQUMsY0FBYyxtQkFBbUI7NkJBQ3JGLENBQUM7eUJBQ0g7cUJBQ0Y7eUJBQU07d0JBQ0wsSUFBSSxRQUFRLElBQUksQ0FBQyxDQUFDLE1BQU0sSUFBSSxNQUFNLENBQUMsSUFBSSxDQUFDLE1BQU0sQ0FBQyxDQUFDLE1BQU0sS0FBSyxDQUFDLENBQUMsRUFBRTs0QkFDN0QsT0FBTztnQ0FDTCxPQUFPLEVBQUUsS0FBSztnQ0FDZCxPQUFPLEVBQUUscUJBQXFCLE9BQU8sQ0FBQyxPQUFPLENBQUMsY0FBYyxtQkFBbUI7NkJBQ2hGLENBQUM7eUJBQ0g7cUJBQ0Y7b0JBQ0QsSUFBSSxPQUFPLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRTt3QkFDL0IsS0FBSyxJQUFJLEtBQUssSUFBSSxPQUFPLENBQUMsT0FBTyxDQUFDLFdBQVcsRUFBRTs0QkFDN0MsSUFBSSxLQUFLLENBQUMsU0FBUyxFQUFFO2dDQUNuQixLQUFLLElBQUksR0FBRyxJQUFJLEtBQUssQ0FBQyxJQUFJLEVBQUU7b0NBQzFCLE1BQU0sTUFBTSxHQUFHLGlCQUFpQixDQUFDLEdBQUcsQ0FBQyxJQUFJLENBQUMsQ0FBQztvQ0FDM0MsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPO3dDQUFFLE9BQU8sTUFBTSxDQUFDO2lDQUNwQzs2QkFDRjt5QkFDRjtxQkFDRjtpQkFDRjtxQkFBTSxJQUFJLE9BQU8sQ0FBQyxVQUFVLEtBQUssU0FBUyxFQUFFO29CQUMzQyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFO3dCQUN4QixLQUFLLElBQUksR0FBRyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFOzRCQUNwQyxNQUFNLE1BQU0sR0FBRyxpQkFBaUIsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7NEJBQzNDLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTztnQ0FBRSxPQUFPLE1BQU0sQ0FBQzt5QkFDcEM7cUJBQ0Y7aUJBQ0Y7cUJBQU0sSUFDTCxDQUFDLE9BQU8sQ0FBQyxVQUFVLEtBQUssZ0JBQWdCLElBQUksT0FBTyxDQUFDLFVBQVUsS0FBSyxZQUFZLENBQUM7b0JBQ2hGLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUNwQjtvQkFDQSxLQUFLLE1BQU0sR0FBRyxJQUFJLE9BQU8sQ0FBQyxPQUFPLENBQUMsSUFBSSxFQUFFO3dCQUN0QyxNQUFNLE1BQU0sR0FBRyxpQkFBaUIsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7d0JBQzNDLElBQUksQ0FBQyxNQUFNLENBQUMsT0FBTzs0QkFBRSxPQUFPLE1BQU0sQ0FBQztxQkFDcEM7aUJBQ0Y7YUFDRjtZQUVELE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxzQ0FBc0MsRUFBRSxDQUFDO1FBQzVFLENBQUMsQ0FBQztRQUVGLEtBQUssSUFBSSxHQUFHLElBQUksSUFBSSxDQUFDLElBQUksRUFBRTtZQUN6QixNQUFNLE1BQU0sR0FBRyxpQkFBaUIsQ0FBQyxHQUFHLENBQUMsSUFBSSxDQUFDLENBQUM7WUFDM0MsSUFBSSxDQUFDLE1BQU0sQ0FBQyxPQUFPLEVBQUU7Z0JBQ25CLE9BQU8sTUFBTSxDQUFDO2FBQ2Y7U0FDRjtRQUVELE9BQU8sRUFBRSxPQUFPLEVBQUUsSUFBSSxFQUFFLE9BQU8sRUFBRSxzQ0FBc0MsRUFBRSxDQUFDO0lBQzVFLENBQUM7K0dBbEVVLHFCQUFxQjttSEFBckIscUJBQXFCLGNBRnBCLE1BQU07OzRGQUVQLHFCQUFxQjtrQkFIakMsVUFBVTttQkFBQztvQkFDVixVQUFVLEVBQUUsTUFBTTtpQkFDbkIiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBJbmplY3RhYmxlIH0gZnJvbSAnQGFuZ3VsYXIvY29yZSc7XG5cbkBJbmplY3RhYmxlKHtcbiAgcHJvdmlkZWRJbjogJ3Jvb3QnXG59KVxuZXhwb3J0IGNsYXNzIEZvcm1WYWxpZGF0aW9uU2VydmljZSB7XG5cbiAgY29uc3RydWN0b3IoKSB7fVxuXG4gIHZhbGlkYXRlUmVxdWlyZWRRdWVzdGlvbnMocGFnZTogYW55KTogeyBpc1ZhbGlkOiBib29sZWFuOyBtZXNzYWdlOiBzdHJpbmcgfSB7XG4gICAgY29uc3QgdmFsaWRhdGVRdWVzdGlvbnMgPSAoXG4gICAgICBmb3JtRWxlbWVudHM6IGFueVtdXG4gICAgKTogeyBpc1ZhbGlkOiBib29sZWFuOyBtZXNzYWdlOiBzdHJpbmcgfSA9PiB7XG4gICAgICBmb3IgKGxldCBlbGVtZW50IG9mIGZvcm1FbGVtZW50cykge1xuICAgICAgICBpZiAoZWxlbWVudC5lbnRpdHlUeXBlID09PSAnUVVFU1RJT04nKSB7XG4gICAgICAgICAgY29uc3QgcmVxdWlyZWQgPSBlbGVtZW50LmVsZW1lbnQucmVxdWlyZWQ7XG4gICAgICAgICAgY29uc3QgYW5zd2VyID0gZWxlbWVudC5lbGVtZW50LmFuc3dlcjtcbiAgICAgICAgICBpZiAoZWxlbWVudC5lbGVtZW50LmZvcm1FbGVtZW50LmVsZW1lbnRUeXBlID09PSAnRklMRV9QSUNLRVInKSB7XG4gICAgICAgICAgICBpZiAocmVxdWlyZWQgJiYgKCFhbnN3ZXIgfHwgYW5zd2VyLmxlbmd0aCA9PT0gMCkpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBpc1ZhbGlkOiBmYWxzZSxcbiAgICAgICAgICAgICAgICBtZXNzYWdlOiBgUmVxdWlyZWQgZmlsZSBxdWVzdGlvbiAke2VsZW1lbnQuZWxlbWVudC5xdWVzdGlvbk51bWJlcn0gaXMgbm90IGFuc3dlcmVkLmAsXG4gICAgICAgICAgICAgIH07XG4gICAgICAgICAgICB9XG4gICAgICAgICAgfSBlbHNlIHtcbiAgICAgICAgICAgIGlmIChyZXF1aXJlZCAmJiAoIWFuc3dlciB8fCBPYmplY3Qua2V5cyhhbnN3ZXIpLmxlbmd0aCA9PT0gMCkpIHtcbiAgICAgICAgICAgICAgcmV0dXJuIHtcbiAgICAgICAgICAgICAgICBpc1ZhbGlkOiBmYWxzZSxcbiAgICAgICAgICAgICAgICBtZXNzYWdlOiBgUmVxdWlyZWQgcXVlc3Rpb24gJHtlbGVtZW50LmVsZW1lbnQucXVlc3Rpb25OdW1iZXJ9IGlzIG5vdCBhbnN3ZXJlZC5gLFxuICAgICAgICAgICAgICB9O1xuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgICBpZiAoZWxlbWVudC5lbGVtZW50LmNoaWxkTG9naWNzKSB7XG4gICAgICAgICAgICBmb3IgKGxldCBsb2dpYyBvZiBlbGVtZW50LmVsZW1lbnQuY2hpbGRMb2dpY3MpIHtcbiAgICAgICAgICAgICAgaWYgKGxvZ2ljLnNob3dMb2dpYykge1xuICAgICAgICAgICAgICAgIGZvciAobGV0IHJvdyBvZiBsb2dpYy5yb3dzKSB7XG4gICAgICAgICAgICAgICAgICBjb25zdCByZXN1bHQgPSB2YWxpZGF0ZVF1ZXN0aW9ucyhyb3cuZ3JpZCk7XG4gICAgICAgICAgICAgICAgICBpZiAoIXJlc3VsdC5pc1ZhbGlkKSByZXR1cm4gcmVzdWx0O1xuICAgICAgICAgICAgICAgIH1cbiAgICAgICAgICAgICAgfVxuICAgICAgICAgICAgfVxuICAgICAgICAgIH1cbiAgICAgICAgfSBlbHNlIGlmIChlbGVtZW50LmVudGl0eVR5cGUgPT09ICdTVUJGT1JNJykge1xuICAgICAgICAgIGlmIChlbGVtZW50LmVsZW1lbnQucm93cykge1xuICAgICAgICAgICAgZm9yIChsZXQgcm93IG9mIGVsZW1lbnQuZWxlbWVudC5yb3dzKSB7XG4gICAgICAgICAgICAgIGNvbnN0IHJlc3VsdCA9IHZhbGlkYXRlUXVlc3Rpb25zKHJvdy5ncmlkKTtcbiAgICAgICAgICAgICAgaWYgKCFyZXN1bHQuaXNWYWxpZCkgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgICAgIH1cbiAgICAgICAgICB9XG4gICAgICAgIH0gZWxzZSBpZiAoXG4gICAgICAgICAgKGVsZW1lbnQuZW50aXR5VHlwZSA9PT0gJ1FVRVNUSU9OX0dST1VQJyB8fCBlbGVtZW50LmVudGl0eVR5cGUgPT09ICdNVUxUSUZJRUxEJykgJiZcbiAgICAgICAgICBlbGVtZW50LmVsZW1lbnQucm93c1xuICAgICAgICApIHtcbiAgICAgICAgICBmb3IgKGNvbnN0IHJvdyBvZiBlbGVtZW50LmVsZW1lbnQucm93cykge1xuICAgICAgICAgICAgY29uc3QgcmVzdWx0ID0gdmFsaWRhdGVRdWVzdGlvbnMocm93LmdyaWQpO1xuICAgICAgICAgICAgaWYgKCFyZXN1bHQuaXNWYWxpZCkgcmV0dXJuIHJlc3VsdDtcbiAgICAgICAgICB9XG4gICAgICAgIH1cbiAgICAgIH1cblxuICAgICAgcmV0dXJuIHsgaXNWYWxpZDogdHJ1ZSwgbWVzc2FnZTogJ0FsbCByZXF1aXJlZCBxdWVzdGlvbnMgYXJlIGFuc3dlcmVkLicgfTtcbiAgICB9O1xuXG4gICAgZm9yIChsZXQgcm93IG9mIHBhZ2Uucm93cykge1xuICAgICAgY29uc3QgcmVzdWx0ID0gdmFsaWRhdGVRdWVzdGlvbnMocm93LmdyaWQpO1xuICAgICAgaWYgKCFyZXN1bHQuaXNWYWxpZCkge1xuICAgICAgICByZXR1cm4gcmVzdWx0O1xuICAgICAgfVxuICAgIH1cblxuICAgIHJldHVybiB7IGlzVmFsaWQ6IHRydWUsIG1lc3NhZ2U6ICdBbGwgcmVxdWlyZWQgcXVlc3Rpb25zIGFyZSBhbnN3ZXJlZC4nIH07XG4gIH1cbn1cbiJdfQ==
|
|
135
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"form-validation.service.js","sourceRoot":"","sources":["../../../../projects/form-submission/src/services/form-validation.service.ts"],"names":[],"mappings":"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","sourcesContent":["import { Injectable } from '@angular/core';\n\n@Injectable({\n  providedIn: 'root'\n})\nexport class FormValidationService {\n\n  constructor() {}\n\n  validateRequiredQuestions(page: any): { isValid: boolean; message: string } {\n    const validateQuestions = (\n      formElements: any[],\n      submissionContext: any = null\n    ): { isValid: boolean; message: string } => {\n      for (let element of formElements) {\n        if (element.entityType === 'QUESTION') {\n          const question = element.element;\n          const required = question.required;\n          const questionId = question.id;\n          const elementType = question.formElement?.elementType;\n  \n          // If inside a subform submission context\n          if (submissionContext?.isInMultipleSubform) {\n            const { submission, questionMap } = submissionContext;\n  \n            const submittedAnswer = submission.answers.find(\n              (ans: any) => ans.questionId === questionId\n            );\n  \n            const answer = submittedAnswer?.answer;\n  \n            const isEmpty =\n              answer === null ||\n              answer === undefined ||\n              (typeof answer === 'string' && answer.trim() === '') ||\n              (typeof answer === 'object' && Object.keys(answer).length === 0);\n  \n            if (required && isEmpty) {\n              return {\n                isValid: false,\n                message: `Required question ${question.questionNumber} is not answered in a subform.`,\n              };\n            }\n          } else {\n            // Normal non-subform validation\n            const answer = question.answer;\n            const isEmpty =\n              answer === null ||\n              answer === undefined ||\n              (typeof answer === 'string' && answer.trim() === '') ||\n              (typeof answer === 'object' && Object.keys(answer).length === 0);\n  \n            if (required && isEmpty) {\n              return {\n                isValid: false,\n                message: `Required question ${question.questionNumber} is not answered.`,\n              };\n            }\n          }\n  \n          // Recursively validate child logics\n          if (question.childLogics) {\n            for (let logic of question.childLogics) {\n              if (logic.showLogic) {\n                for (let row of logic.rows) {\n                  const result = validateQuestions(row.grid, submissionContext);\n                  if (!result.isValid) return result;\n                }\n              }\n            }\n          }\n        } else if (element.entityType === 'SUBFORM') {\n          const subform = element.element;\n  \n          if (subform.property?.subFormStructure === 'multiple') {\n            const submissions = subform.submissions || [];\n          \n            // Check if subform has any required questions\n            let hasRequiredQuestion = false;\n            for (const row of subform.rows) {\n              for (const field of row.grid) {\n                if (field.entityType === 'QUESTION' && field.element.required) {\n                  hasRequiredQuestion = true;\n                  break;\n                }\n              }\n              if (hasRequiredQuestion) break;\n            }\n          \n            if (hasRequiredQuestion && submissions.length === 0) {\n              return {\n                isValid: false,\n                message: `Subform \"${subform.title || 'Untitled'}\" is missing required submissions.`,\n              };\n            }\n          \n            const questionMap = new Map();\n            for (const row of subform.rows) {\n              for (const field of row.grid) {\n                if (field.entityType === 'QUESTION') {\n                  questionMap.set(field.element.id, field.element);\n                }\n              }\n            }\n          \n            for (const submission of submissions) {\n              for (const row of subform.rows) {\n                const result = validateQuestions(row.grid, {\n                  isInMultipleSubform: true,\n                  submission,\n                  questionMap\n                });\n                if (!result.isValid) return result;\n              }\n            }\n          }\n           \n          \n          else {\n            // Single-entry subform\n            for (const row of subform.rows || []) {\n              const result = validateQuestions(row.grid);\n              if (!result.isValid) return result;\n            }\n          }\n        } else if (\n          (element.entityType === 'QUESTION_GROUP' || element.entityType === 'MULTIFIELD') &&\n          element.element.rows\n        ) {\n          for (const row of element.element.rows) {\n            const result = validateQuestions(row.grid, submissionContext);\n            if (!result.isValid) return result;\n          }\n        }\n      }\n  \n      return { isValid: true, message: 'All required questions are answered.' };\n    };\n  \n    // Start validation from top-level rows\n    for (const row of page.rows) {\n      const result = validateQuestions(row.grid);\n      if (!result.isValid) return result;\n    }\n  \n    return { isValid: true, message: 'All required questions are answered.' };\n  }\n  \n  \n}\n"]}
|
|
@@ -430,32 +430,49 @@ function mapOptionToModel(apiOption) {
|
|
|
430
430
|
class FormValidationService {
|
|
431
431
|
constructor() { }
|
|
432
432
|
validateRequiredQuestions(page) {
|
|
433
|
-
const validateQuestions = (formElements) => {
|
|
433
|
+
const validateQuestions = (formElements, submissionContext = null) => {
|
|
434
434
|
for (let element of formElements) {
|
|
435
435
|
if (element.entityType === 'QUESTION') {
|
|
436
|
-
const
|
|
437
|
-
const
|
|
438
|
-
|
|
439
|
-
|
|
436
|
+
const question = element.element;
|
|
437
|
+
const required = question.required;
|
|
438
|
+
const questionId = question.id;
|
|
439
|
+
const elementType = question.formElement?.elementType;
|
|
440
|
+
// If inside a subform submission context
|
|
441
|
+
if (submissionContext?.isInMultipleSubform) {
|
|
442
|
+
const { submission, questionMap } = submissionContext;
|
|
443
|
+
const submittedAnswer = submission.answers.find((ans) => ans.questionId === questionId);
|
|
444
|
+
const answer = submittedAnswer?.answer;
|
|
445
|
+
const isEmpty = answer === null ||
|
|
446
|
+
answer === undefined ||
|
|
447
|
+
(typeof answer === 'string' && answer.trim() === '') ||
|
|
448
|
+
(typeof answer === 'object' && Object.keys(answer).length === 0);
|
|
449
|
+
if (required && isEmpty) {
|
|
440
450
|
return {
|
|
441
451
|
isValid: false,
|
|
442
|
-
message: `Required
|
|
452
|
+
message: `Required question ${question.questionNumber} is not answered in a subform.`,
|
|
443
453
|
};
|
|
444
454
|
}
|
|
445
455
|
}
|
|
446
456
|
else {
|
|
447
|
-
|
|
457
|
+
// Normal non-subform validation
|
|
458
|
+
const answer = question.answer;
|
|
459
|
+
const isEmpty = answer === null ||
|
|
460
|
+
answer === undefined ||
|
|
461
|
+
(typeof answer === 'string' && answer.trim() === '') ||
|
|
462
|
+
(typeof answer === 'object' && Object.keys(answer).length === 0);
|
|
463
|
+
if (required && isEmpty) {
|
|
448
464
|
return {
|
|
449
465
|
isValid: false,
|
|
450
|
-
message: `Required question ${
|
|
466
|
+
message: `Required question ${question.questionNumber} is not answered.`,
|
|
451
467
|
};
|
|
452
468
|
}
|
|
453
469
|
}
|
|
454
|
-
|
|
455
|
-
|
|
470
|
+
// Recursively validate child logics
|
|
471
|
+
if (question.childLogics) {
|
|
472
|
+
for (let logic of question.childLogics) {
|
|
456
473
|
if (logic.showLogic) {
|
|
457
474
|
for (let row of logic.rows) {
|
|
458
|
-
const result = validateQuestions(row.grid);
|
|
475
|
+
const result = validateQuestions(row.grid, submissionContext);
|
|
459
476
|
if (!result.isValid)
|
|
460
477
|
return result;
|
|
461
478
|
}
|
|
@@ -464,8 +481,50 @@ class FormValidationService {
|
|
|
464
481
|
}
|
|
465
482
|
}
|
|
466
483
|
else if (element.entityType === 'SUBFORM') {
|
|
467
|
-
|
|
468
|
-
|
|
484
|
+
const subform = element.element;
|
|
485
|
+
if (subform.property?.subFormStructure === 'multiple') {
|
|
486
|
+
const submissions = subform.submissions || [];
|
|
487
|
+
// Check if subform has any required questions
|
|
488
|
+
let hasRequiredQuestion = false;
|
|
489
|
+
for (const row of subform.rows) {
|
|
490
|
+
for (const field of row.grid) {
|
|
491
|
+
if (field.entityType === 'QUESTION' && field.element.required) {
|
|
492
|
+
hasRequiredQuestion = true;
|
|
493
|
+
break;
|
|
494
|
+
}
|
|
495
|
+
}
|
|
496
|
+
if (hasRequiredQuestion)
|
|
497
|
+
break;
|
|
498
|
+
}
|
|
499
|
+
if (hasRequiredQuestion && submissions.length === 0) {
|
|
500
|
+
return {
|
|
501
|
+
isValid: false,
|
|
502
|
+
message: `Subform "${subform.title || 'Untitled'}" is missing required submissions.`,
|
|
503
|
+
};
|
|
504
|
+
}
|
|
505
|
+
const questionMap = new Map();
|
|
506
|
+
for (const row of subform.rows) {
|
|
507
|
+
for (const field of row.grid) {
|
|
508
|
+
if (field.entityType === 'QUESTION') {
|
|
509
|
+
questionMap.set(field.element.id, field.element);
|
|
510
|
+
}
|
|
511
|
+
}
|
|
512
|
+
}
|
|
513
|
+
for (const submission of submissions) {
|
|
514
|
+
for (const row of subform.rows) {
|
|
515
|
+
const result = validateQuestions(row.grid, {
|
|
516
|
+
isInMultipleSubform: true,
|
|
517
|
+
submission,
|
|
518
|
+
questionMap
|
|
519
|
+
});
|
|
520
|
+
if (!result.isValid)
|
|
521
|
+
return result;
|
|
522
|
+
}
|
|
523
|
+
}
|
|
524
|
+
}
|
|
525
|
+
else {
|
|
526
|
+
// Single-entry subform
|
|
527
|
+
for (const row of subform.rows || []) {
|
|
469
528
|
const result = validateQuestions(row.grid);
|
|
470
529
|
if (!result.isValid)
|
|
471
530
|
return result;
|
|
@@ -475,7 +534,7 @@ class FormValidationService {
|
|
|
475
534
|
else if ((element.entityType === 'QUESTION_GROUP' || element.entityType === 'MULTIFIELD') &&
|
|
476
535
|
element.element.rows) {
|
|
477
536
|
for (const row of element.element.rows) {
|
|
478
|
-
const result = validateQuestions(row.grid);
|
|
537
|
+
const result = validateQuestions(row.grid, submissionContext);
|
|
479
538
|
if (!result.isValid)
|
|
480
539
|
return result;
|
|
481
540
|
}
|
|
@@ -483,11 +542,11 @@ class FormValidationService {
|
|
|
483
542
|
}
|
|
484
543
|
return { isValid: true, message: 'All required questions are answered.' };
|
|
485
544
|
};
|
|
486
|
-
|
|
545
|
+
// Start validation from top-level rows
|
|
546
|
+
for (const row of page.rows) {
|
|
487
547
|
const result = validateQuestions(row.grid);
|
|
488
|
-
if (!result.isValid)
|
|
548
|
+
if (!result.isValid)
|
|
489
549
|
return result;
|
|
490
|
-
}
|
|
491
550
|
}
|
|
492
551
|
return { isValid: true, message: 'All required questions are answered.' };
|
|
493
552
|
}
|