mirror of
https://github.com/Ategon/Jamjar.git
synced 2025-02-12 06:16:21 +00:00
Add image uploads
This commit is contained in:
parent
3bf841d259
commit
8939141f76
2 changed files with 117 additions and 25 deletions
|
@ -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",
|
||||||
|
},
|
||||||
|
]
|
||||||
|
: []),
|
||||||
],
|
],
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
|
@ -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">
|
||||||
|
|
Loading…
Reference in a new issue