@alpic-ai/ui 0.0.0-dev.f8c735b → 0.0.0-dev.f936359
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/accordion-card.d.mts +1 -1
- package/dist/components/accordion.d.mts +1 -1
- package/dist/components/alert.d.mts +1 -1
- package/dist/components/attachment-tile.mjs +1 -1
- package/dist/components/avatar.d.mts +1 -1
- package/dist/components/breadcrumb.d.mts +1 -1
- package/dist/components/button.d.mts +4 -2
- package/dist/components/button.mjs +20 -6
- package/dist/components/card.d.mts +1 -1
- package/dist/components/checkbox.d.mts +1 -1
- package/dist/components/collapsible.d.mts +1 -1
- package/dist/components/combobox.d.mts +1 -1
- package/dist/components/combobox.mjs +1 -1
- package/dist/components/command.d.mts +1 -1
- package/dist/components/copyable.d.mts +1 -1
- package/dist/components/copyable.mjs +1 -1
- package/dist/components/description-list.d.mts +1 -1
- package/dist/components/dialog.d.mts +1 -1
- package/dist/components/dropdown-menu.d.mts +1 -1
- package/dist/components/form.d.mts +1 -1
- package/dist/components/form.mjs +1 -1
- package/dist/components/github-button.d.mts +13 -0
- package/dist/components/github-button.mjs +24 -0
- package/dist/components/input-group.d.mts +1 -1
- package/dist/components/input.d.mts +1 -1
- package/dist/components/input.mjs +1 -1
- package/dist/components/label.d.mts +1 -1
- package/dist/components/page-loader.d.mts +11 -0
- package/dist/components/page-loader.mjs +122 -0
- package/dist/components/pagination.d.mts +1 -1
- package/dist/components/popover.d.mts +1 -1
- package/dist/components/radio-group.d.mts +1 -1
- package/dist/components/scroll-area.d.mts +1 -1
- package/dist/components/select.d.mts +1 -1
- package/dist/components/separator.d.mts +1 -1
- package/dist/components/sheet.d.mts +1 -1
- package/dist/components/sidebar.d.mts +1 -1
- package/dist/components/sidebar.mjs +63 -19
- package/dist/components/sonner.d.mts +1 -1
- package/dist/components/switch.d.mts +1 -1
- package/dist/components/table.d.mts +11 -2
- package/dist/components/table.mjs +2 -2
- package/dist/components/tabs.d.mts +1 -1
- package/dist/components/tabs.mjs +1 -1
- package/dist/components/textarea.d.mts +1 -1
- package/dist/components/textarea.mjs +1 -1
- package/dist/components/toggle-group.d.mts +1 -1
- package/dist/components/toggle-group.mjs +1 -1
- package/dist/components/tooltip-icon-button.mjs +1 -1
- package/dist/components/tooltip.d.mts +1 -1
- package/dist/components/typography.d.mts +1 -1
- package/package.json +10 -10
- package/src/components/button.tsx +13 -9
- package/src/components/combobox.tsx +18 -6
- package/src/components/github-button.tsx +34 -0
- package/src/components/page-loader.tsx +59 -0
- package/src/components/sidebar.tsx +59 -20
- package/src/components/table.tsx +16 -2
- package/src/hooks/use-copy-to-clipboard.ts +6 -2
- package/src/stories/button.stories.tsx +23 -1
- package/src/stories/sidebar.stories.tsx +6 -3
- package/src/stories/table.stories.tsx +2 -2
- package/src/styles/tokens.css +210 -0
|
@@ -2,7 +2,6 @@
|
|
|
2
2
|
|
|
3
3
|
import { Slot } from "@radix-ui/react-slot";
|
|
4
4
|
import { cva, type VariantProps } from "class-variance-authority";
|
|
5
|
-
import { PanelLeftClose, PanelLeftOpen } from "lucide-react";
|
|
6
5
|
import * as React from "react";
|
|
7
6
|
|
|
8
7
|
import { useIsMobile } from "../hooks/use-mobile";
|
|
@@ -20,6 +19,19 @@ const SIDEBAR_WIDTH_MOBILE = "16rem";
|
|
|
20
19
|
const SIDEBAR_WIDTH_ICON = "3.5rem";
|
|
21
20
|
const SIDEBAR_KEYBOARD_SHORTCUT = "b";
|
|
22
21
|
|
|
22
|
+
const INTERACTIVE_SIDEBAR_ELEMENT_SELECTOR = [
|
|
23
|
+
"a",
|
|
24
|
+
"button",
|
|
25
|
+
"input",
|
|
26
|
+
"select",
|
|
27
|
+
"textarea",
|
|
28
|
+
"[role='button']",
|
|
29
|
+
"[role='link']",
|
|
30
|
+
"[role='menuitem']",
|
|
31
|
+
"[contenteditable='true']",
|
|
32
|
+
"[tabindex]:not([tabindex='-1'])",
|
|
33
|
+
].join(", ");
|
|
34
|
+
|
|
23
35
|
type SidebarContextProps = {
|
|
24
36
|
state: "expanded" | "collapsed";
|
|
25
37
|
open: boolean;
|
|
@@ -143,13 +155,31 @@ function Sidebar({
|
|
|
143
155
|
variant?: "sidebar" | "floating" | "inset";
|
|
144
156
|
collapsible?: "offcanvas" | "icon" | "none";
|
|
145
157
|
}) {
|
|
146
|
-
const { isMobile, state, openMobile, setOpenMobile } = useSidebar();
|
|
158
|
+
const { isMobile, state, openMobile, setOpenMobile, open, setOpen } = useSidebar();
|
|
159
|
+
|
|
160
|
+
function handleSurfaceClickCapture(event: React.MouseEvent<HTMLDivElement>) {
|
|
161
|
+
const clickedInteractiveElement =
|
|
162
|
+
event.target instanceof Element && event.target.closest(INTERACTIVE_SIDEBAR_ELEMENT_SELECTOR);
|
|
163
|
+
|
|
164
|
+
if (clickedInteractiveElement) {
|
|
165
|
+
return;
|
|
166
|
+
}
|
|
167
|
+
|
|
168
|
+
if (!open) {
|
|
169
|
+
event.preventDefault();
|
|
170
|
+
event.stopPropagation();
|
|
171
|
+
setOpen(true);
|
|
172
|
+
return;
|
|
173
|
+
}
|
|
174
|
+
|
|
175
|
+
setOpen(false);
|
|
176
|
+
}
|
|
147
177
|
|
|
148
178
|
if (collapsible === "none") {
|
|
149
179
|
return (
|
|
150
180
|
<div
|
|
151
181
|
data-slot="sidebar"
|
|
152
|
-
className={cn("bg-sidebar text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col", className)}
|
|
182
|
+
className={cn("bg-sidebar-surface text-sidebar-foreground flex h-full w-(--sidebar-width) flex-col", className)}
|
|
153
183
|
{...props}
|
|
154
184
|
>
|
|
155
185
|
{children}
|
|
@@ -164,7 +194,7 @@ function Sidebar({
|
|
|
164
194
|
data-sidebar="sidebar"
|
|
165
195
|
data-slot="sidebar"
|
|
166
196
|
data-mobile="true"
|
|
167
|
-
className="bg-sidebar text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
|
|
197
|
+
className="bg-sidebar-surface text-sidebar-foreground w-(--sidebar-width) p-0 [&>button]:hidden"
|
|
168
198
|
style={
|
|
169
199
|
{
|
|
170
200
|
"--sidebar-width": SIDEBAR_WIDTH_MOBILE,
|
|
@@ -213,7 +243,7 @@ function Sidebar({
|
|
|
213
243
|
// Adjust the padding for floating and inset variants.
|
|
214
244
|
variant === "floating" || variant === "inset"
|
|
215
245
|
? "p-2 group-data-[collapsible=icon]:w-[calc(var(--sidebar-width-icon)+(--spacing(4))+2px)]"
|
|
216
|
-
: "group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l",
|
|
246
|
+
: "border-sidebar-border group-data-[collapsible=icon]:w-(--sidebar-width-icon) group-data-[side=left]:border-r group-data-[side=right]:border-l",
|
|
217
247
|
className,
|
|
218
248
|
)}
|
|
219
249
|
{...props}
|
|
@@ -221,7 +251,8 @@ function Sidebar({
|
|
|
221
251
|
<div
|
|
222
252
|
data-sidebar="sidebar"
|
|
223
253
|
data-slot="sidebar-inner"
|
|
224
|
-
className="bg-sidebar group-data-[variant=floating]:border-sidebar-border flex h-full w-full flex-col group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:shadow-sm"
|
|
254
|
+
className="bg-sidebar-surface group-data-[variant=floating]:border-sidebar-border flex h-full w-full cursor-pointer flex-col group-data-[variant=floating]:rounded-lg group-data-[variant=floating]:border group-data-[variant=floating]:shadow-sm"
|
|
255
|
+
onClickCapture={handleSurfaceClickCapture}
|
|
225
256
|
>
|
|
226
257
|
{children}
|
|
227
258
|
</div>
|
|
@@ -231,9 +262,7 @@ function Sidebar({
|
|
|
231
262
|
}
|
|
232
263
|
|
|
233
264
|
function SidebarTrigger({ className, onClick, ...props }: React.ComponentProps<typeof Button>) {
|
|
234
|
-
const {
|
|
235
|
-
|
|
236
|
-
const isOpen = isMobile ? openMobile : state === "expanded";
|
|
265
|
+
const { toggleSidebar } = useSidebar();
|
|
237
266
|
|
|
238
267
|
return (
|
|
239
268
|
<Button
|
|
@@ -248,12 +277,21 @@ function SidebarTrigger({ className, onClick, ...props }: React.ComponentProps<t
|
|
|
248
277
|
}}
|
|
249
278
|
{...props}
|
|
250
279
|
>
|
|
251
|
-
|
|
280
|
+
<SidebarToggleIcon className="size-4.5" />
|
|
252
281
|
<span className="sr-only">Toggle Sidebar</span>
|
|
253
282
|
</Button>
|
|
254
283
|
);
|
|
255
284
|
}
|
|
256
285
|
|
|
286
|
+
function SidebarToggleIcon({ className, ...props }: React.ComponentProps<"svg">) {
|
|
287
|
+
return (
|
|
288
|
+
<svg viewBox="0 0 20 20" fill="none" aria-hidden="true" className={className} {...props}>
|
|
289
|
+
<rect x="2.75" y="2.75" width="14.5" height="14.5" rx="4" stroke="currentColor" strokeWidth="1.5" />
|
|
290
|
+
<path d="M10 4.75V15.25" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round" />
|
|
291
|
+
</svg>
|
|
292
|
+
);
|
|
293
|
+
}
|
|
294
|
+
|
|
257
295
|
function SidebarRail({ className, ...props }: React.ComponentProps<"button">) {
|
|
258
296
|
const { toggleSidebar } = useSidebar();
|
|
259
297
|
|
|
@@ -266,9 +304,7 @@ function SidebarRail({ className, ...props }: React.ComponentProps<"button">) {
|
|
|
266
304
|
onClick={toggleSidebar}
|
|
267
305
|
title="Toggle Sidebar"
|
|
268
306
|
className={cn(
|
|
269
|
-
"
|
|
270
|
-
"in-data-[side=left]:cursor-w-resize in-data-[side=right]:cursor-e-resize",
|
|
271
|
-
"[[data-side=left][data-state=collapsed]_&]:cursor-e-resize [[data-side=right][data-state=collapsed]_&]:cursor-w-resize",
|
|
307
|
+
"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",
|
|
272
308
|
"[@media(hover:hover)]:hover:group-data-[collapsible=offcanvas]:bg-sidebar group-data-[collapsible=offcanvas]:translate-x-0 group-data-[collapsible=offcanvas]:after:left-full",
|
|
273
309
|
"[[data-side=left][data-collapsible=offcanvas]_&]:-right-2",
|
|
274
310
|
"[[data-side=right][data-collapsible=offcanvas]_&]:-left-2",
|
|
@@ -316,16 +352,19 @@ function SidebarHeader({ className, icon, title, children, ...props }: SidebarHe
|
|
|
316
352
|
<div
|
|
317
353
|
data-slot="sidebar-header"
|
|
318
354
|
data-sidebar="header"
|
|
319
|
-
className={cn("flex flex-col gap-2
|
|
355
|
+
className={cn("flex flex-col gap-2 py-2", className)}
|
|
320
356
|
{...props}
|
|
321
357
|
>
|
|
322
358
|
<div className="flex h-8 items-center gap-2 px-3">
|
|
323
|
-
<
|
|
324
|
-
<span className="transition-opacity group-data-[collapsible=icon]:group-hover:opacity-0">
|
|
325
|
-
|
|
326
|
-
|
|
327
|
-
|
|
328
|
-
|
|
359
|
+
<span className="relative flex size-8 shrink-0 items-center justify-center">
|
|
360
|
+
<span className="transition-opacity duration-200 group-data-[collapsible=icon]:group-hover:opacity-0">
|
|
361
|
+
{icon}
|
|
362
|
+
</span>
|
|
363
|
+
<SidebarTrigger
|
|
364
|
+
tabIndex={-1}
|
|
365
|
+
className="absolute inset-0 opacity-0 transition-opacity duration-200 group-data-[collapsible=icon]:group-hover:opacity-100"
|
|
366
|
+
/>
|
|
367
|
+
</span>
|
|
329
368
|
<span className="text-foreground text-md min-w-0 truncate font-medium group-data-[collapsible=icon]:hidden">
|
|
330
369
|
{title}
|
|
331
370
|
</span>
|
package/src/components/table.tsx
CHANGED
|
@@ -66,11 +66,25 @@ function TableHead({ className, ...props }: React.ComponentProps<"th">) {
|
|
|
66
66
|
);
|
|
67
67
|
}
|
|
68
68
|
|
|
69
|
-
|
|
69
|
+
interface TableCellProps extends React.ComponentProps<"td"> {
|
|
70
|
+
/**
|
|
71
|
+
* When true, the cell renders edge-to-edge so the child can act as the
|
|
72
|
+
* interactive surface (e.g. a button or popover trigger filling the cell).
|
|
73
|
+
* Defaults to false (standard padded cell).
|
|
74
|
+
*/
|
|
75
|
+
interactive?: boolean;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
function TableCell({ className, interactive = false, ...props }: TableCellProps) {
|
|
70
79
|
return (
|
|
71
80
|
<td
|
|
72
81
|
data-slot="table-cell"
|
|
73
|
-
className={cn(
|
|
82
|
+
className={cn(
|
|
83
|
+
"align-middle",
|
|
84
|
+
interactive ? "h-px p-0" : "px-6 py-2",
|
|
85
|
+
"[&:has([role=checkbox])]:w-px [&:has([role=checkbox])]:pr-3",
|
|
86
|
+
className,
|
|
87
|
+
)}
|
|
74
88
|
{...props}
|
|
75
89
|
/>
|
|
76
90
|
);
|
|
@@ -8,7 +8,9 @@ export function useCopyToClipboard({ resetDelay = 2000 }: { resetDelay?: number
|
|
|
8
8
|
|
|
9
9
|
useEffect(() => {
|
|
10
10
|
return () => {
|
|
11
|
-
if (timeoutRef.current)
|
|
11
|
+
if (timeoutRef.current) {
|
|
12
|
+
clearTimeout(timeoutRef.current);
|
|
13
|
+
}
|
|
12
14
|
};
|
|
13
15
|
}, []);
|
|
14
16
|
|
|
@@ -16,7 +18,9 @@ export function useCopyToClipboard({ resetDelay = 2000 }: { resetDelay?: number
|
|
|
16
18
|
(text: string) => {
|
|
17
19
|
navigator.clipboard.writeText(text).then(() => {
|
|
18
20
|
setIsCopied(true);
|
|
19
|
-
if (timeoutRef.current)
|
|
21
|
+
if (timeoutRef.current) {
|
|
22
|
+
clearTimeout(timeoutRef.current);
|
|
23
|
+
}
|
|
20
24
|
timeoutRef.current = setTimeout(() => setIsCopied(false), resetDelay);
|
|
21
25
|
});
|
|
22
26
|
},
|
|
@@ -1,4 +1,4 @@
|
|
|
1
|
-
import { Plus } from "lucide-react";
|
|
1
|
+
import { ArrowRight, Plus, Sparkles } from "lucide-react";
|
|
2
2
|
|
|
3
3
|
import { Button } from "../components/button";
|
|
4
4
|
|
|
@@ -323,6 +323,28 @@ export const AllVariants = () => {
|
|
|
323
323
|
</Button>
|
|
324
324
|
</div>
|
|
325
325
|
|
|
326
|
+
{/* ── CTA (animated gradient ring) ────────────────────────────────── */}
|
|
327
|
+
<span className={SECTION_HEADER}>CTA — animated</span>
|
|
328
|
+
<p className="type-text-xs text-muted-foreground -mt-2 max-w-md">
|
|
329
|
+
Hover to rotate the conic gradient around the border and ignite the soft halo.
|
|
330
|
+
</p>
|
|
331
|
+
|
|
332
|
+
<div className="flex items-center gap-6">
|
|
333
|
+
<Button variant="cta" iconTrailing={<ArrowRight />}>
|
|
334
|
+
Get started
|
|
335
|
+
</Button>
|
|
336
|
+
<Button variant="cta" icon={<Sparkles />} iconTrailing={<ArrowRight />}>
|
|
337
|
+
Launch server
|
|
338
|
+
</Button>
|
|
339
|
+
<Button variant="cta">Deploy now</Button>
|
|
340
|
+
<Button variant="cta" iconTrailing={<ArrowRight />} disabled>
|
|
341
|
+
Disabled
|
|
342
|
+
</Button>
|
|
343
|
+
<Button variant="cta" iconTrailing={<ArrowRight />} loading>
|
|
344
|
+
Deploying
|
|
345
|
+
</Button>
|
|
346
|
+
</div>
|
|
347
|
+
|
|
326
348
|
{/* ── asChild ─────────────────────────────────────────────────────── */}
|
|
327
349
|
<span className={SECTION_HEADER}>asChild</span>
|
|
328
350
|
|
|
@@ -107,7 +107,8 @@ export const AllVariants: Story = () => (
|
|
|
107
107
|
<UserFooter />
|
|
108
108
|
</Sidebar>
|
|
109
109
|
<SidebarInset>
|
|
110
|
-
<div className="p-4">
|
|
110
|
+
<div className="flex items-center gap-2 p-4">
|
|
111
|
+
<SidebarTrigger />
|
|
111
112
|
<span className="type-text-sm text-muted-foreground">Sub-menus expand inline under their parent</span>
|
|
112
113
|
</div>
|
|
113
114
|
</SidebarInset>
|
|
@@ -169,7 +170,8 @@ export const AllVariants: Story = () => (
|
|
|
169
170
|
<UserFooter />
|
|
170
171
|
</Sidebar>
|
|
171
172
|
<SidebarInset>
|
|
172
|
-
<div className="p-4">
|
|
173
|
+
<div className="flex items-center gap-2 p-4">
|
|
174
|
+
<SidebarTrigger />
|
|
173
175
|
<span className="type-text-sm text-muted-foreground">Groups with labels, badges, and separator</span>
|
|
174
176
|
</div>
|
|
175
177
|
</SidebarInset>
|
|
@@ -195,7 +197,8 @@ export const AllVariants: Story = () => (
|
|
|
195
197
|
<UserFooter />
|
|
196
198
|
</Sidebar>
|
|
197
199
|
<SidebarInset>
|
|
198
|
-
<div className="p-4">
|
|
200
|
+
<div className="flex items-center gap-2 p-4">
|
|
201
|
+
<SidebarTrigger />
|
|
199
202
|
<span className="type-text-sm text-muted-foreground">Loading state with skeleton placeholders</span>
|
|
200
203
|
</div>
|
|
201
204
|
</SidebarInset>
|
|
@@ -190,8 +190,8 @@ export const AllVariants: Story = () => (
|
|
|
190
190
|
</TableRow>
|
|
191
191
|
</TableHeader>
|
|
192
192
|
<TableBody>
|
|
193
|
-
{USERS.slice(0, 3).map((user,
|
|
194
|
-
<TableRow key={user.id} data-state={
|
|
193
|
+
{USERS.slice(0, 3).map((user, index) => (
|
|
194
|
+
<TableRow key={user.id} data-state={index === 1 ? "selected" : undefined}>
|
|
195
195
|
<TableCell className="font-medium text-foreground">{user.name}</TableCell>
|
|
196
196
|
<TableCell className="text-subtle-foreground">{user.role}</TableCell>
|
|
197
197
|
<TableCell className="text-subtle-foreground">{user.email}</TableCell>
|
package/src/styles/tokens.css
CHANGED
|
@@ -39,6 +39,9 @@
|
|
|
39
39
|
|
|
40
40
|
--color-ring: #f22b79; /* Figma: focus-ring */
|
|
41
41
|
|
|
42
|
+
/* cta — decorative gradient accent, used with --color-primary in the CTA button ring */
|
|
43
|
+
--color-cta-accent: #6eece7; /* Figma: CTA border accent (cyan) */
|
|
44
|
+
|
|
42
45
|
/* sidebar */
|
|
43
46
|
--color-sidebar: #f8fafa; /* Figma: bg-secondary-subtle */
|
|
44
47
|
--color-sidebar-foreground: #3a4848; /* Figma: fg-secondary */
|
|
@@ -106,6 +109,17 @@
|
|
|
106
109
|
/* animations */
|
|
107
110
|
--animate-accordion-down: accordion-down 200ms ease-out;
|
|
108
111
|
--animate-accordion-up: accordion-up 200ms ease-out;
|
|
112
|
+
--animate-beacon-ring-core: beacon-ring-core 2.2s ease-in-out infinite;
|
|
113
|
+
--animate-beacon-ring-pulse: beacon-ring-pulse 2.2s ease-in-out infinite;
|
|
114
|
+
}
|
|
115
|
+
|
|
116
|
+
@keyframes alpic-ride {
|
|
117
|
+
0% {
|
|
118
|
+
transform: translate(0px, 0px);
|
|
119
|
+
}
|
|
120
|
+
100% {
|
|
121
|
+
transform: translate(237px, -64px);
|
|
122
|
+
}
|
|
109
123
|
}
|
|
110
124
|
|
|
111
125
|
@keyframes accordion-down {
|
|
@@ -126,6 +140,186 @@
|
|
|
126
140
|
}
|
|
127
141
|
}
|
|
128
142
|
|
|
143
|
+
@keyframes beacon-ring-core {
|
|
144
|
+
0%,
|
|
145
|
+
100% {
|
|
146
|
+
opacity: 0.55;
|
|
147
|
+
transform: scale(0.98);
|
|
148
|
+
}
|
|
149
|
+
50% {
|
|
150
|
+
opacity: 0.85;
|
|
151
|
+
transform: scale(1.02);
|
|
152
|
+
}
|
|
153
|
+
}
|
|
154
|
+
|
|
155
|
+
@keyframes beacon-ring-pulse {
|
|
156
|
+
0%,
|
|
157
|
+
100% {
|
|
158
|
+
opacity: 0.88;
|
|
159
|
+
transform: scale(1);
|
|
160
|
+
filter: drop-shadow(0 0 0 rgba(59, 130, 246, 0));
|
|
161
|
+
}
|
|
162
|
+
50% {
|
|
163
|
+
opacity: 1;
|
|
164
|
+
transform: scale(1.02);
|
|
165
|
+
filter: drop-shadow(0 0 8px rgba(59, 130, 246, 0.24));
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
|
|
169
|
+
/* ─── CTA button — animated conic gradient ring ───────────────────────────── */
|
|
170
|
+
|
|
171
|
+
@property --cta-angle {
|
|
172
|
+
syntax: "<angle>";
|
|
173
|
+
inherits: false;
|
|
174
|
+
initial-value: 135deg;
|
|
175
|
+
}
|
|
176
|
+
|
|
177
|
+
@keyframes cta-rotate {
|
|
178
|
+
to {
|
|
179
|
+
--cta-angle: 495deg;
|
|
180
|
+
}
|
|
181
|
+
}
|
|
182
|
+
|
|
183
|
+
.button-cta {
|
|
184
|
+
position: relative;
|
|
185
|
+
isolation: isolate;
|
|
186
|
+
/* light mode: whisper-of-gradient surface tint + soft rose drop-shadow */
|
|
187
|
+
background-image: linear-gradient(
|
|
188
|
+
135deg,
|
|
189
|
+
color-mix(in oklab, var(--color-primary) 5%, transparent) 0%,
|
|
190
|
+
color-mix(in oklab, var(--color-cta-accent) 5%, transparent) 100%
|
|
191
|
+
);
|
|
192
|
+
box-shadow:
|
|
193
|
+
0 6px 24px -10px color-mix(in oklab, var(--color-primary) 38%, transparent),
|
|
194
|
+
0 2px 6px -4px color-mix(in oklab, var(--color-cta-accent) 30%, transparent);
|
|
195
|
+
transition:
|
|
196
|
+
box-shadow 400ms ease,
|
|
197
|
+
transform 300ms ease,
|
|
198
|
+
filter 300ms ease;
|
|
199
|
+
}
|
|
200
|
+
|
|
201
|
+
@media (hover: hover) {
|
|
202
|
+
.button-cta:hover {
|
|
203
|
+
box-shadow:
|
|
204
|
+
0 10px 30px -8px color-mix(in oklab, var(--color-primary) 52%, transparent),
|
|
205
|
+
0 3px 10px -3px color-mix(in oklab, var(--color-cta-accent) 40%, transparent);
|
|
206
|
+
}
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
/* dark mode: solid inverted surface, no tint or shadow — let the halo do the work */
|
|
210
|
+
.dark .button-cta {
|
|
211
|
+
background-image: none;
|
|
212
|
+
box-shadow: none;
|
|
213
|
+
}
|
|
214
|
+
|
|
215
|
+
.button-cta::before,
|
|
216
|
+
.button-cta::after {
|
|
217
|
+
content: "";
|
|
218
|
+
position: absolute;
|
|
219
|
+
inset: 0;
|
|
220
|
+
border-radius: inherit;
|
|
221
|
+
pointer-events: none;
|
|
222
|
+
/* Always "running" in browser terms, but paused at rest — freezes at current
|
|
223
|
+
angle on unhover instead of snapping back. */
|
|
224
|
+
animation: cta-rotate 3.2s linear infinite;
|
|
225
|
+
animation-play-state: paused;
|
|
226
|
+
}
|
|
227
|
+
|
|
228
|
+
/* Gradient ring (masked so only the border shows) */
|
|
229
|
+
.button-cta::before {
|
|
230
|
+
padding: 1.5px;
|
|
231
|
+
background: conic-gradient(
|
|
232
|
+
from var(--cta-angle),
|
|
233
|
+
var(--color-cta-accent) 0deg,
|
|
234
|
+
var(--color-primary) 150deg,
|
|
235
|
+
var(--color-cta-accent) 300deg,
|
|
236
|
+
var(--color-cta-accent) 360deg
|
|
237
|
+
);
|
|
238
|
+
-webkit-mask:
|
|
239
|
+
linear-gradient(#000 0 0) content-box,
|
|
240
|
+
linear-gradient(#000 0 0);
|
|
241
|
+
-webkit-mask-composite: xor;
|
|
242
|
+
mask-composite: exclude;
|
|
243
|
+
transition: filter 400ms ease;
|
|
244
|
+
}
|
|
245
|
+
|
|
246
|
+
/* Blurred glow halo behind the button — subtle in light, bolder in dark */
|
|
247
|
+
.button-cta::after {
|
|
248
|
+
z-index: -1;
|
|
249
|
+
background: conic-gradient(
|
|
250
|
+
from var(--cta-angle),
|
|
251
|
+
var(--color-cta-accent) 0deg,
|
|
252
|
+
var(--color-primary) 150deg,
|
|
253
|
+
var(--color-cta-accent) 300deg,
|
|
254
|
+
var(--color-cta-accent) 360deg
|
|
255
|
+
);
|
|
256
|
+
filter: blur(12px);
|
|
257
|
+
opacity: 0.05;
|
|
258
|
+
transition: opacity 400ms ease;
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
.dark .button-cta::after {
|
|
262
|
+
opacity: 0.14;
|
|
263
|
+
}
|
|
264
|
+
|
|
265
|
+
@media (hover: hover) {
|
|
266
|
+
.button-cta:hover::before,
|
|
267
|
+
.button-cta:hover::after {
|
|
268
|
+
animation-play-state: running;
|
|
269
|
+
}
|
|
270
|
+
.button-cta:hover::before {
|
|
271
|
+
filter: saturate(1.15) brightness(1.05);
|
|
272
|
+
}
|
|
273
|
+
.button-cta:hover::after {
|
|
274
|
+
opacity: 0.18;
|
|
275
|
+
}
|
|
276
|
+
.dark .button-cta:hover::after {
|
|
277
|
+
opacity: 0.32;
|
|
278
|
+
}
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
.button-cta:focus-visible::before,
|
|
282
|
+
.button-cta:focus-visible::after {
|
|
283
|
+
animation-play-state: running;
|
|
284
|
+
}
|
|
285
|
+
|
|
286
|
+
.button-cta:disabled::before,
|
|
287
|
+
.button-cta:disabled::after,
|
|
288
|
+
[aria-busy="true"].button-cta::before,
|
|
289
|
+
[aria-busy="true"].button-cta::after {
|
|
290
|
+
animation-play-state: paused;
|
|
291
|
+
}
|
|
292
|
+
.button-cta:disabled::after {
|
|
293
|
+
opacity: 0;
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
@media (prefers-reduced-motion: reduce) {
|
|
297
|
+
.button-cta::before,
|
|
298
|
+
.button-cta::after {
|
|
299
|
+
animation: none;
|
|
300
|
+
}
|
|
301
|
+
}
|
|
302
|
+
|
|
303
|
+
/* Icon slide on hover (applied to [data-cta-icon-trailing]) */
|
|
304
|
+
.button-cta [data-cta-icon-trailing] {
|
|
305
|
+
transition: transform 300ms cubic-bezier(0.2, 0.8, 0.2, 1);
|
|
306
|
+
}
|
|
307
|
+
|
|
308
|
+
@media (hover: hover) {
|
|
309
|
+
.button-cta:hover [data-cta-icon-trailing] {
|
|
310
|
+
transform: translateX(2px);
|
|
311
|
+
}
|
|
312
|
+
}
|
|
313
|
+
|
|
314
|
+
@media (prefers-reduced-motion: reduce) {
|
|
315
|
+
.button-cta [data-cta-icon-trailing] {
|
|
316
|
+
transition: none;
|
|
317
|
+
}
|
|
318
|
+
.button-cta:hover [data-cta-icon-trailing] {
|
|
319
|
+
transform: none;
|
|
320
|
+
}
|
|
321
|
+
}
|
|
322
|
+
|
|
129
323
|
/* ─── Dark mode ───────────────────────────────────────────────────────────── */
|
|
130
324
|
|
|
131
325
|
.dark {
|
|
@@ -163,6 +357,9 @@
|
|
|
163
357
|
|
|
164
358
|
--color-ring: #f22b79; /* Figma: focus-ring */
|
|
165
359
|
|
|
360
|
+
/* cta — decorative gradient accent, used with --color-primary in the CTA button ring */
|
|
361
|
+
--color-cta-accent: #6eece7; /* Figma: CTA border accent (cyan) */
|
|
362
|
+
|
|
166
363
|
/* sidebar */
|
|
167
364
|
--color-sidebar: #0c1c1c; /* Figma: bg-secondary */
|
|
168
365
|
--color-sidebar-foreground: #90a4a4; /* Figma: fg-secondary */
|
|
@@ -221,6 +418,14 @@
|
|
|
221
418
|
|
|
222
419
|
@custom-variant dark (&:where(.dark, .dark *));
|
|
223
420
|
|
|
421
|
+
:root {
|
|
422
|
+
--gradient-sidebar: linear-gradient(0deg, #c9e2e280 0%, #ffffff 70%); /* Figma: bg-nav-gradiant-light */
|
|
423
|
+
}
|
|
424
|
+
|
|
425
|
+
.dark {
|
|
426
|
+
--gradient-sidebar: linear-gradient(0deg, #213535 0%, #121e1e 70%); /* Figma: bg-nav-gradiant-dark */
|
|
427
|
+
}
|
|
428
|
+
|
|
224
429
|
@layer base {
|
|
225
430
|
* {
|
|
226
431
|
@apply border-border shadow-shadow;
|
|
@@ -274,6 +479,11 @@
|
|
|
274
479
|
|
|
275
480
|
/* ─── Type preset utilities ───────────────────────────────────────────────── */
|
|
276
481
|
|
|
482
|
+
@utility bg-sidebar-surface {
|
|
483
|
+
background-color: var(--color-background);
|
|
484
|
+
background-image: var(--gradient-sidebar);
|
|
485
|
+
}
|
|
486
|
+
|
|
277
487
|
@utility type-display-2xl {
|
|
278
488
|
font-family: var(--font-display);
|
|
279
489
|
font-size: var(--font-size-display-2xl);
|