diff --git a/frontend/app/form-tutor/ui/table/columns.tsx b/frontend/app/(form-tutor)/form-tutor/columns.tsx similarity index 92% rename from frontend/app/form-tutor/ui/table/columns.tsx rename to frontend/app/(form-tutor)/form-tutor/columns.tsx index e7c63f9..bb8ea6c 100644 --- a/frontend/app/form-tutor/ui/table/columns.tsx +++ b/frontend/app/(form-tutor)/form-tutor/columns.tsx @@ -2,11 +2,11 @@ import { ColumnDef } from "@tanstack/react-table"; import { Checkbox } from "@/components/ui/checkbox"; -import { TutorFormEventsQuery } from "@/lib/gql/generated/graphql"; import { eventBroker } from "@/lib/eventBroker"; import { Badge } from "@/components/ui/badge"; +import {Event} from "@/lib/gql/generated/graphql"; -export const columns: ColumnDef[] = [ +export const columns: ColumnDef[] = [ { accessorKey: "isSelected", header: "", diff --git a/frontend/app/form-tutor/ui/table/data-table.tsx b/frontend/app/(form-tutor)/form-tutor/data-table.tsx similarity index 100% rename from frontend/app/form-tutor/ui/table/data-table.tsx rename to frontend/app/(form-tutor)/form-tutor/data-table.tsx diff --git a/frontend/app/form-tutor/ui/table/event-table.tsx b/frontend/app/(form-tutor)/form-tutor/event-table.tsx similarity index 57% rename from frontend/app/form-tutor/ui/table/event-table.tsx rename to frontend/app/(form-tutor)/form-tutor/event-table.tsx index 9f49cb4..4771664 100644 --- a/frontend/app/form-tutor/ui/table/event-table.tsx +++ b/frontend/app/(form-tutor)/form-tutor/event-table.tsx @@ -1,16 +1,13 @@ -import {TutorFormEventsQuery} from "@/lib/gql/generated/graphql"; +import {Event} from "@/lib/gql/generated/graphql"; import { columns } from "./columns"; import { DataTable } from "./data-table"; -import { getEvents } from "@/app/actions"; import { useEffect, useState } from "react"; const EventTable = () => { - const [data, setData] = useState(null); + const [data, setData] = useState(null); useEffect(() => { const fetchData = async () => { - const result = await getEvents(); - setData(result); }; fetchData(); @@ -22,7 +19,7 @@ const EventTable = () => { return (
- +
); }; diff --git a/frontend/app/(form-tutor)/form-tutor/layout.tsx b/frontend/app/(form-tutor)/form-tutor/layout.tsx new file mode 100644 index 0000000..828a327 --- /dev/null +++ b/frontend/app/(form-tutor)/form-tutor/layout.tsx @@ -0,0 +1,21 @@ +import type { Metadata } from "next"; +import { Inter } from "next/font/google"; + +const inter = Inter({ subsets: ["latin"] }); + +export const metadata: Metadata = { + title: "Tutor:innen Anmeldung", + description: "Anmeldung für Tutor:innen des Vorkurs", +}; + +export default function RootLayout({ + children, +}: Readonly<{ + children: React.ReactNode; +}>) { + return ( + + {children} + + ); +} diff --git a/frontend/components/tutor-registration-form.tsx b/frontend/app/(form-tutor)/form-tutor/page.tsx similarity index 52% rename from frontend/components/tutor-registration-form.tsx rename to frontend/app/(form-tutor)/form-tutor/page.tsx index 4bb29c4..70992f1 100644 --- a/frontend/components/tutor-registration-form.tsx +++ b/frontend/app/(form-tutor)/form-tutor/page.tsx @@ -1,23 +1,21 @@ -"use client"; +"use client" -import { useFormState } from "react-dom"; -import { SubmitButton } from "./submit-button"; -import { FieldError } from "./field-error"; -import { EMPTY_FORM_STATE } from "@/lib/to-form-state"; -import { addTutor } from "@/app/actions"; +import EventTable from "./event-table"; +import { Button } from "@/components/ui/button"; import { Input } from "@/components/ui/input"; -import dynamic from "next/dynamic"; -const EventTable = dynamic(() => import("@/app/form-tutor/ui/table/event-table"), { - ssr: false, -}); - -const TutorRegistrationForm = () => { +export default function Page() { const inputDivStyling = "w-full my-1 grow-0"; const tableDivStyling = "my-5 md:min-h-32 grow"; - const [formState, action] = useFormState(addTutor, EMPTY_FORM_STATE); + + const action = () => { + alert("hello") + } return ( +
+
+

Anmeldung Vorkurstutor:in

@@ -34,11 +32,9 @@ const TutorRegistrationForm = () => {
- - - + +
+
); -}; - -export { TutorRegistrationForm }; +} diff --git a/frontend/app/(planner)/applications/page.tsx b/frontend/app/(planner)/applications/page.tsx new file mode 100644 index 0000000..9293c20 --- /dev/null +++ b/frontend/app/(planner)/applications/page.tsx @@ -0,0 +1,10 @@ + +export const metadata = { + title: "Anmeldungen", +} + +export default function OverviewPage() { + return ( +

Anmeldungen

+ ) +} diff --git a/frontend/app/(planner)/events/columns.tsx b/frontend/app/(planner)/events/columns.tsx new file mode 100644 index 0000000..ee4cb86 --- /dev/null +++ b/frontend/app/(planner)/events/columns.tsx @@ -0,0 +1,112 @@ +import { DataTableColumnHeader } from "@/components/data-table-column-header"; +import { Badge } from "@/components/ui/badge"; +import { Button } from "@/components/ui/button"; +import { Checkbox } from "@/components/ui/checkbox"; +import { + DropdownMenu, + DropdownMenuContent, + DropdownMenuItem, + DropdownMenuLabel, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import { Event } from "@/lib/gql/generated/graphql"; +import { formatDateToDDMM, formatDateToHHMM } from "@/lib/utils"; +import { ColumnDef } from "@tanstack/react-table"; +import { MoreHorizontal } from "lucide-react"; + +export const columns: ColumnDef[] = [ + { + id: "select", + header: ({ table }) => ( + table.toggleAllPageRowsSelected(!!value)} + aria-label="Alle auswählen" + /> + ), + cell: ({ row }) => ( + row.toggleSelected(!!value)} + aria-label="Reihe auswählen" + /> + ), + }, + { + accessorKey: "title", + header: ({ column }) => ( + + ), + }, + { + accessorKey: "date", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + return formatDateToDDMM(new Date(row.original.from)); + }, + }, + { + accessorKey: "from", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + return formatDateToHHMM(new Date(row.original.from)); + }, + }, + { + accessorKey: "to", + header: ({ column }) => ( + + ), + cell: ({ row }) => { + return formatDateToHHMM(new Date(row.original.to)); + }, + }, + { + accessorKey: "type", + header: "Art", + cell: ({ row }) => ( + + {row.original.type.name} + + ), + }, + { + accessorKey: "topic", + header: "Thema", + cell: ({ row }) => ( + + {row.original.topic.name} + + ), + }, + { + id: "actions", + enableHiding: false, + cell: ({ row }) => { + const event = row.original; + + return ( + + + + + + Optionen + Bearbeiten + Löschen + + + ); + }, + }, +]; diff --git a/frontend/app/(planner)/events/data-table.tsx b/frontend/app/(planner)/events/data-table.tsx new file mode 100644 index 0000000..18f732f --- /dev/null +++ b/frontend/app/(planner)/events/data-table.tsx @@ -0,0 +1,139 @@ +"use client"; + +import { + ColumnDef, + ColumnFiltersState, + SortingState, + VisibilityState, + flexRender, + getCoreRowModel, + getFilteredRowModel, + getSortedRowModel, + useReactTable, +} from "@tanstack/react-table"; + +import { + Table, + TableBody, + TableCell, + TableHead, + TableHeader, + TableRow, +} from "@/components/ui/table"; +import React from "react"; +import { Button } from "@/components/ui/button"; +import { Input } from "@/components/ui/input"; +import { + DropdownMenu, + DropdownMenuCheckboxItem, + DropdownMenuContent, + DropdownMenuTrigger, +} from "@/components/ui/dropdown-menu"; +import {DataTablePagination} from "@/components/data-table-pagination"; +import {DataTableViewOptions} from "@/components/data-table-view-options"; +import {useUmbrella} from "@/components/providers"; + +interface DataTableProps { + columns: ColumnDef[]; + data: TData[]; +} + +export function DataTable({ + columns, + data, +}: DataTableProps) { + const {setCloseupID} = useUmbrella() + + const [sorting, setSorting] = React.useState([]); + const [columnFilters, setColumnFilters] = React.useState( + [] + ); + const [columnVisibility, setColumnVisibility] = + React.useState({}); + const [rowSelection, setRowSelection] = React.useState({}); + + const table = useReactTable({ + data, + columns, + getCoreRowModel: getCoreRowModel(), + onSortingChange: setSorting, + getSortedRowModel: getSortedRowModel(), + onColumnFiltersChange: setColumnFilters, + getFilteredRowModel: getFilteredRowModel(), + onColumnVisibilityChange: setColumnVisibility, + onRowSelectionChange: setRowSelection, + state: { + sorting, + columnFilters, + columnVisibility, + rowSelection, + }, + }); + + return ( +
+
+ + table.getColumn("title")?.setFilterValue(event.target.value) + } + className="max-w-sm" + /> + +
+
+ + + {table.getHeaderGroups().map((headerGroup) => ( + + {headerGroup.headers.map((header) => { + return ( + + {header.isPlaceholder + ? null + : flexRender( + header.column.columnDef.header, + header.getContext() + )} + + ); + })} + + ))} + + + {table.getRowModel().rows?.length ? ( + table.getRowModel().rows.map((row) => ( + + {row.getVisibleCells().map((cell) => ( + + {flexRender( + cell.column.columnDef.cell, + cell.getContext() + )} + + ))} + + )) + ) : ( + + + Keine Ergebnisse. + + + )} + +
+
+ +
+ ); +} diff --git a/frontend/app/(planner)/events/page.tsx b/frontend/app/(planner)/events/page.tsx new file mode 100644 index 0000000..f8d8404 --- /dev/null +++ b/frontend/app/(planner)/events/page.tsx @@ -0,0 +1,66 @@ +"use client"; + +import { useUmbrella } from "@/components/providers"; +import { DataTable } from "./data-table"; +import { + Event, + TableEventsDocument, + TableEventsQuery, + TableEventsQueryVariables, +} from "@/lib/gql/generated/graphql"; +import { client } from "@/lib/graphql"; +import { useEffect, useState } from "react"; +import { columns } from "./columns"; +import { TableSkeleton } from "@/components/table-skeleton"; +import { Button } from "@/components/ui/button"; +import { Plus } from "lucide-react"; +import { ModifyEventDialog } from "@/components/modify-event-dialog"; + +export default function EventsPage() { + const [loading, setLoading] = useState(true); + const [events, setEvents] = useState([]); + + const { umbrellaID } = useUmbrella(); + + useEffect(() => { + const fetchData = async () => { + setLoading(true); + + const vars: TableEventsQueryVariables = { + umbrellaID: [umbrellaID ?? 0], + }; + + const eventData = await client.request( + TableEventsDocument, + vars + ); + + if (eventData.events.length) { + setEvents(eventData.events); + } + + setLoading(false); + }; + + fetchData(); + }, [umbrellaID]); + + return ( + <> +
+ {loading ? ( + + ) : ( + <> + + + + + + )} +
+ + ); +} diff --git a/frontend/app/(planner)/layout.tsx b/frontend/app/(planner)/layout.tsx new file mode 100644 index 0000000..4192e1a --- /dev/null +++ b/frontend/app/(planner)/layout.tsx @@ -0,0 +1,20 @@ +import EventDialog from "@/components/event-dialog"; +import { Footer } from "@/components/footer"; +import { PlannerHeader } from "@/components/planner-selection"; + +interface PlannerLayoutProps { + children: React.ReactNode; +} + +export default function PlannerLayout({ children }: PlannerLayoutProps) { + return ( +
+
+ + + {children} +
+
+
+ ); +} diff --git a/frontend/app/(planner)/overview/page.tsx b/frontend/app/(planner)/overview/page.tsx new file mode 100644 index 0000000..f8f6782 --- /dev/null +++ b/frontend/app/(planner)/overview/page.tsx @@ -0,0 +1,10 @@ + +export const metadata = { + title: "Überblick", +} + +export default function OverviewPage() { + return ( +

Overview

+ ) +} diff --git a/frontend/app/(planner)/page.tsx b/frontend/app/(planner)/page.tsx new file mode 100644 index 0000000..a03d2d1 --- /dev/null +++ b/frontend/app/(planner)/page.tsx @@ -0,0 +1,121 @@ +"use client"; + +import { + Event, + Label, + PlannerEventsDocument, + PlannerEventsQuery, + PlannerEventsQueryVariables, +} from "@/lib/gql/generated/graphql"; +import React, { useCallback, useEffect, useState } from "react"; + +import Filter from "@/components/filter"; +import { usePathname, useRouter, useSearchParams } from "next/navigation"; +import { client } from "@/lib/graphql"; +import { CopyTextArea } from "@/components/copy-text-area"; +import { CardSkeleton } from "@/components/card-skeleton"; +import { Planner } from "@/components/planner"; +import { useUmbrella } from "@/components/providers"; + +export default function IndexPage() { + const router = useRouter(); + const searchParams = useSearchParams(); + const pathname = usePathname(); + + const { umbrellaID } = useUmbrella(); + + const [events, setEvents] = useState([]); + const [types, setTypes] = useState([]); + const [topics, setTopics] = useState([]); + const [toFilter, setToFilter] = useState(searchParams.getAll("to")); + const [tyFilter, setTyFilter] = useState(searchParams.getAll("ty")); + const [icalPath, setIcalPath] = useState(""); + const [loading, setLoading] = useState(true); + + const createQueryString = useCallback((name: string, values: string[]) => { + const params = new URLSearchParams(values.map((v) => [name, v])); + return params.toString(); + }, []); + + useEffect(() => { + const fetchData = async () => { + setLoading(true); + + const vars: PlannerEventsQueryVariables = { + umbrellaID: umbrellaID ?? 0, + topic: toFilter.length == 0 ? undefined : toFilter, + type: tyFilter.length == 0 ? undefined : tyFilter, + }; + + const eventData = await client.request( + PlannerEventsDocument, + vars + ); + + if (eventData.events.length) { + if (eventData.typeLabels.length >= 2) { + setTypes(eventData.typeLabels); + } + if (eventData.topicLabels.length >= 2) { + setTopics(eventData.topicLabels); + } + setEvents( + eventData.events.map((e) => ({ + type: { name: "" }, + needsTutors: true, + ID: e.ID, + title: e.title, + from: e.from, + to: e.to, + topic: { name: "", color: e.topic.color }, + })) + ); + setLoading(false); + } + }; + + router.push( + pathname + + "?" + + createQueryString("e", [umbrellaID?.toString() ?? "0"]) + + (toFilter.length ? "&" : "") + + createQueryString("to", toFilter) + + (tyFilter.length ? "&" : "") + + createQueryString("ty", tyFilter) + ); + fetchData(); + }, [toFilter, tyFilter, umbrellaID]); + + useEffect(() => { + setIcalPath(window.location.origin + "/ical/?" + searchParams); + }, [searchParams]); + + return ( + <> + {events.length > 0 && ( +
+
+ t.name)} + filter={toFilter} + setFilter={setToFilter} + /> + t.name)} + filter={tyFilter} + setFilter={setTyFilter} + /> +
+
+ +
+ )} + +
+ {loading ? : } +
+ + ); +} diff --git a/frontend/app/(planner)/settings/page.tsx b/frontend/app/(planner)/settings/page.tsx new file mode 100644 index 0000000..b08ac83 --- /dev/null +++ b/frontend/app/(planner)/settings/page.tsx @@ -0,0 +1,10 @@ + +export const metadata = { + title: "Einstellungen", +} + +export default function OverviewPage() { + return ( +

Einstellungen

+ ) +} diff --git a/frontend/app/(planner)/tutorials/page.tsx b/frontend/app/(planner)/tutorials/page.tsx new file mode 100644 index 0000000..7cb3fd4 --- /dev/null +++ b/frontend/app/(planner)/tutorials/page.tsx @@ -0,0 +1,10 @@ + +export const metadata = { + title: "Tutorien", +} + +export default function OverviewPage() { + return ( +

Tutorien

+ ) +} diff --git a/frontend/app/(registration)/register/layout.tsx b/frontend/app/(registration)/register/layout.tsx new file mode 100644 index 0000000..fc5e979 --- /dev/null +++ b/frontend/app/(registration)/register/layout.tsx @@ -0,0 +1,16 @@ +import { Footer } from "@/components/footer"; +import Header from "@/components/header"; + +interface RegistrationLayoutProps { + children: React.ReactNode; +} + +export default function RegistrationLayout({ children }: RegistrationLayoutProps) { + return ( +
+
+
{children}
+
+
+ ); +} diff --git a/frontend/app/register/page.tsx b/frontend/app/(registration)/register/page.tsx similarity index 98% rename from frontend/app/register/page.tsx rename to frontend/app/(registration)/register/page.tsx index c8fedd2..8e68460 100644 --- a/frontend/app/register/page.tsx +++ b/frontend/app/(registration)/register/page.tsx @@ -12,7 +12,7 @@ import { RegistrationFormQuery, RegistrationFormQueryVariables, } from "@/lib/gql/generated/graphql"; -import { client } from "@/lib/graphClient"; +import { client } from "@/lib/graphql"; import { Button } from "@/components/ui/button"; import { Card, @@ -37,7 +37,7 @@ import { FormItem, FormMessage, } from "@/components/ui/form"; -import { Toaster, toast } from "sonner"; +import { toast } from "sonner"; type Props = { searchParams: { @@ -87,8 +87,6 @@ const Home = ({ searchParams }: Props) => { eventID: parseInt(eventID), }; - await new Promise((resolve) => setTimeout(resolve, 250)); - const data = await client.request( RegistrationFormDocument, vars @@ -140,8 +138,6 @@ const Home = ({ searchParams }: Props) => { return; } - await new Promise((resolve) => setTimeout(resolve, 250)); - const application: NewUserToEventApplication = { // TODO userMail: "tutor1@example.de", diff --git a/frontend/app/actions.ts b/frontend/app/actions.ts deleted file mode 100644 index d78f225..0000000 --- a/frontend/app/actions.ts +++ /dev/null @@ -1,48 +0,0 @@ -import { - FormState, - fromErrorToFormState, - toFormState, -} from "@/lib/to-form-state"; -import { client } from "@/lib/graphClient"; -import { - AddTutorDocument, - AddTutorMutation, - AddTutorMutationVariables, - TutorFormEventsQuery, - TutorFormEventsDocument, -} from "@/lib/gql/generated/graphql"; -import {eventBroker} from "@/lib/eventBroker"; - -export const getEvents = async (): Promise => { - await new Promise((resolve) => setTimeout(resolve, 250)); - - const data = await client.request( - TutorFormEventsDocument - ); - - return Promise.resolve(data); -}; - -export const addTutor = async (formState: FormState, formData: FormData) => { - await new Promise((resolve) => setTimeout(resolve, 250)); - - const firstName = formData.get("fn")?.toString() || ""; - const lastName = formData.get("sn")?.toString() || ""; - const email = formData.get("email")?.toString() || ""; - const eventsAvailable = eventBroker.getEvents(); - - const vars: AddTutorMutationVariables = { - firstName, - lastName, - email, - eventsAvailable, - }; - - try { - await client.request(AddTutorDocument, vars); - } catch (err) { - return fromErrorToFormState(err); - } - - return toFormState("SUCCESS", "Anmeldung erfolgreich"); -}; diff --git a/frontend/app/error.tsx b/frontend/app/error.tsx deleted file mode 100644 index bab8132..0000000 --- a/frontend/app/error.tsx +++ /dev/null @@ -1,17 +0,0 @@ -"use client"; - -import { buttonVariants } from "@/components/ui/button"; -import Image from "next/image"; -import Link from "next/link"; - -export default function Error() { - return ( -
- Error -

Something went wrong!

- - Go Back - -
- ); -} diff --git a/frontend/app/favicon.ico b/frontend/app/favicon.ico deleted file mode 100644 index 718d6fe..0000000 Binary files a/frontend/app/favicon.ico and /dev/null differ diff --git a/frontend/app/form-tutor/layout.tsx b/frontend/app/form-tutor/layout.tsx deleted file mode 100644 index 8eedc77..0000000 --- a/frontend/app/form-tutor/layout.tsx +++ /dev/null @@ -1,22 +0,0 @@ -import type { Metadata } from "next"; -import { Inter } from "next/font/google"; -import "../globals.css"; - -const inter = Inter({ subsets: ["latin"] }); - -export const metadata: Metadata = { - title: "Tutor:innen Anmeldung", - description: "Anmeldung für Tutor:innen des Vorkurs", -}; - -export default function RootLayout({ - children, - }: Readonly<{ - children: React.ReactNode; -}>) { - return ( - - {children} - - ); -} diff --git a/frontend/app/form-tutor/page.tsx b/frontend/app/form-tutor/page.tsx deleted file mode 100644 index fd3ff86..0000000 --- a/frontend/app/form-tutor/page.tsx +++ /dev/null @@ -1,12 +0,0 @@ -import {TutorRegistrationForm} from "@/components/tutor-registration-form"; - -export default function Page() { - return ( -
-
-

Anmeldung Vorkurstutor:in

- -
-
- ); -} diff --git a/frontend/app/globals.css b/frontend/app/globals.css deleted file mode 100644 index 7239ec5..0000000 --- a/frontend/app/globals.css +++ /dev/null @@ -1,91 +0,0 @@ -@tailwind base; - @tailwind components; - @tailwind utilities; - - @layer base { - :root { - --background: 0 0% 100%; - --foreground: 0 0% 3.9%; - - --card: 0 0% 100%; - --card-foreground: 0 0% 3.9%; - - --popover: 0 0% 100%; - --popover-foreground: 0 0% 3.9%; - - --primary: 0 0% 9%; - --primary-foreground: 0 0% 98%; - - --secondary: 0 0% 96.1%; - --secondary-foreground: 0 0% 9%; - - --muted: 0 0% 96.1%; - --muted-foreground: 0 0% 45.1%; - - --accent: 0 0% 96.1%; - --accent-foreground: 0 0% 9%; - - --destructive: 0 84.2% 60.2%; - --destructive-foreground: 0 0% 98%; - - --border: 0 0% 89.8%; - --input: 0 0% 89.8%; - --ring: 0 0% 3.9%; - - --radius: 0.5rem; - - --chart-1: 12 76% 61%; - - --chart-2: 173 58% 39%; - - --chart-3: 197 37% 24%; - - --chart-4: 43 74% 66%; - - --chart-5: 27 87% 67%; - } - - .dark { - --background: 0 0% 3.9%; - --foreground: 0 0% 98%; - - --card: 0 0% 3.9%; - --card-foreground: 0 0% 98%; - - --popover: 0 0% 3.9%; - --popover-foreground: 0 0% 98%; - - --primary: 0 0% 98%; - --primary-foreground: 0 0% 9%; - - --secondary: 0 0% 14.9%; - --secondary-foreground: 0 0% 98%; - - --muted: 0 0% 14.9%; - --muted-foreground: 0 0% 63.9%; - - --accent: 0 0% 14.9%; - --accent-foreground: 0 0% 98%; - - --destructive: 0 62.8% 30.6%; - --destructive-foreground: 0 0% 98%; - - --border: 0 0% 14.9%; - --input: 0 0% 14.9%; - --ring: 0 0% 83.1%; - --chart-1: 220 70% 50%; - --chart-2: 160 60% 45%; - --chart-3: 30 80% 55%; - --chart-4: 280 65% 60%; - --chart-5: 340 75% 55%; - } - } - - @layer base { - * { - @apply border-border; - } - body { - @apply bg-background text-foreground; - } - } \ No newline at end of file diff --git a/frontend/app/layout.tsx b/frontend/app/layout.tsx index 54ec0f6..4413971 100644 --- a/frontend/app/layout.tsx +++ b/frontend/app/layout.tsx @@ -1,31 +1,74 @@ -import type { Metadata } from "next"; -import { Inter } from "next/font/google"; -import "./globals.css"; +import { Inter as FontSans } from "next/font/google"; +import localFont from "next/font/local"; + +import "@/styles/globals.css"; import { Toaster } from "@/components/ui/sonner"; +import { cn } from "@/lib/utils"; +import { siteConfig } from "@/config/site"; +import { TailwindIndicator } from "@/components/tailwind-indicator"; +import { + UserProvider, + ThemeProvider, + UmbrellaProvider, +} from "@/components/providers"; import Header from "@/components/header"; -import { Providers } from "./providers"; -import {Footer} from "@/components/footer"; +import { Suspense } from "react"; + +const fontSans = FontSans({ + subsets: ["latin"], + variable: "--font-sans", +}); -const inter = Inter({ subsets: ["latin"] }); +const fontHeading = localFont({ + src: "../assets/fonts/CalSans-SemiBold.woff2", + variable: "--font-heading", +}); -export const metadata: Metadata = { - title: "Vorkurs", - description: "Dashboard und Infoseite des Vorkurses", +export const metadata = { + title: { + default: siteConfig.name, + }, + description: siteConfig.description, + keywords: ["pepp"], + creator: "Fachschaft MathPhysInfo", + openGraph: { + type: "website", + locale: "en_US", + url: siteConfig.url, + title: siteConfig.name, + description: siteConfig.description, + siteName: siteConfig.name, + }, }; -export default function RootLayout({ - children, -}: Readonly<{ +interface RootLayoutProps { children: React.ReactNode; -}>) { +} + +export default function RootLayout({ children }: RootLayoutProps) { return ( - - -
-
{children}
- - -