92a3337aeb
CI Pipeline / test (20) (pull_request) Successful in 2m41s
CI Pipeline / test (18) (pull_request) Successful in 3m21s
CI Pipeline / e2e (chromium) (pull_request) Failing after 1m25s
CI Pipeline / e2e (firefox) (pull_request) Failing after 1m24s
CI Pipeline / e2e (webkit) (pull_request) Failing after 1m24s
CI Pipeline / visual-regression (pull_request) Failing after 1m53s
CI Pipeline / performance (pull_request) Failing after 1m31s
CI Pipeline / lint (pull_request) Failing after 1m5s
CI Pipeline / storybook (pull_request) Successful in 1m36s
CI Pipeline / build (pull_request) Failing after 1m19s
215 lines
6.3 KiB
TypeScript
215 lines
6.3 KiB
TypeScript
"use client";
|
|
|
|
import React, { memo, useCallback, forwardRef, useId } from "react";
|
|
|
|
interface InputProps extends Omit<
|
|
React.InputHTMLAttributes<HTMLInputElement>,
|
|
"size" | "onChange" | "onFocus" | "onBlur"
|
|
> {
|
|
size?: "small" | "medium" | "large";
|
|
labelVariant?: "default" | "horizontal";
|
|
state?: "default" | "active" | "hover" | "focus";
|
|
disabled?: boolean;
|
|
error?: boolean;
|
|
label?: string;
|
|
placeholder?: string;
|
|
value?: string;
|
|
onChange?: (e: React.ChangeEvent<HTMLInputElement>) => void;
|
|
onFocus?: (e: React.FocusEvent<HTMLInputElement>) => void;
|
|
onBlur?: (e: React.FocusEvent<HTMLInputElement>) => void;
|
|
className?: string;
|
|
}
|
|
|
|
const Input = forwardRef<HTMLInputElement, InputProps>(
|
|
(
|
|
{
|
|
size = "medium",
|
|
labelVariant = "default",
|
|
state = "default",
|
|
disabled = false,
|
|
error = false,
|
|
label,
|
|
placeholder,
|
|
value,
|
|
onChange,
|
|
onFocus,
|
|
onBlur,
|
|
id,
|
|
name,
|
|
type = "text",
|
|
className = "",
|
|
...props
|
|
},
|
|
ref,
|
|
) => {
|
|
// Generate unique ID for accessibility if not provided
|
|
const generatedId = useId();
|
|
const inputId = id || `input-${generatedId}`;
|
|
|
|
// Size variants
|
|
const sizeStyles: Record<
|
|
string,
|
|
{
|
|
input: string;
|
|
label: string;
|
|
container: string;
|
|
radius: string;
|
|
}
|
|
> = {
|
|
small: {
|
|
input:
|
|
labelVariant === "horizontal"
|
|
? "h-[30px] px-[12px] py-[8px] text-[10px]"
|
|
: "h-[32px] px-[12px] py-[8px] text-[10px]",
|
|
label: "text-[12px] leading-[14px] font-medium",
|
|
container: "gap-[4px]",
|
|
radius: "var(--measures-radius-small)",
|
|
},
|
|
medium: {
|
|
input: "h-[36px] px-[12px] py-[8px] text-[14px] leading-[20px]",
|
|
label: "text-[14px] leading-[16px] font-medium",
|
|
container: "gap-[8px]",
|
|
radius: "var(--measures-radius-medium)",
|
|
},
|
|
large: {
|
|
input: "h-[40px] px-[12px] py-[8px] text-[16px] leading-[24px]",
|
|
label: "text-[16px] leading-[20px] font-medium",
|
|
container: "gap-[12px]",
|
|
radius: "var(--measures-radius-large)",
|
|
},
|
|
};
|
|
|
|
// State styles
|
|
const getStateStyles = (): {
|
|
input: string;
|
|
label: string;
|
|
} => {
|
|
if (disabled) {
|
|
return {
|
|
input:
|
|
"bg-[var(--color-content-default-secondary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-tertiary)] cursor-not-allowed",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
}
|
|
|
|
if (error) {
|
|
return {
|
|
input:
|
|
"bg-[var(--color-surface-default-primary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-utility-negative)]",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
}
|
|
|
|
switch (state) {
|
|
case "active":
|
|
return {
|
|
input:
|
|
"bg-[var(--color-surface-default-primary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-tertiary)]",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
case "hover":
|
|
return {
|
|
input:
|
|
"bg-[var(--color-surface-default-primary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-tertiary)] shadow-[0_0_0_2px_var(--color-border-default-tertiary)]",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
case "focus":
|
|
return {
|
|
input:
|
|
"bg-[var(--color-surface-default-primary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-utility-info)] shadow-[0_0_5px_3px_#3281F8]",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
default:
|
|
return {
|
|
input:
|
|
"bg-[var(--color-surface-default-primary)] text-[var(--color-content-default-primary)] border border-[var(--color-border-default-tertiary)] hover:shadow-[0_0_0_2px_var(--color-border-default-tertiary)]",
|
|
label: "text-[var(--color-content-default-secondary)]",
|
|
};
|
|
}
|
|
};
|
|
|
|
const stateStyles = getStateStyles();
|
|
const currentSize = sizeStyles[size];
|
|
|
|
// Container classes based on label variant
|
|
const containerClasses =
|
|
labelVariant === "horizontal"
|
|
? `flex items-center gap-[12px]`
|
|
: `flex flex-col ${currentSize.container}`;
|
|
|
|
const labelClasses =
|
|
labelVariant === "horizontal"
|
|
? `${currentSize.label} font-inter min-w-fit`
|
|
: `${currentSize.label} font-inter`;
|
|
|
|
const inputClasses = `
|
|
w-full border transition-all duration-200 ease-in-out
|
|
focus:outline-none focus:ring-0
|
|
${currentSize.input}
|
|
${stateStyles.input}
|
|
${className}
|
|
`.trim();
|
|
|
|
const handleChange = useCallback(
|
|
(e: React.ChangeEvent<HTMLInputElement>) => {
|
|
if (!disabled && onChange) {
|
|
onChange(e);
|
|
}
|
|
},
|
|
[disabled, onChange],
|
|
);
|
|
|
|
const handleFocus = useCallback(
|
|
(e: React.FocusEvent<HTMLInputElement>) => {
|
|
if (!disabled && onFocus) {
|
|
onFocus(e);
|
|
}
|
|
},
|
|
[disabled, onFocus],
|
|
);
|
|
|
|
const handleBlur = useCallback(
|
|
(e: React.FocusEvent<HTMLInputElement>) => {
|
|
if (!disabled && onBlur) {
|
|
onBlur(e);
|
|
}
|
|
},
|
|
[disabled, onBlur],
|
|
);
|
|
|
|
return (
|
|
<div className={containerClasses}>
|
|
{label && (
|
|
<label
|
|
htmlFor={inputId}
|
|
className={`${labelClasses} font-inter font-medium text-[var(--color-content-default-secondary)]`}
|
|
>
|
|
{label}
|
|
</label>
|
|
)}
|
|
<div className={disabled ? "opacity-40" : ""}>
|
|
<input
|
|
ref={ref}
|
|
id={inputId}
|
|
name={name}
|
|
type={type}
|
|
value={value}
|
|
placeholder={placeholder}
|
|
onChange={handleChange}
|
|
onFocus={handleFocus}
|
|
onBlur={handleBlur}
|
|
disabled={disabled}
|
|
className={inputClasses}
|
|
style={{ borderRadius: currentSize.radius }}
|
|
{...props}
|
|
/>
|
|
</div>
|
|
</div>
|
|
);
|
|
},
|
|
);
|
|
|
|
Input.displayName = "Input";
|
|
|
|
export default memo(Input);
|