223 lines
7.2 KiB
TypeScript
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",
|
|
},
|
|
],
|
|
};
|