From 51f54ea144ea574ce7cfcab9f35fee6256218304 Mon Sep 17 00:00:00 2001 From: jsomsanith Date: Mon, 15 May 2017 15:04:37 +0200 Subject: [PATCH] Custom validation --- packages/forms/src/UIForm/UIForm.component.js | 92 +++++++++++-------- packages/forms/src/UIForm/utils/validation.js | 33 +++++-- packages/forms/stories/index.js | 9 ++ .../stories/json/core-custom-validation.json | 32 +++++++ 4 files changed, 122 insertions(+), 44 deletions(-) create mode 100644 packages/forms/stories/json/core-custom-validation.json diff --git a/packages/forms/src/UIForm/UIForm.component.js b/packages/forms/src/UIForm/UIForm.component.js index 26609a1327..61fbcc2b26 100644 --- a/packages/forms/src/UIForm/UIForm.component.js +++ b/packages/forms/src/UIForm/UIForm.component.js @@ -1,9 +1,9 @@ import React, { PropTypes } from 'react'; import { reduxForm } from 'redux-form'; -import { merge, validate } from 'talend-json-schema-form-core'; +import { merge } from 'talend-json-schema-form-core'; import Widget from './Widget'; -import { validateAll } from './utils/validation'; +import { validate, validateAll } from './utils/validation'; import { mutateValue } from './utils/properties'; const TRIGGER_AFTER = 'after'; @@ -52,50 +52,62 @@ class UIForm extends React.Component { */ consolidate(event, schema, value) { this.setState( - prevState => ({ - properties: mutateValue(prevState.properties, schema.key, value), - validations: { + (prevState) => { + const properties = mutateValue(prevState.properties, schema.key, value); + const validations = { ...prevState.validations, - [schema.key]: validate(schema, value), - }, - }), - () => { - const { onChange, onTrigger } = this.props; - - if (onChange) { - onChange({ - jsonSchema: this.props.data.jsonSchema, // original jsonSchema - uiSchema: this.props.data.uiSchema, // original uiSchema - properties: this.state.properties, // current properties values - }); - } - - const { key, triggers } = schema; - if (onTrigger && triggers && triggers.indexOf(TRIGGER_AFTER) !== -1) { - onTrigger( - this.state.properties, // current properties values - key[key.length - 1], // field name - value // field value - ); - } - } + [schema.key]: validate(schema, value, properties, this.props.validation), + }; + return { properties, validations }; + }, + () => this.handleChangesCallbacks(schema, value) ); } + /** + * Triggers the onTrigger and onChange if needed + * - onChange : at each field change + * - onTrigger : when schema.trigger : ['after'] + * @param schema The field schema + * @param value The new value + */ + handleChangesCallbacks(schema, value) { + const { onChange, onTrigger } = this.props; + + if (onChange) { + onChange({ + jsonSchema: this.props.data.jsonSchema, // original jsonSchema + uiSchema: this.props.data.uiSchema, // original uiSchema + properties: this.state.properties, // current properties values + }); + } + + const { key, triggers } = schema; + if (onTrigger && triggers && triggers.indexOf(TRIGGER_AFTER) !== -1) { + onTrigger( + this.state.properties, // current properties values + key[key.length - 1], // field name + value // field value + ); + } + } + /** * Triggers a validation and update state. * @returns {boolean} true if the form is valid, false otherwise */ isValid() { - const validations = validateAll(this.state.mergedSchema, this.state.properties); - const keys = Object.keys(validations); - for (const key of keys) { - if (!validations[key].valid) { - this.setState({ validations }); - return false; - } + const validations = validateAll( + this.state.mergedSchema, + this.state.properties, + this.props.validation + ); + + const isValid = Object.keys(validations).every(key => validations[key].valid); + if (!isValid) { + this.setState({ validations }); } - return true; + return isValid; } /** @@ -152,9 +164,17 @@ if (process.env.NODE_ENV !== 'production') { onSubmit: PropTypes.func.isRequired, /** * Tigger > after callback. + * Prototype: function onTrigger(properties, fieldName, value) * This is executed on changes on fields with uiSchema > triggers : ['after'] */ onTrigger: PropTypes.func, + /** + * Custom validation function. + * Prototype: function validation(properties, fieldName, value) + * Return format : { valid: true|false, error: { message: 'my validation message' } } + * This is triggered on fields that has their uiSchema > customValidation : true + */ + validation: PropTypes.func, }; } diff --git a/packages/forms/src/UIForm/utils/validation.js b/packages/forms/src/UIForm/utils/validation.js index 6959ccbedc..443d075463 100644 --- a/packages/forms/src/UIForm/utils/validation.js +++ b/packages/forms/src/UIForm/utils/validation.js @@ -1,22 +1,39 @@ -import { validate } from 'talend-json-schema-form-core'; +import { validate as staticValidate } from 'talend-json-schema-form-core'; import { getValue } from './properties'; /** * Validate values. - * @param mergedSchema The merged schema. + * @param schema The merged schema. + * @param value The value. * @param properties The values. + * @param customValidationFn A custom validation function + * that is applied on schema.customValidation = true + * @returns {object} The validation result. + */ +export function validate(schema, value, properties, customValidationFn) { + const staticResult = staticValidate(schema, value); + if (staticResult.valid && schema.customValidation && customValidationFn) { + return customValidationFn(properties, schema, value); + } + return staticResult; +} + +/** + * Validate values. + * @param mergedSchema The merged schema array. + * @param properties The values. + * @param customValidationFn A custom validation function + * that is applied on schema.customValidation = true * @returns {object} The validation result by field. */ -export function validateAll(mergedSchema, properties) { +export function validateAll(mergedSchema, properties, customValidationFn) { const validations = {}; mergedSchema.forEach((schema) => { const { key, items } = schema; if (key) { - validations[key] = validate( - schema, - getValue(properties, key) - ); + const value = getValue(properties, key); + validations[key] = validate(schema, value, properties, customValidationFn); } if (items) { const subValidations = validateAll(items, properties); @@ -27,7 +44,7 @@ export function validateAll(mergedSchema, properties) { } /** - * Check if a schema value is invalid. + * Check if a schema value is valid. * It is invalid if : * - the schema is an invalid field (validations[key] = { valid: false }) * - the schema has items (ex: fieldset, tabs, ...), and at least one of them is invalid diff --git a/packages/forms/stories/index.js b/packages/forms/stories/index.js index 8c72fe8825..d1e2a45650 100644 --- a/packages/forms/stories/index.js +++ b/packages/forms/stories/index.js @@ -65,6 +65,15 @@ sampleFilenames onTrigger={action('Trigger')} onBlur={action('Blur')} onSubmit={action('Submit')} + validation={(properties, schema, value) => { + action('customValidation')(properties, schema, value); + return { + valid: value.length < 5, + error: { + message: 'Custom validation : The value should be less than 5 chars', + }, + }; + }} /> )); diff --git a/packages/forms/stories/json/core-custom-validation.json b/packages/forms/stories/json/core-custom-validation.json new file mode 100644 index 0000000000..4559c93779 --- /dev/null +++ b/packages/forms/stories/json/core-custom-validation.json @@ -0,0 +1,32 @@ +{ + "jsonSchema": { + "type": "object", + "title": "Comment", + "properties": { + "lastname": { + "type": "string" + }, + "firstname": { + "type": "string" + } + }, + "required": [ + "lastname", + "firstname" + ] + }, + "uiSchema": [ + { + "key": "lastname", + "title": "Last Name", + "description": "This field has custom validation (less than 5 chars)", + "customValidation": true + }, + { + "key": "firstname", + "title": "First Name", + "description": "This field has no custom validation" + } + ], + "properties": {} +}