feat: form hook

This commit is contained in:
2025-04-23 12:32:27 +03:00
parent 2a7d41bba5
commit c629f0dcf8
7 changed files with 62 additions and 41 deletions

View File

@@ -14,6 +14,7 @@
"preact-iso": "^2.9.1", "preact-iso": "^2.9.1",
"primelocale": "^2.1.2", "primelocale": "^2.1.2",
"primereact": "^10.9.4", "primereact": "^10.9.4",
"react-hook-form": "^7.56.1",
"tailwind-merge": "^3.0.2", "tailwind-merge": "^3.0.2",
"tailwind-variants": "^1.0.0", "tailwind-variants": "^1.0.0",
"tailwindcss": "^4.0.17", "tailwindcss": "^4.0.17",
@@ -612,6 +613,8 @@
"react-dom": ["react-dom@19.1.0", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.0" } }, "sha512-Xs1hdnE+DyKgeHJeJznQmYMIBG3TKIHJJT95Q58nHLSrElKlGQqDTR2HQ9fx5CN/Gk6Vh/kupBTDLU11/nDk/g=="], "react-dom": ["react-dom@19.1.0", "", { "dependencies": { "scheduler": "^0.26.0" }, "peerDependencies": { "react": "^19.1.0" } }, "sha512-Xs1hdnE+DyKgeHJeJznQmYMIBG3TKIHJJT95Q58nHLSrElKlGQqDTR2HQ9fx5CN/Gk6Vh/kupBTDLU11/nDk/g=="],
"react-hook-form": ["react-hook-form@7.56.1", "", { "peerDependencies": { "react": "^16.8.0 || ^17 || ^18 || ^19" } }, "sha512-qWAVokhSpshhcEuQDSANHx3jiAEFzu2HAaaQIzi/r9FNPm1ioAvuJSD4EuZzWd7Al7nTRKcKPnBKO7sRn+zavQ=="],
"react-is": ["react-is@18.3.1", "", {}, "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg=="], "react-is": ["react-is@18.3.1", "", {}, "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg=="],
"react-transition-group": ["react-transition-group@4.4.5", "", { "dependencies": { "@babel/runtime": "^7.5.5", "dom-helpers": "^5.0.1", "loose-envify": "^1.4.0", "prop-types": "^15.6.2" }, "peerDependencies": { "react": ">=16.6.0", "react-dom": ">=16.6.0" } }, "sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g=="], "react-transition-group": ["react-transition-group@4.4.5", "", { "dependencies": { "@babel/runtime": "^7.5.5", "dom-helpers": "^5.0.1", "loose-envify": "^1.4.0", "prop-types": "^15.6.2" }, "peerDependencies": { "react": ">=16.6.0", "react-dom": ">=16.6.0" } }, "sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g=="],

View File

@@ -19,6 +19,7 @@
"preact-iso": "^2.9.1", "preact-iso": "^2.9.1",
"primelocale": "^2.1.2", "primelocale": "^2.1.2",
"primereact": "^10.9.4", "primereact": "^10.9.4",
"react-hook-form": "^7.56.1",
"tailwind-merge": "^3.0.2", "tailwind-merge": "^3.0.2",
"tailwind-variants": "^1.0.0", "tailwind-variants": "^1.0.0",
"tailwindcss": "^4.0.17" "tailwindcss": "^4.0.17"

View File

@@ -1,5 +1,5 @@
@reference '../../index.scss'; @reference '../../index.scss';
.button { .button {
@apply rounded-4xl px-4 py-3 text-xl text-black shadow-[0px_4px_4px_0px_rgba(0,0,0,0.25)] transition-colors hover:cursor-pointer; @apply rounded-4xl px-4 py-3 text-xl text-black shadow-[0px_4px_4px_0px_rgba(0,0,0,0.25)] transition-colors hover:cursor-pointer focus:outline-1;
} }

View File

@@ -16,6 +16,7 @@ interface ButtonProps {
color?: "primary" | "secondary" | "red"; color?: "primary" | "secondary" | "red";
onClick?: () => void; onClick?: () => void;
className?: string; className?: string;
type?: "button" | "submit";
} }
const Button: FunctionComponent<ButtonProps> = ({ const Button: FunctionComponent<ButtonProps> = ({
@@ -23,9 +24,10 @@ const Button: FunctionComponent<ButtonProps> = ({
onClick = () => {}, onClick = () => {},
color = "primary", color = "primary",
className = "", className = "",
type = "button",
}) => { }) => {
return ( return (
<button type="button" class={button({ color: color, class: className })} onClick={onClick}> <button type={type} class={button({ color: color, class: className })} onClick={onClick}>
{children} {children}
</button> </button>
); );

View File

@@ -1,5 +1,5 @@
import { cn } from "@/utils/class-merge"; import { cn } from "@/utils/class-merge";
import { FunctionComponent, Ref } from "preact"; import { forwardRef, HTMLProps, useEffect } from "preact/compat";
import { tv } from "tailwind-variants"; import { tv } from "tailwind-variants";
import classes from "./Input.module.scss"; import classes from "./Input.module.scss";
@@ -20,29 +20,19 @@ const input = tv({
}, },
}); });
interface InputProps { interface InputProps extends HTMLProps<HTMLInputElement> {
isPassword?: boolean;
placeholder?: string;
textAlign?: "center" | "left"; textAlign?: "center" | "left";
error?: string; error?: string;
textRef?: Ref<HTMLInputElement> | null;
} }
const Input: FunctionComponent<InputProps> = ({ const Input = forwardRef<HTMLInputElement, InputProps>((props, ref) => {
isPassword = false, const { textAlign, error, type = "text", ...rest } = props;
placeholder = "", useEffect(() => {
textAlign, console.log(`error: ${error}`);
error = "", }, [error]);
textRef = null,
}: InputProps) => {
return ( return (
<div class="flex w-full flex-col items-center gap-1"> <div class="flex w-full flex-col items-center gap-1">
<input <input class={input({ "text-align": textAlign, "border-error": !!error })} ref={ref} type={type} {...rest} />
type={isPassword ? "password" : "text"}
class={input({ "text-align": textAlign, "border-error": error !== "" })}
placeholder={placeholder}
ref={textRef}
/>
<p <p
class={cn("invisible h-10 w-[80%] text-center text-[0.7rem] break-words text-red-500", { class={cn("invisible h-10 w-[80%] text-center text-[0.7rem] break-words text-red-500", {
visible: error !== "", visible: error !== "",
@@ -52,6 +42,8 @@ const Input: FunctionComponent<InputProps> = ({
</p> </p>
</div> </div>
); );
}; });
Input.displayName = "AHInput";
export default Input; export default Input;

4
src/pages/login.dto.ts Normal file
View File

@@ -0,0 +1,4 @@
export interface ILoginForm {
login: string;
password: string;
}

View File

@@ -6,7 +6,9 @@ import { useAppContext } from "@/providers/AuthProvider";
import { FunctionComponent } from "preact"; import { FunctionComponent } from "preact";
import { useLocation } from "preact-iso"; import { useLocation } from "preact-iso";
import "preact/debug"; import "preact/debug";
import { useRef, useState } from "preact/hooks"; import { useState } from "preact/hooks";
import { Controller, SubmitHandler, useForm } from "react-hook-form";
import { ILoginForm } from "./login.dto";
import classes from "./login.module.scss"; import classes from "./login.module.scss";
const testUser = { const testUser = {
@@ -16,37 +18,54 @@ const testUser = {
const LoginPage: FunctionComponent = () => { const LoginPage: FunctionComponent = () => {
const { isLoggedIn } = useAppContext(); const { isLoggedIn } = useAppContext();
const { route } = useLocation(); const { route } = useLocation();
const loginRef = useRef<HTMLInputElement | null>(null);
const passwordRef = useRef<HTMLInputElement | null>(null);
const [loginError, setLoginError] = useState("");
const [passwordError, setPasswordError] = useState(""); const [passwordError, setPasswordError] = useState("");
const login = async () => { const login: SubmitHandler<ILoginForm> = async (data) => {
if (!loginRef.current || !passwordRef.current) return; console.log(data);
setLoginError(""); if (data.login !== testUser.login || data.password !== testUser.password) {
setPasswordError(""); setError("login", { message: "Неверный" }); //TODO: не показывает ошибку
if (!loginRef.current.value.length || !passwordRef.current.value.length) {
if (!loginRef.current.value.length) setLoginError("Введите логин");
if (!passwordRef.current.value.length) setPasswordError("Введите пароль");
return;
}
if (loginRef.current.value !== testUser.login || passwordRef.current.value !== testUser.password) {
setLoginError("Неправильный логин или пароль");
return; return;
} }
isLoggedIn.value = true; isLoggedIn.value = true;
localStorage.setItem("loggedIn", "true"); localStorage.setItem("loggedIn", "true");
route("/profile/tasks", true); route("/profile/tasks", true);
}; };
const { control, handleSubmit, formState, setError } = useForm({
defaultValues: {
login: "",
password: "",
},
mode: "onChange",
});
if (isLoggedIn.value) route("/profile/tasks", true); if (isLoggedIn.value) route("/profile/tasks", true);
return !isLoggedIn.value ? ( return !isLoggedIn.value ? (
<div class={classes.login_container}> <div class={classes.login_container}>
<div class={classes.login_card}> <div class={classes.login_card}>
<p class={classes.login_card_name}>Антихвост</p> <p class={classes.login_card_name}>Антихвост</p>
<Input placeholder="Логин" textAlign="center" textRef={loginRef} error={loginError} /> <form onSubmit={handleSubmit((data) => login(data))}>
<Input isPassword placeholder="Пароль" textAlign="center" textRef={passwordRef} error={passwordError} /> <Controller
<Button color="secondary" onClick={login}> name="login"
control={control}
rules={{
required: "Введите логин",
}}
render={({ field }) => (
<Input placeholder="Логин" textAlign="center" error={formState.errors.login?.message} {...field} />
)}
/>
<Controller
name="password"
control={control}
rules={{
required: "Введите пароль",
}}
render={({ field }) => (
<Input placeholder="Пароль" textAlign="center" type="password" error={passwordError} {...field} />
)}
/>
<Button type="submit" color="secondary" className="w-full">
Войти Войти
</Button> </Button>
</form>
</div> </div>
</div> </div>
) : ( ) : (