This commit is contained in:
2024-09-13 17:08:31 +02:00
parent 4272f14ebf
commit 6a65074c16
17 changed files with 797 additions and 26 deletions

2
.idea/prettier.xml generated
View File

@@ -3,6 +3,6 @@
<component name="PrettierConfiguration">
<option name="myConfigurationMode" value="AUTOMATIC" />
<option name="myRunOnSave" value="true" />
<option name="myFilesPattern" value="**/*.{js,ts,jsx,tsx,cjs,cts,mjs,mts,vue,astro,json}" />
<option name="myFilesPattern" value="**/*.{js,ts,jsx,tsx,cjs,cts,mjs,mts,vue,astro,json,html}" />
</component>
</project>

View File

@@ -12,14 +12,19 @@
},
"dependencies": {
"@fontsource/inter": "^5.0.20",
"@hookform/resolvers": "^3.9.0",
"@radix-ui/react-checkbox": "^1.1.1",
"@radix-ui/react-select": "^2.1.1",
"@radix-ui/react-slot": "^1.1.0",
"class-variance-authority": "^0.7.0",
"clsx": "^2.1.1",
"lucide-react": "^0.439.0",
"next": "14.2.7",
"react": "^18",
"react-dom": "^18",
"tailwind-merge": "^2.5.2"
"react-hook-form": "^7.53.0",
"tailwind-merge": "^2.5.2",
"zod": "^3.23.8"
},
"devDependencies": {
"@chromatic-com/storybook": "1.8.0",

100
pnpm-lock.yaml generated
View File

@@ -11,6 +11,12 @@ importers:
'@fontsource/inter':
specifier: ^5.0.20
version: 5.0.20
'@hookform/resolvers':
specifier: ^3.9.0
version: 3.9.0(react-hook-form@7.53.0(react@18.3.1))
'@radix-ui/react-checkbox':
specifier: ^1.1.1
version: 1.1.1(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
'@radix-ui/react-select':
specifier: ^2.1.1
version: 2.1.1(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
@@ -23,6 +29,9 @@ importers:
clsx:
specifier: ^2.1.1
version: 2.1.1
lucide-react:
specifier: ^0.439.0
version: 0.439.0(react@18.3.1)
next:
specifier: 14.2.7
version: 14.2.7(@babel/core@7.25.2)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
@@ -32,9 +41,15 @@ importers:
react-dom:
specifier: ^18
version: 18.3.1(react@18.3.1)
react-hook-form:
specifier: ^7.53.0
version: 7.53.0(react@18.3.1)
tailwind-merge:
specifier: ^2.5.2
version: 2.5.2
zod:
specifier: ^3.23.8
version: 3.23.8
devDependencies:
'@chromatic-com/storybook':
specifier: 1.8.0
@@ -955,6 +970,11 @@ packages:
'@fontsource/inter@5.0.20':
resolution: {integrity: sha512-rtw2F7xfM7rJmmnncXnR4ADr5wXsp4GyN1O1jmQJ1PMjAK+bm620/ZkQkeOYOkGoa09OksGinOeMA+Mkt6K9PQ==}
'@hookform/resolvers@3.9.0':
resolution: {integrity: sha512-bU0Gr4EepJ/EQsH/IwEzYLsT/PEj5C0ynLQ4m+GSHS+xKH4TfSelhluTgOaoc4kA5s7eCsQbM4wvZLzELmWzUg==}
peerDependencies:
react-hook-form: ^7.0.0
'@humanwhocodes/config-array@0.11.14':
resolution: {integrity: sha512-3T8LkOmg45BV5FICb15QQMsyUSWrQ8AygVfC7ZG32zOalnqrilm018ZVCw0eapXux8FtA33q8PSRSstjee3jSg==}
engines: {node: '>=10.10.0'}
@@ -1233,6 +1253,19 @@ packages:
'@types/react-dom':
optional: true
'@radix-ui/react-checkbox@1.1.1':
resolution: {integrity: sha512-0i/EKJ222Afa1FE0C6pNJxDq1itzcl3HChE9DwskA4th4KRse8ojx8a1nVcOjwJdbpDLcz7uol77yYnQNMHdKw==}
peerDependencies:
'@types/react': '*'
'@types/react-dom': '*'
react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
peerDependenciesMeta:
'@types/react':
optional: true
'@types/react-dom':
optional: true
'@radix-ui/react-collection@1.1.0':
resolution: {integrity: sha512-GZsZslMJEyo1VKm5L1ZJY8tGDxZNPAoUeQUIbKeJfoi7Q4kmig5AsgLMYYuyYbfjd8fBmFORAIwYAkXMnXZgZw==}
peerDependencies:
@@ -1343,6 +1376,19 @@ packages:
'@types/react-dom':
optional: true
'@radix-ui/react-presence@1.1.0':
resolution: {integrity: sha512-Gq6wuRN/asf9H/E/VzdKoUtT8GC9PQc9z40/vEr0VCJ4u5XvvhWIrSsCB6vD2/cH7ugTdSfYq9fLJCcM00acrQ==}
peerDependencies:
'@types/react': '*'
'@types/react-dom': '*'
react: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
react-dom: ^16.8 || ^17.0 || ^18.0 || ^19.0 || ^19.0.0-rc
peerDependenciesMeta:
'@types/react':
optional: true
'@types/react-dom':
optional: true
'@radix-ui/react-primitive@2.0.0':
resolution: {integrity: sha512-ZSpFm0/uHa8zTvKBDjLFWLo8dkr4MBsiDLz0g3gMUwqgLHz9rTaRRGYDgvZPtBJgYCBKXkS9fzmoySgr8CO6Cw==}
peerDependencies:
@@ -3657,6 +3703,11 @@ packages:
lru-cache@5.1.1:
resolution: {integrity: sha512-KpNARQA3Iwv+jTA0utUVVbrh+Jlrr1Fv0e56GGzAFOXN7dk/FviaDW8LHmK52DlcH4WP2n6gI8vN1aesBFgo9w==}
lucide-react@0.439.0:
resolution: {integrity: sha512-PafSWvDTpxdtNEndS2HIHxcNAbd54OaqSYJO90/b63rab2HWYqDbH194j0i82ZFdWOAcf0AHinRykXRRK2PJbw==}
peerDependencies:
react: ^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0-rc
lz-string@1.5.0:
resolution: {integrity: sha512-h5bgJWpxJNswbU7qCrV0tIKQCaS3blPDrqKWx+QxzuzL1zGUzij9XCWLrSLsJPu5t+eWA/ycetzYAO5IOMcWAQ==}
hasBin: true
@@ -4300,6 +4351,12 @@ packages:
react: ^0.14.8 || ^15.0.1 || ^16.0.0 || ^17.0.1 || ^18.0.0
react-dom: ^0.14.8 || ^15.0.1 || ^16.0.0 || ^17.0.1 || ^18.0.0
react-hook-form@7.53.0:
resolution: {integrity: sha512-M1n3HhqCww6S2hxLxciEXy2oISPnAzxY7gvwVPrtlczTM/1dDadXgUxDpHMrMTblDOcm/AXtXxHwZ3jpg1mqKQ==}
engines: {node: '>=18.0.0'}
peerDependencies:
react: ^16.8.0 || ^17 || ^18 || ^19
react-is@16.13.1:
resolution: {integrity: sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==}
@@ -5163,6 +5220,9 @@ packages:
resolution: {integrity: sha512-b4JR1PFR10y1mKjhHY9LaGo6tmrgjit7hxVIeAmyMw3jegXR4dhYqLaQF5zMXZxY7tLpMyJeLjr1C4rLmkVe8g==}
engines: {node: '>=12.20'}
zod@3.23.8:
resolution: {integrity: sha512-XBx9AXhXktjUqnepgTiE5flcKIYWi/rme0Eaj+5Y0lftuGBq+jyRu/md4WnuxqgP1ubdpNCsYEYPxrzVHD8d6g==}
snapshots:
'@adobe/css-tools@4.4.0': {}
@@ -6182,6 +6242,10 @@ snapshots:
'@fontsource/inter@5.0.20': {}
'@hookform/resolvers@3.9.0(react-hook-form@7.53.0(react@18.3.1))':
dependencies:
react-hook-form: 7.53.0(react@18.3.1)
'@humanwhocodes/config-array@0.11.14':
dependencies:
'@humanwhocodes/object-schema': 2.0.3
@@ -6388,6 +6452,22 @@ snapshots:
'@types/react': 18.3.5
'@types/react-dom': 18.3.0
'@radix-ui/react-checkbox@1.1.1(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)':
dependencies:
'@radix-ui/primitive': 1.1.0
'@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1)
'@radix-ui/react-context': 1.1.0(@types/react@18.3.5)(react@18.3.1)
'@radix-ui/react-presence': 1.1.0(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
'@radix-ui/react-primitive': 2.0.0(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)
'@radix-ui/react-use-controllable-state': 1.1.0(@types/react@18.3.5)(react@18.3.1)
'@radix-ui/react-use-previous': 1.1.0(@types/react@18.3.5)(react@18.3.1)
'@radix-ui/react-use-size': 1.1.0(@types/react@18.3.5)(react@18.3.1)
react: 18.3.1
react-dom: 18.3.1(react@18.3.1)
optionalDependencies:
'@types/react': 18.3.5
'@types/react-dom': 18.3.0
'@radix-ui/react-collection@1.1.0(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)':
dependencies:
'@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1)
@@ -6483,6 +6563,16 @@ snapshots:
'@types/react': 18.3.5
'@types/react-dom': 18.3.0
'@radix-ui/react-presence@1.1.0(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)':
dependencies:
'@radix-ui/react-compose-refs': 1.1.0(@types/react@18.3.5)(react@18.3.1)
'@radix-ui/react-use-layout-effect': 1.1.0(@types/react@18.3.5)(react@18.3.1)
react: 18.3.1
react-dom: 18.3.1(react@18.3.1)
optionalDependencies:
'@types/react': 18.3.5
'@types/react-dom': 18.3.0
'@radix-ui/react-primitive@2.0.0(@types/react-dom@18.3.0)(@types/react@18.3.5)(react-dom@18.3.1(react@18.3.1))(react@18.3.1)':
dependencies:
'@radix-ui/react-slot': 1.1.0(@types/react@18.3.5)(react@18.3.1)
@@ -9395,6 +9485,10 @@ snapshots:
dependencies:
yallist: 3.1.1
lucide-react@0.439.0(react@18.3.1):
dependencies:
react: 18.3.1
lz-string@1.5.0: {}
magic-string@0.30.11:
@@ -10055,6 +10149,10 @@ snapshots:
react-dom: 18.3.1(react@18.3.1)
react-is: 18.1.0
react-hook-form@7.53.0(react@18.3.1):
dependencies:
react: 18.3.1
react-is@16.13.1: {}
react-is@17.0.2: {}
@@ -11064,3 +11162,5 @@ snapshots:
yocto-queue@0.1.0: {}
yocto-queue@1.1.1: {}
zod@3.23.8: {}

175
server/.gitignore vendored Normal file
View File

@@ -0,0 +1,175 @@
# Based on https://raw.githubusercontent.com/github/gitignore/main/Node.gitignore
# Logs
logs
_.log
npm-debug.log_
yarn-debug.log*
yarn-error.log*
lerna-debug.log*
.pnpm-debug.log*
# Caches
.cache
# Diagnostic reports (https://nodejs.org/api/report.html)
report.[0-9]_.[0-9]_.[0-9]_.[0-9]_.json
# Runtime data
pids
_.pid
_.seed
*.pid.lock
# Directory for instrumented libs generated by jscoverage/JSCover
lib-cov
# Coverage directory used by tools like istanbul
coverage
*.lcov
# nyc test coverage
.nyc_output
# Grunt intermediate storage (https://gruntjs.com/creating-plugins#storing-task-files)
.grunt
# Bower dependency directory (https://bower.io/)
bower_components
# node-waf configuration
.lock-wscript
# Compiled binary addons (https://nodejs.org/api/addons.html)
build/Release
# Dependency directories
node_modules/
jspm_packages/
# Snowpack dependency directory (https://snowpack.dev/)
web_modules/
# TypeScript cache
*.tsbuildinfo
# Optional npm cache directory
.npm
# Optional eslint cache
.eslintcache
# Optional stylelint cache
.stylelintcache
# Microbundle cache
.rpt2_cache/
.rts2_cache_cjs/
.rts2_cache_es/
.rts2_cache_umd/
# Optional REPL history
.node_repl_history
# Output of 'npm pack'
*.tgz
# Yarn Integrity file
.yarn-integrity
# dotenv environment variable files
.env
.env.development.local
.env.test.local
.env.production.local
.env.local
# parcel-bundler cache (https://parceljs.org/)
.parcel-cache
# Next.js build output
.next
out
# Nuxt.js build / generate output
.nuxt
dist
# Gatsby files
# Comment in the public line in if your project uses Gatsby and not Next.js
# https://nextjs.org/blog/next-9-1#public-directory-support
# public
# vuepress build output
.vuepress/dist
# vuepress v2.x temp and cache directory
.temp
# Docusaurus cache and generated files
.docusaurus
# Serverless directories
.serverless/
# FuseBox cache
.fusebox/
# DynamoDB Local files
.dynamodb/
# TernJS port file
.tern-port
# Stores VSCode versions used for testing VSCode extensions
.vscode-test
# yarn v2
.yarn/cache
.yarn/unplugged
.yarn/build-state.yml
.yarn/install-state.gz
.pnp.*
# IntelliJ based IDEs
.idea
# Finder (MacOS) folder config
.DS_Store

15
server/README.md Normal file
View File

@@ -0,0 +1,15 @@
# server
To install dependencies:
```bash
bun install
```
To run:
```bash
bun run index.ts
```
This project was created using `bun init` in bun v1.1.27. [Bun](https://bun.sh) is a fast all-in-one JavaScript runtime.

BIN
server/bun.lockb Normal file

Binary file not shown.

19
server/dashboard.html Normal file
View File

@@ -0,0 +1,19 @@
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta
name="viewport"
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"
/>
<meta
http-equiv="X-UA-Compatible"
content="ie=edge"
/>
<title>{{username}} | Dashboard</title>
</head>
<body>
<h1>Hello, {{username}}!</h1>
You are <strong>{{age}}</strong> and beautiful!
</body>
</html>

64
server/index.html Normal file
View File

@@ -0,0 +1,64 @@
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta
name="viewport"
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"
/>
<meta
http-equiv="X-UA-Compatible"
content="ie=edge"
/>
<title>Sign In</title>
</head>
<style>
:root {
color-scheme: dark light;
font-family: 'Inter', sans-serif;
}
</style>
<body>
<form>
<div>
<label for="email-input">Email</label>
<input
name="email"
id="email-input"
placeholder="Email"
type="email"
/>
</div>
<div>
<label for="password-input">Password</label>
<input
name="password"
id="password-input"
placeholder="Password"
/>
</div>
<div>
<label for="checkbox-input">Checkbox</label>
<input
required
name="checkbox"
type="checkbox"
id="checkbox-input"
value="true"
/>
</div>
<button type="button">Not submit</button>
<button>Submit</button>
</form>
</body>
<script>
const form = document.querySelector('form')
form.addEventListener('submit', (e) => {
e.preventDefault()
const formObj = Object.fromEntries(new FormData(form))
console.log(formObj)
})
</script>
</html>

57
server/index.ts Normal file
View File

@@ -0,0 +1,57 @@
const users = [
{
name: 'Andrei',
age: '39',
email: 'andres@gmail.com',
password: 'pass',
},
{
name: 'Katya',
age: '18',
email: 'katya@gmail.com',
password: 'another-pass',
},
]
Bun.serve({
port: 3333,
static: {
'/': new Response(await Bun.file('./index.html').bytes(), {
headers: {
'Content-Type': 'text/html',
},
}),
},
async fetch(req) {
const url = new URL(req.url)
if (url.pathname === '/login' && req.method === 'GET') {
const email = url.searchParams.get('email')
const password = url.searchParams.get('password')
const user = users.find(
(u) => u.email === email && u.password === password
)
if (!user) {
return new Response(await Bun.file('./login-incorrect.html').bytes(), {
headers: {
'Content-Type': 'text/html',
},
})
}
const file = await Bun.file('./dashboard.html').text()
return new Response(
file
.replaceAll('{{username}}', user.name)
.replaceAll('{{age}}', user.age),
{
headers: {
'Content-Type': 'text/html',
},
}
)
}
if (url.pathname === '/blog') return new Response('Blog!')
return new Response('404!')
},
})

View File

@@ -0,0 +1,21 @@
<!doctype html>
<html lang="en">
<head>
<meta charset="UTF-8" />
<meta
name="viewport"
content="width=device-width, user-scalable=no, initial-scale=1.0, maximum-scale=1.0, minimum-scale=1.0"
/>
<meta
http-equiv="X-UA-Compatible"
content="ie=edge"
/>
<title>Incorrect login</title>
</head>
<body>
<h1>
Your username or password were incorrect, please try again by
<a href="/login">clicking here</a>
</h1>
</body>
</html>

14
server/package.json Normal file
View File

@@ -0,0 +1,14 @@
{
"name": "server",
"module": "index.ts",
"type": "module",
"devDependencies": {
"@types/bun": "latest"
},
"scripts": {
"dev": "bun --watch run index.ts"
},
"peerDependencies": {
"typescript": "^5.0.0"
}
}

27
server/tsconfig.json Normal file
View File

@@ -0,0 +1,27 @@
{
"compilerOptions": {
// Enable latest features
"lib": ["ESNext", "DOM"],
"target": "ESNext",
"module": "ESNext",
"moduleDetection": "force",
"jsx": "react-jsx",
"allowJs": true,
// Bundler mode
"moduleResolution": "bundler",
"allowImportingTsExtensions": true,
"verbatimModuleSyntax": true,
"noEmit": true,
// Best practices
"strict": true,
"skipLibCheck": true,
"noFallthroughCasesInSwitch": true,
// Some stricter flags (disabled by default)
"noUnusedLocals": false,
"noUnusedParameters": false,
"noPropertyAccessFromIndexSignature": false
}
}

View File

@@ -0,0 +1,35 @@
import { Checkbox } from '@/components/ui/checkbox/checkbox'
import { ComponentPropsWithoutRef } from 'react'
import { Control, useController, FieldValues, FieldPath } from 'react-hook-form'
type Props<T extends FieldValues> = ComponentPropsWithoutRef<
typeof Checkbox
> & {
control: Control<T>
name: FieldPath<T>
}
export const FormCheckbox = <T extends FieldValues>({
control,
name,
errorMessage,
...props
}: Props<T>) => {
const {
field: { onChange, value, ...field },
fieldState: { error },
} = useController({
control,
name,
})
return (
<Checkbox
{...props}
onCheckedChange={onChange}
checked={value}
errorMessage={errorMessage ?? error?.message}
{...field}
/>
)
}

View File

@@ -0,0 +1,39 @@
'use client'
import * as React from 'react'
import * as CheckboxPrimitive from '@radix-ui/react-checkbox'
import { Check } from 'lucide-react'
import { cn } from '@/utils/cn'
import { ReactNode } from 'react'
const Checkbox = React.forwardRef<
React.ElementRef<typeof CheckboxPrimitive.Root>,
React.ComponentPropsWithoutRef<typeof CheckboxPrimitive.Root> & {
label?: ReactNode
errorMessage?: string
}
>(({ className, label, errorMessage, ...props }, ref) => (
<>
<label>
<CheckboxPrimitive.Root
ref={ref}
className={cn(
'peer h-4 w-4 shrink-0 rounded-sm border border-primary ring-offset-background focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:cursor-not-allowed disabled:opacity-50 data-[state=checked]:bg-primary data-[state=checked]:text-primary-foreground',
className
)}
{...props}
>
<CheckboxPrimitive.Indicator
className={cn('flex items-center justify-center text-current')}
>
<Check className='h-4 w-4' />
</CheckboxPrimitive.Indicator>
</CheckboxPrimitive.Root>
{label}
</label>
{errorMessage && <p className={'text-red-500 text-sm'}>{errorMessage}</p>}
</>
))
Checkbox.displayName = CheckboxPrimitive.Root.displayName
export { Checkbox }

View File

@@ -0,0 +1,42 @@
import {
ComponentPropsWithoutRef,
ElementRef,
forwardRef,
ReactNode,
useId,
} from 'react'
import { cn } from '@/utils/cn'
type Props = ComponentPropsWithoutRef<'input'> & {
errorMessage?: string
label?: ReactNode
}
export const TextField = forwardRef<ElementRef<'input'>, Props>(
({ errorMessage, label, className, id, ...rest }, ref) => {
const generatedId = useId()
const idToUse = id ?? generatedId
return (
<div className={'flex flex-col gap-0.5'}>
<label
htmlFor={idToUse}
className={'text-sm text-light-900'}
>
{label}
</label>
<input
{...rest}
className={cn('rounded-md p-4 border', className)}
id={idToUse}
ref={ref}
/>
{errorMessage && (
<p className={'text-red-500 text-sm'}>{errorMessage}</p>
)}
</div>
)
}
)
TextField.displayName = 'TextField'

View File

@@ -1,30 +1,85 @@
import Link from 'next/link'
import { Button, buttonVariants } from '@/components/ui/button/button'
import * as SelectPrimitive from '@radix-ui/react-select'
import { useForm } from 'react-hook-form'
import { z } from 'zod'
import { zodResolver } from '@hookform/resolvers/zod'
import { TextField } from '@/components/ui/text-field/text-field'
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from '@/components/ui/select/select'
const loginSchema = z.object({
email: z
.string()
.min(1, 'Required')
.email('Неверный адрес электронной почты'),
password: z
.string({ required_error: 'Required' })
.min(1, 'Required')
.min(3, 'Минимум 3 символа'),
})
const createLoginSchema = (t: ReturnType<typeof useI18n>) => {
return z.object({
email: z.string().min(1, 'Required').email(t('ERROR_INVALID_EMAIL')),
password: z
.string({ required_error: 'Required' })
.min(1, 'Required')
.min(3, 'Минимум 3 символа'),
})
}
type LoginFields = z.infer<typeof loginSchema>
export default function Login() {
const t = useI18n()
const {
handleSubmit,
register,
formState: { errors },
} = useForm<LoginFields>({
resolver: zodResolver(loginSchema),
})
const onSubmit = handleSubmit((data) => {
console.log(data)
})
return (
<div className={'gap-4 p-8'}>
<Select>
<SelectPrimitive.Trigger asChild>
<button>
<SelectValue placeholder='Theme' />
</button>
</SelectPrimitive.Trigger>
<SelectContent>
<SelectItem value='light'>Light</SelectItem>
<SelectItem value='dark'>Dark</SelectItem>
<SelectItem value='system'>System</SelectItem>
</SelectContent>
</Select>
<div className={'h-screen grid place-items-center'}>
<form
onSubmit={onSubmit}
className={'space-y-10'}
>
<TextField
placeholder='Email'
label={'Email'}
errorMessage={errors.email?.message}
{...register('email')}
/>
<TextField
placeholder='Password'
label={'Password'}
errorMessage={errors.password?.message}
{...register('password')}
/>
<button>Sign In</button>
</form>
</div>
)
}
const useI18n = () => {
const lang = 'ru'
return (key: keyof (typeof translations)['ru']) => {
return translations[lang][key]
}
}
const translations = {
ru: {
ERROR_INVALID_EMAIL: 'Введите валидный адрес эл. почты',
},
en: {
ERROR_INVALID_EMAIL: 'Invalid email',
},
} as const

View File

@@ -1,3 +1,106 @@
import { z } from 'zod'
import { TextField } from '@/components/ui/text-field/text-field'
import { useForm } from 'react-hook-form'
import { zodResolver } from '@hookform/resolvers/zod'
import { FormCheckbox } from '@/components/form/form-checkbox'
const signUpSchema = z
.object({
username: z.string(),
email: z.string().email(),
password: z.string(),
passwordConfirmation: z.string(),
agreesToTOS: z.literal(true, {
errorMap: () => ({ message: 'You have to accept our terms of service' }),
}),
})
.refine((value) => value.password === value.passwordConfirmation, {
message: 'Passwords do not match',
path: ['passwordConfirmation'],
})
type SignUpFields = z.infer<typeof signUpSchema>
export default function SignUp() {
return <div>SignUp</div>
const {
register,
handleSubmit,
formState: { errors },
control,
} = useForm<SignUpFields>({
resolver: zodResolver(signUpSchema),
})
const onSubmit = handleSubmit((data) => {
console.log(data)
})
console.log('render')
return (
<div className={'h-screen grid place-items-center'}>
<form
onSubmit={onSubmit}
className={'space-y-10'}
>
<TextField
placeholder='Username'
label={'Username'}
errorMessage={errors.username?.message}
{...register('username')}
/>
<TextField
placeholder='Email'
label={'Email'}
errorMessage={errors.email?.message}
{...register('email')}
/>
<TextField
placeholder='Password'
label={'Password'}
errorMessage={errors.password?.message}
type={'password'}
{...register('password')}
/>
<TextField
placeholder='Confirm password'
label={'Confirm password'}
type={'password'}
errorMessage={errors.passwordConfirmation?.message}
{...register('passwordConfirmation')}
/>
<div>
<FormCheckbox
className={'mr-3'}
label={'I agree to the Terms of Service and Privacy Policy'}
control={control}
name={'agreesToTOS'}
/>
</div>
{/*<div>*/}
{/* <label className={'block'}>*/}
{/* <Controller*/}
{/* render={({ field: { onChange, value, ...field } }) => {*/}
{/* return (*/}
{/* <Checkbox*/}
{/* className={'mr-3'}*/}
{/* {...field}*/}
{/* onCheckedChange={onChange}*/}
{/* checked={value}*/}
{/* />*/}
{/* )*/}
{/* }}*/}
{/* name={'agreesToTOS'}*/}
{/* control={control}*/}
{/* />*/}
{/* I agree to the Terms of Service and Privacy Policy*/}
{/* </label>*/}
{/* {errors.agreesToTOS && (*/}
{/* <p className={'text-red-500 text-sm'}>*/}
{/* {errors.agreesToTOS.message}*/}
{/* </p>*/}
{/* )}*/}
{/*</div>*/}
<button>Sign Up</button>
</form>
</div>
)
}