@snow-labs/brutal-ui 0.1.1 → 0.2.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.
- package/dist/components/brutal/cta-section.d.ts +7 -2
- package/dist/components/brutal/cta-section.js +136 -32
- package/dist/components/brutal/cta-section.js.map +1 -1
- package/dist/components/brutal/faq.d.ts +16 -0
- package/dist/components/brutal/faq.js +163 -0
- package/dist/components/brutal/faq.js.map +1 -0
- package/dist/components/brutal/feature-grid.d.ts +6 -2
- package/dist/components/brutal/feature-grid.js +73 -10
- package/dist/components/brutal/feature-grid.js.map +1 -1
- package/dist/components/brutal/footer.d.ts +14 -1
- package/dist/components/brutal/footer.js +181 -18
- package/dist/components/brutal/footer.js.map +1 -1
- package/dist/components/brutal/hero.d.ts +9 -2
- package/dist/components/brutal/hero.js +67 -22
- package/dist/components/brutal/hero.js.map +1 -1
- package/dist/components/brutal/index.d.ts +7 -2
- package/dist/components/brutal/index.js +1035 -128
- package/dist/components/brutal/index.js.map +1 -1
- package/dist/components/brutal/integration-grid.d.ts +1 -1
- package/dist/components/brutal/integration-grid.js +13 -5
- package/dist/components/brutal/integration-grid.js.map +1 -1
- package/dist/components/brutal/logo-cloud.d.ts +17 -0
- package/dist/components/brutal/logo-cloud.js +93 -0
- package/dist/components/brutal/logo-cloud.js.map +1 -0
- package/dist/components/brutal/nav.d.ts +3 -1
- package/dist/components/brutal/nav.js +166 -9
- package/dist/components/brutal/nav.js.map +1 -1
- package/dist/components/brutal/newsletter.d.ts +14 -0
- package/dist/components/brutal/newsletter.js +169 -0
- package/dist/components/brutal/newsletter.js.map +1 -0
- package/dist/components/brutal/pricing-table.d.ts +27 -0
- package/dist/components/brutal/pricing-table.js +251 -0
- package/dist/components/brutal/pricing-table.js.map +1 -0
- package/dist/components/brutal/section-divider.d.ts +14 -0
- package/dist/components/brutal/section-divider.js +70 -0
- package/dist/components/brutal/section-divider.js.map +1 -0
- package/dist/components/brutal/section.d.ts +7 -3
- package/dist/components/brutal/section.js +13 -5
- package/dist/components/brutal/section.js.map +1 -1
- package/dist/components/brutal/stats-bar.d.ts +16 -0
- package/dist/components/brutal/stats-bar.js +127 -0
- package/dist/components/brutal/stats-bar.js.map +1 -0
- package/dist/components/brutal/testimonials.d.ts +11 -3
- package/dist/components/brutal/testimonials.js +126 -33
- package/dist/components/brutal/testimonials.js.map +1 -1
- package/dist/components/brutal/wave-divider.d.ts +2 -12
- package/dist/components/brutal/wave-divider.js +54 -26
- package/dist/components/brutal/wave-divider.js.map +1 -1
- package/dist/components/dashboard/activity-feed.d.ts +18 -0
- package/dist/components/dashboard/activity-feed.js +105 -0
- package/dist/components/dashboard/activity-feed.js.map +1 -0
- package/dist/components/dashboard/app-shell.d.ts +19 -0
- package/dist/components/dashboard/app-shell.js +206 -0
- package/dist/components/dashboard/app-shell.js.map +1 -0
- package/dist/components/dashboard/empty-state.d.ts +14 -0
- package/dist/components/dashboard/empty-state.js +96 -0
- package/dist/components/dashboard/empty-state.js.map +1 -0
- package/dist/components/dashboard/file-upload.d.ts +12 -0
- package/dist/components/dashboard/file-upload.js +86 -0
- package/dist/components/dashboard/file-upload.js.map +1 -0
- package/dist/components/dashboard/index.d.ts +10 -0
- package/dist/components/dashboard/index.js +755 -0
- package/dist/components/dashboard/index.js.map +1 -0
- package/dist/components/dashboard/search-bar.d.ts +12 -0
- package/dist/components/dashboard/search-bar.js +49 -0
- package/dist/components/dashboard/search-bar.js.map +1 -0
- package/dist/components/dashboard/sidebar.d.ts +23 -0
- package/dist/components/dashboard/sidebar.js +113 -0
- package/dist/components/dashboard/sidebar.js.map +1 -0
- package/dist/components/dashboard/stat-card.d.ts +13 -0
- package/dist/components/dashboard/stat-card.js +55 -0
- package/dist/components/dashboard/stat-card.js.map +1 -0
- package/dist/components/dashboard/user-menu.d.ts +16 -0
- package/dist/components/dashboard/user-menu.js +179 -0
- package/dist/components/dashboard/user-menu.js.map +1 -0
- package/dist/components/dashboard/view-switcher.d.ts +12 -0
- package/dist/components/dashboard/view-switcher.js +130 -0
- package/dist/components/dashboard/view-switcher.js.map +1 -0
- package/dist/components/views/calendar-view.d.ts +17 -0
- package/dist/components/views/calendar-view.js +182 -0
- package/dist/components/views/calendar-view.js.map +1 -0
- package/dist/components/views/data-table.d.ts +15 -0
- package/dist/components/views/data-table.js +204 -0
- package/dist/components/views/data-table.js.map +1 -0
- package/dist/components/views/grid-view.d.ts +11 -0
- package/dist/components/views/grid-view.js +31 -0
- package/dist/components/views/grid-view.js.map +1 -0
- package/dist/components/views/index.d.ts +7 -0
- package/dist/components/views/index.js +542 -0
- package/dist/components/views/index.js.map +1 -0
- package/dist/components/views/kanban-board.d.ts +21 -0
- package/dist/components/views/kanban-board.js +153 -0
- package/dist/components/views/kanban-board.js.map +1 -0
- package/dist/components/views/list-view.d.ts +19 -0
- package/dist/components/views/list-view.js +96 -0
- package/dist/components/views/list-view.js.map +1 -0
- package/dist/index.d.ts +27 -3
- package/dist/index.js +1881 -142
- package/dist/index.js.map +1 -1
- package/dist/lib/animations.d.ts +68 -0
- package/dist/lib/animations.js +44 -0
- package/dist/lib/animations.js.map +1 -0
- package/dist/templates/dashboard.d.ts +40 -0
- package/dist/templates/dashboard.js +658 -0
- package/dist/templates/dashboard.js.map +1 -0
- package/dist/templates/index.d.ts +4 -0
- package/dist/templates/index.js +2001 -0
- package/dist/templates/index.js.map +1 -0
- package/dist/templates/saas-launch.d.ts +113 -0
- package/dist/templates/saas-launch.js +1394 -0
- package/dist/templates/saas-launch.js.map +1 -0
- package/dist/templates/studio.d.ts +72 -0
- package/dist/templates/studio.js +1099 -0
- package/dist/templates/studio.js.map +1 -0
- package/dist/theme.css +58 -15
- package/package.json +48 -2
|
@@ -0,0 +1,658 @@
|
|
|
1
|
+
"use client";
|
|
2
|
+
import { clsx } from 'clsx';
|
|
3
|
+
import { twMerge } from 'tailwind-merge';
|
|
4
|
+
import * as React from 'react';
|
|
5
|
+
import { createContext, useState, useContext } from 'react';
|
|
6
|
+
import { Dialog } from '@base-ui/react/dialog';
|
|
7
|
+
import { Button as Button$1 } from '@base-ui/react/button';
|
|
8
|
+
import { cva } from 'class-variance-authority';
|
|
9
|
+
import { jsxs, jsx, Fragment } from 'react/jsx-runtime';
|
|
10
|
+
import { XIcon } from 'lucide-react';
|
|
11
|
+
import { Input as Input$1 } from '@base-ui/react/input';
|
|
12
|
+
import { Menu } from '@base-ui/react/menu';
|
|
13
|
+
import { Avatar as Avatar$1 } from '@base-ui/react/avatar';
|
|
14
|
+
import { Toggle } from '@base-ui/react/toggle';
|
|
15
|
+
import { ToggleGroup as ToggleGroup$1 } from '@base-ui/react/toggle-group';
|
|
16
|
+
|
|
17
|
+
function cn(...inputs) {
|
|
18
|
+
return twMerge(clsx(inputs));
|
|
19
|
+
}
|
|
20
|
+
var buttonVariants = cva(
|
|
21
|
+
"group/button inline-flex shrink-0 items-center justify-center rounded-lg whitespace-nowrap font-bold transition-all duration-150 select-none outline-none disabled:pointer-events-none disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
|
|
22
|
+
{
|
|
23
|
+
variants: {
|
|
24
|
+
variant: {
|
|
25
|
+
// Primary: Black bg, white text, offset shadow
|
|
26
|
+
default: "border-brutal border-foreground bg-primary text-primary-foreground shadow-brutal hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal-lg active:translate-x-px active:translate-y-px active:shadow-brutal-sm",
|
|
27
|
+
// CTA: Mint green bg — the Bannerbear signature
|
|
28
|
+
cta: "border-brutal border-foreground bg-cta text-cta-foreground shadow-brutal hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal-lg active:translate-x-px active:translate-y-px active:shadow-brutal-sm",
|
|
29
|
+
// Brand: Dynamic brand color bg
|
|
30
|
+
brand: "border-brutal border-foreground bg-brand text-brand-foreground shadow-brutal hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal-lg active:translate-x-px active:translate-y-px active:shadow-brutal-sm",
|
|
31
|
+
// Outline: White bg, black border, offset shadow
|
|
32
|
+
outline: "border-brutal border-foreground bg-background text-foreground shadow-brutal hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal-lg active:translate-x-px active:translate-y-px active:shadow-brutal-sm",
|
|
33
|
+
// Secondary: Light bg, border, smaller shadow
|
|
34
|
+
secondary: "border-brutal border-foreground bg-secondary text-secondary-foreground shadow-brutal-sm hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal active:translate-x-px active:translate-y-px active:shadow-none",
|
|
35
|
+
// Ghost: No border/shadow, subtle hover
|
|
36
|
+
ghost: "hover:bg-secondary hover:text-foreground",
|
|
37
|
+
// Link: Text only
|
|
38
|
+
link: "text-foreground underline-offset-4 hover:underline",
|
|
39
|
+
// Destructive
|
|
40
|
+
destructive: "border-brutal border-destructive bg-destructive text-destructive-foreground shadow-brutal hover:-translate-x-0.5 hover:-translate-y-0.5 hover:shadow-brutal-lg active:translate-x-px active:translate-y-px active:shadow-brutal-sm",
|
|
41
|
+
// Nav: Thin border, no shadow (for nav Sign In buttons)
|
|
42
|
+
nav: "border border-foreground bg-background text-foreground hover:bg-foreground hover:text-background"
|
|
43
|
+
},
|
|
44
|
+
size: {
|
|
45
|
+
xs: "h-7 gap-1 px-2.5 text-xs",
|
|
46
|
+
sm: "h-8 gap-1.5 px-3 text-sm",
|
|
47
|
+
default: "h-10 gap-2 px-5 text-sm",
|
|
48
|
+
lg: "h-12 gap-2 px-7 text-base",
|
|
49
|
+
xl: "h-14 gap-2.5 px-9 text-lg",
|
|
50
|
+
icon: "size-10",
|
|
51
|
+
"icon-sm": "size-8",
|
|
52
|
+
"icon-lg": "size-12"
|
|
53
|
+
}
|
|
54
|
+
},
|
|
55
|
+
defaultVariants: {
|
|
56
|
+
variant: "default",
|
|
57
|
+
size: "default"
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
);
|
|
61
|
+
function Button({
|
|
62
|
+
className,
|
|
63
|
+
variant = "default",
|
|
64
|
+
size = "default",
|
|
65
|
+
...props
|
|
66
|
+
}) {
|
|
67
|
+
return /* @__PURE__ */ jsx(
|
|
68
|
+
Button$1,
|
|
69
|
+
{
|
|
70
|
+
"data-slot": "button",
|
|
71
|
+
className: cn(buttonVariants({ variant, size, className })),
|
|
72
|
+
...props
|
|
73
|
+
}
|
|
74
|
+
);
|
|
75
|
+
}
|
|
76
|
+
function Sheet({ ...props }) {
|
|
77
|
+
return /* @__PURE__ */ jsx(Dialog.Root, { "data-slot": "sheet", ...props });
|
|
78
|
+
}
|
|
79
|
+
function SheetPortal({ ...props }) {
|
|
80
|
+
return /* @__PURE__ */ jsx(Dialog.Portal, { "data-slot": "sheet-portal", ...props });
|
|
81
|
+
}
|
|
82
|
+
function SheetOverlay({ className, ...props }) {
|
|
83
|
+
return /* @__PURE__ */ jsx(
|
|
84
|
+
Dialog.Backdrop,
|
|
85
|
+
{
|
|
86
|
+
"data-slot": "sheet-overlay",
|
|
87
|
+
className: cn(
|
|
88
|
+
"fixed inset-0 z-50 bg-black/60 transition-opacity duration-150 data-ending-style:opacity-0 data-starting-style:opacity-0",
|
|
89
|
+
className
|
|
90
|
+
),
|
|
91
|
+
...props
|
|
92
|
+
}
|
|
93
|
+
);
|
|
94
|
+
}
|
|
95
|
+
function SheetContent({
|
|
96
|
+
className,
|
|
97
|
+
children,
|
|
98
|
+
side = "right",
|
|
99
|
+
showCloseButton = true,
|
|
100
|
+
...props
|
|
101
|
+
}) {
|
|
102
|
+
return /* @__PURE__ */ jsxs(SheetPortal, { children: [
|
|
103
|
+
/* @__PURE__ */ jsx(SheetOverlay, {}),
|
|
104
|
+
/* @__PURE__ */ jsxs(
|
|
105
|
+
Dialog.Popup,
|
|
106
|
+
{
|
|
107
|
+
"data-slot": "sheet-content",
|
|
108
|
+
"data-side": side,
|
|
109
|
+
className: cn(
|
|
110
|
+
"fixed z-50 flex flex-col gap-4 border-brutal border-foreground bg-background bg-clip-padding text-sm shadow-brutal-lg transition duration-200 ease-in-out data-ending-style:opacity-0 data-starting-style:opacity-0 data-[side=bottom]:inset-x-0 data-[side=bottom]:bottom-0 data-[side=bottom]:h-auto data-[side=bottom]:data-ending-style:translate-y-[2.5rem] data-[side=bottom]:data-starting-style:translate-y-[2.5rem] data-[side=left]:inset-y-0 data-[side=left]:left-0 data-[side=left]:h-full data-[side=left]:w-3/4 data-[side=left]:data-ending-style:translate-x-[-2.5rem] data-[side=left]:data-starting-style:translate-x-[-2.5rem] data-[side=right]:inset-y-0 data-[side=right]:right-0 data-[side=right]:h-full data-[side=right]:w-3/4 data-[side=right]:data-ending-style:translate-x-[2.5rem] data-[side=right]:data-starting-style:translate-x-[2.5rem] data-[side=top]:inset-x-0 data-[side=top]:top-0 data-[side=top]:h-auto data-[side=top]:data-ending-style:translate-y-[-2.5rem] data-[side=top]:data-starting-style:translate-y-[-2.5rem] data-[side=left]:sm:max-w-sm data-[side=right]:sm:max-w-sm",
|
|
111
|
+
className
|
|
112
|
+
),
|
|
113
|
+
...props,
|
|
114
|
+
children: [
|
|
115
|
+
children,
|
|
116
|
+
showCloseButton && /* @__PURE__ */ jsxs(
|
|
117
|
+
Dialog.Close,
|
|
118
|
+
{
|
|
119
|
+
"data-slot": "sheet-close",
|
|
120
|
+
render: /* @__PURE__ */ jsx(
|
|
121
|
+
Button,
|
|
122
|
+
{
|
|
123
|
+
variant: "ghost",
|
|
124
|
+
className: "absolute top-3 right-3",
|
|
125
|
+
size: "icon-sm"
|
|
126
|
+
}
|
|
127
|
+
),
|
|
128
|
+
children: [
|
|
129
|
+
/* @__PURE__ */ jsx(
|
|
130
|
+
XIcon,
|
|
131
|
+
{}
|
|
132
|
+
),
|
|
133
|
+
/* @__PURE__ */ jsx("span", { className: "sr-only", children: "Close" })
|
|
134
|
+
]
|
|
135
|
+
}
|
|
136
|
+
)
|
|
137
|
+
]
|
|
138
|
+
}
|
|
139
|
+
)
|
|
140
|
+
] });
|
|
141
|
+
}
|
|
142
|
+
function SheetTitle({ className, ...props }) {
|
|
143
|
+
return /* @__PURE__ */ jsx(
|
|
144
|
+
Dialog.Title,
|
|
145
|
+
{
|
|
146
|
+
"data-slot": "sheet-title",
|
|
147
|
+
className: cn("text-base font-bold text-foreground", className),
|
|
148
|
+
...props
|
|
149
|
+
}
|
|
150
|
+
);
|
|
151
|
+
}
|
|
152
|
+
var AppShellContext = createContext({
|
|
153
|
+
collapsed: false,
|
|
154
|
+
setCollapsed: () => {
|
|
155
|
+
},
|
|
156
|
+
mobileOpen: false,
|
|
157
|
+
setMobileOpen: () => {
|
|
158
|
+
}
|
|
159
|
+
});
|
|
160
|
+
var useAppShell = () => useContext(AppShellContext);
|
|
161
|
+
function AppShell({
|
|
162
|
+
sidebar,
|
|
163
|
+
header,
|
|
164
|
+
children,
|
|
165
|
+
defaultCollapsed = false,
|
|
166
|
+
className
|
|
167
|
+
}) {
|
|
168
|
+
const [collapsed, setCollapsed] = useState(defaultCollapsed);
|
|
169
|
+
const [mobileOpen, setMobileOpen] = useState(false);
|
|
170
|
+
return /* @__PURE__ */ jsx(
|
|
171
|
+
AppShellContext.Provider,
|
|
172
|
+
{
|
|
173
|
+
value: { collapsed, setCollapsed, mobileOpen, setMobileOpen },
|
|
174
|
+
children: /* @__PURE__ */ jsxs("div", { className: cn("flex h-screen bg-background", className), children: [
|
|
175
|
+
/* @__PURE__ */ jsx(
|
|
176
|
+
"aside",
|
|
177
|
+
{
|
|
178
|
+
className: cn(
|
|
179
|
+
"hidden border-r-brutal border-foreground bg-background transition-all duration-200 md:block",
|
|
180
|
+
collapsed ? "w-16" : "w-64"
|
|
181
|
+
),
|
|
182
|
+
children: sidebar
|
|
183
|
+
}
|
|
184
|
+
),
|
|
185
|
+
/* @__PURE__ */ jsx(Sheet, { open: mobileOpen, onOpenChange: setMobileOpen, children: /* @__PURE__ */ jsxs(SheetContent, { side: "left", className: "w-64 p-0 md:hidden", children: [
|
|
186
|
+
/* @__PURE__ */ jsx(SheetTitle, { className: "sr-only", children: "Navigation" }),
|
|
187
|
+
sidebar
|
|
188
|
+
] }) }),
|
|
189
|
+
/* @__PURE__ */ jsxs("div", { className: "flex flex-1 flex-col overflow-hidden", children: [
|
|
190
|
+
header && /* @__PURE__ */ jsxs("header", { className: "flex h-14 items-center gap-4 border-b-brutal border-foreground px-6", children: [
|
|
191
|
+
/* @__PURE__ */ jsx(
|
|
192
|
+
Button,
|
|
193
|
+
{
|
|
194
|
+
variant: "ghost",
|
|
195
|
+
size: "sm",
|
|
196
|
+
className: "md:hidden",
|
|
197
|
+
onClick: () => setMobileOpen(true),
|
|
198
|
+
children: "\u2630"
|
|
199
|
+
}
|
|
200
|
+
),
|
|
201
|
+
header
|
|
202
|
+
] }),
|
|
203
|
+
/* @__PURE__ */ jsx("main", { className: "flex-1 overflow-auto p-6", children })
|
|
204
|
+
] })
|
|
205
|
+
] })
|
|
206
|
+
}
|
|
207
|
+
);
|
|
208
|
+
}
|
|
209
|
+
function Sidebar({
|
|
210
|
+
logo,
|
|
211
|
+
logoCollapsed,
|
|
212
|
+
groups,
|
|
213
|
+
footer,
|
|
214
|
+
className
|
|
215
|
+
}) {
|
|
216
|
+
const { collapsed, setCollapsed } = useAppShell();
|
|
217
|
+
return /* @__PURE__ */ jsxs("div", { className: cn("flex h-full flex-col", className), children: [
|
|
218
|
+
/* @__PURE__ */ jsxs("div", { className: "flex h-14 items-center justify-between border-b-brutal border-foreground px-4", children: [
|
|
219
|
+
/* @__PURE__ */ jsx("div", { className: "font-black", children: collapsed ? logoCollapsed || logo : logo }),
|
|
220
|
+
/* @__PURE__ */ jsx(
|
|
221
|
+
"button",
|
|
222
|
+
{
|
|
223
|
+
onClick: () => setCollapsed(!collapsed),
|
|
224
|
+
className: "hidden text-xs text-muted-foreground hover:text-foreground md:block",
|
|
225
|
+
children: collapsed ? "\u2192" : "\u2190"
|
|
226
|
+
}
|
|
227
|
+
)
|
|
228
|
+
] }),
|
|
229
|
+
/* @__PURE__ */ jsx("nav", { className: "flex-1 overflow-auto p-2", children: groups.map((group, gi) => /* @__PURE__ */ jsxs("div", { className: "mb-4", children: [
|
|
230
|
+
group.title && !collapsed && /* @__PURE__ */ jsx("p", { className: "brutal-label mb-2 px-2 text-muted-foreground", children: group.title }),
|
|
231
|
+
/* @__PURE__ */ jsx("div", { className: "flex flex-col gap-0.5", children: group.links.map((link) => /* @__PURE__ */ jsxs(
|
|
232
|
+
"a",
|
|
233
|
+
{
|
|
234
|
+
href: link.href,
|
|
235
|
+
className: cn(
|
|
236
|
+
"flex items-center gap-3 rounded-sm px-3 py-2 text-sm font-medium transition-colors",
|
|
237
|
+
link.active ? "bg-brand text-brand-foreground" : "hover:bg-secondary",
|
|
238
|
+
collapsed && "justify-center"
|
|
239
|
+
),
|
|
240
|
+
title: collapsed ? link.label : void 0,
|
|
241
|
+
children: [
|
|
242
|
+
/* @__PURE__ */ jsx("span", { className: "size-5 shrink-0", children: link.icon }),
|
|
243
|
+
!collapsed && /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
244
|
+
/* @__PURE__ */ jsx("span", { className: "flex-1", children: link.label }),
|
|
245
|
+
link.badge !== void 0 && /* @__PURE__ */ jsx("span", { className: "border border-foreground bg-brand-muted px-1.5 font-mono text-[10px] font-bold", children: link.badge })
|
|
246
|
+
] })
|
|
247
|
+
]
|
|
248
|
+
},
|
|
249
|
+
link.href
|
|
250
|
+
)) })
|
|
251
|
+
] }, gi)) }),
|
|
252
|
+
footer && /* @__PURE__ */ jsx("div", { className: "border-t-brutal border-foreground p-3", children: footer })
|
|
253
|
+
] });
|
|
254
|
+
}
|
|
255
|
+
function Input({ className, type, ...props }) {
|
|
256
|
+
return /* @__PURE__ */ jsx(
|
|
257
|
+
Input$1,
|
|
258
|
+
{
|
|
259
|
+
type,
|
|
260
|
+
"data-slot": "input",
|
|
261
|
+
className: cn(
|
|
262
|
+
"h-10 w-full min-w-0 rounded-lg border-brutal border-foreground bg-background px-3 py-2 text-base font-medium shadow-brutal-sm transition-all outline-none placeholder:text-muted-foreground focus:shadow-brutal focus:-translate-x-0.5 focus:-translate-y-0.5 disabled:pointer-events-none disabled:opacity-50 aria-invalid:border-destructive md:text-sm",
|
|
263
|
+
className
|
|
264
|
+
),
|
|
265
|
+
...props
|
|
266
|
+
}
|
|
267
|
+
);
|
|
268
|
+
}
|
|
269
|
+
function SearchBar({
|
|
270
|
+
placeholder = "Search...",
|
|
271
|
+
shortcut = "\u2318K",
|
|
272
|
+
value,
|
|
273
|
+
onChange,
|
|
274
|
+
className
|
|
275
|
+
}) {
|
|
276
|
+
return /* @__PURE__ */ jsxs("div", { className: cn("relative", className), children: [
|
|
277
|
+
/* @__PURE__ */ jsx(
|
|
278
|
+
Input,
|
|
279
|
+
{
|
|
280
|
+
type: "search",
|
|
281
|
+
placeholder,
|
|
282
|
+
value,
|
|
283
|
+
onChange: (e) => onChange?.(e.target.value),
|
|
284
|
+
className: "pr-12"
|
|
285
|
+
}
|
|
286
|
+
),
|
|
287
|
+
shortcut && /* @__PURE__ */ jsx("span", { className: "absolute right-2 top-1/2 -translate-y-1/2 border border-foreground bg-secondary px-1.5 font-mono text-[10px] text-muted-foreground", children: shortcut })
|
|
288
|
+
] });
|
|
289
|
+
}
|
|
290
|
+
function DropdownMenu({ ...props }) {
|
|
291
|
+
return /* @__PURE__ */ jsx(Menu.Root, { "data-slot": "dropdown-menu", ...props });
|
|
292
|
+
}
|
|
293
|
+
function DropdownMenuTrigger({ ...props }) {
|
|
294
|
+
return /* @__PURE__ */ jsx(Menu.Trigger, { "data-slot": "dropdown-menu-trigger", ...props });
|
|
295
|
+
}
|
|
296
|
+
function DropdownMenuContent({
|
|
297
|
+
align = "start",
|
|
298
|
+
alignOffset = 0,
|
|
299
|
+
side = "bottom",
|
|
300
|
+
sideOffset = 4,
|
|
301
|
+
className,
|
|
302
|
+
...props
|
|
303
|
+
}) {
|
|
304
|
+
return /* @__PURE__ */ jsx(Menu.Portal, { children: /* @__PURE__ */ jsx(
|
|
305
|
+
Menu.Positioner,
|
|
306
|
+
{
|
|
307
|
+
className: "isolate z-50 outline-none",
|
|
308
|
+
align,
|
|
309
|
+
alignOffset,
|
|
310
|
+
side,
|
|
311
|
+
sideOffset,
|
|
312
|
+
children: /* @__PURE__ */ jsx(
|
|
313
|
+
Menu.Popup,
|
|
314
|
+
{
|
|
315
|
+
"data-slot": "dropdown-menu-content",
|
|
316
|
+
className: cn("z-50 max-h-(--available-height) w-(--anchor-width) min-w-32 origin-(--transform-origin) overflow-x-hidden overflow-y-auto rounded-lg border-brutal border-foreground bg-popover p-1 text-popover-foreground shadow-brutal duration-100 outline-none data-[side=bottom]:slide-in-from-top-2 data-[side=inline-end]:slide-in-from-left-2 data-[side=inline-start]:slide-in-from-right-2 data-[side=left]:slide-in-from-right-2 data-[side=right]:slide-in-from-left-2 data-[side=top]:slide-in-from-bottom-2 data-open:animate-in data-open:fade-in-0 data-open:zoom-in-95 data-closed:animate-out data-closed:overflow-hidden data-closed:fade-out-0 data-closed:zoom-out-95", className),
|
|
317
|
+
...props
|
|
318
|
+
}
|
|
319
|
+
)
|
|
320
|
+
}
|
|
321
|
+
) });
|
|
322
|
+
}
|
|
323
|
+
function DropdownMenuGroup({ ...props }) {
|
|
324
|
+
return /* @__PURE__ */ jsx(Menu.Group, { "data-slot": "dropdown-menu-group", ...props });
|
|
325
|
+
}
|
|
326
|
+
function DropdownMenuLabel({
|
|
327
|
+
className,
|
|
328
|
+
inset,
|
|
329
|
+
...props
|
|
330
|
+
}) {
|
|
331
|
+
return /* @__PURE__ */ jsx(
|
|
332
|
+
Menu.GroupLabel,
|
|
333
|
+
{
|
|
334
|
+
"data-slot": "dropdown-menu-label",
|
|
335
|
+
"data-inset": inset,
|
|
336
|
+
className: cn(
|
|
337
|
+
"px-1.5 py-1 text-xs font-medium text-muted-foreground data-inset:pl-7",
|
|
338
|
+
className
|
|
339
|
+
),
|
|
340
|
+
...props
|
|
341
|
+
}
|
|
342
|
+
);
|
|
343
|
+
}
|
|
344
|
+
function DropdownMenuItem({
|
|
345
|
+
className,
|
|
346
|
+
inset,
|
|
347
|
+
variant = "default",
|
|
348
|
+
...props
|
|
349
|
+
}) {
|
|
350
|
+
return /* @__PURE__ */ jsx(
|
|
351
|
+
Menu.Item,
|
|
352
|
+
{
|
|
353
|
+
"data-slot": "dropdown-menu-item",
|
|
354
|
+
"data-inset": inset,
|
|
355
|
+
"data-variant": variant,
|
|
356
|
+
className: cn(
|
|
357
|
+
"group/dropdown-menu-item relative flex cursor-default items-center gap-1.5 rounded-md px-1.5 py-1 text-sm outline-hidden select-none focus:bg-accent focus:text-accent-foreground not-data-[variant=destructive]:focus:**:text-accent-foreground data-inset:pl-7 data-[variant=destructive]:text-destructive data-[variant=destructive]:focus:bg-destructive/10 data-[variant=destructive]:focus:text-destructive dark:data-[variant=destructive]:focus:bg-destructive/20 data-disabled:pointer-events-none data-disabled:opacity-50 [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4 data-[variant=destructive]:*:[svg]:text-destructive",
|
|
358
|
+
className
|
|
359
|
+
),
|
|
360
|
+
...props
|
|
361
|
+
}
|
|
362
|
+
);
|
|
363
|
+
}
|
|
364
|
+
function DropdownMenuSeparator({
|
|
365
|
+
className,
|
|
366
|
+
...props
|
|
367
|
+
}) {
|
|
368
|
+
return /* @__PURE__ */ jsx(
|
|
369
|
+
Menu.Separator,
|
|
370
|
+
{
|
|
371
|
+
"data-slot": "dropdown-menu-separator",
|
|
372
|
+
className: cn("-mx-1 my-1 h-px bg-border", className),
|
|
373
|
+
...props
|
|
374
|
+
}
|
|
375
|
+
);
|
|
376
|
+
}
|
|
377
|
+
function Avatar({
|
|
378
|
+
className,
|
|
379
|
+
size = "default",
|
|
380
|
+
...props
|
|
381
|
+
}) {
|
|
382
|
+
return /* @__PURE__ */ jsx(
|
|
383
|
+
Avatar$1.Root,
|
|
384
|
+
{
|
|
385
|
+
"data-slot": "avatar",
|
|
386
|
+
"data-size": size,
|
|
387
|
+
className: cn(
|
|
388
|
+
"group/avatar relative flex size-8 shrink-0 rounded-full select-none after:absolute after:inset-0 after:rounded-full after:border after:border-border after:mix-blend-darken data-[size=lg]:size-10 data-[size=sm]:size-6 dark:after:mix-blend-lighten",
|
|
389
|
+
className
|
|
390
|
+
),
|
|
391
|
+
...props
|
|
392
|
+
}
|
|
393
|
+
);
|
|
394
|
+
}
|
|
395
|
+
function AvatarImage({ className, ...props }) {
|
|
396
|
+
return /* @__PURE__ */ jsx(
|
|
397
|
+
Avatar$1.Image,
|
|
398
|
+
{
|
|
399
|
+
"data-slot": "avatar-image",
|
|
400
|
+
className: cn(
|
|
401
|
+
"aspect-square size-full rounded-full object-cover",
|
|
402
|
+
className
|
|
403
|
+
),
|
|
404
|
+
...props
|
|
405
|
+
}
|
|
406
|
+
);
|
|
407
|
+
}
|
|
408
|
+
function AvatarFallback({
|
|
409
|
+
className,
|
|
410
|
+
...props
|
|
411
|
+
}) {
|
|
412
|
+
return /* @__PURE__ */ jsx(
|
|
413
|
+
Avatar$1.Fallback,
|
|
414
|
+
{
|
|
415
|
+
"data-slot": "avatar-fallback",
|
|
416
|
+
className: cn(
|
|
417
|
+
"flex size-full items-center justify-center rounded-full bg-muted text-sm text-muted-foreground group-data-[size=sm]/avatar:text-xs",
|
|
418
|
+
className
|
|
419
|
+
),
|
|
420
|
+
...props
|
|
421
|
+
}
|
|
422
|
+
);
|
|
423
|
+
}
|
|
424
|
+
function UserMenu({ name, email, avatar, items, onSignOut }) {
|
|
425
|
+
return /* @__PURE__ */ jsxs(DropdownMenu, { children: [
|
|
426
|
+
/* @__PURE__ */ jsxs(
|
|
427
|
+
DropdownMenuTrigger,
|
|
428
|
+
{
|
|
429
|
+
className: "inline-flex shrink-0 cursor-pointer items-center gap-2 rounded-lg px-3 py-1.5 text-sm font-medium outline-none hover:bg-secondary",
|
|
430
|
+
children: [
|
|
431
|
+
/* @__PURE__ */ jsxs(Avatar, { className: "size-7", children: [
|
|
432
|
+
avatar && /* @__PURE__ */ jsx(AvatarImage, { src: avatar, alt: name }),
|
|
433
|
+
/* @__PURE__ */ jsx(AvatarFallback, { children: name[0] })
|
|
434
|
+
] }),
|
|
435
|
+
/* @__PURE__ */ jsx("span", { className: "hidden text-sm font-medium sm:inline", children: name })
|
|
436
|
+
]
|
|
437
|
+
}
|
|
438
|
+
),
|
|
439
|
+
/* @__PURE__ */ jsxs(DropdownMenuContent, { align: "end", className: "w-48", children: [
|
|
440
|
+
/* @__PURE__ */ jsxs(DropdownMenuLabel, { children: [
|
|
441
|
+
/* @__PURE__ */ jsx("p", { className: "text-sm font-bold", children: name }),
|
|
442
|
+
email && /* @__PURE__ */ jsx("p", { className: "text-xs text-muted-foreground", children: email })
|
|
443
|
+
] }),
|
|
444
|
+
/* @__PURE__ */ jsx(DropdownMenuSeparator, {}),
|
|
445
|
+
items && /* @__PURE__ */ jsx(DropdownMenuGroup, { children: items.map((item) => /* @__PURE__ */ jsx(DropdownMenuItem, { onClick: item.onClick, children: item.href ? /* @__PURE__ */ jsx("a", { href: item.href, children: item.label }) : item.label }, item.label)) }),
|
|
446
|
+
onSignOut && /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
447
|
+
/* @__PURE__ */ jsx(DropdownMenuSeparator, {}),
|
|
448
|
+
/* @__PURE__ */ jsx(DropdownMenuItem, { onClick: onSignOut, children: "Sign out" })
|
|
449
|
+
] })
|
|
450
|
+
] })
|
|
451
|
+
] });
|
|
452
|
+
}
|
|
453
|
+
function StatCard({
|
|
454
|
+
label,
|
|
455
|
+
value,
|
|
456
|
+
change,
|
|
457
|
+
changeLabel,
|
|
458
|
+
icon,
|
|
459
|
+
className
|
|
460
|
+
}) {
|
|
461
|
+
return /* @__PURE__ */ jsxs(
|
|
462
|
+
"div",
|
|
463
|
+
{
|
|
464
|
+
className: cn(
|
|
465
|
+
"border-brutal border-foreground bg-background p-5 shadow-brutal",
|
|
466
|
+
className
|
|
467
|
+
),
|
|
468
|
+
children: [
|
|
469
|
+
/* @__PURE__ */ jsxs("div", { className: "mb-3 flex items-center justify-between", children: [
|
|
470
|
+
/* @__PURE__ */ jsx("p", { className: "brutal-label text-muted-foreground", children: label }),
|
|
471
|
+
icon && /* @__PURE__ */ jsx("span", { className: "text-muted-foreground", children: icon })
|
|
472
|
+
] }),
|
|
473
|
+
/* @__PURE__ */ jsx("p", { className: "brutal-h1 mb-1", children: value }),
|
|
474
|
+
change !== void 0 && /* @__PURE__ */ jsxs(
|
|
475
|
+
"p",
|
|
476
|
+
{
|
|
477
|
+
className: cn(
|
|
478
|
+
"flex items-center gap-1 font-mono text-xs font-bold",
|
|
479
|
+
change >= 0 ? "text-cta" : "text-destructive"
|
|
480
|
+
),
|
|
481
|
+
children: [
|
|
482
|
+
/* @__PURE__ */ jsx("span", { children: change >= 0 ? "\u2191" : "\u2193" }),
|
|
483
|
+
/* @__PURE__ */ jsxs("span", { children: [
|
|
484
|
+
Math.abs(change),
|
|
485
|
+
"%"
|
|
486
|
+
] }),
|
|
487
|
+
changeLabel && /* @__PURE__ */ jsx("span", { className: "font-normal text-muted-foreground", children: changeLabel })
|
|
488
|
+
]
|
|
489
|
+
}
|
|
490
|
+
)
|
|
491
|
+
]
|
|
492
|
+
}
|
|
493
|
+
);
|
|
494
|
+
}
|
|
495
|
+
var toggleVariants = cva(
|
|
496
|
+
"group/toggle inline-flex items-center justify-center gap-1 rounded-lg text-sm font-medium whitespace-nowrap transition-all outline-none hover:bg-muted hover:text-foreground focus-visible:ring-2 focus-visible:ring-ring focus-visible:ring-offset-2 disabled:pointer-events-none disabled:opacity-50 aria-invalid:border-destructive aria-invalid:ring-destructive/20 aria-pressed:border-brutal aria-pressed:border-foreground aria-pressed:bg-muted aria-pressed:shadow-brutal-sm data-[state=on]:border-brutal data-[state=on]:border-foreground data-[state=on]:bg-muted data-[state=on]:shadow-brutal-sm [&_svg]:pointer-events-none [&_svg]:shrink-0 [&_svg:not([class*='size-'])]:size-4",
|
|
497
|
+
{
|
|
498
|
+
variants: {
|
|
499
|
+
variant: {
|
|
500
|
+
default: "bg-transparent",
|
|
501
|
+
outline: "border-brutal border-foreground bg-transparent shadow-brutal-sm hover:bg-muted"
|
|
502
|
+
},
|
|
503
|
+
size: {
|
|
504
|
+
default: "h-8 min-w-8 px-2",
|
|
505
|
+
sm: "h-7 min-w-7 rounded-[min(var(--radius-md),12px)] px-1.5 text-[0.8rem]",
|
|
506
|
+
lg: "h-9 min-w-9 px-2.5"
|
|
507
|
+
}
|
|
508
|
+
},
|
|
509
|
+
defaultVariants: {
|
|
510
|
+
variant: "default",
|
|
511
|
+
size: "default"
|
|
512
|
+
}
|
|
513
|
+
}
|
|
514
|
+
);
|
|
515
|
+
var ToggleGroupContext = React.createContext({
|
|
516
|
+
size: "default",
|
|
517
|
+
variant: "default",
|
|
518
|
+
spacing: 0,
|
|
519
|
+
orientation: "horizontal"
|
|
520
|
+
});
|
|
521
|
+
function ToggleGroup({
|
|
522
|
+
className,
|
|
523
|
+
variant,
|
|
524
|
+
size,
|
|
525
|
+
spacing = 0,
|
|
526
|
+
orientation = "horizontal",
|
|
527
|
+
children,
|
|
528
|
+
...props
|
|
529
|
+
}) {
|
|
530
|
+
return /* @__PURE__ */ jsx(
|
|
531
|
+
ToggleGroup$1,
|
|
532
|
+
{
|
|
533
|
+
"data-slot": "toggle-group",
|
|
534
|
+
"data-variant": variant,
|
|
535
|
+
"data-size": size,
|
|
536
|
+
"data-spacing": spacing,
|
|
537
|
+
"data-orientation": orientation,
|
|
538
|
+
style: { "--gap": spacing },
|
|
539
|
+
className: cn(
|
|
540
|
+
"group/toggle-group flex w-fit flex-row items-center gap-[var(--gap)] rounded-lg border-brutal border-foreground data-vertical:flex-col data-vertical:items-stretch",
|
|
541
|
+
className
|
|
542
|
+
),
|
|
543
|
+
...props,
|
|
544
|
+
children: /* @__PURE__ */ jsx(
|
|
545
|
+
ToggleGroupContext.Provider,
|
|
546
|
+
{
|
|
547
|
+
value: { variant, size, spacing, orientation },
|
|
548
|
+
children
|
|
549
|
+
}
|
|
550
|
+
)
|
|
551
|
+
}
|
|
552
|
+
);
|
|
553
|
+
}
|
|
554
|
+
function ToggleGroupItem({
|
|
555
|
+
className,
|
|
556
|
+
children,
|
|
557
|
+
variant = "default",
|
|
558
|
+
size = "default",
|
|
559
|
+
...props
|
|
560
|
+
}) {
|
|
561
|
+
const context = React.useContext(ToggleGroupContext);
|
|
562
|
+
return /* @__PURE__ */ jsx(
|
|
563
|
+
Toggle,
|
|
564
|
+
{
|
|
565
|
+
"data-slot": "toggle-group-item",
|
|
566
|
+
"data-variant": context.variant || variant,
|
|
567
|
+
"data-size": context.size || size,
|
|
568
|
+
"data-spacing": context.spacing,
|
|
569
|
+
className: cn(
|
|
570
|
+
"shrink-0 group-data-[spacing=0]/toggle-group:rounded-none group-data-[spacing=0]/toggle-group:px-2 focus:z-10 focus-visible:z-10 group-data-horizontal/toggle-group:data-[spacing=0]:first:rounded-l-lg group-data-vertical/toggle-group:data-[spacing=0]:first:rounded-t-lg group-data-horizontal/toggle-group:data-[spacing=0]:last:rounded-r-lg group-data-vertical/toggle-group:data-[spacing=0]:last:rounded-b-lg group-data-horizontal/toggle-group:data-[spacing=0]:data-[variant=outline]:border-l-0 group-data-vertical/toggle-group:data-[spacing=0]:data-[variant=outline]:border-t-0 group-data-horizontal/toggle-group:data-[spacing=0]:data-[variant=outline]:first:border-l group-data-vertical/toggle-group:data-[spacing=0]:data-[variant=outline]:first:border-t",
|
|
571
|
+
toggleVariants({
|
|
572
|
+
variant: context.variant || variant,
|
|
573
|
+
size: context.size || size
|
|
574
|
+
}),
|
|
575
|
+
className
|
|
576
|
+
),
|
|
577
|
+
...props,
|
|
578
|
+
children
|
|
579
|
+
}
|
|
580
|
+
);
|
|
581
|
+
}
|
|
582
|
+
var viewLabels = {
|
|
583
|
+
table: "Table",
|
|
584
|
+
kanban: "Board",
|
|
585
|
+
grid: "Grid",
|
|
586
|
+
calendar: "Calendar",
|
|
587
|
+
list: "List"
|
|
588
|
+
};
|
|
589
|
+
function ViewSwitcher({
|
|
590
|
+
value,
|
|
591
|
+
onChange,
|
|
592
|
+
views = ["table", "kanban", "grid", "calendar", "list"],
|
|
593
|
+
className
|
|
594
|
+
}) {
|
|
595
|
+
return /* @__PURE__ */ jsx(
|
|
596
|
+
ToggleGroup,
|
|
597
|
+
{
|
|
598
|
+
value: [value],
|
|
599
|
+
onValueChange: (vals) => {
|
|
600
|
+
if (vals.length > 0) {
|
|
601
|
+
onChange(vals[0]);
|
|
602
|
+
}
|
|
603
|
+
},
|
|
604
|
+
className,
|
|
605
|
+
children: views.map((view) => /* @__PURE__ */ jsx(ToggleGroupItem, { value: view, className: "brutal-label text-xs", children: viewLabels[view] }, view))
|
|
606
|
+
}
|
|
607
|
+
);
|
|
608
|
+
}
|
|
609
|
+
function DashboardTemplate({
|
|
610
|
+
logo,
|
|
611
|
+
logoCollapsed,
|
|
612
|
+
sidebarGroups,
|
|
613
|
+
user,
|
|
614
|
+
userMenuItems,
|
|
615
|
+
onSignOut,
|
|
616
|
+
stats,
|
|
617
|
+
views,
|
|
618
|
+
children,
|
|
619
|
+
sidebarFooter
|
|
620
|
+
}) {
|
|
621
|
+
const [view, setView] = useState("table");
|
|
622
|
+
return /* @__PURE__ */ jsxs(
|
|
623
|
+
AppShell,
|
|
624
|
+
{
|
|
625
|
+
sidebar: /* @__PURE__ */ jsx(
|
|
626
|
+
Sidebar,
|
|
627
|
+
{
|
|
628
|
+
logo,
|
|
629
|
+
logoCollapsed,
|
|
630
|
+
groups: sidebarGroups,
|
|
631
|
+
footer: sidebarFooter
|
|
632
|
+
}
|
|
633
|
+
),
|
|
634
|
+
header: /* @__PURE__ */ jsxs(Fragment, { children: [
|
|
635
|
+
/* @__PURE__ */ jsx(SearchBar, { className: "flex-1 max-w-sm" }),
|
|
636
|
+
/* @__PURE__ */ jsx("div", { className: "ml-auto", children: /* @__PURE__ */ jsx(
|
|
637
|
+
UserMenu,
|
|
638
|
+
{
|
|
639
|
+
name: user.name,
|
|
640
|
+
email: user.email,
|
|
641
|
+
avatar: user.avatar,
|
|
642
|
+
items: userMenuItems,
|
|
643
|
+
onSignOut
|
|
644
|
+
}
|
|
645
|
+
) })
|
|
646
|
+
] }),
|
|
647
|
+
children: [
|
|
648
|
+
stats && stats.length > 0 && /* @__PURE__ */ jsx("div", { className: "mb-6 grid gap-4 sm:grid-cols-2 lg:grid-cols-4", children: stats.map((stat) => /* @__PURE__ */ jsx(StatCard, { ...stat }, stat.label)) }),
|
|
649
|
+
views && views.length > 1 && /* @__PURE__ */ jsx("div", { className: "mb-4", children: /* @__PURE__ */ jsx(ViewSwitcher, { value: view, onChange: setView, views }) }),
|
|
650
|
+
children
|
|
651
|
+
]
|
|
652
|
+
}
|
|
653
|
+
);
|
|
654
|
+
}
|
|
655
|
+
|
|
656
|
+
export { DashboardTemplate };
|
|
657
|
+
//# sourceMappingURL=dashboard.js.map
|
|
658
|
+
//# sourceMappingURL=dashboard.js.map
|