Configure Navigation Layout | Next.js Supabase SaaS Kit

Choose between sidebar and header navigation layouts, configure collapsed states, and customize the navigation experience for your SaaS application.

Makerkit offers two navigation layouts: sidebar (default) and header. You can configure each workspace independently, with separate settings for personal accounts and team accounts. All layout options are controlled through environment variables.

Quick Configuration

Set your preferred layout in .env.local:

# Personal account workspace layout
NEXT_PUBLIC_USER_NAVIGATION_STYLE=sidebar
# Team account workspace layout
NEXT_PUBLIC_TEAM_NAVIGATION_STYLE=sidebar

Available values: sidebar, header, or custom.

Layout Options Compared

The sidebar layout places navigation on the left side of the screen, providing a persistent, vertically-oriented menu.

Best for:

  • Apps with many navigation items (5+)
  • Complex feature sets needing categorized menus
  • Desktop-first applications
  • Enterprise or admin-heavy dashboards

Configuration:

NEXT_PUBLIC_USER_NAVIGATION_STYLE=sidebar
NEXT_PUBLIC_TEAM_NAVIGATION_STYLE=sidebar

Header Layout

The header layout places navigation horizontally at the top of the screen, with a more compact, traditional web app feel.

Best for:

  • Apps with fewer navigation items (3-5)
  • Consumer-facing products
  • Mobile-first designs
  • Simple, focused applications

Configuration:

NEXT_PUBLIC_USER_NAVIGATION_STYLE=header
NEXT_PUBLIC_TEAM_NAVIGATION_STYLE=header

Default Collapsed State

Control whether the sidebar starts expanded or collapsed:

# Personal account sidebar (home workspace)
NEXT_PUBLIC_HOME_SIDEBAR_COLLAPSED=false
# Team account sidebar
NEXT_PUBLIC_TEAM_SIDEBAR_COLLAPSED=false

Set to true to start with a collapsed icon-only sidebar. Users can still expand it manually.

Collapsible Style

Choose how the sidebar collapses and expands:

# Options: offcanvas, icon, none
NEXT_PUBLIC_SIDEBAR_COLLAPSIBLE_STYLE=offcanvas
StyleBehavior
offcanvasSidebar slides in/out as an overlay (mobile-friendly)
iconSidebar collapses to icons only, expanding on hover
noneSidebar cannot be collapsed

Show or hide the sidebar toggle button:

NEXT_PUBLIC_ENABLE_SIDEBAR_TRIGGER=true

Set to false if you want the sidebar to remain in its configured state without user control.

Complete Configuration Example

Here's a full example for a team-focused SaaS with different layouts per workspace:

# Personal account: simple header navigation
NEXT_PUBLIC_USER_NAVIGATION_STYLE=header
# Team workspace: full sidebar with collapsed default
NEXT_PUBLIC_TEAM_NAVIGATION_STYLE=sidebar
NEXT_PUBLIC_TEAM_SIDEBAR_COLLAPSED=true
NEXT_PUBLIC_SIDEBAR_COLLAPSIBLE_STYLE=icon
NEXT_PUBLIC_ENABLE_SIDEBAR_TRIGGER=true
# Theme settings
NEXT_PUBLIC_DEFAULT_THEME_MODE=system
NEXT_PUBLIC_ENABLE_THEME_TOGGLE=true

Customizing Navigation Items

Navigation items are defined in configuration files, not environment variables:

WorkspaceConfiguration File
Personal Accountapps/web/config/personal-account-navigation.config.tsx
Team Accountapps/web/config/team-account-navigation.config.tsx

Personal Account Navigation

import { CreditCard, Home, User, Settings } from 'lucide-react';
import { NavigationConfigSchema } from '@kit/ui/navigation-schema';
import pathsConfig from '~/config/paths.config';
const iconClasses = 'w-4';
const routes = [
{
label: 'common:routes.application',
children: [
{
label: 'common:routes.home',
path: pathsConfig.app.home,
Icon: <Home className={iconClasses} />,
end: true,
},
],
},
{
label: 'common:routes.settings',
children: [
{
label: 'common:routes.profile',
path: pathsConfig.app.personalAccountSettings,
Icon: <User className={iconClasses} />,
},
{
label: 'common:routes.billing',
path: pathsConfig.app.personalAccountBilling,
Icon: <CreditCard className={iconClasses} />,
},
],
},
];
export const personalAccountNavigationConfig = NavigationConfigSchema.parse({
routes,
style: process.env.NEXT_PUBLIC_USER_NAVIGATION_STYLE,
sidebarCollapsed: process.env.NEXT_PUBLIC_HOME_SIDEBAR_COLLAPSED,
sidebarCollapsedStyle: process.env.NEXT_PUBLIC_SIDEBAR_COLLAPSIBLE_STYLE,
});

