From 385263c383577c94a8195396583871f778630918 Mon Sep 17 00:00:00 2001 From: christianalfoni Date: Mon, 23 Feb 2015 08:55:36 +0100 Subject: [PATCH] Merged two pull requests --- README.md | 4 + bower.json | 2 +- build/formsy-react.js | 18 +++- build/specs.js | 162 ++++++++++++++++++++++++++++-------- package.json | 2 +- release/formsy-react.js | 16 +++- release/formsy-react.min.js | 2 +- specs/Formsy-spec.js | 56 +++++-------- src/main.js | 3 + 9 files changed, 188 insertions(+), 77 deletions(-) diff --git a/README.md b/README.md index bafa788..a4dfd87 100644 --- a/README.md +++ b/README.md @@ -68,6 +68,10 @@ The main concept is that forms, inputs and validation is done very differently a ## Changes +**0.7.1** + - Fixed bug where external update of value on pristine form element did not update the form model (Thanks @sdemjanenko) + - Fixed bug where children are null/undefined (Thanks @sdemjanenko) + **0.7.0** - Dynamic form elements. Add them at any point and they will be registered with the form - **onChange()** handler is called whenever an form element has changed its value or a new form element is added to the form diff --git a/bower.json b/bower.json index 5a45961..bf15c70 100644 --- a/bower.json +++ b/bower.json @@ -1,6 +1,6 @@ { "name": "formsy-react", - "version": "0.7.0", + "version": "0.7.1", "main": "src/main.js", "dependencies": { "react": "^0.11.2" diff --git a/build/formsy-react.js b/build/formsy-react.js index 78f641f..80b3ddd 100755 --- a/build/formsy-react.js +++ b/build/formsy-react.js @@ -162,6 +162,18 @@ Formsy.Mixin = { nextProps._validate = this.props._validate; }, + componentDidUpdate: function(prevProps, prevState) { + + // If the input is untouched and something outside changes the value + // update the FORM model by re-attaching to the form + if (this.state._isPristine) { + if (this.props.value !== prevProps.value && this.state._value === prevProps.value) { + this.state._value = this.props.value || ''; + this.props._attachToForm(this); + } + } + }, + // Detach it when component unmounts componentWillUnmount: function () { this.props._detachFromForm(this); @@ -353,13 +365,13 @@ Formsy.Form = React.createClass({displayName: "Form", registerInputs: function (children) { React.Children.forEach(children, function (child) { - if (child.props && child.props.name) { + if (child && child.props && child.props.name) { child.props._attachToForm = this.attachToForm; child.props._detachFromForm = this.detachFromForm; child.props._validate = this.validate; } - if (child.props && child.props.children) { + if (child && child.props && child.props.children) { this.registerInputs(child.props.children); } @@ -517,4 +529,4 @@ module.exports = Formsy; }).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) },{"react":"react"}]},{},["./src/main.js"]) -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","src/main.js"],"names":[],"mappings":"AAAA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})","var React = global.React || require('react');\nvar Formsy = {};\nvar validationRules = {\n  'isValue': function (value) {\n    return value !== '';\n  },\n  'isEmail': function (value) {\n    return value.match(/^((([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+(\\.([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+)*)|((\\x22)((((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(([\\x01-\\x08\\x0b\\x0c\\x0e-\\x1f\\x7f]|\\x21|[\\x23-\\x5b]|[\\x5d-\\x7e]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(\\\\([\\x01-\\x09\\x0b\\x0c\\x0d-\\x7f]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF]))))*(((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(\\x22)))@((([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))\\.)+(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))$/i);\n  },\n  'isTrue': function (value) {\n    return value === true;\n  },\n  'isNumeric': function (value) {\n    if (typeof value === 'number') {\n      return true;\n    } else {\n      return value.match(/^-?[0-9]+$/);\n    }\n  },\n  'isAlpha': function (value) {\n    return value.match(/^[a-zA-Z]+$/);\n  },\n  'isWords': function (value) {\n    return value.match(/^[a-zA-Z\\s]+$/);\n  },\n  'isSpecialWords': function (value) {\n    return value.match(/^[a-zA-Z\\s\\u00C0-\\u017F]+$/);\n  },\n  isLength: function (value, min, max) {\n    if (max !== undefined) {\n      return value.length >= min && value.length <= max;\n    }\n    return value.length >= min;\n  },\n  equals: function (value, eql) {\n    return value == eql;\n  },\n  equalsField: function (value, field) {\n    return value === this[field];\n  }\n};\n\nvar toURLEncoded = function (element, key, list) {\n  var list = list || [];\n  if (typeof (element) == 'object') {\n    for (var idx in element)\n      toURLEncoded(element[idx], key ? key + '[' + idx + ']' : idx, list);\n  } else {\n    list.push(key + '=' + encodeURIComponent(element));\n  }\n  return list.join('&');\n};\n\nvar request = function (method, url, data, contentType, headers) {\n\n  var contentType = contentType === 'urlencoded' ? 'application/' + contentType.replace('urlencoded', 'x-www-form-urlencoded') : 'application/json';\n  data = contentType === 'application/json' ? JSON.stringify(data) : toURLEncoded(data);\n\n  return new Promise(function (resolve, reject) {\n    try {\n      var xhr = new XMLHttpRequest();\n      xhr.open(method, url, true);\n      xhr.setRequestHeader('Accept', 'application/json');\n      xhr.setRequestHeader('Content-Type', contentType);\n\n      // Add passed headers\n      Object.keys(headers).forEach(function (header) {\n        xhr.setRequestHeader(header, headers[header]);\n      });\n\n      xhr.onreadystatechange = function () {\n        if (xhr.readyState === 4) {\n\n          try {\n            var response = xhr.responseText ? JSON.parse(xhr.responseText) : null;\n            if (xhr.status >= 200 && xhr.status < 300) {\n              resolve(response);\n            } else {\n              reject(response);\n            }\n          } catch (e) {\n            reject(e);\n          }\n\n        }\n      };\n      xhr.send(data);\n    } catch (e) {\n      reject(e);\n    }\n  });\n\n};\n\nvar arraysDiffer = function (arrayA, arrayB) {\n  var isDifferent = false;\n  if (arrayA.length !== arrayB.length) {\n    isDifferent = true;\n  } else {\n    arrayA.forEach(function (item, index) {\n      if (item !== arrayB[index]) {\n        isDifferent = true;\n      }\n    });\n  }\n  return isDifferent;\n};\n\nvar ajax = {\n  post: request.bind(null, 'POST'),\n  put: request.bind(null, 'PUT')\n};\nvar options = {};\n\nFormsy.defaults = function (passedOptions) {\n  options = passedOptions;\n};\n\nFormsy.Mixin = {\n  getInitialState: function () {\n    return {\n      _value: this.props.value ? this.props.value : '',\n      _isValid: true,\n      _isPristine: true\n    };\n  },\n  componentWillMount: function () {\n\n    var configure = function () {\n      // Add validations to the store itself as the props object can not be modified\n      this._validations = this.props.validations || '';\n\n      if (this.props.required) {\n        this._validations = this.props.validations ? this.props.validations + ',' : '';\n        this._validations += 'isValue';\n      }\n      this.props._attachToForm(this);\n    }.bind(this);\n\n    if (!this.props.name) {\n      throw new Error('Form Input requires a name property when used');\n    }\n\n    if (!this.props._attachToForm) {\n      return setTimeout(function () {\n        if (!this.props._attachToForm) {\n          throw new Error('Form Mixin requires component to be nested in a Form');\n        }\n        configure();\n      }.bind(this), 0);\n    }\n    configure();\n\n  },\n\n  // We have to make the validate method is kept when new props are added\n  componentWillReceiveProps: function (nextProps) {\n    nextProps._attachToForm = this.props._attachToForm;\n    nextProps._detachFromForm = this.props._detachFromForm;\n    nextProps._validate = this.props._validate;\n  },\n\n  // Detach it when component unmounts\n  componentWillUnmount: function () {\n    this.props._detachFromForm(this);\n  },\n\n  // We validate after the value has been set\n  setValue: function (value) {\n    this.setState({\n      _value: value,\n      _isPristine: false\n    }, function () {\n      this.props._validate(this);\n    }.bind(this));\n  },\n  resetValue: function () {\n    this.setState({\n      _value: '',\n      _isPristine: true\n    }, function () {\n      this.props._validate(this);\n    });\n  },\n  getValue: function () {\n    return this.state._value;\n  },\n  hasValue: function () {\n    return this.state._value !== '';\n  },\n  getErrorMessage: function () {\n    return this.isValid() || this.showRequired() ? null : this.state._serverError || this.props.validationError;\n  },\n  isValid: function () {\n    return this.state._isValid;\n  },\n  isPristine: function () {\n    return this.state._isPristine;\n  },\n  isRequired: function () {\n    return !!this.props.required;\n  },\n  showRequired: function () {\n    return this.isRequired() && this.state._value === '';\n  },\n  showError: function () {\n    return !this.showRequired() && !this.state._isValid;\n  }\n};\n\nFormsy.addValidationRule = function (name, func) {\n  validationRules[name] = func;\n};\n\nFormsy.Form = React.createClass({displayName: \"Form\",\n  getInitialState: function () {\n    return {\n      isValid: true,\n      isSubmitting: false,\n      canChange: false\n    };\n  },\n  getDefaultProps: function () {\n    return {\n      headers: {},\n      onSuccess: function () {},\n      onError: function () {},\n      onSubmit: function () {},\n      onSubmitted: function () {},\n      onValid: function () {},\n      onInvalid: function () {},\n      onChange: function () {}\n    };\n  },\n\n  // Add a map to store the inputs of the form, a model to store\n  // the values of the form and register child inputs\n  componentWillMount: function () {\n    this.inputs = {};\n    this.model = {};\n    this.registerInputs(this.props.children);\n  },\n\n  componentDidMount: function () {\n    this.validateForm();\n  },\n\n  componentWillUpdate: function () {\n    var inputKeys = Object.keys(this.inputs);\n\n    // The updated children array is not available here for some reason,\n    // we need to wait for next event loop\n    setTimeout(function () {\n      this.registerInputs(this.props.children);\n\n      var newInputKeys = Object.keys(this.inputs);\n      if (arraysDiffer(inputKeys, newInputKeys)) {\n        this.validateForm();\n      }\n    }.bind(this), 0);\n  },\n\n  // Update model, submit to url prop and send the model\n  submit: function (event) {\n    event.preventDefault();\n\n    // Trigger form as not pristine.\n    // If any inputs have not been touched yet this will make them dirty\n    // so validation becomes visible (if based on isPristine)\n    this.setFormPristine(false);\n\n    // To support use cases where no async or request operation is needed.\n    // The \"onSubmit\" callback is called with the model e.g. {fieldName: \"myValue\"},\n    // if wanting to reset the entire form to original state, the second param is a callback for this.\n    if (!this.props.url) {\n      this.updateModel();\n      this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n      return;\n    }\n\n    this.updateModel();\n    this.setState({\n      isSubmitting: true\n    });\n\n    this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n\n    var headers = (Object.keys(this.props.headers).length && this.props.headers) || options.headers || {};\n\n    var method = this.props.method && ajax[this.props.method.toLowerCase()] ? this.props.method.toLowerCase() : 'post';\n    ajax[method](this.props.url, this.mapModel(), this.props.contentType || options.contentType || 'json', headers)\n      .then(function (response) {\n        this.props.onSuccess(response);\n        this.props.onSubmitted();\n      }.bind(this))\n      .catch(this.failSubmit);\n  },\n\n  mapModel: function () {\n    return this.props.mapping ? this.props.mapping(this.model) : this.model;\n  },\n\n  // Goes through all registered components and\n  // updates the model values\n  updateModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      var component = this.inputs[name];\n      this.model[name] = component.state._value;\n    }.bind(this));\n  },\n\n  // Reset each key in the model to the original / initial value\n  resetModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      this.inputs[name].resetValue();\n    }.bind(this));\n    this.validateForm();\n  },\n\n  // Go through errors from server and grab the components\n  // stored in the inputs map. Change their state to invalid\n  // and set the serverError message\n  updateInputsWithError: function (errors) {\n    Object.keys(errors).forEach(function (name, index) {\n      var component = this.inputs[name];\n\n      if (!component) {\n        throw new Error('You are trying to update an input that does not exists. Verify errors object with input names. ' + JSON.stringify(errors));\n      }\n\n      var args = [{\n        _isValid: false,\n        _serverError: errors[name]\n      }];\n      component.setState.apply(component, args);\n    }.bind(this));\n  },\n\n  failSubmit: function (errors) {\n    this.updateInputsWithError(errors);\n    this.setState({\n      isSubmitting: false\n    });\n    this.props.onError(errors);\n    this.props.onSubmitted();\n  },\n\n  // Traverse the children and children of children to find\n  // all inputs by checking the name prop. Maybe do a better\n  // check here\n  registerInputs: function (children) {\n    React.Children.forEach(children, function (child) {\n\n      if (child.props && child.props.name) {\n        child.props._attachToForm = this.attachToForm;\n        child.props._detachFromForm = this.detachFromForm;\n        child.props._validate = this.validate;\n      }\n\n      if (child.props && child.props.children) {\n        this.registerInputs(child.props.children);\n      }\n\n    }.bind(this));\n  },\n\n  getCurrentValues: function () {\n    return Object.keys(this.inputs).reduce(function (data, name) {\n      var component = this.inputs[name];\n      data[name] = component.state._value;\n      return data;\n    }.bind(this), {});\n  },\n\n  setFormPristine: function (isPristine) {\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // Iterate through each component and set it as pristine\n    // or \"dirty\".\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      component.setState({\n        _isPristine: isPristine\n      });\n    }.bind(this));\n  },\n\n  // Use the binded values and the actual input value to\n  // validate the input and set its state. Then check the\n  // state of the form itself\n  validate: function (component) {\n\n    // Trigger onChange\n    this.state.canChange && this.props.onChange && this.props.onChange(this.getCurrentValues());\n\n    if (!component.props.required && !component._validations) {\n      return;\n    }\n\n    // Run through the validations, split them up and call\n    // the validator IF there is a value or it is required\n    var isValid = this.runValidation(component);\n\n    component.setState({\n      _isValid: isValid,\n      _serverError: null\n    }, this.validateForm);\n\n  },\n\n  runValidation: function (component) {\n    var isValid = true;\n    if (component._validations.length && (component.props.required || component.state._value !== '')) {\n      component._validations.split(',').forEach(function (validation) {\n        var args = validation.split(':');\n        var validateMethod = args.shift();\n        args = args.map(function (arg) {\n          try {\n            return JSON.parse(arg);\n          } catch (e) {\n            return arg; // It is a string if it can not parse it\n          }\n        });\n        args = [component.state._value].concat(args);\n        if (!validationRules[validateMethod]) {\n          throw new Error('Formsy does not have the validation rule: ' + validateMethod);\n        }\n        if (!validationRules[validateMethod].apply(this.getCurrentValues(), args)) {\n          isValid = false;\n        }\n      }.bind(this));\n    }\n    return isValid;\n  },\n\n  // Validate the form by going through all child input components\n  // and check their state\n  validateForm: function () {\n    var allIsValid = true;\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // We need a callback as we are validating all inputs again. This will\n    // run when the last component has set its state\n    var onValidationComplete = function () {\n      inputKeys.forEach(function (name) {\n        if (!inputs[name].state._isValid) {\n          allIsValid = false;\n        }\n      }.bind(this));\n\n      this.setState({\n        isValid: allIsValid\n      });\n\n      allIsValid && this.props.onValid();\n      !allIsValid && this.props.onInvalid();\n\n      // Tell the form that it can start to trigger change events\n      this.setState({canChange: true});\n\n    }.bind(this);\n\n    // Run validation again in case affected by other inputs. The\n    // last component validated will run the onValidationComplete callback\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      var isValid = this.runValidation(component);\n      component.setState({\n        _isValid: isValid,\n        _serverError: null\n      }, index === inputKeys.length - 1 ? onValidationComplete : null);\n    }.bind(this));\n\n    // If there are no inputs, it is ready to trigger change events\n    if (!inputKeys.length) {\n      this.setState({canChange: true});\n    }\n\n  },\n\n  // Method put on each input component to register\n  // itself to the form\n  attachToForm: function (component) {\n    this.inputs[component.props.name] = component;\n    this.model[component.props.name] = component.state._value;\n    this.validate(component);\n  },\n\n  // Method put on each input component to unregister\n  // itself from the form\n  detachFromForm: function (component) {\n    delete this.inputs[component.props.name];\n    delete this.model[component.props.name];\n  },\n  render: function () {\n\n    return React.DOM.form({\n        onSubmit: this.submit,\n        className: this.props.className\n      },\n      this.props.children\n    );\n\n  }\n});\n\nif (!global.exports && !global.module && (!global.define || !global.define.amd)) {\n  global.Formsy = Formsy;\n}\n\nmodule.exports = Formsy;\n"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","src/main.js"],"names":[],"mappings":"AAAA;;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})","var React = global.React || require('react');\nvar Formsy = {};\nvar validationRules = {\n  'isValue': function (value) {\n    return value !== '';\n  },\n  'isEmail': function (value) {\n    return value.match(/^((([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+(\\.([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+)*)|((\\x22)((((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(([\\x01-\\x08\\x0b\\x0c\\x0e-\\x1f\\x7f]|\\x21|[\\x23-\\x5b]|[\\x5d-\\x7e]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(\\\\([\\x01-\\x09\\x0b\\x0c\\x0d-\\x7f]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF]))))*(((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(\\x22)))@((([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))\\.)+(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))$/i);\n  },\n  'isTrue': function (value) {\n    return value === true;\n  },\n  'isNumeric': function (value) {\n    if (typeof value === 'number') {\n      return true;\n    } else {\n      return value.match(/^-?[0-9]+$/);\n    }\n  },\n  'isAlpha': function (value) {\n    return value.match(/^[a-zA-Z]+$/);\n  },\n  'isWords': function (value) {\n    return value.match(/^[a-zA-Z\\s]+$/);\n  },\n  'isSpecialWords': function (value) {\n    return value.match(/^[a-zA-Z\\s\\u00C0-\\u017F]+$/);\n  },\n  isLength: function (value, min, max) {\n    if (max !== undefined) {\n      return value.length >= min && value.length <= max;\n    }\n    return value.length >= min;\n  },\n  equals: function (value, eql) {\n    return value == eql;\n  },\n  equalsField: function (value, field) {\n    return value === this[field];\n  }\n};\n\nvar toURLEncoded = function (element, key, list) {\n  var list = list || [];\n  if (typeof (element) == 'object') {\n    for (var idx in element)\n      toURLEncoded(element[idx], key ? key + '[' + idx + ']' : idx, list);\n  } else {\n    list.push(key + '=' + encodeURIComponent(element));\n  }\n  return list.join('&');\n};\n\nvar request = function (method, url, data, contentType, headers) {\n\n  var contentType = contentType === 'urlencoded' ? 'application/' + contentType.replace('urlencoded', 'x-www-form-urlencoded') : 'application/json';\n  data = contentType === 'application/json' ? JSON.stringify(data) : toURLEncoded(data);\n\n  return new Promise(function (resolve, reject) {\n    try {\n      var xhr = new XMLHttpRequest();\n      xhr.open(method, url, true);\n      xhr.setRequestHeader('Accept', 'application/json');\n      xhr.setRequestHeader('Content-Type', contentType);\n\n      // Add passed headers\n      Object.keys(headers).forEach(function (header) {\n        xhr.setRequestHeader(header, headers[header]);\n      });\n\n      xhr.onreadystatechange = function () {\n        if (xhr.readyState === 4) {\n\n          try {\n            var response = xhr.responseText ? JSON.parse(xhr.responseText) : null;\n            if (xhr.status >= 200 && xhr.status < 300) {\n              resolve(response);\n            } else {\n              reject(response);\n            }\n          } catch (e) {\n            reject(e);\n          }\n\n        }\n      };\n      xhr.send(data);\n    } catch (e) {\n      reject(e);\n    }\n  });\n\n};\n\nvar arraysDiffer = function (arrayA, arrayB) {\n  var isDifferent = false;\n  if (arrayA.length !== arrayB.length) {\n    isDifferent = true;\n  } else {\n    arrayA.forEach(function (item, index) {\n      if (item !== arrayB[index]) {\n        isDifferent = true;\n      }\n    });\n  }\n  return isDifferent;\n};\n\nvar ajax = {\n  post: request.bind(null, 'POST'),\n  put: request.bind(null, 'PUT')\n};\nvar options = {};\n\nFormsy.defaults = function (passedOptions) {\n  options = passedOptions;\n};\n\nFormsy.Mixin = {\n  getInitialState: function () {\n    return {\n      _value: this.props.value ? this.props.value : '',\n      _isValid: true,\n      _isPristine: true\n    };\n  },\n  componentWillMount: function () {\n\n    var configure = function () {\n      // Add validations to the store itself as the props object can not be modified\n      this._validations = this.props.validations || '';\n\n      if (this.props.required) {\n        this._validations = this.props.validations ? this.props.validations + ',' : '';\n        this._validations += 'isValue';\n      }\n      this.props._attachToForm(this);\n    }.bind(this);\n\n    if (!this.props.name) {\n      throw new Error('Form Input requires a name property when used');\n    }\n\n    if (!this.props._attachToForm) {\n      return setTimeout(function () {\n        if (!this.props._attachToForm) {\n          throw new Error('Form Mixin requires component to be nested in a Form');\n        }\n        configure();\n      }.bind(this), 0);\n    }\n    configure();\n\n  },\n\n  // We have to make the validate method is kept when new props are added\n  componentWillReceiveProps: function (nextProps) {\n    nextProps._attachToForm = this.props._attachToForm;\n    nextProps._detachFromForm = this.props._detachFromForm;\n    nextProps._validate = this.props._validate;\n  },\n\n  componentDidUpdate: function(prevProps, prevState) {\n\n    // If the input is untouched and something outside changes the value\n    // update the FORM model by re-attaching to the form\n    if (this.state._isPristine) {\n      if (this.props.value !== prevProps.value && this.state._value === prevProps.value) {\n        this.state._value = this.props.value || '';\n        this.props._attachToForm(this);\n      }\n    }\n  },\n\n  // Detach it when component unmounts\n  componentWillUnmount: function () {\n    this.props._detachFromForm(this);\n  },\n\n  // We validate after the value has been set\n  setValue: function (value) {\n    this.setState({\n      _value: value,\n      _isPristine: false\n    }, function () {\n      this.props._validate(this);\n    }.bind(this));\n  },\n  resetValue: function () {\n    this.setState({\n      _value: '',\n      _isPristine: true\n    }, function () {\n      this.props._validate(this);\n    });\n  },\n  getValue: function () {\n    return this.state._value;\n  },\n  hasValue: function () {\n    return this.state._value !== '';\n  },\n  getErrorMessage: function () {\n    return this.isValid() || this.showRequired() ? null : this.state._serverError || this.props.validationError;\n  },\n  isValid: function () {\n    return this.state._isValid;\n  },\n  isPristine: function () {\n    return this.state._isPristine;\n  },\n  isRequired: function () {\n    return !!this.props.required;\n  },\n  showRequired: function () {\n    return this.isRequired() && this.state._value === '';\n  },\n  showError: function () {\n    return !this.showRequired() && !this.state._isValid;\n  }\n};\n\nFormsy.addValidationRule = function (name, func) {\n  validationRules[name] = func;\n};\n\nFormsy.Form = React.createClass({displayName: \"Form\",\n  getInitialState: function () {\n    return {\n      isValid: true,\n      isSubmitting: false,\n      canChange: false\n    };\n  },\n  getDefaultProps: function () {\n    return {\n      headers: {},\n      onSuccess: function () {},\n      onError: function () {},\n      onSubmit: function () {},\n      onSubmitted: function () {},\n      onValid: function () {},\n      onInvalid: function () {},\n      onChange: function () {}\n    };\n  },\n\n  // Add a map to store the inputs of the form, a model to store\n  // the values of the form and register child inputs\n  componentWillMount: function () {\n    this.inputs = {};\n    this.model = {};\n    this.registerInputs(this.props.children);\n  },\n\n  componentDidMount: function () {\n    this.validateForm();\n  },\n\n  componentWillUpdate: function () {\n    var inputKeys = Object.keys(this.inputs);\n\n    // The updated children array is not available here for some reason,\n    // we need to wait for next event loop\n    setTimeout(function () {\n      this.registerInputs(this.props.children);\n\n      var newInputKeys = Object.keys(this.inputs);\n      if (arraysDiffer(inputKeys, newInputKeys)) {\n        this.validateForm();\n      }\n    }.bind(this), 0);\n  },\n\n  // Update model, submit to url prop and send the model\n  submit: function (event) {\n    event.preventDefault();\n\n    // Trigger form as not pristine.\n    // If any inputs have not been touched yet this will make them dirty\n    // so validation becomes visible (if based on isPristine)\n    this.setFormPristine(false);\n\n    // To support use cases where no async or request operation is needed.\n    // The \"onSubmit\" callback is called with the model e.g. {fieldName: \"myValue\"},\n    // if wanting to reset the entire form to original state, the second param is a callback for this.\n    if (!this.props.url) {\n      this.updateModel();\n      this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n      return;\n    }\n\n    this.updateModel();\n    this.setState({\n      isSubmitting: true\n    });\n\n    this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n\n    var headers = (Object.keys(this.props.headers).length && this.props.headers) || options.headers || {};\n\n    var method = this.props.method && ajax[this.props.method.toLowerCase()] ? this.props.method.toLowerCase() : 'post';\n    ajax[method](this.props.url, this.mapModel(), this.props.contentType || options.contentType || 'json', headers)\n      .then(function (response) {\n        this.props.onSuccess(response);\n        this.props.onSubmitted();\n      }.bind(this))\n      .catch(this.failSubmit);\n  },\n\n  mapModel: function () {\n    return this.props.mapping ? this.props.mapping(this.model) : this.model;\n  },\n\n  // Goes through all registered components and\n  // updates the model values\n  updateModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      var component = this.inputs[name];\n      this.model[name] = component.state._value;\n    }.bind(this));\n  },\n\n  // Reset each key in the model to the original / initial value\n  resetModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      this.inputs[name].resetValue();\n    }.bind(this));\n    this.validateForm();\n  },\n\n  // Go through errors from server and grab the components\n  // stored in the inputs map. Change their state to invalid\n  // and set the serverError message\n  updateInputsWithError: function (errors) {\n    Object.keys(errors).forEach(function (name, index) {\n      var component = this.inputs[name];\n\n      if (!component) {\n        throw new Error('You are trying to update an input that does not exists. Verify errors object with input names. ' + JSON.stringify(errors));\n      }\n\n      var args = [{\n        _isValid: false,\n        _serverError: errors[name]\n      }];\n      component.setState.apply(component, args);\n    }.bind(this));\n  },\n\n  failSubmit: function (errors) {\n    this.updateInputsWithError(errors);\n    this.setState({\n      isSubmitting: false\n    });\n    this.props.onError(errors);\n    this.props.onSubmitted();\n  },\n\n  // Traverse the children and children of children to find\n  // all inputs by checking the name prop. Maybe do a better\n  // check here\n  registerInputs: function (children) {\n    React.Children.forEach(children, function (child) {\n\n      if (child && child.props && child.props.name) {\n        child.props._attachToForm = this.attachToForm;\n        child.props._detachFromForm = this.detachFromForm;\n        child.props._validate = this.validate;\n      }\n\n      if (child && child.props && child.props.children) {\n        this.registerInputs(child.props.children);\n      }\n\n    }.bind(this));\n  },\n\n  getCurrentValues: function () {\n    return Object.keys(this.inputs).reduce(function (data, name) {\n      var component = this.inputs[name];\n      data[name] = component.state._value;\n      return data;\n    }.bind(this), {});\n  },\n\n  setFormPristine: function (isPristine) {\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // Iterate through each component and set it as pristine\n    // or \"dirty\".\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      component.setState({\n        _isPristine: isPristine\n      });\n    }.bind(this));\n  },\n\n  // Use the binded values and the actual input value to\n  // validate the input and set its state. Then check the\n  // state of the form itself\n  validate: function (component) {\n\n    // Trigger onChange\n    this.state.canChange && this.props.onChange && this.props.onChange(this.getCurrentValues());\n\n    if (!component.props.required && !component._validations) {\n      return;\n    }\n\n    // Run through the validations, split them up and call\n    // the validator IF there is a value or it is required\n    var isValid = this.runValidation(component);\n\n    component.setState({\n      _isValid: isValid,\n      _serverError: null\n    }, this.validateForm);\n\n  },\n\n  runValidation: function (component) {\n    var isValid = true;\n    if (component._validations.length && (component.props.required || component.state._value !== '')) {\n      component._validations.split(',').forEach(function (validation) {\n        var args = validation.split(':');\n        var validateMethod = args.shift();\n        args = args.map(function (arg) {\n          try {\n            return JSON.parse(arg);\n          } catch (e) {\n            return arg; // It is a string if it can not parse it\n          }\n        });\n        args = [component.state._value].concat(args);\n        if (!validationRules[validateMethod]) {\n          throw new Error('Formsy does not have the validation rule: ' + validateMethod);\n        }\n        if (!validationRules[validateMethod].apply(this.getCurrentValues(), args)) {\n          isValid = false;\n        }\n      }.bind(this));\n    }\n    return isValid;\n  },\n\n  // Validate the form by going through all child input components\n  // and check their state\n  validateForm: function () {\n    var allIsValid = true;\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // We need a callback as we are validating all inputs again. This will\n    // run when the last component has set its state\n    var onValidationComplete = function () {\n      inputKeys.forEach(function (name) {\n        if (!inputs[name].state._isValid) {\n          allIsValid = false;\n        }\n      }.bind(this));\n\n      this.setState({\n        isValid: allIsValid\n      });\n\n      allIsValid && this.props.onValid();\n      !allIsValid && this.props.onInvalid();\n\n      // Tell the form that it can start to trigger change events\n      this.setState({canChange: true});\n\n    }.bind(this);\n\n    // Run validation again in case affected by other inputs. The\n    // last component validated will run the onValidationComplete callback\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      var isValid = this.runValidation(component);\n      component.setState({\n        _isValid: isValid,\n        _serverError: null\n      }, index === inputKeys.length - 1 ? onValidationComplete : null);\n    }.bind(this));\n\n    // If there are no inputs, it is ready to trigger change events\n    if (!inputKeys.length) {\n      this.setState({canChange: true});\n    }\n\n  },\n\n  // Method put on each input component to register\n  // itself to the form\n  attachToForm: function (component) {\n    this.inputs[component.props.name] = component;\n    this.model[component.props.name] = component.state._value;\n    this.validate(component);\n  },\n\n  // Method put on each input component to unregister\n  // itself from the form\n  detachFromForm: function (component) {\n    delete this.inputs[component.props.name];\n    delete this.model[component.props.name];\n  },\n  render: function () {\n\n    return React.DOM.form({\n        onSubmit: this.submit,\n        className: this.props.className\n      },\n      this.props.children\n    );\n\n  }\n});\n\nif (!global.exports && !global.module && (!global.define || !global.define.amd)) {\n  global.Formsy = Formsy;\n}\n\nmodule.exports = Formsy;\n"]} diff --git a/build/specs.js b/build/specs.js index d169a23..6d49588 100644 --- a/build/specs.js +++ b/build/specs.js @@ -294,6 +294,42 @@ describe('Formsy', function () { expect(form.getDOMNode().className).toEqual('foo'); }); + it('should allow for null/undefined children', function (done) { + var TestInput = React.createClass({displayName: "TestInput", + mixins: [Formsy.Mixin], + changeValue: function (event) { + this.setValue(event.target.value); + }, + render: function () { + return React.createElement("input", {value: this.getValue(), onChange: this.changeValue}) + } + }); + + var model = null; + var TestForm = React.createClass({displayName: "TestForm", + onSubmit: function (formModel) { + model = formModel; + }, + render: function () { + return ( + React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, + React.createElement("h1", null, "Test"), + null, + undefined, + React.createElement(TestInput, {name: "name", value: 'foo' }) + ) + ); + } + }); + + var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null)); + setTimeout(function () { + TestUtils.Simulate.submit(form.getDOMNode()); + expect(model).toEqual({name: 'foo'}); + done(); + }, 10); + }); + it('should allow for inputs being added dynamically', function (done) { var inputs = []; @@ -400,6 +436,58 @@ describe('Formsy', function () { }); + it('should allow a dynamically updated input to update the form-model', function (done) { + + var forceUpdate = null; + var model = null; + var TestInput = React.createClass({displayName: "TestInput", + mixins: [Formsy.Mixin], + changeValue: function (event) { + this.setValue(event.target.value); + }, + render: function () { + return React.createElement("input", {value: this.getValue(), onChange: this.changeValue}) + } + }); + + var input; + var TestForm = React.createClass({displayName: "TestForm", + componentWillMount: function () { + forceUpdate = this.forceUpdate.bind(this); + }, + onSubmit: function (formModel) { + model = formModel; + }, + render: function () { + input = React.createElement(TestInput, {name: "test", value: this.props.value}); + + return ( + React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, + input + )); + } + }); + var form = TestUtils.renderIntoDocument(React.createElement(TestForm, {value: "foo"})); + + // Wait before changing the input + setTimeout(function () { + form.setProps({value: 'bar'}); + + forceUpdate(function () { + // Wait for next event loop, as that does the form + setTimeout(function () { + TestUtils.Simulate.submit(form.getDOMNode()); + expect(model.test).toBe('bar'); + done(); + }, 0); + + }); + + }, 10); + + }); + + it('should invalidate a valid form if dynamically inserted input is invalid', function (done) { var forceUpdate = null; @@ -441,29 +529,23 @@ describe('Formsy', function () { expect(isInvalid).toBe(false); - // Wait before adding the input - setTimeout(function () { - - - inputs.push(TestInput({ - name: 'test2', - validations: 'isEmail', - value: 'foo@bar' - })); + inputs.push(TestInput({ + name: 'test2', + validations: 'isEmail', + value: 'foo@bar' + })); - forceUpdate(function () { + forceUpdate(function () { - // Wait for next event loop, as that does the form - setTimeout(function () { - TestUtils.Simulate.submit(form.getDOMNode()); - expect(isInvalid).toBe(true); - done(); - }, 0); + // Wait for next event loop, as that does the form + setTimeout(function () { + TestUtils.Simulate.submit(form.getDOMNode()); + expect(isInvalid).toBe(true); + done(); + }, 0); - }); - - }, 10); + }); }); @@ -541,23 +623,19 @@ describe('Formsy', function () { ); // Wait before adding the input - setTimeout(function () { + inputs.push(TestInput({ + name: 'test' + })); - inputs.push(TestInput({ - name: 'test' - })); + forceUpdate(function () { - forceUpdate(function () { + // Wait for next event loop, as that does the form + setTimeout(function () { + expect(hasChanged).toHaveBeenCalled(); + done(); + }, 0); - // Wait for next event loop, as that does the form - setTimeout(function () { - expect(hasChanged).toHaveBeenCalled(); - done(); - }, 0); - - }); - - }, 10); + }); }); @@ -1048,6 +1126,18 @@ Formsy.Mixin = { nextProps._validate = this.props._validate; }, + componentDidUpdate: function(prevProps, prevState) { + + // If the input is untouched and something outside changes the value + // update the FORM model by re-attaching to the form + if (this.state._isPristine) { + if (this.props.value !== prevProps.value && this.state._value === prevProps.value) { + this.state._value = this.props.value || ''; + this.props._attachToForm(this); + } + } + }, + // Detach it when component unmounts componentWillUnmount: function () { this.props._detachFromForm(this); @@ -1239,13 +1329,13 @@ Formsy.Form = React.createClass({displayName: "Form", registerInputs: function (children) { React.Children.forEach(children, function (child) { - if (child.props && child.props.name) { + if (child && child.props && child.props.name) { child.props._attachToForm = this.attachToForm; child.props._detachFromForm = this.detachFromForm; child.props._validate = this.validate; } - if (child.props && child.props.children) { + if (child && child.props && child.props.children) { this.registerInputs(child.props.children); } @@ -1403,4 +1493,4 @@ module.exports = Formsy; }).call(this,typeof global !== "undefined" ? global : typeof self !== "undefined" ? self : typeof window !== "undefined" ? window : {}) },{"react":"react"}]},{},["./specs/Element-spec.js","./specs/Formsy-spec.js","./specs/Submit-spec.js","./specs/Validation-spec.js"]) -//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","specs/Element-spec.js","specs/Formsy-spec.js","specs/Submit-spec.js","specs/Validation-spec.js","src/main.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC9RA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})","var Formsy = require('./../src/main.js');\n\ndescribe('Element', function() {\n\n  it('should return passed and setValue() value when using getValue()', function () {\n    \n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(input.getDOMNode().value).toBe('foo');\n    TestUtils.Simulate.change(input, {target: {value: 'foobar'}});\n    expect(input.getDOMNode().value).toBe('foobar');\n\n  });\n\n  it('should return true or false when calling hasValue() depending on value existance', function () {\n    \n    var reset = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        reset = this.resetValue;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    reset();\n    expect(input.getDOMNode().value).toBe('');\n\n  });\n\n  it('should return error message passed when calling getErrorMessage()', function () {\n    \n    var getErrorMessage = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        getErrorMessage = this.getErrorMessage;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"Has to be email\"})\n      )\n    );\n\n    expect(getErrorMessage()).toBe('Has to be email');\n\n  });\n\n  it('should return server error message when calling getErrorMessage()', function (done) {\n    \n    jasmine.Ajax.install();\n\n    var getErrorMessage = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        getErrorMessage = this.getErrorMessage;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"Has to be email\"})\n      )\n    );\n\n    var form = TestUtils.Simulate.submit(form.getDOMNode());\n\n    jasmine.Ajax.requests.mostRecent().respondWith({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{\"foo\": \"bar\"}'\n    })\n\n    setTimeout(function () {\n      expect(getErrorMessage()).toBe('bar');\n      jasmine.Ajax.uninstall();\n      done();\n    }, 0);\n\n  });\n\n  it('should return true or false when calling isValid() depending on valid state', function () {\n    \n    var isValid = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isValid = this.isValid;\n      },\n      updateValue: function (event) {\n        console.log('event.target.value', event.target.value);\n        this.setValue(event.target.value);\n        setTimeout(function () {\n          console.log('this.getValue()', this.getValue());\n        }.bind(this), 100);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\"})\n      )\n    );\n\n    expect(isValid()).toBe(false);\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(isValid()).toBe(true);\n\n  });\n\n  it('should return true or false when calling isRequired() depending on passed required attribute', function () {\n    \n    var isRequireds = [];\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isRequireds.push(this.isRequired);\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"}), \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", required: true})\n      )\n    );\n\n    expect(isRequireds[0]()).toBe(false);\n    expect(isRequireds[1]()).toBe(true);\n\n  });\n\n  it('should return true or false when calling showRequired() depending on input being empty and required is passed, or not', function () {\n    \n    var showRequireds = [];\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        showRequireds.push(this.showRequired);\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"A\", value: \"foo\"}), \n        React.createElement(TestInput, {name: \"B\", value: \"\", required: true}), \n        React.createElement(TestInput, {name: \"C\", value: \"\"})\n      )\n    );\n\n    expect(showRequireds[0]()).toBe(false);\n    expect(showRequireds[1]()).toBe(true);\n    expect(showRequireds[2]()).toBe(false);\n\n  });\n\n  it('should return true or false when calling showError() depending on value is invalid or a server error has arrived, or not', function (done) {\n\n    var showError = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        showError = this.showError;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"This is not an email\"})\n      )\n    );\n\n    expect(showError()).toBe(true);\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(showError()).toBe(false);\n\n    jasmine.Ajax.install();\n    TestUtils.Simulate.submit(form.getDOMNode());    \n    jasmine.Ajax.requests.mostRecent().respondWith({\n      status: 500,\n      responseType: 'application/json',\n      responseText: '{\"foo\": \"Email already exists\"}'\n    });\n    setTimeout(function () {\n      expect(showError()).toBe(true);\n      jasmine.Ajax.uninstall();\n      done();\n    }, 0);\n  });\n\n  it('should return true or false when calling isPrestine() depending on input has been \"touched\" or not', function () {\n    \n    var isPristine = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isPristine = this.isPristine;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"A\", value: \"foo\"})\n      )\n    );\n\n    expect(isPristine()).toBe(true);\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo'}});\n    expect(isPristine()).toBe(false);\n    \n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Formsy', function () {\n\n  describe('Setting up a form', function () {\n\n    it('should render a form into the document', function () {\n      var form = TestUtils.renderIntoDocument( React.createElement(Formsy.Form, null));\n      expect(form.getDOMNode().tagName).toEqual('FORM');\n    });\n\n    it('should set a class name if passed', function () {\n      var form = TestUtils.renderIntoDocument( React.createElement(Formsy.Form, {className: \"foo\"}));\n      expect(form.getDOMNode().className).toEqual('foo');\n    });\n\n    it('should allow for inputs being added dynamically', function (done) {\n\n      var inputs = [];\n      var forceUpdate = null;\n      var model = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        render: function () {\n          return React.createElement(\"div\", null)\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        inputs.push(TestInput({\n          name: 'test'\n        }));\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(model.test).toBeDefined();\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n    it('should allow dynamically added inputs to update the form-model', function (done) {\n\n      var inputs = [];\n      var forceUpdate = null;\n      var model = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        inputs.push(TestInput({\n          name: 'test'\n        }));\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.change(TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT'), {target: {value: 'foo'}});\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(model.test).toBe('foo');\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n    it('should invalidate a valid form if dynamically inserted input is invalid', function (done) {\n\n      var forceUpdate = null;\n      var isInvalid = false;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n\n\n      var inputs = [TestInput({\n        name: 'test',\n        validations: 'isEmail',\n        value: 'foo@bar.com'\n      })];\n\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        setInvalid: function () {\n          isInvalid = true;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onInvalid: this.setInvalid}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      expect(isInvalid).toBe(false);\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        \n        inputs.push(TestInput({\n          name: 'test2',\n          validations: 'isEmail',\n          value: 'foo@bar'\n        }));\n\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(isInvalid).toBe(true);\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n    it('should not trigger onChange when form is mounted', function () {\n      var hasChanged = jasmine.createSpy('onChange');\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return React.createElement(Formsy.Form, {onChange: this.onChange});\n        }\n      });\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null));\n      expect(hasChanged).not.toHaveBeenCalled();\n    });\n\n    it('should trigger onChange when form element is changed', function () {\n      var hasChanged = jasmine.createSpy('onChange');\n      var MyInput = React.createClass({displayName: \"MyInput\",\n        mixins: [Formsy.Mixin],\n        onChange: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.onChange})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return (\n            React.createElement(Formsy.Form, {onChange: this.onChange}, \n              React.createElement(MyInput, {name: \"foo\"})\n            )\n          );\n        }\n      });\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null));\n      TestUtils.Simulate.change(TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT'), {target: {value: 'bar'}});\n      expect(hasChanged).toHaveBeenCalled();\n    });\n\n    it('should trigger onChange when new input is added to form', function (done) {\n      var hasChanged = jasmine.createSpy('onChange');\n      var inputs = [];\n      var forceUpdate = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onChange: this.onChange}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        inputs.push(TestInput({\n          name: 'test'\n        }));\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            expect(hasChanged).toHaveBeenCalled();\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Ajax', function() {\n\n  beforeEach(function () {\n    jasmine.Ajax.install();\n  });\n\n  afterEach(function () {\n    jasmine.Ajax.uninstall();\n  });\n\n  it('should post to a given url if passed', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().url).toBe('/users');\n    expect(jasmine.Ajax.requests.mostRecent().method).toBe('POST');\n\n  });\n\n  it('should put to a given url if passed a method attribute', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", method: \"PUT\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().url).toBe('/users');\n    expect(jasmine.Ajax.requests.mostRecent().method).toBe('PUT');\n\n  });\n\n  it('should pass x-www-form-urlencoded as contentType when urlencoded is set as contentType', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", contentType: \"urlencoded\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().contentType()).toBe('application/x-www-form-urlencoded');\n\n  });\n\n  it('should run an onSuccess handler, if passed and ajax is successfull. First argument is data from server', function (done) {\n \n    var onSuccess = jasmine.createSpy(\"success\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSuccess: onSuccess}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 200,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSuccess).toHaveBeenCalledWith({});\n      done();\n    }, 0);\n\n  });\n\n  it('should not do ajax request if onSubmit handler is passed, but pass the model as first argument to onSubmit handler', function () {\n    \n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue()})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onSubmit: onSubmit}, \n        React.createElement(TestInput, {name: \"foo\", value: \"bar\"})\n      )\n    );\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    expect(jasmine.Ajax.requests.count()).toBe(0);\n\n    function onSubmit (data) {\n      expect(data).toEqual({\n        foo: 'bar'\n      });\n    }\n\n  });\n\n  it('should trigger an onSubmitted handler, if passed and the submit has responded with SUCCESS', function (done) {\n    \n    var onSubmitted = jasmine.createSpy(\"submitted\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSubmitted: onSubmitted}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 200,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSubmitted).toHaveBeenCalled();\n      done();\n    }, 0);\n\n  });\n\n  it('should trigger an onSubmitted handler, if passed and the submit has responded with ERROR', function (done) {\n    \n    var onSubmitted = jasmine.createSpy(\"submitted\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSubmitted: onSubmitted}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSubmitted).toHaveBeenCalled();\n      done();\n    }, 0);\n\n  });\n\n  it('should trigger an onError handler, if passed and the submit has responded with ERROR', function (done) {\n    \n    var onError = jasmine.createSpy(\"error\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onError: onError}\n      )\n    );\n    \n    // Do not return any error because there are no inputs\n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onError).toHaveBeenCalledWith({});\n      done();\n    }, 0);\n\n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Validation', function() {\n\n  it('should trigger an onValid handler, if passed, when form is valid', function () {\n    \n    var onValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onValid: onValid}, \n        React.createElement(TestInput, {name: \"foo\", required: true})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo'}});\n    expect(onValid).toHaveBeenCalled();\n\n  });\n\n  it('should trigger an onInvalid handler, if passed, when form is invalid', function () {\n    \n    var onInvalid = jasmine.createSpy('invalid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onValid: onInvalid}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: ''}});\n    expect(onInvalid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isEmail', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isNumeric', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isNumeric\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: '123'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isNumeric (actual number)', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(Number(event.target.value));\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isNumeric\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: '123'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n});\n","var React = global.React || require('react');\nvar Formsy = {};\nvar validationRules = {\n  'isValue': function (value) {\n    return value !== '';\n  },\n  'isEmail': function (value) {\n    return value.match(/^((([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+(\\.([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+)*)|((\\x22)((((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(([\\x01-\\x08\\x0b\\x0c\\x0e-\\x1f\\x7f]|\\x21|[\\x23-\\x5b]|[\\x5d-\\x7e]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(\\\\([\\x01-\\x09\\x0b\\x0c\\x0d-\\x7f]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF]))))*(((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(\\x22)))@((([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))\\.)+(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))$/i);\n  },\n  'isTrue': function (value) {\n    return value === true;\n  },\n  'isNumeric': function (value) {\n    if (typeof value === 'number') {\n      return true;\n    } else {\n      return value.match(/^-?[0-9]+$/);\n    }\n  },\n  'isAlpha': function (value) {\n    return value.match(/^[a-zA-Z]+$/);\n  },\n  'isWords': function (value) {\n    return value.match(/^[a-zA-Z\\s]+$/);\n  },\n  'isSpecialWords': function (value) {\n    return value.match(/^[a-zA-Z\\s\\u00C0-\\u017F]+$/);\n  },\n  isLength: function (value, min, max) {\n    if (max !== undefined) {\n      return value.length >= min && value.length <= max;\n    }\n    return value.length >= min;\n  },\n  equals: function (value, eql) {\n    return value == eql;\n  },\n  equalsField: function (value, field) {\n    return value === this[field];\n  }\n};\n\nvar toURLEncoded = function (element, key, list) {\n  var list = list || [];\n  if (typeof (element) == 'object') {\n    for (var idx in element)\n      toURLEncoded(element[idx], key ? key + '[' + idx + ']' : idx, list);\n  } else {\n    list.push(key + '=' + encodeURIComponent(element));\n  }\n  return list.join('&');\n};\n\nvar request = function (method, url, data, contentType, headers) {\n\n  var contentType = contentType === 'urlencoded' ? 'application/' + contentType.replace('urlencoded', 'x-www-form-urlencoded') : 'application/json';\n  data = contentType === 'application/json' ? JSON.stringify(data) : toURLEncoded(data);\n\n  return new Promise(function (resolve, reject) {\n    try {\n      var xhr = new XMLHttpRequest();\n      xhr.open(method, url, true);\n      xhr.setRequestHeader('Accept', 'application/json');\n      xhr.setRequestHeader('Content-Type', contentType);\n\n      // Add passed headers\n      Object.keys(headers).forEach(function (header) {\n        xhr.setRequestHeader(header, headers[header]);\n      });\n\n      xhr.onreadystatechange = function () {\n        if (xhr.readyState === 4) {\n\n          try {\n            var response = xhr.responseText ? JSON.parse(xhr.responseText) : null;\n            if (xhr.status >= 200 && xhr.status < 300) {\n              resolve(response);\n            } else {\n              reject(response);\n            }\n          } catch (e) {\n            reject(e);\n          }\n\n        }\n      };\n      xhr.send(data);\n    } catch (e) {\n      reject(e);\n    }\n  });\n\n};\n\nvar arraysDiffer = function (arrayA, arrayB) {\n  var isDifferent = false;\n  if (arrayA.length !== arrayB.length) {\n    isDifferent = true;\n  } else {\n    arrayA.forEach(function (item, index) {\n      if (item !== arrayB[index]) {\n        isDifferent = true;\n      }\n    });\n  }\n  return isDifferent;\n};\n\nvar ajax = {\n  post: request.bind(null, 'POST'),\n  put: request.bind(null, 'PUT')\n};\nvar options = {};\n\nFormsy.defaults = function (passedOptions) {\n  options = passedOptions;\n};\n\nFormsy.Mixin = {\n  getInitialState: function () {\n    return {\n      _value: this.props.value ? this.props.value : '',\n      _isValid: true,\n      _isPristine: true\n    };\n  },\n  componentWillMount: function () {\n\n    var configure = function () {\n      // Add validations to the store itself as the props object can not be modified\n      this._validations = this.props.validations || '';\n\n      if (this.props.required) {\n        this._validations = this.props.validations ? this.props.validations + ',' : '';\n        this._validations += 'isValue';\n      }\n      this.props._attachToForm(this);\n    }.bind(this);\n\n    if (!this.props.name) {\n      throw new Error('Form Input requires a name property when used');\n    }\n\n    if (!this.props._attachToForm) {\n      return setTimeout(function () {\n        if (!this.props._attachToForm) {\n          throw new Error('Form Mixin requires component to be nested in a Form');\n        }\n        configure();\n      }.bind(this), 0);\n    }\n    configure();\n\n  },\n\n  // We have to make the validate method is kept when new props are added\n  componentWillReceiveProps: function (nextProps) {\n    nextProps._attachToForm = this.props._attachToForm;\n    nextProps._detachFromForm = this.props._detachFromForm;\n    nextProps._validate = this.props._validate;\n  },\n\n  // Detach it when component unmounts\n  componentWillUnmount: function () {\n    this.props._detachFromForm(this);\n  },\n\n  // We validate after the value has been set\n  setValue: function (value) {\n    this.setState({\n      _value: value,\n      _isPristine: false\n    }, function () {\n      this.props._validate(this);\n    }.bind(this));\n  },\n  resetValue: function () {\n    this.setState({\n      _value: '',\n      _isPristine: true\n    }, function () {\n      this.props._validate(this);\n    });\n  },\n  getValue: function () {\n    return this.state._value;\n  },\n  hasValue: function () {\n    return this.state._value !== '';\n  },\n  getErrorMessage: function () {\n    return this.isValid() || this.showRequired() ? null : this.state._serverError || this.props.validationError;\n  },\n  isValid: function () {\n    return this.state._isValid;\n  },\n  isPristine: function () {\n    return this.state._isPristine;\n  },\n  isRequired: function () {\n    return !!this.props.required;\n  },\n  showRequired: function () {\n    return this.isRequired() && this.state._value === '';\n  },\n  showError: function () {\n    return !this.showRequired() && !this.state._isValid;\n  }\n};\n\nFormsy.addValidationRule = function (name, func) {\n  validationRules[name] = func;\n};\n\nFormsy.Form = React.createClass({displayName: \"Form\",\n  getInitialState: function () {\n    return {\n      isValid: true,\n      isSubmitting: false,\n      canChange: false\n    };\n  },\n  getDefaultProps: function () {\n    return {\n      headers: {},\n      onSuccess: function () {},\n      onError: function () {},\n      onSubmit: function () {},\n      onSubmitted: function () {},\n      onValid: function () {},\n      onInvalid: function () {},\n      onChange: function () {}\n    };\n  },\n\n  // Add a map to store the inputs of the form, a model to store\n  // the values of the form and register child inputs\n  componentWillMount: function () {\n    this.inputs = {};\n    this.model = {};\n    this.registerInputs(this.props.children);\n  },\n\n  componentDidMount: function () {\n    this.validateForm();\n  },\n\n  componentWillUpdate: function () {\n    var inputKeys = Object.keys(this.inputs);\n\n    // The updated children array is not available here for some reason,\n    // we need to wait for next event loop\n    setTimeout(function () {\n      this.registerInputs(this.props.children);\n\n      var newInputKeys = Object.keys(this.inputs);\n      if (arraysDiffer(inputKeys, newInputKeys)) {\n        this.validateForm();\n      }\n    }.bind(this), 0);\n  },\n\n  // Update model, submit to url prop and send the model\n  submit: function (event) {\n    event.preventDefault();\n\n    // Trigger form as not pristine.\n    // If any inputs have not been touched yet this will make them dirty\n    // so validation becomes visible (if based on isPristine)\n    this.setFormPristine(false);\n\n    // To support use cases where no async or request operation is needed.\n    // The \"onSubmit\" callback is called with the model e.g. {fieldName: \"myValue\"},\n    // if wanting to reset the entire form to original state, the second param is a callback for this.\n    if (!this.props.url) {\n      this.updateModel();\n      this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n      return;\n    }\n\n    this.updateModel();\n    this.setState({\n      isSubmitting: true\n    });\n\n    this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n\n    var headers = (Object.keys(this.props.headers).length && this.props.headers) || options.headers || {};\n\n    var method = this.props.method && ajax[this.props.method.toLowerCase()] ? this.props.method.toLowerCase() : 'post';\n    ajax[method](this.props.url, this.mapModel(), this.props.contentType || options.contentType || 'json', headers)\n      .then(function (response) {\n        this.props.onSuccess(response);\n        this.props.onSubmitted();\n      }.bind(this))\n      .catch(this.failSubmit);\n  },\n\n  mapModel: function () {\n    return this.props.mapping ? this.props.mapping(this.model) : this.model;\n  },\n\n  // Goes through all registered components and\n  // updates the model values\n  updateModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      var component = this.inputs[name];\n      this.model[name] = component.state._value;\n    }.bind(this));\n  },\n\n  // Reset each key in the model to the original / initial value\n  resetModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      this.inputs[name].resetValue();\n    }.bind(this));\n    this.validateForm();\n  },\n\n  // Go through errors from server and grab the components\n  // stored in the inputs map. Change their state to invalid\n  // and set the serverError message\n  updateInputsWithError: function (errors) {\n    Object.keys(errors).forEach(function (name, index) {\n      var component = this.inputs[name];\n\n      if (!component) {\n        throw new Error('You are trying to update an input that does not exists. Verify errors object with input names. ' + JSON.stringify(errors));\n      }\n\n      var args = [{\n        _isValid: false,\n        _serverError: errors[name]\n      }];\n      component.setState.apply(component, args);\n    }.bind(this));\n  },\n\n  failSubmit: function (errors) {\n    this.updateInputsWithError(errors);\n    this.setState({\n      isSubmitting: false\n    });\n    this.props.onError(errors);\n    this.props.onSubmitted();\n  },\n\n  // Traverse the children and children of children to find\n  // all inputs by checking the name prop. Maybe do a better\n  // check here\n  registerInputs: function (children) {\n    React.Children.forEach(children, function (child) {\n\n      if (child.props && child.props.name) {\n        child.props._attachToForm = this.attachToForm;\n        child.props._detachFromForm = this.detachFromForm;\n        child.props._validate = this.validate;\n      }\n\n      if (child.props && child.props.children) {\n        this.registerInputs(child.props.children);\n      }\n\n    }.bind(this));\n  },\n\n  getCurrentValues: function () {\n    return Object.keys(this.inputs).reduce(function (data, name) {\n      var component = this.inputs[name];\n      data[name] = component.state._value;\n      return data;\n    }.bind(this), {});\n  },\n\n  setFormPristine: function (isPristine) {\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // Iterate through each component and set it as pristine\n    // or \"dirty\".\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      component.setState({\n        _isPristine: isPristine\n      });\n    }.bind(this));\n  },\n\n  // Use the binded values and the actual input value to\n  // validate the input and set its state. Then check the\n  // state of the form itself\n  validate: function (component) {\n\n    // Trigger onChange\n    this.state.canChange && this.props.onChange && this.props.onChange(this.getCurrentValues());\n\n    if (!component.props.required && !component._validations) {\n      return;\n    }\n\n    // Run through the validations, split them up and call\n    // the validator IF there is a value or it is required\n    var isValid = this.runValidation(component);\n\n    component.setState({\n      _isValid: isValid,\n      _serverError: null\n    }, this.validateForm);\n\n  },\n\n  runValidation: function (component) {\n    var isValid = true;\n    if (component._validations.length && (component.props.required || component.state._value !== '')) {\n      component._validations.split(',').forEach(function (validation) {\n        var args = validation.split(':');\n        var validateMethod = args.shift();\n        args = args.map(function (arg) {\n          try {\n            return JSON.parse(arg);\n          } catch (e) {\n            return arg; // It is a string if it can not parse it\n          }\n        });\n        args = [component.state._value].concat(args);\n        if (!validationRules[validateMethod]) {\n          throw new Error('Formsy does not have the validation rule: ' + validateMethod);\n        }\n        if (!validationRules[validateMethod].apply(this.getCurrentValues(), args)) {\n          isValid = false;\n        }\n      }.bind(this));\n    }\n    return isValid;\n  },\n\n  // Validate the form by going through all child input components\n  // and check their state\n  validateForm: function () {\n    var allIsValid = true;\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // We need a callback as we are validating all inputs again. This will\n    // run when the last component has set its state\n    var onValidationComplete = function () {\n      inputKeys.forEach(function (name) {\n        if (!inputs[name].state._isValid) {\n          allIsValid = false;\n        }\n      }.bind(this));\n\n      this.setState({\n        isValid: allIsValid\n      });\n\n      allIsValid && this.props.onValid();\n      !allIsValid && this.props.onInvalid();\n\n      // Tell the form that it can start to trigger change events\n      this.setState({canChange: true});\n\n    }.bind(this);\n\n    // Run validation again in case affected by other inputs. The\n    // last component validated will run the onValidationComplete callback\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      var isValid = this.runValidation(component);\n      component.setState({\n        _isValid: isValid,\n        _serverError: null\n      }, index === inputKeys.length - 1 ? onValidationComplete : null);\n    }.bind(this));\n\n    // If there are no inputs, it is ready to trigger change events\n    if (!inputKeys.length) {\n      this.setState({canChange: true});\n    }\n\n  },\n\n  // Method put on each input component to register\n  // itself to the form\n  attachToForm: function (component) {\n    this.inputs[component.props.name] = component;\n    this.model[component.props.name] = component.state._value;\n    this.validate(component);\n  },\n\n  // Method put on each input component to unregister\n  // itself from the form\n  detachFromForm: function (component) {\n    delete this.inputs[component.props.name];\n    delete this.model[component.props.name];\n  },\n  render: function () {\n\n    return React.DOM.form({\n        onSubmit: this.submit,\n        className: this.props.className\n      },\n      this.props.children\n    );\n\n  }\n});\n\nif (!global.exports && !global.module && (!global.define || !global.define.amd)) {\n  global.Formsy = Formsy;\n}\n\nmodule.exports = Formsy;\n"]} +//# sourceMappingURL=data:application/json;base64,{"version":3,"sources":["node_modules/browserify/node_modules/browser-pack/_prelude.js","specs/Element-spec.js","specs/Formsy-spec.js","specs/Submit-spec.js","specs/Validation-spec.js","src/main.js"],"names":[],"mappings":"AAAA;ACAA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;ACrRA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AC5WA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;AClLA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;;;ACzIA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA;AACA","file":"generated.js","sourceRoot":"","sourcesContent":["(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})","var Formsy = require('./../src/main.js');\n\ndescribe('Element', function() {\n\n  it('should return passed and setValue() value when using getValue()', function () {\n    \n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(input.getDOMNode().value).toBe('foo');\n    TestUtils.Simulate.change(input, {target: {value: 'foobar'}});\n    expect(input.getDOMNode().value).toBe('foobar');\n\n  });\n\n  it('should return true or false when calling hasValue() depending on value existance', function () {\n    \n    var reset = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        reset = this.resetValue;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    reset();\n    expect(input.getDOMNode().value).toBe('');\n\n  });\n\n  it('should return error message passed when calling getErrorMessage()', function () {\n    \n    var getErrorMessage = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        getErrorMessage = this.getErrorMessage;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"Has to be email\"})\n      )\n    );\n\n    expect(getErrorMessage()).toBe('Has to be email');\n\n  });\n\n  it('should return server error message when calling getErrorMessage()', function (done) {\n    \n    jasmine.Ajax.install();\n\n    var getErrorMessage = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        getErrorMessage = this.getErrorMessage;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"Has to be email\"})\n      )\n    );\n\n    var form = TestUtils.Simulate.submit(form.getDOMNode());\n\n    jasmine.Ajax.requests.mostRecent().respondWith({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{\"foo\": \"bar\"}'\n    })\n\n    setTimeout(function () {\n      expect(getErrorMessage()).toBe('bar');\n      jasmine.Ajax.uninstall();\n      done();\n    }, 0);\n\n  });\n\n  it('should return true or false when calling isValid() depending on valid state', function () {\n    \n    var isValid = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isValid = this.isValid;\n      },\n      updateValue: function (event) {\n        console.log('event.target.value', event.target.value);\n        this.setValue(event.target.value);\n        setTimeout(function () {\n          console.log('this.getValue()', this.getValue());\n        }.bind(this), 100);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\"})\n      )\n    );\n\n    expect(isValid()).toBe(false);\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(isValid()).toBe(true);\n\n  });\n\n  it('should return true or false when calling isRequired() depending on passed required attribute', function () {\n    \n    var isRequireds = [];\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isRequireds.push(this.isRequired);\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"}), \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", required: true})\n      )\n    );\n\n    expect(isRequireds[0]()).toBe(false);\n    expect(isRequireds[1]()).toBe(true);\n\n  });\n\n  it('should return true or false when calling showRequired() depending on input being empty and required is passed, or not', function () {\n    \n    var showRequireds = [];\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        showRequireds.push(this.showRequired);\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"A\", value: \"foo\"}), \n        React.createElement(TestInput, {name: \"B\", value: \"\", required: true}), \n        React.createElement(TestInput, {name: \"C\", value: \"\"})\n      )\n    );\n\n    expect(showRequireds[0]()).toBe(false);\n    expect(showRequireds[1]()).toBe(true);\n    expect(showRequireds[2]()).toBe(false);\n\n  });\n\n  it('should return true or false when calling showError() depending on value is invalid or a server error has arrived, or not', function (done) {\n\n    var showError = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        showError = this.showError;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\", validationError: \"This is not an email\"})\n      )\n    );\n\n    expect(showError()).toBe(true);\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(showError()).toBe(false);\n\n    jasmine.Ajax.install();\n    TestUtils.Simulate.submit(form.getDOMNode());    \n    jasmine.Ajax.requests.mostRecent().respondWith({\n      status: 500,\n      responseType: 'application/json',\n      responseText: '{\"foo\": \"Email already exists\"}'\n    });\n    setTimeout(function () {\n      expect(showError()).toBe(true);\n      jasmine.Ajax.uninstall();\n      done();\n    }, 0);\n  });\n\n  it('should return true or false when calling isPrestine() depending on input has been \"touched\" or not', function () {\n    \n    var isPristine = null;\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      componentDidMount: function () {\n        isPristine = this.isPristine;\n      },\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}, \n        React.createElement(TestInput, {name: \"A\", value: \"foo\"})\n      )\n    );\n\n    expect(isPristine()).toBe(true);\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo'}});\n    expect(isPristine()).toBe(false);\n    \n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Formsy', function () {\n\n  describe('Setting up a form', function () {\n\n    it('should render a form into the document', function () {\n      var form = TestUtils.renderIntoDocument( React.createElement(Formsy.Form, null));\n      expect(form.getDOMNode().tagName).toEqual('FORM');\n    });\n\n    it('should set a class name if passed', function () {\n      var form = TestUtils.renderIntoDocument( React.createElement(Formsy.Form, {className: \"foo\"}));\n      expect(form.getDOMNode().className).toEqual('foo');\n    });\n\n    it('should allow for null/undefined children', function (done) {\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n\n      var model = null;\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          return (\n            React.createElement(Formsy.Form, {onSubmit:  this.onSubmit}, \n              React.createElement(\"h1\", null, \"Test\"), \n              null, \n              undefined, \n              React.createElement(TestInput, {name: \"name\", value: 'foo' })\n            )\n          );\n        }\n      });\n\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null));\n      setTimeout(function () {\n        TestUtils.Simulate.submit(form.getDOMNode());\n        expect(model).toEqual({name: 'foo'});\n        done();\n      }, 10);\n    });\n\n    it('should allow for inputs being added dynamically', function (done) {\n\n      var inputs = [];\n      var forceUpdate = null;\n      var model = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        render: function () {\n          return React.createElement(\"div\", null)\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        inputs.push(TestInput({\n          name: 'test'\n        }));\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(model.test).toBeDefined();\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n    it('should allow dynamically added inputs to update the form-model', function (done) {\n\n      var inputs = [];\n      var forceUpdate = null;\n      var model = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      setTimeout(function () {\n\n        inputs.push(TestInput({\n          name: 'test'\n        }));\n\n        forceUpdate(function () {\n\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.change(TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT'), {target: {value: 'foo'}});\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(model.test).toBe('foo');\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n    it('should allow a dynamically updated input to update the form-model', function (done) {\n\n      var forceUpdate = null;\n      var model = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n\n      var input;\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onSubmit: function (formModel) {\n          model = formModel;\n        },\n        render: function () {\n          input = React.createElement(TestInput, {name: \"test\", value:  this.props.value});\n\n          return (\n            React.createElement(Formsy.Form, {onSubmit: this.onSubmit}, \n              input\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, {value: \"foo\"}));\n\n      // Wait before changing the input\n      setTimeout(function () {\n        form.setProps({value: 'bar'});\n\n        forceUpdate(function () {\n          // Wait for next event loop, as that does the form\n          setTimeout(function () {\n            TestUtils.Simulate.submit(form.getDOMNode());\n            expect(model.test).toBe('bar');\n            done();\n          }, 0);\n\n        });\n\n      }, 10);\n\n    });\n\n\n    it('should invalidate a valid form if dynamically inserted input is invalid', function (done) {\n\n      var forceUpdate = null;\n      var isInvalid = false;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n\n\n      var inputs = [TestInput({\n        name: 'test',\n        validations: 'isEmail',\n        value: 'foo@bar.com'\n      })];\n\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        setInvalid: function () {\n          isInvalid = true;\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onInvalid: this.setInvalid}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      expect(isInvalid).toBe(false);\n\n      inputs.push(TestInput({\n        name: 'test2',\n        validations: 'isEmail',\n        value: 'foo@bar'\n      }));\n\n\n      forceUpdate(function () {\n\n        // Wait for next event loop, as that does the form\n        setTimeout(function () {\n          TestUtils.Simulate.submit(form.getDOMNode());\n          expect(isInvalid).toBe(true);\n          done();\n        }, 0);\n\n      });\n\n    });\n\n    it('should not trigger onChange when form is mounted', function () {\n      var hasChanged = jasmine.createSpy('onChange');\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return React.createElement(Formsy.Form, {onChange: this.onChange});\n        }\n      });\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null));\n      expect(hasChanged).not.toHaveBeenCalled();\n    });\n\n    it('should trigger onChange when form element is changed', function () {\n      var hasChanged = jasmine.createSpy('onChange');\n      var MyInput = React.createClass({displayName: \"MyInput\",\n        mixins: [Formsy.Mixin],\n        onChange: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.onChange})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return (\n            React.createElement(Formsy.Form, {onChange: this.onChange}, \n              React.createElement(MyInput, {name: \"foo\"})\n            )\n          );\n        }\n      });\n      var form = TestUtils.renderIntoDocument(React.createElement(TestForm, null));\n      TestUtils.Simulate.change(TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT'), {target: {value: 'bar'}});\n      expect(hasChanged).toHaveBeenCalled();\n    });\n\n    it('should trigger onChange when new input is added to form', function (done) {\n      var hasChanged = jasmine.createSpy('onChange');\n      var inputs = [];\n      var forceUpdate = null;\n      var TestInput = React.createClass({displayName: \"TestInput\",\n        mixins: [Formsy.Mixin],\n        changeValue: function (event) {\n          this.setValue(event.target.value);\n        },\n        render: function () {\n          return React.createElement(\"input\", {value: this.getValue(), onChange: this.changeValue})\n        }\n      });\n      var TestForm = React.createClass({displayName: \"TestForm\",\n        componentWillMount: function () {\n          forceUpdate = this.forceUpdate.bind(this);\n        },\n        onChange: function () {\n          hasChanged();\n        },\n        render: function () {\n          return ( \n            React.createElement(Formsy.Form, {onChange: this.onChange}, \n              inputs\n            ));\n        }\n      });\n      var form = TestUtils.renderIntoDocument( \n        React.createElement(TestForm, null) \n      );\n\n      // Wait before adding the input\n      inputs.push(TestInput({\n        name: 'test'\n      }));\n\n      forceUpdate(function () {\n\n        // Wait for next event loop, as that does the form\n        setTimeout(function () {\n          expect(hasChanged).toHaveBeenCalled();\n          done();\n        }, 0);\n\n      });\n\n    });\n\n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Ajax', function() {\n\n  beforeEach(function () {\n    jasmine.Ajax.install();\n  });\n\n  afterEach(function () {\n    jasmine.Ajax.uninstall();\n  });\n\n  it('should post to a given url if passed', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().url).toBe('/users');\n    expect(jasmine.Ajax.requests.mostRecent().method).toBe('POST');\n\n  });\n\n  it('should put to a given url if passed a method attribute', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", method: \"PUT\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().url).toBe('/users');\n    expect(jasmine.Ajax.requests.mostRecent().method).toBe('PUT');\n\n  });\n\n  it('should pass x-www-form-urlencoded as contentType when urlencoded is set as contentType', function () {\n\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", contentType: \"urlencoded\"}\n      )\n    );\n    \n    TestUtils.Simulate.submit(form.getDOMNode());\n    expect(jasmine.Ajax.requests.mostRecent().contentType()).toBe('application/x-www-form-urlencoded');\n\n  });\n\n  it('should run an onSuccess handler, if passed and ajax is successfull. First argument is data from server', function (done) {\n \n    var onSuccess = jasmine.createSpy(\"success\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSuccess: onSuccess}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 200,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSuccess).toHaveBeenCalledWith({});\n      done();\n    }, 0);\n\n  });\n\n  it('should not do ajax request if onSubmit handler is passed, but pass the model as first argument to onSubmit handler', function () {\n    \n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue()})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onSubmit: onSubmit}, \n        React.createElement(TestInput, {name: \"foo\", value: \"bar\"})\n      )\n    );\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    expect(jasmine.Ajax.requests.count()).toBe(0);\n\n    function onSubmit (data) {\n      expect(data).toEqual({\n        foo: 'bar'\n      });\n    }\n\n  });\n\n  it('should trigger an onSubmitted handler, if passed and the submit has responded with SUCCESS', function (done) {\n    \n    var onSubmitted = jasmine.createSpy(\"submitted\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSubmitted: onSubmitted}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 200,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSubmitted).toHaveBeenCalled();\n      done();\n    }, 0);\n\n  });\n\n  it('should trigger an onSubmitted handler, if passed and the submit has responded with ERROR', function (done) {\n    \n    var onSubmitted = jasmine.createSpy(\"submitted\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onSubmitted: onSubmitted}\n      )\n    );\n    \n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onSubmitted).toHaveBeenCalled();\n      done();\n    }, 0);\n\n  });\n\n  it('should trigger an onError handler, if passed and the submit has responded with ERROR', function (done) {\n    \n    var onError = jasmine.createSpy(\"error\");\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {url: \"/users\", onError: onError}\n      )\n    );\n    \n    // Do not return any error because there are no inputs\n    jasmine.Ajax.stubRequest('/users').andReturn({\n      status: 500,\n      contentType: 'application/json',\n      responseText: '{}'\n    });\n\n    TestUtils.Simulate.submit(form.getDOMNode());\n\n    // Since ajax is returned as a promise (async), move assertion\n    // to end of event loop\n    setTimeout(function () {\n      expect(onError).toHaveBeenCalledWith({});\n      done();\n    }, 0);\n\n  });\n\n});\n","var Formsy = require('./../src/main.js');\n\ndescribe('Validation', function() {\n\n  it('should trigger an onValid handler, if passed, when form is valid', function () {\n    \n    var onValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onValid: onValid}, \n        React.createElement(TestInput, {name: \"foo\", required: true})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: 'foo'}});\n    expect(onValid).toHaveBeenCalled();\n\n  });\n\n  it('should trigger an onInvalid handler, if passed, when form is invalid', function () {\n    \n    var onInvalid = jasmine.createSpy('invalid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, {onValid: onInvalid}, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    TestUtils.Simulate.change(input, {target: {value: ''}});\n    expect(onInvalid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isEmail', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isEmail\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: 'foo@foo.com'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isNumeric', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(event.target.value);\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isNumeric\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: '123'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n  it('RULE: isNumeric (actual number)', function () {\n    \n    var isValid = jasmine.createSpy('valid');\n    var TestInput = React.createClass({displayName: \"TestInput\",\n      mixins: [Formsy.Mixin],\n      updateValue: function (event) {\n        this.setValue(Number(event.target.value));\n      },\n      render: function () {\n        if (this.isValid()) {\n          isValid();\n        }\n        return React.createElement(\"input\", {value: this.getValue(), onChange: this.updateValue})\n      }\n    });\n    var form = TestUtils.renderIntoDocument(\n      React.createElement(Formsy.Form, null, \n        React.createElement(TestInput, {name: \"foo\", value: \"foo\", validations: \"isNumeric\"})\n      )\n    );\n\n    var input = TestUtils.findRenderedDOMComponentWithTag(form, 'INPUT');\n    expect(isValid).not.toHaveBeenCalled();\n    TestUtils.Simulate.change(input, {target: {value: '123'}});\n    expect(isValid).toHaveBeenCalled();\n\n  });\n\n});\n","var React = global.React || require('react');\nvar Formsy = {};\nvar validationRules = {\n  'isValue': function (value) {\n    return value !== '';\n  },\n  'isEmail': function (value) {\n    return value.match(/^((([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+(\\.([a-z]|\\d|[!#\\$%&'\\*\\+\\-\\/=\\?\\^_`{\\|}~]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])+)*)|((\\x22)((((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(([\\x01-\\x08\\x0b\\x0c\\x0e-\\x1f\\x7f]|\\x21|[\\x23-\\x5b]|[\\x5d-\\x7e]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(\\\\([\\x01-\\x09\\x0b\\x0c\\x0d-\\x7f]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF]))))*(((\\x20|\\x09)*(\\x0d\\x0a))?(\\x20|\\x09)+)?(\\x22)))@((([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|\\d|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))\\.)+(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])|(([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])([a-z]|\\d|-|\\.|_|~|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])*([a-z]|[\\u00A0-\\uD7FF\\uF900-\\uFDCF\\uFDF0-\\uFFEF])))$/i);\n  },\n  'isTrue': function (value) {\n    return value === true;\n  },\n  'isNumeric': function (value) {\n    if (typeof value === 'number') {\n      return true;\n    } else {\n      return value.match(/^-?[0-9]+$/);\n    }\n  },\n  'isAlpha': function (value) {\n    return value.match(/^[a-zA-Z]+$/);\n  },\n  'isWords': function (value) {\n    return value.match(/^[a-zA-Z\\s]+$/);\n  },\n  'isSpecialWords': function (value) {\n    return value.match(/^[a-zA-Z\\s\\u00C0-\\u017F]+$/);\n  },\n  isLength: function (value, min, max) {\n    if (max !== undefined) {\n      return value.length >= min && value.length <= max;\n    }\n    return value.length >= min;\n  },\n  equals: function (value, eql) {\n    return value == eql;\n  },\n  equalsField: function (value, field) {\n    return value === this[field];\n  }\n};\n\nvar toURLEncoded = function (element, key, list) {\n  var list = list || [];\n  if (typeof (element) == 'object') {\n    for (var idx in element)\n      toURLEncoded(element[idx], key ? key + '[' + idx + ']' : idx, list);\n  } else {\n    list.push(key + '=' + encodeURIComponent(element));\n  }\n  return list.join('&');\n};\n\nvar request = function (method, url, data, contentType, headers) {\n\n  var contentType = contentType === 'urlencoded' ? 'application/' + contentType.replace('urlencoded', 'x-www-form-urlencoded') : 'application/json';\n  data = contentType === 'application/json' ? JSON.stringify(data) : toURLEncoded(data);\n\n  return new Promise(function (resolve, reject) {\n    try {\n      var xhr = new XMLHttpRequest();\n      xhr.open(method, url, true);\n      xhr.setRequestHeader('Accept', 'application/json');\n      xhr.setRequestHeader('Content-Type', contentType);\n\n      // Add passed headers\n      Object.keys(headers).forEach(function (header) {\n        xhr.setRequestHeader(header, headers[header]);\n      });\n\n      xhr.onreadystatechange = function () {\n        if (xhr.readyState === 4) {\n\n          try {\n            var response = xhr.responseText ? JSON.parse(xhr.responseText) : null;\n            if (xhr.status >= 200 && xhr.status < 300) {\n              resolve(response);\n            } else {\n              reject(response);\n            }\n          } catch (e) {\n            reject(e);\n          }\n\n        }\n      };\n      xhr.send(data);\n    } catch (e) {\n      reject(e);\n    }\n  });\n\n};\n\nvar arraysDiffer = function (arrayA, arrayB) {\n  var isDifferent = false;\n  if (arrayA.length !== arrayB.length) {\n    isDifferent = true;\n  } else {\n    arrayA.forEach(function (item, index) {\n      if (item !== arrayB[index]) {\n        isDifferent = true;\n      }\n    });\n  }\n  return isDifferent;\n};\n\nvar ajax = {\n  post: request.bind(null, 'POST'),\n  put: request.bind(null, 'PUT')\n};\nvar options = {};\n\nFormsy.defaults = function (passedOptions) {\n  options = passedOptions;\n};\n\nFormsy.Mixin = {\n  getInitialState: function () {\n    return {\n      _value: this.props.value ? this.props.value : '',\n      _isValid: true,\n      _isPristine: true\n    };\n  },\n  componentWillMount: function () {\n\n    var configure = function () {\n      // Add validations to the store itself as the props object can not be modified\n      this._validations = this.props.validations || '';\n\n      if (this.props.required) {\n        this._validations = this.props.validations ? this.props.validations + ',' : '';\n        this._validations += 'isValue';\n      }\n      this.props._attachToForm(this);\n    }.bind(this);\n\n    if (!this.props.name) {\n      throw new Error('Form Input requires a name property when used');\n    }\n\n    if (!this.props._attachToForm) {\n      return setTimeout(function () {\n        if (!this.props._attachToForm) {\n          throw new Error('Form Mixin requires component to be nested in a Form');\n        }\n        configure();\n      }.bind(this), 0);\n    }\n    configure();\n\n  },\n\n  // We have to make the validate method is kept when new props are added\n  componentWillReceiveProps: function (nextProps) {\n    nextProps._attachToForm = this.props._attachToForm;\n    nextProps._detachFromForm = this.props._detachFromForm;\n    nextProps._validate = this.props._validate;\n  },\n\n  componentDidUpdate: function(prevProps, prevState) {\n\n    // If the input is untouched and something outside changes the value\n    // update the FORM model by re-attaching to the form\n    if (this.state._isPristine) {\n      if (this.props.value !== prevProps.value && this.state._value === prevProps.value) {\n        this.state._value = this.props.value || '';\n        this.props._attachToForm(this);\n      }\n    }\n  },\n\n  // Detach it when component unmounts\n  componentWillUnmount: function () {\n    this.props._detachFromForm(this);\n  },\n\n  // We validate after the value has been set\n  setValue: function (value) {\n    this.setState({\n      _value: value,\n      _isPristine: false\n    }, function () {\n      this.props._validate(this);\n    }.bind(this));\n  },\n  resetValue: function () {\n    this.setState({\n      _value: '',\n      _isPristine: true\n    }, function () {\n      this.props._validate(this);\n    });\n  },\n  getValue: function () {\n    return this.state._value;\n  },\n  hasValue: function () {\n    return this.state._value !== '';\n  },\n  getErrorMessage: function () {\n    return this.isValid() || this.showRequired() ? null : this.state._serverError || this.props.validationError;\n  },\n  isValid: function () {\n    return this.state._isValid;\n  },\n  isPristine: function () {\n    return this.state._isPristine;\n  },\n  isRequired: function () {\n    return !!this.props.required;\n  },\n  showRequired: function () {\n    return this.isRequired() && this.state._value === '';\n  },\n  showError: function () {\n    return !this.showRequired() && !this.state._isValid;\n  }\n};\n\nFormsy.addValidationRule = function (name, func) {\n  validationRules[name] = func;\n};\n\nFormsy.Form = React.createClass({displayName: \"Form\",\n  getInitialState: function () {\n    return {\n      isValid: true,\n      isSubmitting: false,\n      canChange: false\n    };\n  },\n  getDefaultProps: function () {\n    return {\n      headers: {},\n      onSuccess: function () {},\n      onError: function () {},\n      onSubmit: function () {},\n      onSubmitted: function () {},\n      onValid: function () {},\n      onInvalid: function () {},\n      onChange: function () {}\n    };\n  },\n\n  // Add a map to store the inputs of the form, a model to store\n  // the values of the form and register child inputs\n  componentWillMount: function () {\n    this.inputs = {};\n    this.model = {};\n    this.registerInputs(this.props.children);\n  },\n\n  componentDidMount: function () {\n    this.validateForm();\n  },\n\n  componentWillUpdate: function () {\n    var inputKeys = Object.keys(this.inputs);\n\n    // The updated children array is not available here for some reason,\n    // we need to wait for next event loop\n    setTimeout(function () {\n      this.registerInputs(this.props.children);\n\n      var newInputKeys = Object.keys(this.inputs);\n      if (arraysDiffer(inputKeys, newInputKeys)) {\n        this.validateForm();\n      }\n    }.bind(this), 0);\n  },\n\n  // Update model, submit to url prop and send the model\n  submit: function (event) {\n    event.preventDefault();\n\n    // Trigger form as not pristine.\n    // If any inputs have not been touched yet this will make them dirty\n    // so validation becomes visible (if based on isPristine)\n    this.setFormPristine(false);\n\n    // To support use cases where no async or request operation is needed.\n    // The \"onSubmit\" callback is called with the model e.g. {fieldName: \"myValue\"},\n    // if wanting to reset the entire form to original state, the second param is a callback for this.\n    if (!this.props.url) {\n      this.updateModel();\n      this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n      return;\n    }\n\n    this.updateModel();\n    this.setState({\n      isSubmitting: true\n    });\n\n    this.props.onSubmit(this.mapModel(), this.resetModel, this.updateInputsWithError);\n\n    var headers = (Object.keys(this.props.headers).length && this.props.headers) || options.headers || {};\n\n    var method = this.props.method && ajax[this.props.method.toLowerCase()] ? this.props.method.toLowerCase() : 'post';\n    ajax[method](this.props.url, this.mapModel(), this.props.contentType || options.contentType || 'json', headers)\n      .then(function (response) {\n        this.props.onSuccess(response);\n        this.props.onSubmitted();\n      }.bind(this))\n      .catch(this.failSubmit);\n  },\n\n  mapModel: function () {\n    return this.props.mapping ? this.props.mapping(this.model) : this.model;\n  },\n\n  // Goes through all registered components and\n  // updates the model values\n  updateModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      var component = this.inputs[name];\n      this.model[name] = component.state._value;\n    }.bind(this));\n  },\n\n  // Reset each key in the model to the original / initial value\n  resetModel: function () {\n    Object.keys(this.inputs).forEach(function (name) {\n      this.inputs[name].resetValue();\n    }.bind(this));\n    this.validateForm();\n  },\n\n  // Go through errors from server and grab the components\n  // stored in the inputs map. Change their state to invalid\n  // and set the serverError message\n  updateInputsWithError: function (errors) {\n    Object.keys(errors).forEach(function (name, index) {\n      var component = this.inputs[name];\n\n      if (!component) {\n        throw new Error('You are trying to update an input that does not exists. Verify errors object with input names. ' + JSON.stringify(errors));\n      }\n\n      var args = [{\n        _isValid: false,\n        _serverError: errors[name]\n      }];\n      component.setState.apply(component, args);\n    }.bind(this));\n  },\n\n  failSubmit: function (errors) {\n    this.updateInputsWithError(errors);\n    this.setState({\n      isSubmitting: false\n    });\n    this.props.onError(errors);\n    this.props.onSubmitted();\n  },\n\n  // Traverse the children and children of children to find\n  // all inputs by checking the name prop. Maybe do a better\n  // check here\n  registerInputs: function (children) {\n    React.Children.forEach(children, function (child) {\n\n      if (child && child.props && child.props.name) {\n        child.props._attachToForm = this.attachToForm;\n        child.props._detachFromForm = this.detachFromForm;\n        child.props._validate = this.validate;\n      }\n\n      if (child && child.props && child.props.children) {\n        this.registerInputs(child.props.children);\n      }\n\n    }.bind(this));\n  },\n\n  getCurrentValues: function () {\n    return Object.keys(this.inputs).reduce(function (data, name) {\n      var component = this.inputs[name];\n      data[name] = component.state._value;\n      return data;\n    }.bind(this), {});\n  },\n\n  setFormPristine: function (isPristine) {\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // Iterate through each component and set it as pristine\n    // or \"dirty\".\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      component.setState({\n        _isPristine: isPristine\n      });\n    }.bind(this));\n  },\n\n  // Use the binded values and the actual input value to\n  // validate the input and set its state. Then check the\n  // state of the form itself\n  validate: function (component) {\n\n    // Trigger onChange\n    this.state.canChange && this.props.onChange && this.props.onChange(this.getCurrentValues());\n\n    if (!component.props.required && !component._validations) {\n      return;\n    }\n\n    // Run through the validations, split them up and call\n    // the validator IF there is a value or it is required\n    var isValid = this.runValidation(component);\n\n    component.setState({\n      _isValid: isValid,\n      _serverError: null\n    }, this.validateForm);\n\n  },\n\n  runValidation: function (component) {\n    var isValid = true;\n    if (component._validations.length && (component.props.required || component.state._value !== '')) {\n      component._validations.split(',').forEach(function (validation) {\n        var args = validation.split(':');\n        var validateMethod = args.shift();\n        args = args.map(function (arg) {\n          try {\n            return JSON.parse(arg);\n          } catch (e) {\n            return arg; // It is a string if it can not parse it\n          }\n        });\n        args = [component.state._value].concat(args);\n        if (!validationRules[validateMethod]) {\n          throw new Error('Formsy does not have the validation rule: ' + validateMethod);\n        }\n        if (!validationRules[validateMethod].apply(this.getCurrentValues(), args)) {\n          isValid = false;\n        }\n      }.bind(this));\n    }\n    return isValid;\n  },\n\n  // Validate the form by going through all child input components\n  // and check their state\n  validateForm: function () {\n    var allIsValid = true;\n    var inputs = this.inputs;\n    var inputKeys = Object.keys(inputs);\n\n    // We need a callback as we are validating all inputs again. This will\n    // run when the last component has set its state\n    var onValidationComplete = function () {\n      inputKeys.forEach(function (name) {\n        if (!inputs[name].state._isValid) {\n          allIsValid = false;\n        }\n      }.bind(this));\n\n      this.setState({\n        isValid: allIsValid\n      });\n\n      allIsValid && this.props.onValid();\n      !allIsValid && this.props.onInvalid();\n\n      // Tell the form that it can start to trigger change events\n      this.setState({canChange: true});\n\n    }.bind(this);\n\n    // Run validation again in case affected by other inputs. The\n    // last component validated will run the onValidationComplete callback\n    inputKeys.forEach(function (name, index) {\n      var component = inputs[name];\n      var isValid = this.runValidation(component);\n      component.setState({\n        _isValid: isValid,\n        _serverError: null\n      }, index === inputKeys.length - 1 ? onValidationComplete : null);\n    }.bind(this));\n\n    // If there are no inputs, it is ready to trigger change events\n    if (!inputKeys.length) {\n      this.setState({canChange: true});\n    }\n\n  },\n\n  // Method put on each input component to register\n  // itself to the form\n  attachToForm: function (component) {\n    this.inputs[component.props.name] = component;\n    this.model[component.props.name] = component.state._value;\n    this.validate(component);\n  },\n\n  // Method put on each input component to unregister\n  // itself from the form\n  detachFromForm: function (component) {\n    delete this.inputs[component.props.name];\n    delete this.model[component.props.name];\n  },\n  render: function () {\n\n    return React.DOM.form({\n        onSubmit: this.submit,\n        className: this.props.className\n      },\n      this.props.children\n    );\n\n  }\n});\n\nif (!global.exports && !global.module && (!global.define || !global.define.amd)) {\n  global.Formsy = Formsy;\n}\n\nmodule.exports = Formsy;\n"]} diff --git a/package.json b/package.json index 8930837..4c1bc84 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "formsy-react", - "version": "0.7.0", + "version": "0.7.1", "description": "A form input builder and validator for React JS", "main": "src/main.js", "scripts": { diff --git a/release/formsy-react.js b/release/formsy-react.js index f60b79b..e13dda8 100644 --- a/release/formsy-react.js +++ b/release/formsy-react.js @@ -162,6 +162,18 @@ Formsy.Mixin = { nextProps._validate = this.props._validate; }, + componentDidUpdate: function(prevProps, prevState) { + + // If the input is untouched and something outside changes the value + // update the FORM model by re-attaching to the form + if (this.state._isPristine) { + if (this.props.value !== prevProps.value && this.state._value === prevProps.value) { + this.state._value = this.props.value || ''; + this.props._attachToForm(this); + } + } + }, + // Detach it when component unmounts componentWillUnmount: function () { this.props._detachFromForm(this); @@ -353,13 +365,13 @@ Formsy.Form = React.createClass({displayName: "Form", registerInputs: function (children) { React.Children.forEach(children, function (child) { - if (child.props && child.props.name) { + if (child && child.props && child.props.name) { child.props._attachToForm = this.attachToForm; child.props._detachFromForm = this.detachFromForm; child.props._validate = this.validate; } - if (child.props && child.props.children) { + if (child && child.props && child.props.children) { this.registerInputs(child.props.children); } diff --git a/release/formsy-react.min.js b/release/formsy-react.min.js index 1ae4037..95ab9b5 100644 --- a/release/formsy-react.min.js +++ b/release/formsy-react.min.js @@ -1 +1 @@ -!function t(i,e,n){function s(o,u){if(!e[o]){if(!i[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(r)return r(o,!0);var h=new Error("Cannot find module '"+o+"'");throw h.code="MODULE_NOT_FOUND",h}var p=e[o]={exports:{}};i[o][0].call(p.exports,function(t){var e=i[o][1][t];return s(e?e:t)},p,p.exports,t,i,e,n)}return e[o].exports}for(var r="function"==typeof require&&require,o=0;o=i&&t.length<=e:t.length>=i},equals:function(t,i){return t==i},equalsField:function(t,i){return t===this[i]}},o=function(t,i,e){var e=e||[];if("object"==typeof t)for(var n in t)o(t[n],i?i+"["+n+"]":n,e);else e.push(i+"="+encodeURIComponent(t));return e.join("&")},u=function(t,i,e,n,s){var n="urlencoded"===n?"application/"+n.replace("urlencoded","x-www-form-urlencoded"):"application/json";return e="application/json"===n?JSON.stringify(e):o(e),new Promise(function(r,o){try{var u=new XMLHttpRequest;u.open(t,i,!0),u.setRequestHeader("Accept","application/json"),u.setRequestHeader("Content-Type",n),Object.keys(s).forEach(function(t){u.setRequestHeader(t,s[t])}),u.onreadystatechange=function(){if(4===u.readyState)try{var t=u.responseText?JSON.parse(u.responseText):null;u.status>=200&&u.status<300?r(t):o(t)}catch(i){o(i)}},u.send(e)}catch(a){o(a)}})},a=function(t,i){var e=!1;return t.length!==i.length?e=!0:t.forEach(function(t,n){t!==i[n]&&(e=!0)}),e},h={post:u.bind(null,"POST"),put:u.bind(null,"PUT")},p={};s.defaults=function(t){p=t},s.Mixin={getInitialState:function(){return{_value:this.props.value?this.props.value:"",_isValid:!0,_isPristine:!0}},componentWillMount:function(){var t=function(){this._validations=this.props.validations||"",this.props.required&&(this._validations=this.props.validations?this.props.validations+",":"",this._validations+="isValue"),this.props._attachToForm(this)}.bind(this);if(!this.props.name)throw new Error("Form Input requires a name property when used");return this.props._attachToForm?(t(),void 0):setTimeout(function(){if(!this.props._attachToForm)throw new Error("Form Mixin requires component to be nested in a Form");t()}.bind(this),0)},componentWillReceiveProps:function(t){t._attachToForm=this.props._attachToForm,t._detachFromForm=this.props._detachFromForm,t._validate=this.props._validate},componentWillUnmount:function(){this.props._detachFromForm(this)},setValue:function(t){this.setState({_value:t,_isPristine:!1},function(){this.props._validate(this)}.bind(this))},resetValue:function(){this.setState({_value:"",_isPristine:!0},function(){this.props._validate(this)})},getValue:function(){return this.state._value},hasValue:function(){return""!==this.state._value},getErrorMessage:function(){return this.isValid()||this.showRequired()?null:this.state._serverError||this.props.validationError},isValid:function(){return this.state._isValid},isPristine:function(){return this.state._isPristine},isRequired:function(){return!!this.props.required},showRequired:function(){return this.isRequired()&&""===this.state._value},showError:function(){return!this.showRequired()&&!this.state._isValid}},s.addValidationRule=function(t,i){r[t]=i},s.Form=n.createClass({displayName:"Form",getInitialState:function(){return{isValid:!0,isSubmitting:!1,canChange:!1}},getDefaultProps:function(){return{headers:{},onSuccess:function(){},onError:function(){},onSubmit:function(){},onSubmitted:function(){},onValid:function(){},onInvalid:function(){},onChange:function(){}}},componentWillMount:function(){this.inputs={},this.model={},this.registerInputs(this.props.children)},componentDidMount:function(){this.validateForm()},componentWillUpdate:function(){var t=Object.keys(this.inputs);setTimeout(function(){this.registerInputs(this.props.children);var i=Object.keys(this.inputs);a(t,i)&&this.validateForm()}.bind(this),0)},submit:function(t){if(t.preventDefault(),this.setFormPristine(!1),!this.props.url)return this.updateModel(),this.props.onSubmit(this.mapModel(),this.resetModel,this.updateInputsWithError),void 0;this.updateModel(),this.setState({isSubmitting:!0}),this.props.onSubmit(this.mapModel(),this.resetModel,this.updateInputsWithError);var i=Object.keys(this.props.headers).length&&this.props.headers||p.headers||{},e=this.props.method&&h[this.props.method.toLowerCase()]?this.props.method.toLowerCase():"post";h[e](this.props.url,this.mapModel(),this.props.contentType||p.contentType||"json",i).then(function(t){this.props.onSuccess(t),this.props.onSubmitted()}.bind(this)).catch(this.failSubmit)},mapModel:function(){return this.props.mapping?this.props.mapping(this.model):this.model},updateModel:function(){Object.keys(this.inputs).forEach(function(t){var i=this.inputs[t];this.model[t]=i.state._value}.bind(this))},resetModel:function(){Object.keys(this.inputs).forEach(function(t){this.inputs[t].resetValue()}.bind(this)),this.validateForm()},updateInputsWithError:function(t){Object.keys(t).forEach(function(i){var e=this.inputs[i];if(!e)throw new Error("You are trying to update an input that does not exists. Verify errors object with input names. "+JSON.stringify(t));var n=[{_isValid:!1,_serverError:t[i]}];e.setState.apply(e,n)}.bind(this))},failSubmit:function(t){this.updateInputsWithError(t),this.setState({isSubmitting:!1}),this.props.onError(t),this.props.onSubmitted()},registerInputs:function(t){n.Children.forEach(t,function(t){t.props&&t.props.name&&(t.props._attachToForm=this.attachToForm,t.props._detachFromForm=this.detachFromForm,t.props._validate=this.validate),t.props&&t.props.children&&this.registerInputs(t.props.children)}.bind(this))},getCurrentValues:function(){return Object.keys(this.inputs).reduce(function(t,i){var e=this.inputs[i];return t[i]=e.state._value,t}.bind(this),{})},setFormPristine:function(t){var i=this.inputs,e=Object.keys(i);e.forEach(function(e){var n=i[e];n.setState({_isPristine:t})}.bind(this))},validate:function(t){if(this.state.canChange&&this.props.onChange&&this.props.onChange(this.getCurrentValues()),t.props.required||t._validations){var i=this.runValidation(t);t.setState({_isValid:i,_serverError:null},this.validateForm)}},runValidation:function(t){var i=!0;return t._validations.length&&(t.props.required||""!==t.state._value)&&t._validations.split(",").forEach(function(e){var n=e.split(":"),s=n.shift();if(n=n.map(function(t){try{return JSON.parse(t)}catch(i){return t}}),n=[t.state._value].concat(n),!r[s])throw new Error("Formsy does not have the validation rule: "+s);r[s].apply(this.getCurrentValues(),n)||(i=!1)}.bind(this)),i},validateForm:function(){var t=!0,i=this.inputs,e=Object.keys(i),n=function(){e.forEach(function(e){i[e].state._isValid||(t=!1)}.bind(this)),this.setState({isValid:t}),t&&this.props.onValid(),!t&&this.props.onInvalid(),this.setState({canChange:!0})}.bind(this);e.forEach(function(t,s){var r=i[t],o=this.runValidation(r);r.setState({_isValid:o,_serverError:null},s===e.length-1?n:null)}.bind(this)),e.length||this.setState({canChange:!0})},attachToForm:function(t){this.inputs[t.props.name]=t,this.model[t.props.name]=t.state._value,this.validate(t)},detachFromForm:function(t){delete this.inputs[t.props.name],delete this.model[t.props.name]},render:function(){return n.DOM.form({onSubmit:this.submit,className:this.props.className},this.props.children)}}),e.exports||e.module||e.define&&e.define.amd||(e.Formsy=s),i.exports=s}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{react:"react"}]},{},[1]); \ No newline at end of file +!function t(i,e,n){function s(o,u){if(!e[o]){if(!i[o]){var a="function"==typeof require&&require;if(!u&&a)return a(o,!0);if(r)return r(o,!0);var h=new Error("Cannot find module '"+o+"'");throw h.code="MODULE_NOT_FOUND",h}var p=e[o]={exports:{}};i[o][0].call(p.exports,function(t){var e=i[o][1][t];return s(e?e:t)},p,p.exports,t,i,e,n)}return e[o].exports}for(var r="function"==typeof require&&require,o=0;o=i&&t.length<=e:t.length>=i},equals:function(t,i){return t==i},equalsField:function(t,i){return t===this[i]}},o=function(t,i,e){var e=e||[];if("object"==typeof t)for(var n in t)o(t[n],i?i+"["+n+"]":n,e);else e.push(i+"="+encodeURIComponent(t));return e.join("&")},u=function(t,i,e,n,s){var n="urlencoded"===n?"application/"+n.replace("urlencoded","x-www-form-urlencoded"):"application/json";return e="application/json"===n?JSON.stringify(e):o(e),new Promise(function(r,o){try{var u=new XMLHttpRequest;u.open(t,i,!0),u.setRequestHeader("Accept","application/json"),u.setRequestHeader("Content-Type",n),Object.keys(s).forEach(function(t){u.setRequestHeader(t,s[t])}),u.onreadystatechange=function(){if(4===u.readyState)try{var t=u.responseText?JSON.parse(u.responseText):null;u.status>=200&&u.status<300?r(t):o(t)}catch(i){o(i)}},u.send(e)}catch(a){o(a)}})},a=function(t,i){var e=!1;return t.length!==i.length?e=!0:t.forEach(function(t,n){t!==i[n]&&(e=!0)}),e},h={post:u.bind(null,"POST"),put:u.bind(null,"PUT")},p={};s.defaults=function(t){p=t},s.Mixin={getInitialState:function(){return{_value:this.props.value?this.props.value:"",_isValid:!0,_isPristine:!0}},componentWillMount:function(){var t=function(){this._validations=this.props.validations||"",this.props.required&&(this._validations=this.props.validations?this.props.validations+",":"",this._validations+="isValue"),this.props._attachToForm(this)}.bind(this);if(!this.props.name)throw new Error("Form Input requires a name property when used");return this.props._attachToForm?(t(),void 0):setTimeout(function(){if(!this.props._attachToForm)throw new Error("Form Mixin requires component to be nested in a Form");t()}.bind(this),0)},componentWillReceiveProps:function(t){t._attachToForm=this.props._attachToForm,t._detachFromForm=this.props._detachFromForm,t._validate=this.props._validate},componentDidUpdate:function(t){this.state._isPristine&&this.props.value!==t.value&&this.state._value===t.value&&(this.state._value=this.props.value||"",this.props._attachToForm(this))},componentWillUnmount:function(){this.props._detachFromForm(this)},setValue:function(t){this.setState({_value:t,_isPristine:!1},function(){this.props._validate(this)}.bind(this))},resetValue:function(){this.setState({_value:"",_isPristine:!0},function(){this.props._validate(this)})},getValue:function(){return this.state._value},hasValue:function(){return""!==this.state._value},getErrorMessage:function(){return this.isValid()||this.showRequired()?null:this.state._serverError||this.props.validationError},isValid:function(){return this.state._isValid},isPristine:function(){return this.state._isPristine},isRequired:function(){return!!this.props.required},showRequired:function(){return this.isRequired()&&""===this.state._value},showError:function(){return!this.showRequired()&&!this.state._isValid}},s.addValidationRule=function(t,i){r[t]=i},s.Form=n.createClass({displayName:"Form",getInitialState:function(){return{isValid:!0,isSubmitting:!1,canChange:!1}},getDefaultProps:function(){return{headers:{},onSuccess:function(){},onError:function(){},onSubmit:function(){},onSubmitted:function(){},onValid:function(){},onInvalid:function(){},onChange:function(){}}},componentWillMount:function(){this.inputs={},this.model={},this.registerInputs(this.props.children)},componentDidMount:function(){this.validateForm()},componentWillUpdate:function(){var t=Object.keys(this.inputs);setTimeout(function(){this.registerInputs(this.props.children);var i=Object.keys(this.inputs);a(t,i)&&this.validateForm()}.bind(this),0)},submit:function(t){if(t.preventDefault(),this.setFormPristine(!1),!this.props.url)return this.updateModel(),this.props.onSubmit(this.mapModel(),this.resetModel,this.updateInputsWithError),void 0;this.updateModel(),this.setState({isSubmitting:!0}),this.props.onSubmit(this.mapModel(),this.resetModel,this.updateInputsWithError);var i=Object.keys(this.props.headers).length&&this.props.headers||p.headers||{},e=this.props.method&&h[this.props.method.toLowerCase()]?this.props.method.toLowerCase():"post";h[e](this.props.url,this.mapModel(),this.props.contentType||p.contentType||"json",i).then(function(t){this.props.onSuccess(t),this.props.onSubmitted()}.bind(this)).catch(this.failSubmit)},mapModel:function(){return this.props.mapping?this.props.mapping(this.model):this.model},updateModel:function(){Object.keys(this.inputs).forEach(function(t){var i=this.inputs[t];this.model[t]=i.state._value}.bind(this))},resetModel:function(){Object.keys(this.inputs).forEach(function(t){this.inputs[t].resetValue()}.bind(this)),this.validateForm()},updateInputsWithError:function(t){Object.keys(t).forEach(function(i){var e=this.inputs[i];if(!e)throw new Error("You are trying to update an input that does not exists. Verify errors object with input names. "+JSON.stringify(t));var n=[{_isValid:!1,_serverError:t[i]}];e.setState.apply(e,n)}.bind(this))},failSubmit:function(t){this.updateInputsWithError(t),this.setState({isSubmitting:!1}),this.props.onError(t),this.props.onSubmitted()},registerInputs:function(t){n.Children.forEach(t,function(t){t&&t.props&&t.props.name&&(t.props._attachToForm=this.attachToForm,t.props._detachFromForm=this.detachFromForm,t.props._validate=this.validate),t&&t.props&&t.props.children&&this.registerInputs(t.props.children)}.bind(this))},getCurrentValues:function(){return Object.keys(this.inputs).reduce(function(t,i){var e=this.inputs[i];return t[i]=e.state._value,t}.bind(this),{})},setFormPristine:function(t){var i=this.inputs,e=Object.keys(i);e.forEach(function(e){var n=i[e];n.setState({_isPristine:t})}.bind(this))},validate:function(t){if(this.state.canChange&&this.props.onChange&&this.props.onChange(this.getCurrentValues()),t.props.required||t._validations){var i=this.runValidation(t);t.setState({_isValid:i,_serverError:null},this.validateForm)}},runValidation:function(t){var i=!0;return t._validations.length&&(t.props.required||""!==t.state._value)&&t._validations.split(",").forEach(function(e){var n=e.split(":"),s=n.shift();if(n=n.map(function(t){try{return JSON.parse(t)}catch(i){return t}}),n=[t.state._value].concat(n),!r[s])throw new Error("Formsy does not have the validation rule: "+s);r[s].apply(this.getCurrentValues(),n)||(i=!1)}.bind(this)),i},validateForm:function(){var t=!0,i=this.inputs,e=Object.keys(i),n=function(){e.forEach(function(e){i[e].state._isValid||(t=!1)}.bind(this)),this.setState({isValid:t}),t&&this.props.onValid(),!t&&this.props.onInvalid(),this.setState({canChange:!0})}.bind(this);e.forEach(function(t,s){var r=i[t],o=this.runValidation(r);r.setState({_isValid:o,_serverError:null},s===e.length-1?n:null)}.bind(this)),e.length||this.setState({canChange:!0})},attachToForm:function(t){this.inputs[t.props.name]=t,this.model[t.props.name]=t.state._value,this.validate(t)},detachFromForm:function(t){delete this.inputs[t.props.name],delete this.model[t.props.name]},render:function(){return n.DOM.form({onSubmit:this.submit,className:this.props.className},this.props.children)}}),e.exports||e.module||e.define&&e.define.amd||(e.Formsy=s),i.exports=s}).call(this,"undefined"!=typeof global?global:"undefined"!=typeof self?self:"undefined"!=typeof window?window:{})},{react:"react"}]},{},[1]); \ No newline at end of file diff --git a/specs/Formsy-spec.js b/specs/Formsy-spec.js index 9912e03..df2b298 100755 --- a/specs/Formsy-spec.js +++ b/specs/Formsy-spec.js @@ -249,29 +249,23 @@ describe('Formsy', function () { expect(isInvalid).toBe(false); - // Wait before adding the input - setTimeout(function () { - - - inputs.push(TestInput({ - name: 'test2', - validations: 'isEmail', - value: 'foo@bar' - })); + inputs.push(TestInput({ + name: 'test2', + validations: 'isEmail', + value: 'foo@bar' + })); - forceUpdate(function () { + forceUpdate(function () { - // Wait for next event loop, as that does the form - setTimeout(function () { - TestUtils.Simulate.submit(form.getDOMNode()); - expect(isInvalid).toBe(true); - done(); - }, 0); + // Wait for next event loop, as that does the form + setTimeout(function () { + TestUtils.Simulate.submit(form.getDOMNode()); + expect(isInvalid).toBe(true); + done(); + }, 0); - }); - - }, 10); + }); }); @@ -349,23 +343,19 @@ describe('Formsy', function () { ); // Wait before adding the input - setTimeout(function () { + inputs.push(TestInput({ + name: 'test' + })); - inputs.push(TestInput({ - name: 'test' - })); + forceUpdate(function () { - forceUpdate(function () { + // Wait for next event loop, as that does the form + setTimeout(function () { + expect(hasChanged).toHaveBeenCalled(); + done(); + }, 0); - // Wait for next event loop, as that does the form - setTimeout(function () { - expect(hasChanged).toHaveBeenCalled(); - done(); - }, 0); - - }); - - }, 10); + }); }); diff --git a/src/main.js b/src/main.js index ad3f965..0bd844e 100644 --- a/src/main.js +++ b/src/main.js @@ -161,6 +161,9 @@ Formsy.Mixin = { }, componentDidUpdate: function(prevProps, prevState) { + + // If the input is untouched and something outside changes the value + // update the FORM model by re-attaching to the form if (this.state._isPristine) { if (this.props.value !== prevProps.value && this.state._value === prevProps.value) { this.state._value = this.props.value || '';