1
0
mirror of https://github.com/tabler/tabler.git synced 2025-12-22 01:44:25 +04:00
Files
tabler/site/components/layout/Header.tsx
2023-12-07 02:07:01 +01:00

410 lines
11 KiB
TypeScript

'use client';
import {
Fragment,
useEffect,
useState,
} from 'react';
import { Dialog, Popover } from '@headlessui/react';
import clsx from 'clsx';
import { banner,
blogEnabled,
iconsCountRounded,
sponsorsUrl,
uiGithubUrl,
} from '@/config/site';
import Icon from '@/components/Icon';
import GoToTop from '@/components/layout/GoToTop';
import Link from '@/components/Link';
import NavLink from '@/components/NavLink';
import Shape from '@/components/Shape';
import { usePathname } from 'next/navigation';
import { signOut, useSession } from 'next-auth/react';
import { useRouter } from 'next/navigation';
const NavDropdown = ({ title, children, active, footer = false }) => {
return (
<Popover className="navbar-dropdown">
{({ open }) => (
<>
<Popover.Button className={clsx('navbar-link', active && 'active')}>{title}</Popover.Button>
<Popover.Panel className="navbar-dropdown-menu">
<div className="navbar-dropdown-menu-content">{children}</div>
{footer && <div className="navbar-dropdown-menu-footer">{footer}</div>}
</Popover.Panel>
</>
)}
</Popover>
);
};
const menuLinks = [
{
title: 'UI Kit',
menu: 'ui',
children: [
{
icon: 'home',
href: '/',
title: 'About',
description: 'Develop beautiful web apps with Tabler',
},
{
icon: 'layout-dashboard',
href: '/preview',
title: 'Preview template',
description: 'See what Tabler looks like and offers',
},
{
icon: 'script',
href: '/docs',
title: 'Documentation',
description: 'Read how to develop apps with Tabler',
},
{
icon: 'lego',
href: '/features',
title: 'Features',
description: 'See what kind of features you can find here',
},
{
icon: 'lifebuoy',
href: '/support',
title: 'Support',
description: 'Write to us if you need anything!',
},
{
icon: 'brand-github',
href: uiGithubUrl,
title: 'Source code',
description: 'View Tabler\'s source code ',
props: {
target: '_blank',
rel: 'nofollow',
},
},
],
},
{
href: '/emails',
menu: 'emails',
title: 'Email templates',
},
{
href: '/icons',
menu: 'icons',
title: (
<>
<span className="d-none lg:d-inline">Over {iconsCountRounded} </span>
Icons
</>
),
},
...(blogEnabled ? [{
href: '/blog',
menu: 'blog',
title: <>Blog</>,
}] : []),
{
href: '/docs',
menu: 'docs',
title: 'Documentation',
},
// {
// href: '/guides',
// menu: 'guides',
// title: 'Guides',
// },
{
menu: 'sponsors',
href: sponsorsUrl,
type: 'button',
title: (
<span>
Sponsor<span className="d-none lg:d-inline"> project</span>
</span>
),
icon: <Icon name="heart" filled color="red" />,
},
];
const NavbarLink = (link, menu) => {
// const router = useRouter()
if (link.type === 'button') {
return (
<div className="navbar-item">
<a href={link.href} className="btn" target="_blank" rel="noopener noreferrer">
{link.icon}
{link.title}
</a>
</div>
);
} else if (link.children) {
return (
<NavDropdown title={link.title} active={menu === link.menu}>
{link.children.map((link) => (
<Popover.Button as={Link} href={link.href || ''} className="navbar-dropdown-menu-link" key={link.title} onClick={() => true} {...link.props}>
<div className="row g-3">
<div className="col-auto">
<Shape icon={link.icon} />
</div>
<div className="col">
<h5 className="mb-1">{link.title}</h5>
<p className="font-h6 m-0 text-muted">{link.description}</p>
</div>
</div>
</Popover.Button>
))}
</NavDropdown>
);
}
return (
// router.pathname.replace(/^\//, '').startsWith(link.menu)
<NavLink href={link.href} className="navbar-link">
{link.title}
</NavLink>
);
};
const SidebarLink = (link, menu, onClick) => {
if (link.type === 'button') {
return (
<div className="aside-menu-item mt-4">
<a href={link.href} className="btn btn-block" target="_blank" rel="noopener noreferrer" onClick={onClick}>
{link.icon}
{link.title}
</a>
</div>
);
} else if (link.children) {
return (
<div className="aside-menu-item">
<div className={clsx('aside-menu-title', { active: menu === link.menu })}>{link.title}</div>
<div className="aside-menu-children">
{link.children.map((link) => (
<Link href={link.href || ''} key={link.title} className="aside-menu-link" onClick={onClick} {...link.props}>
{link.title}
</Link>
))}
</div>
</div>
);
}
return (
<Link href={link.href} className={clsx('aside-menu-link', { active: menu === link.menu })} onClick={onClick}>
{link.title}
</Link>
);
};
const NavigationAuth = () => {
const { data: session, status } = useSession();
const router = useRouter();
const image = session?.user?.image;
const name = session?.user?.name;
const email = session?.user?.email;
const signIn = () => {
if (status === 'loading') return;
router.push('/api/auth/signin');
};
return <div className="navbar-item d-flex items-center">
{
!session &&
<a onClick={() => signIn()} className={clsx('btn', { disabled: status === 'loading'})}>
Log in
</a>
}
{
session &&
<Popover className="navbar-dropdown">
{({ open }) => (
<>
<Popover.Button className={clsx('navbar-link d-flex items-center lh-1 text-reset p-0')}>
{
image
? <span
className="avatar avatar"
style={{
backgroundImage: `url(${image})`,
}}
/>
: <span className="avatar avatar text-center">
{name ? name.toUpperCase().substring(0, 1) : 'T'}
</span>
}
<div className="pl-2">
<small className="d-block">{name}</small>
{
email &&
<small className="mt-1 small text-muted">{session.user?.email}</small>
}
</div>
</Popover.Button>
<Popover.Panel className="navbar-dropdown-menu">
<div className="navbar-dropdown-menu-content">
<div onClick={() => router.push('billing')} className="navbar-dropdown-menu-link">
<div className="row items-center g-3">
<div className="col-auto">
<Shape icon='rocket'/>
</div>
<div className="col">
<h5>Billing</h5>
</div>
</div>
</div>
<div onClick={() => signOut()} className="navbar-dropdown-menu-link">
<div className="row items-center g-3">
<div className="col-auto">
<Shape icon='logout'/>
</div>
<div className="col">
<h5>Log out</h5>
</div>
</div>
</div>
</div>
</Popover.Panel>
</>
)}
</Popover>
}
</div>;
};
const Navbar = ({
menu,
opened,
onClick,
...props
}: {
menu?: string;
opened?: boolean;
onClick?: (event: React.MouseEvent) => void;
className?: string
}) => {
return <div className={clsx('navbar', opened && 'opened', props.className)}>
{menuLinks.map((link) => (<Fragment key={link.menu}>{NavbarLink(link, menu)}</Fragment>))}
<NavigationAuth/>
</div>;
};
const Banner = () => {
const [showBanner, setShowBanner] = useState(false);
useEffect(() => {
if (window.localStorage.getItem(`banner-${banner.id}`) !== '1') {
setShowBanner(true);
}
}, []);
function closeBanner() {
localStorage.setItem(`banner-${banner.id}`, '1');
setShowBanner(false);
}
return (
banner.show &&
showBanner && (
<div className="banner">
<div className="container">
<div className="text-truncate">{banner.text}</div>
<a href={banner.link.href} className="ml-5 banner-link" target="_blank">
{banner.link.text}
</a>
</div>
<a onClick={closeBanner} className="banner-close">
<Icon name="x" />
</a>
</div>
)
);
};
export default function Header({ headerStatic, className, pageProps, ...props }: { headerStatic?: boolean; className?: string; pageProps?: any }) {
const [sticky, setSticky] = useState(false);
const [isOpen, setIsOpen] = useState(false);
const pathname = usePathname();
function closeModal() {
setIsOpen(false);
}
function toggleModal() {
setIsOpen(!isOpen);
}
useEffect(() => {
const handleScroll = () => {
setSticky(window.pageYOffset > 0);
};
window.addEventListener('scroll', handleScroll);
handleScroll();
return () => {
window.removeEventListener('scroll', handleScroll);
};
}, []);
return (
<>
<Banner />
<header
className={clsx(
'header',
sticky && 'header-sticky',
pathname.startsWith('/docs') && 'header-docs',
className,
)}
>
<div className="container">
<nav className="row items-center">
<div className="col-auto">
<Link href="/" className={clsx('logo' /*, pageProps.brand ? `logo-${pageProps.brand}` : ''*/)} aria-label="Tabler" />
</div>
<div className="col-auto ml-auto">
<div className="d-none md:d-block">
{/* <Navbar menu={pageProps.menu} /> */}
<Navbar />
</div>
<div className="md:d-none">
<button
className={clsx('navbar-toggle', {
active: isOpen,
})}
onClick={toggleModal}
>
<span />
<span />
<span />
<span />
</button>
</div>
</div>
</nav>
</div>
</header>
<Dialog open={isOpen} onClose={closeModal} className="modal-backdrop">
<Dialog.Panel className="modal modal-side">
<div className={clsx('aside-menu mt-4')}>
{/* {menuLinks.map((link) => (
// <Fragment key={link.menu}>{SidebarLink(link, pageProps.menu, closeModal)}</Fragment>
))} */}
</div>
</Dialog.Panel>
</Dialog>
<GoToTop />
</>
);
}