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}