"use client";
import { OTPInput, OTPInputContext } from "input-otp";
import { MinusIcon } from "lucide-react";
import { type ComponentProps, useContext } from "react";
import { cn } from "@/lib/utils";
function InputOTP({
className,
containerClassName,
...props
}: ComponentProps<typeof OTPInput> & {
containerClassName?: string;
}) {
return (
<OTPInput
className={cn("disabled:cursor-not-allowed", className)}
containerClassName={cn(
"cn-input-otp flex items-center has-disabled:opacity-50",
containerClassName
)}
data-slot="input-otp"
spellCheck={false}
{...props}
/>
);
}
function InputOTPGroup({ className, ...props }: ComponentProps<"div">) {
return (
<div
className={cn(
"flex items-center rounded-md has-aria-invalid:border-destructive has-aria-invalid:ring-2 has-aria-invalid:ring-destructive/20 dark:has-aria-invalid:ring-destructive/40",
className
)}
data-slot="input-otp-group"
{...props}
/>
);
}
function InputOTPSlot({
index,
className,
...props
}: ComponentProps<"div"> & {
index: number;
}) {
const inputOTPContext = useContext(OTPInputContext);
const { char, hasFakeCaret, isActive } = inputOTPContext?.slots[index] ?? {};
return (
<div
className={cn(
"relative flex size-7 items-center justify-center border-input border-y border-e bg-input/20 text-xs/relaxed outline-none transition-all first:rounded-s-md first:border-s last:rounded-e-md aria-invalid:border-destructive data-[active=true]:z-10 data-[active=true]:border-ring data-[active=true]:ring-2 data-[active=true]:ring-ring/30 data-[active=true]:aria-invalid:border-destructive data-[active=true]:aria-invalid:ring-destructive/20 dark:bg-input/30 dark:data-[active=true]:aria-invalid:ring-destructive/40",
className
)}
data-active={isActive}
data-slot="input-otp-slot"
{...props}
>
{char}
{hasFakeCaret && (
<div className="pointer-events-none absolute inset-0 flex items-center justify-center">
<div className="h-4 w-px animate-caret-blink bg-foreground duration-1000" />
</div>
)}
</div>
);
}
function InputOTPSeparator({ ...props }: ComponentProps<"div">) {
return (
<div
aria-hidden="true"
className="flex items-center [&_svg:not([class*='size-'])]:size-4"
data-slot="input-otp-separator"
{...props}
>
<MinusIcon />
</div>
);
}
export { InputOTP, InputOTPGroup, InputOTPSlot, InputOTPSeparator };