@questpie/admin 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (203) hide show
  1. package/.turbo/turbo-build.log +108 -0
  2. package/CHANGELOG.md +10 -0
  3. package/README.md +556 -0
  4. package/STATUS.md +917 -0
  5. package/VALIDATION.md +602 -0
  6. package/components.json +24 -0
  7. package/dist/__tests__/setup.mjs +38 -0
  8. package/dist/__tests__/test-utils.mjs +45 -0
  9. package/dist/__tests__/vitest.d.mjs +3 -0
  10. package/dist/components/admin-app.mjs +69 -0
  11. package/dist/components/fields/array-field.mjs +190 -0
  12. package/dist/components/fields/checkbox-field.mjs +34 -0
  13. package/dist/components/fields/custom-field.mjs +32 -0
  14. package/dist/components/fields/date-field.mjs +41 -0
  15. package/dist/components/fields/datetime-field.mjs +42 -0
  16. package/dist/components/fields/email-field.mjs +37 -0
  17. package/dist/components/fields/embedded-collection.mjs +253 -0
  18. package/dist/components/fields/field-types.mjs +1 -0
  19. package/dist/components/fields/field-utils.mjs +10 -0
  20. package/dist/components/fields/field-wrapper.mjs +34 -0
  21. package/dist/components/fields/index.mjs +23 -0
  22. package/dist/components/fields/json-field.mjs +243 -0
  23. package/dist/components/fields/locale-badge.mjs +16 -0
  24. package/dist/components/fields/number-field.mjs +39 -0
  25. package/dist/components/fields/password-field.mjs +37 -0
  26. package/dist/components/fields/relation-field.mjs +104 -0
  27. package/dist/components/fields/relation-picker.mjs +229 -0
  28. package/dist/components/fields/relation-select.mjs +188 -0
  29. package/dist/components/fields/rich-text-editor/index.mjs +897 -0
  30. package/dist/components/fields/select-field.mjs +41 -0
  31. package/dist/components/fields/switch-field.mjs +34 -0
  32. package/dist/components/fields/text-field.mjs +38 -0
  33. package/dist/components/fields/textarea-field.mjs +38 -0
  34. package/dist/components/index.mjs +59 -0
  35. package/dist/components/primitives/checkbox-input.mjs +127 -0
  36. package/dist/components/primitives/date-input.mjs +303 -0
  37. package/dist/components/primitives/index.mjs +12 -0
  38. package/dist/components/primitives/number-input.mjs +104 -0
  39. package/dist/components/primitives/select-input.mjs +177 -0
  40. package/dist/components/primitives/tag-input.mjs +135 -0
  41. package/dist/components/primitives/text-input.mjs +39 -0
  42. package/dist/components/primitives/textarea-input.mjs +37 -0
  43. package/dist/components/primitives/toggle-input.mjs +31 -0
  44. package/dist/components/primitives/types.mjs +12 -0
  45. package/dist/components/ui/accordion.mjs +55 -0
  46. package/dist/components/ui/avatar.mjs +54 -0
  47. package/dist/components/ui/badge.mjs +34 -0
  48. package/dist/components/ui/button.mjs +48 -0
  49. package/dist/components/ui/card.mjs +58 -0
  50. package/dist/components/ui/checkbox.mjs +21 -0
  51. package/dist/components/ui/combobox.mjs +163 -0
  52. package/dist/components/ui/dialog.mjs +95 -0
  53. package/dist/components/ui/dropdown-menu.mjs +138 -0
  54. package/dist/components/ui/field.mjs +113 -0
  55. package/dist/components/ui/input-group.mjs +82 -0
  56. package/dist/components/ui/input.mjs +17 -0
  57. package/dist/components/ui/label.mjs +15 -0
  58. package/dist/components/ui/popover.mjs +56 -0
  59. package/dist/components/ui/scroll-area.mjs +38 -0
  60. package/dist/components/ui/select.mjs +100 -0
  61. package/dist/components/ui/separator.mjs +16 -0
  62. package/dist/components/ui/sheet.mjs +90 -0
  63. package/dist/components/ui/sidebar.mjs +387 -0
  64. package/dist/components/ui/skeleton.mjs +14 -0
  65. package/dist/components/ui/spinner.mjs +16 -0
  66. package/dist/components/ui/switch.mjs +22 -0
  67. package/dist/components/ui/table.mjs +68 -0
  68. package/dist/components/ui/tabs.mjs +48 -0
  69. package/dist/components/ui/textarea.mjs +15 -0
  70. package/dist/components/ui/tooltip.mjs +44 -0
  71. package/dist/config/component-registry.mjs +38 -0
  72. package/dist/config/index.mjs +129 -0
  73. package/dist/hooks/admin-provider.mjs +70 -0
  74. package/dist/hooks/index.mjs +7 -0
  75. package/dist/hooks/store.mjs +178 -0
  76. package/dist/hooks/use-auth.mjs +76 -0
  77. package/dist/hooks/use-collection-db.mjs +146 -0
  78. package/dist/hooks/use-collection.mjs +112 -0
  79. package/dist/hooks/use-global.mjs +46 -0
  80. package/dist/hooks/use-mobile.mjs +20 -0
  81. package/dist/lib/utils.mjs +10 -0
  82. package/dist/styles/index.css +336 -0
  83. package/dist/styles/index.mjs +1 -0
  84. package/dist/utils/index.mjs +9 -0
  85. package/dist/views/auth/auth-layout.mjs +52 -0
  86. package/dist/views/auth/forgot-password-form.mjs +148 -0
  87. package/dist/views/auth/index.mjs +6 -0
  88. package/dist/views/auth/login-form.mjs +156 -0
  89. package/dist/views/auth/reset-password-form.mjs +184 -0
  90. package/dist/views/collection/auto-form-fields.mjs +525 -0
  91. package/dist/views/collection/collection-form.mjs +91 -0
  92. package/dist/views/collection/collection-list.mjs +76 -0
  93. package/dist/views/collection/form-field.mjs +42 -0
  94. package/dist/views/collection/index.mjs +6 -0
  95. package/dist/views/common/index.mjs +4 -0
  96. package/dist/views/common/locale-switcher.mjs +39 -0
  97. package/dist/views/common/version-history.mjs +272 -0
  98. package/dist/views/index.mjs +9 -0
  99. package/dist/views/layout/admin-layout.mjs +40 -0
  100. package/dist/views/layout/admin-router.mjs +95 -0
  101. package/dist/views/layout/admin-sidebar.mjs +63 -0
  102. package/dist/views/layout/index.mjs +5 -0
  103. package/package.json +276 -0
  104. package/src/__tests__/setup.ts +44 -0
  105. package/src/__tests__/test-utils.tsx +49 -0
  106. package/src/__tests__/vitest.d.ts +9 -0
  107. package/src/components/admin-app.tsx +221 -0
  108. package/src/components/fields/array-field.tsx +237 -0
  109. package/src/components/fields/checkbox-field.tsx +47 -0
  110. package/src/components/fields/custom-field.tsx +50 -0
  111. package/src/components/fields/date-field.tsx +65 -0
  112. package/src/components/fields/datetime-field.tsx +67 -0
  113. package/src/components/fields/email-field.tsx +51 -0
  114. package/src/components/fields/embedded-collection.tsx +315 -0
  115. package/src/components/fields/field-types.ts +162 -0
  116. package/src/components/fields/field-utils.ts +6 -0
  117. package/src/components/fields/field-wrapper.tsx +52 -0
  118. package/src/components/fields/index.ts +66 -0
  119. package/src/components/fields/json-field.tsx +440 -0
  120. package/src/components/fields/locale-badge.tsx +15 -0
  121. package/src/components/fields/number-field.tsx +57 -0
  122. package/src/components/fields/password-field.tsx +51 -0
  123. package/src/components/fields/relation-field.tsx +243 -0
  124. package/src/components/fields/relation-picker.tsx +402 -0
  125. package/src/components/fields/relation-select.tsx +327 -0
  126. package/src/components/fields/rich-text-editor/index.tsx +1337 -0
  127. package/src/components/fields/select-field.tsx +61 -0
  128. package/src/components/fields/switch-field.tsx +47 -0
  129. package/src/components/fields/text-field.tsx +55 -0
  130. package/src/components/fields/textarea-field.tsx +55 -0
  131. package/src/components/index.ts +40 -0
  132. package/src/components/primitives/checkbox-input.tsx +193 -0
  133. package/src/components/primitives/date-input.tsx +401 -0
  134. package/src/components/primitives/index.ts +24 -0
  135. package/src/components/primitives/number-input.tsx +132 -0
  136. package/src/components/primitives/select-input.tsx +296 -0
  137. package/src/components/primitives/tag-input.tsx +200 -0
  138. package/src/components/primitives/text-input.tsx +49 -0
  139. package/src/components/primitives/textarea-input.tsx +46 -0
  140. package/src/components/primitives/toggle-input.tsx +36 -0
  141. package/src/components/primitives/types.ts +235 -0
  142. package/src/components/ui/accordion.tsx +72 -0
  143. package/src/components/ui/avatar.tsx +106 -0
  144. package/src/components/ui/badge.tsx +48 -0
  145. package/src/components/ui/button.tsx +53 -0
  146. package/src/components/ui/card.tsx +94 -0
  147. package/src/components/ui/checkbox.tsx +27 -0
  148. package/src/components/ui/combobox.tsx +290 -0
  149. package/src/components/ui/dialog.tsx +151 -0
  150. package/src/components/ui/dropdown-menu.tsx +254 -0
  151. package/src/components/ui/field.tsx +227 -0
  152. package/src/components/ui/input-group.tsx +149 -0
  153. package/src/components/ui/input.tsx +20 -0
  154. package/src/components/ui/label.tsx +18 -0
  155. package/src/components/ui/popover.tsx +88 -0
  156. package/src/components/ui/scroll-area.tsx +53 -0
  157. package/src/components/ui/select.tsx +192 -0
  158. package/src/components/ui/separator.tsx +23 -0
  159. package/src/components/ui/sheet.tsx +127 -0
  160. package/src/components/ui/sidebar.tsx +723 -0
  161. package/src/components/ui/skeleton.tsx +13 -0
  162. package/src/components/ui/spinner.tsx +10 -0
  163. package/src/components/ui/switch.tsx +32 -0
  164. package/src/components/ui/table.tsx +99 -0
  165. package/src/components/ui/tabs.tsx +82 -0
  166. package/src/components/ui/textarea.tsx +18 -0
  167. package/src/components/ui/tooltip.tsx +70 -0
  168. package/src/config/component-registry.ts +190 -0
  169. package/src/config/index.ts +1099 -0
  170. package/src/hooks/README.md +269 -0
  171. package/src/hooks/admin-provider.tsx +110 -0
  172. package/src/hooks/index.ts +41 -0
  173. package/src/hooks/store.ts +248 -0
  174. package/src/hooks/use-auth.ts +168 -0
  175. package/src/hooks/use-collection-db.ts +209 -0
  176. package/src/hooks/use-collection.ts +156 -0
  177. package/src/hooks/use-global.ts +69 -0
  178. package/src/hooks/use-mobile.ts +21 -0
  179. package/src/lib/utils.ts +6 -0
  180. package/src/styles/index.css +340 -0
  181. package/src/utils/index.ts +6 -0
  182. package/src/views/auth/auth-layout.tsx +77 -0
  183. package/src/views/auth/forgot-password-form.tsx +192 -0
  184. package/src/views/auth/index.ts +21 -0
  185. package/src/views/auth/login-form.tsx +229 -0
  186. package/src/views/auth/reset-password-form.tsx +232 -0
  187. package/src/views/collection/auto-form-fields.tsx +982 -0
  188. package/src/views/collection/collection-form.tsx +186 -0
  189. package/src/views/collection/collection-list.tsx +223 -0
  190. package/src/views/collection/form-field.tsx +52 -0
  191. package/src/views/collection/index.ts +15 -0
  192. package/src/views/common/index.ts +8 -0
  193. package/src/views/common/locale-switcher.tsx +45 -0
  194. package/src/views/common/version-history.tsx +406 -0
  195. package/src/views/index.ts +25 -0
  196. package/src/views/layout/admin-layout.tsx +117 -0
  197. package/src/views/layout/admin-router.tsx +206 -0
  198. package/src/views/layout/admin-sidebar.tsx +185 -0
  199. package/src/views/layout/index.ts +12 -0
  200. package/tsconfig.json +13 -0
  201. package/tsconfig.tsbuildinfo +1 -0
  202. package/tsdown.config.ts +13 -0
  203. package/vitest.config.ts +29 -0
