From 5eefb5aecc0a11bdac7cc9ff566e5f08552f32b9 Mon Sep 17 00:00:00 2001 From: "ICX\\Tatsiana.Hashtold" Date: Tue, 20 Aug 2024 15:30:46 +0300 Subject: [PATCH 1/4] FIO-8810: fixed an issue where user unables to resubmit (change) the form with several levels of nested form with required fields --- src/process/__tests__/process.test.ts | 97 +++++++++++++++++++++++++++ src/utils/formUtil.ts | 14 ++-- 2 files changed, 107 insertions(+), 4 deletions(-) diff --git a/src/process/__tests__/process.test.ts b/src/process/__tests__/process.test.ts index 95225136..3a083159 100644 --- a/src/process/__tests__/process.test.ts +++ b/src/process/__tests__/process.test.ts @@ -2831,6 +2831,103 @@ describe('Process Tests', () => { }); }) + it('Should allow the submission to go through without errors if there is no the subform reference value', async () => { + const form = { + _id: '66bc5cff7ca1729623a182db', + title: 'form2', + name: 'form2', + path: 'form2', + type: 'resource', + display: 'form', + owner: '637b2e6b48c1227e60b1f910', + components: [ + { + label: 'Text Field - form2', + applyMaskOn: 'change', + tableView: true, + validate: { required: true }, + validateWhenHidden: false, + key: 'textFieldForm2', + type: 'textfield', + input: true, + }, + { + label: 'Form', + tableView: true, + form: '66bc5ccd7ca1729623a18063', + useOriginalRevision: false, + key: 'form', + type: 'form', + input: true, + components: [ + { + label: 'Text Field form1', + applyMaskOn: 'change', + tableView: true, + validate: { required: true }, + validateWhenHidden: false, + key: 'textFieldForm1', + type: 'textfield', + input: true, + }, + { + type: 'button', + label: 'Submit', + key: 'submit', + disableOnInvalid: true, + input: true, + tableView: false, + }, + ], + }, + { + type: 'button', + label: 'Submit', + key: 'submit', + disableOnInvalid: true, + input: true, + tableView: false, + }, + ], + project: '669e1c67a40e327e67e7ce55', + _vid: 0, + esign: {}, + created: '2024-08-14T07:30:07.953Z', + modified: '2024-08-14T10:09:13.201Z', + machineName: 'qzdhayddccjauyr:form2', + __v: 1, + }; + + const submission = { + data: { textFieldForm2: '1', form: { _id: '66c455fc0f00757fd4b0e79b' } }, + owner: '637b2e6b48c1227e60b1f910', + access: [], + _fvid: 0, + state: 'submitted', + _id: '66c455fc0f00757fd4b0e79d', + form: '66bc5cff7ca1729623a182db', + }; + + const errors: any = []; + const context = { + form, + submission, + data: submission.data, + components: form.components, + processors: ProcessTargets.submission, + scope: { errors }, + config: { + server: true, + }, + }; + processSync(context); + submission.data = context.data; + context.processors = ProcessTargets.evaluator; + processSync(context); + assert.equal(context.scope.errors.length, 0); + assert.deepEqual(context.data.form, { _id: '66c455fc0f00757fd4b0e79b', data: {} }) + }); + describe('Required component validation in nested form in DataGrid/EditGrid', () => { const nestedForm = { key: 'form', diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts index 7947f7ac..0392012f 100644 --- a/src/utils/formUtil.ts +++ b/src/utils/formUtil.ts @@ -278,8 +278,11 @@ export const eachComponentDataAsync = async ( return true; } if (isComponentModelType(component, 'dataObject')) { - // No need to bother processing all the children data if there is no data for this form. - if (has(data, component.path)) { + // No need to bother processing all the children data if there is no data for this form or the reference value has not been loaded. + const nestedFormValue: any = get(data, component.path); + const noReferenceAttached = nestedFormValue && nestedFormValue._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); + const shouldProcessNestedFormData = nestedFormValue && nestedFormValue._id ? !noReferenceAttached : has(data, component.path); + if (shouldProcessNestedFormData) { // For nested forms, we need to reset the "data" and "path" objects for all of the children components, and then re-establish the data when it is done. const childPath: string = componentDataPath(component, path, compPath); const childData: any = get(data, childPath, null); @@ -331,8 +334,11 @@ export const eachComponentData = ( return true; } if (isComponentModelType(component, 'dataObject')) { - // No need to bother processing all the children data if there is no data for this form. - if (has(data, component.path)) { + // No need to bother processing all the children data if there is no data for this form or the reference value has not been loaded. + const nestedFormValue: any = get(data, component.path); + const noReferenceAttached = nestedFormValue && nestedFormValue._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); + const shouldProcessNestedFormData = nestedFormValue && nestedFormValue._id ? !noReferenceAttached : has(data, component.path); + if (shouldProcessNestedFormData) { // For nested forms, we need to reset the "data" and "path" objects for all of the children components, and then re-establish the data when it is done. const childPath: string = componentDataPath(component, path, compPath); const childData: any = get(data, childPath, {}); From c0018fbdcf28ed84be4e7c4177d1de28387528d3 Mon Sep 17 00:00:00 2001 From: TanyaGashtold <61136841+TanyaGashtold@users.noreply.github.com> Date: Fri, 23 Aug 2024 11:31:13 +0300 Subject: [PATCH 2/4] Update src/utils/formUtil.ts Co-authored-by: John Teague <164385719+johnformio@users.noreply.github.com> --- src/utils/formUtil.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts index 0392012f..dbdd1b59 100644 --- a/src/utils/formUtil.ts +++ b/src/utils/formUtil.ts @@ -280,8 +280,8 @@ export const eachComponentDataAsync = async ( if (isComponentModelType(component, 'dataObject')) { // No need to bother processing all the children data if there is no data for this form or the reference value has not been loaded. const nestedFormValue: any = get(data, component.path); - const noReferenceAttached = nestedFormValue && nestedFormValue._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); - const shouldProcessNestedFormData = nestedFormValue && nestedFormValue._id ? !noReferenceAttached : has(data, component.path); + const noReferenceAttached = nestedFormValue?._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); + const shouldProcessNestedFormData = nestedFormValue?._id ? !noReferenceAttached : has(data, component.path); if (shouldProcessNestedFormData) { // For nested forms, we need to reset the "data" and "path" objects for all of the children components, and then re-establish the data when it is done. const childPath: string = componentDataPath(component, path, compPath); From a497b67f69d94af000defc3b338156b2f4374746 Mon Sep 17 00:00:00 2001 From: TanyaGashtold <61136841+TanyaGashtold@users.noreply.github.com> Date: Fri, 23 Aug 2024 11:31:56 +0300 Subject: [PATCH 3/4] Update src/utils/formUtil.ts Co-authored-by: John Teague <164385719+johnformio@users.noreply.github.com> --- src/utils/formUtil.ts | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts index dbdd1b59..179b216e 100644 --- a/src/utils/formUtil.ts +++ b/src/utils/formUtil.ts @@ -336,8 +336,8 @@ export const eachComponentData = ( if (isComponentModelType(component, 'dataObject')) { // No need to bother processing all the children data if there is no data for this form or the reference value has not been loaded. const nestedFormValue: any = get(data, component.path); - const noReferenceAttached = nestedFormValue && nestedFormValue._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); - const shouldProcessNestedFormData = nestedFormValue && nestedFormValue._id ? !noReferenceAttached : has(data, component.path); + const noReferenceAttached =nestedFormValue?._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); + const shouldProcessNestedFormData = nestedFormValue?._id ? !noReferenceAttached : has(data, component.path); if (shouldProcessNestedFormData) { // For nested forms, we need to reset the "data" and "path" objects for all of the children components, and then re-establish the data when it is done. const childPath: string = componentDataPath(component, path, compPath); From db3957702bb0f82c1be2f17c6bdc90c3bd242c7f Mon Sep 17 00:00:00 2001 From: TanyaGashtold <61136841+TanyaGashtold@users.noreply.github.com> Date: Fri, 23 Aug 2024 11:33:16 +0300 Subject: [PATCH 4/4] Update formUtil.ts --- src/utils/formUtil.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts index 179b216e..94925034 100644 --- a/src/utils/formUtil.ts +++ b/src/utils/formUtil.ts @@ -336,7 +336,7 @@ export const eachComponentData = ( if (isComponentModelType(component, 'dataObject')) { // No need to bother processing all the children data if there is no data for this form or the reference value has not been loaded. const nestedFormValue: any = get(data, component.path); - const noReferenceAttached =nestedFormValue?._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); + const noReferenceAttached = nestedFormValue?._id && isEmpty(nestedFormValue.data) && !has(nestedFormValue, 'form'); const shouldProcessNestedFormData = nestedFormValue?._id ? !noReferenceAttached : has(data, component.path); if (shouldProcessNestedFormData) { // For nested forms, we need to reset the "data" and "path" objects for all of the children components, and then re-establish the data when it is done.