import { useEffect, useRef, useState } from 'preact/hooks'; import { httpCall, httpPost } from '../../lib/http'; import Cookies from 'js-cookie'; import { TOKEN_COOKIE_NAME } from '../../lib/jwt'; interface PreviewFile extends File { preview: string; } export default function UploadProfilePicture({ user, }: { user: { image: string; }; }) { const [file, setFile] = useState(null); const [error, setError] = useState(''); const [isLoading, setIsLoading] = useState(false); const inputRef = useRef(null); const handleFileChange = async (e: Event) => { setError(''); const file = (e.target as HTMLInputElement).files?.[0]; if (!file) return; // Check file size and dimension const dimensions = await new Promise<{ width: number; height: number; }>((resolve) => { const img = new Image(); img.onload = () => { resolve({ width: img.width, height: img.height }); }; img.src = URL.createObjectURL(file); }); // Image can't be larger than 3000x3000 pixels if (dimensions.width > 3000 || dimensions.height > 3000) { setError('Image dimensions are too big. Maximum 3000x3000 pixels.'); return; // Image can't be smaller than 100x100 pixels } else if (dimensions.width < 100 || dimensions.height < 100) { setError('Image dimensions are too small. Minimum 100x100 pixels.'); return; } // Image can't be larger than 1MB if (file.size > 1024 * 1024) { setError('Image size is too big. Maximum 1MB.'); return; } setError(''); setFile( Object.assign(file, { preview: URL.createObjectURL(file), }) ); }; const handleSubmit = async (e: Event) => { e.preventDefault(); setError(''); setIsLoading(true); if (!file) return; const formData = new FormData(); formData.append('name', 'avatar'); formData.append('avatar', file); const res = await fetch( `${import.meta.env.PUBLIC_API_URL}/v1-upload-profile-picture`, { method: 'POST', body: formData, credentials: 'include', } ); const data = await res.json(); if (!res.ok) { setError(data.message || 'Something went wrong'); setIsLoading(false); } // Logout user if token is invalid if (data.status === 401) { Cookies.remove(TOKEN_COOKIE_NAME); window.location.reload(); } window.location.reload(); }; useEffect(() => { // Necessary to revoke the preview URL when the component unmounts for avoiding memory leaks return () => { if (file) URL.revokeObjectURL(file.preview); }; }, [file]); return (
{file && (
)}
{error && (

{error}

)}
); }