useFieldArray:
({ control?: Control, name: string, keyName?: string = 'id' }) => object
Custom hook for working with uncontrolled Field Arrays (dynamic inputs). The motivation is to provide better user experience and form performance. You can watch this short video to compare controlled vs uncontrolled Field Array.
Props
Name | Type | Required | Description |
---|---|---|---|
name | string | ✓ | Name of the field. Important: make sure name is in object shape: |
control | Object | control object is from invoking useForm . it's optional if you are using FormContext. | |
keyName | string = 'id' | field array key value, default to "id", you can change the key name. |
function Test() { const { control, register } = useForm(); const { fields, append, prepend, remove, swap, move, insert } = useFieldArray({ control, // control props comes from useForm (optional: if you are using FormContext) name: "test", // unique name for your Field Array // keyName: "id", default to "id", you can change the key name }); return ( {fields.map((field, index) => ( <input key={field.id} // important to include key with field's id {...register(`test.${index}.value`)} defaultValue={field.value} // make sure to include defaultValue /> ))} ); }
Rules
The
field.id
(and notindex
) must be added as the component key to prevent re-renders breaking the fields:// ✅ correct: {fields.map((field, index) => ( <div key={field.id}> <input ... /> </div> ))} // ✅ correct: {fields.map((field, index) => <input key={field.id} ... />)} // ❌ incorrect: {fields.map((field, index) => <input key={index} ... />)}
defaultValue
must be set for all inputs. SupplieddefaultValues
in theuseForm
hook will prepare thefields
object with default value.You can not call actions one after another. Actions need to be triggered per render.
// ❌ The following is not correct handleChange={() => { if (fields.length === 2) { remove(0); } append({ test: 'test' }); }} // ✅ The following is correct and second action is triggered after next render handleChange={() => { append({ test: 'test' }); }} React.useEffect(() => { if (fields.length === 2) { remove(0); } }, fields)
Each
useFieldArray
is unique and has its own state update, which means you should not have multiple useFieldArray with the samename
.Each input name needs to be unique, if you need to build checkbox or radio with the same name then use it with
useController
orcontroller
.Does not support flat field array.
TypeScript
when register input
name
, you will have to cast them asconst
<input key={field.id} {...register(`test.${index}.test` as const)} defaultValue={field.test} />
we do not support circular reference. Refer to this this Github issue for more detail.
for nested field array, you will have to cast the field array by its name.
const { fields } = useFieldArray({ name: `test.${index}.keyValue` as 'test.0.keyValue' });
Return
Name | Type | Description |
---|---|---|
fields | object & { id: string } | This object contains the defaultValue and key for all your inputs. It's important to assign defaultValue to the inputs.Important: Because each input can be uncontrolled,
|
append |
| Append input/inputs to the end of your fields and focus. The input value will be registered during this action. |
prepend |
| Prepend input/inputs to the start of your fields and focus. The input value will be registered during this action. |
insert |
| Insert input/inputs at particular position and focus. |
swap |
| Swap input/inputs position. |
move |
| Move input/inputs to another position. |
remove |
| Remove input/inputs at particular position, or remove all when no index provided. |
import React from "react"; import { useForm, useFieldArray } from "react-hook-form"; function App() { const { register, control, handleSubmit, reset, trigger, setError } = useForm({ // defaultValues: {}; you can populate the fields by this attribute }); const { fields, append, prepend, remove, swap, move, insert } = useFieldArray({ control, name: "test" }); return ( <form onSubmit={handleSubmit(data => console.log(data))}> <ul> {fields.map((item, index) => ( <li key={item.id}> <input {...register(`test.${index}.firstName`)} defaultValue={item.firstName} // make sure to set up defaultValue /> <Controller render={({ field }) => <input {...field} />} name={`test.${index}.lastName`} control={control} defaultValue={item.lastName} // make sure to set up defaultValue /> <button type="button" onClick={() => remove(index)}>Delete</button> </li> ))} </ul> <button type="button" onClick={() => append({ firstName: "appendBill", lastName: "appendLuo" })} > append </button> <input type="submit" /> </form> ); }
Tips
Custom Register
You can also register
inputs at Controller
without the actual input. This makes useFieldArray
quick flexible to use with complex data structure or the actual data is not stored inside an input.
import * as React from "react"; import { useForm, useFieldArray, Controller, useWatch } from "react-hook-form"; const ConditionalInput = ({ control, index, field }) => { const value = useWatch({ name: "test", control }); return ( <Controller control={control} name={`test.${index}.firstName`} render={({ field }) => value?.[index]?.checkbox === "on" ? <input {...field} /> : null } defaultValue={field.firstName} /> ); }; function App() { const { handleSubmit, control, register } = useForm(); const { fields, append, prepend } = useFieldArray({ control, name: "test" }); const onSubmit = (data) => console.log(data); return ( <form onSubmit={handleSubmit(onSubmit)}> {fields.map((field, index) => ( <section key={field.id}> <input type="checkbox" value="on" {...register(`test.${index}.checkbox`)} defaultChecked={field.checked} /> <ConditionalInput {...{ control, index, field }} /> </section> ))} <button type="button" onClick={() => append({ firstName: "append value" }) } > append </button> <input type="submit" /> </form> ); }
Controlled Field Array
There will be cases where you want to control the entire field array, which means each onChange reflects on the fields
object. You can achieve this by merge with useWatch
or watch
's result.
import * as React from "react"; import { useForm, useFieldArray } from "react-hook-form"; export default function App() { const { register, handleSubmit, control, watch } = useForm<FormValues>(); const { fields, append } = useFieldArray({ control, name: "fieldArray" }); const watchFieldArray = watch("fieldArray"); const controlledFields = fields.map((field, index) => { return { ...field, ...watchFieldArray[index] }; }); return ( <form> {controlledFields.map((field, index) => { return <input {...register(`fieldArray.${index}.name` as const)} />; })} <button type="button" onClick={() => append({ name: "bill" }) } > Append </button> </form> ); }