Skip to content

Zod

Regle offers an adapter for Zod. You can use any zod object schema to validate your state. It offers the same DX as using @regle/rules.

Prerequisites

  • zod version 3.x
sh
pnpm add @regle/zod
sh
npm install @regle/zod
sh
yarn add @regle/zod
sh
bun add @regle/zod

Usage

To use @regle/zod, you only need one composable: useZodRegle.

The useZodRegle composable has the same type definitions as useRegle for state and options. However, instead of passing rules as the second parameter, you provide a Zod schema.

You still benefit from features like dirty tracking and custom error handling.

All schemas are independently parsed using safeParse and safeParseAsync (if your schema includes asynchronous transformations or refinements). Error messages defined in the schema will automatically be retrieved.

:::warn refine and transform and other effects functions on z.object are not supported in Regle.

Regle splits the Zod schema into nested independents schema, so only the field you modify will run it's own schema, and will not rerun the entire object schema for each input.

This prevent the use of effects in a object schema. :::

ts
import { 
useZodRegle
} from '@regle/zod';
import {
z
} from 'zod';
const {
r$
} =
useZodRegle
({
name
: '' },
z
.
object
({
name
:
z
.
string
().
min
(1)
}))

Computed schema

You can also have a computed schema that can be based on other state values.

WARNING

When doing refinements or transform, Vue can't track what the schema depends on because you're in a function callback.

Same way as withParams from @regle/rules, you can use the withDeps helper to force dependencies on any schema

ts
import { 
useZodRegle
, type
toZod
,
withDeps
} from '@regle/zod';
import {
z
} from 'zod';
import {
ref
,
computed
} from 'vue';
type
Form
= {
firstName
?: string;
lastName
?: string
} const
form
=
ref
<
Form
>({
firstName
: '',
lastName
: '' })
const
schema
=
computed
(() =>
z
.
object
({
firstName
:
z
.
string
(),
/** * Important to keep track of the depency change * Without it, the validator wouldn't run if `firstName` changed */
lastName
:
withDeps
(
z
.
string
().
refine
((
v
) =>
v
!==
form
.
value
.
firstName
, {
message
: "Last name can't be equal to first name",
}), [() =>
form
.
value
.
firstName
]
), }) satisfies
toZod
<
Form
>
); const {
r$
} =
useZodRegle
(
form
,
schema
);

Type safe output

Similar to the main useRegle composable, r$.$validate in useZodRegle also returns a type-safe output.

ts
import { 
useZodRegle
,
toZod
} from '@regle/zod';
import {
z
} from 'zod';
import {
ref
,
computed
} from 'vue';
type
Form
= {
firstName
?: string;
lastName
?: string
} const
form
=
ref
<
Form
>({
firstName
: '',
lastName
: '' })
const
schema
=
computed
(() =>
z
.
object
({
firstName
:
z
.
string
().
optional
(),
lastName
:
z
.
string
().
min
(1).
refine
(
v
=>
v
!==
form
.
value
.
firstName
, {
message
: "Last name can't be equal to first name"
}), }) satisfies
toZod
<
Form
>)
const {
r$
} =
useZodRegle
(
form
,
schema
);
async function
submit
() {
const {
result
,
data
} = await
r$
.
$validate
();
if (
result
) {
console
.
log
(
data
);
} }

Released under the MIT License. Logo by Johannes Lacourly