react-form
Advanced tools
Comparing version 0.9.0 to 0.9.1
@@ -188,6 +188,2 @@ 'use strict'; | ||
} | ||
// If all errors have been removed, mark the form as globally clean again | ||
if (!newState.errors && this.state.dirty) { | ||
newState.dirty = false; | ||
} | ||
this.setState(newState, function () { | ||
@@ -258,2 +254,2 @@ _this.props.saveState(_this.state, _this.props); | ||
} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/form.js"],"names":["Form","noop","reop","d","FormDefaultProps","loadState","defaultValues","preValidate","validate","onValidationFail","onChange","saveState","willUnmount","preSubmit","onSubmit","postSubmit","config","Comp","createClass","childContextTypes","formAPI","PropTypes","object","getChildContext","getAPI","getDefaultProps","Object","assign","getInitialState","values","clone","props","touched","errors","nestedErrors","componentWillMount","emitChange","state","componentWillReceiveProps","setFormState","componentWillUnmount","setValue","field","value","noTouch","set","getValue","fallback","val","get","setNestedError","getError","setTouched","getTouched","dirty","addValue","removeValue","index","fieldValue","slice","swapValues","destIndex","fieldValues","setAllTouched","submitForm","e","preventDefault","preSubmitValues","newState","silent","setState","initial","removeNestedErrorValues","cleanErrors","render","err","isObject","resolved","mapValues","found","pickBy","keys","length","undefined","isArray","map","find","recurse","path","i","key"],"mappings":"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","file":"form.js","sourcesContent":["import React from 'react'\nimport _ from './utils'\n\nconst noop = () => {}\nconst reop = d => d\n\nexport const FormDefaultProps = {\n  loadState: noop,\n  defaultValues: {},\n  preValidate: reop,\n  validate: () => null,\n  onValidationFail: noop,\n  onChange: noop,\n  saveState: noop,\n  willUnmount: noop,\n  preSubmit: reop,\n  onSubmit: noop,\n  postSubmit: noop\n}\n\nexport default function Form (config = {}) {\n  return (Comp) => {\n    return React.createClass({\n      childContextTypes: {\n        formAPI: React.PropTypes.object\n      },\n      getChildContext () {\n        return {\n          formAPI: this.getAPI()\n        }\n      },\n      // Lifecycle\n      getDefaultProps () {\n        return Object.assign({}, FormDefaultProps, config)\n      },\n      getInitialState () {\n        const values = Object.assign({}, _.clone(config.defaultValues), _.clone(this.props.values))\n        return this.props.loadState(this.props) || {\n          values,\n          touched: {},\n          errors: this.validate(values),\n          nestedErrors: {}\n        }\n      },\n      componentWillMount () {\n        this.emitChange(this.state, true)\n      },\n      componentWillReceiveProps (props) {\n        if (props.values === this.props.values) {\n          return\n        }\n\n        this.setFormState({\n          values: _.clone(props.values) || {}\n        }, true)\n      },\n      componentWillUnmount () {\n        this.props.willUnmount(this.state, this.props)\n      },\n\n      // API\n      setValue (field, value, noTouch) {\n        const state = this.state\n        const values = _.set(state.values, field, value)\n        // Also set touched since the value is changing\n        if (noTouch) {\n          return this.setFormState({values})\n        }\n        const touched = _.set(state.touched, field, value)\n        this.setFormState({values, touched})\n      },\n      getValue (field, fallback) {\n        const state = this.state\n        const val = _.get(state.values, field)\n        return typeof val !== 'undefined' ? val : fallback\n      },\n      setNestedError (field, value = true) {\n        const nestedErrors = _.set(this.state.nestedErrors, field, value)\n        this.setFormState({nestedErrors})\n      },\n      getError (field) {\n        return _.get(this.state.errors, field)\n      },\n      setTouched (field, value = true) {\n        const touched = _.set(this.state.touched, field, value)\n        this.setFormState({touched})\n      },\n      getTouched (field) {\n        const state = this.state\n        if (this.state.dirty === true || this.props.touched === true) {\n          return true\n        }\n        return _.get(state.touched, field)\n      },\n      addValue (field, value) {\n        const state = this.state\n        const values = _.set(state.values, field, [\n          ..._.get(state.values, field, []),\n          value\n        ])\n        this.setFormState({values})\n      },\n      removeValue (field, index) {\n        const state = this.state\n        const fieldValue = _.get(state.values, field, [])\n        const values = _.set(state.values, field, [\n          ...fieldValue.slice(0, index),\n          ...fieldValue.slice(index + 1)\n        ])\n        this.setFormState({values})\n      },\n      swapValues (field, index, destIndex) {\n        const state = this.state\n        const fieldValues = _.get(state.values, field, [])\n        const values = _.set(state.values, field, [\n          ...fieldValues.slice(0, index),\n          fieldValues[destIndex],\n          ...fieldValues.slice(index + 1, destIndex),\n          fieldValues[index],\n          ...fieldValues.slice(destIndex + 1)\n        ])\n        this.setFormState({values})\n      },\n      setAllTouched (dirty = true) {\n        this.setFormState({dirty: !!dirty})\n      },\n      submitForm (e) {\n        e && e.preventDefault && e.preventDefault(e)\n        const state = this.state\n        const errors = this.validate(state.values, state, this.props)\n        if (errors) {\n          if (!state.dirty) {\n            this.setAllTouched()\n          }\n          return this.props.onValidationFail(state, this.props)\n        }\n        const preSubmitValues = this.props.preSubmit(state.values, state, this.props)\n        this.props.onSubmit(preSubmitValues, state, this.props)\n        this.props.postSubmit(preSubmitValues, state, this.props)\n      },\n\n      // Utils\n      getAPI () {\n        return {\n          setValue: this.setValue,\n          getValue: this.getValue,\n          setNestedError: this.setNestedError,\n          getError: this.getError,\n          setTouched: this.setTouched,\n          getTouched: this.getTouched,\n          addValue: this.addValue,\n          removeValue: this.removeValue,\n          swapValues: this.swapValues,\n          setAllTouched: this.setAllTouched,\n          submitForm: this.submitForm\n        }\n      },\n      setFormState (newState, silent) {\n        if (newState && newState.values) {\n          newState.values = this.props.preValidate(newState.values, newState, this.props)\n          newState.errors = this.validate(newState.values, newState, this.props)\n        }\n        // If all errors have been removed, mark the form as globally clean again\n        if (!newState.errors && this.state.dirty) {\n          newState.dirty = false\n        }\n        this.setState(newState, () => {\n          this.props.saveState(this.state, this.props)\n          if (!silent) {\n            this.emitChange(this.state, this.props)\n          }\n        })\n      },\n      emitChange (state, initial) {\n        this.props.onChange(state, this.props, initial)\n      },\n      validate (values) {\n        const errors = this.props.validate(\n          removeNestedErrorValues(values, this.state ? this.state.nestedErrors : {})\n        )\n        return cleanErrors(errors)\n      },\n      // Render\n      render () {\n        const props = {\n          ...this.props,\n          ...this.state,\n          ...this.getAPI()\n        }\n        return (\n          <Comp {...props} />\n        )\n      }\n    })\n  }\n}\n\n// Utils\n\nfunction cleanErrors (err) {\n  if (_.isObject(err)) {\n    const resolved = _.mapValues(err, cleanErrors)\n    const found = _.pickBy(resolved, d => d)\n    return Object.keys(found).length ? resolved : undefined\n  }\n  if (_.isArray(err)) {\n    const resolved = err.map(cleanErrors)\n    const found = resolved.find(d => d)\n    return found ? resolved : undefined\n  }\n  return err\n}\n\nfunction removeNestedErrorValues (value, nestedErrors) {\n  const recurse = (value, path = []) => {\n    if (_.get(nestedErrors, path)) {\n      return undefined\n    }\n    if (_.isObject(value)) {\n      return _.mapValues(value, (d, i) => {\n        return recurse(d, [...path, i])\n      })\n    }\n    if (_.isArray(value)) {\n      return value.map((d, key) => {\n        return recurse(d, [...path, key])\n      })\n    }\n    return value\n  }\n  return recurse(value)\n}\n"]} | ||
//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["../src/form.js"],"names":["Form","noop","reop","d","FormDefaultProps","loadState","defaultValues","preValidate","validate","onValidationFail","onChange","saveState","willUnmount","preSubmit","onSubmit","postSubmit","config","Comp","createClass","childContextTypes","formAPI","PropTypes","object","getChildContext","getAPI","getDefaultProps","Object","assign","getInitialState","values","clone","props","touched","errors","nestedErrors","componentWillMount","emitChange","state","componentWillReceiveProps","setFormState","componentWillUnmount","setValue","field","value","noTouch","set","getValue","fallback","val","get","setNestedError","getError","setTouched","getTouched","dirty","addValue","removeValue","index","fieldValue","slice","swapValues","destIndex","fieldValues","setAllTouched","submitForm","e","preventDefault","preSubmitValues","newState","silent","setState","initial","removeNestedErrorValues","cleanErrors","render","err","isObject","resolved","mapValues","found","pickBy","keys","length","undefined","isArray","map","find","recurse","path","i","key"],"mappings":"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","file":"form.js","sourcesContent":["import React from 'react'\nimport _ from './utils'\n\nconst noop = () => {}\nconst reop = d => d\n\nexport const FormDefaultProps = {\n  loadState: noop,\n  defaultValues: {},\n  preValidate: reop,\n  validate: () => null,\n  onValidationFail: noop,\n  onChange: noop,\n  saveState: noop,\n  willUnmount: noop,\n  preSubmit: reop,\n  onSubmit: noop,\n  postSubmit: noop\n}\n\nexport default function Form (config = {}) {\n  return (Comp) => {\n    return React.createClass({\n      childContextTypes: {\n        formAPI: React.PropTypes.object\n      },\n      getChildContext () {\n        return {\n          formAPI: this.getAPI()\n        }\n      },\n      // Lifecycle\n      getDefaultProps () {\n        return Object.assign({}, FormDefaultProps, config)\n      },\n      getInitialState () {\n        const values = Object.assign({}, _.clone(config.defaultValues), _.clone(this.props.values))\n        return this.props.loadState(this.props) || {\n          values,\n          touched: {},\n          errors: this.validate(values),\n          nestedErrors: {}\n        }\n      },\n      componentWillMount () {\n        this.emitChange(this.state, true)\n      },\n      componentWillReceiveProps (props) {\n        if (props.values === this.props.values) {\n          return\n        }\n\n        this.setFormState({\n          values: _.clone(props.values) || {}\n        }, true)\n      },\n      componentWillUnmount () {\n        this.props.willUnmount(this.state, this.props)\n      },\n\n      // API\n      setValue (field, value, noTouch) {\n        const state = this.state\n        const values = _.set(state.values, field, value)\n        // Also set touched since the value is changing\n        if (noTouch) {\n          return this.setFormState({values})\n        }\n        const touched = _.set(state.touched, field, value)\n        this.setFormState({values, touched})\n      },\n      getValue (field, fallback) {\n        const state = this.state\n        const val = _.get(state.values, field)\n        return typeof val !== 'undefined' ? val : fallback\n      },\n      setNestedError (field, value = true) {\n        const nestedErrors = _.set(this.state.nestedErrors, field, value)\n        this.setFormState({nestedErrors})\n      },\n      getError (field) {\n        return _.get(this.state.errors, field)\n      },\n      setTouched (field, value = true) {\n        const touched = _.set(this.state.touched, field, value)\n        this.setFormState({touched})\n      },\n      getTouched (field) {\n        const state = this.state\n        if (this.state.dirty === true || this.props.touched === true) {\n          return true\n        }\n        return _.get(state.touched, field)\n      },\n      addValue (field, value) {\n        const state = this.state\n        const values = _.set(state.values, field, [\n          ..._.get(state.values, field, []),\n          value\n        ])\n        this.setFormState({values})\n      },\n      removeValue (field, index) {\n        const state = this.state\n        const fieldValue = _.get(state.values, field, [])\n        const values = _.set(state.values, field, [\n          ...fieldValue.slice(0, index),\n          ...fieldValue.slice(index + 1)\n        ])\n        this.setFormState({values})\n      },\n      swapValues (field, index, destIndex) {\n        const state = this.state\n        const fieldValues = _.get(state.values, field, [])\n        const values = _.set(state.values, field, [\n          ...fieldValues.slice(0, index),\n          fieldValues[destIndex],\n          ...fieldValues.slice(index + 1, destIndex),\n          fieldValues[index],\n          ...fieldValues.slice(destIndex + 1)\n        ])\n        this.setFormState({values})\n      },\n      setAllTouched (dirty = true) {\n        this.setFormState({dirty: !!dirty})\n      },\n      submitForm (e) {\n        e && e.preventDefault && e.preventDefault(e)\n        const state = this.state\n        const errors = this.validate(state.values, state, this.props)\n        if (errors) {\n          if (!state.dirty) {\n            this.setAllTouched()\n          }\n          return this.props.onValidationFail(state, this.props)\n        }\n        const preSubmitValues = this.props.preSubmit(state.values, state, this.props)\n        this.props.onSubmit(preSubmitValues, state, this.props)\n        this.props.postSubmit(preSubmitValues, state, this.props)\n      },\n\n      // Utils\n      getAPI () {\n        return {\n          setValue: this.setValue,\n          getValue: this.getValue,\n          setNestedError: this.setNestedError,\n          getError: this.getError,\n          setTouched: this.setTouched,\n          getTouched: this.getTouched,\n          addValue: this.addValue,\n          removeValue: this.removeValue,\n          swapValues: this.swapValues,\n          setAllTouched: this.setAllTouched,\n          submitForm: this.submitForm\n        }\n      },\n      setFormState (newState, silent) {\n        if (newState && newState.values) {\n          newState.values = this.props.preValidate(newState.values, newState, this.props)\n          newState.errors = this.validate(newState.values, newState, this.props)\n        }\n        this.setState(newState, () => {\n          this.props.saveState(this.state, this.props)\n          if (!silent) {\n            this.emitChange(this.state, this.props)\n          }\n        })\n      },\n      emitChange (state, initial) {\n        this.props.onChange(state, this.props, initial)\n      },\n      validate (values) {\n        const errors = this.props.validate(\n          removeNestedErrorValues(values, this.state ? this.state.nestedErrors : {})\n        )\n        return cleanErrors(errors)\n      },\n      // Render\n      render () {\n        const props = {\n          ...this.props,\n          ...this.state,\n          ...this.getAPI()\n        }\n        return (\n          <Comp {...props} />\n        )\n      }\n    })\n  }\n}\n\n// Utils\n\nfunction cleanErrors (err) {\n  if (_.isObject(err)) {\n    const resolved = _.mapValues(err, cleanErrors)\n    const found = _.pickBy(resolved, d => d)\n    return Object.keys(found).length ? resolved : undefined\n  }\n  if (_.isArray(err)) {\n    const resolved = err.map(cleanErrors)\n    const found = resolved.find(d => d)\n    return found ? resolved : undefined\n  }\n  return err\n}\n\nfunction removeNestedErrorValues (value, nestedErrors) {\n  const recurse = (value, path = []) => {\n    if (_.get(nestedErrors, path)) {\n      return undefined\n    }\n    if (_.isObject(value)) {\n      return _.mapValues(value, (d, i) => {\n        return recurse(d, [...path, i])\n      })\n    }\n    if (_.isArray(value)) {\n      return value.map((d, key) => {\n        return recurse(d, [...path, key])\n      })\n    }\n    return value\n  }\n  return recurse(value)\n}\n"]} |
@@ -46,3 +46,3 @@ 'use strict'; | ||
}, | ||
touched ? error : '' | ||
touched && typeof error === 'string' ? error : '' | ||
); | ||
@@ -53,2 +53,2 @@ } | ||
// | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9mb3JtRXJyb3IuanMiXSwibmFtZXMiOlsiRm9ybUVycm9yIiwiZmllbGQiLCJjbGFzc05hbWUiLCJzdHlsZSIsImdldFRvdWNoZWQiLCJnZXRFcnJvciIsInRvdWNoZWQiLCJlcnJvciIsInN0eWxlcyIsImRpc3BsYXkiLCJjbGFzc2VzIiwiT2JqZWN0IiwiYXNzaWduIl0sIm1hcHBpbmdzIjoiOzs7OztrQkFLd0JBLFM7O0FBTHhCOzs7O0FBQ0E7Ozs7QUFFQTs7Ozs7O0FBRWUsU0FBU0EsU0FBVCxPQUErQztBQUFBLE1BQTFCQyxLQUEwQixRQUExQkEsS0FBMEI7QUFBQSxNQUFuQkMsU0FBbUIsUUFBbkJBLFNBQW1CO0FBQUEsTUFBUkMsS0FBUSxRQUFSQSxLQUFROztBQUM1RCxTQUNFO0FBQUE7QUFBQSxNQUFXLE9BQU9GLEtBQWxCO0FBQ0cscUJBQTRCO0FBQUEsVUFBMUJHLFVBQTBCLFNBQTFCQSxVQUEwQjtBQUFBLFVBQWRDLFFBQWMsU0FBZEEsUUFBYzs7QUFDM0IsVUFBTUMsVUFBVUYsWUFBaEI7QUFDQSxVQUFNRyxRQUFRRixVQUFkO0FBQ0EsVUFBTUcsU0FBUztBQUNiQyxpQkFBU0gsV0FBV0MsS0FBWCxHQUFtQixPQUFuQixHQUE2QjtBQUR6QixPQUFmO0FBR0EsVUFBTUcsVUFBVSwwQkFBVyxXQUFYLEVBQXdCUixTQUF4QixDQUFoQjtBQUNBLGFBQ0U7QUFBQTtBQUFBO0FBQ0UscUJBQVdRLE9BRGI7QUFFRSxpQkFBT0MsT0FBT0MsTUFBUCxDQUFjLEVBQWQsRUFBa0JKLE1BQWxCLEVBQTBCTCxLQUExQjtBQUZUO0FBSUdHLGtCQUFVQyxLQUFWLEdBQWtCO0FBSnJCLE9BREY7QUFRRDtBQWhCSCxHQURGO0FBb0JEO0FBeEJEIiwiZmlsZSI6ImZvcm1FcnJvci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCBjbGFzc25hbWVzIGZyb20gJ2NsYXNzbmFtZXMnXG4vL1xuaW1wb3J0IEZvcm1GaWVsZCBmcm9tICcuL2Zvcm1GaWVsZCdcblxuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gRm9ybUVycm9yICh7ZmllbGQsIGNsYXNzTmFtZSwgc3R5bGV9KSB7XG4gIHJldHVybiAoXG4gICAgPEZvcm1GaWVsZCBmaWVsZD17ZmllbGR9PlxuICAgICAgeyh7Z2V0VG91Y2hlZCwgZ2V0RXJyb3J9KSA9PiB7XG4gICAgICAgIGNvbnN0IHRvdWNoZWQgPSBnZXRUb3VjaGVkKClcbiAgICAgICAgY29uc3QgZXJyb3IgPSBnZXRFcnJvcigpXG4gICAgICAgIGNvbnN0IHN0eWxlcyA9IHtcbiAgICAgICAgICBkaXNwbGF5OiB0b3VjaGVkICYmIGVycm9yID8gJ2Jsb2NrJyA6ICdub25lJ1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IGNsYXNzZXMgPSBjbGFzc25hbWVzKCdGb3JtRXJyb3InLCBjbGFzc05hbWUpXG4gICAgICAgIHJldHVybiAoXG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgY2xhc3NOYW1lPXtjbGFzc2VzfVxuICAgICAgICAgICAgc3R5bGU9e09iamVjdC5hc3NpZ24oe30sIHN0eWxlcywgc3R5bGUpfVxuICAgICAgICAgID5cbiAgICAgICAgICAgIHt0b3VjaGVkID8gZXJyb3IgOiAnJ31cbiAgICAgICAgICA8L2Rpdj5cbiAgICAgICAgKVxuICAgICAgfX1cbiAgICA8L0Zvcm1GaWVsZD5cbiAgKVxufVxuIl19 | ||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJzb3VyY2VzIjpbIi4uL3NyYy9mb3JtRXJyb3IuanMiXSwibmFtZXMiOlsiRm9ybUVycm9yIiwiZmllbGQiLCJjbGFzc05hbWUiLCJzdHlsZSIsImdldFRvdWNoZWQiLCJnZXRFcnJvciIsInRvdWNoZWQiLCJlcnJvciIsInN0eWxlcyIsImRpc3BsYXkiLCJjbGFzc2VzIiwiT2JqZWN0IiwiYXNzaWduIl0sIm1hcHBpbmdzIjoiOzs7OztrQkFLd0JBLFM7O0FBTHhCOzs7O0FBQ0E7Ozs7QUFFQTs7Ozs7O0FBRWUsU0FBU0EsU0FBVCxPQUErQztBQUFBLE1BQTFCQyxLQUEwQixRQUExQkEsS0FBMEI7QUFBQSxNQUFuQkMsU0FBbUIsUUFBbkJBLFNBQW1CO0FBQUEsTUFBUkMsS0FBUSxRQUFSQSxLQUFROztBQUM1RCxTQUNFO0FBQUE7QUFBQSxNQUFXLE9BQU9GLEtBQWxCO0FBQ0cscUJBQTRCO0FBQUEsVUFBMUJHLFVBQTBCLFNBQTFCQSxVQUEwQjtBQUFBLFVBQWRDLFFBQWMsU0FBZEEsUUFBYzs7QUFDM0IsVUFBTUMsVUFBVUYsWUFBaEI7QUFDQSxVQUFNRyxRQUFRRixVQUFkO0FBQ0EsVUFBTUcsU0FBUztBQUNiQyxpQkFBU0gsV0FBV0MsS0FBWCxHQUFtQixPQUFuQixHQUE2QjtBQUR6QixPQUFmO0FBR0EsVUFBTUcsVUFBVSwwQkFBVyxXQUFYLEVBQXdCUixTQUF4QixDQUFoQjtBQUNBLGFBQ0U7QUFBQTtBQUFBO0FBQ0UscUJBQVdRLE9BRGI7QUFFRSxpQkFBT0MsT0FBT0MsTUFBUCxDQUFjLEVBQWQsRUFBa0JKLE1BQWxCLEVBQTBCTCxLQUExQjtBQUZUO0FBSUdHLG1CQUFXLE9BQU9DLEtBQVAsS0FBaUIsUUFBNUIsR0FBdUNBLEtBQXZDLEdBQStDO0FBSmxELE9BREY7QUFRRDtBQWhCSCxHQURGO0FBb0JEO0FBeEJEIiwiZmlsZSI6ImZvcm1FcnJvci5qcyIsInNvdXJjZXNDb250ZW50IjpbImltcG9ydCBSZWFjdCBmcm9tICdyZWFjdCdcbmltcG9ydCBjbGFzc25hbWVzIGZyb20gJ2NsYXNzbmFtZXMnXG4vL1xuaW1wb3J0IEZvcm1GaWVsZCBmcm9tICcuL2Zvcm1GaWVsZCdcblxuZXhwb3J0IGRlZmF1bHQgZnVuY3Rpb24gRm9ybUVycm9yICh7ZmllbGQsIGNsYXNzTmFtZSwgc3R5bGV9KSB7XG4gIHJldHVybiAoXG4gICAgPEZvcm1GaWVsZCBmaWVsZD17ZmllbGR9PlxuICAgICAgeyh7Z2V0VG91Y2hlZCwgZ2V0RXJyb3J9KSA9PiB7XG4gICAgICAgIGNvbnN0IHRvdWNoZWQgPSBnZXRUb3VjaGVkKClcbiAgICAgICAgY29uc3QgZXJyb3IgPSBnZXRFcnJvcigpXG4gICAgICAgIGNvbnN0IHN0eWxlcyA9IHtcbiAgICAgICAgICBkaXNwbGF5OiB0b3VjaGVkICYmIGVycm9yID8gJ2Jsb2NrJyA6ICdub25lJ1xuICAgICAgICB9XG4gICAgICAgIGNvbnN0IGNsYXNzZXMgPSBjbGFzc25hbWVzKCdGb3JtRXJyb3InLCBjbGFzc05hbWUpXG4gICAgICAgIHJldHVybiAoXG4gICAgICAgICAgPGRpdlxuICAgICAgICAgICAgY2xhc3NOYW1lPXtjbGFzc2VzfVxuICAgICAgICAgICAgc3R5bGU9e09iamVjdC5hc3NpZ24oe30sIHN0eWxlcywgc3R5bGUpfVxuICAgICAgICAgID5cbiAgICAgICAgICAgIHt0b3VjaGVkICYmIHR5cGVvZiBlcnJvciA9PT0gJ3N0cmluZycgPyBlcnJvciA6ICcnfVxuICAgICAgICAgIDwvZGl2PlxuICAgICAgICApXG4gICAgICB9fVxuICAgIDwvRm9ybUZpZWxkPlxuICApXG59XG4iXX0= |
{ | ||
"name": "react-form", | ||
"version": "0.9.0", | ||
"version": "0.9.1", | ||
"description": "A fast, lightweight, opinionated table and datagrid built on React", | ||
@@ -5,0 +5,0 @@ "license": "MIT", |
(function(f){if(typeof exports==="object"&&typeof module!=="undefined"){module.exports=f()}else if(typeof define==="function"&&define.amd){define([],f)}else{var g;if(typeof window!=="undefined"){g=window}else if(typeof global!=="undefined"){g=global}else if(typeof self!=="undefined"){g=self}else{g=this}g.reactForm = f()}})(function(){var define,module,exports;return (function e(t,n,r){function s(o,u){if(!n[o]){if(!t[o]){var a=typeof require=="function"&&require;if(!u&&a)return a(o,!0);if(i)return i(o,!0);var f=new Error("Cannot find module '"+o+"'");throw f.code="MODULE_NOT_FOUND",f}var l=n[o]={exports:{}};t[o][0].call(l.exports,function(e){var n=t[o][1][e];return s(n?n:e)},l,l.exports,e,t,n,r)}return n[o].exports}var i=typeof require=="function"&&require;for(var o=0;o<r.length;o++)s(r[o]);return s})({1:[function(require,module,exports){ | ||
"use strict";function _interopRequireDefault(t){return t&&t.__esModule?t:{default:t}}function _toConsumableArray(t){if(Array.isArray(t)){for(var e=0,s=Array(t.length);e<t.length;e++)s[e]=t[e];return s}return Array.from(t)}function Form(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{};return function(e){return _react2.default.createClass({childContextTypes:{formAPI:_react2.default.PropTypes.object},getChildContext:function(){return{formAPI:this.getAPI()}},getDefaultProps:function(){return Object.assign({},FormDefaultProps,t)},getInitialState:function(){var e=Object.assign({},_utils2.default.clone(t.defaultValues),_utils2.default.clone(this.props.values));return this.props.loadState(this.props)||{values:e,touched:{},errors:this.validate(e),nestedErrors:{}}},componentWillMount:function(){this.emitChange(this.state,!0)},componentWillReceiveProps:function(t){t.values!==this.props.values&&this.setFormState({values:_utils2.default.clone(t.values)||{}},!0)},componentWillUnmount:function(){this.props.willUnmount(this.state,this.props)},setValue:function(t,e,s){var r=this.state,a=_utils2.default.set(r.values,t,e);if(s)return this.setFormState({values:a});var u=_utils2.default.set(r.touched,t,e);this.setFormState({values:a,touched:u})},getValue:function(t,e){var s=this.state,r=_utils2.default.get(s.values,t);return"undefined"!=typeof r?r:e},setNestedError:function(t){var e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1],s=_utils2.default.set(this.state.nestedErrors,t,e);this.setFormState({nestedErrors:s})},getError:function(t){return _utils2.default.get(this.state.errors,t)},setTouched:function(t){var e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1],s=_utils2.default.set(this.state.touched,t,e);this.setFormState({touched:s})},getTouched:function(t){var e=this.state;return this.state.dirty===!0||this.props.touched===!0||_utils2.default.get(e.touched,t)},addValue:function(t,e){var s=this.state,r=_utils2.default.set(s.values,t,[].concat(_toConsumableArray(_utils2.default.get(s.values,t,[])),[e]));this.setFormState({values:r})},removeValue:function(t,e){var s=this.state,r=_utils2.default.get(s.values,t,[]),a=_utils2.default.set(s.values,t,[].concat(_toConsumableArray(r.slice(0,e)),_toConsumableArray(r.slice(e+1))));this.setFormState({values:a})},swapValues:function(t,e,s){var r=this.state,a=_utils2.default.get(r.values,t,[]),u=_utils2.default.set(r.values,t,[].concat(_toConsumableArray(a.slice(0,e)),[a[s]],_toConsumableArray(a.slice(e+1,s)),[a[e]],_toConsumableArray(a.slice(s+1))));this.setFormState({values:u})},setAllTouched:function(){var t=!(arguments.length>0&&void 0!==arguments[0])||arguments[0];this.setFormState({dirty:!!t})},submitForm:function(t){t&&t.preventDefault&&t.preventDefault(t);var e=this.state,s=this.validate(e.values,e,this.props);if(s)return e.dirty||this.setAllTouched(),this.props.onValidationFail(e,this.props);var r=this.props.preSubmit(e.values,e,this.props);this.props.onSubmit(r,e,this.props),this.props.postSubmit(r,e,this.props)},getAPI:function(){return{setValue:this.setValue,getValue:this.getValue,setNestedError:this.setNestedError,getError:this.getError,setTouched:this.setTouched,getTouched:this.getTouched,addValue:this.addValue,removeValue:this.removeValue,swapValues:this.swapValues,setAllTouched:this.setAllTouched,submitForm:this.submitForm}},setFormState:function(t,e){var s=this;t&&t.values&&(t.values=this.props.preValidate(t.values,t,this.props),t.errors=this.validate(t.values,t,this.props)),!t.errors&&this.state.dirty&&(t.dirty=!1),this.setState(t,function(){s.props.saveState(s.state,s.props),e||s.emitChange(s.state,s.props)})},emitChange:function(t,e){this.props.onChange(t,this.props,e)},validate:function(t){var e=this.props.validate(removeNestedErrorValues(t,this.state?this.state.nestedErrors:{}));return cleanErrors(e)},render:function(){var t=_extends({},this.props,this.state,this.getAPI());return _react2.default.createElement(e,t)}})}}function cleanErrors(t){if(_utils2.default.isObject(t)){var e=_utils2.default.mapValues(t,cleanErrors),s=_utils2.default.pickBy(e,function(t){return t});return Object.keys(s).length?e:void 0}if(_utils2.default.isArray(t)){var r=t.map(cleanErrors),a=r.find(function(t){return t});return a?r:void 0}return t}function removeNestedErrorValues(t,e){var s=function t(s){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:[];if(!_utils2.default.get(e,r))return _utils2.default.isObject(s)?_utils2.default.mapValues(s,function(e,s){return t(e,[].concat(_toConsumableArray(r),[s]))}):_utils2.default.isArray(s)?s.map(function(e,s){return t(e,[].concat(_toConsumableArray(r),[s]))}):s};return s(t)}Object.defineProperty(exports,"__esModule",{value:!0}),exports.FormDefaultProps=void 0;var _extends=Object.assign||function(t){for(var e=1;e<arguments.length;e++){var s=arguments[e];for(var r in s)Object.prototype.hasOwnProperty.call(s,r)&&(t[r]=s[r])}return t};exports.default=Form;var _react=require("react"),_react2=_interopRequireDefault(_react),_utils=require("./utils"),_utils2=_interopRequireDefault(_utils),noop=function(){},reop=function(t){return t},FormDefaultProps=exports.FormDefaultProps={loadState:noop,defaultValues:{},preValidate:reop,validate:function(){return null},onValidationFail:noop,onChange:noop,saveState:noop,willUnmount:noop,preSubmit:reop,onSubmit:noop,postSubmit:noop}; | ||
"use strict";function _interopRequireDefault(t){return t&&t.__esModule?t:{default:t}}function _toConsumableArray(t){if(Array.isArray(t)){for(var e=0,s=Array(t.length);e<t.length;e++)s[e]=t[e];return s}return Array.from(t)}function Form(){var t=arguments.length>0&&void 0!==arguments[0]?arguments[0]:{};return function(e){return _react2.default.createClass({childContextTypes:{formAPI:_react2.default.PropTypes.object},getChildContext:function(){return{formAPI:this.getAPI()}},getDefaultProps:function(){return Object.assign({},FormDefaultProps,t)},getInitialState:function(){var e=Object.assign({},_utils2.default.clone(t.defaultValues),_utils2.default.clone(this.props.values));return this.props.loadState(this.props)||{values:e,touched:{},errors:this.validate(e),nestedErrors:{}}},componentWillMount:function(){this.emitChange(this.state,!0)},componentWillReceiveProps:function(t){t.values!==this.props.values&&this.setFormState({values:_utils2.default.clone(t.values)||{}},!0)},componentWillUnmount:function(){this.props.willUnmount(this.state,this.props)},setValue:function(t,e,s){var r=this.state,a=_utils2.default.set(r.values,t,e);if(s)return this.setFormState({values:a});var u=_utils2.default.set(r.touched,t,e);this.setFormState({values:a,touched:u})},getValue:function(t,e){var s=this.state,r=_utils2.default.get(s.values,t);return"undefined"!=typeof r?r:e},setNestedError:function(t){var e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1],s=_utils2.default.set(this.state.nestedErrors,t,e);this.setFormState({nestedErrors:s})},getError:function(t){return _utils2.default.get(this.state.errors,t)},setTouched:function(t){var e=!(arguments.length>1&&void 0!==arguments[1])||arguments[1],s=_utils2.default.set(this.state.touched,t,e);this.setFormState({touched:s})},getTouched:function(t){var e=this.state;return this.state.dirty===!0||this.props.touched===!0||_utils2.default.get(e.touched,t)},addValue:function(t,e){var s=this.state,r=_utils2.default.set(s.values,t,[].concat(_toConsumableArray(_utils2.default.get(s.values,t,[])),[e]));this.setFormState({values:r})},removeValue:function(t,e){var s=this.state,r=_utils2.default.get(s.values,t,[]),a=_utils2.default.set(s.values,t,[].concat(_toConsumableArray(r.slice(0,e)),_toConsumableArray(r.slice(e+1))));this.setFormState({values:a})},swapValues:function(t,e,s){var r=this.state,a=_utils2.default.get(r.values,t,[]),u=_utils2.default.set(r.values,t,[].concat(_toConsumableArray(a.slice(0,e)),[a[s]],_toConsumableArray(a.slice(e+1,s)),[a[e]],_toConsumableArray(a.slice(s+1))));this.setFormState({values:u})},setAllTouched:function(){var t=!(arguments.length>0&&void 0!==arguments[0])||arguments[0];this.setFormState({dirty:!!t})},submitForm:function(t){t&&t.preventDefault&&t.preventDefault(t);var e=this.state,s=this.validate(e.values,e,this.props);if(s)return e.dirty||this.setAllTouched(),this.props.onValidationFail(e,this.props);var r=this.props.preSubmit(e.values,e,this.props);this.props.onSubmit(r,e,this.props),this.props.postSubmit(r,e,this.props)},getAPI:function(){return{setValue:this.setValue,getValue:this.getValue,setNestedError:this.setNestedError,getError:this.getError,setTouched:this.setTouched,getTouched:this.getTouched,addValue:this.addValue,removeValue:this.removeValue,swapValues:this.swapValues,setAllTouched:this.setAllTouched,submitForm:this.submitForm}},setFormState:function(t,e){var s=this;t&&t.values&&(t.values=this.props.preValidate(t.values,t,this.props),t.errors=this.validate(t.values,t,this.props)),this.setState(t,function(){s.props.saveState(s.state,s.props),e||s.emitChange(s.state,s.props)})},emitChange:function(t,e){this.props.onChange(t,this.props,e)},validate:function(t){var e=this.props.validate(removeNestedErrorValues(t,this.state?this.state.nestedErrors:{}));return cleanErrors(e)},render:function(){var t=_extends({},this.props,this.state,this.getAPI());return _react2.default.createElement(e,t)}})}}function cleanErrors(t){if(_utils2.default.isObject(t)){var e=_utils2.default.mapValues(t,cleanErrors),s=_utils2.default.pickBy(e,function(t){return t});return Object.keys(s).length?e:void 0}if(_utils2.default.isArray(t)){var r=t.map(cleanErrors),a=r.find(function(t){return t});return a?r:void 0}return t}function removeNestedErrorValues(t,e){var s=function t(s){var r=arguments.length>1&&void 0!==arguments[1]?arguments[1]:[];if(!_utils2.default.get(e,r))return _utils2.default.isObject(s)?_utils2.default.mapValues(s,function(e,s){return t(e,[].concat(_toConsumableArray(r),[s]))}):_utils2.default.isArray(s)?s.map(function(e,s){return t(e,[].concat(_toConsumableArray(r),[s]))}):s};return s(t)}Object.defineProperty(exports,"__esModule",{value:!0}),exports.FormDefaultProps=void 0;var _extends=Object.assign||function(t){for(var e=1;e<arguments.length;e++){var s=arguments[e];for(var r in s)Object.prototype.hasOwnProperty.call(s,r)&&(t[r]=s[r])}return t};exports.default=Form;var _react=require("react"),_react2=_interopRequireDefault(_react),_utils=require("./utils"),_utils2=_interopRequireDefault(_utils),noop=function(){},reop=function(t){return t},FormDefaultProps=exports.FormDefaultProps={loadState:noop,defaultValues:{},preValidate:reop,validate:function(){return null},onValidationFail:noop,onChange:noop,saveState:noop,willUnmount:noop,preSubmit:reop,onSubmit:noop,postSubmit:noop}; | ||
},{"./utils":11,"react":"react"}],2:[function(require,module,exports){ | ||
"use strict";function _interopRequireDefault(e){return e&&e.__esModule?e:{default:e}}function FormError(e){var r=e.field,t=e.className,a=e.style;return _react2.default.createElement(_formField2.default,{field:r},function(e){var r=e.getTouched,l=e.getError,s=r(),u=l(),o={display:s&&u?"block":"none"},i=(0,_classnames2.default)("FormError",t);return _react2.default.createElement("div",{className:i,style:Object.assign({},o,a)},s?u:"")})}Object.defineProperty(exports,"__esModule",{value:!0}),exports.default=FormError;var _react=require("react"),_react2=_interopRequireDefault(_react),_classnames=require("classnames"),_classnames2=_interopRequireDefault(_classnames),_formField=require("./formField"),_formField2=_interopRequireDefault(_formField); | ||
"use strict";function _interopRequireDefault(e){return e&&e.__esModule?e:{default:e}}function FormError(e){var r=e.field,t=e.className,a=e.style;return _react2.default.createElement(_formField2.default,{field:r},function(e){var r=e.getTouched,l=e.getError,s=r(),o=l(),u={display:s&&o?"block":"none"},i=(0,_classnames2.default)("FormError",t);return _react2.default.createElement("div",{className:i,style:Object.assign({},u,a)},s&&"string"==typeof o?o:"")})}Object.defineProperty(exports,"__esModule",{value:!0}),exports.default=FormError;var _react=require("react"),_react2=_interopRequireDefault(_react),_classnames=require("classnames"),_classnames2=_interopRequireDefault(_classnames),_formField=require("./formField"),_formField2=_interopRequireDefault(_formField); | ||
@@ -7,0 +7,0 @@ },{"./formField":3,"classnames":12,"react":"react"}],3:[function(require,module,exports){ |
224
README.md
@@ -28,3 +28,3 @@ <div align="center"> | ||
- [Installation](#installation) | ||
- [Quick Example](#quick-example) | ||
- [Example](#example) | ||
- [API](#api) | ||
@@ -47,6 +47,6 @@ - [{ Form }](#-form-) | ||
## Usage | ||
## Example | ||
```javascript | ||
import React from 'react' | ||
import { Form, Text, Checkbox, Textarea, Select } from 'react-form' | ||
import { Form, Text, Select, Textarea, NestedForm, FormError } from 'react-form' | ||
@@ -60,18 +60,20 @@ // To create a new form, simply call `Form(config)(component)` | ||
defaultValues: { | ||
status: 'single' | ||
friends: [], | ||
} | ||
friends: [] | ||
}, | ||
// Validating your form is super easy, just use the `validate` life-cycle method | ||
validate: values => { | ||
// To create form errors, return an object that maps field names with error messages. | ||
// There are many ways to do this, below is just one example: | ||
const { name, hobby, status, friends, address } = values | ||
return { | ||
// If a field has a validation error, set that field to a string that is the error message for that field | ||
// If the field is valid, return any falsey value to mark it as valid. | ||
name: !values.name ? 'A name is required' : undefined, | ||
hobby: (values.hobby && values.hobby.length < 5) ? 'Your hobby must be at least 5 characters long' : false, | ||
status: !values.status ? 'A status is required' : null, | ||
address: !values.address ? 'A valid address is required' : 0 | ||
// you don't need to return anything at all for fields you don't need to validate | ||
name: !name ? 'A name is required' : undefined, | ||
hobby: (hobby && hobby.length < 5) ? 'Your hobby must be at least 5 characters long' : false, | ||
status: !status ? 'A status is required' : null, | ||
friends: (!friends || !friends.length) ? 'You need at least one friend!' : friends.map(friend => { | ||
const { name, relationship } = friend | ||
return { | ||
name: !name ? 'A name is required' : undefined, | ||
relationship: !relationship ? 'A relationship is required' : undefined | ||
} | ||
}), | ||
address: !address ? 'A valid address is required' : 0 | ||
} | ||
@@ -84,3 +86,3 @@ }, | ||
} | ||
})(({ values, submitForm, addValue, removeValue }) => { | ||
})(({ values, submitForm, addValue, removeValue, getError }) => { | ||
// This is a stateless component, but you can use any valid react component to render your form. | ||
@@ -92,80 +94,110 @@ // Forms also supply plenty of useful props for your components to utilize. See the docs for a complete list. | ||
<Text // This is the built-in Text formInput | ||
field='name' // field is a string version of the field location | ||
placeholder='Full Name' // all other props are sent through to the underlying component, in this case an <input /> | ||
/> | ||
<label> | ||
<span>Full Name</span> | ||
<Text // This is the built-in Text formInput | ||
field='name' // field is a string version of the field location | ||
placeholder='Your name' // all other props are sent through to the underlying component, in this case an <input /> | ||
/> | ||
</label> | ||
<Select // This is the built-in Select formInput | ||
field='status' | ||
options={[{ // You can ship it some options like usual | ||
label: 'Single', | ||
value: 'single' | ||
}, { | ||
label: 'In a Relationship', | ||
value: 'relationship' | ||
}, { | ||
label: 'It\'s Complicated', | ||
value: 'complicated' | ||
}]} | ||
/> | ||
<label> | ||
<span>Relationship Status</span> | ||
<Select // This is the built-in Select formInput | ||
field='status' | ||
options={[{ // You can ship it some options like usual | ||
label: 'Single', | ||
value: 'single' | ||
}, { | ||
label: 'In a Relationship', | ||
value: 'relationship' | ||
}, { | ||
label: 'It\'s Complicated', | ||
value: 'complicated' | ||
}]} | ||
/> | ||
</label> | ||
<Textarea // This is the built-in Textarea formInput | ||
field='bio' | ||
placeholder='Short Bio' | ||
/> | ||
<label> | ||
<span>Short Bio</span> | ||
<Textarea // This is the built-in Textarea formInput | ||
field='bio' | ||
placeholder='Short Bio' | ||
/> | ||
</label> | ||
// Arrays in forms are super easy to handle | ||
{values.friends.map((friends, i) => ( // Loop over the values however you'd like | ||
<div> | ||
<label> | ||
<span>Friends</span> | ||
</label> | ||
<Text | ||
field={['friends', i, 'name']} // You can easily pass an array-style field path. Perfect for passing down as props or nested values | ||
placeholder='Friend Name' | ||
/> | ||
{/* Arrays in forms are super easy to handle */} | ||
{/* This is a custom form error for the root of the friends list (see validation function) */} | ||
<FormError field='friends' /> | ||
<div className='nested'> | ||
{!values.friends.length ? ( | ||
<em>No friends have been added yet</em> | ||
) : values.friends.map((friends, i) => ( // Loop over the values however you'd like | ||
<div key={i}> | ||
<Select | ||
field={`friends.${i}.relationship`} // If you don't like arrays, you can also use a string template | ||
options={[{ | ||
label: 'Friend', | ||
value: 'friend' | ||
}, { | ||
label: 'Acquaintance', | ||
value: 'acquaintance' | ||
}, { | ||
label: 'Colleague', | ||
value: 'colleague' | ||
}]} | ||
/> | ||
<label> | ||
<span>Full Name</span> | ||
<Text | ||
field={['friends', i, 'name']} // You can easily pass an array-style field path. Perfect for passing down as props or nested values | ||
placeholder='Friend Name' | ||
/> | ||
</label> | ||
// This button will remove this friend from the `friends` field | ||
<button | ||
type='button' | ||
onClick={removeValue(['friends', i])} // `removeValue` takes a field location for an item in an array | ||
> | ||
Remove Friend | ||
</button> | ||
<label> | ||
<span>Full Name</span> | ||
<Select | ||
field={`friends.${i}.relationship`} // If you don't like arrays, you can also use a string template | ||
options={[{ | ||
label: 'Friend', | ||
value: 'friend' | ||
}, { | ||
label: 'Acquaintance', | ||
value: 'acquaintance' | ||
}, { | ||
label: 'Colleague', | ||
value: 'colleague' | ||
}]} | ||
/> | ||
</label> | ||
</div> | ||
))} | ||
<button // This button will remove this friend from the `friends` field | ||
type='button' | ||
onClick={() => removeValue('friends', i)} // `removeValue` takes a field location for an array, and the index for the item to remove | ||
> | ||
Remove Friend | ||
</button> | ||
// This button will add a new blank friend item to the `friends` field | ||
<button | ||
</div> | ||
))} | ||
</div> | ||
<button // This button will add a new blank friend item to the `friends` field | ||
type='button' | ||
onClick={addValue('friends', {})} // `addValue` takes an array-like field, and the value to add | ||
> | ||
Remove Friend | ||
onClick={() => addValue('friends', {})} // `addValue` takes an array-like field, and the value to add | ||
> | ||
Add Friend | ||
</button> | ||
// An address has a couple of parts to it, and will probably have its own validation function. | ||
// Let's make it reusable by using a nested form | ||
<br /> | ||
<br /> | ||
<label> | ||
<span>Address</span> | ||
</label> | ||
{/* An address has a couple of parts to it, and will probably have its own validation function. */} | ||
{/* Let's make it reusable by using a nested form */} | ||
<NestedForm | ||
form={AddressForm} // This is just another form that we built below | ||
field='address' // The results of this nested form will be set here on this form. | ||
field='address' // The results of this nested form will be set to this field value on this form. | ||
/> | ||
// Since this is the parent form, let's put a submit button in there ;) | ||
// You can submit your form however you want, as long as you call the `submitForm` callback | ||
{/* // Since this is the parent form, let's put a submit button in there ;) */} | ||
{/* // You can submit your form however you want, as long as you call the `submitForm` callback */} | ||
<button> | ||
Submit | ||
</button> | ||
</form> | ||
@@ -180,8 +212,8 @@ ) | ||
return { | ||
street: !values.street ? 'street' : undefined | ||
city: !values.city ? 'city' : undefined | ||
state: !values.state ? 'state' : undefined | ||
street: !values.street ? 'A street is required' : undefined, | ||
city: !values.city ? 'A city is required' : undefined, | ||
state: !values.state ? 'A state is required' : undefined | ||
} | ||
} | ||
})(({values}) => { | ||
})(() => { | ||
return ( | ||
@@ -191,3 +223,3 @@ <div> | ||
field='street' | ||
placeholder='Address' | ||
placeholder='Street' | ||
/> | ||
@@ -206,14 +238,14 @@ <Text | ||
export default props => { | ||
export default () => { | ||
return ( | ||
<MyForm | ||
// If you wanted to feed your form some existing values, use the `values` prop | ||
values={{ | ||
name: 'Tanner Linsley' | ||
}} | ||
onSubmit={(values) => { | ||
window.alert(JSON.stringify(values, null, 2)) | ||
}} | ||
/> | ||
<div> | ||
<div className='table-wrap'> | ||
<MyForm | ||
onSubmit={(values) => { | ||
window.alert(JSON.stringify(values, null, 2)) | ||
}} | ||
// For more available props, see the docs! | ||
/> | ||
</div> | ||
</div> | ||
) | ||
@@ -406,3 +438,3 @@ } | ||
- `nestedErrors` is an object map indicating any nested forms that did not pass their own validation | ||
- This prop is not recommended for displaying errors. For that, we recommend relying on a `FormInput` or using the `FieldError` component. | ||
- This prop is not recommended for displaying nested form errors. For that, we recommend relying on a `FormInput` or using the `FieldError` component. | ||
- If a nested form contains an error in its own validation lifecycle method, its corresponding `nestedErrors` field will be set to true. | ||
@@ -430,6 +462,6 @@ - Defaults to `{}` | ||
Form()( | ||
({nestedErrors}) => { | ||
console.log(nestedErrors) | ||
({touched}) => { | ||
console.log(touched) | ||
// { | ||
// nestedForm: true // there must be an error in the form located at the "nestedForm" field | ||
// touched: { myField: true } // the `myField` field has been touched | ||
// } | ||
@@ -436,0 +468,0 @@ } |
172212
852
781