feat: add window with profile and logout buttons. add notification component

This commit is contained in:
2026-05-10 18:43:35 +03:00
parent 84e4fcdaec
commit 01685e3811
7 changed files with 273 additions and 20 deletions

View File

@@ -18,3 +18,7 @@ export function registrationStart(payload: RegistrationStartPayload): Promise<st
export function registrationComplete(payload: RegistrationCompletePayload): Promise<string> { export function registrationComplete(payload: RegistrationCompletePayload): Promise<string> {
return api.post<string>('/auth/registration/complete', payload) return api.post<string>('/auth/registration/complete', payload)
} }
export function logout(): Promise<void> {
return api.post<void>('/logout', {})
}

View File

@@ -0,0 +1,108 @@
.notification {
position: fixed;
bottom: 24px;
right: 24px;
z-index: 1000;
display: flex;
align-items: center;
gap: 12px;
padding: 16px 18px;
min-width: 280px;
max-width: 360px;
border-radius: 12px;
background: var(--bg-mid);
border: 1px solid var(--glass-border);
box-shadow: 0 8px 32px rgba(0, 0, 0, 0.4);
animation: slideIn 0.3s cubic-bezier(0.22, 1, 0.36, 1) forwards;
}
.notificationWrapper {
display: flex;
gap: 12px;
}
.notification.closing {
animation: slideOut 0.25s cubic-bezier(0.55, 0, 1, 0.45) forwards;
}
@keyframes slideIn {
from {
transform: translateX(calc(100% + 24px));
opacity: 0;
}
to {
transform: translateX(0);
opacity: 1;
}
}
@keyframes slideOut {
from {
transform: translateX(0);
opacity: 1;
}
to {
transform: translateX(calc(100% + 24px));
opacity: 0;
}
}
.icon {
flex-shrink: 0;
width: 22px;
height: 22px;
border-radius: 50%;
display: flex;
align-items: center;
justify-content: center;
font-size: 12px;
font-weight: 700;
margin-top: 1px;
}
.success .icon {
background: var(--success);
color: #fff;
}
.error .icon {
background: var(--error);
color: #fff;
}
.info .icon {
background: var(--interactive);
color: #fff;
}
.warning .icon {
background: #f59e0b;
color: #fff;
}
.message {
flex: 1;
font-size: 14px;
color: var(--text-secondary);
line-height: 1.5;
margin: 0;
}
.close {
flex-shrink: 0;
background: none;
border: none;
color: var(--text-secondary);
font-size: 14px;
cursor: pointer;
padding: 0;
line-height: 1;
margin-top: 2px;
transition: color 0.15s;
}
.close:hover {
color: var(--text-primary);
}

View File

@@ -0,0 +1,42 @@
import { useState } from 'react'
import styles from './Notification.module.css'
type Status = 'success' | 'error' | 'info' | 'warning'
interface Props {
message: string
status: Status
onClose: () => void
}
const ICONS: Record<Status, string> = {
success: '✓',
error: '✕',
info: 'i',
warning: '!',
}
export function Notification({ message, status, onClose }: Props) {
const [closing, setClosing] = useState(false)
function handleClose() {
setClosing(true)
}
function handleAnimationEnd() {
if (closing) onClose()
}
return (
<div
className={`${styles.notification} ${styles[status]} ${closing ? styles.closing : ''}`}
onAnimationEnd={handleAnimationEnd}
>
<div className={styles.notificationWrapper}>
<span className={styles.icon}>{ICONS[status]}</span>
<p className={styles.message}>{message}</p>
</div>
<button className={styles.close} onClick={handleClose}></button>
</div>
)
}

View File

@@ -0,0 +1 @@
export { Notification } from './Notification'

View File

@@ -1,5 +1,6 @@
export { Button } from './Button' export { Button } from './Button'
export { FormField } from './FormField' export { FormField } from './FormField'
export { Notification } from './Notification'
export { Pill } from './Pill' export { Pill } from './Pill'
export { PrimaryButton } from './PrimaryButton' export { PrimaryButton } from './PrimaryButton'
export { TokenIcon } from './TokenIcon' export { TokenIcon } from './TokenIcon'

View File

