diff --git a/Changelog.md b/Changelog.md index 132f54d4..05b428e5 100644 --- a/Changelog.md +++ b/Changelog.md @@ -15,6 +15,12 @@ - FIO-8645: added tests and translations for validateRequiredDay - FIO-8537: Fixing the filter processor to handle nested component data properly - FIO-8597: fixed an issue with an empty array value for a number component with multiple values enabled + - FIO-8798: updated day component validation + - FIO-8885 & FIO-8886: use strict equality check for conditional component paths rather than Array.prototype.includes + - FIO-8810: fixed an issue where user unables to resubmit (change) the form with several levels of nested forms with required fields + - FIO-8848 fixed validation for TextArea with Save as Json + - FIO-8769 added check for the simpleConditional properties state + - FIO-8901: Fixed incorrect handling of excessive rows in nested array model ## 2.3.0-rc.1 ### Changed diff --git a/package.json b/package.json index 83f09bda..9d23a41e 100644 --- a/package.json +++ b/package.json @@ -94,7 +94,7 @@ "dependencies": { "@types/json-logic-js": "^2.0.7", "browser-cookies": "^1.2.0", - "core-js": "^3.37.1", + "core-js": "^3.38.0", "dayjs": "^1.11.12", "dompurify": "^3.1.6", "eventemitter3": "^5.0.0", diff --git a/src/experimental/base/array/__tests__/ArrayComponent.test.ts b/src/experimental/base/array/__tests__/ArrayComponent.test.ts index 6a1ae32c..dcc5fb40 100644 --- a/src/experimental/base/array/__tests__/ArrayComponent.test.ts +++ b/src/experimental/base/array/__tests__/ArrayComponent.test.ts @@ -52,23 +52,32 @@ describe('ArrayComponent', () => { { firstName: 'Sally', lastName: 'Thompson' + }, + { + firstName: 'Jane', + lastName: 'Doe' } ]; arrComp.dataValue = employees; assert.deepEqual(data, {employees: employees}); assert.deepEqual(arrComp.dataValue, employees); - assert.equal(arrComp.rows.length, 2); + assert.equal(arrComp.rows.length, 3); assert.equal(arrComp.rows[0].length, 2); assert.equal(arrComp.rows[1].length, 2); - assert.equal(arrComp.components.length, 4); + assert.equal(arrComp.rows[2].length, 2); + assert.equal(arrComp.components.length, 6); assert.equal(arrComp.components[0].dataValue, 'Joe'); assert.equal(arrComp.components[1].dataValue, 'Smith'); assert.strictEqual(arrComp.rows[0][0], arrComp.components[0]); assert.strictEqual(arrComp.rows[0][1], arrComp.components[1]); assert.strictEqual(arrComp.rows[1][0], arrComp.components[2]); assert.strictEqual(arrComp.rows[1][1], arrComp.components[3]); + assert.strictEqual(arrComp.rows[2][0], arrComp.components[4]); + assert.strictEqual(arrComp.rows[2][1], arrComp.components[5]); assert.equal(arrComp.components[2].dataValue, 'Sally'); assert.equal(arrComp.components[3].dataValue, 'Thompson'); + assert.equal(arrComp.components[4].dataValue, 'Jane'); + assert.equal(arrComp.components[5].dataValue, 'Doe'); // Ensure it removes rows. employees = [ diff --git a/src/experimental/model/NestedArrayModel.ts b/src/experimental/model/NestedArrayModel.ts index 17a1486f..cac582d3 100644 --- a/src/experimental/model/NestedArrayModel.ts +++ b/src/experimental/model/NestedArrayModel.ts @@ -152,7 +152,7 @@ export function NestedArrayModel(props: any = {}) : ModelDecoratorInterface { // Remove superfluous rows. if (dataValue.length > value.length) { - for (let i = value.length; i < dataValue.length; i++) { + for (let i = dataValue.length - 1; i >= value.length; i--) { this.removeRow(i); } } @@ -192,4 +192,4 @@ export function NestedArrayModel(props: any = {}) : ModelDecoratorInterface { } } } -}; \ No newline at end of file +}; diff --git a/src/process/__tests__/process.test.ts b/src/process/__tests__/process.test.ts index d893c677..dc286f68 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/process/clearHidden.ts b/src/process/clearHidden.ts index 370a844b..03ed8e80 100644 --- a/src/process/clearHidden.ts +++ b/src/process/clearHidden.ts @@ -31,7 +31,7 @@ export const clearHiddenProcess: ProcessorFnSync = (context) = // Check if there's a conditional set for the component and if it's marked as conditionally hidden const isConditionallyHidden = (scope as ConditionsScope).conditionals?.find((cond) => { - return path.includes(cond.path) && cond.conditionallyHidden; + return path === cond.path && cond.conditionallyHidden; }); const shouldClearValueWhenHidden = !component.hasOwnProperty('clearOnHide') || component.clearOnHide; diff --git a/src/process/conditions/__tests__/conditions.test.ts b/src/process/conditions/__tests__/conditions.test.ts index b4956b7e..9b7441a5 100644 --- a/src/process/conditions/__tests__/conditions.test.ts +++ b/src/process/conditions/__tests__/conditions.test.ts @@ -1,5 +1,5 @@ import { expect } from 'chai'; -import { processSync } from '../../process' +import { processSync } from '../../process'; import { conditionProcessInfo } from '../index'; import { ConditionsScope, ProcessContext } from 'types'; import { get } from 'lodash'; @@ -17,127 +17,633 @@ const processForm = (form: any, submission: any) => { }; describe('Condition processor', () => { - it('Should modify component\'s "hidden" property when conditionally visible is false', async () => { - const form = { - components: [ - { - type: 'textfield', - key: 'a', - input: true - }, - { - type: 'textfield', - key: 'b', - input: true, - conditional: { - show: false, - conjunction: 'all', - conditions: [ - { - component: 'a', - operator: 'isEmpty' - } - ] - }, - } - ] - }; - - const submission = { - data: { - a: '', - } - }; - - const context: ProcessContext = processForm(form, submission); - expect(context.components[1]).to.haveOwnProperty('hidden'); - expect(context.components[1].hidden).to.be.true; - }); - - it('Should not define a conditional component (that condition is based on selectBoxes value) as hidden', async () => { - const form1 = { - components: [ - { - label: 'Select Boxes', - optionsLabelPosition: 'right', - tableView: false, - defaultValue: { - '1': false, - '2': false, - '3': false, - test3: false, - }, - values: [ + it('Should modify component\'s "hidden" property when conditionally visible is false', async () => { + const form = { + components: [ + { + type: 'textfield', + key: 'a', + input: true, + }, + { + type: 'textfield', + key: 'b', + input: true, + conditional: { + show: false, + conjunction: 'all', + conditions: [ { - label: '1', - value: '1', - shortcut: '', + component: 'a', + operator: 'isEmpty', }, + ], + }, + }, + ], + }; + + const submission = { + data: { + a: '', + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.components[1]).to.haveOwnProperty('hidden'); + expect(context.components[1].hidden).to.be.true; + }); + + it('Should not define a conditional component (that condition is based on selectBoxes value) as hidden', async () => { + const form1 = { + components: [ + { + label: 'Select Boxes', + optionsLabelPosition: 'right', + tableView: false, + defaultValue: { + '1': false, + '2': false, + '3': false, + test3: false, + }, + values: [ + { + label: '1', + value: '1', + shortcut: '', + }, + { + label: '2', + value: '2', + shortcut: '', + }, + { + label: '3', + value: '3', + shortcut: '', + }, + ], + validateWhenHidden: false, + key: 'selectBoxes', + type: 'selectboxes', + input: true, + inputType: 'checkbox', + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ { - label: '2', - value: '2', - shortcut: '', + component: 'selectBoxes', + operator: 'isEqual', + value: '3', }, + ], + }, + type: 'textfield', + input: true, + }, + { + type: 'button', + label: 'Submit', + key: 'submit', + disableOnInvalid: true, + input: true, + tableView: false, + }, + ], + }; + + const submission1 = { + data: { + selectBoxes: { + '1': false, + '2': false, + '3': true, + }, + textField: 'test', + submit: true, + }, + }; + + const context: ProcessContext = processForm( + form1, + submission1 + ); + + expect(get(context, 'scope.conditionals[0].conditionallyHidden')).to.be + .false; + }); + + it('Should always add components keyed by absolute path to conditional scope (simple components)', async () => { + const form = { + components: [ + { + type: 'textfield', + key: 'a', + input: true, + }, + { + type: 'textfield', + key: 'b', + input: true, + conditional: { + show: false, + conjunction: 'all', + conditions: [ { - label: '3', - value: '3', - shortcut: '', + component: 'a', + operator: 'isEmpty', }, ], - validateWhenHidden: false, - key: 'selectBoxes', - type: 'selectboxes', - input: true, - inputType: 'checkbox', }, - { - label: 'Text Field', - applyMaskOn: 'change', - tableView: true, - validateWhenHidden: false, - key: 'textField', - conditional: { - show: true, - conjunction: 'all', - conditions: [ + }, + ], + }; + + const submission = { + data: { + a: '', + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(1); + expect(context.scope.conditionals?.[0].path).to.equal('b'); + }); + + it('Should always add components keyed by absolute data path to conditional scope (data grid components)', async () => { + const form = { + components: [ + { + label: 'Check Me', + tableView: false, + validateWhenHidden: false, + key: 'checkMe', + type: 'checkbox', + input: true, + defaultValue: false, + }, + { + label: 'Data Grid', + reorder: false, + addAnotherPosition: 'bottom', + layoutFixed: false, + enableRowGroups: false, + initEmpty: false, + tableView: false, + defaultValue: [{}], + validateWhenHidden: false, + key: 'dataGrid', + type: 'datagrid', + input: true, + components: [ + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ + { + component: 'checkMe', + operator: 'isEqual', + value: true, + }, + ], + }, + type: 'textfield', + input: true, + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField1', + type: 'textfield', + input: true, + }, + ], + }, + ], + }; + + const submission = { + data: { + checkMe: false, + dataGrid: [{ textField: 'test' }, { textField: 'test1' }], + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(2); + expect(context.scope.conditionals?.[0].path).to.equal( + 'dataGrid[0].textField' + ); + expect(context.scope.conditionals?.[1].path).to.equal( + 'dataGrid[1].textField' + ); + }); + + it('Should always add components keyed by absolute data path to conditional scope (edit grid components)', async () => { + const form = { + components: [ + { + label: 'Check Me', + tableView: false, + validateWhenHidden: false, + key: 'checkMe', + type: 'checkbox', + input: true, + defaultValue: false, + }, + { + label: 'Edit Grid', + reorder: false, + addAnotherPosition: 'bottom', + layoutFixed: false, + enableRowGroups: false, + initEmpty: false, + tableView: false, + defaultValue: [{}], + validateWhenHidden: false, + key: 'editGrid', + type: 'editgrid', + input: true, + components: [ + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ + { + component: 'checkMe', + operator: 'isEqual', + value: true, + }, + ], + }, + type: 'textfield', + input: true, + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField1', + type: 'textfield', + input: true, + }, + ], + }, + ], + }; + + const submission = { + data: { + checkMe: false, + editGrid: [{ textField: 'test' }, { textField: 'test1' }], + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(2); + expect(context.scope.conditionals?.[0].path).to.equal( + 'editGrid[0].textField' + ); + expect(context.scope.conditionals?.[1].path).to.equal( + 'editGrid[1].textField' + ); + }); + + it('Should always add components keyed by absolute data path to conditional scope (container components)', async () => { + const form = { + components: [ + { + label: 'Check Me', + tableView: false, + validateWhenHidden: false, + key: 'checkMe', + type: 'checkbox', + input: true, + defaultValue: false, + }, + { + label: 'Container', + tableView: false, + key: 'container', + type: 'container', + input: true, + components: [ + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ + { + component: 'checkMe', + operator: 'isEqual', + value: true, + }, + ], + }, + type: 'textfield', + input: true, + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField1', + type: 'textfield', + input: true, + }, + ], + }, + ], + }; + + const submission = { + data: { + checkMe: false, + container: { textField: 'test' }, + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(1); + expect(context.scope.conditionals?.[0].path).to.equal( + 'container.textField' + ); + }); + + it('Should always add components keyed by absolute data path to conditional scope (layout components)', async () => { + const form = { + components: [ + { + label: 'Check Me', + tableView: false, + validateWhenHidden: false, + key: 'checkMe', + type: 'checkbox', + input: true, + defaultValue: false, + }, + { + label: 'Panel', + tableView: false, + key: 'panel', + type: 'panel', + input: true, + components: [ + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ + { + component: 'checkMe', + operator: 'isEqual', + value: true, + }, + ], + }, + type: 'textfield', + input: true, + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField1', + type: 'textfield', + input: true, + }, + ], + }, + ], + }; + + const submission = { + data: { + checkMe: false, + panel: { textField: 'test' }, + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(1); + // Panel components are layout components, so are not pathed + expect(context.scope.conditionals?.[0].path).to.equal('textField'); + }); + + it('Should always add components keyed by absolute data path to conditional scope (deeply nested components)', async () => { + const form = { + components: [ + { + label: 'Tabs', + components: [ + { + label: 'Tab 1', + key: 'tab1', + components: [ { - component: 'selectBoxes', - operator: 'isEqual', - value: '3', + label: 'Check Me', + tableView: false, + validateWhenHidden: false, + key: 'checkMe', + type: 'checkbox', + input: true, + defaultValue: false, }, ], }, - type: 'textfield', - input: true, - }, + { + label: 'Tab 2', + key: 'tab2', + components: [ + { + label: 'Outer Data Grid', + reorder: false, + addAnotherPosition: 'bottom', + layoutFixed: false, + enableRowGroups: false, + initEmpty: false, + tableView: false, + defaultValue: [ + { + outerContainer: { + innerContainer: { + dataGrid: [ + { + textField1: '', + }, + ], + }, + }, + }, + ], + validateWhenHidden: false, + key: 'outerDataGrid', + type: 'datagrid', + input: true, + components: [ + { + label: 'Outer Container', + tableView: false, + validateWhenHidden: false, + key: 'outerContainer', + type: 'container', + input: true, + components: [ + { + label: 'Inner Container', + tableView: false, + validateWhenHidden: false, + key: 'innerContainer', + type: 'container', + input: true, + components: [ + { + label: 'Inner Data Grid', + reorder: false, + addAnotherPosition: 'bottom', + layoutFixed: false, + enableRowGroups: false, + initEmpty: false, + tableView: false, + defaultValue: [ + { + textField1: '', + }, + ], + validateWhenHidden: false, + key: 'innerDataGrid', + type: 'datagrid', + input: true, + components: [ + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField', + conditional: { + show: true, + conjunction: 'all', + conditions: [ + { + component: 'checkMe', + operator: 'isEqual', + value: true, + }, + ], + }, + type: 'textfield', + input: true, + }, + { + label: 'Text Field', + applyMaskOn: 'change', + tableView: true, + validateWhenHidden: false, + key: 'textField1', + type: 'textfield', + input: true, + }, + ], + }, + ], + }, + ], + }, + ], + }, + ], + }, + ], + key: 'tabs', + type: 'tabs', + input: false, + tableView: false, + }, + ], + }; + + const submission = { + data: { + checkMe: false, + outerDataGrid: [ { - type: 'button', - label: 'Submit', - key: 'submit', - disableOnInvalid: true, - input: true, - tableView: false, + outerContainer: { + innerContainer: { + innerDataGrid: [ + { + textField1: 'test', + }, + ], + }, + }, }, ], - }; - - const submission1 = { - data: { - selectBoxes: { - '1': false, - '2': false, - '3': true, - }, - textField: 'test', - submit: true, - }, - }; - - const context: ProcessContext = processForm( - form1, - submission1 - ); - - expect(get(context, 'scope.conditionals[0].conditionallyHidden')).to.be.false; - }); + submit: true, + }, + }; + + const context: ProcessContext = processForm( + form, + submission + ); + expect(context.scope.conditionals).to.have.length(1); + expect(context.scope.conditionals?.[0].path).to.equal( + 'outerDataGrid[0].outerContainer.innerContainer.innerDataGrid[0].textField' + ); + }); }); diff --git a/src/process/conditions/index.ts b/src/process/conditions/index.ts index 211a702e..874a6b79 100644 --- a/src/process/conditions/index.ts +++ b/src/process/conditions/index.ts @@ -79,7 +79,7 @@ export const isConditionallyHidden = (context: ConditionsContext): boolean => { export type ConditionallyHidden = (context: ConditionsContext) => boolean; export const conditionalProcess = (context: ConditionsContext, isHidden: ConditionallyHidden) => { - const { component, row, scope, path } = context; + const { component, scope, path } = context; if (!hasConditions(context)) { return; } diff --git a/src/process/hideChildren.ts b/src/process/hideChildren.ts index 2d2c81de..ec90a3e7 100644 --- a/src/process/hideChildren.ts +++ b/src/process/hideChildren.ts @@ -17,7 +17,7 @@ export const hideChildrenProcessor: ProcessorFnSync = (context) const { component, path, row, scope } = context; // Check if there's a conditional set for the component and if it's marked as conditionally hidden const isConditionallyHidden = scope.conditionals?.find((cond) => { - return path.includes(cond.path) && cond.conditionallyHidden; + return path === cond.path && cond.conditionallyHidden; }); if (component.hidden && isConditionallyHidden) { const info = componentInfo(component); diff --git a/src/process/validation/rules/__tests__/validateDay.test.ts b/src/process/validation/rules/__tests__/validateDay.test.ts index 52abde36..a35f4865 100644 --- a/src/process/validation/rules/__tests__/validateDay.test.ts +++ b/src/process/validation/rules/__tests__/validateDay.test.ts @@ -57,3 +57,73 @@ it('Validating a day component with a valid Date object will return a field erro expect(result).to.be.instanceOf(FieldError); expect(result?.errorKeyOrMessage).to.equal('invalidDay'); }); + +it('Validating a day component with hidden day field with an valid date string value will return null', async () => { + const component = simpleDayField; + component.fields.day.hide = true; + const data = { + component: '03/2023', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.equal(null); +}); + +it('Validating a day component with hidden day field with invalid date will return a field error', async () => { + const component = simpleDayField; + component.fields.day.hide = true; + const data = { + component: '13/2023', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.be.instanceOf(FieldError); + expect(result?.errorKeyOrMessage).to.equal('invalidDay'); +}); + +it('Validating a day component with hidden month field with an valid date string value will return null', async () => { + const component = simpleDayField; + component.fields.month.hide = true; + const data = { + component: '23/2023', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.equal(null); +}); + +it('Validating a day component with hidden month field with invalid date will return a field error', async () => { + const component = simpleDayField; + component.fields.month.hide = true; + const data = { + component: '130/2023', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.be.instanceOf(FieldError); + expect(result?.errorKeyOrMessage).to.equal('invalidDay'); +}); + +it('Validating a day component with hidden year field with an valid date string value will return null', async () => { + const component = simpleDayField; + component.fields.year.hide = true; + const data = { + component: '01/23', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.equal(null); +}); + +it('Validating a day component with hidden year field with invalid date will return a field error', async () => { + const component = simpleDayField; + component.fields.year.hide = true; + const data = { + component: '13/23', + }; + const context = generateProcessorContext(component, data); + const result = await validateDay(context); + expect(result).to.be.instanceOf(FieldError); + expect(result?.errorKeyOrMessage).to.equal('invalidDay'); +}); + diff --git a/src/process/validation/rules/validateDay.ts b/src/process/validation/rules/validateDay.ts index 73ec54bb..d42d63aa 100644 --- a/src/process/validation/rules/validateDay.ts +++ b/src/process/validation/rules/validateDay.ts @@ -50,20 +50,45 @@ export const validateDay: RuleFn = async (context: ValidationContext) => { export const validateDaySync: RuleFnSync = (context: ValidationContext) => { const { component, value } = context; - if (!shouldValidate(context)) { + if (!shouldValidate(context) || !isDayComponent(component)) { return null; } const error = new FieldError('invalidDay', context, 'day'); if (typeof value !== 'string') { return error; } - const [DAY, MONTH, YEAR] = (component as DayComponent).dayFirst ? [0, 1, 2] : [1, 0, 2]; - const values = value.split('/').map((x) => parseInt(x, 10)), - day = values[DAY], - month = values[MONTH], - year = values[YEAR], - maxDay = getDaysInMonthCount(month, year); + let [DAY, MONTH, YEAR] = component.dayFirst ? [0, 1, 2] : [1, 0, 2]; + + const values = value.split('/').map((x) => parseInt(x, 10)); + let day = values[DAY]; + let month = values[MONTH]; + let year = values[YEAR]; + if (values.length !== 3) { + if (component.fields.day.hide) { + MONTH = MONTH === 0 ? 0 : MONTH - 1; + YEAR = YEAR - 1; + day = 0; + month = values[MONTH]; + year = values[YEAR]; + + }; + if (component.fields.month.hide) { + DAY = DAY === 0 ? 0 : DAY - 1; + YEAR = YEAR - 1; + day = values[DAY]; + month = 0; + year = values[YEAR]; + }; + if (component.fields.year.hide) { + day = values[DAY]; + month = values[MONTH]; + year = 0; + }; + } + + const maxDay = getDaysInMonthCount(month, year); + if (isNaN(day) || day < 0 || day > maxDay) { return error; } diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts index a693c886..364814a5 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?._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); 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?._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); const childData: any = get(data, childPath, {}); diff --git a/yarn.lock b/yarn.lock index cbaa9632..ad52440c 100644 --- a/yarn.lock +++ b/yarn.lock @@ -1523,10 +1523,10 @@ core-js@^3.30.2: resolved "https://registry.npmjs.org/core-js/-/core-js-3.30.2.tgz#6528abfda65e5ad728143ea23f7a14f0dcf503fc" integrity sha512-uBJiDmwqsbJCWHAwjrx3cvjbMXP7xD72Dmsn5LOJpiRmE3WbBbN5rCqQ2Qh6Ek6/eOrjlWngEynBWo4VxerQhg== -core-js@^3.37.1: - version "3.37.1" - resolved "https://registry.npmjs.org/core-js/-/core-js-3.37.1.tgz#d21751ddb756518ac5a00e4d66499df981a62db9" - integrity sha512-Xn6qmxrQZyB0FFY8E3bgRXei3lWDJHhvI+u0q9TKIYM49G8pAr0FgnnrFRAmsbptZL1yxRADVXn+x5AGsbBfyw== +core-js@^3.38.0: + version "3.38.1" + resolved "https://registry.yarnpkg.com/core-js/-/core-js-3.38.1.tgz#aa375b79a286a670388a1a363363d53677c0383e" + integrity sha512-OP35aUorbU3Zvlx7pjsFdu1rGNnD4pgw/CWoYzRY3t2EzoVT7shKHY1dlAy3f41cGIO7ZDPQimhGFTlEYkG/Hw== core-util-is@1.0.2: version "1.0.2"