fosrl.pangolin/src/components/SidebarNav.tsx

130 lines
4.2 KiB
TypeScript
Raw Normal View History

"use client";
import React, { useEffect } from "react";
import Link from "next/link";
import { useParams, usePathname, useRouter } from "next/navigation";
2025-01-01 21:41:31 -05:00
import { cn } from "@app/lib/cn";
import { buttonVariants } from "@/components/ui/button";
import {
Select,
SelectContent,
SelectItem,
SelectTrigger,
SelectValue,
} from "@/components/ui/select";
2024-10-13 16:18:54 -04:00
interface SidebarNavProps extends React.HTMLAttributes<HTMLElement> {
2024-10-13 23:42:09 -04:00
items: {
href: string;
title: string;
2024-11-16 12:18:22 -05:00
icon?: React.ReactNode;
}[];
disabled?: boolean;
2024-10-13 16:18:54 -04:00
}
export function SidebarNav({
className,
items,
disabled = false,
...props
}: SidebarNavProps) {
2024-10-13 23:42:09 -04:00
const pathname = usePathname();
const params = useParams();
const orgId = params.orgId as string;
2024-10-14 22:26:32 -04:00
const niceId = params.niceId as string;
2024-10-13 23:42:09 -04:00
const resourceId = params.resourceId as string;
2024-11-09 23:59:19 -05:00
const userId = params.userId as string;
2024-10-13 16:18:54 -04:00
const [selectedValue, setSelectedValue] = React.useState<string>(getSelectedValue());
useEffect(() => {
setSelectedValue(getSelectedValue());
}, [usePathname()]);
const router = useRouter();
const handleSelectChange = (value: string) => {
if (!disabled) {
router.push(value);
}
};
function getSelectedValue() {
const item = items.find((item) => hydrateHref(item.href) === pathname);
return hydrateHref(item?.href || "");
}
function hydrateHref(val: string): string {
return val
.replace("{orgId}", orgId)
.replace("{niceId}", niceId)
2024-11-09 23:59:19 -05:00
.replace("{resourceId}", resourceId)
.replace("{userId}", userId);
}
2024-10-13 23:42:09 -04:00
return (
<div>
<div className="block lg:hidden">
<Select
defaultValue={selectedValue}
value={selectedValue}
onValueChange={handleSelectChange}
disabled={disabled}
2024-10-13 23:42:09 -04:00
>
<SelectTrigger>
<SelectValue placeholder="Select an option" />
</SelectTrigger>
<SelectContent>
{items.map((item) => (
<SelectItem
key={hydrateHref(item.href)}
value={hydrateHref(item.href)}
>
{item.title}
</SelectItem>
))}
</SelectContent>
</Select>
</div>
<nav
className={cn(
2025-01-04 20:22:01 -05:00
"hidden lg:flex space-x-2 lg:flex-col lg:space-x-0 lg:space-y-3 pr-8",
disabled && "opacity-50 pointer-events-none",
className
)}
{...props}
>
{items.map((item) => (
<Link
2024-11-09 23:59:19 -05:00
key={hydrateHref(item.href)}
href={hydrateHref(item.href)}
className={cn(
buttonVariants({ variant: "ghost" }),
2024-11-09 23:59:19 -05:00
pathname === hydrateHref(item.href) &&
!pathname.includes("create")
2025-01-04 20:22:01 -05:00
? "bg-accent hover:bg-accent dark:bg-border dark:hover:bg-border"
: "hover:bg-transparent hover:underline",
"justify-start",
disabled && "cursor-not-allowed"
)}
onClick={
disabled ? (e) => e.preventDefault() : undefined
}
tabIndex={disabled ? -1 : undefined}
aria-disabled={disabled}
>
2024-11-16 12:18:22 -05:00
{item.icon ? (
<div className="flex items-center space-x-2">
{item.icon}
<span>{item.title}</span>
</div>
) : (
item.title
)}
</Link>
))}
</nav>
</div>
);
2024-10-17 22:12:02 -04:00
}