feat: add sidebar layout from TailwindUI
This commit is contained in:
		
							parent
							
								
									a20b4128d0
								
							
						
					
					
						commit
						349b73a992
					
				
					 4 changed files with 203 additions and 33 deletions
				
			
		
							
								
								
									
										18
									
								
								src/components/Layout.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										18
									
								
								src/components/Layout.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,18 @@ | |||
| import { Sidebar } from "~/components/Sidebar"; | ||||
| 
 | ||||
| export type LayoutProps = { | ||||
|   children: React.ReactNode; | ||||
| }; | ||||
| 
 | ||||
| export function Layout({ children }: LayoutProps) { | ||||
|   return ( | ||||
|     <> | ||||
|         <Sidebar /> | ||||
|         <main className="py-10 lg:pl-72"> | ||||
|           <div className="px-4 sm:px-6 lg:px-8"> | ||||
|             {children} | ||||
|           </div> | ||||
|         </main> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
							
								
								
									
										166
									
								
								src/components/Sidebar.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										166
									
								
								src/components/Sidebar.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,166 @@ | |||
| import { Fragment, useState } from 'react'; | ||||
| import { Dialog, Transition } from '@headlessui/react'; | ||||
| import { | ||||
|   Bars3Icon, | ||||
|   CalendarIcon, | ||||
|   ChartPieIcon, | ||||
|   DocumentDuplicateIcon, | ||||
|   FolderIcon, | ||||
|   HomeIcon, | ||||
|   UsersIcon, | ||||
|   XMarkIcon, | ||||
|   ArrowRightOnRectangleIcon, | ||||
| } from '@heroicons/react/24/outline'; | ||||
| 
 | ||||
| const navigation = [ | ||||
|   { name: 'Dashboard', href: '#', icon: HomeIcon, current: true }, | ||||
|   { name: 'Team', href: '#', icon: UsersIcon, current: false }, | ||||
|   { name: 'Projects', href: '#', icon: FolderIcon, current: false }, | ||||
|   { name: 'Calendar', href: '#', icon: CalendarIcon, current: false }, | ||||
|   { name: 'Documents', href: '#', icon: DocumentDuplicateIcon, current: false }, | ||||
|   { name: 'Reports', href: '#', icon: ChartPieIcon, current: false }, | ||||
| ] | ||||
| 
 | ||||
| function classNames(...classes: any) { | ||||
|   return classes.filter(Boolean).join(' ') | ||||
| } | ||||
| 
 | ||||
| export function Sidebar() { | ||||
|   const [sidebarOpen, setSidebarOpen] = useState(false) | ||||
| 
 | ||||
|   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}> | ||||
|                               <a | ||||
|                                 href={item.href} | ||||
|                                 className={classNames( | ||||
|                                   item.current | ||||
|                                     ? '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} | ||||
|                               </a> | ||||
|                             </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}> | ||||
|                       <a | ||||
|                         href={item.href} | ||||
|                         className={classNames( | ||||
|                           item.current | ||||
|                             ? '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} | ||||
|                       </a> | ||||
|                     </li> | ||||
|                   ))} | ||||
|                 </ul> | ||||
|               </li> | ||||
|               <li className="-mx-6 mt-auto"> | ||||
|                 <a | ||||
|                   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> | ||||
|                 </a> | ||||
|               </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> | ||||
|         <a href="#"> | ||||
|           <span className="sr-only">Sign out</span> | ||||
|           <ArrowRightOnRectangleIcon className="h-6 w-6 text-white" /> | ||||
|         </a> | ||||
|       </div> | ||||
|     </> | ||||
|   ) | ||||
| } | ||||
							
								
								
									
										13
									
								
								src/pages/_document.tsx
									
										
									
									
									
										Normal file
									
								
							
							
						
						
									
										13
									
								
								src/pages/_document.tsx
									
										
									
									
									
										Normal file
									
								
							|  | @ -0,0 +1,13 @@ | |||
| import { Html, Head, Main, NextScript } from 'next/document' | ||||
| 
 | ||||
| export default function Document() { | ||||
|   return ( | ||||
|     <Html className="h-full bg-white"> | ||||
|       <Head /> | ||||
|       <body className="h-full"> | ||||
|         <Main /> | ||||
|         <NextScript /> | ||||
|       </body> | ||||
|     </Html> | ||||
|   ) | ||||
| } | ||||
|  | @ -1,45 +1,18 @@ | |||
| import Head from "next/head"; | ||||
| import Link from "next/link"; | ||||
| 
 | ||||
| import { Layout } from "~/components/Layout"; | ||||
| 
 | ||||
| export default function Home() { | ||||
|   return ( | ||||
|     <> | ||||
|       <Head> | ||||
|         <title>Create T3 App</title> | ||||
|         <title>MaresHQ backoffice</title> | ||||
|         <meta name="description" content="Generated by create-t3-app" /> | ||||
|         <link rel="icon" href="/favicon.ico" /> | ||||
|       </Head> | ||||
|       <main className="flex min-h-screen flex-col items-center justify-center bg-gradient-to-b from-[#2e026d] to-[#15162c]"> | ||||
|         <div className="container flex flex-col items-center justify-center gap-12 px-4 py-16 "> | ||||
|           <h1 className="text-5xl font-extrabold tracking-tight text-white sm:text-[5rem]"> | ||||
|             Create <span className="text-[hsl(280,100%,70%)]">T3</span> App | ||||
|           </h1> | ||||
|           <div className="grid grid-cols-1 gap-4 sm:grid-cols-2 md:gap-8"> | ||||
|             <Link | ||||
|               className="flex max-w-xs flex-col gap-4 rounded-xl bg-white/10 p-4 text-white hover:bg-white/20" | ||||
|               href="https://create.t3.gg/en/usage/first-steps" | ||||
|               target="_blank" | ||||
|             > | ||||
|               <h3 className="text-2xl font-bold">First Steps →</h3> | ||||
|               <div className="text-lg"> | ||||
|                 Just the basics - Everything you need to know to set up your | ||||
|                 database and authentication. | ||||
|               </div> | ||||
|             </Link> | ||||
|             <Link | ||||
|               className="flex max-w-xs flex-col gap-4 rounded-xl bg-white/10 p-4 text-white hover:bg-white/20" | ||||
|               href="https://create.t3.gg/en/introduction" | ||||
|               target="_blank" | ||||
|             > | ||||
|               <h3 className="text-2xl font-bold">Documentation →</h3> | ||||
|               <div className="text-lg"> | ||||
|                 Learn more about Create T3 App, the libraries it uses, and how | ||||
|                 to deploy it. | ||||
|               </div> | ||||
|             </Link> | ||||
|           </div> | ||||
|         </div> | ||||
|       </main> | ||||
|       <Layout> | ||||
|         <div></div> | ||||
|       </Layout> | ||||
|     </> | ||||
|   ); | ||||
| } | ||||
|  |  | |||
		Reference in a new issue