2026-04-28 16:37:12 +02:00

223 lines
7.2 KiB
TypeScript

import logo from "../assets/Logo.png";
import { useState } from "react";
import { Button, useMediaQuery } from "@relume_io/relume-ui";
import type { ButtonProps } from "@relume_io/relume-ui";
import { AnimatePresence, motion } from "framer-motion";
import { RxChevronDown } from "react-icons/rx";
import { Link } from "react-router-dom";
type NavButtonProps = ButtonProps & { url?: string };
type ImageProps = {
url?: string;
src: string;
alt?: string;
};
type NavLink = {
url?: string;
title: string;
subMenuLinks?: NavLink[];
};
type Props = {
logo: ImageProps;
navLinks: NavLink[];
buttons: NavButtonProps[];
};
export type Navbar3Props = React.ComponentPropsWithoutRef<"section"> & Partial<Props>;
export const Navbar3 = (props: Navbar3Props) => {
const { logo, navLinks, buttons } = {
...Navbar3Defaults,
...props,
};
const [isMobileMenuOpen, setIsMobileMenuOpen] = useState(false);
const isMobile = useMediaQuery("(max-width: 991px)");
return (
<section
id="relume"
className="z-[999] grid w-full grid-cols-[1fr_max-content_1fr] items-center justify-between border-b border-neutral-dark bg-tech-navy px-[5%] md:min-h-18 font-barlow"
>
<button
className="flex size-12 flex-col justify-center lg:hidden"
onClick={() => setIsMobileMenuOpen((prev) => !prev)}
>
{Array(3)
.fill(null)
.map((_, index) => (
<span key={index} className="my-[3px] h-0.5 w-6 bg-cloud-white lg:hidden" />
))}
</button>
<AnimatePresence>
<motion.div
initial="closed"
animate={isMobileMenuOpen ? "open" : "closed"}
exit="closed"
variants={{
closed: {
x: "-100%",
opacity: 1,
transition: { type: "spring", duration: 0.6, bounce: 0 },
transitionEnd: {
opacity: "var(--opacity-closed, 0%)",
x: "var(--x-closed, -100%)",
},
},
open: {
x: 0,
opacity: 1,
transition: { type: "spring", duration: 0.4, bounce: 0 },
},
}}
className="absolute left-0 top-0 z-50 flex h-dvh w-[90%] flex-col border-r border-border-primary bg-tech-navy px-[5%] pb-4 md:w-[80%] lg:visible lg:static lg:-ml-4 lg:flex lg:h-auto lg:w-auto lg:flex-row lg:border-none lg:px-0 lg:pb-0 lg:[--opacity-closed:100%] lg:[--x-closed:0%]"
>
<a href={logo.url} className="mb-8 mt-10 flex flex-shrink-0 lg:hidden">
<img src={logo.src} alt={logo.alt} />
</a>
{navLinks.map((navLink, index) => (
<div key={index} className="w-full lg:w-auto">
{navLink.subMenuLinks && navLink.subMenuLinks.length > 0 ? (
<SubMenu navLink={navLink} isMobile={isMobile} />
) : (
<a
href={navLink.url}
className="relative block py-3 text-md text-cloud-white lg:px-4 lg:py-2 lg:text-base"
>
{navLink.title}
</a>
)}
</div>
))}
<div className="mt-6 lg:hidden">
{buttons.map(({ url, ...button }, index) =>
url ? (
<Link key={index} to={url}>
<Button {...button} className="w-full">{button.title}</Button>
</Link>
) : (
<Button key={index} {...button} className="w-full">{button.title}</Button>
)
)}
</div>
</motion.div>
{isMobileMenuOpen && (
<motion.div
initial={{ opacity: 0 }}
exit={{ opacity: 0 }}
animate={{ opacity: 0.5 }}
transition={{ duration: 0.2 }}
className="fixed inset-0 z-40 bg-black lg:hidden"
onClick={() => setIsMobileMenuOpen(false)}
/>
)}
</AnimatePresence>
<a href={logo.url} className="flex min-h-16 flex-shrink-0 items-center">
<img src={logo.src} alt={logo.alt} />
</a>
<div className="flex min-h-16 items-center justify-end gap-x-4">
<div>
{buttons.map(({ url, ...button }, index) =>
url ? (
<Link key={index} to={url}>
<Button {...button} className="px-5 py-2 bg-electric-violet text-cloud-white rounded-2xl border-electric-violet font-barlow font-semibold">
{button.title}
</Button>
</Link>
) : (
<Button key={index} {...button} className="px-5 py-2 bg-electric-violet text-cloud-white rounded-2xl border-electric-violet font-barlow font-semibold">
{button.title}
</Button>
)
)}
</div>
</div>
</section>
);
};
const SubMenu = ({ navLink, isMobile }: { navLink: NavLink; isMobile: boolean }) => {
const [isDropdownOpen, setIsDropdownOpen] = useState(false);
return (
<div
onMouseEnter={() => !isMobile && setIsDropdownOpen(true)}
onMouseLeave={() => !isMobile && setIsDropdownOpen(false)}
>
<button
className="flex w-full items-center justify-between gap-2 py-3 text-md lg:flex-none lg:justify-start lg:px-4 lg:py-2 lg:text-base"
onClick={() => setIsDropdownOpen((prev) => !prev)}
>
<span>{navLink.title}</span>
<AnimatePresence>
<motion.span
animate={isDropdownOpen ? "rotated" : "initial"}
variants={{
rotated: { rotate: 180 },
initial: { rotate: 0 },
}}
transition={{ duration: 0.3 }}
>
<RxChevronDown />
</motion.span>
</AnimatePresence>
</button>
{isDropdownOpen && (
<AnimatePresence>
<motion.nav
variants={{
open: {
visibility: "visible",
opacity: "var(--opacity-open, 100%)",
y: 0,
},
close: {
visibility: "hidden",
opacity: "var(--opacity-close, 0)",
y: "var(--y-close, 0%)",
},
}}
initial="close"
exit="close"
animate={isDropdownOpen ? "open" : "close"}
className="bg-background-primary lg:absolute lg:z-50 lg:border lg:border-border-primary lg:p-2 lg:[--y-close:25%]"
>
{navLink.subMenuLinks?.map((subMenuLink, index) => (
<a
key={index}
href={subMenuLink.url}
className="block py-3 pl-[5%] text-md lg:py-2 lg:pl-4 lg:pr-8 lg:text-left lg:text-base"
>
{subMenuLink.title}
</a>
))}
</motion.nav>
</AnimatePresence>
)}
</div>
);
};
export const Navbar3Defaults: Props = {
logo: {
url: "/",
src: logo,
alt: "UXplore Logo",
},
navLinks: [
{ title: "Programm", url: "/programm" },
{ title: "Speaker", url: "/speaker" },
{ title: "Location", url: "#" },
],
buttons: [
{
title: "Tickets",
size: "sm",
variant: "primary",
url: "/tickets",
},
],
};