161 lines
		
	
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
			
		
		
	
	
			161 lines
		
	
	
	
		
			6.7 KiB
		
	
	
	
		
			TypeScript
		
	
	
	
	
	
| import Link from 'next/link';
 | |
| import { useRouter } from 'next/router';
 | |
| import { Fragment, useState } from 'react';
 | |
| import { Dialog, Transition } from '@headlessui/react';
 | |
| import {
 | |
|   Bars3Icon,
 | |
|   HomeIcon,
 | |
|   XMarkIcon,
 | |
|   AcademicCapIcon,
 | |
|   ArrowRightOnRectangleIcon,
 | |
| } from '@heroicons/react/24/outline';
 | |
| 
 | |
| const navigation = [
 | |
|   { name: 'Dashboard', href: '/', icon: HomeIcon },
 | |
|   { name: 'Trainings', href: '/trainings', icon: AcademicCapIcon },
 | |
| ]
 | |
| 
 | |
| function classNames(...classes: any) {
 | |
|   return classes.filter(Boolean).join(' ')
 | |
| }
 | |
| 
 | |
| export function Sidebar() {
 | |
|   const [sidebarOpen, setSidebarOpen] = useState(false)
 | |
|   const router = useRouter();
 | |
| 
 | |
|   return (
 | |
|     <>
 | |
|       <Transition.Root show={sidebarOpen} as={Fragment}>
 | |
|         <Dialog as="div" className="relative z-50 lg:hidden" onClose={setSidebarOpen}>
 | |
|           <Transition.Child
 | |
|             as={Fragment}
 | |
|             enter="transition-opacity ease-linear duration-300"
 | |
|             enterFrom="opacity-0"
 | |
|             enterTo="opacity-100"
 | |
|             leave="transition-opacity ease-linear duration-300"
 | |
|             leaveFrom="opacity-100"
 | |
|             leaveTo="opacity-0"
 | |
|           >
 | |
|             <div className="fixed inset-0 bg-black/80" />
 | |
|           </Transition.Child>
 | |
| 
 | |
|           <div className="fixed inset-0 flex">
 | |
|             <Transition.Child
 | |
|               as={Fragment}
 | |
|               enter="transition ease-in-out duration-300 transform"
 | |
|               enterFrom="-translate-x-full"
 | |
|               enterTo="translate-x-0"
 | |
|               leave="transition ease-in-out duration-300 transform"
 | |
|               leaveFrom="translate-x-0"
 | |
|               leaveTo="-translate-x-full"
 | |
|             >
 | |
|               <Dialog.Panel className="relative mr-16 flex w-full max-w-xs flex-1">
 | |
|                 <Transition.Child
 | |
|                   as={Fragment}
 | |
|                   enter="ease-in-out duration-300"
 | |
|                   enterFrom="opacity-0"
 | |
|                   enterTo="opacity-100"
 | |
|                   leave="ease-in-out duration-300"
 | |
|                   leaveFrom="opacity-100"
 | |
|                   leaveTo="opacity-0"
 | |
|                 >
 | |
|                   <div className="absolute left-full top-0 flex w-16 justify-center pt-5">
 | |
|                     <button type="button" className="-m-2.5 p-2.5" onClick={() => setSidebarOpen(false)}>
 | |
|                       <span className="sr-only">Close sidebar</span>
 | |
|                       <XMarkIcon className="h-6 w-6 text-white" aria-hidden="true" />
 | |
|                     </button>
 | |
|                   </div>
 | |
|                 </Transition.Child>
 | |
|                 {/* Sidebar component, swap this element with another sidebar if you like */}
 | |
|                 <div className="flex grow flex-col gap-y-5 overflow-y-auto bg-black px-6 pb-2 ring-1 ring-white/10">
 | |
|                   <div className="flex h-16 shrink-0 items-center">
 | |
|                     <span className="text-amber-600 text-2xl font-bold">MaresHQ <sup>backoffice</sup></span>
 | |
|                   </div>
 | |
|                   <nav className="flex flex-1 flex-col">
 | |
|                     <ul role="list" className="flex flex-1 flex-col gap-y-7">
 | |
|                       <li>
 | |
|                         <ul role="list" className="-mx-2 space-y-1">
 | |
|                           {navigation.map((item) => (
 | |
|                             <li key={item.name}>
 | |
|                               <Link
 | |
|                                 href={item.href}
 | |
|                                 className={classNames(
 | |
|                                   router.pathname === item.href
 | |
|                                     ? 'bg-gray-800 text-white'
 | |
|                                     : 'text-gray-400 hover:text-white hover:bg-gray-800',
 | |
|                                   'group flex gap-x-3 rounded-md p-2 text-sm leading-6 font-semibold'
 | |
|                                 )}
 | |
|                               >
 | |
|                                 <item.icon className="h-6 w-6 shrink-0" aria-hidden="true" />
 | |
|                                 {item.name}
 | |
|                               </Link>
 | |
|                             </li>
 | |
|                           ))}
 | |
|                         </ul>
 | |
|                       </li>
 | |
|                     </ul>
 | |
|                   </nav>
 | |
|                 </div>
 | |
|               </Dialog.Panel>
 | |
|             </Transition.Child>
 | |
|           </div>
 | |
|         </Dialog>
 | |
|       </Transition.Root>
 | |
| 
 | |
|       {/* Static sidebar for desktop */}
 | |
|       <div className="hidden lg:fixed lg:inset-y-0 lg:z-50 lg:flex lg:w-72 lg:flex-col">
 | |
|         {/* Sidebar component, swap this element with another sidebar if you like */}
 | |
|         <div className="flex grow flex-col gap-y-5 overflow-y-auto bg-black px-6">
 | |
|           <div className="flex h-16 shrink-0 items-center">
 | |
|             <span className="text-amber-600 text-2xl font-bold">MaresHQ <sup>backoffice</sup></span>
 | |
|           </div>
 | |
|           <nav className="flex flex-1 flex-col">
 | |
|             <ul role="list" className="flex flex-1 flex-col gap-y-7">
 | |
|               <li>
 | |
|                 <ul role="list" className="-mx-2 space-y-1">
 | |
|                   {navigation.map((item) => (
 | |
|                     <li key={item.name}>
 | |
|                       <Link
 | |
|                         href={item.href}
 | |
|                         className={classNames(
 | |
|                           router.pathname === item.href
 | |
|                             ? 'bg-gray-800 text-white'
 | |
|                             : 'text-gray-400 hover:text-white hover:bg-gray-800',
 | |
|                           'group flex gap-x-3 rounded-md p-2 text-sm leading-6 font-semibold'
 | |
|                         )}
 | |
|                       >
 | |
|                         <item.icon className="h-6 w-6 shrink-0" aria-hidden="true" />
 | |
|                         {item.name}
 | |
|                       </Link>
 | |
|                     </li>
 | |
|                   ))}
 | |
|                 </ul>
 | |
|               </li>
 | |
|               <li className="-mx-6 mt-auto">
 | |
|                 <Link
 | |
|                   href="#"
 | |
|                   className="flex items-center gap-x-4 px-6 py-3 text-sm font-semibold leading-6 text-white hover:bg-gray-800"
 | |
|                 >
 | |
|                   <ArrowRightOnRectangleIcon className="h-6 w-6" />
 | |
|                   <span>Sign out</span>
 | |
|                 </Link>
 | |
|               </li>
 | |
|             </ul>
 | |
|           </nav>
 | |
|         </div>
 | |
|       </div>
 | |
| 
 | |
|       <div className="sticky top-0 z-40 flex items-center gap-x-6 bg-black px-4 py-4 shadow-sm sm:px-6 lg:hidden">
 | |
|         <button type="button" className="-m-2.5 p-2.5 text-gray-400 lg:hidden" onClick={() => setSidebarOpen(true)}>
 | |
|           <span className="sr-only">Open sidebar</span>
 | |
|           <Bars3Icon className="h-6 w-6" aria-hidden="true" />
 | |
|         </button>
 | |
|         <div className="flex-1 text-sm font-semibold leading-6 text-white">Dashboard</div>
 | |
|         <Link href="#">
 | |
|           <span className="sr-only">Sign out</span>
 | |
|           <ArrowRightOnRectangleIcon className="h-6 w-6 text-white" />
 | |
|         </Link>
 | |
|       </div>
 | |
|     </>
 | |
|   )
 | |
| }
 |