mirror of
https://github.com/fosrl/pangolin.git
synced 2026-01-28 22:00:51 +00:00
179 lines
7.4 KiB
TypeScript
179 lines
7.4 KiB
TypeScript
"use client";
|
|
|
|
import React, { useEffect, useState } from "react";
|
|
import { SidebarNav } from "@app/components/SidebarNav";
|
|
import { OrgSelector } from "@app/components/OrgSelector";
|
|
import { cn } from "@app/lib/cn";
|
|
import { ListUserOrgsResponse } from "@server/routers/org";
|
|
import SupporterStatus from "@app/components/SupporterStatus";
|
|
import { ExternalLink, Server, BookOpenText } from "lucide-react";
|
|
import Link from "next/link";
|
|
import { usePathname } from "next/navigation";
|
|
import { useUserContext } from "@app/hooks/useUserContext";
|
|
import { useLicenseStatusContext } from "@app/hooks/useLicenseStatusContext";
|
|
import { useEnvContext } from "@app/hooks/useEnvContext";
|
|
import { useTranslations } from "next-intl";
|
|
import type { SidebarNavSection } from "@app/app/navigation";
|
|
import {
|
|
Tooltip,
|
|
TooltipContent,
|
|
TooltipProvider,
|
|
TooltipTrigger
|
|
} from "@app/components/ui/tooltip";
|
|
|
|
interface LayoutSidebarProps {
|
|
orgId?: string;
|
|
orgs?: ListUserOrgsResponse["orgs"];
|
|
navItems: SidebarNavSection[];
|
|
defaultSidebarCollapsed: boolean;
|
|
}
|
|
|
|
export function LayoutSidebar({
|
|
orgId,
|
|
orgs,
|
|
navItems,
|
|
defaultSidebarCollapsed
|
|
}: LayoutSidebarProps) {
|
|
const [isSidebarCollapsed, setIsSidebarCollapsed] = useState(
|
|
defaultSidebarCollapsed
|
|
);
|
|
const pathname = usePathname();
|
|
const isAdminPage = pathname?.startsWith("/admin");
|
|
const { user } = useUserContext();
|
|
const { isUnlocked } = useLicenseStatusContext();
|
|
const { env } = useEnvContext();
|
|
const t = useTranslations();
|
|
|
|
const setSidebarStateCookie = (collapsed: boolean) => {
|
|
if (typeof window !== "undefined") {
|
|
const isSecure = window.location.protocol === "https:";
|
|
document.cookie = `pangolin-sidebar-state=${collapsed ? "collapsed" : "expanded"}; path=/; max-age=${60 * 60 * 24 * 30}; samesite=lax${isSecure ? "; secure" : ""}`;
|
|
}
|
|
};
|
|
|
|
useEffect(() => {
|
|
setSidebarStateCookie(isSidebarCollapsed);
|
|
}, [isSidebarCollapsed]);
|
|
|
|
return (
|
|
<div
|
|
className={cn(
|
|
"hidden md:flex border-r bg-card flex-col h-full shrink-0 relative",
|
|
isSidebarCollapsed ? "w-16" : "w-64"
|
|
)}
|
|
>
|
|
<div className="flex-1 overflow-y-auto">
|
|
<div className="p-4">
|
|
<OrgSelector
|
|
orgId={orgId}
|
|
orgs={orgs}
|
|
isCollapsed={isSidebarCollapsed}
|
|
/>
|
|
</div>
|
|
<div className="px-4 pt-1">
|
|
{!isAdminPage && user.serverAdmin && (
|
|
<div className="pb-4">
|
|
<Link
|
|
href="/admin"
|
|
className={cn(
|
|
"flex items-center rounded transition-colors text-muted-foreground hover:text-foreground text-sm w-full hover:bg-secondary/50 dark:hover:bg-secondary/20 rounded-md",
|
|
isSidebarCollapsed
|
|
? "px-2 py-2 justify-center"
|
|
: "px-3 py-1.5"
|
|
)}
|
|
title={
|
|
isSidebarCollapsed
|
|
? t("serverAdmin")
|
|
: undefined
|
|
}
|
|
>
|
|
<span
|
|
className={cn(
|
|
"flex-shrink-0",
|
|
!isSidebarCollapsed && "mr-2"
|
|
)}
|
|
>
|
|
<Server className="h-4 w-4" />
|
|
</span>
|
|
{!isSidebarCollapsed && (
|
|
<span>{t("serverAdmin")}</span>
|
|
)}
|
|
</Link>
|
|
</div>
|
|
)}
|
|
<SidebarNav
|
|
sections={navItems}
|
|
isCollapsed={isSidebarCollapsed}
|
|
/>
|
|
</div>
|
|
</div>
|
|
<div className="p-4 space-y-4 border-t shrink-0">
|
|
<SupporterStatus isCollapsed={isSidebarCollapsed} />
|
|
{!isSidebarCollapsed && (
|
|
<div className="space-y-2">
|
|
<div className="text-xs text-muted-foreground text-center">
|
|
<Link
|
|
href="https://github.com/fosrl/pangolin"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="flex items-center justify-center gap-1"
|
|
>
|
|
{!isUnlocked()
|
|
? t("communityEdition")
|
|
: t("commercialEdition")}
|
|
<ExternalLink size={12} />
|
|
</Link>
|
|
</div>
|
|
<div className="text-xs text-muted-foreground ">
|
|
<Link
|
|
href="https://docs.fossorial.io/"
|
|
target="_blank"
|
|
rel="noopener noreferrer"
|
|
className="flex items-center justify-center gap-1"
|
|
>
|
|
{t("documentation")}
|
|
<BookOpenText size={12} />
|
|
</Link>
|
|
</div>
|
|
{env?.app?.version && (
|
|
<div className="text-xs text-muted-foreground text-center">
|
|
v{env.app.version}
|
|
</div>
|
|
)}
|
|
</div>
|
|
)}
|
|
</div>
|
|
|
|
{/* Collapse button */}
|
|
<TooltipProvider>
|
|
<Tooltip>
|
|
<TooltipTrigger asChild>
|
|
<button
|
|
onClick={() =>
|
|
setIsSidebarCollapsed(!isSidebarCollapsed)
|
|
}
|
|
className="cursor-pointer absolute -right-2.5 top-1/2 transform -translate-y-1/2 w-2 h-8 rounded-full flex items-center justify-center transition-all duration-200 ease-in-out hover:scale-110 group z-[60]"
|
|
aria-label={
|
|
isSidebarCollapsed
|
|
? "Expand sidebar"
|
|
: "Collapse sidebar"
|
|
}
|
|
>
|
|
<div className="w-0.5 h-4 bg-current opacity-30 group-hover:opacity-100 transition-opacity duration-200" />
|
|
</button>
|
|
</TooltipTrigger>
|
|
<TooltipContent side="right" sideOffset={8}>
|
|
<p>
|
|
{isSidebarCollapsed
|
|
? t("sidebarExpand")
|
|
: t("sidebarCollapse")}
|
|
</p>
|
|
</TooltipContent>
|
|
</Tooltip>
|
|
</TooltipProvider>
|
|
</div>
|
|
);
|
|
}
|
|
|
|
export default LayoutSidebar;
|