@@ -0,0 +1,723 @@
1
+ "use client"
2
+
3
+ import * as React from "react"
4
+ import { mergeProps } from "@base-ui/react/merge-props"
5
+ import { useRender } from "@base-ui/react/use-render"
6
+ import { cva, type VariantProps } from "class-variance-authority"
7
+
8
+ import { cn } from "../../lib/utils"
9
+ import { Button } from "./button"
10
+ import { Input } from "./input"
11
+ import { Separator } from "./separator"
12
+ import {
13
+ Sheet,
14
+ SheetContent,
15
+ SheetDescription,
16
+ SheetHeader,
17
+ SheetTitle,
18
+ } from "./sheet"
19
+ import { Skeleton } from "./skeleton"
20
+ import {
21
+ Tooltip,
22
+ TooltipContent,
23
+ TooltipTrigger,
24
+ } from "./tooltip"
25
+ import { useIsMobile } from "../../hooks/use-mobile"
26
+ import { SidebarIcon } from "@phosphor-icons/react"
27
+
28
+ const SIDEBAR_COOKIE_NAME = "sidebar_state"
29
+ const SIDEBAR_COOKIE_MAX_AGE = 60 * 60 * 24 * 7
30
+ const SIDEBAR_WIDTH = "16rem"
31
+ const SIDEBAR_WIDTH_MOBILE = "18rem"
32
+ const SIDEBAR_WIDTH_ICON = "3rem"
33
+ const SIDEBAR_KEYBOARD_SHORTCUT = "b"
34
+
35
+ type SidebarContextProps = {
36
+ state: "expanded" | "collapsed"
37
+ open: boolean
38
+ setOpen: (open: boolean) => void
39
+ openMobile: boolean
40
+ setOpenMobile: (open: boolean) => void
41
+ isMobile: boolean
42
+ toggleSidebar: () => void
43
+ }
44
+
45
+ const SidebarContext = React.createContext<SidebarContextProps | null>(null)
46
+
47
+ function useSidebar() {
48
+ const context = React.useContext(SidebarContext)
49
+ if (!context) {
50
+ throw new Error("useSidebar must be used within a SidebarProvider.")
51
+ }
52
+
53
+ return context
54
+ }
55
+
56
+ function SidebarProvider({
57
+ defaultOpen = true,
58
+ open: openProp,
59
+ onOpenChange: setOpenProp,
60
+ className,
61
+ style,
62
+ children,
63
+ ...props
64
+ }: React.ComponentProps<"div"> & {
65
+ defaultOpen?: boolean
66
+ open?: boolean
67
+ onOpenChange?: (open: boolean) => void
68
+ }) {
69
+ const isMobile = useIsMobile()
70
+ const [openMobile, setOpenMobile] = React.useState(false)
71
+
72
+ // This is the internal state of the sidebar.
73
+ // We use openProp and setOpenProp for control from outside the component.
74
+ const [_open, _setOpen] = React.useState(defaultOpen)
75
+ const open = openProp ?? _open
76
+ const setOpen = React.useCallback(
77
+ (value: boolean | ((value: boolean) => boolean)) => {
78
+ const openState = typeof value === "function" ? value(open) : value
79
+ if (setOpenProp) {
80
+ setOpenProp(openState)
81
+ } else {
82
+ _setOpen(openState)
83
+ }
84
+
85
+ // This sets the cookie to keep the sidebar state.
86
+ document.cookie = `${SIDEBAR_COOKIE_NAME}=${openState}; path=/; max-age=${SIDEBAR_COOKIE_MAX_AGE}`
87
+ },
88
+ [setOpenProp, open]
89
+ )
90
+
91
+ // Helper to toggle the sidebar.
92
+ const toggleSidebar = React.useCallback(() => {
93
+ return isMobile ? setOpenMobile((open) => !open) : setOpen((open) => !open)
94
+ }, [isMobile, setOpen, setOpenMobile])
95
+
96
+ // Adds a keyboard shortcut to toggle the sidebar.
97
+ React.useEffect(() => {
98
+ const handleKeyDown = (event: KeyboardEvent) => {
99
+ if (
100
+ event.key === SIDEBAR_KEYBOARD_SHORTCUT &&
101
+ (event.metaKey || event.ctrlKey)
102
+ ) {
103
+ event.preventDefault()
104
+ toggleSidebar()
105
+ }
106
+ }
107
+
108
+ window.addEventListener("keydown", handleKeyDown)
109
+ return () => window.removeEventListener("keydown", handleKeyDown)
110
+ }, [toggleSidebar])
111
+
112
+ // We add a state so that we can do data-state="expanded" or "collapsed".
113
+ // This makes it easier to style the sidebar with Tailwind classes.
114
+ const state = open ? "expanded" : "collapsed"
115
+
116
+ const contextValue = React.useMemo<SidebarContextProps>(
117
+ () => ({
118
+ state,
119
+ open,
120
+ setOpen,
121
+ isMobile,
122
+ openMobile,
123
+ setOpenMobile,
124
+ toggleSidebar,
125
+ }),
126
+ [state, open, setOpen, isMobile, openMobile, setOpenMobile, toggleSidebar]
127
+ )
128
+
129
+ return (
130
+ <SidebarContext.Provider value={contextValue}>
131
+ <div
132
+ data-slot="sidebar-wrapper"
133
+ style={
134
+ {
135
+ "--sidebar-width": SIDEBAR_WIDTH,
136
+ "--sidebar-width-icon": SIDEBAR_WIDTH_ICON,
137
+ ...style,
138
+ } as React.CSSProperties
139
+ }
140
+ className={cn(
141
+ "group/sidebar-wrapper has-data-[variant=inset]:bg-sidebar flex min-h-svh w-full",
142
+ className
143
+ )}
144
+ {...props}
145
+ >
146
+ {children}
147
+ </div>
148
+ </SidebarContext.Provider>
149
+ )
150
+ }
151
+
152
+ function Sidebar({
153
+ side = "left",
154
+ variant = "sidebar",
155
+ collapsible = "offExamples",
156
+ className,
157
+ children,
158
+ ...props
159
+ }: React.ComponentProps<"div"> & {
160
+ side?: "left" | "right"
161
+ variant?: "sidebar" | "floating" | "inset"
162
+ collapsible?: "offExamples" | "icon" | "none"
163
+ }) {
164
+ const { isMobile, state, openMobile, setOpenMobile } = useSidebar()
165
+
166
+ if (collapsible === "none") {
167
+ return (
168
+ <div
169
+ data-slot="sidebar"
170
+ className={cn(
171
+ "bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col",
172
+ className
173
+ )}
174
+ {...props}
175
+ >
176
+ {children}
177
+ </div>
178
+ )
179
+ }
180
+
181
+ if (isMobile) {
182
+ return (
183
+ <Sheet open={openMobile} onOpenChange={setOpenMobile} {...props}>
184
+ <SheetContent
185
+ data-sidebar="sidebar"
186
+ data-slot="sidebar"
187
+ data-mobile="true"
188
+ className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
189
+ style={
190
+ {
191
+ "--sidebar-width": SIDEBAR_WIDTH_MOBILE,
192
+ } as React.CSSProperties
193
+ }
194
+ side={side}
195
+ >
196
+ <SheetHeader className="sr-only">
197
+ <SheetTitle>Sidebar</SheetTitle>
198
+ <SheetDescription>Displays the mobile sidebar.</SheetDescription>
199
+ </SheetHeader>
200
+ <div className="flex h-full w-full flex-col">{children}</div>
201
+ </SheetContent>
202
+ </Sheet>
203
+ )
204
+ }
205
+
206
+ return (
207
+ <div
208
+ className="group peer text-sidebar-foreground hidden md:block"
209
+ data-state={state}
210
+ data-collapsible={state === "collapsed" ? collapsible : ""}
211
+ data-variant={variant}
212
+ data-side={side}
213
+ data-slot="sidebar"
214
+ >
215
+ {/* This is what handles the sidebar gap on desktop */}
216
+ <div
217
+ data-slot="sidebar-gap"
218
+ className={cn(
219
+ "transition-[width] duration-200 ease-linear relative w-(--sidebar-width) bg-transparent",
220
+ "group-data-[collapsible=offExamples]:w-0",
221
+ "group-data-[side=right]:rotate-180",
222
+ variant === "floating" || variant === "inset"
223
+ ? "group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4)))]"
224
+ : "group-data-[collapsible=icon]:w-(--sidebar-width-icon)"
225
+ )}
226
+ />
227
+ <div
228
+ data-slot="sidebar-container"
229
+ className={cn(
230
+ "fixed inset-y-0 z-10 hidden h-svh w-(--sidebar-width) transition-[left,right,width] duration-200 ease-linear md:flex",
231
+ side === "left"
232
+ ? "left-0 group-data-[collapsible=offExamples]:left-[calc(var(--sidebar-width)*-1)]"
233
+ : "right-0 group-data-[collapsible=offExamples]:right-[calc(var(--sidebar-width)*-1)]",
234
+ // Adjust the padding for floating and inset variants.
235
+ variant === "floating" || variant === "inset"
236
+ ? "p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]"
237
+ : "group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l",
238
+ className
239
+ )}
240
+ {...props}
241
+ >
242
+ <div
243
+ data-sidebar="sidebar"
244
+ data-slot="sidebar-inner"
245
+ className="bg-sidebar group-data-[variant=floating]:ring-sidebar-border group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:shadow-sm group-data-[variant=floating]:ring-1 flex size-full flex-col"
246
+ >
247
+ {children}
248
+ </div>
249
+ </div>
250
+ </div>
251
+ )
252
+ }
253
+
254
+ function SidebarTrigger({
255
+ className,
256
+ onClick,
257
+ ...props
258
+ }: React.ComponentProps<typeof Button>) {
259
+ const { toggleSidebar } = useSidebar()
260
+
261
+ return (
262
+ <Button
263
+ data-sidebar="trigger"
264
+ data-slot="sidebar-trigger"
265
+ variant="ghost"
266
+ size="icon-sm"
267
+ className={cn(className)}
268
+ onClick={(event) => {
269
+ onClick?.(event)
270
+ toggleSidebar()
271
+ }}
272
+ {...props}
273
+ >
274
+ <SidebarIcon
275
+ />
276
+ <span className="sr-only">Toggle Sidebar</span>
277
+ </Button>
278
+ )
279
+ }
280
+
281
+ function SidebarRail({ className, ...props }: React.ComponentProps<"button">) {
282
+ const { toggleSidebar } = useSidebar()
283
+
284
+ return (
285
+ <button
286
+ data-sidebar="rail"
287
+ data-slot="sidebar-rail"
288
+ aria-label="Toggle Sidebar"
289
+ tabIndex={-1}
290
+ onClick={toggleSidebar}
291
+ title="Toggle Sidebar"
292
+ className={cn(
293
+ "hover:after:bg-sidebar-border absolute inset-y-0 z-20 hidden w-4 -translate-x-1/2 transition-all ease-linear group-data-[side=left]:-right-4 group-data-[side=right]:left-0 after:absolute after:inset-y-0 after:left-1/2 after:w-[2px] sm:flex",
294
+ "in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize",
295
+ "[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize",
296
+ "hover:group-data-[collapsible=offExamples]:bg-sidebar group-data-[collapsible=offExamples]:translate-x-0 group-data-[collapsible=offExamples]:after:left-full",
297
+ "[[data-side=left][data-collapsible=offExamples]_&]:-right-2",
298
+ "[[data-side=right][data-collapsible=offExamples]_&]:-left-2",
299
+ className
300
+ )}
301
+ {...props}
302
+ />
303
+ )
304
+ }
305
+
306
+ function SidebarInset({ className, ...props }: React.ComponentProps<"main">) {
307
+ return (
308
+ <main
309
+ data-slot="sidebar-inset"
310
+ className={cn(
311
+ "bg-background md:peer-data-[variant=inset]:m-2 md:peer-data-[variant=inset]:ml-0 md:peer-data-[variant=inset]:rounded-xl md:peer-data-[variant=inset]:shadow-sm md:peer-data-[variant=inset]:peer-data-[state=collapsed]:ml-2 relative flex w-full flex-1 flex-col",
312
+ className
313
+ )}
314
+ {...props}
315
+ />
316
+ )
317
+ }
318
+
319
+ function SidebarInput({
320
+ className,
321
+ ...props
322
+ }: React.ComponentProps<typeof Input>) {
323
+ return (
324
+ <Input
325
+ data-slot="sidebar-input"
326
+ data-sidebar="input"
327
+ className={cn("bg-muted/20 dark:bg-muted/30 border-input h-8 w-full", className)}
328
+ {...props}
329
+ />
330
+ )
331
+ }
332
+
333
+ function SidebarHeader({ className, ...props }: React.ComponentProps<"div">) {
334
+ return (
335
+ <div
336
+ data-slot="sidebar-header"
337
+ data-sidebar="header"
338
+ className={cn("gap-2 p-2 flex flex-col", className)}
339
+ {...props}
340
+ />
341
+ )
342
+ }
343
+
344
+ function SidebarFooter({ className, ...props }: React.ComponentProps<"div">) {
345
+ return (
346
+ <div
347
+ data-slot="sidebar-footer"
348
+ data-sidebar="footer"
349
+ className={cn("gap-2 p-2 flex flex-col", className)}
350
+ {...props}
351
+ />
352
+ )
353
+ }
354
+
355
+ function SidebarSeparator({
356
+ className,
357
+ ...props
358
+ }: React.ComponentProps<typeof Separator>) {
359
+ return (
360
+ <Separator
361
+ data-slot="sidebar-separator"
362
+ data-sidebar="separator"
363
+ className={cn("bg-sidebar-border mx-2 w-auto", className)}
364
+ {...props}
365
+ />
366
+ )
367
+ }
368
+
369
+ function SidebarContent({ className, ...props }: React.ComponentProps<"div">) {
370
+ return (
371
+ <div
372
+ data-slot="sidebar-content"
373
+ data-sidebar="content"
374
+ className={cn(
375
+ "no-scrollbar gap-0 flex min-h-0 flex-1 flex-col overflow-auto group-data-[collapsible=icon]:overflow-hidden",
376
+ className
377
+ )}
378
+ {...props}
379
+ />
380
+ )
381
+ }
382
+
383
+ function SidebarGroup({ className, ...props }: React.ComponentProps<"div">) {
384
+ return (
385
+ <div
386
+ data-slot="sidebar-group"
387
+ data-sidebar="group"
388
+ className={cn(
389
+ "px-2 py-1 relative flex w-full min-w-0 flex-col",
390
+ className
391
+ )}
392
+ {...props}
393
+ />
394
+ )
395
+ }
396
+
397
+ function SidebarGroupLabel({
398
+ className,
399
+ render,
400
+ ...props
401
+ }: useRender.ComponentProps<"div"> & React.ComponentProps<"div">) {
402
+ return useRender({
403
+ defaultTagName: "div",
404
+ props: mergeProps<"div">(
405
+ {
406
+ className: cn(
407
+ "text-sidebar-foreground/70 ring-sidebar-ring h-8 rounded-md px-2 text-xs transition-[margin,opacity] duration-200 ease-linear group-data-[collapsible=icon]:-mt-8 group-data-[collapsible=icon]:opacity-0 focus-visible:ring-2 [&>svg]:size-4 flex shrink-0 items-center outline-hidden [&>svg]:shrink-0",
408
+ className
409
+ ),
410
+ },
411
+ props
412
+ ),
413
+ render,
414
+ state: {
415
+ slot: "sidebar-group-label",
416
+ sidebar: "group-label",
417
+ },
418
+ })
419
+ }
420
+
421
+ function SidebarGroupAction({
422
+ className,
423
+ render,
424
+ ...props
425
+ }: useRender.ComponentProps<"button"> & React.ComponentProps<"button">) {
426
+ return useRender({
427
+ defaultTagName: "button",
428
+ props: mergeProps<"button">(
429
+ {
430
+ className: cn(
431
+ "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground absolute top-3.5 right-3 w-5 rounded-md p-0 focus-visible:ring-2 [&>svg]:size-4 flex aspect-square items-center justify-center outline-hidden transition-transform [&>svg]:shrink-0 after:absolute after:-inset-2 md:after:hidden group-data-[collapsible=icon]:hidden",
432
+ className
433
+ ),
434
+ },
435
+ props
436
+ ),
437
+ render,
438
+ state: {
439
+ slot: "sidebar-group-action",
440
+ sidebar: "group-action",
441
+ },
442
+ })
443
+ }
444
+
445
+ function SidebarGroupContent({
446
+ className,
447
+ ...props
448
+ }: React.ComponentProps<"div">) {
449
+ return (
450
+ <div
451
+ data-slot="sidebar-group-content"
452
+ data-sidebar="group-content"
453
+ className={cn("text-xs w-full", className)}
454
+ {...props}
455
+ />
456
+ )
457
+ }
458
+
459
+ function SidebarMenu({ className, ...props }: React.ComponentProps<"ul">) {
460
+ return (
461
+ <ul
462
+ data-slot="sidebar-menu"
463
+ data-sidebar="menu"
464
+ className={cn("gap-px flex w-full min-w-0 flex-col", className)}
465
+ {...props}
466
+ />
467
+ )
468
+ }
469
+
470
+ function SidebarMenuItem({ className, ...props }: React.ComponentProps<"li">) {
471
+ return (
472
+ <li
473
+ data-slot="sidebar-menu-item"
474
+ data-sidebar="menu-item"
475
+ className={cn("group/menu-item relative", className)}
476
+ {...props}
477
+ />
478
+ )
479
+ }
480
+
481
+ const sidebarMenuButtonVariants = cva(
482
+ "ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground data-active:bg-sidebar-accent data-active:text-sidebar-accent-foreground data-open:hover:bg-sidebar-accent data-open:hover:text-sidebar-accent-foreground gap-2 rounded-[calc(var(--radius-sm)+2px)] p-2 text-left text-xs transition-[width,height,padding] group-has-data-[sidebar=menu-action]/menu-item:pr-8 group-data-[collapsible=icon]:size-8! group-data-[collapsible=icon]:p-2! focus-visible:ring-2 data-active:font-medium peer/menu-button flex w-full items-center overflow-hidden outline-hidden group/menu-button disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&_svg]:size-4 [&_svg]:shrink-0",
483
+ {
484
+ variants: {
485
+ variant: {
486
+ default: "hover:bg-sidebar-accent hover:text-sidebar-accent-foreground",
487
+ outline: "bg-background hover:bg-sidebar-accent hover:text-sidebar-accent-foreground shadow-[0_0_0_1px_hsl(var(--sidebar-border))] hover:shadow-[0_0_0_1px_hsl(var(--sidebar-accent))]",
488
+ },
489
+ size: {
490
+ default: "h-8 text-xs",
491
+ sm: "h-7 text-xs",
492
+ lg: "h-12 text-xs group-data-[collapsible=icon]:p-0!",
493
+ },
494
+ },
495
+ defaultVariants: {
496
+ variant: "default",
497
+ size: "default",
498
+ },
499
+ }
500
+ )
501
+
502
+ function SidebarMenuButton({
503
+ render,
504
+ isActive = false,
505
+ variant = "default",
506
+ size = "default",
507
+ tooltip,
508
+ className,
509
+ ...props
510
+ }: useRender.ComponentProps<"button"> &
511
+ React.ComponentProps<"button"> & {
512
+ isActive?: boolean
513
+ tooltip?: string | React.ComponentProps<typeof TooltipContent>
514
+ } & VariantProps<typeof sidebarMenuButtonVariants>) {
515
+ const { isMobile, state } = useSidebar()
516
+ const comp = useRender({
517
+ defaultTagName: "button",
518
+ props: mergeProps<"button">(
519
+ {
520
+ className: cn(sidebarMenuButtonVariants({ variant, size }), className),
521
+ },
522
+ props
523
+ ),
524
+ render: !tooltip ? render : TooltipTrigger,
525
+ state: {
526
+ slot: "sidebar-menu-button",
527
+ sidebar: "menu-button",
528
+ size,
529
+ active: isActive,
530
+ },
531
+ })
532
+
533
+ if (!tooltip) {
534
+ return comp
535
+ }
536
+
537
+ if (typeof tooltip === "string") {
538
+ tooltip = {
539
+ children: tooltip,
540
+ }
541
+ }
542
+
543
+ return (
544
+ <Tooltip>
545
+ {comp}
546
+ <TooltipContent
547
+ side="right"
548
+ align="center"
549
+ hidden={state !== "collapsed" || isMobile}
550
+ {...tooltip}
551
+ />
552
+ </Tooltip>
553
+ )
554
+ }
555
+
556
+ function SidebarMenuAction({
557
+ className,
558
+ render,
559
+ showOnHover = false,
560
+ ...props
561
+ }: useRender.ComponentProps<"button"> &
562
+ React.ComponentProps<"button"> & {
563
+ showOnHover?: boolean
564
+ }) {
565
+ return useRender({
566
+ defaultTagName: "button",
567
+ props: mergeProps<"button">(
568
+ {
569
+ className: cn(
570
+ "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground peer-hover/menu-button:text-sidebar-accent-foreground absolute top-1.5 right-1 aspect-square w-5 rounded-[calc(var(--radius-sm)-2px)] p-0 peer-data-[size=default]/menu-button:top-1.5 peer-data-[size=lg]/menu-button:top-2.5 peer-data-[size=sm]/menu-button:top-1 focus-visible:ring-2 [&>svg]:size-4 flex items-center justify-center outline-hidden transition-transform group-data-[collapsible=icon]:hidden after:absolute after:-inset-2 md:after:hidden [&>svg]:shrink-0",
571
+ showOnHover &&
572
+ "peer-data-active/menu-button:text-sidebar-accent-foreground group-focus-within/menu-item:opacity-100 group-hover/menu-item:opacity-100 data-open:opacity-100 md:opacity-0",
573
+ className
574
+ ),
575
+ },
576
+ props
577
+ ),
578
+ render,
579
+ state: {
580
+ slot: "sidebar-menu-action",
581
+ sidebar: "menu-action",
582
+ },
583
+ })
584
+ }
585
+
586
+ function SidebarMenuBadge({
587
+ className,
588
+ ...props
589
+ }: React.ComponentProps<"div">) {
590
+ return (
591
+ <div
592
+ data-slot="sidebar-menu-badge"
593
+ data-sidebar="menu-badge"
594
+ className={cn(
595
+ "text-sidebar-foreground peer-hover/menu-button:text-sidebar-accent-foreground peer-data-active/menu-button:text-sidebar-accent-foreground pointer-events-none absolute right-1 flex h-5 min-w-5 rounded-[calc(var(--radius-sm)-2px)] px-1 text-xs font-medium peer-data-[size=default]/menu-button:top-1.5 peer-data-[size=lg]/menu-button:top-2.5 peer-data-[size=sm]/menu-button:top-1 flex items-center justify-center tabular-nums select-none group-data-[collapsible=icon]:hidden",
596
+ className
597
+ )}
598
+ {...props}
599
+ />
600
+ )
601
+ }
602
+
603
+ function SidebarMenuSkeleton({
604
+ className,
605
+ showIcon = false,
606
+ ...props
607
+ }: React.ComponentProps<"div"> & {
608
+ showIcon?: boolean
609
+ }) {
610
+ // Random width between 50 to 90%.
611
+ const [width] = React.useState(() => {
612
+ return `${Math.floor(Math.random() * 40) + 50}%`
613
+ })
614
+
615
+ return (
616
+ <div
617
+ data-slot="sidebar-menu-skeleton"
618
+ data-sidebar="menu-skeleton"
619
+ className={cn("h-8 gap-2 rounded-md px-2 flex items-center", className)}
620
+ {...props}
621
+ >
622
+ {showIcon && (
623
+ <Skeleton
624
+ className="size-4 rounded-md"
625
+ data-sidebar="menu-skeleton-icon"
626
+ />
627
+ )}
628
+ <Skeleton
629
+ className="h-4 max-w-(--skeleton-width) flex-1"
630
+ data-sidebar="menu-skeleton-text"
631
+ style={
632
+ {
633
+ "--skeleton-width": width,
634
+ } as React.CSSProperties
635
+ }
636
+ />
637
+ </div>
638
+ )
639
+ }
640
+
641
+ function SidebarMenuSub({ className, ...props }: React.ComponentProps<"ul">) {
642
+ return (
643
+ <ul
644
+ data-slot="sidebar-menu-sub"
645
+ data-sidebar="menu-sub"
646
+ className={cn("border-sidebar-border mx-3.5 translate-x-px gap-1 border-l px-2.5 py-0.5 group-data-[collapsible=icon]:hidden flex min-w-0 flex-col", className)}
647
+ {...props}
648
+ />
649
+ )
650
+ }
651
+
652
+ function SidebarMenuSubItem({
653
+ className,
654
+ ...props
655
+ }: React.ComponentProps<"li">) {
656
+ return (
657
+ <li
658
+ data-slot="sidebar-menu-sub-item"
659
+ data-sidebar="menu-sub-item"
660
+ className={cn("group/menu-sub-item relative", className)}
661
+ {...props}
662
+ />
663
+ )
664
+ }
665
+
666
+ function SidebarMenuSubButton({
667
+ render,
668
+ size = "md",
669
+ isActive = false,
670
+ className,
671
+ ...props
672
+ }: useRender.ComponentProps<"a"> &
673
+ React.ComponentProps<"a"> & {
674
+ size?: "sm" | "md"
675
+ isActive?: boolean
676
+ }) {
677
+ return useRender({
678
+ defaultTagName: "a",
679
+ props: mergeProps<"a">(
680
+ {
681
+ className: cn(
682
+ "text-sidebar-foreground ring-sidebar-ring hover:bg-sidebar-accent hover:text-sidebar-accent-foreground active:bg-sidebar-accent active:text-sidebar-accent-foreground [&>svg]:text-sidebar-accent-foreground data-active:bg-sidebar-accent data-active:text-sidebar-accent-foreground h-7 gap-2 rounded-md px-2 focus-visible:ring-2 data-[size=md]:text-xs data-[size=sm]:text-xs [&>svg]:size-4 flex min-w-0 -translate-x-px items-center overflow-hidden outline-hidden group-data-[collapsible=icon]:hidden disabled:pointer-events-none disabled:opacity-50 aria-disabled:pointer-events-none aria-disabled:opacity-50 [&>span:last-child]:truncate [&>svg]:shrink-0",
683
+ className
684
+ ),
685
+ },
686
+ props
687
+ ),
688
+ render,
689
+ state: {
690
+ slot: "sidebar-menu-sub-button",
691
+ sidebar: "menu-sub-button",
692
+ size,
693
+ active: isActive,
694
+ },
695
+ })
696
+ }
697
+
698
+ export {
699
+ Sidebar,
700
+ SidebarContent,
701
+ SidebarFooter,
702
+ SidebarGroup,
703
+ SidebarGroupAction,
704
+ SidebarGroupContent,
705
+ SidebarGroupLabel,
706
+ SidebarHeader,
707
+ SidebarInput,
708
+ SidebarInset,
709
+ SidebarMenu,
710
+ SidebarMenuAction,
711
+ SidebarMenuBadge,
712
+ SidebarMenuButton,
713
+ SidebarMenuItem,
714
+ SidebarMenuSkeleton,
715
+ SidebarMenuSub,
716
+ SidebarMenuSubButton,
717
+ SidebarMenuSubItem,
718
+ SidebarProvider,
719
+ SidebarRail,
720
+ SidebarSeparator,
721
+ SidebarTrigger,
722
+ useSidebar,
723
+ }