Skip to content

Commit 583b615

Browse files
committed
feat: separated hooks from view in the sign up page
1 parent aaf954e commit 583b615

File tree

9 files changed

+187
-170
lines changed

9 files changed

+187
-170
lines changed

src/app/components/button-form.tsx

+2-1
Original file line numberDiff line numberDiff line change
@@ -39,7 +39,8 @@ export function ButtonForm({
3939
${color.color}
4040
${color.hover}
4141
text-white
42-
font-bold
42+
font-bold
43+
ml-2
4344
py-2
4445
px-4
4546
rounded

src/app/components/button-wrapper/button-wrapper.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ export interface ButtonWrapperProps {
77
}
88

99
export function ButtonWrapper({ children }: ButtonWrapperProps) {
10-
return <div className="flex items-center justify-evenly">{children}</div>;
10+
return <div className="flex items-center justify-end">{children}</div>;
1111
}

src/app/components/loading.tsx

+4-7
Original file line numberDiff line numberDiff line change
@@ -1,7 +1,7 @@
11
"use client";
22

33
import { useAboutCtx } from "@/app/about/hooks/use-about";
4-
import ButtonForm from "@/app/components/button-form";
4+
import { ButtonForm } from "@/app/components/button-form";
55

66
export default function Loading() {
77
const { showLoading } = useAboutCtx();
@@ -10,12 +10,9 @@ export default function Loading() {
1010
<div
1111
className={!showLoading ? "hidden" : "flex justify-center items-center"}
1212
>
13-
<ButtonForm
14-
model="warning"
15-
placeholder="Loading..."
16-
handleOnClick={() => null}
17-
disabled={true}
18-
/>
13+
<ButtonForm model="warning" disabled={true}>
14+
Loading...
15+
</ButtonForm>
1916
<svg className="animate-spin h-5 w-5 mr-3 ..." viewBox="0 0 24 24" />
2017
</div>
2118
);

src/app/signup/components/content/content.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,5 +7,5 @@ export interface SignUpContentProps {
77
}
88

99
export function SignUpContent({ children }: SignUpContentProps) {
10-
return <>{children}</>;
10+
return <div className="flex justify-center items-center">{children}</div>;
1111
}

src/app/signup/components/root/root.tsx

+1-5
Original file line numberDiff line numberDiff line change
@@ -7,9 +7,5 @@ export interface SignUpRootProps {
77
}
88

99
export function SignUpRoot({ children }: SignUpRootProps) {
10-
return (
11-
<div className="w-full h-full flex justify-center items-center">
12-
{children}
13-
</div>
14-
);
10+
return <div className="h-screen w-screen overflow-hidden">{children}</div>;
1511
}

src/app/signup/components/sign-up-form.tsx

-146
This file was deleted.
+57
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,57 @@
1+
"use client"
2+
3+
import { MouseEvent, useCallback } from "react"
4+
import { useSignUpCtx } from "@/app/signup/hooks/use-sign-up"
5+
import { useSignUpDispatch } from "@/app/signup/hooks/use-sign-up-dispatch"
6+
import { useRouter } from "next/navigation"
7+
import { useForm } from "react-hook-form"
8+
import { zodResolver } from "@hookform/resolvers/zod"
9+
import { SignUpSchema } from "@/app/signup/schemas/sign-up-schema"
10+
import { SignUpFormSchema } from "@/app/signup/interfaces/sign-up-form-schema";
11+
12+
export function useSignUpForm() {
13+
const {
14+
register,
15+
handleSubmit,
16+
watch,
17+
formState: { errors },
18+
} = useForm<SignUpFormSchema>({
19+
resolver: zodResolver(SignUpSchema),
20+
});
21+
22+
const file = watch("file");
23+
24+
const router = useRouter();
25+
26+
const { error, success, showLoading } = useSignUpCtx();
27+
28+
const { signUpUser } = useSignUpDispatch();
29+
30+
const memoizedHandleAfterSignUp = useCallback(() => {
31+
router.push("/");
32+
}, [router]);
33+
34+
const handleFormSubmit = (data: SignUpFormSchema) => {
35+
const { success } = SignUpSchema.safeParse(data);
36+
37+
if (success) signUpUser(data);
38+
};
39+
40+
const handleCancelSignUp = (event: MouseEvent<HTMLButtonElement>) => {
41+
event.preventDefault();
42+
memoizedHandleAfterSignUp();
43+
};
44+
45+
return {
46+
register,
47+
handleSubmit,
48+
file,
49+
handleCancelSignUp,
50+
handleFormSubmit,
51+
errors,
52+
showLoading,
53+
success,
54+
error,
55+
memoizedHandleAfterSignUp
56+
}
57+
}

src/app/signup/page.tsx

+117-8
Original file line numberDiff line numberDiff line change
@@ -1,12 +1,121 @@
1-
import React from "react";
2-
import SignUpForm from "@/app/signup/components/sign-up-form";
1+
"use client";
2+
3+
import React, { useEffect } from "react";
4+
import { toast } from "react-toastify";
5+
import Image from "next/image";
6+
import { getObjectErrors } from "@/app/utils/get-object-errors";
7+
import { SignUp } from "@/app/signup/components/index";
8+
import { useSignUpForm } from "@/app/signup/hooks/use-sign-up-form";
9+
10+
export default function SignUpPage() {
11+
const {
12+
register,
13+
handleSubmit,
14+
file,
15+
handleCancelSignUp,
16+
handleFormSubmit,
17+
errors,
18+
showLoading,
19+
success,
20+
error,
21+
memoizedHandleAfterSignUp,
22+
} = useSignUpForm();
23+
24+
useEffect(() => {
25+
if (errors) {
26+
getObjectErrors(errors);
27+
}
28+
}, [errors]);
29+
30+
useEffect(() => {
31+
if (success !== null && success === true && error === null) {
32+
toast.success("User signed up with success!");
33+
memoizedHandleAfterSignUp();
34+
}
35+
}, [error, success, memoizedHandleAfterSignUp]);
336

4-
export default function Home() {
537
return (
6-
<div className="grid grid-rows-[20px_1fr_20px] items-center justify-items-center min-h-screen p-8 pb-20 gap-16 sm:p-20 font-[family-name:var(--font-geist-sans)]">
7-
<main className="flex flex-col gap-8 row-start-2 items-center sm:items-start">
8-
<SignUpForm />
9-
</main>
10-
</div>
38+
<SignUp.Root>
39+
<SignUp.Content>
40+
<SignUp.Form onSubmit={handleSubmit(handleFormSubmit)}>
41+
<Image
42+
alt="User profile image"
43+
src={(() => {
44+
if (file && file.length > 0) {
45+
const url = URL.createObjectURL(file[0]);
46+
return url;
47+
}
48+
49+
return "/images/default_avatar.png";
50+
})()}
51+
quality={100}
52+
width={200}
53+
height={200}
54+
crossOrigin="use-credentials"
55+
/>
56+
57+
<SignUp.InputWrapper>
58+
<SignUp.Label id={"file"}>Profile image</SignUp.Label>
59+
<SignUp.Input
60+
register={register("file", { required: false })}
61+
id="file"
62+
type="file"
63+
/>
64+
</SignUp.InputWrapper>
65+
66+
<SignUp.InputWrapper>
67+
<SignUp.Label id={"name"}>Name</SignUp.Label>
68+
<SignUp.Input
69+
register={register("name", { required: true })}
70+
id="name"
71+
/>
72+
</SignUp.InputWrapper>
73+
74+
<SignUp.InputWrapper>
75+
<SignUp.Label id={"email"}>Email</SignUp.Label>
76+
<SignUp.Input
77+
id="email"
78+
type="email"
79+
register={register("email", { required: true })}
80+
/>
81+
</SignUp.InputWrapper>
82+
83+
<SignUp.InputWrapper>
84+
<SignUp.Label id={"username"}>Username</SignUp.Label>
85+
<SignUp.Input
86+
id="username"
87+
register={register("username", { required: true })}
88+
/>
89+
</SignUp.InputWrapper>
90+
91+
<SignUp.InputWrapper>
92+
<SignUp.Label id={"password"}>Password</SignUp.Label>
93+
<SignUp.Input
94+
id="password"
95+
type="password"
96+
register={register("password", { required: true })}
97+
/>
98+
</SignUp.InputWrapper>
99+
100+
<SignUp.ButtonWrapper>
101+
<SignUp.Button
102+
disabled={showLoading}
103+
type="submit"
104+
model={showLoading ? "disabled" : "success"}
105+
>
106+
Sign up
107+
</SignUp.Button>
108+
109+
<SignUp.Button
110+
disabled={showLoading}
111+
onClick={handleCancelSignUp}
112+
model={showLoading ? "disabled" : "danger"}
113+
>
114+
Cancel
115+
</SignUp.Button>
116+
</SignUp.ButtonWrapper>
117+
</SignUp.Form>
118+
</SignUp.Content>
119+
</SignUp.Root>
11120
);
12121
}

0 commit comments

Comments
 (0)