@@ -39,10 +39,55 @@
color: #ff4d4d; color: #ff4d4d;
} }
.accountWrapper {
position: relative;
}
.account { .account {
display: flex; display: flex;
align-items: center; align-items: center;
gap: 10px; gap: 10px;
background: none;
border: none;
cursor: pointer;
padding: 0;
}
.dropdown {
position: absolute;
top: calc(100% + 12px);
right: 0;
background: var(--bg-card, #1a1a2e);
border: 1px solid var(--glass-border);
border-radius: 10px;
overflow: hidden;
min-width: 180px;
z-index: 100;
display: flex;
flex-direction: column;
}
.dropdownItem {
display: block;
padding: 12px 16px;
font-size: 14px;
color: var(--text-secondary);
text-decoration: none;
background: none;
border: none;
cursor: pointer;
text-align: left;
width: 100%;
transition: background 0.15s, color 0.15s;
}
.dropdownItem:hover {
background: var(--glass-border);
color: var(--text-primary);
}
.dropdownItem.danger:hover {
color: var(--error);
} }
.avatar { .avatar {

View File

@@ -2,6 +2,11 @@ import logo from '@shared/assets/logo-full-white.png'
import styles from './WalletHeader.module.css' import styles from './WalletHeader.module.css'
import { Link } from 'react-router-dom' import { Link } from 'react-router-dom'
import { ROUTES } from '@shared/config/routes' import { ROUTES } from '@shared/config/routes'
import { useEffect, useRef, useState } from 'react'
import { useMutation } from '@tanstack/react-query'
import { useNavigate } from 'react-router-dom'
import { logout } from '@features/auth/api/registrationApi'
import { Notification } from '@shared/ui'
const TICKERS = [ const TICKERS = [
{ symbol: 'BTC', price: '$66,916.00', change: 0.12, }, { symbol: 'BTC', price: '$66,916.00', change: 0.12, },
@@ -10,26 +15,73 @@ const TICKERS = [
] ]
export function WalletHeader() { export function WalletHeader() {
const [open, setOpen] = useState(false)
const [error, setError] = useState(false)
const ref = useRef<HTMLDivElement>(null)
const navigate = useNavigate()
const { mutate: logoutMutate } = useMutation({
mutationFn: logout,
onSuccess: () => navigate(ROUTES.HOME),
onError: () => setError(true),
})
useEffect(() => {
function handleClickOutside(e: MouseEvent) {
if (ref.current && !ref.current.contains(e.target as Node)) {
setOpen(false)
}
}
document.addEventListener('mousedown', handleClickOutside)
return () => document.removeEventListener('mousedown', handleClickOutside)
}, [])
function handleLogout() {
logoutMutate()
setOpen(false)
}
return ( return (
<nav className={styles.nav}> <>
<a href="/" className={styles.logo}> <nav className={styles.nav}>
<img src={logo} alt="ЭКСА" /> <a href="/" className={styles.logo}>
</a> <img src={logo} alt="ЭКСА" />
<div className={styles.ticker}> </a>
{TICKERS.map(({ symbol, price, change }) => ( <div className={styles.ticker}>
<div key={symbol} className={styles.tick}> {TICKERS.map(({ symbol, price, change }) => (
<b>{symbol}</b> <div key={symbol} className={styles.tick}>
<span>{price}</span> <b>{symbol}</b>
<span className={change >= 0 ? styles.up : styles.dn}> <span>{price}</span>
{change >= 0 ? '+' : ''}{change}% <span className={change >= 0 ? styles.up : styles.dn}>
</span> {change >= 0 ? '+' : ''}{change}%
</div> </span>
))} </div>
</div> ))}
<Link to={ROUTES.PROFILE} className={styles.account}> </div>
<div className={styles.avatar} /> <div className={styles.accountWrapper} ref={ref}>
<span>Test account</span> <button className={styles.account} onClick={() => setOpen(v => !v)}>
</Link> <div className={styles.avatar} />
</nav> <span>Test account</span>
</button>
{open && (
<div className={styles.dropdown}>
<Link to={ROUTES.PROFILE} className={styles.dropdownItem} onClick={() => setOpen(false)}>
Личный кабинет
</Link>
<button className={`${styles.dropdownItem} ${styles.danger}`} onClick={handleLogout}>
Выйти
</button>
</div>
)}
</div>
</nav>
{error && (
<Notification
status="error"
message="Произошла ошибка сервера"
onClose={() => setError(false)}
/>
)}
</>
) )
} }