computer-scienceangular-roadmapbackend-roadmapblockchain-roadmapdba-roadmapdeveloper-roadmapdevops-roadmapfrontend-roadmapgo-roadmaphactoberfestjava-roadmapjavascript-roadmapnodejs-roadmappython-roadmapqa-roadmapreact-roadmaproadmapstudy-planvue-roadmapweb3-roadmap
You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
105 lines
2.9 KiB
105 lines
2.9 KiB
import { |
|
BookOpen, |
|
BookOpenText, |
|
CheckSquare, |
|
FileQuestion, |
|
Menu, |
|
Shirt, |
|
Video, |
|
Waypoints, |
|
} from 'lucide-react'; |
|
import { useRef, useState } from 'react'; |
|
import { cn } from '../lib/classname.ts'; |
|
import { useOutsideClick } from '../hooks/use-outside-click.ts'; |
|
|
|
const links = [ |
|
{ |
|
link: '/roadmaps', |
|
label: 'Roadmaps', |
|
description: 'Step by step learning paths', |
|
Icon: Waypoints, |
|
}, |
|
{ |
|
link: '/best-practices', |
|
label: 'Best Practices', |
|
description: "Do's and don'ts", |
|
Icon: CheckSquare, |
|
}, |
|
{ |
|
link: '/questions', |
|
label: 'Questions', |
|
description: 'Test and Practice your knowledge', |
|
Icon: FileQuestion, |
|
}, |
|
{ |
|
link: '/guides', |
|
label: 'Guides', |
|
description: 'In-depth articles and tutorials', |
|
Icon: BookOpenText, |
|
}, |
|
{ |
|
link: 'https://youtube.com/@roadmapsh', |
|
label: 'Videos', |
|
description: 'Animated and interactive content', |
|
Icon: Video, |
|
isExternal: true, |
|
}, |
|
{ |
|
link: 'https://cottonbureau.com/people/roadmapsh', |
|
label: 'Shop', |
|
description: 'Get some cool swag', |
|
Icon: Shirt, |
|
isExternal: true, |
|
}, |
|
]; |
|
|
|
export function NavigationDropdown() { |
|
const dropdownRef = useRef<HTMLDivElement>(null); |
|
const [isOpen, setIsOpen] = useState(false); |
|
|
|
useOutsideClick(dropdownRef, () => { |
|
setIsOpen(false); |
|
}); |
|
|
|
return ( |
|
<div className="relative flex items-center" ref={dropdownRef}> |
|
<button |
|
className={cn('text-gray-400 hover:text-white', { |
|
'text-white': isOpen, |
|
})} |
|
onClick={() => setIsOpen(true)} |
|
onMouseOver={() => setIsOpen(true)} |
|
> |
|
<Menu className="h-5 w-5" /> |
|
</button> |
|
<div |
|
className={cn( |
|
'absolute pointer-events-none left-0 top-full z-[999] mt-2 w-48 min-w-[320px] -translate-y-1 rounded-lg bg-slate-800 py-2 opacity-0 shadow-xl transition-all duration-100', |
|
{ |
|
'pointer-events-auto translate-y-2.5 opacity-100': isOpen, |
|
}, |
|
)} |
|
> |
|
{links.map((link) => ( |
|
<a |
|
href={link.link} |
|
target={link.isExternal ? '_blank' : undefined} |
|
rel={link.isExternal ? 'noopener noreferrer' : undefined} |
|
key={link.link} |
|
className="group flex items-center gap-3 px-4 py-2.5 text-gray-400 transition-colors hover:bg-slate-700" |
|
> |
|
<span className="flex h-[40px] w-[40px] items-center justify-center rounded-full bg-slate-600 transition-colors group-hover:bg-slate-500 group-hover:text-slate-100"> |
|
<link.Icon className="inline-block h-5 w-5" /> |
|
</span> |
|
<span className="flex flex-col"> |
|
<span className="font-medium text-slate-300 transition-colors group-hover:text-slate-100"> |
|
{link.label} |
|
</span> |
|
<span className="text-sm">{link.description}</span> |
|
</span> |
|
</a> |
|
))} |
|
</div> |
|
</div> |
|
); |
|
}
|
|
|