Allow images in posts

This commit is contained in:
Ategon 2025-02-10 20:57:13 -05:00
parent 333bb8b387
commit 21bb6b7188
9 changed files with 435 additions and 326 deletions

12
package-lock.json generated
View file

@ -59,6 +59,7 @@
"react-select": "^5.9.0", "react-select": "^5.9.0",
"react-toastify": "^11.0.3", "react-toastify": "^11.0.3",
"sanitize-html": "^2.14.0", "sanitize-html": "^2.14.0",
"tiptap-extension-resize-image": "^1.2.1",
"tiptap-markdown": "^0.8.10" "tiptap-markdown": "^0.8.10"
}, },
"devDependencies": { "devDependencies": {
@ -12654,6 +12655,17 @@
"@popperjs/core": "^2.9.0" "@popperjs/core": "^2.9.0"
} }
}, },
"node_modules/tiptap-extension-resize-image": {
"version": "1.2.1",
"resolved": "https://registry.npmjs.org/tiptap-extension-resize-image/-/tiptap-extension-resize-image-1.2.1.tgz",
"integrity": "sha512-SLMAujDa+0LN/6Iv2HtU4Uk0BL6LMh4b/r85frpdnjFDW2i6pIOfTVG8jzJQ8T1EgYHNn2YG1U2HoVAGuwLc3Q==",
"license": "MIT",
"peerDependencies": {
"@tiptap/core": "^2.0.0",
"@tiptap/extension-image": "^2.0.0",
"@tiptap/pm": "^2.0.0"
}
},
"node_modules/tiptap-markdown": { "node_modules/tiptap-markdown": {
"version": "0.8.10", "version": "0.8.10",
"resolved": "https://registry.npmjs.org/tiptap-markdown/-/tiptap-markdown-0.8.10.tgz", "resolved": "https://registry.npmjs.org/tiptap-markdown/-/tiptap-markdown-0.8.10.tgz",

View file

@ -60,6 +60,7 @@
"react-select": "^5.9.0", "react-select": "^5.9.0",
"react-toastify": "^11.0.3", "react-toastify": "^11.0.3",
"sanitize-html": "^2.14.0", "sanitize-html": "^2.14.0",
"tiptap-extension-resize-image": "^1.2.1",
"tiptap-markdown": "^0.8.10" "tiptap-markdown": "^0.8.10"
}, },
"devDependencies": { "devDependencies": {

View file

@ -6,18 +6,14 @@ import { Button, Form, Input, Spacer } from "@nextui-org/react";
import { LoaderCircle } from "lucide-react"; import { LoaderCircle } from "lucide-react";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
import sanitizeHtml from "sanitize-html";
import { Select, SelectItem } from "@nextui-org/react"; import { Select, SelectItem } from "@nextui-org/react";
import Timers from "@/components/timers"; import Timers from "@/components/timers";
import Streams from "@/components/streams"; import Streams from "@/components/streams";
import { UserType } from "@/types/UserType"; import { UserType } from "@/types/UserType";
import { useRouter } from 'next/navigation'; import { useRouter } from "next/navigation";
import { GameType } from "@/types/GameType"; import { GameType } from "@/types/GameType";
import { PlatformType, DownloadLinkType } from "@/types/DownloadLinkType"; import { PlatformType, DownloadLinkType } from "@/types/DownloadLinkType";
import { sanitize } from "@/helpers/sanitize";
export default function CreateGamePage() { export default function CreateGamePage() {
const router = useRouter(); const router = useRouter();
@ -53,8 +49,8 @@ export default function CreateGamePage() {
const sanitizeSlug = (value: string): string => { const sanitizeSlug = (value: string): string => {
return value return value
.toLowerCase() // Convert to lowercase .toLowerCase() // Convert to lowercase
.replace(/\s+/g, '-') // Replace whitespace with hyphens .replace(/\s+/g, "-") // Replace whitespace with hyphens
.replace(/[^a-z0-9-]/g, '') // Only allow lowercase letters, numbers, and hyphens .replace(/[^a-z0-9-]/g, "") // Only allow lowercase letters, numbers, and hyphens
.substring(0, 50); // Limit length to 50 characters .substring(0, 50); // Limit length to 50 characters
}; };
@ -150,15 +146,18 @@ export default function CreateGamePage() {
*/ */
}; };
load(); load();
},[]); }, []);
useEffect(() => { useEffect(() => {
const checkExistingGame = async () => { const checkExistingGame = async () => {
const response = await fetch( const response = await fetch(
process.env.NEXT_PUBLIC_MODE === "PROD" process.env.NEXT_PUBLIC_MODE === "PROD"
? `https://d2jam.com/api/v1/self/current-game?username=${getCookie("user")}` ? `https://d2jam.com/api/v1/self/current-game?username=${getCookie(
: `http://localhost:3005/api/v1/self/current-game?username=${getCookie("user")}`, "user"
)}`
: `http://localhost:3005/api/v1/self/current-game?username=${getCookie(
"user"
)}`,
{ {
headers: { authorization: `Bearer ${getCookie("token")}` }, headers: { authorization: `Bearer ${getCookie("token")}` },
credentials: "include", credentials: "include",
@ -177,21 +176,18 @@ export default function CreateGamePage() {
setThumbnailUrl(gameData.thumbnail); setThumbnailUrl(gameData.thumbnail);
setDownloadLinks(gameData.downloadLinks); setDownloadLinks(gameData.downloadLinks);
setGame(gameData); setGame(gameData);
const uniqueAuthors = [gameData.author, ...gameData.contributors] const uniqueAuthors = [
.filter((author, index, self) => gameData.author,
...gameData.contributors,
].filter(
(author, index, self) =>
index === self.findIndex((a) => a.id === author.id) index === self.findIndex((a) => a.id === author.id)
); );
setSelectedAuthors(uniqueAuthors); setSelectedAuthors(uniqueAuthors);
} else {
}
else
{
setSelectedAuthors(user ? [user] : []); setSelectedAuthors(user ? [user] : []);
} }
} } else {
else
{
setEditGame(false); setEditGame(false);
setTitle(""); setTitle("");
setGameSlug(""); setGameSlug("");
@ -199,20 +195,16 @@ export default function CreateGamePage() {
setEditorKey((prev) => prev + 1); setEditorKey((prev) => prev + 1);
setThumbnailUrl(""); setThumbnailUrl("");
setDownloadLinks([]); setDownloadLinks([]);
} }
}; };
if (mounted && user) { if (mounted && user) {
checkExistingGame(); checkExistingGame();
} }
}, [user, mounted]);
},[user,mounted]);
return ( return (
<div className="static flex items-top mt-20 justify-center top-0 left-0"> <div className="static flex items-top mt-20 justify-center top-0 left-0">
<Form <Form
className="w-full max-w-2xl flex flex-col gap-4" className="w-full max-w-2xl flex flex-col gap-4"
validationErrors={errors} validationErrors={errors}
@ -245,7 +237,7 @@ export default function CreateGamePage() {
return; return;
} }
const sanitizedHtml = sanitizeHtml(content); const sanitizedHtml = sanitize(content);
setWaitingPost(true); setWaitingPost(true);
try { try {
@ -285,7 +277,11 @@ export default function CreateGamePage() {
} }
if (response.ok) { if (response.ok) {
toast.success(gameSlug ? "Game updated successfully!" : "Game created successfully!"); toast.success(
gameSlug
? "Game updated successfully!"
: "Game created successfully!"
);
setWaitingPost(false); setWaitingPost(false);
router.push(`/games/${gameSlug || sanitizeSlug(title)}`); router.push(`/games/${gameSlug || sanitizeSlug(title)}`);
} else { } else {
@ -332,7 +328,7 @@ export default function CreateGamePage() {
description="This will be used in the URL: d2jam.com/games/your-game-name" description="This will be used in the URL: d2jam.com/games/your-game-name"
/> />
<div className="flex flex-col gap-2"> <div className="flex flex-col gap-2">
<label className="text-sm font-medium">Add Authors</label> <label className="text-sm font-medium">Add Authors</label>
<Input <Input
placeholder="Search users..." placeholder="Search users..."
@ -349,7 +345,7 @@ export default function CreateGamePage() {
key={user.id} key={user.id}
className="flex justify-between items-center p-2 hover:bg-gray-200 dark:hover:bg-gray-700 rounded cursor-pointer" className="flex justify-between items-center p-2 hover:bg-gray-200 dark:hover:bg-gray-700 rounded cursor-pointer"
onClick={() => { onClick={() => {
if (!selectedAuthors.some(a => a.id === user.id)) { if (!selectedAuthors.some((a) => a.id === user.id)) {
setSelectedAuthors([...selectedAuthors, user]); setSelectedAuthors([...selectedAuthors, user]);
} }
setSearchResults([]); setSearchResults([]);
@ -368,9 +364,14 @@ export default function CreateGamePage() {
className="bg-blue-100 dark:bg-blue-800 text-blue-800 dark:text-blue-100 px-3 py-1 rounded-full flex items-center gap-2" className="bg-blue-100 dark:bg-blue-800 text-blue-800 dark:text-blue-100 px-3 py-1 rounded-full flex items-center gap-2"
> >
<span>{author.name}</span> <span>{author.name}</span>
{((game && author.id !== game.authorId) || (!game && author.id !== user?.id)) && ( {((game && author.id !== game.authorId) ||
(!game && author.id !== user?.id)) && (
<button <button
onClick={() => setSelectedAuthors(selectedAuthors.filter(a => a.id !== author.id))} onClick={() =>
setSelectedAuthors(
selectedAuthors.filter((a) => a.id !== author.id)
)
}
className="text-sm hover:text-red-500" className="text-sm hover:text-red-500"
> >
× ×
@ -378,18 +379,19 @@ export default function CreateGamePage() {
)} )}
</div> </div>
))} ))}
</div> </div>
</div> </div>
<label className="text-sm font-medium">Game Description</label> <label className="text-sm font-medium">Game Description</label>
<Editor key={editorKey} content={content} setContent={setContent} gameEditor /> <Editor
key={editorKey}
content={content}
setContent={setContent}
gameEditor
/>
<Spacer /> <Spacer />
<div className="flex flex-col gap-4"> <div className="flex flex-col gap-4">
<Input <Input
label="Thumbnail URL" label="Thumbnail URL"
labelPlacement="outside" labelPlacement="outside"
@ -414,22 +416,27 @@ export default function CreateGamePage() {
}} }}
onBlur={() => { onBlur={() => {
if (!urlRegex.test(downloadLinks[index].url)) { if (!urlRegex.test(downloadLinks[index].url)) {
toast.error("Please enter a valid URL starting with http:// or https://"); toast.error(
"Please enter a valid URL starting with http:// or https://"
);
if (!downloadLinks[index].url.startsWith("http://") && !downloadLinks[index].url.startsWith("https://")) { if (
const newUrl = "https://" + downloadLinks[index].url; !downloadLinks[index].url.startsWith("http://") &&
!downloadLinks[index].url.startsWith("https://")
) {
const newUrl =
"https://" + downloadLinks[index].url;
const newLinks = [...downloadLinks]; const newLinks = [...downloadLinks];
newLinks[index].url = newUrl; newLinks[index].url = newUrl;
setDownloadLinks(newLinks); setDownloadLinks(newLinks);
const input = document.querySelector<HTMLInputElement>( const input =
document.querySelector<HTMLInputElement>(
`#download-link-${index}` `#download-link-${index}`
); );
if (input) { if (input) {
input.value = newUrl; input.value = newUrl;
} }
} }
} }
}} }}
/> />
@ -439,7 +446,8 @@ export default function CreateGamePage() {
aria-label="Select platform" // Add this to fix accessibility warning aria-label="Select platform" // Add this to fix accessibility warning
onSelectionChange={(value) => { onSelectionChange={(value) => {
const newLinks = [...downloadLinks]; const newLinks = [...downloadLinks];
newLinks[index].platform = value as unknown as PlatformType; newLinks[index].platform =
value as unknown as PlatformType;
setDownloadLinks(newLinks); setDownloadLinks(newLinks);
}} }}
> >
@ -466,7 +474,9 @@ export default function CreateGamePage() {
color="danger" color="danger"
variant="light" variant="light"
onPress={() => { onPress={() => {
setDownloadLinks(downloadLinks.filter((l) => l.id !== link.id)); setDownloadLinks(
downloadLinks.filter((l) => l.id !== link.id)
);
}} }}
> >
× ×
@ -510,7 +520,6 @@ export default function CreateGamePage() {
<Streams /> <Streams />
</div> </div>
)} )}
</div> </div>
); );
} }

