better profie load

This commit is contained in:
songtianlun 2025-07-28 23:19:23 +08:00
parent 0288bf8eda
commit a58b9222c3
3 changed files with 473 additions and 266 deletions

View File

@ -9,6 +9,8 @@ import { Input } from '@/components/ui/input'
import { Label } from '@/components/ui/label'
import { Textarea } from '@/components/ui/textarea'
import { Avatar } from '@/components/ui/avatar'
import { LoadingSpinner, LoadingOverlay } from '@/components/ui/loading-spinner'
import { AvatarSkeleton, FormFieldSkeleton, TextAreaSkeleton } from '@/components/ui/skeleton'
import { Camera, Save, Eye, EyeOff, Globe } from 'lucide-react'
interface UserProfile {
@ -45,6 +47,9 @@ export default function ProfilePage() {
})
const [saveStatus, setSaveStatus] = useState<{type: 'success' | 'error' | null, message: string}>({type: null, message: ''})
const [isLoading, setIsLoading] = useState(false)
const [profileLoading, setProfileLoading] = useState(true)
const [fieldLoading, setFieldLoading] = useState<{[key: string]: boolean}>({})
const [avatarUploading, setAvatarUploading] = useState(false)
const supabase = createClient()
@ -57,6 +62,7 @@ export default function ProfilePage() {
const loadProfile = async () => {
if (!user) return
setProfileLoading(true)
try {
// Get user metadata and profile data
const { data: { user: userData }, error: userError } = await supabase.auth.getUser()
@ -85,12 +91,15 @@ export default function ProfilePage() {
} catch (error) {
console.error('Error loading profile:', error)
setSaveStatus({type: 'error', message: 'Failed to load profile'})
} finally {
setProfileLoading(false)
}
}
const updateProfile = async (field: string, value: string) => {
if (!user) return
setFieldLoading(prev => ({ ...prev, [field]: true }))
setIsLoading(true)
setSaveStatus({type: null, message: ''})
@ -120,6 +129,7 @@ export default function ProfilePage() {
setSaveStatus({type: 'error', message: (error instanceof Error ? error.message : 'Unknown error') || `Failed to update ${field}`})
} finally {
setIsLoading(false)
setFieldLoading(prev => ({ ...prev, [field]: false }))
}
}
@ -134,6 +144,7 @@ export default function ProfilePage() {
return
}
setFieldLoading(prev => ({ ...prev, password: true }))
setIsLoading(true)
setSaveStatus({type: null, message: ''})
@ -157,6 +168,7 @@ export default function ProfilePage() {
setSaveStatus({type: 'error', message: (error instanceof Error ? error.message : 'Unknown error') || 'Failed to update password'})
} finally {
setIsLoading(false)
setFieldLoading(prev => ({ ...prev, password: false }))
}
}
@ -164,6 +176,7 @@ export default function ProfilePage() {
const file = event.target.files?.[0]
if (!file || !user) return
setAvatarUploading(true)
setIsLoading(true)
setSaveStatus({type: null, message: ''})
@ -187,12 +200,14 @@ export default function ProfilePage() {
await loadProfile()
setSaveStatus({type: 'success', message: 'Avatar updated successfully'})
setIsLoading(false)
setAvatarUploading(false)
}
reader.readAsDataURL(file)
} catch (error: unknown) {
setSaveStatus({type: 'error', message: (error instanceof Error ? error.message : 'Unknown error') || 'Failed to upload avatar'})
setIsLoading(false)
setAvatarUploading(false)
}
}
@ -241,6 +256,10 @@ export default function ProfilePage() {
<div className="grid grid-cols-1 lg:grid-cols-3 gap-8">
{/* Avatar Section */}
<div className="lg:col-span-1">
{profileLoading ? (
<AvatarSkeleton />
) : (
<LoadingOverlay isLoading={avatarUploading}>
<div className="bg-card p-6 rounded-lg border border-border">
<h2 className="text-xl font-semibold text-foreground mb-4">Profile Picture</h2>
<div className="flex flex-col items-center">
@ -251,27 +270,37 @@ export default function ProfilePage() {
size={96}
className="w-24 h-24"
/>
<label className="absolute -bottom-2 -right-2 bg-primary text-primary-foreground p-2 rounded-full cursor-pointer hover:bg-primary/90">
<label className={`absolute -bottom-2 -right-2 bg-primary text-primary-foreground p-2 rounded-full cursor-pointer hover:bg-primary/90 ${avatarUploading ? 'pointer-events-none opacity-50' : ''}`}>
{avatarUploading ? (
<LoadingSpinner size="sm" className="text-primary-foreground" />
) : (
<Camera className="w-4 h-4" />
)}
<input
type="file"
accept="image/*"
className="hidden"
onChange={uploadAvatar}
disabled={isLoading}
disabled={isLoading || avatarUploading}
/>
</label>
</div>
<p className="text-sm text-muted-foreground mt-2 text-center">
Click the camera icon to upload a new picture
{avatarUploading ? 'Uploading...' : 'Click the camera icon to upload a new picture'}
</p>
</div>
</div>
</LoadingOverlay>
)}
</div>
{/* Profile Information */}
<div className="lg:col-span-2 space-y-6">
{/* Username */}
{profileLoading ? (
<FormFieldSkeleton />
) : (
<LoadingOverlay isLoading={fieldLoading.username}>
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold text-foreground">Username</h3>
@ -280,6 +309,7 @@ export default function ProfilePage() {
variant="outline"
size="sm"
onClick={() => setIsEditing(prev => ({ ...prev, username: true }))}
disabled={isLoading || avatarUploading}
>
Edit
</Button>
@ -292,14 +322,19 @@ export default function ProfilePage() {
value={formData.username}
onChange={(e) => setFormData(prev => ({ ...prev, username: e.target.value }))}
placeholder="Enter username"
disabled={fieldLoading.username}
/>
<div className="flex gap-2">
<Button
size="sm"
onClick={() => updateProfile('username', formData.username)}
disabled={isLoading}
disabled={isLoading || fieldLoading.username}
>
{fieldLoading.username ? (
<LoadingSpinner size="sm" className="mr-2" />
) : (
<Save className="w-4 h-4 mr-2" />
)}
Save
</Button>
<Button
@ -309,6 +344,7 @@ export default function ProfilePage() {
setIsEditing(prev => ({ ...prev, username: false }))
setFormData(prev => ({ ...prev, username: profile?.username || '' }))
}}
disabled={fieldLoading.username}
>
Cancel
</Button>
@ -318,8 +354,14 @@ export default function ProfilePage() {
<p className="text-foreground">{profile?.username || 'No username set'}</p>
)}
</div>
</LoadingOverlay>
)}
{/* Email */}
{profileLoading ? (
<FormFieldSkeleton />
) : (
<LoadingOverlay isLoading={fieldLoading.email}>
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold text-foreground">Email</h3>
@ -328,6 +370,7 @@ export default function ProfilePage() {
variant="outline"
size="sm"
onClick={() => setIsEditing(prev => ({ ...prev, email: true }))}
disabled={isLoading || avatarUploading}
>
Edit
</Button>
@ -341,14 +384,19 @@ export default function ProfilePage() {
value={formData.email}
onChange={(e) => setFormData(prev => ({ ...prev, email: e.target.value }))}
placeholder="Enter email"
disabled={fieldLoading.email}
/>
<div className="flex gap-2">
<Button
size="sm"
onClick={() => updateProfile('email', formData.email)}
disabled={isLoading}
disabled={isLoading || fieldLoading.email}
>
{fieldLoading.email ? (
<LoadingSpinner size="sm" className="mr-2" />
) : (
<Save className="w-4 h-4 mr-2" />
)}
Save
</Button>
<Button
@ -358,6 +406,7 @@ export default function ProfilePage() {
setIsEditing(prev => ({ ...prev, email: false }))
setFormData(prev => ({ ...prev, email: profile?.email || '' }))
}}
disabled={fieldLoading.email}
>
Cancel
</Button>
@ -367,8 +416,14 @@ export default function ProfilePage() {
<p className="text-foreground">{profile?.email}</p>
)}
</div>
</LoadingOverlay>
)}
{/* Bio */}
{profileLoading ? (
<TextAreaSkeleton />
) : (
<LoadingOverlay isLoading={fieldLoading.bio}>
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold text-foreground">Bio</h3>
@ -377,6 +432,7 @@ export default function ProfilePage() {
variant="outline"
size="sm"
onClick={() => setIsEditing(prev => ({ ...prev, bio: true }))}
disabled={isLoading || avatarUploading}
>
Edit
</Button>
@ -391,6 +447,7 @@ export default function ProfilePage() {
placeholder="Tell us about yourself"
className="min-h-[100px]"
maxLength={500}
disabled={fieldLoading.bio}
/>
<div className="text-sm text-muted-foreground text-right">
{formData.bio.length}/500 characters
@ -399,9 +456,13 @@ export default function ProfilePage() {
<Button
size="sm"
onClick={() => updateProfile('bio', formData.bio)}
disabled={isLoading}
disabled={isLoading || fieldLoading.bio}
>
{fieldLoading.bio ? (
<LoadingSpinner size="sm" className="mr-2" />
) : (
<Save className="w-4 h-4 mr-2" />
)}
Save
</Button>
<Button
@ -411,6 +472,7 @@ export default function ProfilePage() {
setIsEditing(prev => ({ ...prev, bio: false }))
setFormData(prev => ({ ...prev, bio: profile?.bio || '' }))
}}
disabled={fieldLoading.bio}
>
Cancel
</Button>
@ -420,12 +482,22 @@ export default function ProfilePage() {
<p className="text-foreground">{profile?.bio || 'No bio added yet'}</p>
)}
</div>
</LoadingOverlay>
)}
{/* Language */}
{profileLoading ? (
<FormFieldSkeleton />
) : (
<LoadingOverlay isLoading={fieldLoading.language}>
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold text-foreground">Language</h3>
{fieldLoading.language ? (
<LoadingSpinner size="sm" />
) : (
<Globe className="w-5 h-5 text-muted-foreground" />
)}
</div>
<select
@ -436,14 +508,20 @@ export default function ProfilePage() {
updateProfile('language', newLanguage)
}}
className="w-full px-3 py-2 border border-border rounded-md bg-input text-foreground focus:outline-none focus:ring-2 focus:ring-ring"
disabled={isLoading}
disabled={isLoading || fieldLoading.language || avatarUploading}
>
<option value="en">English</option>
<option value="zh"></option>
</select>
</div>
</LoadingOverlay>
)}
{/* Password */}
{profileLoading ? (
<FormFieldSkeleton />
) : (
<LoadingOverlay isLoading={fieldLoading.password}>
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<h3 className="text-lg font-semibold text-foreground">Password</h3>
@ -452,6 +530,7 @@ export default function ProfilePage() {
variant="outline"
size="sm"
onClick={() => setIsEditing(prev => ({ ...prev, password: true }))}
disabled={isLoading || avatarUploading}
>
Change Password
</Button>
@ -470,11 +549,13 @@ export default function ProfilePage() {
onChange={(e) => setFormData(prev => ({ ...prev, newPassword: e.target.value }))}
placeholder="Enter new password"
className="pr-10"
disabled={fieldLoading.password}
/>
<button
type="button"
onClick={() => setShowPasswords(prev => ({ ...prev, new: !prev.new }))}
className="absolute right-3 top-1/2 transform -translate-y-1/2 text-muted-foreground hover:text-foreground"
disabled={fieldLoading.password}
>
{showPasswords.new ? <EyeOff className="h-4 w-4" /> : <Eye className="h-4 w-4" />}
</button>
@ -491,11 +572,13 @@ export default function ProfilePage() {
onChange={(e) => setFormData(prev => ({ ...prev, confirmPassword: e.target.value }))}
placeholder="Confirm new password"
className="pr-10"
disabled={fieldLoading.password}
/>
<button
type="button"
onClick={() => setShowPasswords(prev => ({ ...prev, confirm: !prev.confirm }))}
className="absolute right-3 top-1/2 transform -translate-y-1/2 text-muted-foreground hover:text-foreground"
disabled={fieldLoading.password}
>
{showPasswords.confirm ? <EyeOff className="h-4 w-4" /> : <Eye className="h-4 w-4" />}
</button>
@ -506,9 +589,13 @@ export default function ProfilePage() {
<Button
size="sm"
onClick={updatePassword}
disabled={isLoading}
disabled={isLoading || fieldLoading.password}
>
{fieldLoading.password ? (
<LoadingSpinner size="sm" className="mr-2" />
) : (
<Save className="w-4 h-4 mr-2" />
)}
Update Password
</Button>
<Button
@ -523,6 +610,7 @@ export default function ProfilePage() {
confirmPassword: ''
}))
}}
disabled={fieldLoading.password}
>
Cancel
</Button>
@ -532,6 +620,8 @@ export default function ProfilePage() {
<p className="text-muted-foreground"></p>
)}
</div>
</LoadingOverlay>
)}
</div>
</div>
</div>

View File

@ -0,0 +1,47 @@
import { cn } from '@/lib/utils'
import { Loader2 } from 'lucide-react'
interface LoadingSpinnerProps {
size?: 'sm' | 'md' | 'lg'
className?: string
}
export function LoadingSpinner({ size = 'md', className }: LoadingSpinnerProps) {
const sizeClasses = {
sm: 'h-4 w-4',
md: 'h-6 w-6',
lg: 'h-8 w-8'
}
return (
<Loader2
className={cn(
'animate-spin text-muted-foreground',
sizeClasses[size],
className
)}
/>
)
}
interface LoadingOverlayProps {
isLoading: boolean
children: React.ReactNode
className?: string
}
export function LoadingOverlay({ isLoading, children, className }: LoadingOverlayProps) {
return (
<div className={cn("relative", className)}>
{children}
{isLoading && (
<div className="absolute inset-0 bg-background/80 backdrop-blur-sm flex items-center justify-center rounded-lg">
<div className="flex items-center gap-2 text-sm text-muted-foreground">
<LoadingSpinner size="sm" />
<span>Loading...</span>
</div>
</div>
)}
</div>
)
}

View File

@ -0,0 +1,70 @@
import { cn } from '@/lib/utils'
interface SkeletonProps {
className?: string
}
export function Skeleton({ className }: SkeletonProps) {
return (
<div
className={cn(
"animate-pulse rounded-md bg-muted",
className
)}
/>
)
}
// Specific skeleton components for different profile sections
export function ProfileCardSkeleton() {
return (
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<Skeleton className="h-6 w-24" />
<Skeleton className="h-8 w-12" />
</div>
<Skeleton className="h-4 w-full" />
</div>
)
}
export function AvatarSkeleton() {
return (
<div className="bg-card p-6 rounded-lg border border-border">
<Skeleton className="h-6 w-32 mb-4" />
<div className="flex flex-col items-center">
<div className="relative">
<Skeleton className="w-24 h-24 rounded-full" />
<div className="absolute -bottom-2 -right-2 w-10 h-10 rounded-full bg-primary/20">
<Skeleton className="w-full h-full rounded-full" />
</div>
</div>
<Skeleton className="h-3 w-48 mt-2" />
</div>
</div>
)
}
export function FormFieldSkeleton() {
return (
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<Skeleton className="h-6 w-20" />
<Skeleton className="h-8 w-12" />
</div>
<Skeleton className="h-10 w-full" />
</div>
)
}
export function TextAreaSkeleton() {
return (
<div className="bg-card p-6 rounded-lg border border-border">
<div className="flex items-center justify-between mb-4">
<Skeleton className="h-6 w-16" />
<Skeleton className="h-8 w-12" />
</div>
<Skeleton className="h-24 w-full" />
</div>
)
}