Compare commits

..

1 Commits

Author SHA1 Message Date
m1000
54330f2137 Merge pull request 'feature/auth' (#2) from feature/auth into main
Reviewed-on: Toogether/Client#2
2024-12-13 12:55:23 +00:00
54 changed files with 369 additions and 3639 deletions

View File

@ -10,4 +10,3 @@ OAUTH_AUTHORIZATION_URL=
OAUTH_TOKEN_URL= OAUTH_TOKEN_URL=
OAUTH_USERINFO_URL= OAUTH_USERINFO_URL=
OAUTH_JWKS_ENDPOINT= OAUTH_JWKS_ENDPOINT=
OAUTH_SCOPES="openid email profile"

View File

@ -1,5 +0,0 @@
{
"tabWidth": 4,
"useTabs": true,
"semi": true
}

View File

@ -1,6 +1,6 @@
# syntax=docker.io/docker/dockerfile:1 # syntax=docker.io/docker/dockerfile:1
FROM node:20-alpine AS base FROM node:18-alpine AS base
# Install dependencies only when needed # Install dependencies only when needed
FROM base AS deps FROM base AS deps

View File

@ -3,3 +3,8 @@ services:
image: toogether/webapp image: toogether/webapp
ports: ports:
- "80:3000" - "80:3000"
build:
context: ./
dockerfile: Dockerfile
environment:
- NEXT_PUBLIC_API_URL=http://api:3000

View File

@ -1,33 +1,21 @@
{ {
"name": "@toogether/webapp", "name": "client",
"version": "0.1.0", "version": "0.1.0",
"private": true, "private": true,
"scripts": { "scripts": {
"dev": "next dev -p 4000", "dev": "next dev --turbopack",
"build": "next build", "build": "next build",
"build:docker": "docker build -t toogether/webapp .", "start": "next start",
"start": "next start -p 4000", "lint": "next lint"
"start:prod": "docker compose up --force-recreate -d",
"lint": "next lint",
"format": "prettier --write ."
}, },
"dependencies": { "dependencies": {
"@internationalized/date": "^3.6.0",
"@nextui-org/navbar": "^2.2.7",
"@nextui-org/react": "^2.6.10",
"@nextui-org/system": "^2.4.5",
"@nextui-org/theme": "^2.4.4",
"axios": "^1.7.9", "axios": "^1.7.9",
"axios-hooks": "^5.1.0",
"framer-motion": "^11.15.0",
"jsonwebtoken": "^9.0.2", "jsonwebtoken": "^9.0.2",
"moment-timezone": "^0.5.46", "moment": "^2.30.1",
"next": "15.0.3", "next": "15.0.3",
"next-auth": "^4.24.10", "next-auth": "^4.24.10",
"next-themes": "^0.4.4",
"react": "19.0.0-rc-66855b96-20241106", "react": "19.0.0-rc-66855b96-20241106",
"react-dom": "19.0.0-rc-66855b96-20241106", "react-dom": "19.0.0-rc-66855b96-20241106",
"react-icons": "^5.4.0",
"zustand": "^5.0.2" "zustand": "^5.0.2"
}, },
"devDependencies": { "devDependencies": {
@ -38,7 +26,6 @@
"eslint": "^8", "eslint": "^8",
"eslint-config-next": "15.0.3", "eslint-config-next": "15.0.3",
"postcss": "^8", "postcss": "^8",
"prettier": "^3.4.2",
"tailwindcss": "^3.4.1", "tailwindcss": "^3.4.1",
"typescript": "^5" "typescript": "^5"
} }

View File

@ -1,10 +0,0 @@
import { ClassList } from "@/app/components/Class";
import { Metadata } from "next";
export const metadata: Metadata = {
title: "Classes | Toogether",
};
export default async function Page() {
return <ClassList />;
}

View File

@ -1,50 +0,0 @@
"use client";
import { Sidebar } from "@/app/components/Sidebar";
import { Divider } from "@nextui-org/react";
import { usePathname } from "next/navigation";
import { FaDoorClosed, FaRegUser } from "react-icons/fa";
import { FiSettings } from "react-icons/fi";
import { IoMdStats } from "react-icons/io";
import { MdInbox } from "react-icons/md";
import { SidebarItemProps } from "../components/Sidebar/item";
export default function Layout({ children }: { children: React.ReactNode }) {
const pathName = usePathname();
const sidebarItems = [
[
{ href: "/admin", title: "Dashboard", icon: <IoMdStats /> },
{ href: "/admin/users", title: "Users", icon: <FaRegUser /> },
{ href: "/admin/classes", title: "Classes", icon: <MdInbox /> },
{ href: "/admin/rooms", title: "Rooms", icon: <FaDoorClosed /> },
],
[
{
href: "/admin/settings",
title: "Settings",
icon: <FiSettings />,
},
],
] as SidebarItemProps[][];
return (
<div className="flex">
<Sidebar items={sidebarItems} />
<main className="p-7 w-full">
<div className="flex items-center gap-2">
<h1 className="text-2xl font-semibold">
{
sidebarItems
.flat()
.find((item) => item.href === pathName)?.title
}
</h1>
</div>
<Divider className="mt-2 mb-5 bg-foreground-300" />
{children}
</main>
</div>
);
}

View File

@ -1,11 +0,0 @@
import { Metadata } from "next";
export const metadata: Metadata = {
title: "Admin | Toogether",
};
export default async function Page() {
return (
<p>Hello world!</p>
);
}

View File

@ -1,11 +0,0 @@
import { Metadata } from "next";
export const metadata: Metadata = {
title: "Rooms | Toogether",
};
export default async function Page() {
return (
<p>Hello world!</p>
);
}

View File

@ -1,11 +0,0 @@
import { Metadata } from "next";
export const metadata: Metadata = {
title: "Settings | Toogether",
};
export default async function Page() {
return (
<p>Hello world!</p>
);
}

View File

@ -1,10 +0,0 @@
import { UserList } from "@/app/components/Users";
import { Metadata } from "next";
export const metadata: Metadata = {
title: "Users | Toogether",
};
export default async function Page() {
return <UserList />;
}

View File

@ -1,11 +1,10 @@
import { Metadata } from "next"; "use client";
import { ProvidersList } from "./providersList"; import { authOptions } from "@/authOptions";
import { signIn } from "next-auth/react";
export const metadata: Metadata = { const LoginPage = () => {
title: "Toogether | Connexion", const provider = authOptions.providers[0];
};
export default function LoginPage() {
return ( return (
<div className="flex flex-col h-screen items-center justify-center bg-gradient-to-br from-green-700 to-green-950 bg-[length:200%_200%] animate-gradient-x text-white"> <div className="flex flex-col h-screen items-center justify-center bg-gradient-to-br from-green-700 to-green-950 bg-[length:200%_200%] animate-gradient-x text-white">
<div className="flex flex-col justify-center items-center gap-6 bg-black bg-opacity-40 rounded-md p-5 mx-5 w-2/3"> <div className="flex flex-col justify-center items-center gap-6 bg-black bg-opacity-40 rounded-md p-5 mx-5 w-2/3">
@ -26,8 +25,19 @@ export default function LoginPage() {
<div className="w-full border-t border-white"></div> <div className="w-full border-t border-white"></div>
<h3 className="font-bold text-xl">Via</h3> <h3 className="font-bold text-xl">Via</h3>
<ProvidersList /> <ul>
<li key={provider.id}>
<button
className="bg-white text-black p-2 rounded-md"
onClick={() => signIn(provider.id)}
>
{provider.name}
</button>
</li>
</ul>
</div> </div>
</div> </div>
); );
} };
export default LoginPage;

View File

@ -1,22 +0,0 @@
"use client";
import { authOptions } from "@/authOptions";
import { signIn } from "next-auth/react";
export const ProvidersList = () => {
const providers = authOptions.providers;
return (
<div className="flex gap-2">
{providers.map((provider) => (
<button
key={provider.id}
className="bg-white text-black p-2 rounded-md"
onClick={() => signIn(provider.id)}
>
{provider.name}
</button>
))}
</div>
);
};

View File

@ -1,24 +0,0 @@
"use client";
import { NextUIProvider } from "@nextui-org/react";
import { Session } from "next-auth";
import { SessionProvider } from "next-auth/react";
import { ThemeProvider as NextThemesProvider } from "next-themes";
export default function AppWrapper({
session,
children,
}: {
session: Session | null;
children: React.ReactNode;
}) {
return (
<SessionProvider session={session} refetchOnWindowFocus={false}>
<NextUIProvider>
<NextThemesProvider attribute="class" defaultTheme="dark">
{children}
</NextThemesProvider>
</NextUIProvider>
</SessionProvider>
);
}

View File

@ -1,71 +0,0 @@
"use client";
import { Class } from "@/app/interface/class";
import { classesService } from "@/app/services/classes.service";
import {
Card,
Skeleton,
Table,
TableBody,
TableCell,
TableColumn,
TableHeader,
TableRow,
} from "@nextui-org/react";
import { useEffect, useState } from "react";
export const ClassList = () => {
const [classes, setClasses] = useState<Class[]>();
useEffect(() => {
classesService.getAll().then((response) => {
setClasses(response.data);
});
}, []);
if (!classes) {
return (
<Card className="w-full space-y-5 p-4" radius="lg">
<Skeleton className="rounded-lg">
<div className="h-10 rounded-lg bg-default-300" />
</Skeleton>
<div className="space-y-3">
<Skeleton className="w-3/5 rounded-lg">
<div className="h-3 w-3/5 rounded-lg bg-default-200" />
</Skeleton>
<Skeleton className="w-4/5 rounded-lg">
<div className="h-3 w-4/5 rounded-lg bg-default-200" />
</Skeleton>
<Skeleton className="w-2/5 rounded-lg">
<div className="h-3 w-2/5 rounded-lg bg-default-300" />
</Skeleton>
</div>
</Card>
);
}
if (classes.length === 0) {
return (
<Card className="w-full p-4" radius="lg">
<p className="text-center text-lg text-gray-500 dark:text-gray-400">
No classes found
</p>
</Card>
);
}
return (
<Table aria-label="List of users" className="w-full">
<TableHeader>
<TableColumn>NAME</TableColumn>
</TableHeader>
<TableBody>
{classes.map((Class) => (
<TableRow key={Class.id.toString()}>
<TableCell>{Class.name}</TableCell>
</TableRow>
))}
</TableBody>
</Table>
);
};

View File

@ -0,0 +1,29 @@
"use client";
import { axiosInstance } from "@/app/lib/axios";
import { useRef } from "react";
export const FetchApi = () => {
const listRef = useRef<HTMLUListElement>(null);
const handleFetch = async () => {
const response = await axiosInstance.get<{
message: string;
}>("/ping");
const li = document.createElement("li");
li.textContent = response.data.message;
listRef.current?.appendChild(li);
};
return (
<div>
<button
className="text-white px-2 py-2 bg-green-500 rounded-md"
onClick={handleFetch}
>
Fetch API
</button>
<ul ref={listRef}></ul>
</div>
);
};

View File

@ -0,0 +1,28 @@
import { getServerSession, Session } from "next-auth";
import Link from "next/link";
const Header = async () => {
const session = (await getServerSession()) as Session;
return (
<header className="bg-gray-900 text-white p-4">
<div className="container mx-auto flex justify-between items-center">
<h1 className="text-xl font-bold">Toogether</h1>
<div className="flex items-center gap-2">
<p>
Logged in as <strong>{session.user.name}</strong>
</p>
<nav className="space-x-4">
<Link href="/auth/logout">
<button className="bg-gray-800 px-4 py-2 rounded hover:scale-105 transition duration-200">
Logout
</button>
</Link>
</nav>
</div>
</div>
</header>
);
};
export default Header;

View File

@ -1,128 +0,0 @@
"use client";
import { useClassStore } from "@/app/stores/classStore";
import { useUserStore } from "@/app/stores/userStore";
import { User } from "@/app/types/next-auth";
import { getInitials } from "@/app/utils/initial";
import {
Autocomplete,
AutocompleteItem,
Avatar,
Button,
Dropdown,
DropdownItem,
DropdownMenu,
DropdownTrigger,
Navbar,
NavbarBrand,
NavbarContent,
NavbarItem,
} from "@nextui-org/react";
import { useRouter } from "next/navigation";
import { useEffect } from "react";
import { ThemeSwitcher } from "../ThemeSwitcher/ThemeSwitcher";
export const HeaderContent = ({ user }: { user?: User }) => {
const router = useRouter();
const { classes, fetchClass } = useClassStore();
const {
currentClassId: selectedClassId,
setCurrentClassId: setSelectedClassId,
} = useUserStore();
useEffect(() => {
fetchClass();
}, [fetchClass]);
const selectedClass = classes.find((Class) => Class.id === selectedClassId);
useEffect(() => {
if (!selectedClass && classes.length > 0) {
setSelectedClassId(classes[0].id);
}
}, [selectedClass, classes, setSelectedClassId]);
return (
<Navbar className="mb-2">
<NavbarBrand>
<p className="font-bold text-inherit">Toogether</p>
</NavbarBrand>
<NavbarContent as="div" justify="end">
{user?.roles.includes("admin") ? (
<NavbarItem>
<Button
size="sm"
variant="flat"
className="min-w-0"
onPress={() => router.push("/admin")}
>
🔧
</Button>
</NavbarItem>
) : null}
<NavbarItem>
<ThemeSwitcher />
</NavbarItem>
<Dropdown placement="bottom-end">
<DropdownTrigger>
<Avatar
isBordered
as="button"
className="transition-transform"
color="secondary"
name={user?.name ? getInitials(user.name) : ""}
size="sm"
/>
</DropdownTrigger>
<DropdownMenu
aria-label="Profile Actions"
variant="flat"
closeOnSelect={false}
>
<DropdownItem
key="profile"
className="h-14 gap-2"
showDivider
>
<p>Signed in as</p>
<p className="font-semibold">{user?.name}</p>
</DropdownItem>
<DropdownItem key="class" className="h-14 gap-2">
<Autocomplete
size="sm"
label="Select an class"
value={selectedClass?.name}
selectedKey={selectedClass?.id}
onSelectionChange={(selectedId) => {
const inputSelectedClass = classes.find(
(Class) => Class.id === selectedId,
);
if (inputSelectedClass)
setSelectedClassId(
inputSelectedClass.id,
);
}}
>
{classes.map((Class) => (
<AutocompleteItem key={Class.id}>
{Class.name}
</AutocompleteItem>
))}
</Autocomplete>
</DropdownItem>
<DropdownItem key="settings">Settings</DropdownItem>
<DropdownItem
key="logout"
color="danger"
href="/auth/logout"
>
Logout
</DropdownItem>
</DropdownMenu>
</Dropdown>
</NavbarContent>
</Navbar>
);
};

View File

@ -1,9 +0,0 @@
import { getServerSession } from "next-auth";
import { authOptions } from "@/authOptions";
import { HeaderContent } from "./contents";
export const Header = async () => {
const session = await getServerSession(authOptions);
return <HeaderContent user={session?.user} />;
};

View File

@ -1,90 +0,0 @@
"use client";
import { parseDate, Time } from "@internationalized/date";
import {
Button,
Card,
CardBody,
CardHeader,
DateInput,
Divider,
TimeInput,
} from "@nextui-org/react";
import { Room } from "../../interface/room";
import moment from "moment";
import { useRouter } from "next/navigation";
export const RoomCard = ({ id, name, date, Times, Presentator }: Room) => {
const router = useRouter();
return (
<Card className="w-[300px]">
<CardHeader>
<div className="flex flex-col min-h-20">
<p className="text-md">{name}</p>
<p className="text-small text-default-500">
{Presentator.username}
</p>
</div>
</CardHeader>
<Divider />
<CardBody>
<div className="flex flex-col gap-2" key={`times.${id}`}>
<DateInput
isReadOnly
label="Date"
value={parseDate(moment(date).format("YYYY-MM-DD"))}
/>
{Times.map((time) => (
<div key={`time.${time.id}`}>
<div className="flex items-center gap-2">
<TimeInput
isReadOnly
label="Start"
hourCycle={24}
value={
new Time(
moment(time.startTime).hours(),
moment(time.startTime).minutes(),
)
}
/>
<span>-</span>
<TimeInput
isReadOnly
label="End"
hourCycle={24}
value={
new Time(
moment(time.endTime).hours(),
moment(time.endTime).minutes(),
)
}
/>
</div>
</div>
))}
</div>
</CardBody>
{moment(date).dayOfYear() === moment().dayOfYear() && (
<div className="flex p-2">
<Button
isDisabled={moment(
moment(Times[0].startTime).subtract(5, "minutes"),
).isAfter(moment())}
color="primary"
radius="full"
size="sm"
variant={"flat"}
onPress={() => {
router.push(`/room/${id}`);
}}
>
Join
</Button>
</div>
)}
</Card>
);
};

View File

@ -1,87 +0,0 @@
"use client";
import { useEffect, useRef } from "react";
import { RoomCard } from "./Card";
import { Room } from "../../interface/room";
import { SkeletonRoomCard } from "./SkeletonRoomCard";
export const RoomList = ({ rooms }: { rooms: Room[] | null }) => {
const scrollContainerRef = useRef<HTMLUListElement>(null);
const handleWheel = (event: WheelEvent) => {
if (event.deltaY === 0) return;
if (event.ctrlKey || event.shiftKey || event.altKey) return;
const scrollContainer = scrollContainerRef.current;
if (!scrollContainer) return;
const goLeft = event.deltaY < 0;
const isEnd = goLeft
? scrollContainer.scrollLeft === 0
: scrollContainer.scrollLeft + scrollContainer.clientWidth >=
scrollContainer.scrollWidth;
if (isEnd) return;
event.preventDefault();
const scrollAmount = 10;
const direction = event.deltaY > 0 ? 1 : -1;
let scrollCount = 0;
const interval = setInterval(() => {
scrollContainer.scrollLeft += scrollAmount * direction;
scrollCount += scrollAmount;
if (scrollCount >= 100) {
clearInterval(interval);
}
}, 10);
};
useEffect(() => {
const scrollContainer = scrollContainerRef.current;
if (!scrollContainer) return;
scrollContainer.addEventListener("wheel", handleWheel);
return () => {
scrollContainer.removeEventListener("wheel", handleWheel);
};
}, []);
if (rooms?.length == 0)
return (
<div className="flex gap-2 rounded-xl bg-default-100 p-4">
<p className="text-default-500 text-center w-full">
No rooms found
</p>
</div>
);
return (
<ul
ref={scrollContainerRef}
className="flex gap-2 overflow-x-auto scrollbar-hide rounded-xl bg-default-100 p-1"
>
{rooms ? (
rooms.map((room) => (
<li key={room.id}>
<RoomCard
id={room.id}
name={room.name}
date={room.date}
Presentator={room.Presentator}
Times={room.Times}
/>
</li>
))
) : (
<>
{Array.from({ length: 3 }).map((_, i) => (
<li key={i}>
<SkeletonRoomCard />
</li>
))}
</>
)}
</ul>
);
};

View File

@ -1,30 +0,0 @@
"use client";
import { Card, Skeleton, Divider } from "@nextui-org/react";
export const SkeletonRoomCard = () => {
return (
<Card className="w-[200px] space-y-5 p-4" radius="lg">
<div className="flex flex-col gap-2">
<Skeleton className="w-4/5 rounded-lg">
<div className="h-3 w-4/5 rounded-lg bg-default-200" />
</Skeleton>
<Skeleton className="w-2/5 rounded-lg">
<div className="h-3 w-2/5 rounded-lg bg-default-300" />
</Skeleton>
</div>
<Divider />
<Skeleton className="rounded-lg">
<div className="h-12 rounded-lg bg-default-300" />
</Skeleton>
<div className="flex items-center gap-2">
<Skeleton className="rounded-lg w-1/2">
<div className="h-10 rounded-lg bg-default-300" />
</Skeleton>
<span>-</span>
<Skeleton className="rounded-lg w-1/2">
<div className="h-10 rounded-lg bg-default-300" />
</Skeleton>
</div>
</Card>
);
};

View File

@ -1,31 +0,0 @@
"use client";
import { useRoomStore } from "@/app/stores/roomStore";
import { useUserStore } from "@/app/stores/userStore";
import { useEffect } from "react";
import { RoomList } from "./List";
export const RoomTable = () => {
const { fetchRooms, actual, future, past } = useRoomStore();
const { currentClassId: selectedClassId } = useUserStore();
useEffect(() => {
if (selectedClassId) fetchRooms();
}, [fetchRooms, selectedClassId]);
return (
<div className="flex flex-col gap-4">
<section className="flex flex-col gap-2">
<h2 className="font-semibold text-lg">Upcoming</h2>
<RoomList rooms={future} />
</section>
<section className="flex flex-col gap-2">
<h2 className="font-semibold text-lg">Today</h2>
<RoomList rooms={actual} />
</section>
<section className="flex flex-col gap-2">
<h2 className="font-semibold text-lg">Past</h2>
<RoomList rooms={past} />
</section>
</div>
);
};

View File

@ -0,0 +1,11 @@
"use client";
import { SessionProvider } from "next-auth/react";
export default function SessionProviderWrapper({
children,
}: {
children: React.ReactNode;
}) {
return <SessionProvider>{children}</SessionProvider>;
}

View File

@ -1,129 +0,0 @@
import { Divider, Link } from "@nextui-org/react";
import { usePathname, useRouter } from "next/navigation";
import { useEffect, useState } from "react";
import {
BsLayoutSidebarInset,
BsReverseLayoutSidebarInsetReverse,
} from "react-icons/bs";
import { ThemeSwitcher } from "../ThemeSwitcher/ThemeSwitcher";
import { SidebarItem } from "./item";
import { useUserStore } from "@/app/stores/userStore";
export const Sidebar = ({
items,
}: {
items: {
href: string;
title: string;
icon: React.ReactNode;
}[][];
}) => {
const router = useRouter();
const pathName = usePathname();
// State to manage menu openness
const { setSidebarIsOpen, sidebarIsOpen } = useUserStore();
const [isMobile, setIsMobile] = useState(false);
useEffect(() => {
const handleResize = () => {
if (sidebarIsOpen) return;
if (window.innerWidth <= 1024 && !isMobile) {
setIsMobile(true);
} else if (window.innerWidth > 1024 && isMobile) {
setIsMobile(false);
}
};
handleResize();
window.addEventListener("resize", handleResize);
return () => window.removeEventListener("resize", handleResize);
});
useEffect(() => {
if (isMobile) {
setSidebarIsOpen(false);
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [isMobile]);
return (
<aside
className={`
${sidebarIsOpen ? "w-96" : "w-20"} z-40 h-screen transition-all bg-foreground-100
`}
>
{/* items-center lg:items-start */}
<div
className={`flex flex-col h-full px-3 py-4 overflow-y-auto scrollbar-hide ${sidebarIsOpen ? "items-start" : "items-center"}`}
>
<div
className={`flex flex-col gap-2 p-2 mb-4 w-full ${!sidebarIsOpen ? "items-center" : ""}`}
>
{sidebarIsOpen ? (
<div className="flex items-center justify-between w-full">
<Link
className="text-2xl font-semibold text-gray-900 dark:text-white rounded-lg cursor-pointer"
onPress={() => router.push("/")}
>
Toogether
</Link>
<button
onClick={() => setSidebarIsOpen(!sidebarIsOpen)}
>
<BsLayoutSidebarInset className="text-xl" />
</button>
</div>
) : (
<button
onClick={() => setSidebarIsOpen(!sidebarIsOpen)}
>
<BsReverseLayoutSidebarInsetReverse className="text-xl" />
</button>
)}
{sidebarIsOpen && (
<p className="text-sm text-foreground-400 dark:text-gray-400">
Manage classes, rooms, and users
</p>
)}
</div>
{items.map((group, index) => (
<div key={index} className={sidebarIsOpen ? "w-full" : ""}>
<section className="flex flex-col">
<ul className={`flex flex-col gap-2 w-full`}>
{group.map((item, index) => (
<li
key={index}
className={`${
pathName === item.href
? "bg-foreground-300"
: "hover:bg-foreground-200"
} rounded-md cursor-pointer w-full flex `}
>
<SidebarItem
href={item.href}
title={item.title}
icon={item.icon}
setOpen={setSidebarIsOpen}
showTitle={sidebarIsOpen}
isMobile={isMobile}
/>
</li>
))}
</ul>
</section>
{index < items.length - 1 && (
<Divider className="bg-foreground-300 my-4" />
)}
</div>
))}
<section
className={`mt-auto ${!sidebarIsOpen && "justify-center"}`}
>
<ThemeSwitcher />
</section>
</div>
</aside>
);
};

View File

@ -1,37 +0,0 @@
"use client";
import { Link } from "@nextui-org/react";
import { useRouter } from "next/navigation";
export type SidebarItemProps = {
title: string;
icon: React.ReactNode;
href: string;
setOpen: (open: boolean) => void;
showTitle: boolean;
isMobile: boolean;
};
export const SidebarItem = ({
title,
icon,
href,
setOpen,
showTitle = true,
isMobile
}: SidebarItemProps) => {
const router = useRouter();
return (
<Link
onPress={() => {
router.push(href);
if (isMobile) setOpen(false);
}}
color="foreground"
className="w-full p-2 gap-3 text-md"
>
<span className="text-xl">{icon}</span>
{showTitle && <span className="text-sm">{title}</span>}
</Link>
);
};

View File

@ -1,26 +0,0 @@
"use client";
import { Button } from "@nextui-org/react";
import { useTheme } from "next-themes";
import { useEffect, useState } from "react";
export const ThemeSwitcher = () => {
const [mounted, setMounted] = useState(false);
const { theme, setTheme } = useTheme();
useEffect(() => {
setMounted(true);
}, []);
return (
<Button
size="sm"
variant="flat"
className="min-w-0"
onPress={() =>
mounted && setTheme(theme === "light" ? "dark" : "light")
}
>
{mounted && theme === "light" ? "🌑" : "☀️"}
</Button>
);
};

View File

@ -1,81 +0,0 @@
"use client";
import { User } from "@/app/interface/user";
import { usersService } from "@/app/services/users.service";
import {
Card,
Skeleton,
Table,
TableBody,
TableCell,
TableColumn,
TableHeader,
TableRow,
} from "@nextui-org/react";
import { useEffect, useState } from "react";
export const UserList = () => {
const [users, setUsers] = useState<User[]>();
useEffect(() => {
usersService.getAll().then((response) => {
setUsers(response.data);
});
}, []);
if (!users) {
return (
<Card className="w-full space-y-5 p-4" radius="lg">
<Skeleton className="rounded-lg">
<div className="h-10 rounded-lg bg-default-300" />
</Skeleton>
<div className="space-y-3">
<Skeleton className="w-3/5 rounded-lg">
<div className="h-3 w-3/5 rounded-lg bg-default-200" />
</Skeleton>
<Skeleton className="w-4/5 rounded-lg">
<div className="h-3 w-4/5 rounded-lg bg-default-200" />
</Skeleton>
<Skeleton className="w-2/5 rounded-lg">
<div className="h-3 w-2/5 rounded-lg bg-default-300" />
</Skeleton>
</div>
</Card>
);
}
if (users.length === 0) {
return (
<Card className="w-full p-4" radius="lg">
<p className="text-center text-lg text-gray-500 dark:text-gray-400">
No users found
</p>
</Card>
);
}
return (
<Table aria-label="List of users" className="w-full">
<TableHeader>
<TableColumn>USERNAME</TableColumn>
<TableColumn>CLASS</TableColumn>
</TableHeader>
<TableBody>
{users.map((user) => (
<TableRow key={user.id.toString()}>
<TableCell>{user.username}</TableCell>
<TableCell className="flex flex-wrap gap-1">
{user.Class.map((c) => (
<span
className="px-2 rounded-md bg-foreground-100"
key={c.id.toString()}
>
{c.name}
</span>
))}
</TableCell>
</TableRow>
))}
</TableBody>
</Table>
);
};

View File

@ -1,16 +0,0 @@
const NEXT_PUBLIC_API_URL = process.env.NEXT_PUBLIC_API_URL;
export const API_URLS = {
class: {
all: `${NEXT_PUBLIC_API_URL}/@me/class`,
},
room: {
all: (classId: string) =>
`${NEXT_PUBLIC_API_URL}/@me/class/${classId}/rooms`,
},
admin: {
user: {
all: `${NEXT_PUBLIC_API_URL}/user`,
},
},
};

View File

@ -1,19 +0,0 @@
import { authOptions } from "@/authOptions";
import { getServerSession } from "next-auth";
import { HeaderContent } from "../components/Header/contents";
export default async function Layout({
children,
}: {
children: React.ReactNode;
}) {
const session = await getServerSession(authOptions);
return (
<div>
<HeaderContent user={session?.user} />
<main>{children}</main>
</div>
);
}

View File

@ -1,16 +0,0 @@
import { Metadata } from "next";
import { RoomTable } from "../components/Room/Table";
export const metadata: Metadata = {
title: "Toogether | Home",
description:
"Toogether is a platform that allows you to create and join rooms to study together.",
};
export default async function Page() {
return (
<div className="flex flex-col gap-8 p-4">
<RoomTable />
</div>
);
}

View File

@ -1,5 +0,0 @@
export interface Class {
id: string;
name: string;
createdAt: string;
}

View File

@ -1,17 +0,0 @@
export interface Room {
id: string;
name: string;
date: string;
Times: {
id: number;
startTime: string;
endTime: string;
roomId: string;
}[];
Presentator: {
id: string;
username: string;
role: string;
createdAt: string;
};
}

View File

@ -1,10 +0,0 @@
export interface User {
id: string;
username: string;
Class: {
id: string;
name: string;
createdAt: string;
}[];
}

View File

@ -1,18 +1,15 @@
import { getSession } from "next-auth/react"; import SessionProviderWrapper from "./components/SessionProviderWrapper";
import AppWrapper from "./components/AppWrapper";
import "./globals.css"; import "./globals.css";
export default async function RootLayout({ export default function RootLayout({
children, children,
}: { }: {
children: React.ReactNode; children: React.ReactNode;
}) { }) {
const session = await getSession()
return ( return (
<html lang="fr" suppressHydrationWarning> <html lang="fr">
<body> <body>
<AppWrapper session={session}>{children}</AppWrapper> <SessionProviderWrapper>{children}</SessionProviderWrapper>
</body> </body>
</html> </html>
); );

View File

@ -1,8 +1,6 @@
import axios from "axios"; import axios from "axios";
import moment, { Moment } from "moment"; import moment, { Moment } from "moment";
import { getSession, signOut } from "next-auth/react"; import { getSession } from "next-auth/react";
moment.locale("fr");
let cachedAccessToken: string | null = null; let cachedAccessToken: string | null = null;
let tokenExpirationAt: Moment | null = null; let tokenExpirationAt: Moment | null = null;
@ -11,45 +9,20 @@ export const axiosInstance = axios.create({
baseURL: process.env.NEXT_PUBLIC_API_URL, baseURL: process.env.NEXT_PUBLIC_API_URL,
}); });
let isRefreshing = false;
let refreshQueue: Array<(token: string) => void> = [];
axiosInstance.interceptors.request.use(async (config) => { axiosInstance.interceptors.request.use(async (config) => {
if (tokenExpirationAt && moment().isBefore(tokenExpirationAt)) { if (tokenExpirationAt && moment().isBefore(tokenExpirationAt)) {
config.headers.Authorization = `Bearer ${cachedAccessToken}`; config.headers.Authorization = `Bearer ${cachedAccessToken}`;
return config; return config;
} }
if (isRefreshing) {
// Add the request to the queue
return new Promise((resolve) => {
refreshQueue.push((token: string) => {
config.headers.Authorization = `Bearer ${token}`;
resolve(config);
});
});
}
isRefreshing = true;
try {
const session = await getSession(); const session = await getSession();
if (!session || session.error) { if (!session) {
console.log("No session found, redirecting to login page"); throw new Error("User is not authenticated");
await signOut({ callbackUrl: "/auth/login" });
// cancel the request
return new Promise(() => {});
} }
cachedAccessToken = session.accessToken; cachedAccessToken = session.accessToken;
tokenExpirationAt = moment(session.accessTokenExpires); tokenExpirationAt = moment(session.accessTokenExpires);
// Execute the queue
refreshQueue.forEach((cb) => cb(cachedAccessToken!));
refreshQueue = [];
config.headers.Authorization = `Bearer ${cachedAccessToken}`; config.headers.Authorization = `Bearer ${cachedAccessToken}`;
return config; return config;
} finally {
isRefreshing = false;
}
}); });

View File

@ -0,0 +1,3 @@
export const UppercaseFirstLetter = (str: string) => {
return str.slice(0, 1).toLocaleUpperCase() + str.slice(1);
}

View File

@ -1,20 +1,13 @@
import { Metadata } from "next"; import Header from "./components/Header";
import { Header } from "./components/Header"; import { FetchApi } from "./components/FetchApi";
import { RoomTable } from "./components/Room/Table";
export const metadata: Metadata = { const HomePage = () => {
title: "Toogether | Home",
description:
"Toogether is a platform that allows you to create and join rooms to study together.",
};
export default async function Page() {
return ( return (
<> <>
<Header /> <Header />
<main className="flex flex-col gap-8 p-4"> <FetchApi />
<RoomTable />
</main>
</> </>
); );
} };
export default HomePage;

View File

@ -1,7 +0,0 @@
import { API_URLS } from "../constants/apiUrl.constant";
import { Class } from "../interface/class";
import { axiosInstance } from "../lib/axios";
const getAll = () => axiosInstance.get<Class[]>(API_URLS.class.all);
export const classesService = { getAll };

View File

@ -1,7 +0,0 @@
import { API_URLS } from "../constants/apiUrl.constant";
import { axiosInstance } from "../lib/axios";
import { Room } from "../interface/room";
const getAll = (classId: string) => axiosInstance.get<Room[]>(API_URLS.room.all(classId));
export const roomsService = { getAll };

View File

@ -1,7 +0,0 @@
import { API_URLS } from "../constants/apiUrl.constant";
import { User } from "../interface/user";
import { axiosInstance } from "../lib/axios";
const getAll = () => axiosInstance.get<User[]>(API_URLS.admin.user.all);
export const usersService = { getAll };

View File

@ -1,36 +0,0 @@
import { create } from "zustand";
import { classesService } from "../services/classes.service";
import { Class } from "../interface/class";
type ClassStoreState = {
classes: Class[];
};
type ClassStoreActions = {
_setClass: (classes: Class[]) => void;
fetchClass: () => Promise<Class[]>;
checkClass: (id: string) => boolean;
};
type ClassStore = ClassStoreState & ClassStoreActions;
const defaultState: ClassStoreState = {
classes: [],
};
export const useClassStore = create<ClassStore>()((set) => ({
...defaultState,
_setClass: (classes: Class[]) => {
set(() => ({
classes: classes,
}));
},
fetchClass: async () => {
const classResponse = await classesService.getAll();
useClassStore.getState()._setClass(classResponse.data);
return classResponse.data;
},
checkClass: () => {
return true
},
}));

View File

@ -1,49 +0,0 @@
import moment from "moment-timezone";
import { create } from "zustand";
import { Room } from "../interface/room";
import { useUserStore } from "./userStore";
import { roomsService } from "../services/rooms.service";
moment.tz.setDefault("UTC");
type RoomStoreState = {
future: Room[] | null;
actual: Room[] | null;
past: Room[] | null;
};
type RoomStoreActions = {
_setRooms: (rooms: Room[]) => void;
fetchRooms: () => void;
};
type RoomStore = RoomStoreState & RoomStoreActions;
const defaultState: RoomStoreState = {
future: null,
actual: null,
past: null,
};
export const useRoomStore = create<RoomStore>()((set) => ({
...defaultState,
_setRooms: (rooms) => {
const future = rooms.filter((room) =>
moment(room.date).isAfter(moment(), "day"),
);
const actual = rooms.filter((room) =>
moment(room.date).isSame(moment(), "day"),
);
const past = rooms.filter((room) =>
moment(room.date).isBefore(moment(), "day"),
);
set({ future, actual, past });
},
fetchRooms: async () => {
const selectedClassId = useUserStore.getState().currentClassId;
if (!selectedClassId) return;
const roomResponse = await roomsService.getAll(selectedClassId);
useRoomStore.getState()._setRooms(roomResponse.data);
},
}));

View File

@ -1,37 +0,0 @@
import { create } from "zustand";
import { persist, createJSONStorage } from "zustand/middleware";
type UserStoreState = {
currentClassId: string | null;
sidebarIsOpen: boolean;
};
type UserStoreActions = {
setCurrentClassId: (classId: string) => void;
setSidebarIsOpen: (isOpen: boolean) => void;
};
type UserStore = UserStoreState & UserStoreActions;
const defaultState: UserStoreState = {
currentClassId: null,
sidebarIsOpen: false,
};
export const useUserStore = create<UserStore>()(
persist(
(set) => ({
...defaultState,
setCurrentClassId: (classId: string) => {
set({ currentClassId: classId });
},
setSidebarIsOpen: (isOpen: boolean) => {
set({ sidebarIsOpen: isOpen });
},
}),
{
name: "userStore",
storage: createJSONStorage(() => localStorage),
},
),
);

View File

@ -8,6 +8,5 @@ declare namespace NodeJS {
OAUTH_USERINFO_URL: string; OAUTH_USERINFO_URL: string;
OAUTH_ISSUER: string; OAUTH_ISSUER: string;
OAUTH_JWKS_ENDPOINT: string; OAUTH_JWKS_ENDPOINT: string;
OAUTH_SCOPES: string;
} }
} }

View File

@ -1,7 +1,7 @@
/* eslint-disable @typescript-eslint/no-unused-vars */ /* eslint-disable @typescript-eslint/no-unused-vars */
import { Moment } from "moment"; import { Moment } from "moment";
import { DefaultSession } from "next-auth"; import NextAuth, { DefaultSession } from "next-auth";
import "next-auth/jwt"; import { JWT } from "next-auth/jwt";
interface User { interface User {
id: string; id: string;
@ -10,14 +10,6 @@ interface User {
preferred_username: string; preferred_username: string;
given_name: string; given_name: string;
family_name: string; family_name: string;
email: string;
roles: string[];
}
export interface JWTDecoded {
realm_access: {
roles: string[];
};
} }
declare module "next-auth" { declare module "next-auth" {
@ -25,39 +17,26 @@ declare module "next-auth" {
accessToken: string; accessToken: string;
refreshToken: string; refreshToken: string;
accessTokenExpires: Moment; accessTokenExpires: Moment;
error?: string; error?: Error;
user: User; user: User;
} }
interface Account { interface Account {
provider: string;
type: string;
providerAccountId: string;
access_token: string;
expires_at: number; expires_at: number;
refresh_expires_in: number; access_token: string;
refresh_token: string; refresh_token: string;
token_type: string;
id_token: string;
session_state: string;
scope: string;
} }
} }
declare module "next-auth/jwt" { declare module "next-auth/jwt" {
interface JWT { interface JWT {
// Default properties
name: string;
email: string;
picture: string;
sub: string;
// Custom properties
accessToken: string; accessToken: string;
accessTokenExpires: Moment; accessTokenExpires: Moment;
refreshToken: string; refreshToken: string;
refreshTokenExpires: Moment | undefined; error?: Error;
error?: string;
user: User | AdapterUser; user: User | AdapterUser;
iat: number;
exp: number;
jti: string;
} }
} }

View File

@ -1,13 +0,0 @@
export const getInitials = (name: string) => {
if (!name) return "";
const nameParts = name.split(" ");
if (nameParts.length === 1) {
return name;
}
const firstInitial = nameParts[0]?.[0] || "";
const secondInitial = nameParts[1]?.[0] || nameParts[0]?.[1] || "";
return firstInitial + secondInitial;
};

View File

@ -1,11 +1,7 @@
import axios, { AxiosResponse } from "axios"; import axios from "axios";
import moment from "moment-timezone"; import moment from "moment";
import { AuthOptions, Session } from "next-auth"; import { AuthOptions, Session } from "next-auth";
import { JWT } from "next-auth/jwt"; import { JWT } from "next-auth/jwt";
import jsonwebtoken from "jsonwebtoken";
import { JWTDecoded } from "./app/types/next-auth";
moment.tz.setDefault("America/New_York");
export const authOptions: AuthOptions = { export const authOptions: AuthOptions = {
providers: [ providers: [
@ -18,7 +14,7 @@ export const authOptions: AuthOptions = {
authorization: { authorization: {
url: process.env.OAUTH_AUTHORIZATION_URL, url: process.env.OAUTH_AUTHORIZATION_URL,
params: { params: {
scope: process.env.OAUTH_SCOPES, scope: "openid profile offline_access",
response_type: "code", response_type: "code",
}, },
}, },
@ -28,7 +24,6 @@ export const authOptions: AuthOptions = {
userinfo: process.env.OAUTH_USERINFO_URL, userinfo: process.env.OAUTH_USERINFO_URL,
issuer: process.env.OAUTH_ISSUER, issuer: process.env.OAUTH_ISSUER,
jwks_endpoint: process.env.OAUTH_JWKS_ENDPOINT, jwks_endpoint: process.env.OAUTH_JWKS_ENDPOINT,
wellKnown: `${process.env.OAUTH_ISSUER}/.well-known/openid-configuration`,
profile(profile: Session["user"]) { profile(profile: Session["user"]) {
return { return {
id: profile.sub || profile.id, id: profile.sub || profile.id,
@ -36,51 +31,24 @@ export const authOptions: AuthOptions = {
profile.name || profile.name ||
profile.preferred_username || profile.preferred_username ||
`${profile.given_name} ${profile.family_name}`, `${profile.given_name} ${profile.family_name}`,
email: profile.email,
}; };
}, },
}, },
], ],
callbacks: { callbacks: {
async jwt({ token, account, user: profile }) { async jwt({ token, account, user }) {
// Initial sign in if (account && user) {
if (account && profile) {
token.accessToken = account.access_token; token.accessToken = account.access_token;
token.accessTokenExpires = moment(account.expires_at * 1000).subtract(5, "s");
token.refreshToken = account.refresh_token; token.refreshToken = account.refresh_token;
token.user = user;
token.accessTokenExpires = moment.unix(account.expires_at);
token.refreshTokenExpires =
account.refresh_expires_in != 0
? moment().add(account.refresh_expires_in, "seconds")
: undefined;
const accessTokenDecode = jsonwebtoken.decode(
account.access_token,
) as JWTDecoded;
token.user = {
...profile,
roles: accessTokenDecode.realm_access.roles,
};
return token; return token;
} }
if ( if (moment().isBefore(moment(token.accessTokenExpires))) {
token.refreshTokenExpires &&
moment().isAfter(token.refreshTokenExpires)
)
return {
...token,
error: "Refresh token has expired",
};
// Return previous token if the access token has not expired yet
if (moment().isBefore(moment(token.accessTokenExpires)))
return token; return token;
}
// Access token has expired, try to refresh it
return refreshAccessToken(token); return refreshAccessToken(token);
}, },
async session({ session, token }) { async session({ session, token }) {
@ -97,23 +65,14 @@ export const authOptions: AuthOptions = {
pages: { pages: {
signIn: "/auth/login", signIn: "/auth/login",
signOut: "/auth/logout", signOut: "/auth/logout",
}, }
}; };
const refreshAccessToken = async (token: JWT): Promise<JWT> => { const refreshAccessToken = async (token: JWT): Promise<JWT> => {
try {
const response = await axios.post<{ const response = await axios.post<{
access_token: string; access_token: string;
expires_in: number; expires_in: number;
id_token: string;
"not-before-policy": number;
refresh_expires_in: number;
refresh_token: string; refresh_token: string;
scope: string;
session_state: string;
token_type: string;
error?: string;
error_description?: string; error_description?: string;
}>( }>(
process.env.OAUTH_TOKEN_URL, process.env.OAUTH_TOKEN_URL,
@ -127,27 +86,19 @@ const refreshAccessToken = async (token: JWT): Promise<JWT> => {
headers: { headers: {
"Content-Type": "application/x-www-form-urlencoded", "Content-Type": "application/x-www-form-urlencoded",
}, },
}, }
); );
if (response.status !== 200) throw response.data; if (response.status != 200) {
throw new Error(
response.data.error_description || "Failed to refresh access token"
);
}
return { return {
...token, ...token,
accessToken: response.data.access_token, accessToken: response.data.access_token,
accessTokenExpires: moment() accessTokenExpires: moment().add(response.data.expires_in, "seconds").subtract(5, "s"),
.add(response.data.expires_in, "seconds")
.subtract(5, "s"),
refreshToken: response.data.refresh_token, refreshToken: response.data.refresh_token,
}; };
} catch (e) {
console.error(
"Error refreshing access token",
(e as AxiosResponse).data,
);
return {
...token,
error: "RefreshAccessTokenError",
};
}
}; };

View File

@ -17,14 +17,6 @@ export async function middleware(req: NextRequest) {
return NextResponse.redirect(url); return NextResponse.redirect(url);
} }
if (
!token?.user.roles.includes("admin") &&
url.pathname.startsWith("/admin")
) {
url.pathname = "/";
return NextResponse.redirect(url);
}
return NextResponse.next(); return NextResponse.next();
} }

View File

@ -1,12 +1,10 @@
import type { Config } from "tailwindcss"; import type { Config } from "tailwindcss";
import { nextui } from "@nextui-org/react";
export default { export default {
content: [ content: [
"./src/pages/**/*.{js,ts,jsx,tsx,mdx}", "./src/pages/**/*.{js,ts,jsx,tsx,mdx}",
"./src/components/**/*.{js,ts,jsx,tsx,mdx}", "./src/components/**/*.{js,ts,jsx,tsx,mdx}",
"./src/app/**/*.{js,ts,jsx,tsx,mdx}", "./src/app/**/*.{js,ts,jsx,tsx,mdx}",
"./node_modules/@nextui-org/theme/dist/**/*.{js,ts,jsx,tsx}",
], ],
theme: { theme: {
extend: { extend: {
@ -14,8 +12,17 @@ export default {
background: "var(--background)", background: "var(--background)",
foreground: "var(--foreground)", foreground: "var(--foreground)",
}, },
animation: {
"gradient-x": "gradient-x 5s ease infinite",
},
keyframes: {
"gradient-x": {
"0%": { "background-position": "0% 50%" },
"50%": { "background-position": "100% 50%" },
"100%": { "background-position": "0% 50%" },
}, },
}, },
darkMode: "class", },
plugins: [nextui()], },
plugins: [],
} satisfies Config; } satisfies Config;

2133
yarn.lock

File diff suppressed because it is too large Load Diff