add settings navigation

This commit is contained in:
Philipinho 2023-09-04 19:38:40 +01:00
parent 6af5c9a9ca
commit 03a9b81a80
10 changed files with 315 additions and 108 deletions

View File

@ -10,15 +10,20 @@
"lint": "next lint"
},
"dependencies": {
"@hookform/resolvers": "^3.3.0",
"@hookform/resolvers": "^3.3.1",
"@radix-ui/react-dialog": "^1.0.4",
"@radix-ui/react-icons": "^1.3.0",
"@radix-ui/react-label": "^2.0.2",
"@radix-ui/react-popover": "^1.0.6",
"@radix-ui/react-scroll-area": "^1.0.4",
"@radix-ui/react-select": "^1.2.2",
"@radix-ui/react-separator": "^1.0.3",
"@radix-ui/react-slot": "^1.0.2",
"@radix-ui/react-toast": "^1.1.4",
"@radix-ui/react-tooltip": "^1.0.6",
"@tabler/icons-react": "^2.32.0",
"@tanstack/react-query": "^4.33.0",
"@tanstack/react-table": "^8.9.3",
"@types/node": "20.4.8",
"@types/react": "18.2.18",
"@types/react-dom": "18.2.7",
@ -26,9 +31,11 @@
"axios": "^1.4.0",
"class-variance-authority": "^0.7.0",
"clsx": "^2.0.0",
"cmdk": "^0.2.0",
"eslint": "8.46.0",
"eslint-config-next": "13.4.13",
"jotai": "^2.3.1",
"jotai-optics": "^0.3.1",
"js-cookie": "^3.0.5",
"next": "13.4.13",
"next-themes": "^0.2.1",
@ -36,12 +43,15 @@
"react": "18.2.0",
"react-dom": "18.2.0",
"react-hook-form": "^7.45.4",
"react-hot-toast": "^2.4.1",
"tailwind-merge": "^1.14.0",
"tailwindcss": "3.3.3",
"tailwindcss-animate": "^1.0.6",
"typescript": "5.1.6"
"typescript": "5.1.6",
"zod": "^3.22.2"
},
"devDependencies": {
"@types/js-cookie": "^3.0.3"
"@types/js-cookie": "^3.0.3",
"optics-ts": "^2.4.1"
}
}

View File

@ -1,37 +0,0 @@
import { ReactNode } from 'react';
import {
IconHome,
IconSearch,
IconSettings,
IconFilePlus,
} from '@tabler/icons-react';
export type NavigationMenuType = {
label: string;
path: string;
icon: ReactNode;
isActive?: boolean;
onClick?: React.MouseEventHandler<HTMLAnchorElement | HTMLButtonElement>;
};
export const navigationMenu: NavigationMenuType[] = [
{
label: 'Home',
path: '',
icon: <IconHome size={16} />,
},
{
label: 'Search',
path: '',
icon: <IconSearch size={16} />,
},
{
label: 'Settings',
path: '',
icon: <IconSettings size={16} />,
},
{
label: 'New Page',
path: '',
icon: <IconFilePlus size={16} />,
},
];

View File

@ -0,0 +1,68 @@
import React, { ReactNode } from 'react';
import {
IconHome,
IconSearch,
IconSettings,
IconFilePlus,
} from '@tabler/icons-react';
import NavigationLink from "@/components/sidebar/navigation/navigation-link";
import ButtonWithIcon from "@/components/ui/button-with-icon";
export type NavigationMenuType = {
label: string;
path: string;
icon: ReactNode;
target?: string,
onClick?: React.MouseEventHandler<HTMLAnchorElement | HTMLButtonElement>;
};
export const navigationMenu: NavigationMenuType[] = [
{
label: 'Home',
path: '',
icon: <IconHome size={16} />,
target: '/home',
},
{
label: 'Search',
path: '',
icon: <IconSearch size={16} />,
},
{
label: 'Settings',
path: '',
icon: <IconSettings size={16} />,
target: '/settings/account'
},
{
label: 'New Page',
path: '',
icon: <IconFilePlus size={16} />,
},
];
export const renderMenuItem = (menu, index) => {
if (menu.target) {
return (
<NavigationLink
href={menu.target}
icon={menu.icon}
className="w-full flex flex-1 justify-start items-center"
>
{menu.label}
</NavigationLink>
);
}
return (
<ButtonWithIcon
key={index}
icon={menu.icon}
variant="ghost"
className="w-full flex flex-1 justify-start items-center"
// onClick={}
>
<span className="text-ellipsis overflow-hidden">{menu.label}</span>
</ButtonWithIcon>
);
};

View File

@ -0,0 +1,25 @@
import { ReactNode } from "react";
import { buttonVariants } from "@/components/ui/button";
import Link from "next/link";
import { cn } from "@/lib/utils";
interface NavigationLinkProps {
children: ReactNode,
href: string;
icon?: ReactNode;
variant?: "default" | "ghost" | "outline";
className?: string;
}
export default function NavigationLink({ children, href, icon, variant = "ghost", className }: NavigationLinkProps) {
return (
<Link href={href} className={cn(buttonVariants({ variant: variant }), className)}>
{icon && <span className="mr-[8px]">
{icon}
</span>}
<span className="text-ellipsis overflow-hidden">
{children}
</span>
</Link>
);
}

