Forms are central in modern web-apps, react-distributed-forms let you write them in the easiest way. HTML is cleaner, changes control is centralized.
- No CSS inside, you can use yours!
- 2-ways data-binding included
- You can granularly customize behaviors with context
- You can put Forms in some files and fields in other files
- ...without pass props or wiring them.
- You can simply replace your HTML components
- It just works!
This 20 rows are a Form with data-binding with the state of the Component, that save changes when click on Submit and when user stop editing fields.
import React from "react";
import { Form, Input, Selext, Textarea, Button } from "react-distributed-forms";
class Story extends React.Component {
  state = {};
  saveData() {
    const formData = this.state; // send formData to backend
  }
  render() {
    return (
      <Form
        binding={this}
        onFieldDidChanged={this.saveData}
        onSubmit={this.saveData}
      >
        <Input name="author" />
        <Input name="private" type="checkbox" />
        <Textarea name="story" />
      </Form>
    );
  }
}Live Demo of this example: https://codesandbox.io/s/k0n810zymo;
Add react-distributed-forms to your project.
npm i react-distributed-forms --saveimport { Form, Input, Selext, Textarea, Button } from "react-distributed-forms";Basic <Input>
<Input />
// <input type="text"></input><Input> with values
<Form values={{ username: "Hero2018" }}>
  <Input name="username" />
</Form>
// <input type="text" name="username" value="Hero2018"></input><Input> with onChange listener
<Form onFieldChange={({ name, value }) => {}}>
  <Input name="username" />
</Form>
// <input type="text" name="username"></input>Get notified when user remove focus from field, if the field value is changed
<Form onFieldDidChanged={({ name, value }) => {}}>
  <Input name="username" />
  <Select name="genre">
    <option value="m">Male</option>
    <option value="f">Female</option>
  </Select>
</Form>
// <input type="text" name="username"></input>
// <select name="genre">
//   <option value="m">Male</option>
//   <option value="f">Female</option>
// </select>With react-distributed-forms you can pass a React Component to <Form>, this will be used to write every fields'change into the component state.
This is the simplest scenario, when you bind the state of the component:
class SomeComponent extends React.Component {
  state = {
    first_name: "George"
  };
  render() {
    return (
      <Form binding={this}>
        <Input name="first_name" />
      </Form>
    );
  }
}You can also bind Form values to a different key in the state object:
class SomeComponent extends React.Component {
  state = {
    formdata: {
      first_name: "George"
    }
  };
  render() {
    return (
      <Form binding={[this, "formdata"]}>
        <Input name="first_name" />
      </Form>
    );
  }
}You can have a <Form> inside another <Form>, in this case the Forms will receive changes from all the fields contained in the Form itself or in a Form down in the hierarchy.
In the example Form with id="parent" will receive changes from all the fields inside: username, genre, and also "privacy". The Form with id="child" will receive only the changes of the field "privacy".
<Form id="parent" onFieldChange={({ name, value }) => {}}>
  <Input name="username" />
  <Select name="genre">
    <option value="m">Male</option>
    <option value="f">Female</option>
  </Select>
  <Form id="child" onFieldChange={({ name, value }) => {}}>
    <label>Privacy Agreement</label>
    <Input type="checkbox" name="privacy" />
  </Form>
</Form>If you don't want that a Form propagate changes to upper levels, use the prop "stopPropagation" prop. In the example the Form with id="parent" will receive changes only from "username" and "genre" because the nested Form has "stopPropagation" set
<Form id="parent" onFieldChange={({ name, value }) => {}}>
  <Input name="username" />
  <Select name="genre">
    <option value="m">Male</option>
    <option value="f">Female</option>
  </Select>
  <Form onFieldChange={({ name, value }) => {}} stopPropagation>
    <label>Privacy Agreement</label>
    <Input type="checkbox" name="privacy" />
  </Form>
