37679e83e6
- Next.js 16 + Appwrite multi-tenant emlak CRM - Database: kovakemlak-db (properties, customers, customer_searches, property_matches, presentations, investors, activities, tenant_settings) - Same stack as isletmem-kovakcrm (shadcn/ui template base) - Modules: portföy, müşteri takibi, arama kriterleri, otomatik eşleştirme, sunum linki, yatırımcı portalı
102 lines
3.2 KiB
TypeScript
102 lines
3.2 KiB
TypeScript
"use client"
|
|
|
|
import { ChevronRight, type LucideIcon } from "lucide-react"
|
|
import Link from "next/link"
|
|
import { usePathname } from "next/navigation"
|
|
|
|
import {
|
|
Collapsible,
|
|
CollapsibleContent,
|
|
CollapsibleTrigger,
|
|
} from "@/components/ui/collapsible"
|
|
import {
|
|
SidebarGroup,
|
|
SidebarGroupLabel,
|
|
SidebarMenu,
|
|
SidebarMenuButton,
|
|
SidebarMenuItem,
|
|
SidebarMenuSub,
|
|
SidebarMenuSubButton,
|
|
SidebarMenuSubItem,
|
|
} from "@/components/ui/sidebar"
|
|
|
|
export function NavMain({
|
|
label,
|
|
items,
|
|
}: {
|
|
label: string
|
|
items: {
|
|
title: string
|
|
url: string
|
|
icon?: LucideIcon
|
|
isActive?: boolean
|
|
items?: {
|
|
title: string
|
|
url: string
|
|
isActive?: boolean
|
|
}[]
|
|
}[]
|
|
}) {
|
|
const pathname = usePathname()
|
|
|
|
// Check if any subitem is active to determine if parent should be open
|
|
const shouldBeOpen = (item: typeof items[0]) => {
|
|
if (item.isActive) return true
|
|
return item.items?.some(subItem => pathname === subItem.url) || false
|
|
}
|
|
|
|
return (
|
|
<SidebarGroup>
|
|
<SidebarGroupLabel>{label}</SidebarGroupLabel>
|
|
<SidebarMenu>
|
|
{items.map((item) => (
|
|
<Collapsible
|
|
key={item.title}
|
|
asChild
|
|
defaultOpen={shouldBeOpen(item)}
|
|
className="group/collapsible"
|
|
>
|
|
<SidebarMenuItem>
|
|
{item.items?.length ? (
|
|
<>
|
|
<CollapsibleTrigger asChild>
|
|
<SidebarMenuButton tooltip={item.title} className="cursor-pointer">
|
|
{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.items?.map((subItem) => (
|
|
<SidebarMenuSubItem key={subItem.title}>
|
|
<SidebarMenuSubButton asChild className="cursor-pointer" isActive={pathname === subItem.url}>
|
|
<Link
|
|
href={subItem.url}
|
|
target={(item.title === "Auth Pages" || item.title === "Errors") ? "_blank" : undefined}
|
|
rel={(item.title === "Auth Pages" || item.title === "Errors") ? "noopener noreferrer" : undefined}
|
|
>
|
|
<span>{subItem.title}</span>
|
|
</Link>
|
|
</SidebarMenuSubButton>
|
|
</SidebarMenuSubItem>
|
|
))}
|
|
</SidebarMenuSub>
|
|
</CollapsibleContent>
|
|
</>
|
|
) : (
|
|
<SidebarMenuButton asChild tooltip={item.title} className="cursor-pointer" isActive={pathname === item.url}>
|
|
<Link href={item.url}>
|
|
{item.icon && <item.icon />}
|
|
<span>{item.title}</span>
|
|
</Link>
|
|
</SidebarMenuButton>
|
|
)}
|
|
</SidebarMenuItem>
|
|
</Collapsible>
|
|
))}
|
|
</SidebarMenu>
|
|
</SidebarGroup>
|
|
)
|
|
}
|