From 024ba1beaa19df4b52ed8b5d3cc2c44f2240145e Mon Sep 17 00:00:00 2001 From: Sergei Maertens Date: Mon, 23 Oct 2023 14:49:56 +0200 Subject: [PATCH 1/2] :test_tube: [#2656] Add regression test for address autofill in editgrid --- src/formio/components/TextField.mocks.js | 8 ++ src/formio/components/TextField.stories.js | 136 ++++++++++++++++++++- 2 files changed, 141 insertions(+), 3 deletions(-) create mode 100644 src/formio/components/TextField.mocks.js diff --git a/src/formio/components/TextField.mocks.js b/src/formio/components/TextField.mocks.js new file mode 100644 index 000000000..a9208aafa --- /dev/null +++ b/src/formio/components/TextField.mocks.js @@ -0,0 +1,8 @@ +import {rest} from 'msw'; + +import {BASE_URL} from 'api-mocks'; + +export const mockAddressAutoCompleteGet = (street = 'Keizersgracht', city = 'Amsterdam') => + rest.get(`${BASE_URL}location/get-street-name-and-city`, (req, res, ctx) => { + return res(ctx.json({streetName: street, city})); + }); diff --git a/src/formio/components/TextField.stories.js b/src/formio/components/TextField.stories.js index dc76e9bd5..923a11ec6 100644 --- a/src/formio/components/TextField.stories.js +++ b/src/formio/components/TextField.stories.js @@ -1,10 +1,15 @@ -import {withUtrechtDocument} from 'story-utils/decorators'; +import {expect} from '@storybook/jest'; +import {userEvent, waitFor, within} from '@storybook/testing-library'; -import {SingleFormioComponent} from './story-util'; +import {ConfigDecorator, withUtrechtDocument} from 'story-utils/decorators'; +import {sleep} from 'utils'; + +import {mockAddressAutoCompleteGet} from './TextField.mocks'; +import {MultipleFormioComponents, SingleFormioComponent} from './story-util'; export default { title: 'Form.io components / Vanilla / TextField', - decorators: [withUtrechtDocument], + decorators: [withUtrechtDocument, ConfigDecorator], args: { type: 'textfield', extraComponentProperties: {}, @@ -22,6 +27,7 @@ export default { }, parameters: { controls: {sort: 'requiredFirst'}, + msw: {handlers: [mockAddressAutoCompleteGet('Keizersgracht', 'Amsterdam')]}, }, }; @@ -32,3 +38,127 @@ export const TextField = { label: 'Onderwerp', }, }; + +export const TextFieldsWithLocation = { + render: MultipleFormioComponents, + args: { + components: [ + { + type: 'textfield', + key: 'postcode', + label: 'Postcode', + inputMask: '9999 AA', + }, + { + type: 'textfield', + key: 'houseNumber', + label: 'Number', + }, + { + type: 'textfield', + key: 'streetname', + label: 'Street', + deriveStreetName: true, + derivePostcode: 'postcode', + deriveHouseNumber: 'houseNumber', + }, + { + type: 'textfield', + key: 'streetname', + label: 'City', + deriveCity: true, + derivePostcode: 'postcode', + deriveHouseNumber: 'houseNumber', + }, + ], + evalContext: {}, + }, + + play: async ({canvasElement, step}) => { + const canvas = within(canvasElement); + // formio... :thisisfine: + await sleep(100); + + await step('Fill out postcode and number', async () => { + await userEvent.type(canvas.getByLabelText('Postcode'), '1015 CJ'); + await userEvent.type(canvas.getByLabelText('Number'), '117'); + }); + + await step('Check that street and city are autofilled', async () => { + await waitFor(async () => { + expect(canvas.getByLabelText('Street')).toHaveDisplayValue('Keizersgracht'); + }); + expect(canvas.getByLabelText('City')).toHaveDisplayValue('Amsterdam'); + }); + }, +}; + +export const TextFieldsWithLocationInEditGrid = { + render: SingleFormioComponent, + args: { + type: 'editgrid', + key: 'addresses', + label: 'Addresses', + groupLabel: '', + maxLength: null, + extraComponentProperties: { + hideLabel: false, + components: [ + { + type: 'textfield', + key: 'postcode', + label: 'Postcode', + inputMask: '9999 AA', + }, + { + type: 'textfield', + key: 'houseNumber', + label: 'Number', + }, + { + type: 'textfield', + key: 'streetname', + label: 'Street', + deriveStreetName: true, + derivePostcode: 'postcode', + deriveHouseNumber: 'houseNumber', + }, + { + type: 'textfield', + key: 'streetname', + label: 'City', + deriveCity: true, + derivePostcode: 'postcode', + deriveHouseNumber: 'houseNumber', + }, + ], + inlineEdit: false, + description: '', + disableAddingRemovingRows: false, + addAnother: 'Add another', + saveRow: 'Confirm', + removeRow: 'Delete', + }, + evalContext: {}, + }, + + play: async ({canvasElement, step}) => { + const canvas = within(canvasElement); + // formio... :thisisfine: + await sleep(100); + + await userEvent.click(canvas.getByRole('button', {name: 'Add another'})); + + await step('Fill out postcode and number', async () => { + await userEvent.type(canvas.getByLabelText('Postcode'), '1015 CJ'); + await userEvent.type(canvas.getByLabelText('Number'), '117'); + }); + + await step('Check that street and city are autofilled', async () => { + await waitFor(async () => { + expect(canvas.getByLabelText('Street')).toHaveDisplayValue('Keizersgracht'); + }); + expect(canvas.getByLabelText('City')).toHaveDisplayValue('Amsterdam'); + }); + }, +}; From 9a6ea7793caae94a4bcb0d6e44b64b2624b49e74 Mon Sep 17 00:00:00 2001 From: Sergei Maertens Date: Mon, 23 Oct 2023 15:01:57 +0200 Subject: [PATCH 2/2] :bug: [open-formulieren/open-forms#2656] Fix address autofill scope The 'row' and 'data' value containers are identical when used outside of a container component, but when inside a container the row provides the local scope. Edit grid is such a container. --- src/formio/components/TextField.js | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/formio/components/TextField.js b/src/formio/components/TextField.js index a046d5dcd..c24e0e1cc 100644 --- a/src/formio/components/TextField.js +++ b/src/formio/components/TextField.js @@ -91,7 +91,7 @@ class TextField extends Formio.Components.components.textfield { fieldLogic(data, row) { const changed = super.fieldLogic(data, row); - this.handleSettingLocationData(data); + this.handleSettingLocationData(row); return changed; } }