Files
gh-jezweb-claude-skills-ski…/references/rhf-api-reference.md
2025-11-30 08:25:27 +08:00

7.6 KiB

React Hook Form API Reference

Complete API reference for React Hook Form v7.65.0


useForm Hook

const {
  register,
  handleSubmit,
  watch,
  formState,
  setValue,
  getValues,
  reset,
  trigger,
  control,
  setError,
  clearErrors,
  setFocus,
} = useForm<FormData>(options)

Options

Option Type Description
resolver Resolver Schema validation resolver (zodResolver, etc.)
mode 'onSubmit' | 'onChange' | 'onBlur' | 'all' When to validate (default: 'onSubmit')
reValidateMode 'onChange' | 'onBlur' When to re-validate after error
defaultValues object | () => object | Promise<object> Initial form values
values object Controlled form values
resetOptions object Options for reset behavior
shouldUnregister boolean Unregister fields when unmounted
shouldFocusError boolean Focus first error on submit
criteriaMode 'firstError' | 'all' Return first error or all
delayError number Delay error display (ms)

register

Register input and apply validation rules.

<input {...register('fieldName', options)} />

Options:

  • required: boolean | string
  • min: number | { value: number, message: string }
  • max: number | { value: number, message: string }
  • minLength: number | { value: number, message: string }
  • maxLength: number | { value: number, message: string }
  • pattern: RegExp | { value: RegExp, message: string }
  • validate: (value) => boolean | string | object
  • valueAsNumber: boolean
  • valueAsDate: boolean
  • disabled: boolean
  • onChange: (e) => void
  • onBlur: (e) => void

handleSubmit

Wraps your form submission handler.

<form onSubmit={handleSubmit(onSubmit, onError)}>

function onSubmit(data: FormData) {
  // Valid data
}

function onError(errors: FieldErrors) {
  // Validation errors
}

watch

Watch specified inputs and return their values.

// Watch all fields
const values = watch()

// Watch specific field
const email = watch('email')

// Watch multiple fields
const [email, password] = watch(['email', 'password'])

// Watch with callback
useEffect(() => {
  const subscription = watch((value, { name, type }) => {
    console.log(value, name, type)
  })
  return () => subscription.unsubscribe()
}, [watch])

formState

Form state object.

const {
  isDirty,        // Form has been modified
  dirtyFields,    // Object of modified fields
  touchedFields,  // Object of touched fields
  isSubmitted,    // Form has been submitted
  isSubmitSuccessful, // Last submission successful
  isSubmitting,   // Form is currently submitting
  isValidating,   // Form is validating
  isValid,        // Form is valid
  errors,         // Validation errors
  submitCount,    // Number of submissions
} = formState

setValue

Set field value programmatically.

setValue('fieldName', value, options)

// Options
{
  shouldValidate: boolean,  // Trigger validation
  shouldDirty: boolean,     // Mark as dirty
  shouldTouch: boolean,     // Mark as touched
}

getValues

Get current form values.

// Get all values
const values = getValues()

// Get specific field
const email = getValues('email')

// Get multiple fields
const [email, password] = getValues(['email', 'password'])

reset

Reset form to default values.

reset() // Reset to defaultValues

reset({ email: '', password: '' }) // Reset to specific values

reset(undefined, {
  keepErrors: boolean,
  keepDirty: boolean,
  keepIsSubmitted: boolean,
  keepTouched: boolean,
  keepIsValid: boolean,
  keepSubmitCount: boolean,
})

trigger

Manually trigger validation.

// Trigger all fields
await trigger()

// Trigger specific field
await trigger('email')

// Trigger multiple fields
await trigger(['email', 'password'])

setError

Set field error manually.

setError('fieldName', {
  type: 'manual',
  message: 'Error message',
})

// Root error (not tied to specific field)
setError('root', {
  type: 'server',
  message: 'Server error',
})

clearErrors

Clear field errors.

clearErrors() // Clear all errors

clearErrors('email') // Clear specific field

clearErrors(['email', 'password']) // Clear multiple fields

setFocus

Focus on specific field.

setFocus('fieldName', { shouldSelect: true })

Controller

For controlled components (third-party UI libraries).

import { Controller } from 'react-hook-form'

<Controller
  name="fieldName"
  control={control}
  defaultValue=""
  rules={{ required: true }}
  render={({ field, fieldState, formState }) => (
    <CustomInput
      {...field}
      error={fieldState.error}
    />
  )}
/>

render props:

  • field: { value, onChange, onBlur, ref, name }
  • fieldState: { invalid, isTouched, isDirty, error }
  • formState: Full form state

useController

Hook version of Controller (for reusable components).

import { useController } from 'react-hook-form'

function CustomInput({ name, control }) {
  const {
    field,
    fieldState: { invalid, isTouched, isDirty, error },
    formState: { touchedFields, dirtyFields }
  } = useController({
    name,
    control,
    rules: { required: true },
    defaultValue: '',
  })

  return <input {...field} />
}

useFieldArray

Manage dynamic field arrays.

import { useFieldArray } from 'react-hook-form'

const { fields, append, prepend, remove, insert, update, replace } = useFieldArray({
  control,
  name: 'items',
  keyName: 'id', // Default: 'id'
})

Methods:

  • append(value) - Add to end
  • prepend(value) - Add to beginning
  • insert(index, value) - Insert at index
  • remove(index) - Remove at index
  • update(index, value) - Update at index
  • replace(values) - Replace entire array

Important: Use field.id as key, not array index!

{fields.map((field, index) => (
  <div key={field.id}> {/* Use field.id! */}
    <input {...register(`items.${index}.name`)} />
  </div>
))}

useWatch

Subscribe to input changes without re-rendering entire form.

import { useWatch } from 'react-hook-form'

const email = useWatch({
  control,
  name: 'email',
  defaultValue: '',
})

useFormState

Subscribe to form state without re-rendering entire form.

import { useFormState } from 'react-hook-form'

const { isDirty, isValid } = useFormState({ control })

useFormContext

Access form context (for deeply nested components).

import { useFormContext } from 'react-hook-form'

function NestedComponent() {
  const { register, formState: { errors } } = useFormContext()

  return <input {...register('email')} />
}

// Wrap form with FormProvider
import { FormProvider, useForm } from 'react-hook-form'

function App() {
  const methods = useForm()

  return (
    <FormProvider {...methods}>
      <form>
        <NestedComponent />
      </form>
    </FormProvider>
  )
}

ErrorMessage

Helper component for displaying errors (from @hookform/error-message).

import { ErrorMessage } from '@hookform/error-message'

<ErrorMessage
  errors={errors}
  name="email"
  render={({ message }) => <span className="error">{message}</span>}
/>

DevTool

Development tool for debugging (from @hookform/devtools).

import { DevTool } from '@hookform/devtools'

<DevTool control={control} />

Official Docs: https://react-hook-form.com/