see updated post example for using v6+ - https://dev.to/aaronksaunders/using-react-hook-form-with-ionic-react-components-update-1463
Setting up react-hook-form is pretty straight forward; You get started by importing the library and defining and initializing the custom hook with any default values.
Not going to cover too much of the basics since there is extensive documentation provided on the library's website: Getting Started
// the import
import { useForm, Controller } from "react-hook-form";
// set the default values for the controls
let initialValues = {
rangeInfo: -100,
fullName: "",
gender: "",
techCos: "",
email: ""
};
const App () => {
const { control, register, handleSubmit, errors, formState } = useForm({
defaultValues: initialValues
});
return (<div></div>)
}
and then we have the onSubmit
function that is called when the form is submitted we use this functaion as a way to the values from the form. Finally we also are managing the state locally using useState
. We are storing the local state information in the variable data
.
// the import
import { useForm, Controller } from "react-hook-form";
const App () => {
const { control, register, handleSubmit, errors, formState } = useForm({
defaultValues: initialValues
});
const [data, setData] = useState();
const onSubmit = data => {
alert(JSON.stringify(data, null, 2));
setData(data);
};
return (<div></div>)
}
Next we set up the form for use in the application; please note the use of the onSubmit
function in the form
I have excluded a lot of the Ionic components for setting up the page, the header and such but they are included in the project and sample code provided at the end of the post
// the import
import { useForm, Controller } from "react-hook-form";
const App () => {
const { control, register, handleSubmit, errors, formState } = useForm({
defaultValues: initialValues
});
const [data, setData] = useState();
const onSubmit = data => {
alert(JSON.stringify(data, null, 2));
setData(data);
};
return (
<form onSubmit={handleSubmit(onSubmit)} >
{/* here is where the Ionic Components will go /*}
</form>
)
}
Most of the Ionic Framework components basic functionality will work fine, track the errors and provide the values without all of the additional useState
boilerplate code you often see in react applications but to get the real benefit of validation and error checking you need to wrap the Ionic Components in the Controller
Component
We will start out first with the basic use on the react-hook-form before we dive in to a control wrapped Ionic Component.
<IonItem>
<IonLabel>Gender</IonLabel>
<IonSelect
placeholder="Select One"
name="gender"
ref={register({ required: true })}
>
<IonSelectOption value="FEMALE">Female</IonSelectOption>
<IonSelectOption value="MALE">Male</IonSelectOption>
</IonSelect>
</IonItem>
{showError("gender")}
As you can see here the simple IonInput
is handled out of the box
<IonItem>
<IonLabel>Name</IonLabel>
<IonInput name="name" ref={register({ required: true })}></IonInput>
</IonItem>
{showError("name")}
I created a simple error handler function to display the error message from the react-hook-form hook. The library creates an object as part of the hook that holds the errors that are generated when the form is validated.
const showError = (_fieldName: string) => {
{
return (
(errors as any)[_fieldName] && (
<div
style={{
color: "red",
padding: 5,
paddingLeft: 12,
fontSize: "smaller"
}}
>
This field is required
</div>
)
);
}
};
Using The React-Hook-Form Control Component
An example of where you have to use the Controller
Component is with the IonRange
Component
Using the IonRange
Component requires the use of the react-hook-form controller
property and listening for the onIonChange
event to get the appropriate value from the IonRange
Component.
We get the value from the IonRange component using the selected.detail.value
property and set the object appropriately and let the react-hook-form hook handle it from there.
<IonItem>
<Controller
as={
<IonRange min={-200} max={200} color="secondary" >
<IonLabel slot="start">-200</IonLabel>
<IonLabel slot="end">200</IonLabel>
</IonRange>
}
control={control}
onChangeName="onIonChange"
onChange={([selected]: any) => {
return selected.detail.value;
}}
name="rangeInfo"
rules={{ required: true }}
/>
</IonItem>
In the end to get the true value from the library and Ionic Framework's Web Components, I suggest you just wrap everything. I was picking and choosing specific components to wrap as needed and when I came to checking the form's state to see if the form was valid, or not I just went all in.
Wrapping Everything in a Control
<IonItem>
<IonLabel>Name - IonInput</IonLabel>
<Controller
as={IonInput}
control={control}
onChangeName="onIonChange"
onChange={([selected]) => {
console.log("fullName", selected.detail.value);
return selected.detail.value;
}}
name="fullName"
rules={{
required: true,
minLength: { value: 4, message: "Must be 4 chars long" }
}}
/>
</IonItem>
{showError("fullName")} {/* USING THE showError FUNCTION */}
A more complex control IonRadioGroup
we cannot just wrap the component name like we did above since there are child components in play here.
<Controller
as={
<IonRadioGroup>
<IonListHeader>
<IonLabel>
<h1>Manufacturers</h1>
</IonLabel>
</IonListHeader>
<IonItem>
<IonLabel>Apple</IonLabel>
<IonRadio value="apple" />
</IonItem>
<IonItem>
<IonLabel>Amazon</IonLabel>
<IonRadio value="amazon" />
</IonItem>
<IonItem>
<IonLabel>Microsoft</IonLabel>
<IonRadio value="microsoft" />
</IonItem>
</IonRadioGroup>
}
control={control}
name="techCos"
rules={{ required: true }}
onChangeName="onIonChange"
onChange={([selected]) => {
console.log(selected.detail.value);
return selected.detail.value;
}}
/>
{/* we can get the error and potentially a custom message */}
{ errors.techCos && (errors.techCos.message || <span>Field Is Required</span>)}
Error Checking and Form Validation
For verifying the contents of the form, you can get access to the formState
object to determine of the form is valid. You can use it to keep the submit button disabled.
<IonButton type="submit" disabled={formState.isValid === false}>
submit
</IonButton>
If you are going to check for errors, you set the mode
of when errors are check...
const { control, register, handleSubmit, errors, formState } = useForm({
defaultValues: initialValues,
mode : 'onChange' // when the values change... check for errors
});
or we can check when fields are blurred, more information is available in the react-form-hooks documentation.
const { control, register, handleSubmit, errors, formState } = useForm({
defaultValues: initialValues,
mode : 'onBlur' // when the you blur... check for errors
});
Source Code / Project / Video
Top comments (18)
I noticed when I ported the code, that react-hook-form in codesandbox.io is older than the latest version. In the latest version, there is a breaking change.
Before:
now:
github.com/react-hook-form/react-h...
@aaronksaunders Thank you. This article was a life saver. Is there a way to handle
IonCheckbox
? I tried the approach similar to Radio button but, doesn't work.just seeing this, did you get it resolved yet?
Not yet. I'm currently working on something else. Will update you by the end of the day.
Thanks for the post, very useful. If I actually want to submit the data to send to an email address, can I just use something like Formspree to do so and it will work in the native iOs and Android apps? Or would it be better to put it in something like firebase and send it from there?
you can use formspree, from what I have seen it should work
great, thanks! Will give it a go. And Thanks for replying so quickly.
Thank you very much for writing up this awesome blog post.
your welcome, extremely helpful hook for efficiently managing forms in react... working on a new one showing hook use in modals and overmindjs for state management
looking forward for that!
This is great info, not many examples are out there. I posted comment on other areas where your content is displayed for other users, but this will break if you are using react-hook-form v6+.
In order to use the method from this post, you would have to use v5.7.2 or below
working on an updated post - dev.to/ionic/using-react-hook-form...
Added a gist with the typescript version of the code to the blog post
gist.github.com/aaronksaunders/862...
Really useful, you saved me.
glad it was helpful to you!
formState.isValid: {(formState.isValid === true).toString()}
Why is it inside a label?
not sure I understand the question
Great article. You may want to do an article on Formtik as well.