- Redux Form
- Examples
- Material Ui Form Example
Material UI Example
This is a simple demonstration of how to connect all the standard material-ui form elements to redux-form
.
For the most part, it is a matter of wrapping each form control in a <Field>
component as custom component.
For controls like SelectField
we need to simulate the onChange
manually. As props have been exposed in redux-form
you can fire onChange
manually. Read more here.
The delay between when you click "Submit" and when the alert dialog pops up is intentional, to simulate server latency.
Running this example locally
To run this example locally on your machine clone the redux-form
repository, then cd redux-form
to change to the repo directory, and run npm install
.
Then run npm run example:material-ui
or manually run the following commands:
cd ./examples/material-ui
npm install
npm start
Then open localhost:3030
in your browser to view the example running locally on your machine.
Field Renderers
Notice that we define simple functions, like renderTextField
, renderCheckbox
, and renderSelectField
to form a bridge between redux-form
and the Material UI input components. You would only need to define these in one place in your application and reuse them in each form.
For Material UI, @erikras
has published a set of wrapper components to use Material UI: redux-form-material-ui
.
How to use async validation in form:
- Emails that will fail validation:
foo@foo.com
,bar@bar.com
.
Form
Values
undefined
Code
asyncValidate.js
const sleep = ms => new Promise(resolve => setTimeout(resolve, ms))
const asyncValidate = (values/*, dispatch */) => {
return sleep(1000) // simulate server latency
.then(() => {
if ([ 'foo@foo.com', 'bar@bar.com' ].includes(values.email)) {
throw { email: 'Email already Exists' }
}
})
}
export default asyncValidate
MaterialUiForm.js
import React from 'react'
import { Field, reduxForm } from 'redux-form'
import TextField from 'material-ui/TextField'
import { RadioButton, RadioButtonGroup } from 'material-ui/RadioButton'
import Checkbox from 'material-ui/Checkbox'
import SelectField from 'material-ui/SelectField'
import MenuItem from 'material-ui/MenuItem'
import asyncValidate from './asyncValidate'
const validate = values => {
const errors = {}
const requiredFields = [ 'firstName', 'lastName', 'email', 'favoriteColor', 'notes' ]
requiredFields.forEach(field => {
if (!values[ field ]) {
errors[ field ] = 'Required'
}
})
if (values.email && !/^[A-Z0-9._%+-]+@[A-Z0-9.-]+\.[A-Z]{2,4}$/i.test(values.email)) {
errors.email = 'Invalid email address'
}
return errors
}
const renderTextField = ({ input, label, meta: { touched, error }, ...custom }) => (
<TextField hintText={label}
floatingLabelText={label}
errorText={touched && error}
{...input}
{...custom}
/>
)
const renderCheckbox = ({ input, label }) => (
<Checkbox label={label}
checked={input.value ? true : false}
onCheck={input.onChange}/>
)
const renderRadioGroup = ({ input, ...rest }) => (
<RadioButtonGroup {...input} {...rest}
valueSelected={input.value}
onChange={(event, value) => input.onChange(value)}/>
)
const renderSelectField = ({ input, label, meta: { touched, error }, children }) => (
<SelectField
floatingLabelText={label}
errorText={touched && error}
{...input}
onChange={(event, index, value) => input.onChange(value)}
children={children}/>
)
const MaterialUiForm = props => {
const { handleSubmit, pristine, reset, submitting } = props
return (
<form onSubmit={handleSubmit}>
<div>
<Field name="firstName" component={renderTextField} label="First Name"/>
</div>
<div>
<Field name="lastName" component={renderTextField} label="Last Name"/>
</div>
<div>
<Field name="email" component={renderTextField} label="Email"/>
</div>
<div>
<Field name="sex" component={renderRadioGroup}>
<RadioButton value="male" label="male"/>
<RadioButton value="female" label="female"/>
</Field>
</div>
<div>
<Field name="favoriteColor" component={renderSelectField} label="Favorite Color">
<MenuItem value={'ff0000'} primaryText="Red"/>
<MenuItem value={'00ff00'} primaryText="Green"/>
<MenuItem value={'0000ff'} primaryText="Blue"/>
</Field>
</div>
<div>
<Field name="employed" component={renderCheckbox} label="Employed"/>
</div>
<div>
<Field name="notes" component={renderTextField} label="Notes" multiLine={true} rows={2}/>
</div>
<div>
<button type="submit" disabled={pristine || submitting}>Submit</button>
<button type="button" disabled={pristine || submitting} onClick={reset}>Clear Values
</button>
</div>
</form>
)
}
export default reduxForm({
form: 'MaterialUiForm', // a unique identifier for this form
validate,
asyncValidate
})(MaterialUiForm)