mirror of
https://github.com/DeveloLongScript/MHSF.git
synced 2026-05-07 16:44:59 -05:00
97 lines
2.5 KiB
TypeScript
97 lines
2.5 KiB
TypeScript
"use client";
|
|
|
|
import type React from "react";
|
|
import { Label } from "@/components/ui/label";
|
|
import { generateID } from "@/lib/form-helper";
|
|
|
|
export type Size = "sm" | "md" | "lg";
|
|
export type Shadow = "sm" | "none";
|
|
|
|
const sizeClass = {
|
|
sm: "px-3 py-1.5",
|
|
md: "px-4 py-2",
|
|
lg: "px-5 py-3",
|
|
};
|
|
|
|
const shadowClass = {
|
|
sm: "shadow-xs",
|
|
none: "shadow-none",
|
|
};
|
|
|
|
interface InputWithLabelProps {
|
|
label?: string;
|
|
value?: string;
|
|
id?: string;
|
|
placeholder?: string;
|
|
disabled?: boolean;
|
|
required?: boolean;
|
|
size?: Size;
|
|
suffix?: React.ReactNode;
|
|
prefix?: React.ReactNode;
|
|
shadow?: Shadow;
|
|
className?: string;
|
|
onChange?: (e: React.ChangeEvent<HTMLInputElement>) => void;
|
|
}
|
|
|
|
const Input: React.FC<InputWithLabelProps> = ({
|
|
label,
|
|
value,
|
|
placeholder = "",
|
|
disabled = false,
|
|
required = false,
|
|
id = generateID(),
|
|
size = "md",
|
|
suffix,
|
|
prefix,
|
|
shadow = "none",
|
|
className = "",
|
|
onChange,
|
|
}) => {
|
|
const borderClass =
|
|
"border border-slate-200 border-b-slate-300 dark:border-zinc-800 dark:border-t-zinc-700";
|
|
|
|
return (
|
|
<div className={`flex flex-col gap-1 ${className}`}>
|
|
{label && (
|
|
<Label
|
|
htmlFor={id}
|
|
className={`peer-invalid:text-red-500 ${
|
|
required ? "after:content-['*'] after:text-red-500 after:ml-1" : ""
|
|
}`}
|
|
>
|
|
{label}
|
|
</Label>
|
|
)}
|
|
<div
|
|
className={`${shadowClass[shadow-sm]} border ${borderClass} focus-within:border-primary-900 dark:focus-within:border-primary-100 focus-within:ring-2 ring-slate-300 dark:ring-zinc-700 transition-colors rounded-lg flex flex-row items-center text-sm ${className}`}
|
|
>
|
|
{prefix && (
|
|
<div
|
|
className={`rounded-lg rounded-r-none text-slate-600 dark:text-zinc-400 ${sizeClass[size]}`}
|
|
>
|
|
{prefix}
|
|
</div>
|
|
)}
|
|
<input
|
|
type="text"
|
|
id={id}
|
|
placeholder={placeholder}
|
|
disabled={disabled}
|
|
value={value}
|
|
onChange={onChange}
|
|
className={`${sizeClass[size]} bg-white dark:bg-zinc-950 focus:outline-hidden rounded-lg text-sm w-full disabled:bg-slate-100 disabled:cursor-not-allowed invalid:border-red-500! peer invalid:text-red-500 z-10 ${className}`}
|
|
/>
|
|
{suffix && (
|
|
<div
|
|
className={`rounded-lg rounded-l-none text-slate-600 dark:text-zinc-400 ${sizeClass[size]}`}
|
|
>
|
|
{suffix}
|
|
</div>
|
|
)}
|
|
</div>
|
|
</div>
|
|
);
|
|
};
|
|
|
|
export { Input };
|