In many cases, you want to change the validation rules based depending on the state of the form or other conditions. The most popular example of this is when you want to validate a field differently based on whether the user has submitted the form for the first time or not.
We support this through our onDynamic validation function.
import { revalidateLogic, createForm } from '@tanstack/svelte-form'
// ...
const form = createForm(() => ({
  defaultValues: {
    firstName: '',
    lastName: '',
  },
  // If this is omitted, onDynamic will not be called
  validationLogic: revalidateLogic(),
  validators: {
    onDynamic: ({ value }) => {
      if (!value.firstName) {
        return { firstName: 'A first name is required' }
      }
      return undefined
    },
  },
}))
import { revalidateLogic, createForm } from '@tanstack/svelte-form'
// ...
const form = createForm(() => ({
  defaultValues: {
    firstName: '',
    lastName: '',
  },
  // If this is omitted, onDynamic will not be called
  validationLogic: revalidateLogic(),
  validators: {
    onDynamic: ({ value }) => {
      if (!value.firstName) {
        return { firstName: 'A first name is required' }
      }
      return undefined
    },
  },
}))
By default onDynamic is not called, so you need to pass revalidateLogic() to the validationLogic option of createForm.
revalidateLogic allows you to specify when validation should be run and change the validation rules dynamically based on the current submission state of the form.
It takes two arguments:
mode: The mode of validation prior to the first form submission. This can be one of the following:
modeAfterSubmission: The mode of validation after the form has been submitted. This can be one of the following:
You can, for example, use the following to revalidate on blur after the first submission:
const form = createForm(() => ({
  // ...
  validationLogic: revalidateLogic({
    mode: 'submit',
    modeAfterSubmission: 'blur',
  }),
  // ...
}))
const form = createForm(() => ({
  // ...
  validationLogic: revalidateLogic({
    mode: 'submit',
    modeAfterSubmission: 'blur',
  }),
  // ...
}))
Just as you might access errors from an onChange or onBlur validation, you can access the errors from the onDynamic validation function using the form.state.errorMap object.
<script lang="ts">
  import { createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    // ...
    validationLogic: revalidateLogic(),
    validators: {
      onDynamic: ({ value }) => {
        if (!value.firstName) {
          return { firstName: 'A first name is required' }
        }
        return undefined
      },
    },
  }))
</script>
<p>{form.state.errorMap.onDynamic?.firstName}</p>
<script lang="ts">
  import { createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    // ...
    validationLogic: revalidateLogic(),
    validators: {
      onDynamic: ({ value }) => {
        if (!value.firstName) {
          return { firstName: 'A first name is required' }
        }
        return undefined
      },
    },
  }))
</script>
<p>{form.state.errorMap.onDynamic?.firstName}</p>
You can use onDynamic validation alongside other validation logic, such as onChange or onBlur.
<script lang="ts">
  import { revalidateLogic, createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    defaultValues: {
      firstName: '',
      lastName: '',
    },
    validationLogic: revalidateLogic(),
    validators: {
      onChange: ({ value }) => {
        if (!value.firstName) {
          return { firstName: 'A first name is required' }
        }
        return undefined
      },
      onDynamic: ({ value }) => {
        if (!value.lastName) {
          return { lastName: 'A last name is required' }
        }
        return undefined
      },
    },
  }))
</script>
<div>
  <p>{form.state.errorMap.onChange?.firstName}</p>
  <p>{form.state.errorMap.onDynamic?.lastName}</p>
</div>
<script lang="ts">
  import { revalidateLogic, createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    defaultValues: {
      firstName: '',
      lastName: '',
    },
    validationLogic: revalidateLogic(),
    validators: {
      onChange: ({ value }) => {
        if (!value.firstName) {
          return { firstName: 'A first name is required' }
        }
        return undefined
      },
      onDynamic: ({ value }) => {
        if (!value.lastName) {
          return { lastName: 'A last name is required' }
        }
        return undefined
      },
    },
  }))
</script>
<div>
  <p>{form.state.errorMap.onChange?.firstName}</p>
  <p>{form.state.errorMap.onDynamic?.lastName}</p>
