1
+ import { useEffect , useState } from 'react'
1
2
import { useForm } from 'react-hook-form'
2
3
3
4
import { Button , Card , Fieldset , FormWrapper , Input , Message , MessageTheme , StyledLink } from '@/components'
4
5
import { Floating1ColumnLayout , TranslationStore } from '@/views'
5
6
import { zodResolver } from '@hookform/resolvers/zod'
7
+ import { makeValidationUtils } from '@utils/validation'
6
8
import { z } from 'zod'
7
9
8
10
import { Agreements } from './components/agreements'
@@ -15,53 +17,60 @@ interface SignUpPageProps {
15
17
error ?: string
16
18
}
17
19
18
- const makeSignUpSchema = ( t : TranslationStore [ 't' ] ) =>
19
- z
20
+ const makeSignUpSchema = ( t : TranslationStore [ 't' ] ) => {
21
+ const { required, maxLength, minLength, specialSymbols, noSpaces, invalid } = makeValidationUtils ( t )
22
+
23
+ return z
20
24
. object ( {
21
25
userId : z
22
26
. string ( )
23
27
. trim ( )
24
- . nonempty ( t ( 'views:signUp.validation.userIDNoEmpty' , 'User ID can’t be blank' ) )
25
- . max ( 100 , t ( 'views:signUp.validation.userIDMax' , 'User ID must be no longer than 100 characters' ) )
26
- . regex (
27
- / ^ [ a - z A - Z 0 - 9 . _ - \s ] + $ / ,
28
- t (
29
- 'views:signUp.validation.userIDRegex' ,
30
- 'User ID must contain only letters, numbers, and the characters: - _ .'
31
- )
32
- )
33
- . refine (
34
- data => ! data . includes ( ' ' ) ,
35
- t ( 'views:signUp.validation.userIDNoSpaces' , 'User ID cannot contain spaces' )
36
- ) ,
28
+ . nonempty ( required ( t ( 'views:signUp.userIDPlaceholder' ) ) )
29
+ . max ( ...maxLength ( 100 , t ( 'views:signUp.userIDPlaceholder' ) ) )
30
+ . regex ( ...specialSymbols ( t ( 'views:signUp.userIDPlaceholder' ) ) )
31
+ . refine ( ...noSpaces ( t ( 'views:signUp.userIDPlaceholder' ) ) ) ,
37
32
email : z
38
33
. string ( )
39
- . email ( t ( 'views:signUp.validation.emailNoEmpty ' , 'Invalid email address' ) )
40
- . max ( 250 , t ( 'views:signUp.validation.emailMax ' , 'Email must be no longer than 250 characters' ) ) ,
34
+ . email ( invalid ( t ( 'views:signUp.emailLabel ' , 'Email' ) ) )
35
+ . max ( ... maxLength ( 250 , t ( 'views:signUp.emailLabel ' , 'Email' ) ) ) ,
41
36
password : z
42
37
. string ( )
43
- . min ( 6 , t ( 'views:signUp.validation.passwordNoEmpty ' , 'Password must be at least 6 characters' ) )
44
- . max ( 128 , t ( 'views:signUp.validation.passwordMax ' , 'Password must be no longer than 128 characters' ) ) ,
45
- confirmPassword : z . string ( )
38
+ . min ( ... minLength ( 6 , t ( 'views:signUp.passwordLabel ' , 'Password' ) ) )
39
+ . max ( ... maxLength ( 128 , t ( 'views:signUp.passwordLabel ' , 'Password' ) ) ) ,
40
+ confirmPassword : z . string ( ) . min ( ... minLength ( 6 , t ( 'views:signUp.passwordLabel' , 'Password' ) ) )
46
41
} )
47
42
. refine ( data => data . password === data . confirmPassword , {
48
43
message : t ( 'views:signUp.validation.passwordsCheck' , "Passwords don't match" ) ,
49
44
path : [ 'confirmPassword' ]
50
45
} )
46
+ }
51
47
52
48
export type SignUpData = z . infer < ReturnType < typeof makeSignUpSchema > >
53
49
54
50
export function SignUpPage ( { isLoading, handleSignUp, useTranslationStore, error } : SignUpPageProps ) {
51
+ const [ serverError , setServerError ] = useState < string | null > ( null )
52
+
55
53
const { t } = useTranslationStore ( )
56
54
const {
57
55
register,
58
56
handleSubmit,
59
57
formState : { errors }
60
58
} = useForm < SignUpData > ( {
59
+ mode : 'onChange' ,
61
60
resolver : zodResolver ( makeSignUpSchema ( t ) )
62
61
} )
63
62
64
- const hasError = Object . keys ( errors ) . length > 0 || ! ! error
63
+ const handleInputChange = async ( ) => {
64
+ if ( ! serverError ) return
65
+ setServerError ( null )
66
+ }
67
+
68
+ useEffect ( ( ) => {
69
+ if ( ! error ) return
70
+ setServerError ( error )
71
+ } , [ error ] )
72
+
73
+ const hasError = Object . keys ( errors ) . length > 0 || ! ! serverError
65
74
66
75
return (
67
76
< Floating1ColumnLayout
@@ -88,7 +97,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
88
97
< Input
89
98
id = "userId"
90
99
type = "text"
91
- { ...register ( 'userId' ) }
100
+ { ...register ( 'userId' , { onChange : handleInputChange } ) }
92
101
placeholder = { t ( 'views:signUp.userIDPlaceholder' , 'User ID' ) }
93
102
label = { t ( 'views:signUp.userIDLabel' , 'User ID' ) }
94
103
size = "md"
@@ -98,7 +107,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
98
107
< Input
99
108
id = "email"
100
109
type = "email"
101
- { ...register ( 'email' ) }
110
+ { ...register ( 'email' , { onChange : handleInputChange } ) }
102
111
placeholder = { t ( 'views:signUp.emailPlaceholder' , 'Your email' ) }
103
112
label = { t ( 'views:signUp.emailLabel' , 'Email' ) }
104
113
size = "md"
@@ -110,7 +119,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
110
119
placeholder = { t ( 'views:signUp.passwordPlaceholder' , 'Password (6+ characters)' ) }
111
120
label = { t ( 'views:signUp.passwordLabel' , 'Password' ) }
112
121
size = "md"
113
- { ...register ( 'password' ) }
122
+ { ...register ( 'password' , { onChange : handleInputChange } ) }
114
123
error = { errors . password ?. message ?. toString ( ) }
115
124
/>
116
125
< Input
@@ -119,7 +128,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
119
128
placeholder = { t ( 'views:signUp.confirmPasswordPlaceholder' , 'Confirm password' ) }
120
129
label = { t ( 'views:signUp.confirmPasswordLabel' , 'Confirm password' ) }
121
130
size = "md"
122
- { ...register ( 'confirmPassword' ) }
131
+ { ...register ( 'confirmPassword' , { onChange : handleInputChange } ) }
123
132
error = { errors . confirmPassword ?. message ?. toString ( ) }
124
133
/>
125
134
</ Fieldset >
0 commit comments