mirror of
https://github.com/Ategon/Jamjar.git
synced 2025-02-12 06:16:21 +00:00
Compare commits
22 commits
68e4fb7a71
...
e0319729c8
Author | SHA1 | Date | |
---|---|---|---|
![]() |
e0319729c8 | ||
![]() |
8002647ec7 | ||
![]() |
c74f207d1a | ||
![]() |
9cc4f99e3d | ||
![]() |
c1693a3166 | ||
![]() |
4642ea9032 | ||
![]() |
a879690129 | ||
![]() |
0e35c13292 | ||
![]() |
2d2ffa7d12 | ||
![]() |
047d630b14 | ||
![]() |
2178675f2a | ||
![]() |
aaa9305c09 | ||
![]() |
fc5de7ba12 | ||
![]() |
3a2d5b1cb3 | ||
![]() |
ad7311247b | ||
![]() |
51d92b2fc4 | ||
![]() |
e61dd263c5 | ||
![]() |
2c05a7464f | ||
![]() |
6d857ffc49 | ||
![]() |
606a0695d8 | ||
![]() |
e63f40c69b | ||
![]() |
2523a1052b |
21 changed files with 920 additions and 81 deletions
21
.devcontainer/devcontainer.json
Normal file
21
.devcontainer/devcontainer.json
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
// For format details, see https://aka.ms/devcontainer.json. For config options, see the
|
||||||
|
// README at: https://github.com/devcontainers/templates/tree/main/src/typescript-node
|
||||||
|
{
|
||||||
|
"name": "Node.js & TypeScript",
|
||||||
|
// Or use a Dockerfile or Docker Compose file. More info: https://containers.dev/guide/dockerfile
|
||||||
|
"image": "mcr.microsoft.com/devcontainers/typescript-node:1-22-bookworm",
|
||||||
|
// https://github.com/csci430-os/vscode-remote-devcontainer/issues/2#issuecomment-1939950604
|
||||||
|
"mounts": [
|
||||||
|
"type=bind,source=${localEnv:HOME}/.ssh,target=/home/node/.ssh,readonly"
|
||||||
|
]
|
||||||
|
// Features to add to the dev container. More info: https://containers.dev/features.
|
||||||
|
// "features": {},
|
||||||
|
// Use 'forwardPorts' to make a list of ports inside the container available locally.
|
||||||
|
// "forwardPorts": [],
|
||||||
|
// Use 'postCreateCommand' to run commands after the container is created.
|
||||||
|
// "postCreateCommand": "yarn install",
|
||||||
|
// Configure tool-specific properties.
|
||||||
|
// "customizations": {},
|
||||||
|
// Uncomment to connect as root instead. More info: https://aka.ms/dev-containers-non-root.
|
||||||
|
// "remoteUser": "root"
|
||||||
|
}
|
18
.vscode/launch.json
vendored
Normal file
18
.vscode/launch.json
vendored
Normal file
|
@ -0,0 +1,18 @@
|
||||||
|
{
|
||||||
|
// Use IntelliSense to learn about possible attributes.
|
||||||
|
// Hover to view descriptions of existing attributes.
|
||||||
|
// For more information, visit: https://go.microsoft.com/fwlink/?linkid=830387
|
||||||
|
"version": "0.2.0",
|
||||||
|
"configurations": [
|
||||||
|
{
|
||||||
|
"type": "node",
|
||||||
|
"request": "launch",
|
||||||
|
"name": "Run dev",
|
||||||
|
"runtimeExecutable": "next",
|
||||||
|
"cwd": "${workspaceFolder}",
|
||||||
|
"args": [
|
||||||
|
"dev"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
36
README.md
36
README.md
|
@ -2,15 +2,37 @@
|
||||||
|
|
||||||
Frontend for a game jam site
|
Frontend for a game jam site
|
||||||
|
|
||||||
To run using next.js (for development)
|
## Things used
|
||||||
|
|
||||||
```
|
- Typescript (language)
|
||||||
npm i
|
- Next.js (web framework)
|
||||||
npm run dev
|
- Tailwind (css framework)
|
||||||
```
|
- Lucide (icons)
|
||||||
|
- Eslint (static code analysis)
|
||||||
|
- Framer motion (animations)
|
||||||
|
- React Toastify (toasts)
|
||||||
|
|
||||||
To run using docker and docker compose
|
## Running for development
|
||||||
|
|
||||||
|
Prerequisites:
|
||||||
|
- node.js or equivalent
|
||||||
|
|
||||||
|
To start up the site locally for development you need to:
|
||||||
|
|
||||||
|
1. Go to a spot you want to be the parent folder for where the folder for Jamjar goes (e.g. navigate to it in terminal)
|
||||||
|
2. Clone the repository aka get a local copy of the files (e.g. by running `git clone https://github.com/Dare2Jam/Jamjar.git`)
|
||||||
|
3. Go into the folder you just cloned in (e.g. using `cd Jamjar`)
|
||||||
|
4. Install dependencies needed for the site (e.g. `npm i`)
|
||||||
|
5. Create a `.env` file in the folder which is used for environment variables. In this you would set NEXT_PUBLIC_MODE to either PROD or DEV depending on what backend data you want to load in (dev loads it from a locally running jamcore, PROD loads it from the production site)
|
||||||
```
|
```
|
||||||
docker compose up --build -d
|
NEXT_PUBLIC_MODE=DEV
|
||||||
```
|
```
|
||||||
|
6. Run the site using `npm run dev` which will start up a dev server that will hot reload as you make changes (most of the time)
|
||||||
|
7. Go to https://localhost:3000 (or another port if it says it started up the site on a different port)
|
||||||
|
|
||||||
|
## Running using docker
|
||||||
|
|
||||||
|
Prerequisites:
|
||||||
|
- docker
|
||||||
|
|
||||||
|
If you want to start up the frontend using docker instead of what is above (either for development or for a production site) you can run `docker compose up --build -d` to build the image and then run it in the background. This will need to be done after any changes you make to rebuild the image
|
||||||
|
|
25
package-lock.json
generated
25
package-lock.json
generated
|
@ -15,7 +15,8 @@
|
||||||
"lucide-react": "^0.453.0",
|
"lucide-react": "^0.453.0",
|
||||||
"next": "15.0.1",
|
"next": "15.0.1",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0"
|
"react-dom": "^18.2.0",
|
||||||
|
"react-toastify": "^11.0.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^20",
|
"@types/node": "^20",
|
||||||
|
@ -7605,6 +7606,28 @@
|
||||||
"react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
|
"react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"node_modules/react-toastify": {
|
||||||
|
"version": "11.0.3",
|
||||||
|
"resolved": "https://registry.npmjs.org/react-toastify/-/react-toastify-11.0.3.tgz",
|
||||||
|
"integrity": "sha512-cbPtHJPfc0sGqVwozBwaTrTu1ogB9+BLLjd4dDXd863qYLj7DGrQ2sg5RAChjFUB4yc3w8iXOtWcJqPK/6xqRQ==",
|
||||||
|
"license": "MIT",
|
||||||
|
"dependencies": {
|
||||||
|
"clsx": "^2.1.1"
|
||||||
|
},
|
||||||
|
"peerDependencies": {
|
||||||
|
"react": "^18 || ^19",
|
||||||
|
"react-dom": "^18 || ^19"
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"node_modules/react-toastify/node_modules/clsx": {
|
||||||
|
"version": "2.1.1",
|
||||||
|
"resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz",
|
||||||
|
"integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==",
|
||||||
|
"license": "MIT",
|
||||||
|
"engines": {
|
||||||
|
"node": ">=6"
|
||||||
|
}
|
||||||
|
},
|
||||||
"node_modules/read-cache": {
|
"node_modules/read-cache": {
|
||||||
"version": "1.0.0",
|
"version": "1.0.0",
|
||||||
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
||||||
|
|
|
@ -16,7 +16,8 @@
|
||||||
"lucide-react": "^0.453.0",
|
"lucide-react": "^0.453.0",
|
||||||
"next": "15.0.1",
|
"next": "15.0.1",
|
||||||
"react": "^18.2.0",
|
"react": "^18.2.0",
|
||||||
"react-dom": "^18.2.0"
|
"react-dom": "^18.2.0",
|
||||||
|
"react-toastify": "^11.0.3"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@types/node": "^20",
|
"@types/node": "^20",
|
||||||
|
|
BIN
public/images/dare2jam.png
Normal file
BIN
public/images/dare2jam.png
Normal file
Binary file not shown.
After Width: | Height: | Size: 25 KiB |
Binary file not shown.
Before Width: | Height: | Size: 56 KiB |
|
@ -1,29 +0,0 @@
|
||||||
import Posts from "@/components/posts";
|
|
||||||
import Timers from "@/components/timers";
|
|
||||||
import { Image } from "@nextui-org/react";
|
|
||||||
|
|
||||||
export default function Home() {
|
|
||||||
return (
|
|
||||||
<div>
|
|
||||||
<div className="absolute left-0 top-0 w-full h-full z-0">
|
|
||||||
<Image
|
|
||||||
src="/images/bg.jpg"
|
|
||||||
alt="Home background"
|
|
||||||
className="object-cover w-full h-full"
|
|
||||||
radius="none"
|
|
||||||
loading="eager"
|
|
||||||
removeWrapper
|
|
||||||
/>
|
|
||||||
<div className="absolute left-0 top-0 w-full h-full bg-gradient-to-r from-black/50 to-transparent z-10" />
|
|
||||||
</div>
|
|
||||||
<div className="z-10 relative flex">
|
|
||||||
<div>
|
|
||||||
<Posts />
|
|
||||||
</div>
|
|
||||||
<div className="w-1/3 flex justify-end">
|
|
||||||
<Timers />
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
);
|
|
||||||
}
|
|
108
src/app/create-post/page.tsx
Normal file
108
src/app/create-post/page.tsx
Normal file
|
@ -0,0 +1,108 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { getCookie, hasCookie } from "@/helpers/cookie";
|
||||||
|
import { Button, Form, Input, Textarea } from "@nextui-org/react";
|
||||||
|
import { redirect } from "next/navigation";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
|
export default function CreatePostPage() {
|
||||||
|
const [title, setTitle] = useState("");
|
||||||
|
const [content, setContent] = useState("");
|
||||||
|
const [errors, setErrors] = useState({});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="absolute flex items-center justify-center top-0 left-0 w-screen h-screen">
|
||||||
|
<Form
|
||||||
|
className="w-full max-w-xs flex flex-col gap-4"
|
||||||
|
validationErrors={errors}
|
||||||
|
onReset={() => {
|
||||||
|
setTitle("");
|
||||||
|
setContent("");
|
||||||
|
}}
|
||||||
|
onSubmit={async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (!title && !content) {
|
||||||
|
setErrors({
|
||||||
|
title: "Please enter a valid title",
|
||||||
|
content: "Please enter valid content",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!title) {
|
||||||
|
setErrors({ title: "Please enter a valid title" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!content) {
|
||||||
|
setErrors({ content: "Please enter valid content" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!hasCookie("token")) {
|
||||||
|
setErrors({ content: "You are not logged in" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
? "https://d2jam.com/api/v1/post"
|
||||||
|
: "http://localhost:3005/api/v1/post",
|
||||||
|
{
|
||||||
|
body: JSON.stringify({
|
||||||
|
title: title,
|
||||||
|
content: content,
|
||||||
|
username: getCookie("user"),
|
||||||
|
}),
|
||||||
|
method: "POST",
|
||||||
|
headers: {
|
||||||
|
"Content-Type": "application/json",
|
||||||
|
authorization: `Bearer ${getCookie("token")}`,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.status == 401) {
|
||||||
|
setErrors({ content: "Invalid user" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
toast.success("Successfully created post");
|
||||||
|
|
||||||
|
redirect("/");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Title"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="title"
|
||||||
|
placeholder="Enter a title"
|
||||||
|
type="text"
|
||||||
|
value={title}
|
||||||
|
onValueChange={setTitle}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Textarea
|
||||||
|
isRequired
|
||||||
|
label="Content"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="content"
|
||||||
|
placeholder="Enter the post body"
|
||||||
|
value={content}
|
||||||
|
onValueChange={setContent}
|
||||||
|
/>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button color="primary" type="submit">
|
||||||
|
Create
|
||||||
|
</Button>
|
||||||
|
<Button type="reset" variant="flat">
|
||||||
|
Reset
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Form>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
Binary file not shown.
Before Width: | Height: | Size: 25 KiB After Width: | Height: | Size: 15 KiB |
|
@ -3,11 +3,12 @@ import { Inter } from "next/font/google";
|
||||||
import "./globals.css";
|
import "./globals.css";
|
||||||
import Navbar from "../components/navbar";
|
import Navbar from "../components/navbar";
|
||||||
import Providers from "./providers";
|
import Providers from "./providers";
|
||||||
|
import { ToastContainer } from "react-toastify";
|
||||||
|
|
||||||
const inter = Inter({ subsets: ["latin"] });
|
const inter = Inter({ subsets: ["latin"] });
|
||||||
|
|
||||||
export const metadata: Metadata = {
|
export const metadata: Metadata = {
|
||||||
title: "Edikoyo Jam",
|
title: "Dare2Jam",
|
||||||
description: "A community built game jam!",
|
description: "A community built game jam!",
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -24,6 +25,7 @@ export default function RootLayout({
|
||||||
<div className="bg-zinc-100 dark:bg-zinc-950 min-h-screen">
|
<div className="bg-zinc-100 dark:bg-zinc-950 min-h-screen">
|
||||||
<Navbar />
|
<Navbar />
|
||||||
<div className="max-w-8xl mx-auto">{children}</div>
|
<div className="max-w-8xl mx-auto">{children}</div>
|
||||||
|
<ToastContainer />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</Providers>
|
</Providers>
|
||||||
|
|
102
src/app/login/page.tsx
Normal file
102
src/app/login/page.tsx
Normal file
|
@ -0,0 +1,102 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { Button, Form, Input } from "@nextui-org/react";
|
||||||
|
import { redirect } from "next/navigation";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
|
export default function UserPage() {
|
||||||
|
const [username, setUsername] = useState("");
|
||||||
|
const [password, setPassword] = useState("");
|
||||||
|
const [errors, setErrors] = useState({});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="absolute flex items-center justify-center top-0 left-0 w-screen h-screen">
|
||||||
|
<Form
|
||||||
|
className="w-full max-w-xs flex flex-col gap-4"
|
||||||
|
validationErrors={errors}
|
||||||
|
onReset={() => {
|
||||||
|
setUsername("");
|
||||||
|
setPassword("");
|
||||||
|
}}
|
||||||
|
onSubmit={async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (!username && !password) {
|
||||||
|
setErrors({
|
||||||
|
username: "Please enter a valid username",
|
||||||
|
password: "Please enter a valid password",
|
||||||
|
});
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!username) {
|
||||||
|
setErrors({ username: "Please enter a valid username" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!password) {
|
||||||
|
setErrors({ password: "Please enter a valid password" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
? "https://d2jam.com/api/v1/login"
|
||||||
|
: "http://localhost:3005/api/v1/login",
|
||||||
|
{
|
||||||
|
body: JSON.stringify({ username: username, password: password }),
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.status == 401) {
|
||||||
|
setErrors({ password: "Invalid username or password" });
|
||||||
|
setPassword("");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { token, user } = await response.json();
|
||||||
|
|
||||||
|
document.cookie = `token=${token}`;
|
||||||
|
document.cookie = `user=${user.slug}`;
|
||||||
|
|
||||||
|
toast.success("Successfully logged in");
|
||||||
|
|
||||||
|
redirect("/");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Username"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="username"
|
||||||
|
placeholder="Enter your username"
|
||||||
|
type="text"
|
||||||
|
value={username}
|
||||||
|
onValueChange={setUsername}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Password"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="password"
|
||||||
|
placeholder="Enter your password"
|
||||||
|
type="password"
|
||||||
|
value={password}
|
||||||
|
onValueChange={setPassword}
|
||||||
|
/>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button color="primary" type="submit">
|
||||||
|
Submit
|
||||||
|
</Button>
|
||||||
|
<Button type="reset" variant="flat">
|
||||||
|
Reset
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Form>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
21
src/app/logout/page.tsx
Normal file
21
src/app/logout/page.tsx
Normal file
|
@ -0,0 +1,21 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { redirect } from "next/navigation";
|
||||||
|
import React, { useEffect } from "react";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
|
export default function UserPage() {
|
||||||
|
useEffect(() => {
|
||||||
|
document.cookie = "token=; expires=Thu, 01 Jan 1970 00:00:00 UTC; path=/;";
|
||||||
|
|
||||||
|
toast.success("Successfully logged out");
|
||||||
|
|
||||||
|
redirect("/");
|
||||||
|
});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="absolute flex items-center justify-center top-0 left-0 w-screen h-screen">
|
||||||
|
<p>Logging out...</p>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,11 +1,14 @@
|
||||||
import { Image } from "@nextui-org/image";
|
import { Image } from "@nextui-org/image";
|
||||||
import { Button } from "@nextui-org/button";
|
import { Button } from "@nextui-org/button";
|
||||||
import { SiDiscord } from "@icons-pack/react-simple-icons";
|
import { SiDiscord } from "@icons-pack/react-simple-icons";
|
||||||
|
import { Link } from "@nextui-org/react";
|
||||||
|
import Posts from "@/components/posts";
|
||||||
|
import Timers from "@/components/timers";
|
||||||
|
|
||||||
export default async function Home() {
|
export default async function Home() {
|
||||||
return (
|
return (
|
||||||
<div>
|
<div className="w-full">
|
||||||
<div className="absolute left-0 top-0 w-full h-full z-0">
|
<div className="fixed left-0 top-0 w-full h-full z-0">
|
||||||
<Image
|
<Image
|
||||||
src="/images/bg.jpg"
|
src="/images/bg.jpg"
|
||||||
alt="Home background"
|
alt="Home background"
|
||||||
|
@ -16,17 +19,13 @@ export default async function Home() {
|
||||||
/>
|
/>
|
||||||
<div className="absolute left-0 top-0 w-full h-full bg-gradient-to-r from-black/50 to-transparent z-10" />
|
<div className="absolute left-0 top-0 w-full h-full bg-gradient-to-r from-black/50 to-transparent z-10" />
|
||||||
</div>
|
</div>
|
||||||
<div className="relative left-0 top-0 z-10 px-8">
|
<div className="z-10 relative flex w-full flex-wrap">
|
||||||
<div className="flex gap-20">
|
<div>
|
||||||
<div className="flex flex-col gap-4 py-16 sm:py-36 md:py-72">
|
<div className="flex flex-col gap-4 py-4 sm:py-8 md:py-12 pl-16">
|
||||||
<h1 className="text-3xl sm:text-4xl md:text-5xl">Edikoyo Jam</h1>
|
<h1 className="text-3xl sm:text-4xl md:text-5xl">Dare2Jam</h1>
|
||||||
<p className="text-lg sm:text-xl">April 4th - 7th</p>
|
<p className="text-lg sm:text-xl">April 4th - 7th</p>
|
||||||
<div className="flex gap-2">
|
<div className="flex gap-2">
|
||||||
<a
|
<Link href="https://discord.gg/rfmKzM6ASw" target="_blank">
|
||||||
href="https://discord.gg/rfmKzM6ASw"
|
|
||||||
target="_blank"
|
|
||||||
rel="noopener noreferrer"
|
|
||||||
>
|
|
||||||
<Button
|
<Button
|
||||||
variant="bordered"
|
variant="bordered"
|
||||||
className="border-white/50 text-white"
|
className="border-white/50 text-white"
|
||||||
|
@ -34,11 +33,50 @@ export default async function Home() {
|
||||||
>
|
>
|
||||||
Join the Discord
|
Join the Discord
|
||||||
</Button>
|
</Button>
|
||||||
</a>
|
</Link>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
|
<Posts />
|
||||||
|
</div>
|
||||||
|
<div className="w-1/3 flex justify-end py-4 sm:py-8 md:py-12 flex-grow">
|
||||||
|
<Timers />
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
{
|
||||||
|
/* <div>
|
||||||
|
<div className="absolute left-0 top-0 w-full h-full z-0">
|
||||||
|
<Image
|
||||||
|
src="/images/bg.jpg"
|
||||||
|
alt="Home background"
|
||||||
|
className="object-cover w-full h-full"
|
||||||
|
radius="none"
|
||||||
|
loading="eager"
|
||||||
|
removeWrapper
|
||||||
|
/>
|
||||||
|
<div className="absolute left-0 top-0 w-full h-full bg-gradient-to-r from-black/50 to-transparent z-10" />
|
||||||
|
</div>
|
||||||
|
<div className="relative left-0 top-0 z-10 px-8">
|
||||||
|
<div className="flex gap-20">
|
||||||
|
<div className="flex flex-col gap-4 py-16 sm:py-36 md:py-72">
|
||||||
|
<h1 className="text-3xl sm:text-4xl md:text-5xl">Dare2Jam</h1>
|
||||||
|
<p className="text-lg sm:text-xl">April 4th - 7th</p>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Link href="https://discord.gg/rfmKzM6ASw" target="_blank">
|
||||||
|
<Button
|
||||||
|
variant="bordered"
|
||||||
|
className="border-white/50 text-white"
|
||||||
|
startContent={<SiDiscord />}
|
||||||
|
>
|
||||||
|
Join the Discord
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
</div> */
|
||||||
|
}
|
||||||
|
|
137
src/app/signup/page.tsx
Normal file
137
src/app/signup/page.tsx
Normal file
|
@ -0,0 +1,137 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
|
import { Button, Form, Input } from "@nextui-org/react";
|
||||||
|
import { redirect } from "next/navigation";
|
||||||
|
import { useState } from "react";
|
||||||
|
import { toast } from "react-toastify";
|
||||||
|
|
||||||
|
export default function UserPage() {
|
||||||
|
const [username, setUsername] = useState("");
|
||||||
|
const [password, setPassword] = useState("");
|
||||||
|
const [password2, setPassword2] = useState("");
|
||||||
|
const [errors, setErrors] = useState({});
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="absolute flex items-center justify-center top-0 left-0 w-screen h-screen">
|
||||||
|
<Form
|
||||||
|
className="w-full max-w-xs flex flex-col gap-4"
|
||||||
|
validationErrors={errors}
|
||||||
|
onReset={() => {
|
||||||
|
setUsername("");
|
||||||
|
setPassword("");
|
||||||
|
setPassword2("");
|
||||||
|
}}
|
||||||
|
onSubmit={async (e) => {
|
||||||
|
e.preventDefault();
|
||||||
|
|
||||||
|
if (!password || !username || !password2) {
|
||||||
|
const localErrors: Record<string, string> = {};
|
||||||
|
|
||||||
|
if (password && password.length < 8) {
|
||||||
|
setPassword2("");
|
||||||
|
setErrors({ password: "Password must be minimum 8 characters" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!password) {
|
||||||
|
localErrors["password"] = "Please enter a valid password";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!password2) {
|
||||||
|
localErrors["password2"] = "Please reenter your password";
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!username) {
|
||||||
|
localErrors["username"] = "Please enter a valid username";
|
||||||
|
}
|
||||||
|
setErrors(localErrors);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (username.length > 32) {
|
||||||
|
setPassword2("");
|
||||||
|
setErrors({ password: "Usernames can be maximum 32 characters" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (password.length < 8) {
|
||||||
|
setPassword2("");
|
||||||
|
setErrors({ password: "Password must be minimum 8 characters" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (password != password2) {
|
||||||
|
setPassword2("");
|
||||||
|
setErrors({ password2: "Passwords do not match" });
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
? "https://d2jam.com/api/v1/signup"
|
||||||
|
: "http://localhost:3005/api/v1/signup",
|
||||||
|
{
|
||||||
|
body: JSON.stringify({ username: username, password: password }),
|
||||||
|
method: "POST",
|
||||||
|
headers: { "Content-Type": "application/json" },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.status == 409) {
|
||||||
|
setErrors({ username: "User already exists" });
|
||||||
|
setPassword2("");
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const { token, user } = await response.json();
|
||||||
|
document.cookie = `token=${token}`;
|
||||||
|
document.cookie = `user=${user.slug}`;
|
||||||
|
|
||||||
|
toast.success("Successfully signed up");
|
||||||
|
|
||||||
|
redirect("/");
|
||||||
|
}}
|
||||||
|
>
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Username"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="username"
|
||||||
|
placeholder="Enter your username"
|
||||||
|
type="text"
|
||||||
|
value={username}
|
||||||
|
onValueChange={setUsername}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Password"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="password"
|
||||||
|
placeholder="Enter your password"
|
||||||
|
type="password"
|
||||||
|
value={password}
|
||||||
|
onValueChange={setPassword}
|
||||||
|
/>
|
||||||
|
<Input
|
||||||
|
isRequired
|
||||||
|
label="Password Confirmation"
|
||||||
|
labelPlacement="outside"
|
||||||
|
name="password2"
|
||||||
|
placeholder="Reenter your password"
|
||||||
|
type="password"
|
||||||
|
value={password2}
|
||||||
|
onValueChange={setPassword2}
|
||||||
|
/>
|
||||||
|
<div className="flex gap-2">
|
||||||
|
<Button color="primary" type="submit">
|
||||||
|
Submit
|
||||||
|
</Button>
|
||||||
|
<Button type="reset" variant="flat">
|
||||||
|
Reset
|
||||||
|
</Button>
|
||||||
|
</div>
|
||||||
|
</Form>
|
||||||
|
</div>
|
||||||
|
);
|
||||||
|
}
|
|
@ -1,3 +1,5 @@
|
||||||
|
"use client";
|
||||||
|
|
||||||
import {
|
import {
|
||||||
Navbar as NavbarBase,
|
Navbar as NavbarBase,
|
||||||
NavbarBrand,
|
NavbarBrand,
|
||||||
|
@ -6,10 +8,63 @@ import {
|
||||||
} from "@nextui-org/navbar";
|
} from "@nextui-org/navbar";
|
||||||
import { Link } from "@nextui-org/link";
|
import { Link } from "@nextui-org/link";
|
||||||
import { Divider } from "@nextui-org/divider";
|
import { Divider } from "@nextui-org/divider";
|
||||||
import { Image, Tooltip } from "@nextui-org/react";
|
import {
|
||||||
|
Avatar,
|
||||||
|
Button,
|
||||||
|
Dropdown,
|
||||||
|
DropdownItem,
|
||||||
|
DropdownMenu,
|
||||||
|
DropdownTrigger,
|
||||||
|
Image,
|
||||||
|
Spacer,
|
||||||
|
Tooltip,
|
||||||
|
} from "@nextui-org/react";
|
||||||
import { SiDiscord, SiForgejo, SiGithub } from "@icons-pack/react-simple-icons";
|
import { SiDiscord, SiForgejo, SiGithub } from "@icons-pack/react-simple-icons";
|
||||||
|
import { LogInIcon, Menu, NotebookPen, SquarePen } from "lucide-react";
|
||||||
|
import { useEffect, useState } from "react";
|
||||||
|
import { hasCookie, getCookie } from "@/helpers/cookie";
|
||||||
|
import { usePathname } from "next/navigation";
|
||||||
|
import { UserType } from "@/types/UserType";
|
||||||
|
|
||||||
export default function Navbar() {
|
export default function Navbar() {
|
||||||
|
const [user, setUser] = useState<UserType>();
|
||||||
|
const pathname = usePathname();
|
||||||
|
const [isMobile, setIsMobile] = useState(false);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
loadUser();
|
||||||
|
async function loadUser() {
|
||||||
|
if (!hasCookie("token")) {
|
||||||
|
setUser(undefined);
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await fetch(
|
||||||
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
? `https://d2jam.com/api/v1/self?username=${getCookie("user")}`
|
||||||
|
: `http://localhost:3005/api/v1/self?username=${getCookie("user")}`,
|
||||||
|
{
|
||||||
|
headers: { authorization: `Bearer ${getCookie("token")}` },
|
||||||
|
}
|
||||||
|
);
|
||||||
|
|
||||||
|
if (response.status == 200) {
|
||||||
|
setUser(await response.json());
|
||||||
|
} else {
|
||||||
|
setUser(undefined);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}, [pathname]);
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const handleResize = () => {
|
||||||
|
setIsMobile(window.innerWidth <= 768); // Adjust breakpoint as needed
|
||||||
|
};
|
||||||
|
handleResize();
|
||||||
|
window.addEventListener("resize", handleResize);
|
||||||
|
return () => window.removeEventListener("resize", handleResize);
|
||||||
|
}, []);
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<NavbarBase
|
<NavbarBase
|
||||||
shouldHideOnScroll
|
shouldHideOnScroll
|
||||||
|
@ -21,20 +76,264 @@ export default function Navbar() {
|
||||||
href="/"
|
href="/"
|
||||||
className="duration-500 ease-in-out transition-all transform hover:scale-110"
|
className="duration-500 ease-in-out transition-all transform hover:scale-110"
|
||||||
>
|
>
|
||||||
<Image src="/images/edikoyo.png" alt="Edikoyo logo" width={160} />
|
<Image src="/images/dare2jam.png" alt="Dare2Jam logo" width={80} />
|
||||||
</Link>
|
</Link>
|
||||||
</NavbarBrand>
|
</NavbarBrand>
|
||||||
<NavbarContent>
|
|
||||||
<NavbarItem>
|
|
||||||
<Link
|
|
||||||
href="/app"
|
|
||||||
className="text-white flex justify-center duration-500 ease-in-out transition-all transform hover:scale-110"
|
|
||||||
>
|
|
||||||
Beta Site
|
|
||||||
</Link>
|
|
||||||
</NavbarItem>
|
|
||||||
</NavbarContent>
|
|
||||||
<NavbarContent justify="end">
|
<NavbarContent justify="end">
|
||||||
|
{isMobile ? (
|
||||||
|
user ? (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Avatar src={user.profilePicture} />
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu className="text-black">
|
||||||
|
<DropdownItem key="create-post" href="/create-post">
|
||||||
|
Create Post
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
key="github"
|
||||||
|
href="https://github.com/Ategon/Jamjar"
|
||||||
|
>
|
||||||
|
GitHub
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
key="forgejo"
|
||||||
|
href="https://git.edikoyo.com/Ategon/Jamjar"
|
||||||
|
>
|
||||||
|
Forgejo
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
key="discord"
|
||||||
|
href="https://discord.gg/rfmKzM6ASw"
|
||||||
|
>
|
||||||
|
Discord
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="logout" color="danger" href="/logout">
|
||||||
|
Logout
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
) : (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Menu />
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu className="text-black">
|
||||||
|
<DropdownItem
|
||||||
|
key="github"
|
||||||
|
href="https://github.com/Ategon/Jamjar"
|
||||||
|
>
|
||||||
|
GitHub
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
key="forgejo"
|
||||||
|
href="https://git.edikoyo.com/Ategon/Jamjar"
|
||||||
|
>
|
||||||
|
Forgejo
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
key="discord"
|
||||||
|
href="https://discord.gg/rfmKzM6ASw"
|
||||||
|
>
|
||||||
|
Discord
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="login" href="/login">
|
||||||
|
Log In
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="signup" href="/signup">
|
||||||
|
Sign Up
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
<div className="flex gap-3 items-center">
|
||||||
|
{user && (
|
||||||
|
<NavbarItem>
|
||||||
|
<Link href="/create-post">
|
||||||
|
<Button
|
||||||
|
endContent={<SquarePen />}
|
||||||
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
|
variant="bordered"
|
||||||
|
>
|
||||||
|
Create Post
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Spacer x={32} />
|
||||||
|
</NavbarItem>
|
||||||
|
)}
|
||||||
|
<NavbarItem>
|
||||||
|
<Tooltip
|
||||||
|
delay={1000}
|
||||||
|
content={
|
||||||
|
<div className="px-1 py-2 text-black text-center">
|
||||||
|
<div className="text-small font-bold">GitHub</div>
|
||||||
|
<div className="text-tiny">Source Code</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Link
|
||||||
|
href="https://github.com/Ategon/Jamjar"
|
||||||
|
className="text-white flex justify-center duration-500 ease-in-out transition-all transform hover:scale-125 hover:text-red-100"
|
||||||
|
isExternal
|
||||||
|
>
|
||||||
|
<SiGithub title="" />
|
||||||
|
</Link>
|
||||||
|
</Tooltip>
|
||||||
|
</NavbarItem>
|
||||||
|
<NavbarItem>
|
||||||
|
<Tooltip
|
||||||
|
delay={1000}
|
||||||
|
content={
|
||||||
|
<div className="px-1 py-2 text-black text-center">
|
||||||
|
<div className="text-small font-bold">Forgejo</div>
|
||||||
|
<div className="text-tiny">Source Code</div>
|
||||||
|
</div>
|
||||||
|
}
|
||||||
|
>
|
||||||
|
<Link
|
||||||
|
href="https://git.edikoyo.com/Ategon/Jamjar"
|
||||||
|
className="text-white flex justify-center duration-500 ease-in-out transition-all transform hover:scale-125 hover:text-red-100"
|
||||||
|
isExternal
|
||||||
|
>
|
||||||
|
<SiForgejo title="" />
|
||||||
|
</Link>
|
||||||
|
</Tooltip>
|
||||||
|
</NavbarItem>
|
||||||
|
<NavbarItem>
|
||||||
|
<Link
|
||||||
|
href="https://discord.gg/rfmKzM6ASw"
|
||||||
|
className="text-white flex justify-center duration-500 ease-in-out transition-all transform hover:scale-125 hover:text-indigo-100"
|
||||||
|
isExternal
|
||||||
|
>
|
||||||
|
<SiDiscord />
|
||||||
|
</Link>
|
||||||
|
</NavbarItem>
|
||||||
|
<Divider orientation="vertical" className="h-1/2" />
|
||||||
|
{!user ? (
|
||||||
|
<div className="flex gap-3 items-center">
|
||||||
|
<NavbarItem>
|
||||||
|
<Link href="/login">
|
||||||
|
<Button
|
||||||
|
endContent={<LogInIcon />}
|
||||||
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
|
variant="bordered"
|
||||||
|
>
|
||||||
|
Log In
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</NavbarItem>
|
||||||
|
<NavbarItem>
|
||||||
|
<Link href="/signup">
|
||||||
|
<Button
|
||||||
|
endContent={<NotebookPen />}
|
||||||
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
|
variant="bordered"
|
||||||
|
>
|
||||||
|
Sign up
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</NavbarItem>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Avatar src={user.profilePicture} />
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu>
|
||||||
|
{/* <DropdownItem
|
||||||
|
key="profile"
|
||||||
|
className="text-black"
|
||||||
|
href="/profile"
|
||||||
|
>
|
||||||
|
Profile
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem
|
||||||
|
showDivider
|
||||||
|
key="settings"
|
||||||
|
className="text-black"
|
||||||
|
href="/settings"
|
||||||
|
>
|
||||||
|
Settings
|
||||||
|
</DropdownItem> */}
|
||||||
|
<DropdownItem
|
||||||
|
key="logout"
|
||||||
|
color="danger"
|
||||||
|
className="text-danger"
|
||||||
|
href="/logout"
|
||||||
|
>
|
||||||
|
Logout
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
)}
|
||||||
|
</NavbarContent>
|
||||||
|
</NavbarBase>
|
||||||
|
);
|
||||||
|
}
|
||||||
|
/*
|
||||||
|
|
||||||
|
{isMobile ? (
|
||||||
|
// Mobile view
|
||||||
|
user ? (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Avatar src={user.profilePicture} />
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu>
|
||||||
|
<DropdownItem key="create-post" href="/create-post">
|
||||||
|
Create Post
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="logout" color="danger" href="/logout">
|
||||||
|
Logout
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
) : (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Button auto flat className="text-white">
|
||||||
|
☰
|
||||||
|
</Button>
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu>
|
||||||
|
<DropdownItem key="github" href="https://github.com/Ategon/Jamjar" isExternal>
|
||||||
|
GitHub
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="forgejo" href="https://git.edikoyo.com/Ategon/Jamjar" isExternal>
|
||||||
|
Forgejo
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="discord" href="https://discord.gg/rfmKzM6ASw" isExternal>
|
||||||
|
Discord
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="login" href="/login">
|
||||||
|
Log In
|
||||||
|
</DropdownItem>
|
||||||
|
<DropdownItem key="signup" href="/signup">
|
||||||
|
Sign Up
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
)
|
||||||
|
) : (
|
||||||
|
|
||||||
|
|
||||||
|
user && (
|
||||||
|
<NavbarItem>
|
||||||
|
<Link href="/create-post">
|
||||||
|
<Button
|
||||||
|
endContent={<SquarePen />}
|
||||||
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
|
variant="bordered"
|
||||||
|
>
|
||||||
|
Create Post
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
<Spacer x={32} />
|
||||||
|
</NavbarItem>
|
||||||
|
)
|
||||||
<NavbarItem>
|
<NavbarItem>
|
||||||
<Tooltip
|
<Tooltip
|
||||||
delay={1000}
|
delay={1000}
|
||||||
|
@ -83,16 +382,48 @@ export default function Navbar() {
|
||||||
</Link>
|
</Link>
|
||||||
</NavbarItem>
|
</NavbarItem>
|
||||||
<Divider orientation="vertical" className="h-1/2" />
|
<Divider orientation="vertical" className="h-1/2" />
|
||||||
{/* <NavbarItem>
|
{!user ? (
|
||||||
<Button
|
<div className="flex gap-3 items-center">
|
||||||
endContent={<LogInIcon />}
|
<NavbarItem>
|
||||||
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
<Link href="/login">
|
||||||
variant="bordered"
|
<Button
|
||||||
>
|
endContent={<LogInIcon />}
|
||||||
Log In
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
</Button>
|
variant="bordered"
|
||||||
</NavbarItem> */}
|
>
|
||||||
</NavbarContent>
|
Log In
|
||||||
</NavbarBase>
|
</Button>
|
||||||
);
|
</Link>
|
||||||
}
|
</NavbarItem>
|
||||||
|
<NavbarItem>
|
||||||
|
<Link href="/signup">
|
||||||
|
<Button
|
||||||
|
endContent={<NotebookPen />}
|
||||||
|
className="text-white border-white/50 hover:border-green-100/50 hover:text-green-100 hover:scale-110 transition-all transform duration-500 ease-in-out"
|
||||||
|
variant="bordered"
|
||||||
|
>
|
||||||
|
Sign up
|
||||||
|
</Button>
|
||||||
|
</Link>
|
||||||
|
</NavbarItem>
|
||||||
|
</div>
|
||||||
|
) : (
|
||||||
|
<Dropdown>
|
||||||
|
<DropdownTrigger>
|
||||||
|
<Avatar src={user.profilePicture} />
|
||||||
|
</DropdownTrigger>
|
||||||
|
<DropdownMenu>
|
||||||
|
|
||||||
|
<DropdownItem
|
||||||
|
key="logout"
|
||||||
|
color="danger"
|
||||||
|
className="text-danger"
|
||||||
|
href="/logout"
|
||||||
|
>
|
||||||
|
Logout
|
||||||
|
</DropdownItem>
|
||||||
|
</DropdownMenu>
|
||||||
|
</Dropdown>
|
||||||
|
)}
|
||||||
|
)
|
||||||
|
</NavbarContent> */
|
||||||
|
|
|
@ -1,5 +1,4 @@
|
||||||
import { Avatar, Button, Card, CardBody, Spacer } from "@nextui-org/react";
|
import { Avatar, Card, CardBody, Spacer } from "@nextui-org/react";
|
||||||
import { Heart, MessageCircle } from "lucide-react";
|
|
||||||
import { formatDistance } from "date-fns";
|
import { formatDistance } from "date-fns";
|
||||||
import Link from "next/link";
|
import Link from "next/link";
|
||||||
import { PostType } from "@/types/PostType";
|
import { PostType } from "@/types/PostType";
|
||||||
|
@ -36,14 +35,14 @@ export default function PostCard({ post }: { post: PostType }) {
|
||||||
|
|
||||||
<Spacer y={4} />
|
<Spacer y={4} />
|
||||||
|
|
||||||
<div className="flex gap-3">
|
{/* <div className="flex gap-3">
|
||||||
<Button size="sm">
|
<Button size="sm">
|
||||||
<Heart size={16} /> {post.likers.length}
|
<Heart size={16} /> {post.likers.length}
|
||||||
</Button>
|
</Button>
|
||||||
<Button size="sm">
|
<Button size="sm">
|
||||||
<MessageCircle size={16} /> {0}
|
<MessageCircle size={16} /> {0}
|
||||||
</Button>
|
</Button>
|
||||||
</div>
|
</div> */}
|
||||||
</CardBody>
|
</CardBody>
|
||||||
</Card>
|
</Card>
|
||||||
);
|
);
|
||||||
|
|
|
@ -9,7 +9,11 @@ export default function Posts() {
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
const fetchPosts = async () => {
|
const fetchPosts = async () => {
|
||||||
const response = await fetch("https://jam.edikoyo.com/api/v1/posts");
|
const response = await fetch(
|
||||||
|
process.env.NEXT_PUBLIC_MODE === "PROD"
|
||||||
|
? "https://d2jam.com/api/v1/posts"
|
||||||
|
: "http://localhost:3005/api/v1/posts"
|
||||||
|
);
|
||||||
setPosts(await response.json());
|
setPosts(await response.json());
|
||||||
};
|
};
|
||||||
|
|
||||||
|
|
|
@ -1,3 +1,4 @@
|
||||||
|
import { Spacer } from "@nextui-org/react";
|
||||||
import Timer from "./Timer";
|
import Timer from "./Timer";
|
||||||
|
|
||||||
export default function Timers() {
|
export default function Timers() {
|
||||||
|
@ -7,6 +8,8 @@ export default function Timers() {
|
||||||
name="Jam Start"
|
name="Jam Start"
|
||||||
targetDate={new Date("2025-04-04T18:00:00-05:00")}
|
targetDate={new Date("2025-04-04T18:00:00-05:00")}
|
||||||
/>
|
/>
|
||||||
|
<Spacer y={8} />
|
||||||
|
<p>Site under construction</p>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
|
31
src/helpers/cookie.ts
Normal file
31
src/helpers/cookie.ts
Normal file
|
@ -0,0 +1,31 @@
|
||||||
|
export function getCookies() {
|
||||||
|
const pairs = document.cookie.split(";");
|
||||||
|
const cookies: Record<string, string> = {};
|
||||||
|
for (let i = 0; i < pairs.length; i++) {
|
||||||
|
const pair = pairs[i].split("=");
|
||||||
|
cookies[(pair[0] + "").trim()] = unescape(pair.slice(1).join("="));
|
||||||
|
}
|
||||||
|
return cookies;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function getCookie(cookie: string) {
|
||||||
|
const pairs = document.cookie.split(";");
|
||||||
|
for (let i = 0; i < pairs.length; i++) {
|
||||||
|
const pair = pairs[i].trim().split("=");
|
||||||
|
if (pair[0] == cookie) {
|
||||||
|
return pair[1];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
export function hasCookie(cookie: string) {
|
||||||
|
const pairs = document.cookie.split(";");
|
||||||
|
for (let i = 0; i < pairs.length; i++) {
|
||||||
|
const pair = pairs[i].trim().split("=");
|
||||||
|
if (pair[0] == cookie) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
}
|
7
src/types/UserType.ts
Normal file
7
src/types/UserType.ts
Normal file
|
@ -0,0 +1,7 @@
|
||||||
|
export interface UserType {
|
||||||
|
id: number;
|
||||||
|
slug: string;
|
||||||
|
name: string;
|
||||||
|
profilePicture: string;
|
||||||
|
createdAt: Date;
|
||||||
|
}
|
Loading…
Reference in a new issue