Skip to content

Commit 84b80cd

Browse files
reuse validation and fix things after review
1 parent 3e9540b commit 84b80cd

29 files changed

+633
-719
lines changed

packages/ui/locales/en/views.json

+61-142
Large diffs are not rendered by default.

packages/ui/locales/fr/views.json

+64-143
Large diffs are not rendered by default.

packages/ui/src/components/alert/AlertContainer.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -18,7 +18,7 @@ import { cva, type VariantProps } from 'class-variance-authority'
1818
import { Icon } from '../icon'
1919

2020
const alertVariants = cva(
21-
'relative w-full rounded-lg border px-4 py-3 text-sm [&>svg+div]:translate-y-[-3px] [&>svg]:absolute [&>svg]:left-4 [&>svg]:top-4 [&>svg]:text-foreground [&>svg~*]:pl-7',
21+
'[&>svg]:text-foreground relative grid w-full gap-1 rounded-lg border px-4 py-3 text-sm [&>svg+div]:translate-y-[-3px] [&>svg]:absolute [&>svg]:left-4 [&>svg]:top-4 [&>svg~*]:pl-7',
2222
{
2323
variants: {
2424
variant: {

packages/ui/src/components/alert/AlertTitle.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ export interface AlertTitleProps extends PropsWithChildren<React.HTMLAttributes<
77
}
88

99
export const AlertTitle = forwardRef<HTMLHeadingElement, AlertTitleProps>(({ className, children }, ref) => (
10-
<h5 ref={ref} className={cn('mb-1 font-medium leading-none tracking-tight', className)}>
10+
<h5 ref={ref} className={cn('font-medium leading-none tracking-tight', className)}>
1111
{children}
1212
</h5>
1313
))

packages/ui/src/components/button-with-options.tsx

+2-2
Original file line numberDiff line numberDiff line change
@@ -123,7 +123,7 @@ export const ButtonWithOptions = <T extends string>({
123123
control={<RadioButton className="mt-px" value={String(option.value)} id={String(option.value)} />}
124124
id={String(option.value)}
125125
label={option.label}
126-
aria-selected={selectedValue === option.value}
126+
ariaSelected={selectedValue === option.value}
127127
description={option?.description}
128128
/>
129129
</DropdownMenu.Item>
@@ -139,7 +139,7 @@ export const ButtonWithOptions = <T extends string>({
139139
onClick={() => handleOptionChange(option.value)}
140140
>
141141
<span className="flex flex-col gap-y-1.5">
142-
<span className="leading-none text-foreground-8">{option.label}</span>
142+
<span className="text-foreground-8 leading-none">{option.label}</span>
143143
{option?.description && <span className="text-foreground-4">{option.description}</span>}
144144
</span>
145145
</DropdownMenu.Item>

packages/ui/src/components/git-commit-dialog/git-commit-dialog.tsx

+5-5
Original file line numberDiff line numberDiff line change
@@ -175,17 +175,17 @@ export const GitCommitDialog: FC<GitCommitDialogProps> = ({
175175
Commit directly to the
176176
<span
177177
className="
178-
relative mx-1.5 inline-flex gap-1 px-2.5 text-foreground-8
179-
before:absolute before:-top-1 before:left-0 before:z-[-1] before:h-6 before:w-full before:rounded before:bg-background-8
178+
text-foreground-8 before:bg-background-8 relative mx-1.5 inline-flex gap-1
179+
px-2.5 before:absolute before:-top-1 before:left-0 before:z-[-1] before:h-6 before:w-full before:rounded
180180
"
181181
>
182-
<Icon className="translate-y-0.5 text-icons-9" name="branch" size={14} />
182+
<Icon className="text-icons-9 translate-y-0.5" name="branch" size={14} />
183183
{currentBranch}
184184
</span>
185185
branch
186186
</span>
187187
}
188-
aria-selected={commitToGitRefValue === CommitToGitRefOption.DIRECTLY}
188+
ariaSelected={commitToGitRefValue === CommitToGitRefOption.DIRECTLY}
189189
/>
190190
<Option
191191
control={
@@ -197,7 +197,7 @@ export const GitCommitDialog: FC<GitCommitDialogProps> = ({
197197
}
198198
id={CommitToGitRefOption.NEW_BRANCH}
199199
label="Create a new branch for this commit and start a pull request"
200-
aria-selected={commitToGitRefValue === CommitToGitRefOption.NEW_BRANCH}
200+
ariaSelected={commitToGitRefValue === CommitToGitRefOption.NEW_BRANCH}
201201
description={
202202
// TODO: Add correct path
203203
<StyledLink to="/">Learn more about pull requests</StyledLink>

packages/ui/src/components/option.tsx

+13-2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,7 @@ interface OptionProps extends HTMLAttributes<HTMLDivElement> {
1212
description?: string | ReactNode
1313
disabled?: boolean
1414
error?: string
15+
ariaSelected?: boolean
1516
}
1617

1718
/**
@@ -24,13 +25,23 @@ interface OptionProps extends HTMLAttributes<HTMLDivElement> {
2425
* description="Description for Option 1"
2526
* />
2627
*/
27-
export const Option: FC<OptionProps> = ({ control, id, label, description, className, disabled, error, ...props }) => {
28+
export const Option: FC<OptionProps> = ({
29+
control,
30+
id,
31+
label,
32+
description,
33+
className,
34+
disabled,
35+
error,
36+
ariaSelected,
37+
...props
38+
}) => {
2839
return (
2940
<div
3041
className={cn('flex items-start', className)}
3142
role="option"
3243
aria-labelledby={`${id}-label`}
33-
aria-selected={false}
44+
aria-selected={ariaSelected}
3445
{...props}
3546
>
3647
{control}

packages/ui/src/components/textarea.tsx

+1-1
Original file line numberDiff line numberDiff line change
@@ -27,7 +27,7 @@ const Textarea = forwardRef<HTMLTextAreaElement, TextareaProps>(
2727
)}
2828
<textarea
2929
className={cn(
30-
'placeholder:text-foreground-4 flex min-h-[74px] w-full rounded border bg-input-background px-3 py-2 text-sm focus-visible:outline-none focus-visible:rounded disabled:cursor-not-allowed',
30+
'text-foreground-1 placeholder:text-foreground-4 flex min-h-[74px] w-full rounded border bg-input-background px-3 py-2 text-sm focus-visible:outline-none focus-visible:rounded disabled:cursor-not-allowed',
3131
resizable ? 'resize-y [field-sizing:content] whitespace-pre-wrap [word-break:break-word]' : 'resize-none',
3232
className,
3333
error

packages/ui/src/utils/validation.ts

+74
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,74 @@
1+
import { TranslationStore } from '@/views'
2+
import { z } from 'zod'
3+
4+
export const makeValidationUtils = (t: TranslationStore['t']) => ({
5+
required: (name: string): string => t('views:utils.validation.required', `${name} is required`, { name }),
6+
invalid: (name: string): string => t('views:utils.validation.invalid', `${name} is invalid`, { name }),
7+
minLength: (length: number, name: string): [number, string] => [
8+
length,
9+
t('views:utils.validation.minLength', `${name} must be at least ${length} characters`, { name, length })
10+
],
11+
maxLength: (length: number, name: string): [number, string] => [
12+
length,
13+
t('views:utils.validation.maxLength', `${name} must be no longer than ${length} characters`, { name, length })
14+
],
15+
specialSymbols: (name: string): [RegExp, string] => [
16+
/^[a-zA-Z0-9._-\s]+$/,
17+
t(
18+
'views:utils.validation.specialSymbols',
19+
`${name} must contain only letters, numbers, and the characters: - _ .`,
20+
{ name }
21+
)
22+
],
23+
noSpaces: (name: string): [(data: string) => boolean, string] => [
24+
data => !data.includes(' '),
25+
t('views:utils.validation.noSpaces', `${name} cannot contain spaces`, { name })
26+
]
27+
})
28+
29+
export const makeFloatValidationUtils =
30+
(t: TranslationStore['t'], ctx: z.RefinementCtx) =>
31+
({ value: v, path, name }: { value: string | undefined; path: string; name: string }) => {
32+
const { required, invalid, maxLength, specialSymbols, noSpaces } = makeValidationUtils(t)
33+
const value = v?.trim()
34+
35+
const addIssue = (message: string) => ctx.addIssue({ code: 'custom', path: [path], message })
36+
37+
return {
38+
requiredFloat: (): void => {
39+
if (value) return
40+
41+
addIssue(required(name))
42+
},
43+
maxLengthFloat: (length: number): void => {
44+
if (!value) return
45+
const [maxLen, errorMessage] = maxLength(length, name)
46+
if (value.length <= maxLen) return
47+
48+
addIssue(errorMessage)
49+
},
50+
specialSymbolsFloat: (): void => {
51+
if (!value) return
52+
const [regex, errorMessage] = specialSymbols(name)
53+
if (regex.test(value)) return
54+
55+
addIssue(errorMessage)
56+
},
57+
noSpacesFloat: (): void => {
58+
if (!value) return
59+
const [checkFn, errorMessage] = noSpaces(name)
60+
if (checkFn(value)) return
61+
62+
addIssue(errorMessage)
63+
},
64+
urlFloat: (): void => {
65+
if (!value) return
66+
67+
try {
68+
new URL(value)
69+
} catch (error) {
70+
addIssue(invalid(name))
71+
}
72+
}
73+
}
74+
}

packages/ui/src/views/auth/signin-page.tsx

+32-9
Original file line numberDiff line numberDiff line change
@@ -1,9 +1,10 @@
1-
import { useMemo } from 'react'
1+
import { useEffect, useMemo, useState } from 'react'
22
import { useForm } from 'react-hook-form'
33

44
import { Button, Card, Fieldset, FormWrapper, Input, Message, MessageTheme, StyledLink } from '@/components'
55
import { Floating1ColumnLayout, TranslationStore } from '@/views'
66
import { zodResolver } from '@hookform/resolvers/zod'
7+
import { makeValidationUtils } from '@utils/validation'
78
import { z } from 'zod'
89

910
import { Agreements } from './components/agreements'
@@ -16,27 +17,49 @@ interface SignInPageProps {
1617
error?: string
1718
}
1819

19-
const makeSignInSchema = (t: TranslationStore['t']) =>
20-
z.object({
21-
email: z.string().trim().nonempty(t('views:signIn.validation.emailNoEmpty', 'Field can’t be blank')),
22-
password: z.string().nonempty(t('views:signIn.validation.passwordNoEmpty', 'Password can’t be blank'))
20+
const makeSignInSchema = (t: TranslationStore['t']) => {
21+
const { required } = makeValidationUtils(t)
22+
23+
return z.object({
24+
email: z
25+
.string()
26+
.trim()
27+
.nonempty(required(t('views:signIn.emailTitle'))),
28+
password: z.string().nonempty(required(t('views:signIn.passwordTitle')))
2329
})
30+
}
2431

2532
export type SignInData = z.infer<ReturnType<typeof makeSignInSchema>>
2633

2734
export function SignInPage({ handleSignIn, useTranslationStore, isLoading, error }: SignInPageProps) {
35+
const [serverError, setServerError] = useState<string | null>(null)
36+
2837
const { t } = useTranslationStore()
2938
const {
3039
register,
3140
handleSubmit,
3241
formState: { errors }
3342
} = useForm<SignInData>({
43+
mode: 'onChange',
3444
resolver: zodResolver(makeSignInSchema(t))
3545
})
3646

37-
const errorMessage = useMemo(() => (error?.includes('Not Found') ? 'Please check your details' : error), [error])
47+
const errorMessage = useMemo(
48+
() => (error?.includes('Not Found') ? t('views:signIn.checkDetails', 'Please check your details') : error),
49+
[error, t]
50+
)
51+
52+
const handleInputChange = async () => {
53+
if (!serverError) return
54+
setServerError(null)
55+
}
56+
57+
useEffect(() => {
58+
if (!error) return
59+
setServerError(error)
60+
}, [error])
3861

39-
const hasError = Object.keys(errors).length > 0 || !!error
62+
const hasError = Object.keys(errors).length > 0 || !!serverError
4063

4164
return (
4265
<Floating1ColumnLayout
@@ -65,14 +88,14 @@ export function SignInPage({ handleSignIn, useTranslationStore, isLoading, error
6588
label={t('views:signIn.emailTitle', 'Username/Email')}
6689
placeholder={t('views:signIn.emailDescription', 'Your email')}
6790
size="md"
68-
{...register('email')}
91+
{...register('email', { onChange: handleInputChange })}
6992
error={errors.email?.message?.toString()}
7093
autoFocus
7194
/>
7295
<Input
7396
id="password"
7497
type="password"
75-
{...register('password')}
98+
{...register('password', { onChange: handleInputChange })}
7699
label={t('views:signIn.passwordTitle', 'Password')}
77100
placeholder={t('views:signIn.passwordDescription', 'Password')}
78101
size="md"

packages/ui/src/views/auth/signup-page.tsx

+34-25
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,10 @@
1+
import { useEffect, useState } from 'react'
12
import { useForm } from 'react-hook-form'
23

34
import { Button, Card, Fieldset, FormWrapper, Input, Message, MessageTheme, StyledLink } from '@/components'
45
import { Floating1ColumnLayout, TranslationStore } from '@/views'
56
import { zodResolver } from '@hookform/resolvers/zod'
7+
import { makeValidationUtils } from '@utils/validation'
68
import { z } from 'zod'
79

810
import { Agreements } from './components/agreements'
@@ -15,53 +17,60 @@ interface SignUpPageProps {
1517
error?: string
1618
}
1719

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
2024
.object({
2125
userId: z
2226
.string()
2327
.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-zA-Z0-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'))),
3732
email: z
3833
.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'))),
4136
password: z
4237
.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')))
4641
})
4742
.refine(data => data.password === data.confirmPassword, {
4843
message: t('views:signUp.validation.passwordsCheck', "Passwords don't match"),
4944
path: ['confirmPassword']
5045
})
46+
}
5147

5248
export type SignUpData = z.infer<ReturnType<typeof makeSignUpSchema>>
5349

5450
export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error }: SignUpPageProps) {
51+
const [serverError, setServerError] = useState<string | null>(null)
52+
5553
const { t } = useTranslationStore()
5654
const {
5755
register,
5856
handleSubmit,
5957
formState: { errors }
6058
} = useForm<SignUpData>({
59+
mode: 'onChange',
6160
resolver: zodResolver(makeSignUpSchema(t))
6261
})
6362

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
6574

6675
return (
6776
<Floating1ColumnLayout
@@ -88,7 +97,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
8897
<Input
8998
id="userId"
9099
type="text"
91-
{...register('userId')}
100+
{...register('userId', { onChange: handleInputChange })}
92101
placeholder={t('views:signUp.userIDPlaceholder', 'User ID')}
93102
label={t('views:signUp.userIDLabel', 'User ID')}
94103
size="md"
@@ -98,7 +107,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
98107
<Input
99108
id="email"
100109
type="email"
101-
{...register('email')}
110+
{...register('email', { onChange: handleInputChange })}
102111
placeholder={t('views:signUp.emailPlaceholder', 'Your email')}
103112
label={t('views:signUp.emailLabel', 'Email')}
104113
size="md"
@@ -110,7 +119,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
110119
placeholder={t('views:signUp.passwordPlaceholder', 'Password (6+ characters)')}
111120
label={t('views:signUp.passwordLabel', 'Password')}
112121
size="md"
113-
{...register('password')}
122+
{...register('password', { onChange: handleInputChange })}
114123
error={errors.password?.message?.toString()}
115124
/>
116125
<Input
@@ -119,7 +128,7 @@ export function SignUpPage({ isLoading, handleSignUp, useTranslationStore, error
119128
placeholder={t('views:signUp.confirmPasswordPlaceholder', 'Confirm password')}
120129
label={t('views:signUp.confirmPasswordLabel', 'Confirm password')}
121130
size="md"
122-
{...register('confirmPassword')}
131+
{...register('confirmPassword', { onChange: handleInputChange })}
123132
error={errors.confirmPassword?.message?.toString()}
124133
/>
125134
</Fieldset>

0 commit comments

Comments
 (0)