1<script lang="ts">
2 import * as Sheet from '$lib/components/ui/sheet/index.js';
3 import { cn, type WithElementRef } from '$lib/components/ui/utils.js';
4 import type { HTMLAttributes } from 'svelte/elements';
5 import { SIDEBAR_WIDTH_MOBILE } from './constants.js';
6 import { useSidebar } from './context.svelte.js';
7
8 let {
9 ref = $bindable(null),
10 side = 'left',
11 variant = 'sidebar',
12 collapsible = 'offcanvas',
13 class: className,
14 children,
15 ...restProps
16 }: WithElementRef<HTMLAttributes<HTMLDivElement>> & {
17 side?: 'left' | 'right';
18 variant?: 'sidebar' | 'floating' | 'inset';
19 collapsible?: 'offcanvas' | 'icon' | 'none';
20 } = $props();
21
22 const sidebar = useSidebar();
23</script>
24
25{#if collapsible === 'none'}
26 <div
27 class={cn(
28 'flex h-full w-(--sidebar-width) flex-col bg-sidebar text-sidebar-foreground',
29 className
30 )}
31 bind:this={ref}
32 {...restProps}
33 >
34 {@render children?.()}
35 </div>
36{:else if sidebar.isMobile}
37 <Sheet.Root bind:open={() => sidebar.openMobile, (v) => sidebar.setOpenMobile(v)} {...restProps}>
38 <Sheet.Content
39 data-sidebar="sidebar"
40 data-slot="sidebar"
41 data-mobile="true"
42 class="z-99999 w-(--sidebar-width) bg-sidebar p-0 text-sidebar-foreground sm:z-99 [&>button]:hidden"
43 style="--sidebar-width: {SIDEBAR_WIDTH_MOBILE};"
44 {side}
45 >
46 <Sheet.Header class="sr-only">
47 <Sheet.Title>Sidebar</Sheet.Title>
48 <Sheet.Description>Displays the mobile sidebar.</Sheet.Description>
49 </Sheet.Header>
50 <div class="flex h-full w-full flex-col">
51 {@render children?.()}
52 </div>
53 </Sheet.Content>
54 </Sheet.Root>
55{:else}
56 <div
57 bind:this={ref}
58 class="group peer hidden text-sidebar-foreground md:block"
59 data-state={sidebar.state}
60 data-collapsible={sidebar.state === 'collapsed' ? collapsible : ''}
61 data-variant={variant}
62 data-side={side}
63 data-slot="sidebar"
64 >
65 <!-- This is what handles the sidebar gap on desktop -->
66 <div
67 data-slot="sidebar-gap"
68 class={cn(
69 'relative w-(--sidebar-width) bg-transparent transition-[width] duration-200 ease-linear',
70 'group-data-[collapsible=offcanvas]:w-0',
71 'group-data-[side=right]:rotate-180',
72 variant === 'floating' || variant === 'inset'
73 ? 'group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]'
74 : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)'
75 )}
76 ></div>
77 <div
78 data-slot="sidebar-container"
79 class={cn(
80 'fixed inset-y-0 z-999 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:z-0 md:flex',
81 side === 'left'
82 ? 'left-0 group-data-[collapsible=offcanvas]:left-[calc(var(--sidebar-width)*-1)]'
83 : 'right-0 group-data-[collapsible=offcanvas]:right-[calc(var(--sidebar-width)*-1)]',
84 // Adjust the padding for floating and inset variants.
85 variant === 'floating' || variant === 'inset'
86 ? 'p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]'
87 : 'group-data-[collapsible=icon]:w-(--sidebar-width-icon)',
88 className
89 )}
90 {...restProps}
91 >
92 <div
93 data-sidebar="sidebar"
94 data-slot="sidebar-inner"
95 class="flex h-full w-full flex-col bg-sidebar group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:border-sidebar-border group-data-[variant=floating]:shadow-sm"
96 >
97 {@render children?.()}
98 </div>
99 </div>
100 </div>
101{/if}