2025-12-13 12:35:03 +08:00
|
|
|
|
/**
|
|
|
|
|
|
* Account Page
|
|
|
|
|
|
* Cyberpunk Terminal Aesthetic
|
|
|
|
|
|
*/
|
|
|
|
|
|
|
2025-11-28 01:06:01 +08:00
|
|
|
|
import { useState, useEffect } from "react";
|
|
|
|
|
|
import { useNavigate } from "react-router-dom";
|
2026-01-05 14:45:00 +08:00
|
|
|
|
import { useAuth } from "@/shared/context/AuthContext";
|
2025-11-28 01:06:01 +08:00
|
|
|
|
import { Button } from "@/components/ui/button";
|
|
|
|
|
|
import { Input } from "@/components/ui/input";
|
|
|
|
|
|
import { Label } from "@/components/ui/label";
|
|
|
|
|
|
import { Avatar, AvatarFallback, AvatarImage } from "@/components/ui/avatar";
|
|
|
|
|
|
import { AlertDialog, AlertDialogAction, AlertDialogCancel, AlertDialogContent, AlertDialogDescription, AlertDialogFooter, AlertDialogHeader, AlertDialogTitle } from "@/components/ui/alert-dialog";
|
|
|
|
|
|
import {
|
|
|
|
|
|
User,
|
|
|
|
|
|
Mail,
|
|
|
|
|
|
Phone,
|
|
|
|
|
|
Shield,
|
|
|
|
|
|
Calendar,
|
|
|
|
|
|
Save,
|
|
|
|
|
|
KeyRound,
|
|
|
|
|
|
LogOut,
|
|
|
|
|
|
UserPlus,
|
2025-12-13 12:35:03 +08:00
|
|
|
|
GitBranch,
|
2025-12-26 20:51:00 +08:00
|
|
|
|
Terminal
|
2025-11-28 01:06:01 +08:00
|
|
|
|
} from "lucide-react";
|
|
|
|
|
|
import { apiClient } from "@/shared/api/serverClient";
|
|
|
|
|
|
import { toast } from "sonner";
|
|
|
|
|
|
import type { Profile } from "@/shared/types";
|
|
|
|
|
|
|
|
|
|
|
|
export default function Account() {
|
|
|
|
|
|
const navigate = useNavigate();
|
2026-01-05 14:45:00 +08:00
|
|
|
|
const { logout } = useAuth();
|
2025-11-28 01:06:01 +08:00
|
|
|
|
const [profile, setProfile] = useState<Profile | null>(null);
|
|
|
|
|
|
const [loading, setLoading] = useState(true);
|
|
|
|
|
|
const [saving, setSaving] = useState(false);
|
|
|
|
|
|
const [showLogoutDialog, setShowLogoutDialog] = useState(false);
|
|
|
|
|
|
const [form, setForm] = useState({
|
|
|
|
|
|
full_name: "",
|
|
|
|
|
|
phone: "",
|
|
|
|
|
|
github_username: "",
|
|
|
|
|
|
gitlab_username: "",
|
2026-01-05 13:47:48 +08:00
|
|
|
|
gitea_username: "",
|
2025-11-28 01:06:01 +08:00
|
|
|
|
});
|
|
|
|
|
|
const [passwordForm, setPasswordForm] = useState({
|
|
|
|
|
|
current_password: "",
|
|
|
|
|
|
new_password: "",
|
|
|
|
|
|
confirm_password: "",
|
|
|
|
|
|
});
|
|
|
|
|
|
const [changingPassword, setChangingPassword] = useState(false);
|
|
|
|
|
|
|
|
|
|
|
|
useEffect(() => {
|
|
|
|
|
|
loadProfile();
|
|
|
|
|
|
}, []);
|
|
|
|
|
|
|
|
|
|
|
|
const loadProfile = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
setLoading(true);
|
|
|
|
|
|
const res = await apiClient.get('/users/me');
|
|
|
|
|
|
setProfile(res.data);
|
|
|
|
|
|
setForm({
|
|
|
|
|
|
full_name: res.data.full_name || "",
|
|
|
|
|
|
phone: res.data.phone || "",
|
|
|
|
|
|
github_username: res.data.github_username || "",
|
|
|
|
|
|
gitlab_username: res.data.gitlab_username || "",
|
2026-01-05 13:47:48 +08:00
|
|
|
|
gitea_username: res.data.gitea_username || "",
|
2025-11-28 01:06:01 +08:00
|
|
|
|
});
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('Failed to load profile:', error);
|
|
|
|
|
|
toast.error("加载账号信息失败");
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setLoading(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleSave = async () => {
|
|
|
|
|
|
try {
|
|
|
|
|
|
setSaving(true);
|
|
|
|
|
|
const res = await apiClient.put('/users/me', form);
|
|
|
|
|
|
setProfile(res.data);
|
|
|
|
|
|
toast.success("账号信息已更新");
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('Failed to update profile:', error);
|
|
|
|
|
|
toast.error("更新失败");
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setSaving(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleChangePassword = async () => {
|
|
|
|
|
|
if (!passwordForm.new_password || !passwordForm.confirm_password) {
|
|
|
|
|
|
toast.error("请填写新密码");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (passwordForm.new_password !== passwordForm.confirm_password) {
|
|
|
|
|
|
toast.error("两次输入的密码不一致");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
if (passwordForm.new_password.length < 6) {
|
|
|
|
|
|
toast.error("密码长度至少6位");
|
|
|
|
|
|
return;
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
try {
|
|
|
|
|
|
setChangingPassword(true);
|
|
|
|
|
|
await apiClient.put('/users/me', { password: passwordForm.new_password });
|
|
|
|
|
|
toast.success("密码已更新");
|
|
|
|
|
|
setPasswordForm({ current_password: "", new_password: "", confirm_password: "" });
|
|
|
|
|
|
} catch (error) {
|
|
|
|
|
|
console.error('Failed to change password:', error);
|
|
|
|
|
|
toast.error("密码更新失败");
|
|
|
|
|
|
} finally {
|
|
|
|
|
|
setChangingPassword(false);
|
|
|
|
|
|
}
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const formatDate = (dateString?: string) => {
|
|
|
|
|
|
if (!dateString) return "-";
|
|
|
|
|
|
return new Date(dateString).toLocaleDateString('zh-CN', {
|
|
|
|
|
|
year: 'numeric',
|
|
|
|
|
|
month: 'long',
|
|
|
|
|
|
day: 'numeric'
|
|
|
|
|
|
});
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const getInitials = (name?: string, email?: string) => {
|
|
|
|
|
|
if (name) return name.charAt(0).toUpperCase();
|
|
|
|
|
|
if (email) return email.charAt(0).toUpperCase();
|
|
|
|
|
|
return "U";
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleLogout = () => {
|
2026-01-05 14:45:00 +08:00
|
|
|
|
logout();
|
2025-11-28 01:06:01 +08:00
|
|
|
|
toast.success("已退出登录");
|
|
|
|
|
|
navigate('/login');
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
const handleSwitchAccount = () => {
|
2026-01-05 14:45:00 +08:00
|
|
|
|
logout();
|
2025-11-28 01:06:01 +08:00
|
|
|
|
navigate('/login');
|
|
|
|
|
|
};
|
|
|
|
|
|
|
|
|
|
|
|
if (loading) {
|
|
|
|
|
|
return (
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="flex items-center justify-center min-h-screen cyber-bg-elevated">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="text-center space-y-4">
|
|
|
|
|
|
<div className="loading-spinner mx-auto" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<p className="text-muted-foreground font-mono text-sm uppercase tracking-wider">加载中...</p>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
return (
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="space-y-6 p-6 cyber-bg-elevated min-h-screen font-mono relative">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
{/* Grid background */}
|
|
|
|
|
|
<div className="absolute inset-0 cyber-grid-subtle pointer-events-none" />
|
2025-11-28 01:06:01 +08:00
|
|
|
|
|
|
|
|
|
|
<div className="relative z-10 grid grid-cols-1 lg:grid-cols-3 gap-6">
|
|
|
|
|
|
{/* Profile Card */}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="cyber-card p-0">
|
|
|
|
|
|
<div className="cyber-card-header">
|
|
|
|
|
|
<User className="w-5 h-5 text-primary" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<h3 className="text-lg font-bold uppercase tracking-wider text-foreground">用户信息</h3>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="p-6 text-center">
|
|
|
|
|
|
<div className="relative inline-block mb-4">
|
|
|
|
|
|
<Avatar className="w-24 h-24 border-2 border-primary/30">
|
|
|
|
|
|
<AvatarImage src={profile?.avatar_url} />
|
|
|
|
|
|
<AvatarFallback className="bg-primary/20 text-primary text-2xl font-bold">
|
|
|
|
|
|
{getInitials(profile?.full_name, profile?.email)}
|
|
|
|
|
|
</AvatarFallback>
|
|
|
|
|
|
</Avatar>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="absolute -bottom-1 -right-1 w-6 h-6 bg-emerald-500 rounded-full border-2 border-background flex items-center justify-center">
|
|
|
|
|
|
<div className="w-2 h-2 bg-foreground rounded-full animate-pulse" />
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<h4 className="text-lg font-bold text-foreground uppercase mb-1">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
{profile?.full_name || "未设置姓名"}
|
|
|
|
|
|
</h4>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<p className="text-muted-foreground text-sm">{profile?.email}</p>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="mt-6 pt-6 border-t border-border space-y-3 text-left">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="flex items-center gap-3 text-sm">
|
|
|
|
|
|
<Shield className="w-4 h-4 text-violet-400" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<span className="text-muted-foreground">角色:</span>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<span className="text-violet-400 font-bold uppercase">
|
|
|
|
|
|
{profile?.role === 'admin' ? '管理员' : '成员'}
|
|
|
|
|
|
</span>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</div>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="flex items-center gap-3 text-sm">
|
|
|
|
|
|
<Calendar className="w-4 h-4 text-sky-400" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<span className="text-muted-foreground">注册时间:</span>
|
|
|
|
|
|
<span className="text-foreground font-mono">{formatDate(profile?.created_at)}</span>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="mt-6 pt-6 border-t border-border space-y-2">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
variant="outline"
|
2025-11-28 16:16:29 +08:00
|
|
|
|
onClick={handleSwitchAccount}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="w-full cyber-btn-outline h-10"
|
2025-11-28 16:16:29 +08:00
|
|
|
|
>
|
|
|
|
|
|
<UserPlus className="w-4 h-4 mr-2" />
|
|
|
|
|
|
切换账号
|
|
|
|
|
|
</Button>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
variant="destructive"
|
2025-11-28 16:16:29 +08:00
|
|
|
|
onClick={() => setShowLogoutDialog(true)}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="w-full bg-rose-500/20 hover:bg-rose-500/30 text-rose-400 border border-rose-500/30 h-10"
|
2025-11-28 16:16:29 +08:00
|
|
|
|
>
|
|
|
|
|
|
<LogOut className="w-4 h-4 mr-2" />
|
|
|
|
|
|
退出登录
|
|
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
|
|
|
|
|
|
{/* Edit Form */}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="lg:col-span-2 cyber-card p-0">
|
|
|
|
|
|
<div className="cyber-card-header">
|
|
|
|
|
|
<Terminal className="w-5 h-5 text-primary" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<h3 className="text-lg font-bold uppercase tracking-wider text-foreground">基本信息</h3>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="p-6 space-y-6">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="email" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<Mail className="w-3 h-3" /> 邮箱
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="email"
|
|
|
|
|
|
value={profile?.email || ""}
|
|
|
|
|
|
disabled
|
2025-12-18 19:57:43 +08:00
|
|
|
|
className="cyber-input bg-muted text-muted-foreground cursor-not-allowed"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<p className="text-xs text-muted-foreground">邮箱不可修改</p>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="full_name" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<User className="w-3 h-3" /> 姓名
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="full_name"
|
|
|
|
|
|
value={form.full_name}
|
|
|
|
|
|
onChange={(e) => setForm({ ...form, full_name: e.target.value })}
|
|
|
|
|
|
placeholder="请输入姓名"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="phone" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<Phone className="w-3 h-3" /> 手机号
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="phone"
|
|
|
|
|
|
value={form.phone}
|
|
|
|
|
|
onChange={(e) => setForm({ ...form, phone: e.target.value })}
|
|
|
|
|
|
placeholder="请输入手机号"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<div className="pt-6 border-t border-border">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<h3 className="section-title text-sm mb-4 flex items-center gap-2">
|
|
|
|
|
|
<GitBranch className="w-4 h-4" />
|
|
|
|
|
|
代码托管账号
|
|
|
|
|
|
</h3>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-2 gap-4">
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="github" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<GitBranch className="w-3 h-3" /> GitHub 用户名
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="github"
|
|
|
|
|
|
value={form.github_username}
|
|
|
|
|
|
onChange={(e) => setForm({ ...form, github_username: e.target.value })}
|
|
|
|
|
|
placeholder="your-github-username"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="gitlab" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<GitBranch className="w-3 h-3" /> GitLab 用户名
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="gitlab"
|
|
|
|
|
|
value={form.gitlab_username}
|
|
|
|
|
|
onChange={(e) => setForm({ ...form, gitlab_username: e.target.value })}
|
|
|
|
|
|
placeholder="your-gitlab-username"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2026-01-05 13:47:48 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
|
|
|
|
|
<Label htmlFor="gitea" className="text-xs font-bold text-muted-foreground uppercase flex items-center gap-2">
|
|
|
|
|
|
<GitBranch className="w-3 h-3" /> Gitea 用户名
|
|
|
|
|
|
</Label>
|
|
|
|
|
|
<Input
|
|
|
|
|
|
id="gitea"
|
|
|
|
|
|
value={form.gitea_username}
|
|
|
|
|
|
onChange={(e) => setForm({ ...form, gitea_username: e.target.value })}
|
|
|
|
|
|
placeholder="your-gitea-username"
|
|
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
<div className="flex justify-end pt-4">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<Button onClick={handleSave} disabled={saving} className="cyber-btn-primary h-10">
|
|
|
|
|
|
{saving ? (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<div className="loading-spinner w-4 h-4 mr-2" />
|
|
|
|
|
|
保存中...
|
|
|
|
|
|
</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<Save className="w-4 h-4 mr-2" />
|
|
|
|
|
|
保存修改
|
|
|
|
|
|
</>
|
|
|
|
|
|
)}
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
|
|
|
|
|
|
{/* Password Change */}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<div className="lg:col-span-3 cyber-card p-0">
|
|
|
|
|
|
<div className="cyber-card-header">
|
|
|
|
|
|
<KeyRound className="w-5 h-5 text-amber-400" />
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<h3 className="text-lg font-bold uppercase tracking-wider text-foreground">修改密码</h3>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
<div className="p-6">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<div className="grid grid-cols-1 md:grid-cols-3 gap-4">
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="new_password" className="text-xs font-bold text-muted-foreground uppercase">新密码</Label>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<Input
|
|
|
|
|
|
id="new_password"
|
|
|
|
|
|
type="password"
|
|
|
|
|
|
value={passwordForm.new_password}
|
|
|
|
|
|
onChange={(e) => setPasswordForm({ ...passwordForm, new_password: e.target.value })}
|
|
|
|
|
|
placeholder="输入新密码"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="space-y-2">
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<Label htmlFor="confirm_password" className="text-xs font-bold text-muted-foreground uppercase">确认密码</Label>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<Input
|
|
|
|
|
|
id="confirm_password"
|
|
|
|
|
|
type="password"
|
|
|
|
|
|
value={passwordForm.confirm_password}
|
|
|
|
|
|
onChange={(e) => setPasswordForm({ ...passwordForm, confirm_password: e.target.value })}
|
|
|
|
|
|
placeholder="再次输入新密码"
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="cyber-input"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
/>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
<div className="flex items-end">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<Button
|
|
|
|
|
|
onClick={handleChangePassword}
|
|
|
|
|
|
disabled={changingPassword}
|
|
|
|
|
|
className="cyber-btn-outline h-10"
|
|
|
|
|
|
>
|
|
|
|
|
|
{changingPassword ? (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<div className="loading-spinner w-4 h-4 mr-2" />
|
|
|
|
|
|
更新中...
|
|
|
|
|
|
</>
|
|
|
|
|
|
) : (
|
|
|
|
|
|
<>
|
|
|
|
|
|
<KeyRound className="w-4 h-4 mr-2" />
|
|
|
|
|
|
更新密码
|
|
|
|
|
|
</>
|
|
|
|
|
|
)}
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</Button>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
</div>
|
2025-11-28 01:06:01 +08:00
|
|
|
|
</div>
|
|
|
|
|
|
|
|
|
|
|
|
{/* Logout Confirmation Dialog */}
|
|
|
|
|
|
<AlertDialog open={showLogoutDialog} onOpenChange={setShowLogoutDialog}>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<AlertDialogContent className="cyber-card border-rose-500/30 cyber-dialog">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
<AlertDialogHeader>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<AlertDialogTitle className="text-lg font-bold uppercase text-foreground flex items-center gap-2">
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<LogOut className="w-5 h-5 text-rose-400" />
|
|
|
|
|
|
确认退出登录?
|
|
|
|
|
|
</AlertDialogTitle>
|
2025-12-18 19:57:43 +08:00
|
|
|
|
<AlertDialogDescription className="text-muted-foreground">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
退出后需要重新登录才能访问系统。
|
|
|
|
|
|
</AlertDialogDescription>
|
|
|
|
|
|
</AlertDialogHeader>
|
|
|
|
|
|
<AlertDialogFooter>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<AlertDialogCancel className="cyber-btn-outline">
|
2025-11-28 01:06:01 +08:00
|
|
|
|
取消
|
|
|
|
|
|
</AlertDialogCancel>
|
2025-12-13 12:35:03 +08:00
|
|
|
|
<AlertDialogAction
|
2025-11-28 01:06:01 +08:00
|
|
|
|
onClick={handleLogout}
|
2025-12-13 12:35:03 +08:00
|
|
|
|
className="bg-rose-500/20 hover:bg-rose-500/30 text-rose-400 border border-rose-500/30"
|
2025-11-28 01:06:01 +08:00
|
|
|
|
>
|
|
|
|
|
|
确认退出
|
|
|
|
|
|
</AlertDialogAction>
|
|
|
|
|
|
</AlertDialogFooter>
|
|
|
|
|
|
</AlertDialogContent>
|
|
|
|
|
|
</AlertDialog>
|
|
|
|
|
|
</div>
|
|
|
|
|
|
);
|
|
|
|
|
|
}
|