45 lines
1.4 KiB
TypeScript
45 lines
1.4 KiB
TypeScript
import { type InputHTMLAttributes, forwardRef } from "react"
|
|
import { cn } from "@/lib/utils"
|
|
|
|
interface FormInputProps extends InputHTMLAttributes<HTMLInputElement> {
|
|
label?: string
|
|
error?: string
|
|
hint?: string
|
|
}
|
|
|
|
export const FormInput = forwardRef<HTMLInputElement, FormInputProps>(
|
|
({ label, error, hint, className, id, ...props }, ref) => {
|
|
const inputId = id || label?.toLowerCase().replace(/\s+/g, "-")
|
|
|
|
return (
|
|
<div className="flex flex-col gap-1.5">
|
|
{label && (
|
|
<label
|
|
htmlFor={inputId}
|
|
className="text-sm font-medium text-foreground"
|
|
>
|
|
{label}
|
|
</label>
|
|
)}
|
|
<input
|
|
ref={ref}
|
|
id={inputId}
|
|
className={cn(
|
|
"rounded-lg border border-input bg-card px-3.5 py-2.5 text-sm text-foreground placeholder:text-muted-foreground",
|
|
"outline-none transition-colors focus:border-primary focus:ring-2 focus:ring-primary/20",
|
|
"disabled:cursor-not-allowed disabled:opacity-50",
|
|
error && "border-destructive focus:border-destructive focus:ring-destructive/20",
|
|
className
|
|
)}
|
|
{...props}
|
|
/>
|
|
{hint && !error && (
|
|
<p className="text-xs text-muted-foreground">{hint}</p>
|
|
)}
|
|
{error && <p className="text-xs text-destructive">{error}</p>}
|
|
</div>
|
|
)
|
|
}
|
|
)
|
|
FormInput.displayName = "FormInput"
|