"use client"; import { useEffect, useState, useCallback, useRef } from "react"; import { useRouter } from "next/navigation"; import { useAuth } from "@/context/AuthContext"; const API_BASE = ""; const SPECIES_BREEDS = { Dog: ["Beagle", "Boxer", "Bulldog", "Chihuahua", "Dachshund", "German Shepherd", "Golden Retriever", "Labrador Retriever", "Poodle", "Rottweiler", "Shih Tzu", "Siberian Husky", "Yorkshire Terrier", "Mixed / Other"], Cat: ["Abyssinian", "Bengal", "British Shorthair", "Maine Coon", "Persian", "Ragdoll", "Scottish Fold", "Siamese", "Sphynx", "Mixed / Other"], Bird: ["Canary", "Cockatiel", "Cockatoo", "Finch", "Lovebird", "Macaw", "Parakeet", "Parrot", "Other"], Rabbit: ["Dutch", "Flemish Giant", "Holland Lop", "Lionhead", "Mini Rex", "Other"], Hamster: ["Dwarf", "Roborovski", "Syrian", "Other"], "Guinea Pig": ["Abyssinian", "American", "Peruvian", "Teddy", "Other"], Reptile: ["Ball Python", "Bearded Dragon", "Blue-tongued Skink", "Corn Snake", "Leopard Gecko", "Other"], Fish: ["Angelfish", "Betta", "Cichlid", "Clownfish", "Goldfish", "Guppy", "Tetra", "Other"], Other: ["Other"], }; 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?status=Owned`, { headers: { Authorization: `Bearer ${token}` }, }); if (!response.ok) { throw new Error(`Request failed (${response.status})`); } const petData = await response.json(); clearPetImageObjectUrls(); const ownedPets = Array.isArray(petData) ? petData : []; const petsWithResolvedImages = await Promise.all( ownedPets.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 { const res = await fetch(`${API_BASE}/api/v1/my-pets/${id}`, { method: "DELETE", headers: { Authorization: `Bearer ${token}` }, }); if (!res.ok) { const data = await res.json().catch(() => null); throw new Error(data?.message || `Failed to remove pet (${res.status})`); } loadPets(); } catch (err) { alert(err.message); } } 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}}
))}
)}
)}
); }