"use client"; import { useEffect, useState, useCallback, useRef } from "react"; import { useRouter } from "next/navigation"; import { useAuth } from "@/context/AuthContext"; const API_BASE = ""; //Species and breed options for the add/edit pet form 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"], }; const labelCls = "flex flex-col gap-[0.35rem] text-[0.9rem] font-semibold text-[#444]"; const inputCls = "px-[0.85rem] py-[0.6rem] bg-white border border-[#ddd] rounded-lg text-base outline-none transition-all focus:border-[#e68672] focus:shadow-[0_0_0_3px_rgba(230,134,114,0.2)]"; const selectCls = `custom-select ${inputCls} bg-white cursor-pointer`; const errorCls = "bg-[#fff0f0] border border-[#f5c6c6] text-[#c0392b] rounded-lg px-4 py-3 text-[0.9rem]"; const successCls = "bg-[#f0fdf4] border border-[#bbf7d0] text-[#16a34a] rounded-lg px-4 py-3 text-[0.9rem]"; const submitBtnCls = "py-3 bg-[#e68672] text-white border-none rounded-lg text-base font-bold cursor-pointer transition-all hover:bg-[#d4705e] active:scale-[0.98] disabled:opacity-60 disabled:cursor-not-allowed"; //Profile page - shows user info, edit form, avatar upload, owned pets, and order history 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 [orders, setOrders] = useState([]); const [loadingOrders, setLoadingOrders] = 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 [petAge, setPetAge] = useState("1"); const [submitting, setSubmitting] = useState(false); const [petError, setPetError] = useState(null); const [avatarObjectUrl, setAvatarObjectUrl] = useState(null); const [profileForm, setProfileForm] = useState({ firstName: "", lastName: "", email: "", phone: "", password: "", confirmPassword: "" }); const [profileSubmitting, setProfileSubmitting] = useState(false); const [profileError, setProfileError] = useState(null); const [profileSuccess, setProfileSuccess] = useState(null); const [avatarSubmitting, setAvatarSubmitting] = useState(false); //Revokes all blob URLs created for pet images to free up memory 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({ firstName: user?.firstName || "", lastName: user?.lastName || "", email: user?.email || "", phone: user?.phone || "", password: "", confirmPassword: "", }); }, [user]); //Fetches the user's owned pets and resolves their images into blob URLs 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]); //Fetches the user's recent order history const loadOrders = useCallback(async () => { if (!token) return; setLoadingOrders(true); try { const res = await fetch(`${API_BASE}/api/v1/sales/my?size=20&sort=saleDate,desc`, { headers: { Authorization: `Bearer ${token}` }, }); if (!res.ok) return; const data = await res.json(); setOrders(data.content ?? []); } catch { } finally { setLoadingOrders(false); } }, [token]); useEffect(() => { if (user?.role === "CUSTOMER" || user?.role === "ADMIN") { loadPets(); loadOrders(); } }, [user, loadPets, loadOrders]); useEffect(() => { let objectUrl = null; if (user?.avatarUrl && token) { fetch(`${API_BASE}${user.avatarUrl}`, { headers: { Authorization: `Bearer ${token}` }, }) .then((res) => (res.ok ? res.blob() : null)) .then((blob) => { if (blob) { objectUrl = URL.createObjectURL(blob); setAvatarObjectUrl(objectUrl); } else { setAvatarObjectUrl(null); } }) .catch(() => setAvatarObjectUrl(null)); } else { setAvatarObjectUrl(null); } return () => { if (objectUrl) URL.revokeObjectURL(objectUrl); }; }, [user?.avatarUrl, token]); //Logs out and sends the user to the home page function handleLogout() { logout(); router.push("/"); } //Saves changes to the user's name, email, phone, or password async function handleProfileSubmit(e) { e.preventDefault(); setProfileError(null); if (profileForm.password && profileForm.password !== profileForm.confirmPassword) { setProfileError("Passwords do not match."); return; } setProfileSubmitting(true); setProfileSuccess(null); const payload = { firstName: profileForm.firstName, lastName: profileForm.lastName, email: profileForm.email, phone: profileForm.phone, }; if (profileForm.password) { payload.password = profileForm.password; } try { const res = await fetch(`${API_BASE}/api/v1/auth/me`, { method: "PUT", headers: { "Content-Type": "application/json", Authorization: `Bearer ${token}`, }, body: JSON.stringify(payload), }); const data = await res.json().catch(() => null); if (!res.ok) { throw new Error(data?.message || `Request failed (${res.status})`); } await refreshUser(); setProfileForm((prev) => ({ ...prev, password: "", confirmPassword: "" })); setProfileSuccess("Profile updated successfully."); } catch (err) { setProfileError(err.message); } finally { setProfileSubmitting(false); } } //Uploads a new avatar image for the user 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); } } //Removes the user's avatar 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); } } //Opens the add pet form with blank fields function openAddForm() { setEditingPet(null); setPetName(""); setSpecies(""); setBreed(""); setPetAge("1"); setPetError(null); setShowForm(true); } //Opens the edit pet form pre-filled with the selected pet's details function openEditForm(pet) { setEditingPet(pet); setPetName(pet.petName); setSpecies(pet.species); setBreed(pet.breed || ""); setPetAge(pet.petAge != null ? String(pet.petAge) : "1"); setPetError(null); setShowForm(true); } function closeForm() { setShowForm(false); setEditingPet(null); setPetError(null); } //Creates a new pet or saves edits to an existing one 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, petAge: Number(petAge) }), }); 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); } } //Confirms with the user then removes the pet profile 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); } } //Uploads a photo for a specific pet profile 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 displayName = [user.firstName, user.lastName].filter(Boolean).join(" ") || user.username; const fields = [ {label: "First Name", value: user.firstName || "N/A"}, {label: "Last Name", value: user.lastName || "N/A"}, {label: "Username", value: user.username}, {label: "Email", value: user.email}, {label: "Phone", value: user.phone || "N/A"}, ...(user.storeName ? [{ label: "Store", value: user.storeName }] : []), ...(user.role === "CUSTOMER" ? [{ label: "Loyalty Points", value: user.loyaltyPoints ?? 0 }] : []), ]; return (
{/* Profile card */}
{avatarObjectUrl ? ( {displayName} ) : ( displayName.charAt(0).toUpperCase() )}

{displayName}

{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}} {pet.petAge != null && Age: {pet.petAge === 0 ? "< 1 yr" : `${pet.petAge} yr${pet.petAge !== 1 ? "s" : ""}`}}
))}
)}
)} {(user.role === "CUSTOMER" || user.role === "ADMIN") && (

Order History

{loadingOrders ? (

Loading orders...

) : orders.length === 0 ? (

No orders yet.

) : (
{orders.map((order) => (
{new Date(order.saleDate).toLocaleDateString([], { year: "numeric", month: "short", day: "numeric" })} ${Number(order.totalAmount).toFixed(2)}
{order.storeName} {order.paymentMethod && {order.paymentMethod}}
{order.items?.length > 0 && (
    {order.items.map((item) => (
  • {item.productName} × {item.quantity} ${(Number(item.unitPrice) * item.quantity).toFixed(2)}
  • ))}
)}
))}
)}
)}
); }