</div>
You can also use onDynamic validation with fields, just like you would with other validation logic.
<script lang="ts">
  import { createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    defaultValues: {
      name: '',
      age: 0,
    },
    validationLogic: revalidateLogic(),
    onSubmit({ value }) {
      alert(JSON.stringify(value))
    },
  }))
</script>
<form
  onsubmit={(e) => {
    e.preventDefault()
    e.stopPropagation()
    form.handleSubmit()
  }}
>
  <form.Field
    name={'age'}
    validators={{
      onDynamic: ({ value }) =>
        value > 18 ? undefined : 'Age must be greater than 18',
    }}
  >
    {#snippet children(field)}
      <div>
        <input
          type="number"
          onchange={(e) => field.handleChange(e.target.valueAsNumber)}
          onblur={field.handleBlur}
          value={field.state.value}
        />
        <p style="color: red">
          {field.state.meta.errorMap.onDynamic}
        </p>
      </div>
    {/snippet}
  </form.Field>
  <button type="submit">Submit</button>
</form>
<script lang="ts">
  import { createForm } from '@tanstack/svelte-form'
  const form = createForm(() => ({
    defaultValues: {
      name: '',
      age: 0,
    },
    validationLogic: revalidateLogic(),
    onSubmit({ value }) {
      alert(JSON.stringify(value))
    },
  }))
</script>
<form
  onsubmit={(e) => {
    e.preventDefault()
    e.stopPropagation()
    form.handleSubmit()
  }}
>
  <form.Field
    name={'age'}
    validators={{
      onDynamic: ({ value }) =>
        value > 18 ? undefined : 'Age must be greater than 18',
    }}
  >
    {#snippet children(field)}
      <div>
        <input
          type="number"
          onchange={(e) => field.handleChange(e.target.valueAsNumber)}
          onblur={field.handleBlur}
          value={field.state.value}
        />
        <p style="color: red">
          {field.state.meta.errorMap.onDynamic}
        </p>
      </div>
    {/snippet}
  </form.Field>
  <button type="submit">Submit</button>
</form>
Async validation can also be used with onDynamic just like with other validation logic. You can even debounce the async validation to avoid excessive calls.
const form = createForm(() => ({
  defaultValues: {
    username: '',
  },
  validationLogic: revalidateLogic(),
  validators: {
    onDynamicAsyncDebounceMs: 500, // Debounce the async validation by 500ms
    onDynamicAsync: async ({ value }) => {
      if (!value.username) {
        return { username: 'Username is required' }
      }
      // Simulate an async validation
      const isValid = await validateUsername(value.username)
      return isValid ? undefined : { username: 'Username is already taken' }
    },
  },
}))
const form = createForm(() => ({
  defaultValues: {
    username: '',
  },
  validationLogic: revalidateLogic(),
  validators: {
    onDynamicAsyncDebounceMs: 500, // Debounce the async validation by 500ms
    onDynamicAsync: async ({ value }) => {
      if (!value.username) {
        return { username: 'Username is required' }
      }
      // Simulate an async validation
      const isValid = await validateUsername(value.username)
      return isValid ? undefined : { username: 'Username is already taken' }
    },
  },
}))
You can also use standard schema validation libraries like Valibot or Zod with onDynamic validation. This allows you to define complex validation rules that can change dynamically based on the form state.
import { z } from 'zod'
const schema = z.object({
  firstName: z.string().min(1, 'A first name is required'),
  lastName: z.string().min(1, 'A last name is required'),
})
const form = createForm(() => ({
  defaultValues: {
    firstName: '',
    lastName: '',
  },
  validationLogic: revalidateLogic(),
  validators: {
    onDynamic: schema,
  },
}))
import { z } from 'zod'
const schema = z.object({
  firstName: z.string().min(1, 'A first name is required'),
  lastName: z.string().min(1, 'A last name is required'),
})
const form = createForm(() => ({
  defaultValues: {
    firstName: '',
    lastName: '',
  },
  validationLogic: revalidateLogic(),
  validators: {
    onDynamic: schema,
  },
}))
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.
Your weekly dose of JavaScript news. Delivered every Monday to over 100,000 devs, for free.
