header.tsx 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260
  1. "use client";
  2. import Link from "next/link";
  3. import Image from "next/image";
  4. import { LoginLink, LogoutLink, RegisterLink } from "@kinde-oss/kinde-auth-nextjs/server";
  5. import { useKindeBrowserClient } from "@kinde-oss/kinde-auth-nextjs";
  6. import {
  7. NavigationMenu,
  8. NavigationMenuContent,
  9. NavigationMenuItem,
  10. NavigationMenuLink,
  11. NavigationMenuList,
  12. NavigationMenuTrigger,
  13. } from "@/components/ui/navigation-menu";
  14. import { Button } from "@/components/ui/button";
  15. import { Sheet, SheetContent, SheetTrigger } from "@/components/ui/sheet";
  16. import { Menu, FileText, Upload, Home, Folder, Settings, User, LogOut, LogIn } from "lucide-react";
  17. import { cn } from "@/lib/utils";
  18. import React from "react";
  19. const navigationItems = [
  20. {
  21. title: "Files",
  22. href: "/files",
  23. description: "Browse and manage your files",
  24. icon: Folder,
  25. },
  26. {
  27. title: "Upload",
  28. href: "/upload",
  29. description: "Upload new files to the platform",
  30. icon: Upload,
  31. },
  32. {
  33. title: "API Docs",
  34. href: "/api-docs",
  35. description: "View API documentation",
  36. icon: FileText,
  37. },
  38. ];
  39. const userMenuItems = [
  40. {
  41. title: "Profile",
  42. href: "/profile",
  43. description: "Manage your profile settings",
  44. icon: User,
  45. },
  46. {
  47. title: "Settings",
  48. href: "/settings",
  49. description: "Configure your preferences",
  50. icon: Settings,
  51. },
  52. ];
  53. export default function Header() {
  54. const { isAuthenticated, getUser } = useKindeBrowserClient();
  55. const user = getUser();
  56. return (
  57. <header className="sticky top-0 z-50 w-full border-b bg-background/95 backdrop-blur supports-[backdrop-filter]:bg-background/60">
  58. <div className="container flex h-16 items-center justify-between">
  59. <div className="flex items-center gap-6">
  60. <Link href="/" className="flex items-center space-x-2">
  61. <Image
  62. src="/vtorio.svg"
  63. alt="vtor.io"
  64. width={32}
  65. height={32}
  66. className="h-8 w-auto"
  67. />
  68. <span className="font-bold text-xl hidden sm:inline">vtor.io</span>
  69. </Link>
  70. <NavigationMenu className="hidden md:flex">
  71. <NavigationMenuList>
  72. <NavigationMenuItem>
  73. <Link href="/" legacyBehavior passHref>
  74. <NavigationMenuLink className={cn(
  75. "group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium transition-colors hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground focus:outline-none disabled:pointer-events-none disabled:opacity-50"
  76. )}>
  77. <Home className="mr-2 h-4 w-4" />
  78. Home
  79. </NavigationMenuLink>
  80. </Link>
  81. </NavigationMenuItem>
  82. {navigationItems.map((item) => (
  83. <NavigationMenuItem key={item.title}>
  84. <Link href={item.href} legacyBehavior passHref>
  85. <NavigationMenuLink className={cn(
  86. "group inline-flex h-9 w-max items-center justify-center rounded-md bg-background px-4 py-2 text-sm font-medium transition-colors hover:bg-accent hover:text-accent-foreground focus:bg-accent focus:text-accent-foreground focus:outline-none disabled:pointer-events-none disabled:opacity-50"
  87. )}>
  88. <item.icon className="mr-2 h-4 w-4" />
  89. {item.title}
  90. </NavigationMenuLink>
  91. </Link>
  92. </NavigationMenuItem>
  93. ))}
  94. </NavigationMenuList>
  95. </NavigationMenu>
  96. </div>
  97. <div className="flex items-center gap-4">
  98. {isAuthenticated ? (
  99. <>
  100. <NavigationMenu className="hidden md:flex">
  101. <NavigationMenuList>
  102. <NavigationMenuItem>
  103. <NavigationMenuTrigger className="h-9">
  104. <div className="flex items-center gap-2">
  105. <Image
  106. className="rounded-full"
  107. src={user?.picture || "/default-avatar.png"}
  108. alt="User Avatar"
  109. width={24}
  110. height={24}
  111. />
  112. <span className="text-sm font-medium">{user?.given_name || user?.email}</span>
  113. </div>
  114. </NavigationMenuTrigger>
  115. <NavigationMenuContent>
  116. <ul className="grid gap-3 p-6 w-[300px]">
  117. {userMenuItems.map((item) => (
  118. <li key={item.title}>
  119. <NavigationMenuLink asChild>
  120. <Link
  121. href={item.href}
  122. className="block select-none space-y-1 rounded-md p-3 leading-none no-underline outline-none transition-colors hover:bg-accent hover:text-accent-foreground"
  123. >
  124. <div className="flex items-center gap-2">
  125. <item.icon className="h-4 w-4" />
  126. <div className="text-sm font-medium leading-none">{item.title}</div>
  127. </div>
  128. <p className="line-clamp-2 text-sm leading-snug text-muted-foreground">
  129. {item.description}
  130. </p>
  131. </Link>
  132. </NavigationMenuLink>
  133. </li>
  134. ))}
  135. <li>
  136. <LogoutLink className="block select-none space-y-1 rounded-md p-3 leading-none no-underline outline-none transition-colors hover:bg-accent hover:text-accent-foreground">
  137. <div className="flex items-center gap-2">
  138. <LogOut className="h-4 w-4" />
  139. <div className="text-sm font-medium leading-none">Logout</div>
  140. </div>
  141. <p className="line-clamp-2 text-sm leading-snug text-muted-foreground">
  142. Sign out of your account
  143. </p>
  144. </LogoutLink>
  145. </li>
  146. </ul>
  147. </NavigationMenuContent>
  148. </NavigationMenuItem>
  149. </NavigationMenuList>
  150. </NavigationMenu>
  151. <div className="hidden md:flex items-center gap-2">
  152. <LogoutLink>
  153. <Button variant="ghost" size="sm">
  154. <LogOut className="h-4 w-4 mr-2" />
  155. Logout
  156. </Button>
  157. </LogoutLink>
  158. </div>
  159. </>
  160. ) : (
  161. <div className="hidden md:flex items-center gap-2">
  162. <LoginLink>
  163. <Button variant="ghost" size="sm">
  164. <LogIn className="h-4 w-4 mr-2" />
  165. Login
  166. </Button>
  167. </LoginLink>
  168. <RegisterLink>
  169. <Button size="sm">Sign Up</Button>
  170. </RegisterLink>
  171. </div>
  172. )}
  173. {/* Mobile Menu */}
  174. <Sheet>
  175. <SheetTrigger asChild className="md:hidden">
  176. <Button variant="ghost" size="icon">
  177. <Menu className="h-5 w-5" />
  178. <span className="sr-only">Toggle menu</span>
  179. </Button>
  180. </SheetTrigger>
  181. <SheetContent side="right" className="w-[300px]">
  182. <div className="flex flex-col space-y-4 mt-8">
  183. <Link href="/" className="flex items-center space-x-2">
  184. <Home className="h-4 w-4" />
  185. <span>Home</span>
  186. </Link>
  187. {navigationItems.map((item) => (
  188. <Link
  189. key={item.title}
  190. href={item.href}
  191. className="flex items-center space-x-2"
  192. >
  193. <item.icon className="h-4 w-4" />
  194. <span>{item.title}</span>
  195. </Link>
  196. ))}
  197. {isAuthenticated ? (
  198. <>
  199. <div className="border-t pt-4">
  200. <div className="flex items-center space-x-2 mb-4">
  201. <Image
  202. className="rounded-full"
  203. src={user?.picture || "/default-avatar.png"}
  204. alt="User Avatar"
  205. width={32}
  206. height={32}
  207. />
  208. <span className="font-medium">{user?.given_name || user?.email}</span>
  209. </div>
  210. {userMenuItems.map((item) => (
  211. <Link
  212. key={item.title}
  213. href={item.href}
  214. className="flex items-center space-x-2 py-2"
  215. >
  216. <item.icon className="h-4 w-4" />
  217. <span>{item.title}</span>
  218. </Link>
  219. ))}
  220. <LogoutLink className="flex items-center space-x-2 py-2 text-red-600">
  221. <LogOut className="h-4 w-4" />
  222. <span>Logout</span>
  223. </LogoutLink>
  224. </div>
  225. </>
  226. ) : (
  227. <div className="border-t pt-4 space-y-2">
  228. <LoginLink>
  229. <Button variant="ghost" className="w-full justify-start">
  230. <LogIn className="h-4 w-4 mr-2" />
  231. Login
  232. </Button>
  233. </LoginLink>
  234. <RegisterLink>
  235. <Button className="w-full">Sign Up</Button>
  236. </RegisterLink>
  237. </div>
  238. )}
  239. </div>
  240. </SheetContent>
  241. </Sheet>
  242. </div>
  243. </div>
  244. </header>
  245. );
  246. }