Add image uploads

This commit is contained in:
Ategon 2025-02-10 18:25:42 -05:00
parent 3bf841d259
commit 8939141f76
2 changed files with 117 additions and 25 deletions

View file

@ -7,6 +7,14 @@ const nextConfig: NextConfig = {
protocol: "https", protocol: "https",
hostname: "**", hostname: "**",
}, },
...(process.env.NEXT_PUBLIC_MODE === "DEV"
? [
{
protocol: "http" as "http",
hostname: "localhost",
},
]
: []),
], ],
}, },
}; };

View file

@ -4,7 +4,7 @@ import Editor from "@/components/editor";
import sanitizeHtml from "sanitize-html"; import sanitizeHtml from "sanitize-html";
import { getCookie, hasCookie } from "@/helpers/cookie"; import { getCookie, hasCookie } from "@/helpers/cookie";
import { UserType } from "@/types/UserType"; import { UserType } from "@/types/UserType";
import { Avatar, Button, Form, Input } from "@nextui-org/react"; import { Avatar, Button, Form, Input, Spacer } from "@nextui-org/react";
import { redirect, usePathname } from "next/navigation"; import { redirect, usePathname } from "next/navigation";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
@ -13,9 +13,9 @@ import Image from "next/image";
export default function UserPage() { export default function UserPage() {
const [user, setUser] = useState<UserType>(); const [user, setUser] = useState<UserType>();
const [profilePicture, setProfilePicture] = useState(""); const [profilePicture, setProfilePicture] = useState<string | null>(null);
const [name, setName] = useState(""); const [name, setName] = useState("");
const [bannerPicture, setBannerPicture] = useState(""); const [bannerPicture, setBannerPicture] = useState<string | null>(null);
const [bio, setBio] = useState(""); const [bio, setBio] = useState("");
const [errors] = useState({}); const [errors] = useState({});
const pathname = usePathname(); const pathname = usePathname();
@ -104,7 +104,7 @@ export default function UserPage() {
if (response.ok) { if (response.ok) {
toast.success("Changed settings"); toast.success("Changed settings");
setUser(await response.json()); setUser((await response.json()).data);
setWaitingSave(false); setWaitingSave(false);
} else { } else {
toast.error("Failed to update settings"); toast.error("Failed to update settings");
@ -127,31 +127,104 @@ export default function UserPage() {
<p>Bio</p> <p>Bio</p>
<Editor content={bio} setContent={setBio} /> <Editor content={bio} setContent={setBio} />
<Input <p>Profile Picture</p>
label="Profile Picture" <input
labelPlacement="outside" type="file"
name="profilePicture" accept="image/*"
placeholder="Enter a url to an image" onChange={async (e) => {
type="text" const file = e.target.files?.[0];
value={profilePicture} if (!file) return;
onValueChange={setProfilePicture}
const formData = new FormData();
formData.append("upload", file);
try {
const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/image"
: "http://localhost:3005/api/v1/image",
{
method: "POST",
body: formData,
headers: {
authorization: `Bearer ${getCookie("token")}`,
},
credentials: "include",
}
);
if (response.ok) {
const data = await response.json();
setProfilePicture(data.data);
toast.success(data.message);
} else {
toast.error("Failed to upload image");
}
} catch (error) {
console.error(error);
toast.error("Error uploading image");
}
}}
/> />
{profilePicture && <Avatar src={profilePicture} />} {profilePicture && (
<div>
<Avatar src={profilePicture} />
<Spacer y={3} />
<Button
color="danger"
size="sm"
onPress={() => {
setProfilePicture(null);
}}
>
Remove Profile Picture
</Button>
</div>
)}
<Input <p>Banner Image</p>
label="Banner Picture" <input
labelPlacement="outside" type="file"
name="bannerPicture" accept="image/*"
placeholder="Enter a url to an image" onChange={async (e) => {
type="text" const file = e.target.files?.[0];
value={bannerPicture} if (!file) return;
onValueChange={setBannerPicture}
const formData = new FormData();
formData.append("upload", file);
try {
const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD"
? "https://d2jam.com/api/v1/image"
: "http://localhost:3005/api/v1/image",
{
method: "POST",
body: formData,
headers: {
authorization: `Bearer ${getCookie("token")}`,
},
credentials: "include",
}
);
if (response.ok) {
const data = await response.json();
setBannerPicture(data.data);
toast.success(data.message);
} else {
toast.error("Failed to upload image");
}
} catch (error) {
console.error(error);
toast.error("Error uploading image");
}
}}
/> />
{bannerPicture && {bannerPicture && (
bannerPicture.startsWith("https://") && <div className="w-full">
bannerPicture.length > 8 && (
<div className="bg-[#222222] h-28 w-full relative"> <div className="bg-[#222222] h-28 w-full relative">
<Image <Image
src={bannerPicture} src={bannerPicture}
@ -160,7 +233,18 @@ export default function UserPage() {
fill fill
/> />
</div> </div>
)} <Spacer y={3} />
<Button
color="danger"
size="sm"
onPress={() => {
setBannerPicture(null);
}}
>
Remove Banner Picture
</Button>
</div>
)}
<div className="flex gap-2"> <div className="flex gap-2">
<Button color="primary" type="submit"> <Button color="primary" type="submit">