Form/form-validation-

PreviousNext

A form/form-validation- example

Docs
shadcnblocksblock

Preview

Loading preview…
code/example/form/form-validation-5.tsx
"use client";

import { zodResolver } from "@hookform/resolvers/zod";
import type { CheckedState } from "@radix-ui/react-checkbox";
import { useForm } from "react-hook-form";
import { z } from "zod";

import { Button } from "@/components/ui/button";
import { Checkbox } from "@/components/ui/checkbox";
import {
  Form,
  FormControl,
  FormDescription,
  FormField,
  FormItem,
  FormLabel,
  FormMessage,
} from "@/components/ui/form";
import { Input } from "@/components/ui/input";

export const title = "Conditional Validation";

const formSchema = z
  .object({
    hasCompany: z.boolean(),
    companyName: z.string().optional(),
    companyWebsite: z.string().url().optional().or(z.literal("")),
  })
  .refine(
    (data) => {
      if (data.hasCompany) {
        return data.companyName && data.companyName.length >= 2;
      }
      return true;
    },
    {
      message: "Company name is required",
      path: ["companyName"],
    },
  );

const Example = () => {
  const form = useForm<z.infer<typeof formSchema>>({
    resolver: zodResolver(formSchema),
    defaultValues: {
      hasCompany: false,
      companyName: "",
      companyWebsite: "",
    },
  });

  const hasCompany = form.watch("hasCompany");

  function onSubmit(values: z.infer<typeof formSchema>) {
    console.log(values);
  }

  return (
    <div className="w-full max-w-md">
      <Form {...form}>
        <form className="space-y-4" onSubmit={form.handleSubmit(onSubmit)}>
          <FormField
            control={form.control}
            name="hasCompany"
            render={({ field }) => (
              <FormItem className="flex flex-row items-start space-y-0 space-x-3">
                <FormControl>
                  <Checkbox
                    checked={field.value}
                    onCheckedChange={(checked: CheckedState) =>
                      field.onChange(checked === true)
                    }
                  />
                </FormControl>
                <div className="space-y-1 leading-none">
                  <FormLabel>I have a company</FormLabel>
                  <FormDescription>
                    Check this if you represent a company.
                  </FormDescription>
                </div>
              </FormItem>
            )}
          />
          {hasCompany && (
            <>
              <FormField
                control={form.control}
                name="companyName"
                render={({ field }) => (
                  <FormItem>
                    <FormLabel>Company Name</FormLabel>
                    <FormControl>
                      <Input
                        className="bg-background"
                        placeholder="Acme Inc"
                        {...field}
                      />
                    </FormControl>
                    <FormMessage />
                  </FormItem>
                )}
              />
              <FormField
                control={form.control}
                name="companyWebsite"
                render={({ field }) => (
                  <FormItem>
                    <FormLabel>Company Website (Optional)</FormLabel>
                    <FormControl>
                      <Input
                        className="bg-background"
                        placeholder="https://example.com"
                        type="url"
                        {...field}
                      />
                    </FormControl>
                    <FormMessage />
                  </FormItem>
                )}
              />
            </>
          )}
          <Button type="submit">Submit</Button>
        </form>
      </Form>
    </div>
  );
};

export default Example;

Installation

npx shadcn@latest add @shadcnblocks/form-form-validation-5

Usage

import { FormFormValidation5 } from "@/components/form-form-validation-5"
<FormFormValidation5 />