View file

@ -14,12 +14,12 @@ import { LoaderCircle } from "lucide-react";
import { redirect } from "next/navigation"; import { redirect } from "next/navigation";
import { ReactNode, useEffect, useState } from "react"; import { ReactNode, useEffect, useState } from "react";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
import sanitizeHtml from "sanitize-html";
import Select, { MultiValue, StylesConfig } from "react-select"; import Select, { MultiValue, StylesConfig } from "react-select";
import { useTheme } from "next-themes"; import { useTheme } from "next-themes";
import Timers from "@/components/timers"; import Timers from "@/components/timers";
import Streams from "@/components/streams"; import Streams from "@/components/streams";
import { UserType } from "@/types/UserType"; import { UserType } from "@/types/UserType";
import { sanitize } from "@/helpers/sanitize";
export default function CreatePostPage() { export default function CreatePostPage() {
const [title, setTitle] = useState(""); const [title, setTitle] = useState("");
@ -118,7 +118,6 @@ export default function CreatePostPage() {
return () => window.removeEventListener("resize", handleResize); return () => window.removeEventListener("resize", handleResize);
}, []); }, []);
const styles: StylesConfig< const styles: StylesConfig<
{ {
value: string; value: string;
@ -213,7 +212,7 @@ export default function CreatePostPage() {
return; return;
} }
const sanitizedHtml = sanitizeHtml(content); const sanitizedHtml = sanitize(content);
setWaitingPost(true); setWaitingPost(true);
const tags = []; const tags = [];

View file

@ -37,8 +37,8 @@ import { redirect, useParams } from "next/navigation";
import { useEffect, useState } from "react"; import { useEffect, useState } from "react";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
import Editor from "@/components/editor"; import Editor from "@/components/editor";
import sanitizeHtml from "sanitize-html";
import CommentCard from "@/components/posts/CommentCard"; import CommentCard from "@/components/posts/CommentCard";
import { sanitize } from "@/helpers/sanitize";
export default function PostPage() { export default function PostPage() {
const [post, setPost] = useState<PostType>(); const [post, setPost] = useState<PostType>();
@ -579,7 +579,7 @@ export default function PostPage() {
return; return;
} }
const sanitizedHtml = sanitizeHtml(content); const sanitizedHtml = sanitize(content);
setWaitingPost(true); setWaitingPost(true);
const response = await fetch( const response = await fetch(

View file

@ -1,7 +1,6 @@
"use client"; "use client";
import Editor from "@/components/editor"; import Editor from "@/components/editor";
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, Spacer } from "@nextui-org/react"; import { Avatar, Button, Form, Input, Spacer } from "@nextui-org/react";
@ -10,6 +9,7 @@ import { useEffect, useState } from "react";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
import { LoaderCircle } from "lucide-react"; import { LoaderCircle } from "lucide-react";
import Image from "next/image"; import Image from "next/image";
import { sanitize } from "@/helpers/sanitize";
export default function UserPage() { export default function UserPage() {
const [user, setUser] = useState<UserType>(); const [user, setUser] = useState<UserType>();
@ -71,7 +71,7 @@ export default function UserPage() {
onSubmit={async (e) => { onSubmit={async (e) => {
e.preventDefault(); e.preventDefault();
const sanitizedBio = sanitizeHtml(bio); const sanitizedBio = sanitize(bio);
if (!name) { if (!name) {
toast.error("You need to enter a name"); toast.error("You need to enter a name");

View file

@ -37,6 +37,9 @@ import CodeBlock from "@tiptap/extension-code-block";
import { Spacer } from "@nextui-org/react"; import { Spacer } from "@nextui-org/react";
import { useTheme } from "next-themes"; import { useTheme } from "next-themes";
import Link from "@tiptap/extension-link"; import Link from "@tiptap/extension-link";
import ImageResize from "tiptap-extension-resize-image";
import { toast } from "react-toastify";
import { getCookie } from "@/helpers/cookie";
type EditorProps = { type EditorProps = {
content: string; content: string;
@ -95,6 +98,7 @@ export default function Editor({
Youtube, Youtube,
CodeBlock, CodeBlock,
Link, Link,
ImageResize,
], ],
content: content, content: content,
immediatelyRender: false, immediatelyRender: false,
@ -110,6 +114,69 @@ export default function Editor({
: "min-h-[150px] max-h-[400px]") + : "min-h-[150px] max-h-[400px]") +
" overflow-y-auto cursor-text rounded-md border p-5 focus-within:outline-none focus-within:border-blue-500 !duration-250 !ease-linear !transition-all", " overflow-y-auto cursor-text rounded-md border p-5 focus-within:outline-none focus-within:border-blue-500 !duration-250 !ease-linear !transition-all",
}, },
handleDrop: (view, event, slice, moved) => {
if (
!moved &&
event.dataTransfer &&
event.dataTransfer.files &&
event.dataTransfer.files[0]
) {
const file = event.dataTransfer.files[0];
const filesize = parseInt((file.size / 1024 / 1024).toFixed(4));
if (file.type !== "image/jpeg" && file.type !== "image/png") {
toast.error("Invalid file format");
return false;
}
console.log(filesize);
if (filesize > 8) {
toast.error("Image is too big");
return false;
}
const formData = new FormData();
formData.append("upload", event.dataTransfer.files[0]);
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",
}
).then((response) => {
if (response.ok) {
response.json().then((data) => {
toast.success(data.message);
const { schema } = view.state;
const coordinates = view.posAtCoords({
left: event.clientX,
top: event.clientY,
});
if (!coordinates) {
toast.error("Error getting coordinates");
return;
}
const node = schema.nodes.image.create({ src: data.data });
const transaction = view.state.tr.insert(coordinates.pos, node);
return view.dispatch(transaction);
});
} else {
toast.error("Failed to upload image");
}
});
}
return false;
},
}, },
}); });

View file

@ -7,8 +7,8 @@ import { useState } from "react";
import Editor from "../editor"; import Editor from "../editor";
import { toast } from "react-toastify"; import { toast } from "react-toastify";
import { getCookie, hasCookie } from "@/helpers/cookie"; import { getCookie, hasCookie } from "@/helpers/cookie";
import sanitizeHtml from "sanitize-html";
import LikeButton from "./LikeButton"; import LikeButton from "./LikeButton";
import { sanitize } from "@/helpers/sanitize";
export default function CommentCard({ comment }: { comment: CommentType }) { export default function CommentCard({ comment }: { comment: CommentType }) {
const [creatingReply, setCreatingReply] = useState<boolean>(false); const [creatingReply, setCreatingReply] = useState<boolean>(false);
@ -90,7 +90,7 @@ export default function CommentCard({ comment }: { comment: CommentType }) {
return; return;
} }
const sanitizedHtml = sanitizeHtml(content); const sanitizedHtml = sanitize(content);
setWaitingPost(true); setWaitingPost(true);
const response = await fetch( const response = await fetch(

21
src/helpers/sanitize.ts Normal file
View file

@ -0,0 +1,21 @@
import sanitizeHtml from "sanitize-html";
export function sanitize(content: string) {
return sanitizeHtml(content, {
allowedTags: sanitizeHtml.defaults.allowedTags.concat(["img"]),
allowedAttributes: {
img: ["src", "style"],
p: ["style"],
},
allowedStyles: {
img: {
width: [/^\d+(px|%)?|auto$/],
height: [/^\d+(px|%)?|auto$/],
margin: [/^\d+(px|%)?|auto$/],
},
p: {
"text-align": [/^right|left|center$/],
},
},
});
}