ro-form is a minimalistic Form utility library for React that helps you build forms faster. It handles the data flow in forms and does not feature any UI. As examples below show, it is extremely simple to use. It can be used both with regular stateful React
components and redux
;
yarn:
yarn add ro-form
npm:
npm install --save ro-form
import React, { Component } from 'react';
import { Form, FormControl } from 'ro-form';
class LoginScreen extends Component {
constructor(props) {
super(props);
this.state = {
data: {
username: '',
password: ''
}
};
}
changeData = newData => this.setState({ data: newData });
submit = () => {
const { data } = this.state;
console.log('Perform login with data:', data);
}
render() {
const { data } = this.state;
return (
<Form value={data} onChange={this.changeData} onSubmit={this.submit}>
<div>
Username:
<FormControl
control={TextInput}
path={'username'}
/>
</div>
<div>
Password:
<FormControl
control={TextInput}
controlProps={{ type: 'password' }}
path={'password'}
/>
</div>
<button
type={'submit'}
disabled={data.password.length === 0 && data.username.length === 0}
>
Login
</button>
</Form>
);
}
}
const TextInput = ({ onChange, type, value }) => (
<input type={type || 'text'} onChange={e => onChange(e.target.value)} value={value}>
);
import React, { Component } from 'react';
import { Form, FormControl } from 'ro-form';
import { connect } from 'react-redux';
const ReduxLoginScreen = ({ formData, changeFormData, onSubmit }) => (
<Form value={data} onChange={changeFormData} onSubmit={onSubmit}>
<div>
Username:
<FormControl
control={TextInput}
path={'username'}
/>
</div>
<div>
Password:
<FormControl
control={TextInput}
controlProps={{ type: 'password' }}
path={'password'}
/>
</div>
<button
type={'submit'}
disabled={data.password.length === 0 && data.username.length === 0}
>
Login
</button>
</Form>
);
const TextInput = ({ onChange, type, value }) => (
<input type={type || 'text'} onChange={e => onChange(e.target.value)} value={value}>
);
const mapStateToProps = state => {
formData: state.myFormData,
};
const mapDispatchToProps = dispatch => {
changeFormData: (newFormData) => dispatch(changeMyFormDataAction(newFormData)),
onSubmit: () => dispatch(performLoginAction()),
};
ReduxLoginScreen = connect(mapStateToProps, mapDispatchToProps)(ReduxLoginScreen);
ro-form is a really simple library that contains only two components: Form and FormControl.
Form component's purpose is to bind your component's data to underlying FormControls.
Props:
- Required:
value
(Object): Object that contains data that needs to be provided to underlying FormControls - Optional:
onChange(newValue)
(Function): Function that is called every time data in any of the FormControls changes. It receives one argument, which is the newvalue
object. - Optional:
onFieldChange(fieldName, newFieldValue)
(Function): Function that is called every time data in any of the FormControls changes. The difference withonChange
prop is that this one is called with just single field's new value. - Optional:
onSubmit
(Function): It is called once Form is submitted (eg. when user pressesReturn
key on keyboard while one of form's inputs is focused). If you don't want to allow this behaviour, you do not have to provide this parameter.
FormControl handles updates of parts of the value
object provided to parent Form.
Props:
- Required:
control
(Component): Any component that hasvalue
(Any) prop andonChange
(Function) prop. Control'sonChange
should be passed one argument, which is the newvalue
. - Optional:
controlProps
(Object): Any additional props that should be provided tocontrol
.
Besides control, it requires either:
path
(String): Name of component's data part in the Form'svalue
object. FormControl will then automatically providevalue
andonChange
to your input control.
Or:
value
(Any): Manually provide value to underlying inputonChange(newValue)
(Function): Manually handle input data changes
These two examples are identical:
<Form value={data} onChange={this.changeData}>
{/* This is the same */}
<FormControl
control={TextInput}
path={'username'}
/>
{/* as this */}
<FormControl
control={TextInput}
value={data.username}
onChange={newUsername => this.changeData({ data: { ...data, username: newUsername } })}
/>
</Form>
Any component that receives onChange
and value
props can be used as control
in FormControl
:
const TextInput = ({ onChange, type, value }) => (
<input type={type || 'text'} onChange={e => onChange(e.target.value)} value={value}>
);
<FormControl
control={TextInput}
path={'some_field'}
/>
const BabyNamePicker = ({ onChange, value, names }) => (
<ul>
{names.map(name => (
<li
key={name}
onClick={() => onChange(name)}
className={value === name ? 'active' : 'inactive'}
>
{name}
</li>
))}
</ul>
);
<FormControl
control={BabyNamePicker}
controlProps={{ names: ['Edwin', 'Stephanie', 'Angela', 'Robert'] }}
path={'name'}
/>