diff --git a/src/process/__tests__/process.test.ts b/src/process/__tests__/process.test.ts
index e6847316..bb17faf9 100644
--- a/src/process/__tests__/process.test.ts
+++ b/src/process/__tests__/process.test.ts
@@ -67,6 +67,991 @@ describe('Process Tests', () => {
*/
describe('Process Tests', () => {
+ it('Should submit data within a nested form.', async () => {
+ const form = {
+ _id: {
+ },
+ title: "parent-fio-8023",
+ name: "parentFio8023",
+ path: "parentfio8023",
+ type: "form",
+ display: "wizard",
+ tags: [
+ ],
+ deleted: null,
+ access: [
+ {
+ type: "read_all",
+ roles: [
+ {
+ },
+ {
+ },
+ {
+ },
+ ],
+ },
+ ],
+ submissionAccess: [
+ ],
+ owner: {
+ },
+ components: [
+ {
+ title: "Nested Wizard",
+ breadcrumbClickable: true,
+ buttonSettings: {
+ previous: true,
+ cancel: true,
+ next: true,
+ },
+ collapsible: false,
+ key: "page1",
+ type: "panel",
+ label: "Nested Wizard",
+ tableView: false,
+ components: [
+ {
+ label: "HTML",
+ attrs: [
+ {
+ attr: "",
+ value: "",
+ },
+ ],
+ content: "- Nested Wizard inside Parent Form\n
- Nested Wizard pages should follow the Parent Wizard page\n
- No Nested Wizard pages should display in a Parent Wizard page\n
- Should not be able to submit the Parent Wizard until all the Child and Parent field validation is satisified",
+ refreshOnChange: false,
+ key: "html",
+ type: "htmlelement",
+ tableView: false,
+ input: false,
+ },
+ {
+ label: "Parent Text",
+ tableView: true,
+ validate: {
+ required: true,
+ },
+ key: "parentText",
+ type: "textfield",
+ input: true,
+ },
+ {
+ label: "Parent Number",
+ mask: false,
+ spellcheck: true,
+ tableView: false,
+ delimiter: false,
+ requireDecimal: false,
+ inputFormat: "plain",
+ validate: {
+ required: true,
+ },
+ key: "parentNumber",
+ type: "number",
+ input: true,
+ },
+ {
+ label: "Signature",
+ tableView: false,
+ validate: {
+ required: true,
+ },
+ key: "signature",
+ type: "signature",
+ input: true,
+ },
+ ],
+ input: false,
+ },
+ {
+ title: "Nested EditGrid/DataGrid",
+ breadcrumbClickable: true,
+ buttonSettings: {
+ previous: true,
+ cancel: true,
+ next: true,
+ },
+ collapsible: false,
+ tableView: false,
+ key: "page2",
+ type: "panel",
+ label: "Page 2",
+ input: false,
+ components: [
+ {
+ label: "Form",
+ tableView: true,
+ form: "65ea3662705068f84a93c781",
+ useOriginalRevision: false,
+ key: "form1",
+ type: "form",
+ revision: "1",
+ input: true,
+ components: [
+ {
+ title: "Basic Components",
+ theme: "info",
+ breadcrumbClickable: true,
+ buttonSettings: {
+ previous: true,
+ cancel: true,
+ next: true,
+ },
+ collapsible: false,
+ key: "page1",
+ type: "panel",
+ label: "Basic Components",
+ tableView: false,
+ components: [
+ {
+ label: "Number",
+ mask: false,
+ spellcheck: true,
+ tableView: false,
+ delimiter: false,
+ requireDecimal: false,
+ inputFormat: "plain",
+ validate: {
+ required: true,
+ },
+ key: "number",
+ type: "number",
+ input: true,
+ },
+ {
+ label: "Checkbox",
+ tableView: false,
+ validate: {
+ required: true,
+ },
+ key: "checkbox",
+ type: "checkbox",
+ input: true,
+ defaultValue: false,
+ },
+ {
+ label: "Select Boxes",
+ optionsLabelPosition: "right",
+ tableView: false,
+ values: [
+ {
+ label: "SB - A",
+ value: "sbA",
+ shortcut: "",
+ },
+ {
+ label: "SB - B",
+ value: "sbB",
+ shortcut: "",
+ },
+ {
+ label: "SB - C",
+ value: "sbC",
+ shortcut: "",
+ },
+ {
+ label: "SB - D",
+ value: "sbD",
+ shortcut: "",
+ },
+ ],
+ validate: {
+ required: true,
+ },
+ key: "selectBoxes1",
+ type: "selectboxes",
+ input: true,
+ inputType: "checkbox",
+ defaultValue: {
+ "": false,
+ sbA: false,
+ sbB: false,
+ sbC: false,
+ sbD: false,
+ },
+ },
+ {
+ label: "Select",
+ widget: "choicesjs",
+ tableView: true,
+ data: {
+ values: [
+ {
+ label: "SA",
+ value: "sa",
+ },
+ {
+ label: "Sb",
+ value: "sb",
+ },
+ {
+ label: "SC",
+ value: "sc",
+ },
+ ],
+ },
+ validate: {
+ required: true,
+ },
+ key: "select1",
+ type: "select",
+ input: true,
+ },
+ {
+ label: "Select - URL",
+ widget: "choicesjs",
+ tableView: true,
+ dataSrc: "url",
+ data: {
+ url: "https://cdn.rawgit.com/mshafrir/2646763/raw/states_titlecase.json",
+ headers: [
+ {
+ key: "",
+ value: "",
+ },
+ ],
+ },
+ template: "{{ item.name }}",
+ validate: {
+ required: true,
+ },
+ key: "selectUrl",
+ type: "select",
+ input: true,
+ disableLimit: false,
+ },
+ {
+ label: "Radio",
+ optionsLabelPosition: "right",
+ inline: false,
+ tableView: false,
+ values: [
+ {
+ label: "Ra",
+ value: "ra",
+ shortcut: "",
+ },
+ {
+ label: "Rb",
+ value: "rb",
+ shortcut: "",
+ },
+ {
+ label: "Rc",
+ value: "rc",
+ shortcut: "",
+ },
+ ],
+ validate: {
+ required: true,
+ },
+ key: "radio1",
+ type: "radio",
+ input: true,
+ },
+ ],
+ input: false,
+ },
+ {
+ title: "Advanced",
+ theme: "primary",
+ breadcrumbClickable: true,
+ buttonSettings: {
+ previous: true,
+ cancel: true,
+ next: true,
+ },
+ collapsible: false,
+ key: "page2",
+ type: "panel",
+ label: "Advanced",
+ tableView: false,
+ input: false,
+ components: [
+ {
+ label: "Email",
+ tableView: true,
+ validate: {
+ required: true,
+ },
+ key: "email",
+ type: "email",
+ input: true,
+ },
+ {
+ label: "Url",
+ tableView: true,
+ validate: {
+ required: true,
+ },
+ key: "url",
+ type: "url",
+ input: true,
+ },
+ {
+ label: "Phone Number",
+ tableView: true,
+ validate: {
+ required: true,
+ },
+ key: "phoneNumber",
+ type: "phoneNumber",
+ input: true,
+ },
+ {
+ label: "Tags",
+ tableView: false,
+ validate: {
+ required: true,
+ },
+ key: "tags",
+ type: "tags",
+ input: true,
+ },
+ {
+ label: "Address",
+ tableView: false,
+ provider: "google",
+ validate: {
+ required: true,
+ },
+ key: "address",
+ type: "address",
+ providerOptions: {
+ params: {
+ key: "AIzaSyBNL2e4MnmyPj9zN7SVAe428nCSLP1X144",
+ region: "",
+ autocompleteOptions: {
+ },
+ },
+ },
+ input: true,
+ components: [
+ {
+ label: "Address 1",
+ tableView: false,
+ key: "address1",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Address 2",
+ tableView: false,
+ key: "address2",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "City",
+ tableView: false,
+ key: "city",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "State",
+ tableView: false,
+ key: "state",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Country",
+ tableView: false,
+ key: "country",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Zip Code",
+ tableView: false,
+ key: "zip",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ ],
+ },
+ {
+ label: "Date / Time",
+ tableView: false,
+ enableMinDateInput: false,
+ datePicker: {
+ disableWeekends: false,
+ disableWeekdays: false,
+ },
+ enableMaxDateInput: false,
+ validate: {
+ required: true,
+ },
+ key: "dateTime",
+ type: "datetime",
+ input: true,
+ widget: {
+ type: "calendar",
+ displayInTimezone: "viewer",
+ locale: "en",
+ useLocaleSettings: false,
+ allowInput: true,
+ mode: "single",
+ enableTime: true,
+ noCalendar: false,
+ format: "yyyy-MM-dd hh:mm a",
+ hourIncrement: 1,
+ minuteIncrement: 1,
+ time_24hr: false,
+ minDate: null,
+ disableWeekends: false,
+ disableWeekdays: false,
+ maxDate: null,
+ },
+ },
+ {
+ label: "Day",
+ hideInputLabels: false,
+ inputsLabelPosition: "top",
+ useLocaleSettings: false,
+ tableView: false,
+ fields: {
+ day: {
+ hide: false,
+ required: true,
+ },
+ month: {
+ hide: false,
+ required: true,
+ },
+ year: {
+ hide: false,
+ required: true,
+ },
+ },
+ key: "day",
+ type: "day",
+ input: true,
+ defaultValue: "00/00/0000",
+ },
+ {
+ label: "Time",
+ tableView: true,
+ dataFormat: "HH:mm:ss a",
+ validate: {
+ required: true,
+ },
+ key: "time",
+ type: "time",
+ input: true,
+ inputMask: "99:99",
+ },
+ {
+ label: "Currency",
+ mask: false,
+ spellcheck: true,
+ tableView: false,
+ currency: "USD",
+ inputFormat: "plain",
+ validate: {
+ required: true,
+ },
+ key: "currency",
+ type: "currency",
+ input: true,
+ delimiter: true,
+ },
+ {
+ label: "Signature",
+ tableView: false,
+ validate: {
+ required: true,
+ },
+ key: "signature",
+ type: "signature",
+ input: true,
+ },
+ ],
+ },
+ {
+ title: "DataGrid / EditGrid",
+ theme: "danger",
+ breadcrumbClickable: true,
+ buttonSettings: {
+ previous: true,
+ cancel: true,
+ next: true,
+ },
+ collapsible: false,
+ key: "page3",
+ type: "panel",
+ label: "DataGrid / EditGrid",
+ tableView: false,
+ input: false,
+ components: [
+ {
+ label: "Data Grid",
+ reorder: false,
+ addAnotherPosition: "bottom",
+ defaultOpen: false,
+ layoutFixed: false,
+ enableRowGroups: false,
+ tableView: false,
+ defaultValue: [
+ {
+ },
+ ],
+ key: "dataGrid",
+ type: "datagrid",
+ input: true,
+ components: [
+ {
+ label: "Checkbox",
+ tableView: false,
+ validate: {
+ required: true,
+ },
+ key: "checkbox",
+ type: "checkbox",
+ input: true,
+ defaultValue: false,
+ },
+ {
+ label: "Select",
+ widget: "choicesjs",
+ tableView: true,
+ dataSrc: "resource",
+ data: {
+ resource: "65e9eb1aee138974f569d619",
+ },
+ valueProperty: "data.text",
+ template: "{{ item.data.text }}",
+ validate: {
+ select: false,
+ },
+ key: "select",
+ type: "select",
+ searchField: "data.text__regex",
+ input: true,
+ addResource: false,
+ reference: false,
+ },
+ {
+ label: "Radio",
+ optionsLabelPosition: "right",
+ inline: false,
+ tableView: false,
+ values: [
+ {
+ label: "Ra",
+ value: "ra",
+ shortcut: "",
+ },
+ {
+ label: "Rb",
+ value: "rb",
+ shortcut: "",
+ },
+ {
+ label: "Rc",
+ value: "rc",
+ shortcut: "",
+ },
+ ],
+ validate: {
+ required: true,
+ },
+ key: "radio1",
+ type: "radio",
+ input: true,
+ },
+ ],
+ },
+ {
+ label: "Edit Grid",
+ tableView: true,
+ rowDrafts: false,
+ key: "editGrid",
+ type: "editgrid",
+ input: true,
+ components: [
+ {
+ label: "Date / Time",
+ tableView: true,
+ enableMinDateInput: false,
+ datePicker: {
+ disableWeekends: false,
+ disableWeekdays: false,
+ },
+ enableMaxDateInput: false,
+ validate: {
+ required: true,
+ },
+ key: "dateTime",
+ type: "datetime",
+ input: true,
+ widget: {
+ type: "calendar",
+ displayInTimezone: "viewer",
+ locale: "en",
+ useLocaleSettings: false,
+ allowInput: true,
+ mode: "single",
+ enableTime: true,
+ noCalendar: false,
+ format: "yyyy-MM-dd hh:mm a",
+ hourIncrement: 1,
+ minuteIncrement: 1,
+ time_24hr: false,
+ minDate: null,
+ disableWeekends: false,
+ disableWeekdays: false,
+ maxDate: null,
+ },
+ },
+ {
+ label: "Address",
+ tableView: true,
+ provider: "google",
+ key: "address",
+ type: "address",
+ providerOptions: {
+ params: {
+ key: "AIzaSyBNL2e4MnmyPj9zN7SVAe428nCSLP1X144",
+ region: "",
+ autocompleteOptions: {
+ },
+ },
+ },
+ input: true,
+ components: [
+ {
+ label: "Address 1",
+ tableView: false,
+ key: "address1",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Address 2",
+ tableView: false,
+ key: "address2",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "City",
+ tableView: false,
+ key: "city",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "State",
+ tableView: false,
+ key: "state",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Country",
+ tableView: false,
+ key: "country",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ {
+ label: "Zip Code",
+ tableView: false,
+ key: "zip",
+ type: "textfield",
+ input: true,
+ customConditional: "show = _.get(instance, 'parent.manualMode', false);",
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ },
+ ],
+ settings: {
+ share: {
+ theme: "",
+ showHeader: true,
+ },
+ },
+ properties: {
+ },
+ project: {
+ },
+ controller: "",
+ revisions: "",
+ submissionRevisions: "",
+ _vid: 0,
+ created: "2024-03-07T21:50:03.872Z",
+ modified: "2024-03-07T21:50:03.879Z",
+ machineName: "authoring-oaomxjpqpoigtqg:parentFio8023",
+ __v: 0,
+ };
+ const submission = {
+ data: {
+ parentText: "test",
+ signature: "",
+ form1: {
+ form: "65ea3662705068f84a93c781",
+ owner: "65ea3601c3792e416cabcb2a",
+ roles: [
+ ],
+ access: [
+ ],
+ metadata: {
+ selectData: {
+ form1: {
+ data: {
+ select1: {
+ label: "Sb",
+ },
+ },
+ },
+ },
+ timezone: "America/Chicago",
+ offset: -360,
+ origin: "http://localhost:3000",
+ referrer: "",
+ browserName: "Netscape",
+ userAgent: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36",
+ pathName: "/",
+ onLine: true,
+ headers: {
+ host: "localhost:3000",
+ connection: "keep-alive",
+ "content-length": "9020",
+ pragma: "no-cache",
+ "cache-control": "no-cache",
+ "sec-ch-ua": "\"Chromium\";v=\"122\", \"Not(A:Brand\";v=\"24\", \"Brave\";v=\"122\"",
+ accept: "application/json",
+ "content-type": "application/json",
+ "sec-ch-ua-mobile": "?0",
+ "user-agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36",
+ "sec-ch-ua-platform": "\"macOS\"",
+ "sec-gpc": "1",
+ "accept-language": "en-US,en",
+ origin: "http://localhost:3000",
+ "sec-fetch-site": "same-origin",
+ "sec-fetch-mode": "cors",
+ "sec-fetch-dest": "empty",
+ referer: "http://localhost:3000/",
+ "accept-encoding": "gzip, deflate, br",
+ },
+ },
+ data: {
+ number: 23,
+ checkbox: true,
+ selectBoxes1: {
+ sbA: false,
+ sbB: true,
+ sbC: false,
+ sbD: false,
+ },
+ select1: "sb",
+ selectUrl: {
+ name: "Alaska",
+ abbreviation: "AK",
+ },
+ radio1: "ra",
+ email: "travis@form.io",
+ url: "google.com",
+ phoneNumber: "(234) 234-2342",
+ tags: "test",
+ address: {
+ address_components: [
+ {
+ long_name: "12342",
+ short_name: "12342",
+ types: [
+ "street_number",
+ ],
+ },
+ {
+ long_name: "Coit Road",
+ short_name: "Coit Rd",
+ types: [
+ "route",
+ ],
+ },
+ {
+ long_name: "North Dallas",
+ short_name: "North Dallas",
+ types: [
+ "neighborhood",
+ "political",
+ ],
+ },
+ {
+ long_name: "Dallas",
+ short_name: "Dallas",
+ types: [
+ "locality",
+ "political",
+ ],
+ },
+ {
+ long_name: "Dallas County",
+ short_name: "Dallas County",
+ types: [
+ "administrative_area_level_2",
+ "political",
+ ],
+ },
+ {
+ long_name: "Texas",
+ short_name: "TX",
+ types: [
+ "administrative_area_level_1",
+ "political",
+ ],
+ },
+ {
+ long_name: "United States",
+ short_name: "US",
+ types: [
+ "country",
+ "political",
+ ],
+ },
+ {
+ long_name: "75243",
+ short_name: "75243",
+ types: [
+ "postal_code",
+ ],
+ },
+ {
+ long_name: "2308",
+ short_name: "2308",
+ types: [
+ "postal_code_suffix",
+ ],
+ },
+ ],
+ formatted_address: "12342 Coit Rd, Dallas, TX 75243, USA",
+ geometry: {
+ location: {
+ lat: 32.9165814,
+ lng: -96.76889729999999,
+ },
+ viewport: {
+ south: 32.9151396697085,
+ west: -96.7703730302915,
+ north: 32.9178376302915,
+ east: -96.76767506970849,
+ },
+ },
+ place_id: "ChIJrbdWEhUgTIYRl5rVJe8Zl6A",
+ plus_code: {
+ compound_code: "W68J+JC Dallas, TX, USA",
+ global_code: "8645W68J+JC",
+ },
+ types: [
+ "street_address",
+ ],
+ formattedPlace: "12342 Coit Rd, Dallas, TX 75243, USA",
+ },
+ dateTime: "2024-03-14T17:00:00.000Z",
+ day: "03/23/2029",
+ time: "15:30:00 pm",
+ currency: 2,
+ signature: "",
+ dataGrid: [
+ {
+ checkbox: true,
+ select: "",
+ radio1: "rb",
+ },
+ ],
+ },
+ _id: "65ea36dd705068f84a93c9c3",
+ _fvid: 1,
+ project: "65ea3620705068f84a93c694",
+ state: "submitted",
+ externalIds: [
+ ],
+ created: "2024-03-07T21:51:25.110Z",
+ modified: "2024-03-07T21:51:25.110Z",
+ },
+ parentNumber: 234,
+ },
+ owner: "65ea3601c3792e416cabcb2a",
+ access: [
+ ],
+ metadata: {
+ timezone: "America/Chicago",
+ offset: -360,
+ origin: "http://localhost:3000",
+ referrer: "",
+ browserName: "Netscape",
+ userAgent: "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36",
+ pathName: "/",
+ onLine: true,
+ headers: {
+ "accept-language": "en-US,en",
+ "cache-control": "no-cache",
+ connection: "keep-alive",
+ origin: "http://localhost:3000",
+ pragma: "no-cache",
+ referer: "http://localhost:3000/",
+ "sec-fetch-dest": "empty",
+ "sec-fetch-mode": "cors",
+ "sec-fetch-site": "same-origin",
+ "sec-gpc": "1",
+ "user-agent": "Mozilla/5.0 (Macintosh; Intel Mac OS X 10_15_7) AppleWebKit/537.36 (KHTML, like Gecko) Chrome/122.0.0.0 Safari/537.36",
+ accept: "application/json",
+ "content-type": "application/json",
+ "sec-ch-ua": "\"Chromium\";v=\"122\", \"Not(A:Brand\";v=\"24\", \"Brave\";v=\"122\"",
+ "sec-ch-ua-mobile": "?0",
+ "sec-ch-ua-platform": "\"macOS\"",
+ host: "localhost:3000",
+ "accept-encoding": "gzip, deflate, br",
+ "content-length": "18055",
+ },
+ },
+ _vnote: "",
+ state: "submitted",
+ form: "65ea368b705068f84a93c87a",
+ };
+
+ 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);
+ });
+
it('Should process nested form data correctly', async () => {
const submission = {
data: {
diff --git a/src/utils/formUtil.ts b/src/utils/formUtil.ts
index e1ebdbb1..f54c8e80 100644
--- a/src/utils/formUtil.ts
+++ b/src/utils/formUtil.ts
@@ -369,7 +369,16 @@ export function eachComponent(
writable: true,
value: JSON.parse(JSON.stringify(parent))
});
- component.parent.path = parent.path;
+ Object.defineProperty(component.parent, 'parent', {
+ enumerable: false,
+ writable: true,
+ value: parent.parent
+ });
+ Object.defineProperty(component.parent, 'path', {
+ enumerable: false,
+ writable: true,
+ value: parent.path
+ });
delete component.parent.components;
delete component.parent.componentMap;
delete component.parent.columns;
@@ -447,7 +456,16 @@ export async function eachComponentAsync(
writable: true,
value: JSON.parse(JSON.stringify(parent))
});
- component.parent.path = parent.path;
+ Object.defineProperty(component.parent, 'parent', {
+ enumerable: false,
+ writable: true,
+ value: parent.parent
+ });
+ Object.defineProperty(component.parent, 'path', {
+ enumerable: false,
+ writable: true,
+ value: parent.path
+ });
delete component.parent.components;
delete component.parent.componentMap;
delete component.parent.columns;