View File

@ -0,0 +1,40 @@
import { SidebarSection } from "@/components/sidebar/sidebar-section";
import { navigationMenu, renderMenuItem } from "@/components/sidebar/navigation/navigation-items";
import { ScrollArea } from "@/components/ui/scroll-area";
import NavigationLink from "@/components/sidebar/navigation/navigation-link";
import { IconFileText } from "@tabler/icons-react";
import React from "react";
export default function Navigation() {
return (
<div className="pt-8">
<PrimaryNavigation />
<SecondaryNavigationArea />
</div>
);
}
function PrimaryNavigation() {
return (
<SidebarSection className="pb-2 mb-4 select-none border-b">
{navigationMenu.map(renderMenuItem)}
</SidebarSection>
);
}
function SecondaryNavigationArea() {
return (
<ScrollArea className="h-[70vh]">
<div className="space-y-1">
<NavigationLink
href="#"
className="w-full justify-start"
icon={<IconFileText size={16} />}
>
Welcome page
</NavigationLink>
</div>
</ScrollArea>
);
}

View File

@ -32,10 +32,16 @@ export default function SidebarToggleButton({
return (
<ButtonWithIcon
className={cn(className, 'z-[100]')}
className={cn(className, 'z-50')}
icon={<SidebarIcon size={20} />}
variant={'ghost'}
onClick={toggleSidebar}
></ButtonWithIcon>
/>
);
}
export function MobileSidebarToggle({ isSidebarOpen }) {
return (
<SidebarToggleButton className={`absolute top-0 ${isSidebarOpen ? "right-0" : "left-0"} right-0 m-4`} />
);
}

View File

@ -1,79 +1,56 @@
import { useIsMobile } from '@/hooks/use-is-mobile';
import { useAtom } from 'jotai';
import { useIsMobile } from "@/hooks/use-is-mobile";
import { useAtom } from "jotai";
import {
desktopSidebarAtom,
mobileSidebarAtom,
} from '@/components/sidebar/atoms/sidebar-atom';
import { ScrollArea } from '@/components/ui/scroll-area';
import { IconFileText } from '@tabler/icons-react';
import { SidebarSection } from '@/components/sidebar/sidebar-section';
import {
navigationMenu,
NavigationMenuType,
} from '@/components/sidebar/actions/sidebar-actions';
import ButtonWithIcon from '@/components/ui/button-with-icon';
import SidebarToggleButton from './sidebar-toggle-button';
} from "@/components/sidebar/atoms/sidebar-atom";
import { MobileSidebarToggle } from "./sidebar-toggle-button";
import SettingsNav from "@/features/settings/nav/settings-nav";
import { usePathname } from "next/navigation";
import React, { useEffect } from "react";
import Navigation from "@/components/sidebar/navigation/navigation";
export default function Sidebar() {
const isMobile = useIsMobile();
const pathname = usePathname();
const [isSidebarOpen, setIsSidebarOpen] = useAtom(isMobile ? mobileSidebarAtom : desktopSidebarAtom);
const isSettings = pathname.startsWith("/settings");
const [isSidebarOpen] = useAtom(
isMobile ? mobileSidebarAtom : desktopSidebarAtom
);
const mobileClass = "fixed top-0 left-0 h-screen z-50 bg-background";
const sidebarWidth = isSidebarOpen ? "w-[270px]" : "w-[0px]";
const closeSidebar = () => {
setIsSidebarOpen(false);
};
useEffect(() => {
if (isMobile) {
setIsSidebarOpen(false);
}
}, [pathname, isMobile, setIsSidebarOpen]);
return (
<nav
className={`${
isSidebarOpen ? (isMobile ? 'w-full' : 'w-[270px]') : 'w-[0px]'
} ${
isMobile && isSidebarOpen
? 'fixed top-0 left-0 h-screen z-[99] bg-background'
: ''
} flex-grow-0 flex-shrink-0 overflow-hidden border-r duration-500 ease-in-out`}
>
{isMobile && (
<>
<SidebarToggleButton
className={`absolute top-0 ${
isSidebarOpen ? 'right-0' : 'left-0'
} right-0 m-4`}
/>
<div className="mt-[20px]"></div>
</>
<>
{isMobile && isSidebarOpen && (
<div className="fixed top-0 left-0 w-full h-screen z-[50] bg-black/60"
onClick={closeSidebar}>
</div>
)}
<div className={`flex flex-col flex-shrink-0 gap-0.5 p-[10px]`}>
<div className="h-full">
<div className="mt-[20px]"></div>
<nav
className={`${sidebarWidth} ${isMobile && isSidebarOpen ? mobileClass : ""}
flex-grow-0 flex-shrink-0 overflow-hidden border-r duration-300 z-49`}>
<SidebarSection className="pb-2 mb-4 select-none border-b">
{navigationMenu.map((menu: NavigationMenuType, index: number) => (
<ButtonWithIcon
key={index}
icon={menu.icon}
variant={'ghost'}
className="w-full flex flex-1 justify-start items-center"
>
<span className="text-ellipsis overflow-hidden">
{menu.label}
</span>
</ButtonWithIcon>
))}
</SidebarSection>
{isMobile && (
<MobileSidebarToggle isSidebarOpen={isSidebarOpen} />
)}
<ScrollArea className="h-[70vh]">
<div className="space-y-1">
<ButtonWithIcon
variant="ghost"
className="w-full justify-start"
icon={<IconFileText size={16} />}
>
Welcome page
</ButtonWithIcon>
</div>
</ScrollArea>
<div className="flex flex-col flex-shrink-0 gap-0.5 p-[10px]">
<div className="h-full mt-[8px]">
{isSettings ? <SettingsNav /> : <Navigation />}
</div>
</div>
</div>
</nav>
</nav>
</>
);
}

View File

@ -7,12 +7,13 @@ export default function TopBar() {
const isMobile = useIsMobile();
return (
<header className="max-w-full z-50 select-none">
<div className="w-full max-w-full h-[50px] opacity-100 relative duration-700 ease-in">
<header className="max-w-full z-10 select-none">
<div className="w-full max-w-full h-[50px] relative">
<div className="flex justify-between items-center h-full overflow-hidden py-0 px-1 gap-2.5 border-b">
<div className="flex items-center leading-tight h-full flex-grow-0 mr-[8px] min-w-0 font-semibold text-sm">
<div className="flex items-center h-full flex-grow-0 mr-[8px] min-w-0">
{!isMobile && <SidebarToggleButton />}
</div>
</div>
</div>
</header>

View File

@ -0,0 +1,50 @@
'use client'
import { ReactNode } from 'react';
import { IconUserCircle, IconUser, IconUsers,
IconBuilding, IconSettingsCog } from '@tabler/icons-react';
export interface SettingsNavMenuSection {
heading: string;
icon: ReactNode;
items: SettingsNavMenuItem[];
}
export interface SettingsNavMenuItem {
label: string;
icon: ReactNode;
target?: string;
}
export type SettingsNavItem = SettingsNavMenuSection[];
export const settingsNavItems: SettingsNavItem = [
{
heading: 'Account',
icon: <IconUserCircle size={20}/>,
items: [
{
label: 'My account',
icon: <IconUser size={16}/>,
target: '/settings/account',
},
],
},
{
heading: 'Workspace',
icon: <IconBuilding size={20}/>,
items: [
{
label: 'General',
icon: <IconSettingsCog size={16}/>,
target: '/settings/workspace',
},
{
label: 'Members',
icon: <IconUsers size={16}/>,
target: '/settings/workspace/members',
},
],
},
];

View File

@ -0,0 +1,67 @@
"use client";
import {
SettingsNavItem,
SettingsNavMenuItem, SettingsNavMenuSection,
settingsNavItems
} from "@/features/settings/nav/settings-nav-items";
import { usePathname } from "next/navigation";
import Link from "next/link";
import React from "react";
import { cn } from "@/lib/utils";
import { buttonVariants } from "@/components/ui/button";
import { ChevronLeftIcon } from "@radix-ui/react-icons";
interface SettingsNavProps {
menu: SettingsNavItem;
}
function RenderNavItem({ label, icon, target }: SettingsNavMenuItem): React.ReactNode {
const pathname = usePathname();
const isActive = pathname === target;
return (
<div className="ml-2">
<Link href={target} className={` ${isActive ? "bg-foreground/10 rounded-md" : ""}
w-full flex flex-1 justify-start items-center text-sm font-medium px-3 py-2`}>
<span className="mr-1">{icon}</span>
<span className="text-ellipsis overflow-hidden">
{label}
</span>
</Link>
</div>
);
}
function SettingsNavItems({ menu }: SettingsNavProps): React.ReactNode {
return (
<>
<div>
<Link
href="/home"
className={cn(
buttonVariants({ variant: "ghost" }),
"relative")} style={{marginLeft: '-5px', top:'-5px'}}>
<ChevronLeftIcon className="mr-2 h-4 w-4" /> Back
</Link>
</div>
<div className="p-5 pt-0">
{menu.map((section: SettingsNavMenuSection, index: number) => (
<div key={index}>
<h3 className="flex items-center py-2 text-sm font-semibold text-muted-foreground">
<span className="mr-1">{section.icon}</span> {section.heading}
</h3>
{section.items.map((item: SettingsNavMenuItem, itemIndex: number) => (
<RenderNavItem key={itemIndex} {...item} />
))}
</div>
))}
</div>
</>
);
}
export default function SettingsNav() {
return <SettingsNavItems menu={settingsNavItems} />
}