diff --git a/packages/form-js-editor/test/spec/features/properties-panel/PropertiesPanel.spec.js b/packages/form-js-editor/test/spec/features/properties-panel/PropertiesPanel.spec.js index f9157d738..82e0ab829 100644 --- a/packages/form-js-editor/test/spec/features/properties-panel/PropertiesPanel.spec.js +++ b/packages/form-js-editor/test/spec/features/properties-panel/PropertiesPanel.spec.js @@ -3537,6 +3537,7 @@ describe('properties panel', function() { expectPanelStructure(container, { 'General': [ 'Key', + 'Do not submit', 'Target value', 'Compute on' ], @@ -3554,6 +3555,41 @@ describe('properties panel', function() { }); + describe('js function field', function() { + + it('entries', function() { + + // given + const field = schema.components.find(({ type }) => type === 'script'); + + bootstrapPropertiesPanel({ + container, + field + }); + + // then + expectPanelStructure(container, { + 'General': [ + 'Key', + 'Do not submit', + 'Function parameters', + 'Javascript code', + 'Compute on' + ], + 'Condition': [ + 'Deactivate if' + ], + 'Layout': [ + 'Columns' + ], + 'Custom properties': [] + }); + + }); + + }); + + describe('iframe', function() { it('entries', function() { diff --git a/packages/form-js-viewer/test/spec/render/components/form-fields/JSFunctionField.spec.js b/packages/form-js-viewer/test/spec/render/components/form-fields/JSFunctionField.spec.js new file mode 100644 index 000000000..e15ca6398 --- /dev/null +++ b/packages/form-js-viewer/test/spec/render/components/form-fields/JSFunctionField.spec.js @@ -0,0 +1,163 @@ +import { + render +} from '@testing-library/preact/pure'; + +import { JSFunctionField } from '../../../../../src/render/components/form-fields/JSFunctionField'; + +import { MockFormContext } from '../helper'; + +import { act } from 'preact/test-utils'; + +import { + createFormContainer +} from '../../../../TestHelper'; + +let container; + +describe('JSFunctionField', function() { + + beforeEach(function() { + container = createFormContainer(); + }); + + + afterEach(function() { + container.remove(); + }); + + + it('should evaluate with setValue', async function() { + + // given + const onChangeSpy = sinon.spy(); + const field = defaultField; + const passedData = { value : 42 }; + + const services = { + expressionLanguage: { + isExpression: () => true, + evaluate: () => { + return passedData; + } + } + }; + + // when + act(() => { + createJSFunctionField({ field, onChange: onChangeSpy, services }); + }); + + // wait for the iframe to compute the expression and pass it back + await new Promise(r => setTimeout(r, 100)).then(() => { + + // then + expect(onChangeSpy).to.be.calledOnce; + expect(onChangeSpy).to.be.calledWith({ field, value: 42 }); + }); + + }); + + + it('should evaluate with return', async function() { + + // given + const onChangeSpy = sinon.spy(); + const field = { + ...defaultField, + jsFunction: 'return data.value' + }; + const passedData = { value : 42 }; + + const services = { + expressionLanguage: { + isExpression: () => true, + evaluate: () => { + return passedData; + } + } + }; + + // when + act(() => { + createJSFunctionField({ field, onChange: onChangeSpy, services }); + }); + + // wait for the iframe to compute the expression and pass it back + await new Promise(r => setTimeout(r, 100)).then(() => { + + // then + expect(onChangeSpy).to.be.calledOnce; + expect(onChangeSpy).to.be.calledWith({ field, value: 42 }); + }); + + }); + + + it('should evaluate multiple times when using interval', async function() { + + // given + const onChangeSpy = sinon.spy(); + const field = { + ...defaultField, + computeOn: 'interval', + interval: 100 + }; + const passedData = { value : 42 }; + + const services = { + expressionLanguage: { + isExpression: () => true, + evaluate: () => { + return passedData; + } + } + }; + + // when + act(() => { + createJSFunctionField({ field, onChange: onChangeSpy, services }); + }); + + // wait for the iframe to compute the expression and pass it back + await new Promise(r => setTimeout(r, 500)).then(() => { + + // then + + // deliberately underestimating the number of calls to account for potential timing issues + expect(onChangeSpy.callCount > 3).to.be.true; + expect(onChangeSpy).to.be.calledWith({ field, value: 42 }); + }); + + + }); + +}); + +// helpers ////////// + +const defaultField = { + type: 'script', + key: 'jsfunction', + jsFunction: 'setValue(data.value)', + computeOn: 'load' +}; + +function createJSFunctionField({ services, ...restOptions } = {}) { + const options = { + field: defaultField, + onChange: () => {}, + ...restOptions + }; + + return render( + + + , { + container: options.container || container.querySelector('.fjs-form') + } + ); +} \ No newline at end of file