feat: add input
This commit is contained in:
parent
b61f12eac7
commit
a2c0f201d9
111
packages/react/components/Input.tsx
Normal file
111
packages/react/components/Input.tsx
Normal file
@ -0,0 +1,111 @@
|
||||
import React from "react";
|
||||
import { VariantProps, vcn } from "../shared";
|
||||
|
||||
const inputColors = {
|
||||
background: {
|
||||
default: "bg-neutral-50 dark:bg-neutral-900",
|
||||
hover: "hover:bg-neutral-100 dark:hover:bg-neutral-800",
|
||||
invalid:
|
||||
"invalid:bg-red-100 invalid:dark:bg-red-900 has-[input:invalid]:bg-red-100 dark:has-[input:invalid]:bg-red-900",
|
||||
invalidHover:
|
||||
"hover:invalid:bg-red-200 dark:hover:invalid:bg-red-800 has-[input:invalid:hover]:bg-red-200 dark:has-[input:invalid:hover]:bg-red-800",
|
||||
},
|
||||
border: {
|
||||
default: "border-neutral-400 dark:border-neutral-600",
|
||||
invalid:
|
||||
"invalid:border-red-400 invalid:dark:border-red-600 has-[input:invalid]:border-red-400 dark:has-[input:invalid]:border-red-600",
|
||||
},
|
||||
ring: {
|
||||
default: "ring-transparent focus-within:ring-current",
|
||||
invalid:
|
||||
"invalid:focus-within:ring-red-400 invalid:focus-within:dark:ring-red-600 has-[input:invalid]:focus-within:ring-red-400 dark:has-[input:invalid]:focus-within:ring-red-600",
|
||||
},
|
||||
};
|
||||
|
||||
const [inputVariant, resolveInputVariantProps] = vcn({
|
||||
base: `rounded-md p-2 border ring-1 outline-none transition-all duration-200 [appearance:textfield] disabled:brightness-50 disabled:saturate-0 disabled:cursor-not-allowed ${inputColors.background.default} ${inputColors.background.hover} ${inputColors.border.default} ${inputColors.ring.default} ${inputColors.background.invalid} ${inputColors.background.invalidHover} ${inputColors.border.invalid} ${inputColors.ring.invalid} [&:has(input)]:flex [&:has(input)]:w-fit`,
|
||||
variants: {
|
||||
unstyled: {
|
||||
true: "bg-transparent border-none p-0 ring-0 hover:bg-transparent invalid:hover:bg-transparent invalid:focus-within:bg-transparent invalid:focus-within:ring-0",
|
||||
false: "",
|
||||
},
|
||||
},
|
||||
defaults: {
|
||||
unstyled: false,
|
||||
},
|
||||
});
|
||||
|
||||
interface InputFrameProps
|
||||
extends VariantProps<typeof inputVariant>,
|
||||
React.ComponentPropsWithoutRef<"label"> {
|
||||
children?: React.ReactNode;
|
||||
}
|
||||
|
||||
const InputFrame = React.forwardRef<HTMLLabelElement, InputFrameProps>(
|
||||
(props, ref) => {
|
||||
const [variantProps, otherPropsUnsafe] = resolveInputVariantProps(props);
|
||||
const { children, ...otherPropsSafe } = otherPropsUnsafe;
|
||||
|
||||
return (
|
||||
<label
|
||||
ref={ref}
|
||||
className={`group/input-frame ${inputVariant(variantProps)}`}
|
||||
{...otherPropsSafe}
|
||||
>
|
||||
{children}
|
||||
</label>
|
||||
);
|
||||
}
|
||||
);
|
||||
|
||||
interface InputProps
|
||||
extends VariantProps<typeof inputVariant>,
|
||||
React.ComponentPropsWithoutRef<"input"> {
|
||||
type: Exclude<
|
||||
React.InputHTMLAttributes<HTMLInputElement>["type"],
|
||||
| "button"
|
||||
| "checkbox"
|
||||
| "color"
|
||||
| "date"
|
||||
| "datetime-local"
|
||||
| "file"
|
||||
| "radio"
|
||||
| "range"
|
||||
| "reset"
|
||||
| "image"
|
||||
| "submit"
|
||||
| "time"
|
||||
>;
|
||||
invalid?: string;
|
||||
}
|
||||
|
||||
const Input = React.forwardRef<HTMLInputElement, InputProps>((props, ref) => {
|
||||
const [variantProps, otherPropsUnsafe] = resolveInputVariantProps(props);
|
||||
const { type, invalid, ...otherPropsSafe } = otherPropsUnsafe;
|
||||
|
||||
const innerRef = React.useRef<HTMLInputElement | null>(null);
|
||||
|
||||
React.useEffect(() => {
|
||||
if (innerRef && innerRef.current) {
|
||||
innerRef.current.setCustomValidity(invalid ?? "");
|
||||
}
|
||||
}, [invalid]);
|
||||
|
||||
return (
|
||||
<input
|
||||
type={type}
|
||||
ref={(el) => {
|
||||
innerRef.current = el;
|
||||
if (typeof ref === "function") {
|
||||
ref(el);
|
||||
} else if (ref) {
|
||||
ref.current = el;
|
||||
}
|
||||
}}
|
||||
className={inputVariant(variantProps)}
|
||||
{...otherPropsSafe}
|
||||
/>
|
||||
);
|
||||
});
|
||||
|
||||
export { InputFrame, Input };
|
115
packages/react/stories/Input.stories.tsx
Normal file
115
packages/react/stories/Input.stories.tsx
Normal file
@ -0,0 +1,115 @@
|
||||
import React from "react";
|
||||
import { Input, InputFrame } from "../components/Input";
|
||||
import { Button } from "../components/Button";
|
||||
|
||||
export default {
|
||||
title: "React/Input",
|
||||
};
|
||||
|
||||
export const TextInput = () => {
|
||||
return <Input type="text" placeholder="Type Here..." />;
|
||||
};
|
||||
|
||||
export const PasswordInput = () => {
|
||||
return <Input type="password" />;
|
||||
};
|
||||
|
||||
export const InvalidInput = () => {
|
||||
return <Input type="text" invalid="Invalid" />;
|
||||
};
|
||||
|
||||
export const DisabledInput = () => {
|
||||
return <Input type="text" disabled />;
|
||||
};
|
||||
|
||||
export const InputWithFrame = () => {
|
||||
const [passwordState, setPasswordState] = React.useState({ visible: false });
|
||||
|
||||
return (
|
||||
<InputFrame>
|
||||
<Input type={passwordState.visible ? "text" : "password"} unstyled />
|
||||
<Button
|
||||
preset="default"
|
||||
size="icon"
|
||||
onClick={() =>
|
||||
setPasswordState((prev) => ({ ...prev, visible: !prev.visible }))
|
||||
}
|
||||
>
|
||||
{passwordState.visible ? (
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1em"
|
||||
height="1em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 17.5c-3.8 0-7.2-2.1-8.8-5.5H1c1.7 4.4 6 7.5 11 7.5s9.3-3.1 11-7.5h-2.2c-1.6 3.4-5 5.5-8.8 5.5"
|
||||
></path>
|
||||
</svg>
|
||||
) : (
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1em"
|
||||
height="1em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 9a3 3 0 0 1 3 3a3 3 0 0 1-3 3a3 3 0 0 1-3-3a3 3 0 0 1 3-3m0-4.5c5 0 9.27 3.11 11 7.5c-1.73 4.39-6 7.5-11 7.5S2.73 16.39 1 12c1.73-4.39 6-7.5 11-7.5M3.18 12a9.821 9.821 0 0 0 17.64 0a9.821 9.821 0 0 0-17.64 0"
|
||||
></path>
|
||||
</svg>
|
||||
)}
|
||||
</Button>
|
||||
</InputFrame>
|
||||
);
|
||||
};
|
||||
|
||||
export const InputWithFrameInvalid = () => {
|
||||
const [passwordState, setPasswordState] = React.useState({ visible: false });
|
||||
|
||||
return (
|
||||
<InputFrame>
|
||||
<Input
|
||||
type={passwordState.visible ? "text" : "password"}
|
||||
unstyled
|
||||
invalid="Invalid"
|
||||
/>
|
||||
<Button
|
||||
preset="default"
|
||||
size="icon"
|
||||
onClick={() =>
|
||||
setPasswordState((prev) => ({ ...prev, visible: !prev.visible }))
|
||||
}
|
||||
background="error"
|
||||
border="error"
|
||||
>
|
||||
{passwordState.visible ? (
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1em"
|
||||
height="1em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 17.5c-3.8 0-7.2-2.1-8.8-5.5H1c1.7 4.4 6 7.5 11 7.5s9.3-3.1 11-7.5h-2.2c-1.6 3.4-5 5.5-8.8 5.5"
|
||||
></path>
|
||||
</svg>
|
||||
) : (
|
||||
<svg
|
||||
xmlns="http://www.w3.org/2000/svg"
|
||||
width="1em"
|
||||
height="1em"
|
||||
viewBox="0 0 24 24"
|
||||
>
|
||||
<path
|
||||
fill="currentColor"
|
||||
d="M12 9a3 3 0 0 1 3 3a3 3 0 0 1-3 3a3 3 0 0 1-3-3a3 3 0 0 1 3-3m0-4.5c5 0 9.27 3.11 11 7.5c-1.73 4.39-6 7.5-11 7.5S2.73 16.39 1 12c1.73-4.39 6-7.5 11-7.5M3.18 12a9.821 9.821 0 0 0 17.64 0a9.821 9.821 0 0 0-17.64 0"
|
||||
></path>
|
||||
</svg>
|
||||
)}
|
||||
</Button>
|
||||
</InputFrame>
|
||||
);
|
||||
};
|
Loading…
x
Reference in New Issue
Block a user