Auto-infer form fields from zod schema and render them with react-hook-form with E2E type safety.
npm install -S zod-form-renderer zod react-hook-form @hookform/resolvers
The zod form renderer uses the zod type inference to map schema properties to form input fields.
This library might be useful to you, if you
- use TypeScript
- use zod and react-hook-form
- know your zod schema at build time
- have multiple forms in your application
- want type-safe form fields
- want a clean forms API without any
formik
orreact-hook-form
clutter.
Make sure you have
"strict": true"
in yourtsconfig.json
!
Start by creating your zod validation schema.
export const mySchema = z.object({
title: z.enum(['', 'Dr.', 'Prof.']),
name: z.string(),
birthday: z.coerce.date(),
age: z.number(),
accept: z.boolean(),
});
As you can see, a schema may contain different zod types. We will use these to create separate field renderers.
To give you an example, we use a simple TextInputRenderer
for zod strings.
import { ComponentPropsWithRef } from 'react';
import { useFieldRendererContext } from 'zod-form-renderer';
// Use input props as for any React component
export type TextRendererProps = ComponentPropsWithRef<'input'> & {
label: string;
};
export const TextRenderer = (props: TextRendererProps) => {
// The zod-form-renderer will automatically provide the field
// name, schema and form context to use in the renderer.
const { name, schema, form } = useFieldRendererContext();
// React to errors in the field state
const error = form.formState.errors?.[name];
return (
<div>
<label htmlFor={name}>
{props.label}
{schema.isOptional() && ` (Optional)`}
</label>
<br />
<input id={name} {...form.register(name)} {...props} />
<br />
<p style={{ color: 'red' }}>{error?.message?.toString()}</p>
</div>
);
};
A field renderer is a simple React component which displays an input field. You can apply any styling or additional behavior.
Use type="number"
for z.number()
types or <select />
for z.enum()
. Further examples can be found in /test/support/renderers
.
Once you have defined field renderers for all zod primitive types, combine them in a map.
import { createRendererMap } from 'zod-form-renderer';
// Provide renderers for all these required types
export const myRendererMap = createRendererMap({
Enum: SelectRenderer,
String: TextRenderer,
Number: NumberRenderer,
Boolean: CheckboxRenderer,
Date: DatepickerRenderer,
Default: DefaultRenderer,
Submit: SubmitButton,
});
Now you're ready to set up your first form renderer instance.
import { FormRenderer } from 'zod-form-renderer';
<FormRenderer
schema={mySchema}
typeRendererMap={myRendererMap}
useFormProps={{
// Under the hood, react-hook-form is used.
// Apply any form behavior you'd like
defaultValues: {
name: 'John Doe',
},
}}
onSubmit={(values) => {
console.log(values);
}}
>
{({ controls: { Title, Name, Birthday, Age, Accept, Submit } }) => (
<>
<Title
label="My Title"
options={[
{ value: '', label: 'None' },
{ value: 'Dr.', label: 'Dr.' },
{ value: 'Prof.', label: 'Prof.' },
]}
/>
<Name label="My Name" />
<Birthday label="My Birthday" />
<Age label="My Age" />
<Accept label="I Accept" />
<Submit>{"Let's go!"}</Submit>
</>
)}
</FormRenderer>;
The form renderer returns a controls
property with React components. These components are directly and type-safely inferred from your schema
and rendererMap
.
Any required properties from your field renderers will be enforced. No wiring-up or registering is required for react-hook-form
, you simply add your submit handler and that's it!
As seen before, any react-hook-form
configuration will be passed through.
The <FormRenderer />
also returns a reference to the hook-form, so you have access to all instance methods there.
<FormRenderer /** props... **/>
{({
controls: {
/** ... **/
},
form,
}) => {
const hasAccepted = form.watch('accept');
return (
<>
{/** Other form fields **/}
<Accept label="I Accept" />
<Submit disabled={!hasAccepted}>Let's go!</Submit>
</>
);
}}
</FormRenderer>
You might ask yourself, what about custom fields like a file upload? Or you want both a select box and a radio button group for your z.enum()
type within the same form?
No worries, we got you covered. Overwriting single fields is possible with the optional fieldRendererMap
property.
// Define a FileUploadRenderer with <input type="file">.
import { FileUploadRenderer, FileUploadRendererProps } from '...';
<FormRenderer
schema={...}
typeRendererMap={myRendererMap}
fieldRendererMap={{
myImage: FileUploadRenderer,
}}
onSubmit={...}
>
{({ controls: { MyImage, Submit } }) => (
<>
<MyImage<FileUploadRendererProps> />
<Submit>Upload</Submit>
</>
)}
</FormRenderer>
Overwriting any form field is always possible. You can create as many renderers as you like and apply them where needed. Please note that you have to provide the props type manually as a generic in this case. We are not able to infer that yet.
Please read our Code of conduct to keep our community open and respectable. 💖
Want to report a bug, contribute some code, or improve the documentation? Excellent!
Read up on our guidelines for contributing and then check out one of our issues labeled as help wanted
or good first issue
.
If you believe you have found a security vulnerability, we encourage you to responsibly disclose this and not open a public issue. Security issues in this open source project can be safely reported via opensource@thepeaklab.com.
This project is MIT-licensed.
Developed with 💖 at the peak lab.