Skip to content

nerd-coder/svelte-zod-form

Repository files navigation

🌡 Svelte Zod Form

version license bundlejs codecov

Building forms in Svelte with breeze, using Zod

Example

REPL: Simple login form REPL: Simple login form

Installation

npm i @nerd-coder/svelte-zod-form

How to use

First you need to create a Zod's schema

import { z } from 'zod'

const loginSchema = z.object({
  email: z.string().email(),
  pass: z.string().min(4),
})

Then pass the schema to ZodFormStore:

const form = new ZodFormStore(loginSchema, { onSubmit: (v) => console.log('Submitted values:', v) })

All the field's handler, value will be generated and typed for you:

// We need pull the generated field store out, in order
// to use the Svelte's "auto subscription" feature "$"
const { email_value, email_error, email_dirty, pass_value, pass_error, pass_dirty } = form.stores

Finally, use it in html

<form on:submit|preventDefault={form.triggerSubmit}>
  <fieldset>
    <input
      name="email"
      on:input={form.fields.email.handleChange}
      on:blur={form.fields.email.handleBlur}
      value={$email_value || ''}
      class:invalid={!!$email_error}
      class:valid={!$email_error && !!$email_dirty}
    />
    {#if $email_error}<p>{$email_error}</p>{/if}
  </fieldset>

  <fieldset>
    <input
      name="pass"
      type="password"
      on:input={form.fields.pass.handleChange}
      on:blur={form.fields.pass.handleBlur}
      value={$pass_value || ''}
      class:invalid={!!$pass_error}
      class:valid={!$pass_error && !!$pass_dirty}
    />
    {#if $pass_error}<p>{$pass_error}</p>{/if}
  </fieldset>

  <button type="submit">Sign In</button>
</form>

Configuration

initialValue

  • type: Partial<T>
  • required: false
  • default: undefined

The initial data in the form. Will revert to this value if call form.reset.

const form = new ZodFormStore(schema, {
  initialValue: { email: 'my@email.com' },
  ...
})

onSubmit

  • type: (v: T) => Promise<void | string> | string | void
  • required: true

Async callback to handle submmition of the form. Should return nothing, or an string contain error message

const form = new ZodFormStore(schema, {
  onSubmit: (values) => console.log('Submitted values:', values),
  ...
})

autoSubmitAfter

  • type: number
  • required: false
  • default: undefined

Auto trigger submit when any data changed, after the delay in ms. Passing falsy value (0 or undefined) to disabled.

const form = new ZodFormStore(schema, {
  autoSubmitAfter: 200,
  ...
})

debounceDelay

  • type: number
  • required: false
  • default: undefined

Debounce the value update, in ms. Passing falsy value (0 or undefined) to disabled.

const form = new ZodFormStore(schema, {
  debounceDelay: 200,
  ...
})

debug

  • type: boolean
  • required: false
  • default: false

Print various debug messages.

const form = new ZodFormStore(schema, {
  debug: true,
  ...
})

API

Prop Type Description
model Readable<T> Form's data. Will be passed to onSubmit handler
options readonly ICreateFormOptions<T> Form settings. Should not be update
triggerSubmit () => Promise<void> Function to start parsing, validating and submit the form's data
reset () => void Function to reset the form to original state.
submitting Readable<boolean> True of submitting the form.
error Readable<string> Error message returned from onSubmit handler, or custom validation message.
errors Readable<string[]> Array of string contains all error messages (including fields's errors and error return from onSubmit handler).
dirty Readable<boolean> Indicate if the form is edited or submitted.
valid Readable<boolean> Indicate if the form is valid.
fields object Generated fields's functions.
stores object Generated fields's stores.

Generated stores's props

Prop Type Description
stores.fieldname_value Readable<T['fieldname']> Readable store holding field's value
stores.fieldname_touched Readable<boolean> The field have been touched or not
stores.fieldname_dirty Readable<boolean> The field value been changed or not
stores.fieldname_error Readable<string> The field validation error message, if any
stores.fieldname_valid Readable<boolean> The field value is valid or not

Generated field's functions

Prop Type Description
fields.fieldname.updateValue (updater: Updater<T['fieldname']>) => void Function to update field's value
stores.fieldname.setValue (val: T['fieldname']) => void Function to set field's value
stores.fieldname.handleChange (val: unknown) => void Callback to update field's value
stores.fieldname.handleBlur () => void Callback to mark field as touched
stores.fieldname.reset () => void Reset field to original state
stores.fieldname.setError (msg: string) => void Manually set field error
stores.fieldname.setTouched (val: boolean) => void Manually set touched state

Features

  • Use Svelte native stores
  • Fast: only update what changed, and you only subscribe to what you need
  • Minimal - only ~1.55Kb in size (gzipped)
  • Build-in validation using Zod
  • TypeScript

Extra

Why the cactus?

> For it resilience

TODO

  • More tests
  • Support Array

Contributions

Any contributions are highly appreciate, please make a pull-request. If you would like to discuses a new feature, please create an issue first.