</Form>react-distributed-forms is built on top of react-distributed-context that implements React context API.
From React Context page:
Context provides a way to pass data through the component tree without having to pass props down manually at every level.
This means that even if the Fields or the Nested Form you put inside a Form is elsewhere in your code, you don't have to wire it passing props, or building it in the same component.
You just create an <Input>, or <Select>, or <Textarea>, or <Button>, or a <Form> itself, wherever you need in your code, and if it has a <Form>as an ancestor in the tree, it will start to talk with him.
Demo:
Example:
Animal.js
const Animal = () => (
  <Form onFieldChange={({ name, value }) => {}}>
    <Select name="type">
      <option>Cat</option>
      <option>Fish</option>
      <option>Bird</option>
    </Select>
    {/* render the right animal component */}
  </Form>
);Cat.js
// you don't need a <Form>,
// because you will render <Cat>
// inside a component that already has a <Form>
// and it will receive this component fields changes
const Cat = () => (
  <div className="Cat">
    <Input type="number" name="number_of_legs" />
    <label>
      {`Is registered by law to cats'registry?`}
      <Input type="checkbox" name="cat_registration" />
    </label>
  </div>
);Fish.js
// same as Cat you don't need a <Form>,
const Fish = () => (
  <div className="Fish">
    <Input type="number" name="number_of_fins" />
  </div>
);Bird.js
// same as Cat you don't need a <Form>,
// but if you want you can have a Nested Form,
// in this case the Animal <Form> will receive all the changes,
// and the Bird <Form> only the changes of "number_of_legs" and "number_of_wings"
const Bird = () => (
  <div className="Bird">
    <Form onFieldChange={({ name, value }) => {}}>
      <Input type="number" name="number_of_legs" />
      <Input type="number" name="number_of_wings" />
    </Form>
  </div>
);Triggered every time a field changes its value
- you can use it to update your state.
Triggered every time a field is focused
- you can use for changing styles of focused input
Triggered every time a field lose focused
- you can use for removing styles of focuesd input
Triggered every time a field has changed its value and user has finished editing it, for example when a <Select> changes value or an <Input type="text"> lose focus after editing
- you can use it to call your backend automatically on form editing.
Triggered every time a Button inside a from is clicked
A Key-Value object to automatically set the value of the fields, based on their "name" attributes.
A React Component to bind state with, you can pass only the component like binding={this} or if you want to bind an internal key of state you can pass an array binding={[this, 'the_state_key_to_bind']}
if set <Form stopPropagation> the forms that are upper in the hierarchy won't receive data from fields inside this Form.
With binding set to this, every change in the Form will be written to the Component state.
import React from 'react';
import { render } from "react-dom";
import { Form, Input, Selext, Textarea, Button } from 'react-distributed-forms';
// You can set a variable and check it if you want to auto-save when user finished editing a field
const AUTO_SAVE_ON_CHANGE_ENABLED = true;
class UserInfoForm extends React.Component {
  state = {
    first_name: 'John',
    last_name: 'Doe'
  };
  /**
   * onFieldDidChanged
   * called from Form's fields to indicate user has finished editing the input
   */
  onFieldDidChanged({name, value}) {
    if (AUTO_SAVE_ON_CHANGE_ENABLED) {
      this.setState({
        [name]: value
      }, () => {
        // after state's updated, send new data to your API endpoint.
        ajaxPOST(YOUR_API_ENDPOINT, this.state);
      });
    }
  },
  /**
   * onSubmit
   * called from Form's fields when form has to be submitted
   */
  onSubmit({name, value}) {
    if (name === 'submit') {
      ajaxPOST(YOUR_API_ENDPOINT, this.state);
    }
  },
  render () {
    return (
      <Form
        binding={this}
        onFieldDidChanged={this.onFieldDidChanged}
        onSubmit={this.onSubmit}
      >
        <div>
          <label for="first_name">First Name:</label>
          <Input type="text" id="first_name" name="first_name" />
        </div>
        <div>
          <label for="last_name">Last Name:</label>
          <Input type="text" id="last_name" name="last_name" />
        </div>
        <Button name="submit">Save</Button>
      </Form>
    );
  }
}
render(<UserInfoForm />, document.querySelector("#root"));

