MIF_E31221222/sigap-website/components/nav-main.tsx

148 lines
3.7 KiB
TypeScript

"use client";
import { ChevronRight } from "lucide-react";
import {
Collapsible,
CollapsibleContent,
CollapsibleTrigger,
} from "@/components/ui/collapsible";
import {
SidebarGroup,
SidebarGroupLabel,
SidebarMenu,
SidebarMenuButton,
SidebarMenuItem,
SidebarMenuSub,
} from "@/components/ui/sidebar";
import * as TablerIcons from "@tabler/icons-react";
interface SubSubItem {
title: string;
url: string;
}
interface SubItem {
title: string;
url: string;
icon?: TablerIcons.Icon;
subSubItems?: SubSubItem[];
}
interface NavItem {
title: string;
url: string;
icon?: TablerIcons.Icon;
isActive?: boolean;
subItems?: SubItem[];
}
function SubSubItemComponent({ item }: { item: SubSubItem }) {
return (
<SidebarMenuItem>
<SidebarMenuButton asChild>
<a href={item.url}>
<span>{item.title}</span>
</a>
</SidebarMenuButton>
</SidebarMenuItem>
);
}
function SubItemComponent({ item }: { item: SubItem }) {
const hasSubSubItems = item.subSubItems && item.subSubItems.length > 0;
if (!hasSubSubItems) {
return (
<SidebarMenuItem>
<SidebarMenuButton asChild>
<a href={item.url}>
{item.icon && <item.icon />}
<span>{item.title}</span>
</a>
</SidebarMenuButton>
</SidebarMenuItem>
);
}
return (
<Collapsible asChild className="group/collapsible">
<SidebarMenuItem>
<CollapsibleTrigger asChild>
<SidebarMenuButton>
{item.icon && <item.icon />}
<span>{item.title}</span>
<ChevronRight className="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" />
</SidebarMenuButton>
</CollapsibleTrigger>
<CollapsibleContent>
<SidebarMenuSub>
{item.subSubItems!.map((subSubItem) => (
<SubSubItemComponent key={subSubItem.title} item={subSubItem} />
))}
</SidebarMenuSub>
</CollapsibleContent>
</SidebarMenuItem>
</Collapsible>
);
}
function RecursiveNavItem({ item, index }: { item: NavItem; index: number }) {
const hasSubItems = item.subItems && item.subItems.length > 0;
if (!hasSubItems) {
return (
<SidebarMenuItem>
<SidebarMenuButton tooltip={item.title} asChild>
<a href={item.url}>
{item.icon && <item.icon />}
<span>{item.title}</span>
</a>
</SidebarMenuButton>
</SidebarMenuItem>
);
}
return (
<Collapsible
key={item.title}
asChild
defaultOpen={index === 1}
className="group/collapsible"
>
<SidebarMenuItem>
<CollapsibleTrigger asChild>
<SidebarMenuButton tooltip={item.title}>
{item.icon && <item.icon />}
<span>{item.title}</span>
{hasSubItems && (
<ChevronRight className="ml-auto transition-transform duration-200 group-data-[state=open]/collapsible:rotate-90" />
)}
</SidebarMenuButton>
</CollapsibleTrigger>
<CollapsibleContent>
<SidebarMenuSub>
{item.subItems!.map((subItem) => (
<SubItemComponent key={subItem.title} item={subItem} />
))}
</SidebarMenuSub>
</CollapsibleContent>
</SidebarMenuItem>
</Collapsible>
);
}
export function NavMain({ items }: { items: NavItem[] }) {
return (
<SidebarGroup>
<SidebarGroupLabel>Platform</SidebarGroupLabel>
<SidebarMenu>
{items.map((item, index) => (
<RecursiveNavItem key={item.title} item={item} index={index} />
))}
</SidebarMenu>
</SidebarGroup>
);
}