148 lines
3.7 KiB
TypeScript
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>
|
|
);
|
|
}
|