- 创建 Toast 通知组件,替换所有 alert() 调用 - 修复 useReview hook 内存泄漏(setInterval 清理) - 移除所有 console.error 和 console.log 语句 - 为复制操作失败添加用户友好的 toast 提示 Co-Authored-By: Claude Opus 4.5 <noreply@anthropic.com>
278 lines
8.5 KiB
TypeScript
278 lines
8.5 KiB
TypeScript
'use client'
|
|
|
|
import React, { useState } from 'react'
|
|
import { useRouter } from 'next/navigation'
|
|
import {
|
|
CircleUser,
|
|
Settings,
|
|
BellRing,
|
|
History,
|
|
MessageCircleQuestion,
|
|
PlusCircle,
|
|
ChevronRight,
|
|
LogOut,
|
|
Copy,
|
|
Check
|
|
} from 'lucide-react'
|
|
import { ResponsiveLayout } from '@/components/layout/ResponsiveLayout'
|
|
import { cn } from '@/lib/utils'
|
|
import { useToast } from '@/components/ui/Toast'
|
|
|
|
// 用户数据
|
|
const mockUser = {
|
|
name: '李小红',
|
|
initial: '李',
|
|
creatorId: 'CR123456', // 达人ID
|
|
role: '抖音达人 · 已认证',
|
|
stats: {
|
|
completed: 28,
|
|
passRate: 92,
|
|
inProgress: 3,
|
|
},
|
|
}
|
|
|
|
// 菜单项数据
|
|
const menuItems = [
|
|
{
|
|
id: 'personal',
|
|
icon: CircleUser,
|
|
iconColor: 'text-accent-indigo',
|
|
bgColor: 'bg-accent-indigo',
|
|
title: '个人信息',
|
|
subtitle: '头像、昵称、绑定账号',
|
|
},
|
|
{
|
|
id: 'account',
|
|
icon: Settings,
|
|
iconColor: 'text-accent-green',
|
|
bgColor: 'bg-accent-green',
|
|
title: '账户设置',
|
|
subtitle: '修改密码、账号安全',
|
|
},
|
|
{
|
|
id: 'notification',
|
|
icon: BellRing,
|
|
iconColor: 'text-accent-blue',
|
|
bgColor: 'bg-accent-blue',
|
|
title: '消息设置',
|
|
subtitle: '通知开关、提醒偏好',
|
|
},
|
|
{
|
|
id: 'history',
|
|
icon: History,
|
|
iconColor: 'text-accent-coral',
|
|
bgColor: 'bg-accent-coral',
|
|
title: '历史记录',
|
|
subtitle: '已完成和过期的任务',
|
|
},
|
|
{
|
|
id: 'help',
|
|
icon: MessageCircleQuestion,
|
|
iconColor: 'text-text-secondary',
|
|
bgColor: 'bg-bg-elevated',
|
|
title: '帮助与反馈',
|
|
subtitle: '常见问题、联系客服',
|
|
},
|
|
{
|
|
id: 'appeal',
|
|
icon: PlusCircle,
|
|
iconColor: 'text-accent-indigo',
|
|
bgColor: 'bg-accent-indigo',
|
|
title: '申诉次数',
|
|
subtitle: '查看各任务申诉次数 · 申请增加',
|
|
},
|
|
]
|
|
|
|
// 用户卡片组件
|
|
function UserCard() {
|
|
const toast = useToast()
|
|
const [copied, setCopied] = useState(false)
|
|
|
|
// 复制达人ID
|
|
const handleCopyId = async () => {
|
|
try {
|
|
await navigator.clipboard.writeText(mockUser.creatorId)
|
|
setCopied(true)
|
|
setTimeout(() => setCopied(false), 2000)
|
|
} catch {
|
|
toast.error('复制失败,请重试')
|
|
}
|
|
}
|
|
|
|
return (
|
|
<div className="bg-bg-card rounded-2xl p-6 card-shadow flex flex-col gap-5">
|
|
{/* 头像和信息 */}
|
|
<div className="flex items-center gap-5">
|
|
{/* 头像 */}
|
|
<div
|
|
className="w-20 h-20 rounded-full flex items-center justify-center"
|
|
style={{
|
|
background: 'linear-gradient(135deg, #6366F1 0%, #4F46E5 100%)',
|
|
}}
|
|
>
|
|
<span className="text-[32px] font-bold text-white">{mockUser.initial}</span>
|
|
</div>
|
|
{/* 用户信息 */}
|
|
<div className="flex flex-col gap-1.5">
|
|
<span className="text-xl font-semibold text-text-primary">{mockUser.name}</span>
|
|
<span className="text-sm text-text-secondary">{mockUser.role}</span>
|
|
{/* 达人ID */}
|
|
<div className="flex items-center gap-2 mt-1">
|
|
<span className="text-xs text-text-tertiary">达人ID:</span>
|
|
<div className="flex items-center gap-1.5 px-2 py-1 rounded-md bg-bg-elevated">
|
|
<span className="text-xs font-mono font-medium text-accent-indigo">{mockUser.creatorId}</span>
|
|
<button
|
|
type="button"
|
|
onClick={handleCopyId}
|
|
className="p-0.5 hover:bg-bg-card rounded transition-colors"
|
|
title={copied ? '已复制' : '复制达人ID'}
|
|
>
|
|
{copied ? (
|
|
<Check size={12} className="text-accent-green" />
|
|
) : (
|
|
<Copy size={12} className="text-text-tertiary hover:text-text-secondary" />
|
|
)}
|
|
</button>
|
|
</div>
|
|
{copied && (
|
|
<span className="text-xs text-accent-green animate-fade-in">已复制</span>
|
|
)}
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
{/* 统计数据 */}
|
|
<div className="flex items-center justify-around pt-4 border-t border-border-subtle">
|
|
<div className="flex flex-col items-center gap-1">
|
|
<span className="text-2xl font-bold text-text-primary">{mockUser.stats.completed}</span>
|
|
<span className="text-xs text-text-secondary">完成任务</span>
|
|
</div>
|
|
<div className="flex flex-col items-center gap-1">
|
|
<span className="text-2xl font-bold text-accent-green">{mockUser.stats.passRate}%</span>
|
|
<span className="text-xs text-text-secondary">通过率</span>
|
|
</div>
|
|
<div className="flex flex-col items-center gap-1">
|
|
<span className="text-2xl font-bold text-accent-indigo">{mockUser.stats.inProgress}</span>
|
|
<span className="text-xs text-text-secondary">进行中</span>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
// 菜单项组件
|
|
function MenuItem({ item, onClick }: { item: typeof menuItems[0]; onClick: () => void }) {
|
|
const Icon = item.icon
|
|
const isPlainBg = item.bgColor === 'bg-bg-elevated'
|
|
|
|
return (
|
|
<button
|
|
type="button"
|
|
onClick={onClick}
|
|
className="flex items-center justify-between py-4 w-full text-left hover:bg-bg-elevated/30 transition-colors rounded-lg px-2 -mx-2"
|
|
>
|
|
<div className="flex items-center gap-4">
|
|
{/* 图标背景 */}
|
|
<div
|
|
className={cn(
|
|
'w-10 h-10 rounded-[10px] flex items-center justify-center',
|
|
isPlainBg ? item.bgColor : `${item.bgColor}/15`
|
|
)}
|
|
>
|
|
<Icon size={20} className={item.iconColor} />
|
|
</div>
|
|
{/* 文字 */}
|
|
<div className="flex flex-col gap-0.5">
|
|
<span className="text-[15px] font-medium text-text-primary">{item.title}</span>
|
|
<span className="text-[13px] text-text-tertiary">{item.subtitle}</span>
|
|
</div>
|
|
</div>
|
|
<ChevronRight size={20} className="text-text-tertiary" />
|
|
</button>
|
|
)
|
|
}
|
|
|
|
// 菜单卡片组件
|
|
function MenuCard({ onMenuClick }: { onMenuClick: (id: string) => void }) {
|
|
return (
|
|
<div className="bg-bg-card rounded-2xl p-6 card-shadow flex flex-col">
|
|
{menuItems.map((item, index) => (
|
|
<div key={item.id}>
|
|
<MenuItem item={item} onClick={() => onMenuClick(item.id)} />
|
|
{index < menuItems.length - 1 && (
|
|
<div className="h-px bg-border-subtle" />
|
|
)}
|
|
</div>
|
|
))}
|
|
</div>
|
|
)
|
|
}
|
|
|
|
// 退出卡片组件
|
|
function LogoutCard({ onLogout }: { onLogout: () => void }) {
|
|
return (
|
|
<div className="bg-bg-card rounded-2xl p-6 card-shadow">
|
|
<button
|
|
type="button"
|
|
onClick={onLogout}
|
|
className="w-full flex items-center justify-center gap-2 py-4 rounded-xl border-[1.5px] border-accent-coral text-accent-coral font-medium hover:bg-accent-coral/10 transition-colors"
|
|
>
|
|
<LogOut size={20} />
|
|
<span>退出登录</span>
|
|
</button>
|
|
</div>
|
|
)
|
|
}
|
|
|
|
export default function CreatorProfilePage() {
|
|
const router = useRouter()
|
|
|
|
// 菜单项点击处理
|
|
const handleMenuClick = (menuId: string) => {
|
|
const routes: Record<string, string> = {
|
|
personal: '/creator/profile/edit',
|
|
account: '/creator/settings/account',
|
|
notification: '/creator/settings/notification',
|
|
history: '/creator/history',
|
|
help: '/creator/help',
|
|
appeal: '/creator/appeal-quota',
|
|
}
|
|
const route = routes[menuId]
|
|
if (route) {
|
|
router.push(route)
|
|
}
|
|
}
|
|
|
|
// 退出登录
|
|
const handleLogout = () => {
|
|
// TODO: 实际退出逻辑
|
|
router.push('/login')
|
|
}
|
|
|
|
return (
|
|
<ResponsiveLayout role="creator">
|
|
<div className="flex flex-col gap-6 h-full">
|
|
{/* 顶部栏 */}
|
|
<div className="flex flex-col gap-1">
|
|
<h1 className="text-2xl lg:text-[28px] font-bold text-text-primary">个人中心</h1>
|
|
<p className="text-sm lg:text-[15px] text-text-secondary">管理您的账户信息和偏好设置</p>
|
|
</div>
|
|
|
|
{/* 内容区 - 响应式布局 */}
|
|
<div className="flex flex-col lg:flex-row gap-6 flex-1 min-h-0 overflow-y-auto lg:overflow-visible">
|
|
{/* 用户卡片 */}
|
|
<div className="lg:w-[360px] lg:flex-shrink-0">
|
|
<UserCard />
|
|
</div>
|
|
|
|
{/* 菜单和退出 */}
|
|
<div className="flex-1 flex flex-col gap-5 lg:overflow-y-auto lg:pr-2">
|
|
<MenuCard onMenuClick={handleMenuClick} />
|
|
<LogoutCard onLogout={handleLogout} />
|
|
</div>
|
|
</div>
|
|
</div>
|
|
</ResponsiveLayout>
|
|
)
|
|
}
|