Search Components...

Select - React Hook Form

A Select component that uses react-hook-form Controller.

Installation

npx shadcn@latest add /registry/rhf-select.json

Manual Installation

This component is dependent on floating-button component. Please add that component first before adding this one.

1. Copy and paste the following code into your project.

"use client";
import { type ComponentPropsWithoutRef } from "react";
import { Controller, type FieldPath, type FieldValues, useFormContext } from "react-hook-form";
import { cn } from "@/lib/utils";
import { Select, SelectContent, SelectItem } from "@/components/ui/select";
import { Trigger } from "@radix-ui/react-select";
import { FloatingLabelButon } from "@/components/ui/buttons/floating-label-button";
import { ChevronsUpDown } from "lucide-react";
 
// ----------------------------------------------------------------------
 
type SelectOption = {
	label: string;
	value: string;
};
 
interface CustomRHFSelectProps<TFieldValues extends FieldValues>
	extends Omit<ComponentPropsWithoutRef<typeof Controller>, "name" | "control" | "render">,
		Omit<ComponentPropsWithoutRef<typeof Select>, "name" | "defaultValue"> {
	name: FieldPath<TFieldValues>;
	helperText?: string;
	containerClass?: string;
	label?: string;
	buttonClass?: string;
	options: SelectOption[];
}
 
export default function CustomRHFSelect<TFieldValues extends FieldValues>({
	name,
	label,
	helperText,
	defaultValue,
	rules,
	containerClass,
	buttonClass,
	options,
}: CustomRHFSelectProps<TFieldValues>) {
	const { control } = useFormContext();
 
	const handleValueChange = (value: string, onChange: (value: string) => void) => {
		if (value !== "empty") {
			onChange(value);
			return;
		}
		onChange("");
	};
 
	return (
		<Controller
			name={name}
			control={control}
			defaultValue={defaultValue}
			rules={rules}
			render={({ field, fieldState: { error } }) => (
				<div className={cn("space-y-2", containerClass)}>
					<Select onValueChange={(value) => handleValueChange(value, field.onChange)} defaultValue={field.value}>
						<Trigger asChild>
							<FloatingLabelButon
								label={label ?? ""}
								value={field.value}
								variant="outline"
								className={cn("w-full", buttonClass)}
								endIcon={<ChevronsUpDown className="ml-2 h-4 w-4 shrink-0 opacity-50" />}
								error={!!error}
								size="md"
							/>
						</Trigger>
						<SelectContent>
							<SelectItem value="empty" className="rounded-sm focus:bg-accent outline-none">
								<span className="italic text-foreground">Empty</span>
							</SelectItem>
							{options.map((o, idx) => (
								<SelectItem key={idx} value={o.value} className="font-medium">
									{o.label}
								</SelectItem>
							))}
						</SelectContent>
					</Select>
					{!!helperText && <p className="text-sm text-muted-foreground ml-1.5">{helperText}</p>}
					{!!error && <p className="text-sm text-error ml-1.5">{error?.message}</p>}
				</div>
			)}
		/>
	);
}

2. Add this to your @/components/ui/hook-form/index.tsx file for easy import.

export { default as RHFSelect } from "./rhf-select";

Shadcn

This select component is built on top of the excellent foundation provided by Shadcn/UI.

For a deeper dive into the core concepts and building blocks, check out the original Shadcn select component.