Comparing version 2.0.0 to 2.1.0
@@ -36,7 +36,7 @@ "use strict"; | ||
function Field(name) { | ||
function Field() { | ||
var _this = this; | ||
var opts = arguments.length > 1 && arguments[1] !== undefined ? arguments[1] : {}; | ||
var form = arguments.length > 2 ? arguments[2] : undefined; | ||
var opts = arguments.length > 0 && arguments[0] !== undefined ? arguments[0] : {}; | ||
var form = arguments.length > 1 ? arguments[1] : undefined; | ||
(0, _classCallCheck2["default"])(this, Field); | ||
@@ -147,4 +147,4 @@ (0, _defineProperty2["default"])(this, "form", void 0); | ||
this.form = form; | ||
this.name = name; | ||
this.opts = opts; | ||
this.name = opts.name; | ||
this.node = opts.node; | ||
@@ -242,2 +242,2 @@ this.readOnly = opts.readOnly || false; | ||
exports["default"] = _default; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Field.ts"],"names":["Field","value","undefined","name","opts","form","_events","dispatch","validators","length","error","setError","setState","isValid","isValidating","isValidated","state","_cancelablePromise","cancel","CancelablePromise","resolve","validator","fields","Boolean","then","node","readOnly","validate","debounceValidate","validationDelay","modifyValue","isChanged","Events","_initialValue","values","addEventListener","handleBlur","removeEventListener","modifiedValue","eventName","handler","subscribe","unsubscribe"],"mappings":"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","sourcesContent":["import Events from './Events'\nimport Form from './Form'\nimport { asyncSome, debounce, CancelablePromise } from './util'\n\n\nexport type FieldOpts = {\n  node?: HTMLElement\n  value?: string\n  validate?: Array<Function>\n  readOnly?: boolean\n  validationDelay?: number\n}\n\ntype State = {\n  value: any\n  error: any\n  isChanged: boolean\n  isValidating: boolean\n  isValidated: boolean\n  isValid: boolean\n}\n\nclass Field {\n\n  form: Form\n  name: string\n  opts: FieldOpts\n  node: HTMLElement\n  validators?: Array<Function>\n  readOnly: boolean\n  debounceValidate: Function\n  state: State\n\n  private _events: Events\n  private _initialValue: any\n  private _cancelablePromise: CancelablePromise\n\n  static modifyValue(value) {\n    return value === undefined || value === null ? '' : value\n  }\n\n  constructor(name: string, opts: FieldOpts = {}, form?: Form) {\n    this.form                 = form\n    this.name                 = name\n    this.opts                 = opts\n    this.node                 = opts.node\n    this.readOnly             = opts.readOnly || false\n    this.validators           = opts.validate || []\n    this.debounceValidate     = this.opts.validationDelay ? debounce(this.validate, this.opts.validationDelay) : this.validate\n\n    this.state = {\n      value: Field.modifyValue(opts.value),\n      error: null,\n      isChanged: false,\n      isValidating: false,\n      isValidated: false,\n      isValid: true,\n    }\n\n    this._events              = new Events()\n    this._initialValue        = this.state.value\n    this._cancelablePromise   = null\n\n    // TODO how to detect this?\n    // this.hasAsyncValidators = this.validators.some((validator) => (\n    //   validator.constructor.name === 'AsyncFunction'\n    // ))\n  }\n\n  setState(values: Partial<State>) {\n    this.state = { ...this.state, ...values }\n    this._events.dispatch('state change', this.state)\n  }\n\n  setRef(node) {\n    this.node = node\n\n    if (this.node) {\n      this.node.addEventListener('blur', this.handleBlur)\n    }\n  }\n\n  unsetRef() {\n    if (this.node) {\n      this.node.removeEventListener('blur', this.handleBlur)\n    }\n\n    this.node = null\n  }\n\n  handleBlur = () => {\n    this._events.dispatch('blur')\n  }\n\n  set(value: any) {\n    const modifiedValue = Field.modifyValue(value)\n\n    if (modifiedValue !== this.state.value && !this.readOnly) {\n      this.setState({\n        value: modifiedValue,\n        isChanged: true,\n      })\n\n      this._events.dispatch('set', this.state.value)\n      this._events.dispatch('change', this.state.value) // @deprecated\n    }\n  }\n\n  unset() {\n    this.setState({\n      value: this._initialValue,\n      error: null,\n      isChanged: false,\n      isValidating: false,\n      isValidated: false,\n      isValid: true,\n    })\n\n    this._events.dispatch('unset')\n  }\n\n  validate = (): CancelablePromise => {\n    if (!this.validators || !this.validators.length) {\n      return\n    }\n\n    let error\n\n    this._events.dispatch('start validate')\n\n    const setError = (error: string) => {\n      this.setState({\n        error,\n        isValid: !error,\n        isValidating: false,\n        isValidated: true,\n      })\n\n      this._events.dispatch('validate', this.state.error)\n    }\n\n    this.setState({\n      isValidating: true,\n    })\n\n    if (this._cancelablePromise) {\n      this._cancelablePromise.cancel()\n    }\n\n    this._cancelablePromise = new CancelablePromise(async (resolve) => {\n      await asyncSome(this.validators, async (validator) => {\n        error = await validator(this.state.value, this.form && this.form.fields) // error here is String error message\n        return Boolean(error)\n      })\n      resolve()\n    })\n\n    return this._cancelablePromise.then(() => {\n      setError(error)\n\n      return error\n    })\n  }\n\n  on(eventName: string, handler: Function) {\n    this._events.subscribe(eventName, handler)\n  }\n\n  off(eventName: string, handler: Function) {\n    this._events.unsubscribe(eventName, handler)\n  }\n}\n\n\nexport default Field\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Field.ts"],"names":["Field","value","undefined","opts","form","_events","dispatch","validators","length","error","setError","setState","isValid","isValidating","isValidated","state","_cancelablePromise","cancel","CancelablePromise","resolve","validator","fields","Boolean","then","name","node","readOnly","validate","debounceValidate","validationDelay","modifyValue","isChanged","Events","_initialValue","values","addEventListener","handleBlur","removeEventListener","modifiedValue","eventName","handler","subscribe","unsubscribe"],"mappings":"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","sourcesContent":["import Events from './Events'\nimport Form from './Form'\nimport { asyncSome, debounce, CancelablePromise } from './util'\n\n\nexport type FieldOpts = {\n  name?: string\n  node?: HTMLElement\n  value?: string\n  validate?: Array<Function>\n  readOnly?: boolean\n  validationDelay?: number\n}\n\ntype State = {\n  value: any\n  error: any\n  isChanged: boolean\n  isValidating: boolean\n  isValidated: boolean\n  isValid: boolean\n}\n\nclass Field {\n\n  form: Form\n  name?: string\n  opts: FieldOpts\n  node: HTMLElement\n  validators?: Array<Function>\n  readOnly: boolean\n  debounceValidate: Function\n  state: State\n\n  private _events: Events\n  private _initialValue: any\n  private _cancelablePromise: CancelablePromise\n\n  static modifyValue(value) {\n    return value === undefined || value === null ? '' : value\n  }\n\n  constructor(opts: FieldOpts = {}, form?: Form) {\n    this.form                 = form\n    this.opts                 = opts\n    this.name                 = opts.name\n    this.node                 = opts.node\n    this.readOnly             = opts.readOnly || false\n    this.validators           = opts.validate || []\n    this.debounceValidate     = this.opts.validationDelay ? debounce(this.validate, this.opts.validationDelay) : this.validate\n\n    this.state = {\n      value: Field.modifyValue(opts.value),\n      error: null,\n      isChanged: false,\n      isValidating: false,\n      isValidated: false,\n      isValid: true,\n    }\n\n    this._events              = new Events()\n    this._initialValue        = this.state.value\n    this._cancelablePromise   = null\n\n    // TODO how to detect this?\n    // this.hasAsyncValidators = this.validators.some((validator) => (\n    //   validator.constructor.name === 'AsyncFunction'\n    // ))\n  }\n\n  setState(values: Partial<State>) {\n    this.state = { ...this.state, ...values }\n    this._events.dispatch('state change', this.state)\n  }\n\n  setRef(node) {\n    this.node = node\n\n    if (this.node) {\n      this.node.addEventListener('blur', this.handleBlur)\n    }\n  }\n\n  unsetRef() {\n    if (this.node) {\n      this.node.removeEventListener('blur', this.handleBlur)\n    }\n\n    this.node = null\n  }\n\n  handleBlur = () => {\n    this._events.dispatch('blur')\n  }\n\n  set(value: any) {\n    const modifiedValue = Field.modifyValue(value)\n\n    if (modifiedValue !== this.state.value && !this.readOnly) {\n      this.setState({\n        value: modifiedValue,\n        isChanged: true,\n      })\n\n      this._events.dispatch('set', this.state.value)\n      this._events.dispatch('change', this.state.value) // @deprecated\n    }\n  }\n\n  unset() {\n    this.setState({\n      value: this._initialValue,\n      error: null,\n      isChanged: false,\n      isValidating: false,\n      isValidated: false,\n      isValid: true,\n    })\n\n    this._events.dispatch('unset')\n  }\n\n  validate = (): CancelablePromise => {\n    if (!this.validators || !this.validators.length) {\n      return\n    }\n\n    let error\n\n    this._events.dispatch('start validate')\n\n    const setError = (error: string) => {\n      this.setState({\n        error,\n        isValid: !error,\n        isValidating: false,\n        isValidated: true,\n      })\n\n      this._events.dispatch('validate', this.state.error)\n    }\n\n    this.setState({\n      isValidating: true,\n    })\n\n    if (this._cancelablePromise) {\n      this._cancelablePromise.cancel()\n    }\n\n    this._cancelablePromise = new CancelablePromise(async (resolve) => {\n      await asyncSome(this.validators, async (validator) => {\n        error = await validator(this.state.value, this.form && this.form.fields) // error here is String error message\n        return Boolean(error)\n      })\n      resolve()\n    })\n\n    return this._cancelablePromise.then(() => {\n      setError(error)\n\n      return error\n    })\n  }\n\n  on(eventName: string, handler: Function) {\n    this._events.subscribe(eventName, handler)\n  }\n\n  off(eventName: string, handler: Function) {\n    this._events.unsubscribe(eventName, handler)\n  }\n}\n\n\nexport default Field\n"]} |
@@ -67,2 +67,3 @@ "use strict"; | ||
} : fieldOpts; | ||
fieldOpts.name = fieldName; | ||
@@ -73,3 +74,3 @@ if (typeof initialValue !== 'undefined') { | ||
var field = new _Field["default"](fieldName, fieldOpts, _this); | ||
var field = new _Field["default"](fieldOpts, _this); | ||
_this.fields[fieldName] = field; | ||
@@ -244,2 +245,2 @@ field.on('change', function () { | ||
exports["default"] = _default; | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Form.ts"],"names":["defaultOptions","initialValues","Form","opts","name","fields","state","isValid","isChanged","isValidating","isSubmitting","isSubmitted","_events","Events","_setupFields","Object","keys","forEach","fieldName","initialValue","fieldOpts","Array","isArray","validate","value","field","Field","on","dispatch","values","set","setState","unset","errors","error","promises","map","Promise","all","every","getValues","reject","getErrors","eventName","handler","subscribe","unsubscribe"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAAA;;AACA;;AAiBA,IAAMA,cAAc,GAAG;AACrBC,EAAAA,aAAa,EAAE;AADM,CAAvB;;IAIMC,I;;;AAQJ,gBAAYC,IAAZ,EAA4B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAC1B,SAAKC,IAAL,GAAcD,IAAI,CAACC,IAAnB;AACA,SAAKD,IAAL,iCAAmBH,cAAnB,MAAsCG,IAAtC;AACA,SAAKE,MAAL,GAAc,EAAd;AAEA,SAAKC,KAAL,GAAa;AACXC,MAAAA,OAAO,EAAE,IADE;AAEXC,MAAAA,SAAS,EAAE,KAFA;AAEQ;AACnBC,MAAAA,YAAY,EAAE,KAHH;AAIXC,MAAAA,YAAY,EAAE,KAJH;AAKXC,MAAAA,WAAW,EAAE;AALF,KAAb;AAQA,SAAKC,OAAL,GAAe,IAAIC,kBAAJ,EAAf;;AAEA,SAAKC,YAAL;AACD;;;;mCAEsB;AAAA;;AACrBC,MAAAA,MAAM,CAACC,IAAP,CAAY,KAAKb,IAAL,CAAUE,MAAtB,EAA8BY,OAA9B,CAAsC,UAACC,SAAD,EAAe;AACnD,YAAMC,YAAY,GAAI,KAAI,CAAChB,IAAL,CAAUF,aAAV,IAA2B,KAAI,CAACE,IAAL,CAAUF,aAAV,CAAwBiB,SAAxB,CAAjD;AACA,YAAIE,SAAS,GAAS,KAAI,CAACjB,IAAL,CAAUE,MAAV,CAAiBa,SAAjB,CAAtB;AAEAE,QAAAA,SAAS,GAAGC,KAAK,CAACC,OAAN,CAAcF,SAAd,IAA2B;AAAEG,UAAAA,QAAQ,EAAEH;AAAZ,SAA3B,GAAqDA,SAAjE;;AAEA,YAAI,OAAOD,YAAP,KAAwB,WAA5B,EAAyC;AACvCC,UAAAA,SAAS,CAACI,KAAV,GAAkBL,YAAlB;AACD;;AAED,YAAMM,KAAK,GAAG,IAAIC,iBAAJ,CAAUR,SAAV,EAAqBE,SAArB,EAAgC,KAAhC,CAAd;AAEA,QAAA,KAAI,CAACf,MAAL,CAAYa,SAAZ,IAAyBO,KAAzB;AAEAA,QAAAA,KAAK,CAACE,EAAN,CAAS,QAAT,EAAmB,YAAM;AACvB,UAAA,KAAI,CAACf,OAAL,CAAagB,QAAb,CAAsB,QAAtB,EAAgCH,KAAhC;AACD,SAFD;AAIAA,QAAAA,KAAK,CAACE,EAAN,CAAS,MAAT,EAAiB,YAAM;AACrB,UAAA,KAAI,CAACf,OAAL,CAAagB,QAAb,CAAsB,MAAtB,EAA8BH,KAA9B;AACD,SAFD;AAGD,OArBD;AAsBD;;;6BAEQI,M,EAAwB;AAC/B,WAAKvB,KAAL,iCAAkB,KAAKA,KAAvB,MAAiCuB,MAAjC;;AACA,WAAKjB,OAAL,CAAagB,QAAb,CAAsB,cAAtB,EAAsC,KAAKtB,KAA3C;AACD;;;8BAESuB,M,EAAgB;AAAA;;AACxB;AACAd,MAAAA,MAAM,CAACC,IAAP,CAAYa,MAAZ,EAAoBZ,OAApB,CAA4B,UAACC,SAAD,EAAe;AACzC,YAAMO,KAAY,GAAG,MAAI,CAACpB,MAAL,CAAYa,SAAZ,CAArB;;AAEA,YAAIO,KAAJ,EAAW;AACTA,UAAAA,KAAK,CAACK,GAAN,CAAUD,MAAM,CAACX,SAAD,CAAhB;AACD;AACF,OAND;AAOD;;;gCAEmB;AAAA;;AAClB,UAAMW,MAAM,GAAG,EAAf;AAEAd,MAAAA,MAAM,CAACC,IAAP,CAAY,KAAKX,MAAjB,EAAyBY,OAAzB,CAAiC,UAACC,SAAD,EAAe;AAC9CW,QAAAA,MAAM,CAACX,SAAD,CAAN,GAAoB,MAAI,CAACb,MAAL,CAAYa,SAAZ,EAAuBZ,KAAvB,CAA6BkB,KAAjD;AACD,OAFD;AAIA,aAAOK,MAAP;AACD;;;kCAEa;AAAA;;AACZ,WAAKE,QAAL,CAAc;AACZvB,QAAAA,SAAS,EAAE,KADC;AAEZD,QAAAA,OAAO,EAAE;AAFG,OAAd;AAKAQ,MAAAA,MAAM,CAACC,IAAP,CAAY,KAAKX,MAAjB,EAAyBY,OAAzB,CAAiC,UAACC,SAAD,EAAe;AAC9C,QAAA,MAAI,CAACb,MAAL,CAAYa,SAAZ,EAAuBc,KAAvB;AACD,OAFD;AAGD;;;gCAEmB;AAAA;;AAClB,UAAMC,MAAM,GAAG,EAAf;AAEAlB,MAAAA,MAAM,CAACC,IAAP,CAAY,KAAKX,MAAjB,EAAyBY,OAAzB,CAAiC,UAACC,SAAD,EAAe;AAC9Ce,QAAAA,MAAM,CAACf,SAAD,CAAN,GAAoB,MAAI,CAACb,MAAL,CAAYa,SAAZ,EAAuBZ,KAAvB,CAA6B4B,KAAjD;AACD,OAFD;AAIA,aAAOD,MAAP;AACD;;;;;;;;;;;;;;AAGOE,gBAAAA,Q,GAAYpB,MAAM,CAACC,IAAP,CAAY,KAAKX,MAAjB,EAAyB+B,GAAzB,CAA6B,UAAClB,SAAD;AAAA,yBAAe,MAAI,CAACb,MAAL,CAAYa,SAAZ,EAAuBK,QAAvB,EAAf;AAAA,iBAA7B,C;;uBACMc,OAAO,CAACC,GAAR,CAAYH,QAAZ,C;;;AAAlBF,gBAAAA,M;AACA1B,gBAAAA,O,GAAY0B,MAAM,CAACM,KAAP,CAAa,UAACL,KAAD;AAAA,yBAAW,CAACA,KAAZ;AAAA,iBAAb,C;AAElB,qBAAKH,QAAL,CAAc;AAAExB,kBAAAA,OAAO,EAAPA;AAAF,iBAAd;iDAEOA,O;;;;;;;;;;;;;;;;;;;;;;;;;;;AAIP;AACA;AACA;AACA;AACMsB,gBAAAA,M,GAAS,KAAKW,SAAL,E,EAEf;;;uBACM,KAAKjB,QAAL,E;;;oBAED,KAAKjB,KAAL,CAAWC,O;;;;;kDACP8B,OAAO,CAACI,MAAR,CAAe,KAAKC,SAAL,EAAf,C;;;kDAGFb,M;;;;;;;;;;;;;;;;;;uBAGNc,S,EAAmBC,O,EAAmB;AACvC,WAAKhC,OAAL,CAAaiC,SAAb,CAAuBF,SAAvB,EAAkCC,OAAlC;AACD;;;wBAEGD,S,EAAmBC,O,EAAmB;AACxC,WAAKhC,OAAL,CAAakC,WAAb,CAAyBH,SAAzB,EAAoCC,OAApC;AACD;;;;;eAIY1C,I","sourcesContent":["import Events from './Events'\nimport Field from './Field'\n\n\nexport type FormOpts = {\n  name?: string\n  fields: object\n  initialValues?: object\n}\n\ntype State = {\n  isValid: boolean\n  isChanged: boolean\n  isValidating: boolean\n  isSubmitting: boolean\n  isSubmitted: boolean\n}\n\nconst defaultOptions = {\n  initialValues: {},\n}\n\nclass Form {\n\n  private _events: Events\n  name: string\n  opts: FormOpts\n  fields: object\n  state: State\n\n  constructor(opts: FormOpts) {\n    this.name   = opts.name\n    this.opts   = { ...defaultOptions, ...opts }\n    this.fields = {}\n\n    this.state = {\n      isValid: true,\n      isChanged: false,  // TODO connect to Field\n      isValidating: false,\n      isSubmitting: false,\n      isSubmitted: false,\n    }\n\n    this._events = new Events()\n\n    this._setupFields()\n  }\n\n  private _setupFields() {\n    Object.keys(this.opts.fields).forEach((fieldName) => {\n      const initialValue  = this.opts.initialValues && this.opts.initialValues[fieldName]\n      let fieldOpts       = this.opts.fields[fieldName]\n\n      fieldOpts = Array.isArray(fieldOpts) ? { validate: fieldOpts } : fieldOpts\n\n      if (typeof initialValue !== 'undefined') {\n        fieldOpts.value = initialValue\n      }\n\n      const field = new Field(fieldName, fieldOpts, this)\n\n      this.fields[fieldName] = field\n\n      field.on('change', () => {\n        this._events.dispatch('change', field)\n      })\n\n      field.on('blur', () => {\n        this._events.dispatch('blur', field)\n      })\n    })\n  }\n\n  setState(values: Partial<State>) {\n    this.state = { ...this.state, ...values }\n    this._events.dispatch('state change', this.state)\n  }\n\n  setValues(values: object) {\n    // TODO should we mark form as changed and validate it?\n    Object.keys(values).forEach((fieldName) => {\n      const field: Field = this.fields[fieldName]\n\n      if (field) {\n        field.set(values[fieldName])\n      }\n    })\n  }\n\n  getValues(): object {\n    const values = {}\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      values[fieldName] = this.fields[fieldName].state.value\n    })\n\n    return values\n  }\n\n  unsetValues() {\n    this.setState({\n      isChanged: false,\n      isValid: true,\n    })\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      this.fields[fieldName].unset()\n    })\n  }\n\n  getErrors(): object {\n    const errors = {}\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      errors[fieldName] = this.fields[fieldName].state.error\n    })\n\n    return errors\n  }\n\n  async validate(): Promise<boolean> {\n    const promises  = Object.keys(this.fields).map((fieldName) => this.fields[fieldName].validate())\n    const errors    = await Promise.all(promises)\n    const isValid   = errors.every((error) => !error)\n\n    this.setState({ isValid })\n\n    return isValid\n  }\n\n  async submit(): Promise<object> {\n    // validation takes values on start but user may change form values after this moment and before the validation end\n    // so if getValues is called after validate() - values may be different in validation and result of sumbit\n    // so we should get form values before async validation\n    // TODO lock fields on validations start\n    const values = this.getValues()\n\n    // TODO don't validate if all fields are changed and valid\n    await this.validate()\n\n    if (!this.state.isValid) {\n      return Promise.reject(this.getErrors())\n    }\n\n    return values\n  }\n\n  on(eventName: string, handler: Function) {\n    this._events.subscribe(eventName, handler)\n  }\n\n  off(eventName: string, handler: Function) {\n    this._events.unsubscribe(eventName, handler)\n  }\n}\n\n\nexport default Form\n"]} | ||
//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"sources":["../src/Form.ts"],"names":["defaultOptions","initialValues","Form","opts","name","fields","state","isValid","isChanged","isValidating","isSubmitting","isSubmitted","_events","Events","_setupFields","Object","keys","forEach","fieldName","initialValue","fieldOpts","Array","isArray","validate","value","field","Field","on","dispatch","values","set","setState","unset","errors","error","promises","map","Promise","all","every","getValues","reject","getErrors","eventName","handler","subscribe","unsubscribe"],"mappings":"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","sourcesContent":["import Events from './Events'\nimport Field from './Field'\n\n\nexport type FormOpts = {\n  name?: string\n  fields: object\n  initialValues?: object\n}\n\ntype State = {\n  isValid: boolean\n  isChanged: boolean\n  isValidating: boolean\n  isSubmitting: boolean\n  isSubmitted: boolean\n}\n\nconst defaultOptions = {\n  initialValues: {},\n}\n\nclass Form {\n\n  private _events: Events\n  name: string\n  opts: FormOpts\n  fields: object\n  state: State\n\n  constructor(opts: FormOpts) {\n    this.name   = opts.name\n    this.opts   = { ...defaultOptions, ...opts }\n    this.fields = {}\n\n    this.state = {\n      isValid: true,\n      isChanged: false,  // TODO connect to Field\n      isValidating: false,\n      isSubmitting: false,\n      isSubmitted: false,\n    }\n\n    this._events = new Events()\n\n    this._setupFields()\n  }\n\n  private _setupFields() {\n    Object.keys(this.opts.fields).forEach((fieldName) => {\n      const initialValue  = this.opts.initialValues && this.opts.initialValues[fieldName]\n      let fieldOpts       = this.opts.fields[fieldName]\n\n      fieldOpts = Array.isArray(fieldOpts) ? { validate: fieldOpts } : fieldOpts\n      fieldOpts.name = fieldName\n\n      if (typeof initialValue !== 'undefined') {\n        fieldOpts.value = initialValue\n      }\n\n      const field = new Field(fieldOpts, this)\n\n      this.fields[fieldName] = field\n\n      field.on('change', () => {\n        this._events.dispatch('change', field)\n      })\n\n      field.on('blur', () => {\n        this._events.dispatch('blur', field)\n      })\n    })\n  }\n\n  setState(values: Partial<State>) {\n    this.state = { ...this.state, ...values }\n    this._events.dispatch('state change', this.state)\n  }\n\n  setValues(values: object) {\n    // TODO should we mark form as changed and validate it?\n    Object.keys(values).forEach((fieldName) => {\n      const field: Field = this.fields[fieldName]\n\n      if (field) {\n        field.set(values[fieldName])\n      }\n    })\n  }\n\n  getValues(): object {\n    const values = {}\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      values[fieldName] = this.fields[fieldName].state.value\n    })\n\n    return values\n  }\n\n  unsetValues() {\n    this.setState({\n      isChanged: false,\n      isValid: true,\n    })\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      this.fields[fieldName].unset()\n    })\n  }\n\n  getErrors(): object {\n    const errors = {}\n\n    Object.keys(this.fields).forEach((fieldName) => {\n      errors[fieldName] = this.fields[fieldName].state.error\n    })\n\n    return errors\n  }\n\n  async validate(): Promise<boolean> {\n    const promises  = Object.keys(this.fields).map((fieldName) => this.fields[fieldName].validate())\n    const errors    = await Promise.all(promises)\n    const isValid   = errors.every((error) => !error)\n\n    this.setState({ isValid })\n\n    return isValid\n  }\n\n  async submit(): Promise<object> {\n    // validation takes values on start but user may change form values after this moment and before the validation end\n    // so if getValues is called after validate() - values may be different in validation and result of sumbit\n    // so we should get form values before async validation\n    // TODO lock fields on validations start\n    const values = this.getValues()\n\n    // TODO don't validate if all fields are changed and valid\n    await this.validate()\n\n    if (!this.state.isValid) {\n      return Promise.reject(this.getErrors())\n    }\n\n    return values\n  }\n\n  on(eventName: string, handler: Function) {\n    this._events.subscribe(eventName, handler)\n  }\n\n  off(eventName: string, handler: Function) {\n    this._events.unsubscribe(eventName, handler)\n  }\n}\n\n\nexport default Form\n"]} |
{ | ||
"name": "formular", | ||
"version": "2.0.0", | ||
"version": "2.1.0", | ||
"author": "Pavel Ivanov", | ||
@@ -5,0 +5,0 @@ "description": "Build forms in React. Easy-Peasy!", |
325
README.md
@@ -1,308 +0,51 @@ | ||
# Form | ||
# Formular | ||
## Usage | ||
Easy way to work with forms in React. Using React Hooks 😏 | ||
### Form initialization | ||
```js | ||
import Form from 'formular' | ||
## Quick start | ||
const form = new Form({ | ||
fields: { | ||
email: { | ||
validate: [ required, email ], | ||
}, | ||
}, | ||
initialValues: { | ||
email: 'test-email@test.com', | ||
}, | ||
}) | ||
#### Installation | ||
``` | ||
npm install | ||
``` | ||
### In React | ||
#### Usage | ||
```jsx | ||
import { Input } from 'formular/react/tags' | ||
```jsx harmony | ||
const App = () => { | ||
const form = useMemo(() => ( | ||
new Form({ | ||
fields: { | ||
email: [ required ], | ||
password: [ required ], | ||
}, | ||
}) | ||
), []) | ||
const handleSubmit = () => form.submit() | ||
const Form = () => ( | ||
<form> | ||
return ( | ||
<Input field={form.fields.email} /> | ||
<button onClick={handleSubmit)>Submit</button> | ||
</form> | ||
) | ||
) | ||
} | ||
``` | ||
## Form | ||
Or if you need only one field you can just do | ||
### Properties | ||
```jsx harmony | ||
const App = () => { | ||
const field = useMemo(() => ( | ||
new Field({ | ||
validate: [ required ], | ||
}) | ||
), []) | ||
`opts: object` | ||
Your form options. | ||
`fields: object` | ||
Your form fields. | ||
`isValid: boolean` | ||
Keeps form validation state. Default is `true`. | ||
`isChanged: boolean` | ||
Tells if form has been changed. Default is `false`. | ||
### Methods | ||
`setInitialValues: (values: object) => void` | ||
Calls `field.setInitialValue(value)` for every form field. | ||
`setValues: (values: object) => void` | ||
Calls `field.set(value)` for every form field. | ||
`unsetValues: () => void` | ||
Calls `field.unset()` for every form field. | ||
`getValues: () => object` | ||
Returns current form field values. | ||
`getErrors: () => object` | ||
Returns current form field errors. | ||
`validate: async () => boolean` | ||
Fires form validation and returns resulted validation state. | ||
`submit: async () => object` | ||
Calls `validate` method and then if form passed the validation it returns `getValues` result. If not, it returns `getErrors` result. | ||
`on: (eventName: string, handler: Function) => void` | ||
Subscribes `handler` to the event with certain `eventName`. | ||
`off: (eventName: string, handler: Function) => void` | ||
Unsubscribes `handler` from the event with certain `eventName`. | ||
## Field | ||
### Usage | ||
```js | ||
const options = { | ||
validate: [ required, email ], | ||
value: 'test-email@test.com', | ||
return ( | ||
<Input field={field} /> | ||
) | ||
} | ||
const emailField = new Field('email', options) | ||
const passwordField = new Field('password', [ required ]) | ||
``` | ||
Your field options can be either object or array. If it's an array, it will be used as `options.validate` param. | ||
## Examples | ||
`options.validate: array` | ||
An array of your field validators. | ||
`options.value?: any` | ||
Your field initial value. | ||
`options.hasAsyncValidators?: boolean` | ||
Determines if your field has async validators. Default is `false`. | ||
### Properties | ||
`node: Node` | ||
Field's html node ref. | ||
`name: string` | ||
Field's name. | ||
`validators: array` | ||
Array of field's validators. | ||
`value: any` | ||
Field's value. | ||
`initialValue: any` | ||
Field's initial value. | ||
`error: string` | ||
Field's error. | ||
`isChanged: boolean` | ||
Determines whether field was changed. Default is `false`. | ||
`isValid: boolean` | ||
Determines whether field passed validation. Default is `true`. | ||
`cancelablePromise: Promise` | ||
TODO | ||
`hasAsyncValidators: boolean` | ||
Determines whether field has async validators. Default is `false`. | ||
`debounceValidate: Function` | ||
TODO | ||
### Methods | ||
`setInitialValue: (value: any) => void` | ||
Sets field's initial value. | ||
`setRef: (node: Node) => void` | ||
Sets field's node ref. | ||
`unsetRef: (node: Node) => void` | ||
Unsets field's node ref. | ||
`setInitialValue: (value: any) => void` | ||
Sets field's initial value. | ||
`handleBlur: () => void` | ||
TODO | ||
`set: (value: any) => void` | ||
Sets field's new value. New value is set when it is not equal to current value. | ||
`unset: () => void` | ||
Unsets field. | ||
`validate: async () => string` | ||
Fires field validation and returns validation error. | ||
`on: (eventName: string, handler: Function) => void` | ||
Subscribes `handler` to the event with certain `eventName`. | ||
`off: (eventName: string, handler: Function) => void` | ||
Unsubscribes `handler` from the event with certain `eventName`. | ||
## FormGroup | ||
### Usage | ||
If you are using several forms and you need to control them in one place you can combine them using `FormGroup` | ||
```js | ||
import Form, { FormGroup } from 'formular' | ||
const shippingAddressForm = new Form({ | ||
fields: { | ||
zip: [ required ], | ||
address: [ required ], | ||
city: [ required ], | ||
country: [ required ], | ||
}, | ||
}) | ||
const creditCardForm = new Form({ | ||
fields: { | ||
cardNumber: [ required ], | ||
expDate: [ required ], | ||
cvv: [ required ], | ||
}, | ||
}) | ||
const formGroup = new FormGroup({ shippingAddressForm, creditCardForm }) | ||
``` | ||
### Properties | ||
`forms: object` | ||
An object containing forms instances. | ||
### Methods | ||
`replace: (forms: object) => void` | ||
Reinitialises your form group with new `forms`. | ||
`validate: () => boolean` | ||
Calls `validate` method for each form in form group and returns validation result. If one of the forms is not valid then the result is `false`, otherwise the result is `true`. | ||
`setValues: (values: object) => void` | ||
Calls `setValues` method for each form in form group if `values['yourFormName'] exists. | ||
`getValues: () => object` | ||
Calls `getValues` for each form and returns an object containing these values. | ||
`unsetValues: () => object` | ||
Calls `unsetValues` for each form. | ||
`submit: () => object` | ||
Calls `validate` method and then if forms passed the validation returns `getValues` result. If not, it returns `getErrors` result. | ||
`on: (eventName: string, handler: Function) => void` | ||
Subscribes `handler` to the event with certain `eventName`. | ||
`off: (eventName: string, handler: Function) => void` | ||
Unsubscribes `handler` from the event with certain `eventName`. | ||
## To run examples | ||
``` | ||
npx babel-node examples/validate-form.js | ||
``` | ||
## TODOs | ||
- [x] Pass initial values in options | ||
- [x] Set initial values | ||
- [x] Validate form | ||
- [x] Validate one field | ||
- [x] Group forms to one container (group) | ||
- [x] Validate forms group | ||
- [x] Async validation | ||
- [x] Validate field on blur | ||
- [x] Validate field on each change if it has been already validated | ||
- [x] Validate field on each change if form was submitted | ||
- [ ] 'isChanged' on field, form, forms group | ||
- [ ] Get forms group state (values, isChanged, etc) | ||
- [ ] Set state to forms group | ||
- [ ] Reset form to default state, options, etc | ||
- [ ] Add option to `setValues()` to allow set values without validation each field | ||
- [ ] Checkbox and Radio support (boolean checked, value) | ||
- [ ] Memo for async validators (decorator) | ||
- [ ] Option for validation debounce timeout | ||
- [Basics](https://codesandbox.io/s/formular-basics-cke7r) |
@@ -7,2 +7,3 @@ import Events from './Events' | ||
export type FieldOpts = { | ||
name?: string | ||
node?: HTMLElement | ||
@@ -27,3 +28,3 @@ value?: string | ||
form: Form | ||
name: string | ||
name?: string | ||
opts: FieldOpts | ||
@@ -44,6 +45,6 @@ node: HTMLElement | ||
constructor(name: string, opts: FieldOpts = {}, form?: Form) { | ||
constructor(opts: FieldOpts = {}, form?: Form) { | ||
this.form = form | ||
this.name = name | ||
this.opts = opts | ||
this.name = opts.name | ||
this.node = opts.node | ||
@@ -50,0 +51,0 @@ this.readOnly = opts.readOnly || false |
@@ -55,2 +55,3 @@ import Events from './Events' | ||
fieldOpts = Array.isArray(fieldOpts) ? { validate: fieldOpts } : fieldOpts | ||
fieldOpts.name = fieldName | ||
@@ -61,3 +62,3 @@ if (typeof initialValue !== 'undefined') { | ||
const field = new Field(fieldName, fieldOpts, this) | ||
const field = new Field(fieldOpts, this) | ||
@@ -64,0 +65,0 @@ this.fields[fieldName] = field |
Major refactor
Supply chain riskPackage has recently undergone a major refactor. It may be unstable or indicate significant internal changes. Use caution when updating to versions that include significant changes.
Found 1 instance in 1 package
2728
0
261851
52