7.5 KiB
7.5 KiB
Настройки профиля в Ласточке
Обзор
Настройки профиля позволяют пользователю управлять своей учётной записью:
- 👤 Профиль — имя, аватар, био
- 🔒 Безопасность — смена пароля
- 📱 Контакты — email и телефон
- ⚙️ Приватность — настройки видимости
Компонент ProfileSettings
Вкладки
1. Профиль
Поля:
- Аватар — загрузка изображения (макс 5MB)
- Отображаемое имя — как вас видят другие
- О себе — краткая информация
Функции:
- Предпросмотр аватара
- Редактирование полей
- Сохранение изменений
- Отмена редактирования
2. Безопасность
Поля:
- Текущий пароль
- Новый пароль (минимум 6 символов)
- Подтверждение пароля
Функции:
- Показать/скрыть пароль
- Валидация сложности пароля
- Проверка совпадения паролей
Props
interface ProfileSettingsProps {
onClose?: () => void
}
API
updateProfile
Обновление информации о профиле.
import { updateProfile } from '@/lib/email-auth'
const result = await updateProfile({
displayName: 'Новое имя',
avatar: 'data:image/png;base64,...',
bio: 'О себе',
})
if (result.success) {
console.log('Профиль обновлён')
} else {
console.error(result.error)
}
changePassword
Смена пароля.
import { changePassword } from '@/lib/email-auth'
const result = await changePassword('old-password', 'new-password')
if (result.success) {
console.log('Пароль изменён')
} else {
console.error(result.error)
}
Загрузка аватара
Обработка файла
const handleAvatarUpload = (e: React.ChangeEvent<HTMLInputElement>) => {
const file = e.target.files?.[0]
if (!file) return
// Проверка размера (макс 5MB)
if (file.size > 5 * 1024 * 1024) {
setError('Размер файла не должен превышать 5MB')
return
}
// Проверка типа
if (!file.type.startsWith('image/')) {
setError('Загрузите изображение')
return
}
// Конвертация в base64
const reader = new FileReader()
reader.onload = (event) => {
setAvatar(event.target?.result as string)
}
reader.readAsDataURL(file)
}
Требования к изображению
| Параметр | Значение |
|---|---|
| Формат | PNG, JPEG, GIF, WebP |
| Размер | до 5 MB |
| Разрешение | от 100x100 px |
| Соотношение | 1:1 (квадрат) |
Валидация
Имя
if (name.trim().length < 2) {
error: 'Имя должно быть не менее 2 символов'
}
Пароль
// Минимальная длина
if (newPassword.length < 6) {
error: 'Пароль должен быть не менее 6 символов'
}
// Совпадение
if (newPassword !== confirmPassword) {
error: 'Пароли не совпадают'
}
// Текущий пароль
if (!currentPassword) {
error: 'Введите текущий пароль'
}
Интеграция с Tinode
Обновление профиля
const me = tn.getMeTopic()
await me.setMeta({
desc: {
public: {
fn: displayName, // Отображаемое имя
photo: { // Аватар
type: 'image',
data: avatarData,
},
note: bio, // Био
},
},
})
Смена пароля
await tn.setMeta({
private: {
password: {
old: oldPassword,
new: newPassword,
},
},
})
Примеры использования
Открытие настроек
import { useState } from 'react'
import ProfileSettings from '@/components/ui/ProfileSettings'
function App() {
const [showSettings, setShowSettings] = useState(false)
return (
<>
<button onClick={() => setShowSettings(true)}>
Настройки
</button>
{showSettings && (
<ProfileSettings onClose={() => setShowSettings(false)} />
)}
</>
)
}
Интеграция в Sidebar
import ProfileSettings from '@/components/ui/ProfileSettings'
function Sidebar() {
const [showSettings, setShowSettings] = useState(false)
return (
<>
{/* Кнопка настроек */}
<button onClick={() => setShowSettings(true)}>
<Settings size={20} />
</button>
{/* Модальное окно */}
{showSettings && (
<div className="modal">
<ProfileSettings onClose={() => setShowSettings(false)} />
</div>
)}
</>
)
}
Состояния
Успешное обновление
const [success, setSuccess] = useState('')
if (result.success) {
setSuccess('Профиль успешно обновлён')
setTimeout(() => setSuccess(''), 3000)
}
Ошибка
const [error, setError] = useState('')
if (!result.success) {
setError(result.error || 'Ошибка обновления')
}
Загрузка
const [isLoading, setIsLoading] = useState(false)
setIsLoading(true)
try {
await updateProfile({...})
} finally {
setIsLoading(false)
}
Советы по UX
1. Debounced сохранение
Автоматическое сохранение через 1 секунду после последнего изменения:
useEffect(() => {
const timer = setTimeout(async () => {
if (isDirty) {
await saveProfile()
}
}, 1000)
return () => clearTimeout(timer)
}, [name, bio, isDirty])
2. Предпросмотр изменений
Показ изменений до сохранения:
const [preview, setPreview] = useState({
name: displayName,
bio: bio,
avatar: avatar,
})
3. Подтверждение важных действий
Запрос подтверждения перед сменой пароля:
const handleChangePassword = () => {
if (!window.confirm('Вы уверены, что хотите изменить пароль?')) {
return
}
// Смена пароля
}
Безопасность
Требования к паролю
- Минимум 6 символов
- Рекомендуется: буквы + цифры
- Не рекомендуется: простые комбинации (123456, password)
Защита от CSRF
Все запросы на изменение данных должны включать CSRF-токен.
Сессии
При смене пароля:
- Завершить все другие сессии
- Отправить уведомление на email
- Запросить повторный вход на других устройствах
Ласточка — народный мессенджер с открытым кодом 🕊️