Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension


Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
4 changes: 4 additions & 0 deletions CLAUDE.md
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,10 @@ npm run test:unit # Vitest
npm run test:e2e # Playwright
npm run test # All tests (Vitest + Playwright)

npm run analyse # Bundle analyser UI (why is X bundled? bloat? splits?)

fuser -k 3000/tcp 2>/dev/null; rm -f .next/dev/lock # Kill dev server

vercel list # Recent deployments and status
vercel env ls # Check env vars are configured
vercel whoami # Verify CLI is authenticated
Expand Down
7 changes: 0 additions & 7 deletions app/(root)/(no-right-sidebar)/layout.tsx

This file was deleted.

11 changes: 0 additions & 11 deletions app/(root)/(with-right-sidebar)/layout.tsx

This file was deleted.

File renamed without changes.
45 changes: 34 additions & 11 deletions app/(root)/layout.tsx
Original file line number Diff line number Diff line change
@@ -1,16 +1,39 @@
import { LeftSidebar } from "@/components/navigation/left-sidebar";
import { Navbar } from "@/components/navigation/navbar";
import { SidebarProvider } from "@/components/sidebar-provider";
import { cookies } from "next/headers";
import { AppSidebar } from "@/components/app-sidebar";
import { ContentTopBar } from "@/components/navigation/content-top-bar";
import { MobileHeader } from "@/components/navigation/mobile-header";
import { SidebarProvider } from "@/components/ui/sidebar";
import {
CONTENT_HORIZONTAL_PADDING,
cn,
MOBILE_HEADER_TOP_OFFSET,
} from "@/lib/utils";

