"use client"; import { useEffect, useState, useCallback, useRef } from "react"; import { useRouter } from "next/navigation"; import { useAuth } from "@/context/AuthContext"; const API_BASE = ""; export default function ProfilePage() { const {user, token, loading, logout, refreshUser} = useAuth(); const router = useRouter(); const petImageObjectUrlsRef = useRef([]); const [pets, setPets] = useState([]); const [loadingPets, setLoadingPets] = useState(false); const [showForm, setShowForm] = useState(false); const [editingPet, setEditingPet] = useState(null); const [petName, setPetName] = useState(""); const [species, setSpecies] = useState(""); const [breed, setBreed] = useState(""); const [submitting, setSubmitting] = useState(false); const [petError, setPetError] = useState(null); const [profileForm, setProfileForm] = useState({ fullName: "", email: "", phone: "" }); const [profileSubmitting, setProfileSubmitting] = useState(false); const [profileError, setProfileError] = useState(null); const [profileSuccess, setProfileSuccess] = useState(null); const [avatarSubmitting, setAvatarSubmitting] = useState(false); const clearPetImageObjectUrls = useCallback(() => { for (const objectUrl of petImageObjectUrlsRef.current) { URL.revokeObjectURL(objectUrl); } petImageObjectUrlsRef.current = []; }, []); useEffect(() => { if (!loading && !user) { router.replace(`/login?next=${encodeURIComponent("/profile")}`); } }, [user, loading, router]); useEffect(() => { setProfileForm({ fullName: user?.fullName || "", email: user?.email || "", phone: user?.phone || "", }); }, [user]); const loadPets = useCallback(async () => { if (!token) return; setLoadingPets(true); try { const response = await fetch(`${API_BASE}/api/v1/my-pets`, { headers: { Authorization: `Bearer ${token}` }, }); if (!response.ok) { throw new Error(`Request failed (${response.status})`); } const petData = await response.json(); clearPetImageObjectUrls(); const petsWithResolvedImages = await Promise.all( (Array.isArray(petData) ? petData : []).map(async (pet) => { if (!pet.imageUrl) { return pet; } try { const imageResponse = await fetch(`${API_BASE}${pet.imageUrl}`, { headers: { Authorization: `Bearer ${token}` }, }); if (!imageResponse.ok) { return { ...pet, imageUrl: null }; } const blob = await imageResponse.blob(); const objectUrl = URL.createObjectURL(blob); petImageObjectUrlsRef.current.push(objectUrl); return { ...pet, imageUrl: objectUrl }; } catch { return { ...pet, imageUrl: null }; } }) ); setPets(petsWithResolvedImages); } catch { } finally { setLoadingPets(false); } }, [token, clearPetImageObjectUrls]); useEffect(() => { return () => { clearPetImageObjectUrls(); }; }, [clearPetImageObjectUrls]); useEffect(() => { if (user?.role === "CUSTOMER" || user?.role === "ADMIN") { loadPets(); } }, [user, loadPets]); function handleLogout() { logout(); router.push("/"); } async function handleProfileSubmit(e) { e.preventDefault(); setProfileSubmitting(true); setProfileError(null); setProfileSuccess(null); try { const res = await fetch(`${API_BASE}/api/v1/auth/me`, { method: "PUT", headers: { "Content-Type": "application/json", Authorization: `Bearer ${token}`, }, body: JSON.stringify(profileForm), }); const data = await res.json().catch(() => null); if (!res.ok) { throw new Error(data?.message || `Request failed (${res.status})`); } await refreshUser(); setProfileSuccess("Profile updated successfully."); } catch (err) { setProfileError(err.message); } finally { setProfileSubmitting(false); } } async function handleAvatarUpload(file) { if (!file) { return; } const formData = new FormData(); formData.append("avatar", file); setAvatarSubmitting(true); setProfileError(null); setProfileSuccess(null); try { const res = await fetch(`${API_BASE}/api/v1/auth/me/avatar`, { method: "POST", headers: { Authorization: `Bearer ${token}` }, body: formData, }); const data = await res.json().catch(() => null); if (!res.ok) { throw new Error(data?.message || "Failed to upload avatar"); } await refreshUser(); setProfileSuccess(data?.message || "Avatar updated successfully."); } catch (err) { setProfileError(err.message); } finally { setAvatarSubmitting(false); } } async function handleAvatarDelete() { setAvatarSubmitting(true); setProfileError(null); setProfileSuccess(null); try { const res = await fetch(`${API_BASE}/api/v1/auth/me/avatar`, { method: "DELETE", headers: { Authorization: `Bearer ${token}` }, }); const data = await res.json().catch(() => null); if (!res.ok) { throw new Error(data?.message || "Failed to delete avatar"); } await refreshUser(); setProfileSuccess(data?.message || "Avatar removed successfully."); } catch (err) { setProfileError(err.message); } finally { setAvatarSubmitting(false); } } function openAddForm() { setEditingPet(null); setPetName(""); setSpecies(""); setBreed(""); setPetError(null); setShowForm(true); } function openEditForm(pet) { setEditingPet(pet); setPetName(pet.petName); setSpecies(pet.species); setBreed(pet.breed || ""); setPetError(null); setShowForm(true); } function closeForm() { setShowForm(false); setEditingPet(null); setPetError(null); } async function handlePetSubmit(e) { e.preventDefault(); setPetError(null); setSubmitting(true); const url = editingPet ? `${API_BASE}/api/v1/my-pets/${editingPet.customerPetId}` : `${API_BASE}/api/v1/my-pets`; try { const res = await fetch(url, { method: editingPet ? "PUT" : "POST", headers: { "Content-Type": "application/json", Authorization: `Bearer ${token}`, }, body: JSON.stringify({ petName, species, breed: breed || null }), }); if (!res.ok) { const data = await res.json().catch(() => null); throw new Error(data?.message || `Request failed (${res.status})`); } closeForm(); loadPets(); } catch (err) { setPetError(err.message); } finally { setSubmitting(false); } } async function handleDeletePet(id) { if (!confirm("Remove this pet profile?")) { return; } try { await fetch(`${API_BASE}/api/v1/my-pets/${id}`, { method: "DELETE", headers: { Authorization: `Bearer ${token}` }, }); loadPets(); } catch { } } async function handleImageUpload(petId, file) { const formData = new FormData(); formData.append("image", file); try { const res = await fetch(`${API_BASE}/api/v1/my-pets/${petId}/image`, { method: "POST", headers: { Authorization: `Bearer ${token}` }, body: formData, }); if (!res.ok) { const data = await res.json().catch(() => null); alert(data?.message || "Failed to upload image"); return; } loadPets(); } catch { alert("Failed to upload image"); } } if (loading || !user) { return

Loading…

; } const fields = [ {label: "Full Name", value: user.fullName}, {label: "Username", value: user.username}, {label: "Email", value: user.email}, {label: "Phone", value: user.phone || "—"}, {label: "Role", value: user.role}, ...(user.storeName ? [{ label: "Store", value: user.storeName }] : []), ]; return (
{user.avatarUrl ? ( {user.fullName ) : ( (user.fullName || user.username).charAt(0).toUpperCase() )}

{user.fullName || user.username}

{user.role}
{fields.map(({ label, value }) => (
{label}
{value}
))}

Update Profile

{profileError &&
{profileError}
} {profileSuccess &&
{profileSuccess}
}
{user.avatarUrl && ( )}
{(user.role === "CUSTOMER" || user.role === "ADMIN") && (

My Pets

{showForm && (

{editingPet ? "Edit Pet" : "Add a New Pet"}

{petError &&
{petError}
}
)} {loadingPets ? (

Loading pets...

) : pets.length === 0 && !showForm ? (

No pet profiles yet. Add your first pet above!

) : (
{pets.map((pet) => (
{pet.imageUrl ? ( {pet.petName} ) : (
🐾
)}
{pet.petName} {pet.species} {pet.breed && {pet.breed}}
))}
)}
)}
); }