Adding a New Navigation Item

To add a custom page to the navigation:

import { BarChart3 } from 'lucide-react';
const routes = [
{
label: 'common:routes.application',
children: [
{
label: 'common:routes.home',
path: pathsConfig.app.home,
Icon: <Home className={iconClasses} />,
end: true,
},
// Add your custom navigation item
{
label: 'common:routes.analytics',
path: '/home/analytics',
Icon: <BarChart3 className={iconClasses} />,
},
],
},
// ... rest of routes
];

Remember to add the translation key to your locale files:

{
"routes": {
"analytics": "Analytics"
}
}

The NavigationConfigSchema supports these properties:

interface NavigationConfig {
routes: {
label: string; // Translation key for section header
collapsible?: boolean; // Allow section to collapse (default: true)
children: {
label: string; // Translation key for item
path: string; // Route path
Icon?: ReactNode; // Lucide icon component
end?: boolean; // Exact path matching
}[];
}[];
style?: 'sidebar' | 'header' | 'custom';
sidebarCollapsed?: boolean | string;
sidebarCollapsedStyle?: 'offcanvas' | 'icon' | 'none';
}

Environment Variables Reference

VariableDefaultOptionsDescription
NEXT_PUBLIC_USER_NAVIGATION_STYLEsidebarsidebar, header, customPersonal account layout
NEXT_PUBLIC_TEAM_NAVIGATION_STYLEsidebarsidebar, header, customTeam account layout
NEXT_PUBLIC_HOME_SIDEBAR_COLLAPSEDfalsetrue, falsePersonal sidebar default state
NEXT_PUBLIC_TEAM_SIDEBAR_COLLAPSEDfalsetrue, falseTeam sidebar default state
NEXT_PUBLIC_SIDEBAR_COLLAPSIBLE_STYLEoffcanvasoffcanvas, icon, noneCollapse behavior
NEXT_PUBLIC_ENABLE_SIDEBAR_TRIGGERtruetrue, falseShow collapse toggle

Common Mistakes

Mixing layout styles inconsistently: If personal accounts use header layout but teams use sidebar, the experience can feel disjointed. Consider the transition between workspaces.

Too many items in header layout: Header navigation works best with 3-5 top-level items. More than that causes horizontal overflow or cramped spacing. Use sidebar layout for complex navigation.

Forgetting mobile behavior: Sidebar layout automatically converts to a slide-out drawer on mobile. Test both layouts on narrow viewports.

Not updating translations: Navigation labels use translation keys. Adding items without corresponding translations shows raw keys like common:routes.analytics.

Verification

After changing layout configuration:

  1. Clear your browser's local storage (layout preferences are cached)
  2. Restart the dev server for environment variable changes
  3. Test both personal account and team account workspaces
  4. Verify mobile responsiveness at 375px viewport width
  5. Check that sidebar collapse/expand works correctly

Frequently Asked Questions

Can I use different layouts for personal and team accounts?
Yes. Set NEXT_PUBLIC_USER_NAVIGATION_STYLE and NEXT_PUBLIC_TEAM_NAVIGATION_STYLE to different values. This is useful when team workspaces need more navigation complexity than personal accounts.
How do I create a completely custom layout?
Set the navigation style to 'custom' and implement your own layout component. You'll need to modify the layout files in apps/web/app/home/ to use your custom navigation component.
Why isn't my sidebar staying collapsed?
User preferences are stored in local storage and override environment defaults. Clear local storage or use the browser's incognito mode to test default behavior.
How do I add icons to navigation items?
Import icons from lucide-react and pass them as the Icon prop. Use className='w-4' to maintain consistent sizing with other navigation icons.

Next Steps