const RootLayout = async ({ children }: { children: React.ReactNode }) => {
const cookieStore = await cookies();
const defaultOpen = cookieStore.get("sidebar_state")?.value !== "false";

const RootLayout = ({ children }: { children: React.ReactNode }) => {
return (
<SidebarProvider>
<div className="flex min-h-screen flex-col">
<Navbar />
<div className="flex flex-1">
<LeftSidebar />
<main className="flex-1">{children}</main>
</div>
<SidebarProvider defaultOpen={defaultOpen}>
{/* Mobile-only fixed header */}
<MobileHeader />

{/* Full-height sidebar */}
<AppSidebar />

{/* Content area */}
<div className="flex min-h-screen flex-1 flex-col">
{/* Desktop-only top bar */}
<ContentTopBar />
<main
className={cn(
"flex-1 pb-10 sm:pt-10",
MOBILE_HEADER_TOP_OFFSET,
CONTENT_HORIZONTAL_PADDING,
)}
>
{children}
</main>
</div>
</SidebarProvider>
);
Expand Down
File renamed without changes.
File renamed without changes.
2 changes: 1 addition & 1 deletion app/layout.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,7 +30,7 @@ export default function RootLayout({
<body className="flex min-h-full flex-col antialiased">
<ThemeProvider
attribute="class"
defaultTheme="system"
defaultTheme="dark"
enableSystem
disableTransitionOnChange
>
Expand Down
2 changes: 1 addition & 1 deletion components.json
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
"tsx": true,
"tailwind": {
"config": "",
"css": "globals.css",
"css": "app/globals.css",
"baseColor": "slate",
"cssVariables": true,
"prefix": ""
Expand Down
117 changes: 117 additions & 0 deletions components/app-sidebar.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,117 @@
"use client";

import { SignedIn, UserButton } from "@clerk/nextjs";
import Image from "next/image";
import Link from "next/link";
import { usePathname } from "next/navigation";
import { ThemeLogo } from "@/components/navigation/full-logo";
import { NAV_LINKS } from "@/components/navigation/nav-links.constants";
import { SidebarToggleButton } from "@/components/navigation/sidebar-toggle-button";
import {
Sidebar,
SidebarContent,
SidebarFooter,
SidebarGroup,
SidebarGroupContent,
SidebarHeader,
SidebarMenu,
SidebarMenuButton,
SidebarMenuItem,
SidebarRail,
} from "@/components/ui/sidebar";
import {
cn,
getNavIconClasses,
isRouteActive,
NAV_LINK_ACTIVE_CLASSES,
NAV_LINK_INACTIVE_CLASSES,
} from "@/lib/utils";

export function AppSidebar() {
const pathname = usePathname();

return (
<Sidebar id="app-sidebar" collapsible="icon">
{/* Header: Logo - h-14 matches ContentTopBar height */}
<SidebarHeader className="h-14 flex-row items-center px-3">
<Link
href="/"
aria-label="DevFlow sidebar logo"
className="flex items-center group-data-[collapsible=icon]:size-6 group-data-[collapsible=icon]:justify-center"
>
{/* Icon-only when collapsed */}
{/* biome-ignore lint/a11y/useAltText: Decorative logo, aria-label on parent link */}
{/* biome-ignore lint/performance/noImgElement: SVG logo doesn't benefit from next/image optimisation */}
<img
src="/images/site-logo.svg"
className="size-6 group-data-[collapsible=icon]:block hidden"
/>
{/* Full logo when expanded */}
<ThemeLogo className="group-data-[collapsible=icon]:hidden" />
</Link>
</SidebarHeader>

{/* Main Navigation */}
<SidebarContent>
<SidebarGroup>
<SidebarGroupContent>
<SidebarMenu className="gap-3">
{NAV_LINKS.map((link) => {
const isActive = isRouteActive(pathname, link.route);

return (
<SidebarMenuItem key={link.route}>
<SidebarMenuButton
asChild
isActive={isActive}
tooltip={link.label}
className={cn(
"rounded-lg",
isActive
? `${NAV_LINK_ACTIVE_CLASSES} hover:bg-(image:--gradient-primary) hover:text-primary-foreground`
: NAV_LINK_INACTIVE_CLASSES,
)}
>
<Link href={link.route}>
<Image
src={link.imgURL}
alt=""
width={20}
height={20}
className={getNavIconClasses(isActive)}
/>
<span>{link.label}</span>
</Link>
</SidebarMenuButton>
</SidebarMenuItem>
);
})}
</SidebarMenu>
</SidebarGroupContent>
</SidebarGroup>
</SidebarContent>

{/* Footer: UserButton + Toggle */}
<SidebarFooter className="p-2">
<div
className={cn(
"flex items-center gap-2",
// Expanded: horizontal row with space between
"justify-between",
// Collapsed (icon mode): vertical stack
"group-data-[collapsible=icon]:flex-col",
"group-data-[collapsible=icon]:justify-start",
)}
>
<SignedIn>
<UserButton />
</SignedIn>
<SidebarToggleButton />
</div>
</SidebarFooter>

{/* Edge-click rail to toggle */}
<SidebarRail />
</Sidebar>
);
}
2 changes: 1 addition & 1 deletion components/clerk-provider.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -19,7 +19,7 @@ export function ClerkProvider({
},
elements: {
formButtonPrimary:
"bg-[image:var(--gradient-primary)] text-white hover:opacity-90",
"bg-(image:--gradient-primary) text-white hover:opacity-90",
footer: "bg-card",
avatarBox: "size-8",
},
Expand Down
34 changes: 34 additions & 0 deletions components/navigation/content-top-bar.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,34 @@
import { SignedOut, SignInButton, SignUpButton } from "@clerk/nextjs";
import { ThemeToggle } from "@/components/navigation/theme-toggle";
import { Button } from "@/components/ui/button";
import { CONTENT_HORIZONTAL_PADDING, cn, HEADER_HEIGHT } from "@/lib/utils";

export function ContentTopBar() {
return (
<header
className={cn(
"sticky top-0 z-40 hidden items-center gap-4 border-b bg-background sm:flex",
HEADER_HEIGHT,
CONTENT_HORIZONTAL_PADDING,
)}
>
{/* Left: Search - grows and centres */}
<div className="flex flex-1 justify-center">
<p className="text-muted-foreground">Global Search</p>
</div>

{/* Right: Theme + Auth - fixed width */}
<div className="flex flex-none items-center gap-2">
<ThemeToggle />
<SignedOut>
<SignInButton>
<Button variant="ghost">Sign in</Button>
</SignInButton>
<SignUpButton>
<Button>Sign up</Button>
</SignUpButton>
</SignedOut>
</div>
</header>
);
}
2 changes: 1 addition & 1 deletion components/navigation/full-logo.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -17,7 +17,7 @@ export function ThemeLogo({ className }: ThemeLogoProps) {
<span
style={{ aspectRatio: `${LOGO_WIDTH}/${LOGO_HEIGHT}` }}
className={cn(
"h-7 bg-(image:--logo-full-themed) bg-contain bg-no-repeat",
"h-6 bg-(image:--logo-full-themed) bg-contain bg-no-repeat",
className,
)}
role="img"
Expand Down
59 changes: 0 additions & 59 deletions components/navigation/left-sidebar.tsx

This file was deleted.

28 changes: 28 additions & 0 deletions components/navigation/mobile-header.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,28 @@
import Link from "next/link";
import { MobileNav } from "@/components/navigation/mobile-nav";
import { ThemeToggle } from "@/components/navigation/theme-toggle";
import { cn, HEADER_HEIGHT } from "@/lib/utils";

export function MobileHeader() {
return (
<header
className={cn(
"fixed top-0 z-50 flex w-full items-center justify-between bg-sidebar px-4 shadow-sm sm:hidden",
HEADER_HEIGHT,
)}
>
{/* Left: Logo icon */}
<Link href="/" aria-label="DevFlow mobile logo">
{/* biome-ignore lint/a11y/useAltText: Decorative logo, aria-label on parent link */}
{/* biome-ignore lint/performance/noImgElement: SVG logo doesn't benefit from next/image optimisation */}
<img src="/images/site-logo.svg" className="size-7" />
</Link>

{/* Right: Theme toggle + hamburger */}
<div className="flex items-center gap-2">
<ThemeToggle />
<MobileNav />
</div>
</header>
);
}
2 changes: 1 addition & 1 deletion components/navigation/mobile-nav.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -75,7 +75,7 @@ export function MobileNav() {
</SheetClose>

{/* Navigation Links */}
<nav className="flex flex-col gap-3 pt-9">
<nav className="flex flex-col gap-3 pt-4">
{NAV_LINKS.map((link) => (
<SheetClose key={link.route} asChild>
<NavLink
Expand Down
Loading
Loading