Skip to content

useController

Build reusable controlled inputs with full form state access.

</> useController: (props?: UseControllerProps) => { field: object, fieldState: object, formState: object }

This custom hook powers Controller. Additionally, it shares the same props and methods as Controller. It's useful for creating reusable Controlled input.

Props


The following table contains information about the arguments for useController.

NameTypeRequiredDescription
nameFieldPathUnique name of your input.
controlControlcontrol object provided by invoking useForm. Optional when using FormProvider.
rulesObjectValidation rules in the same format for register, which includes: required, min, max, minLength, maxLength, pattern, validate.

rules={{ required: true }}
shouldUnregisterboolean = falseInput will be unregistered after unmount and defaultValues will be removed as well. Note: this prop should be avoided when using with useFieldArray as unregister function gets called after input unmount/remount and reorder.
disabledboolean = falsedisabled prop will be returned from field prop. Controlled input will be disabled and its value will be omitted from the submission data.
defaultValueunknownImportant: Can not apply undefined to defaultValue or defaultValues at useForm.
  • You need to either set defaultValue at the field-level or useForm's defaultValues. undefined is not a valid value. If you used defaultValues at useForm, skip using this prop.
  • If your form will invoke reset with default values, you will need to provide useForm with defaultValues.
exactboolean = falseThis prop will enable an exact match for input name subscriptions, default to true.

Return


The following table contains information about properties which useController produces.

Object NameNameTypeDescription
fieldonChange(value: any) => voidA function which sends the input's value to the library. It should be assigned to the onChange prop of the input and value should not be undefined. This prop updates formState and you should avoid manually invoking setValue or other API related to field update.
fieldonBlur() => voidA function which sends the input's onBlur event to the library. It should be assigned to the input's onBlur prop.
fieldvalueunknownThe current value of the controlled component.
fielddisabledbooleanThe disabled state of the input.
fieldnamestringInput's name being registered.
fieldrefReact.RefA ref used to connect hook form to the input. Assign ref to component's input ref to allow hook form to focus the error input.
fieldStateinvalidbooleanInvalid state for current input.
fieldStateisTouchedbooleanTouched state for current controlled input.
fieldStateisDirtybooleanDirty state for current controlled input.
fieldStateerrorobjecterror for this specific input.
formStateisDirtybooleanSet to true after the user modifies any of the inputs. Important: Make sure to provide all inputs' defaultValues at useForm, so hook form can have a single source of truth to compare whether the form is dirty.
formStatedirtyFieldsobjectAn object with the user-modified fields. Make sure to provide all inputs' defaultValues via useForm, so the library can compare against the defaultValues.
formStatetouchedFieldsobjectAn object containing all the inputs the user has interacted with.
formStatedefaultValuesobjectThe value which has been set at useForm's defaultValues or updated defaultValues via reset API.
formStateisSubmittedbooleanSet to true after the form is submitted. Will remain true until the reset method is invoked.
formStateisSubmitSuccessfulbooleanIndicate the form was successfully submitted without any runtime error.
formStateisSubmittingbooleantrue if the form is currently being submitted. false otherwise.
formStateisLoadingbooleantrue if the form is currently loading async default values. Important: this prop is only applicable to async defaultValues.
formStatesubmitCountnumberNumber of times the form was submitted.
formStateisValidbooleanSet to true if the form doesn't have any errors. setError has no effect on isValid formState, isValid will always derived via the entire form validation result.
formStateisValidatingbooleanSet to true during validation.
formStatevalidatingFieldsobjectCapture fields which are getting async validation.
formStateerrorsobjectAn object with field errors. There is also an ErrorMessage component to retrieve error message easily.
formStatedisabledbooleanSet to true if the form is disabled via the disabled prop in useForm.

Examples


import { TextField } from "@material-ui/core"
import { useController, useForm } from "react-hook-form"
function Input({ control, name }) {
const {
field,
fieldState: { invalid, isTouched, isDirty },
formState: { touchedFields, dirtyFields },
} = useController({
name,
control,
rules: { required: true },
})
return (
<TextField
onChange={field.onChange} // send value to hook form
onBlur={field.onBlur} // notify when input is touched/blur
value={field.value} // input value
name={field.name} // send down the input name
inputRef={field.ref} // send input ref, so we can focus on input when error appear
/>
)
}
import * as React from "react"
import { useForm, useController, UseControllerProps } from "react-hook-form"
type FormValues = {
FirstName: string
}
function Input(props: UseControllerProps<FormValues>) {
const { field, fieldState } = useController(props)
return (
<div>
<input {...field} placeholder={props.name} />
<p>{fieldState.isTouched && "Touched"}</p>
<p>{fieldState.isDirty && "Dirty"}</p>
<p>{fieldState.invalid ? "invalid" : "valid"}</p>
</div>
)
}
export default function App() {
const { handleSubmit, control } = useForm<FormValues>({
defaultValues: {
FirstName: "",
},
mode: "onChange",
})
const onSubmit = (data: FormValues) => console.log(data)
return (
<form onSubmit={handleSubmit(onSubmit)}>
<Input control={control} name="FirstName" rules={{ required: true }} />
<input type="submit" />
</form>
)
}
import * as React from "react"
import { useController, useForm } from "react-hook-form"
const Checkboxes = ({ options, control, name }) => {
const { field } = useController({
control,
name,
})
const [value, setValue] = React.useState(field.value || [])
return (
<>
{options.map((option, index) => (
<input
onChange={(e) => {
const valueCopy = [...value]
// update checkbox value
valueCopy[index] = e.target.checked ? e.target.value : null
// send data to react hook form
field.onChange(valueCopy)
// update local state
setValue(valueCopy)
}}
key={option}
checked={value.includes(option)}
type="checkbox"
value={option}
/>
))}
</>
)
}
export default function App() {
const { register, handleSubmit, control } = useForm({
defaultValues: {
controlled: [],
uncontrolled: [],
},
})
const onSubmit = (data) => console.log(data)
return (
<form onSubmit={handleSubmit(onSubmit)}>
<section>
<h2>uncontrolled</h2>
<input {...register("uncontrolled")} type="checkbox" value="A" />
<input {...register("uncontrolled")} type="checkbox" value="B" />
<input {...register("uncontrolled")} type="checkbox" value="C" />
</section>
<section>
<h2>controlled</h2>
<Checkboxes
options={["a", "b", "c"]}
control={control}
name="controlled"
/>
</section>
<input type="submit" />
</form>
)
}

Tips


  • It's important to be aware of each prop's responsibility when working with external controlled components, such as MUI, AntD, Chakra UI. Its job is to spy on the input, report, and set its value.

    • onChange: send data back to hook form
    • onBlur: report that the input has been interacted with (focus and blur)
    • value: set up input initial and updated value
    • ref: allow input to be focused with error
    • name: give input an unique name

    It's fine to host your state and combined with useController.

    const { field } = useController();
    const [value, setValue] = useState(field.value);
    onChange={(event) => {
    field.onChange(parseInt(event.target.value)) // data send back to hook form
    setValue(event.target.value) // UI state
    }}
  • Do not register input again. This custom hook is designed to take care of the registration process.

    const { field } = useController({ name: 'test' })
    <input {...field} /> // ✅
    <input {...field} {...register('test')} /> // ❌ double up the registration
  • It's ideal to use a single useController per component. If you need to use more than one, make sure you rename the prop. May want to consider using Controller instead.

    const { field: input } = useController({ name: 'test' })
    const { field: checkbox } = useController({ name: 'test1' })
    <input {...input} />
    <input {...checkbox} />

Thank you for your support

If you find React Hook Form to be useful in your project, please consider to star and support it.