-
Notifications
You must be signed in to change notification settings - Fork 302
Expand file tree
/
Copy pathSidebar.vue
More file actions
78 lines (71 loc) · 2.33 KB
/
Sidebar.vue
File metadata and controls
78 lines (71 loc) · 2.33 KB
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
<template>
<div
class="flex h-full flex-col flex-shrink-0 overflow-y-auto overflow-x-hidden border-r border-outline-gray-1 bg-surface-menu-bar transition-all duration-300 ease-in-out p-2"
:class="shouldCollapse ? 'w-12' : 'w-60'"
>
<slot name="header">
<SidebarHeader
v-if="props.header"
:isCollapsed="shouldCollapse"
:title="props.header.title"
:subtitle="props.header.subtitle"
:logo="props.header.logo"
:menu-items="props.header.menuItems"
>
<template #logo>
<slot name="header-logo"></slot>
</template>
</SidebarHeader>
</slot>
<SidebarSection
v-for="section in props.sections"
:key="section.label"
:label="section.label"
:items="section.items"
:collapsible="section.collapsible"
>
<template #sidebar-item="{ item, isCollapsed }"
><slot name="sidebar-item" :item :isCollapsed></slot
></template>
</SidebarSection>
<div class="mt-auto flex flex-col gap-2">
<slot
name="footer-items"
v-bind="{ isCollapsed: shouldCollapse, isMobile }"
/>
<SidebarItem
v-if="!props.disableCollapse"
:label="shouldCollapse ? 'Expand' : 'Collapse'"
:isCollapsed="shouldCollapse"
@click="isCollapsed = !isCollapsed"
>
<template #icon>
<LucidePanelRightOpen
class="size-4 text-ink-gray-6 duration-300 ease-in-out"
:class="{ 'rotate-180': shouldCollapse }"
/>
</template>
</SidebarItem>
</div>
</div>
</template>
<script setup lang="ts">
import { breakpointsTailwind, useBreakpoints } from '@vueuse/core'
import { provide, computed } from 'vue'
import SidebarHeader from './SidebarHeader.vue'
import SidebarItem from './SidebarItem.vue'
import { SidebarProps } from './types'
import LucidePanelRightOpen from '~icons/lucide/panel-right-open'
import SidebarSection from './SidebarSection.vue'
const props = defineProps<SidebarProps>()
const isCollapsed = defineModel('collapsed', {
type: Boolean,
default: null,
})
provide('isSidebarCollapsed', isCollapsed)
const shouldCollapse = computed(
() => (isCollapsed.value ?? isMobile.value) && !props.disableCollapse,
)
const breakpoints = useBreakpoints(breakpointsTailwind)
const isMobile = breakpoints.smaller('sm')
</script>