@vllnt/ui 0.1.11 → 0.2.0-canary.4abf494

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 (100) hide show
  1. package/CHANGELOG.md +104 -0
  2. package/README.md +106 -1
  3. package/dist/components/activity-heatmap/activity-heatmap.js +168 -0
  4. package/dist/components/activity-heatmap/index.js +6 -0
  5. package/dist/components/activity-log/activity-log.js +256 -0
  6. package/dist/components/activity-log/index.js +6 -0
  7. package/dist/components/ai-chat-input/ai-chat-input.js +107 -0
  8. package/dist/components/ai-chat-input/index.js +4 -0
  9. package/dist/components/ai-message-bubble/ai-message-bubble.js +119 -0
  10. package/dist/components/ai-message-bubble/index.js +6 -0
  11. package/dist/components/ai-source-citation/ai-source-citation.js +39 -0
  12. package/dist/components/ai-source-citation/index.js +6 -0
  13. package/dist/components/ai-streaming-text/ai-streaming-text.js +41 -0
  14. package/dist/components/ai-streaming-text/index.js +6 -0
  15. package/dist/components/ai-tool-call-display/ai-tool-call-display.js +93 -0
  16. package/dist/components/ai-tool-call-display/index.js +6 -0
  17. package/dist/components/animated-text/animated-text.js +328 -0
  18. package/dist/components/animated-text/index.js +4 -0
  19. package/dist/components/annotation/annotation.js +49 -0
  20. package/dist/components/annotation/index.js +8 -0
  21. package/dist/components/avatar-group/avatar-group.js +82 -0
  22. package/dist/components/avatar-group/index.js +10 -0
  23. package/dist/components/border-beam/border-beam.js +51 -0
  24. package/dist/components/border-beam/index.js +4 -0
  25. package/dist/components/candlestick-chart/candlestick-chart.js +215 -0
  26. package/dist/components/candlestick-chart/index.js +6 -0
  27. package/dist/components/combobox/combobox.js +130 -0
  28. package/dist/components/combobox/index.js +4 -0
  29. package/dist/components/countdown-timer/countdown-timer.js +184 -0
  30. package/dist/components/countdown-timer/index.js +4 -0
  31. package/dist/components/credit-badge/credit-badge.js +59 -0
  32. package/dist/components/credit-badge/index.js +6 -0
  33. package/dist/components/data-list/data-list.js +99 -0
  34. package/dist/components/data-list/index.js +16 -0
  35. package/dist/components/data-table/data-table.js +242 -0
  36. package/dist/components/data-table/index.js +6 -0
  37. package/dist/components/date-picker/date-picker.js +74 -0
  38. package/dist/components/date-picker/index.js +4 -0
  39. package/dist/components/file-upload/file-upload.js +227 -0
  40. package/dist/components/file-upload/index.js +4 -0
  41. package/dist/components/flashcard/flashcard.js +66 -0
  42. package/dist/components/flashcard/index.js +4 -0
  43. package/dist/components/index.js +172 -1
  44. package/dist/components/live-feed/index.js +4 -0
  45. package/dist/components/live-feed/live-feed.js +168 -0
  46. package/dist/components/market-treemap/index.js +6 -0
  47. package/dist/components/market-treemap/market-treemap.js +100 -0
  48. package/dist/components/marquee/index.js +4 -0
  49. package/dist/components/marquee/marquee.js +98 -0
  50. package/dist/components/metric-gauge/index.js +6 -0
  51. package/dist/components/metric-gauge/metric-gauge.js +213 -0
  52. package/dist/components/model-selector/model-selector.js +11 -2
  53. package/dist/components/number-input/index.js +4 -0
  54. package/dist/components/number-input/number-input.js +167 -0
  55. package/dist/components/number-ticker/index.js +4 -0
  56. package/dist/components/number-ticker/number-ticker.js +63 -0
  57. package/dist/components/order-book/index.js +6 -0
  58. package/dist/components/order-book/order-book.js +128 -0
  59. package/dist/components/password-input/index.js +4 -0
  60. package/dist/components/password-input/password-input.js +45 -0
  61. package/dist/components/plan-badge/index.js +6 -0
  62. package/dist/components/plan-badge/plan-badge.js +67 -0
  63. package/dist/components/rating/index.js +4 -0
  64. package/dist/components/rating/rating.js +121 -0
  65. package/dist/components/role-badge/index.js +6 -0
  66. package/dist/components/role-badge/role-badge.js +50 -0
  67. package/dist/components/scope-selector/index.js +6 -0
  68. package/dist/components/scope-selector/scope-selector.js +336 -0
  69. package/dist/components/severity-badge/index.js +8 -0
  70. package/dist/components/severity-badge/severity-badge.js +163 -0
  71. package/dist/components/sparkline-grid/index.js +6 -0
  72. package/dist/components/sparkline-grid/sparkline-grid.js +92 -0
  73. package/dist/components/spinner/index.js +5 -1
  74. package/dist/components/spinner/unicode-spinner.js +708 -0
  75. package/dist/components/stat-card/index.js +5 -0
  76. package/dist/components/stat-card/stat-card.js +102 -0
  77. package/dist/components/status-board/index.js +6 -0
  78. package/dist/components/status-board/status-board.js +138 -0
  79. package/dist/components/status-indicator/index.js +10 -0
  80. package/dist/components/status-indicator/status-indicator.js +175 -0
  81. package/dist/components/stepper/index.js +4 -0
  82. package/dist/components/stepper/stepper.js +117 -0
  83. package/dist/components/subscription-card/index.js +6 -0
  84. package/dist/components/subscription-card/subscription-card.js +161 -0
  85. package/dist/components/ticker-tape/index.js +6 -0
  86. package/dist/components/ticker-tape/ticker-tape.js +106 -0
  87. package/dist/components/tour/index.js +4 -0
  88. package/dist/components/tour/tour.js +157 -0
  89. package/dist/components/usage-breakdown/index.js +6 -0
  90. package/dist/components/usage-breakdown/usage-breakdown.js +140 -0
  91. package/dist/components/wallet-card/index.js +4 -0
  92. package/dist/components/wallet-card/wallet-card.js +115 -0
  93. package/dist/components/watchlist/index.js +6 -0
  94. package/dist/components/watchlist/watchlist.js +110 -0
  95. package/dist/components/world-clock-bar/index.js +6 -0
  96. package/dist/components/world-clock-bar/world-clock-bar.js +101 -0
  97. package/dist/index.d.ts +1173 -7
  98. package/dist/test-setup.js +19 -0
  99. package/package.json +27 -6
  100. package/styles.css +55 -0
@@ -0,0 +1,328 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import * as React from "react";
3
+ import { cn } from "../../lib/utils";
4
+ const GLYPH_SEGMENTER = new Intl.Segmenter(void 0, {
5
+ granularity: "grapheme"
6
+ });
7
+ const ASCII_RANDOM_CHARACTERS = Array.from(
8
+ { length: 94 },
9
+ (_, index) => String.fromCodePoint(index + 33)
10
+ ).join("");
11
+ const TERMINAL_RANDOM_CHARACTERS = "\u2502\u2503\u2500\u2501\u2504\u2505\u2508\u2509\u250C\u2510\u2514\u2518\u251C\u2524\u252C\u2534\u253C\u256D\u256E\u256F\u2570\u2571\u2572\u2573";
12
+ const BLOCK_RANDOM_CHARACTERS = "\u2591\u2592\u2593\u2588\u258C\u2590\u2580\u2584\u25A0\u25A1\u25AA\u25AB\u2596\u2597\u2598\u2599\u259A\u259B\u259C\u259D\u259E\u259F";
13
+ const UNICODE_SYMBOL_RANDOM_CHARACTERS = "\u25C6\u25C7\u25C8\u25CB\u25CF\u25CE\u25C9\u25CC\u25CD\u25D0\u25D1\u25D2\u25D3\u25D4\u25D5\u25E2\u25E3\u25E4\u25E5\u25E6\u203B\u2726\u2727\u2731\u2736\u2737\u2739";
14
+ const MATRIX_RANDOM_CHARACTERS = `${ASCII_RANDOM_CHARACTERS}${TERMINAL_RANDOM_CHARACTERS}${BLOCK_RANDOM_CHARACTERS}${UNICODE_SYMBOL_RANDOM_CHARACTERS}`;
15
+ const ANIMATED_TEXT_RANDOM_CHARACTER_PRESETS = {
16
+ ascii: ASCII_RANDOM_CHARACTERS,
17
+ binary: "01",
18
+ blocks: BLOCK_RANDOM_CHARACTERS,
19
+ matrix: MATRIX_RANDOM_CHARACTERS,
20
+ symbols: UNICODE_SYMBOL_RANDOM_CHARACTERS,
21
+ terminal: TERMINAL_RANDOM_CHARACTERS
22
+ };
23
+ const DEFAULT_RANDOM_CHARACTERS = ANIMATED_TEXT_RANDOM_CHARACTER_PRESETS.matrix;
24
+ function getSegments(text, splitBy) {
25
+ if (splitBy === "character") {
26
+ const segmenter = new Intl.Segmenter(void 0, {
27
+ granularity: "grapheme"
28
+ });
29
+ return Array.from(segmenter.segment(text), ({ segment }) => segment);
30
+ }
31
+ return text.match(/\S+\s*/g) ?? [];
32
+ }
33
+ function getGlyphs(text) {
34
+ return Array.from(GLYPH_SEGMENTER.segment(text), ({ segment }) => segment);
35
+ }
36
+ function getRandomMatrixGlyph(randomCharacters) {
37
+ const glyphs = getGlyphs(randomCharacters);
38
+ return glyphs[Math.floor(Math.random() * glyphs.length)] ?? glyphs[0] ?? "0";
39
+ }
40
+ function getResolvedRandomCharacters(randomCharacters, randomCharactersPreset) {
41
+ if (randomCharacters && randomCharacters.length > 0) {
42
+ return randomCharacters;
43
+ }
44
+ return ANIMATED_TEXT_RANDOM_CHARACTER_PRESETS[randomCharactersPreset] ?? DEFAULT_RANDOM_CHARACTERS;
45
+ }
46
+ function getCursorToneClass(variant) {
47
+ return variant === "matrix" || variant === "decipher" ? "text-primary" : "text-foreground";
48
+ }
49
+ function buildRevealFrames(segments, stagger) {
50
+ return segments.map((segment, index) => ({
51
+ content: segment,
52
+ isRevealed: true,
53
+ key: `${segment}-${index}`,
54
+ style: {
55
+ animationDelay: `${index * stagger}ms`
56
+ }
57
+ }));
58
+ }
59
+ function buildIndexOrder(direction, length) {
60
+ if (direction === "end") {
61
+ return Array.from({ length }, (_, index) => length - index - 1);
62
+ }
63
+ if (direction === "random") {
64
+ return Array.from({ length }, (_, index) => index).sort(
65
+ () => Math.random() - 0.5
66
+ );
67
+ }
68
+ if (direction === "center-out") {
69
+ const center = (length - 1) / 2;
70
+ return Array.from({ length }, (_, index) => index).sort((left, right) => {
71
+ const leftDistance = Math.abs(left - center);
72
+ const rightDistance = Math.abs(right - center);
73
+ if (leftDistance === rightDistance) {
74
+ return left - right;
75
+ }
76
+ return leftDistance - rightDistance;
77
+ });
78
+ }
79
+ return Array.from({ length }, (_, index) => index);
80
+ }
81
+ function buildRevealPlan(direction, length, randomness) {
82
+ const orderedIndices = buildIndexOrder(direction, length);
83
+ const revealPlan = Array.from({ length }, () => 0);
84
+ const jitterRange = Math.max(0, Math.round(randomness * 4));
85
+ orderedIndices.forEach((segmentIndex, revealIndex) => {
86
+ const jitter = jitterRange > 0 ? Math.floor(Math.random() * (jitterRange + 1)) : 0;
87
+ revealPlan[segmentIndex] = revealIndex + jitter;
88
+ });
89
+ return revealPlan;
90
+ }
91
+ function useRevealProgress(active, length, stagger) {
92
+ const [progress, setProgress] = React.useState(0);
93
+ React.useEffect(() => {
94
+ if (!active) {
95
+ setProgress(length);
96
+ return;
97
+ }
98
+ setProgress(0);
99
+ const revealInterval = window.setInterval(
100
+ () => {
101
+ setProgress((current) => {
102
+ if (current >= length + 4) {
103
+ window.clearInterval(revealInterval);
104
+ return current;
105
+ }
106
+ return current + 1;
107
+ });
108
+ },
109
+ Math.max(16, stagger)
110
+ );
111
+ return () => {
112
+ window.clearInterval(revealInterval);
113
+ };
114
+ }, [active, length, stagger]);
115
+ return progress;
116
+ }
117
+ function useMatrixFrame({
118
+ active,
119
+ progress,
120
+ randomCharacters,
121
+ revealPlan,
122
+ segments
123
+ }) {
124
+ const [matrixFrame, setMatrixFrame] = React.useState(
125
+ () => segments.map(() => getRandomMatrixGlyph(randomCharacters))
126
+ );
127
+ React.useEffect(() => {
128
+ if (!active) {
129
+ return;
130
+ }
131
+ setMatrixFrame(segments.map(() => getRandomMatrixGlyph(randomCharacters)));
132
+ const scrambleInterval = window.setInterval(() => {
133
+ setMatrixFrame(
134
+ (current) => current.map((glyph, index) => {
135
+ const isWhitespace = /^\s+$/.test(segments[index] ?? "");
136
+ const isRevealed = progress >= (revealPlan[index] ?? 0);
137
+ if (isWhitespace || isRevealed) {
138
+ return glyph;
139
+ }
140
+ return getRandomMatrixGlyph(randomCharacters);
141
+ })
142
+ );
143
+ }, 48);
144
+ return () => {
145
+ window.clearInterval(scrambleInterval);
146
+ };
147
+ }, [active, progress, randomCharacters, revealPlan, segments]);
148
+ return matrixFrame;
149
+ }
150
+ function buildOldSchoolFrames({
151
+ matrixFrame,
152
+ progress,
153
+ randomCharacters,
154
+ revealPlan,
155
+ segments,
156
+ variant
157
+ }) {
158
+ return segments.map((segment, index) => {
159
+ const isWhitespace = /^\s+$/.test(segment);
160
+ const revealStep = revealPlan[index] ?? 0;
161
+ const isRevealed = progress >= revealStep;
162
+ let content = "";
163
+ if (variant === "matrix" || variant === "decipher") {
164
+ content = isWhitespace ? segment : isRevealed ? segment : matrixFrame[index] ?? getRandomMatrixGlyph(randomCharacters);
165
+ } else if (isRevealed) {
166
+ content = segment;
167
+ }
168
+ return {
169
+ content,
170
+ isRevealed,
171
+ key: `${segment}-${index}`
172
+ };
173
+ });
174
+ }
175
+ function useAnimatedTextFrames({
176
+ direction,
177
+ randomCharacters,
178
+ randomness,
179
+ segments,
180
+ stagger,
181
+ variant
182
+ }) {
183
+ const isOldSchool = variant !== "reveal";
184
+ const revealPlan = React.useMemo(
185
+ () => isOldSchool ? buildRevealPlan(direction, segments.length, randomness) : Array.from({ length: segments.length }, (_, index) => index),
186
+ [direction, isOldSchool, randomness, segments.length]
187
+ );
188
+ const progress = useRevealProgress(isOldSchool, segments.length, stagger);
189
+ const matrixFrame = useMatrixFrame({
190
+ active: variant === "matrix" || variant === "decipher",
191
+ progress,
192
+ randomCharacters,
193
+ revealPlan,
194
+ segments
195
+ });
196
+ return React.useMemo(() => {
197
+ if (!isOldSchool) {
198
+ return buildRevealFrames(segments, stagger);
199
+ }
200
+ return buildOldSchoolFrames({
201
+ matrixFrame,
202
+ progress,
203
+ randomCharacters,
204
+ revealPlan,
205
+ segments,
206
+ variant
207
+ });
208
+ }, [
209
+ isOldSchool,
210
+ matrixFrame,
211
+ progress,
212
+ randomCharacters,
213
+ revealPlan,
214
+ segments,
215
+ stagger,
216
+ variant
217
+ ]);
218
+ }
219
+ function getSegmentClasses(variant, isRevealed) {
220
+ if (variant === "reveal") {
221
+ return "inline-block whitespace-pre opacity-0 [animation-duration:var(--vllnt-animated-text-duration)] [animation-fill-mode:forwards] [animation-name:vllnt-animated-text-reveal] [animation-timing-function:cubic-bezier(0.16,1,0.3,1)]";
222
+ }
223
+ if (variant === "matrix" || variant === "decipher") {
224
+ return cn(
225
+ "inline-block whitespace-pre font-mono tracking-[0.08em] transition-colors duration-150",
226
+ isRevealed ? "text-foreground" : "text-primary/75"
227
+ );
228
+ }
229
+ return "inline-block whitespace-pre font-mono";
230
+ }
231
+ function getContainerClasses(variant) {
232
+ if (variant === "matrix" || variant === "decipher") {
233
+ return "flex flex-wrap font-mono leading-relaxed tracking-[0.08em]";
234
+ }
235
+ if (variant === "terminal" || variant === "typewriter") {
236
+ return "flex flex-wrap font-mono leading-relaxed";
237
+ }
238
+ return "flex flex-wrap leading-relaxed";
239
+ }
240
+ function AnimatedTextCursor({
241
+ cursorChar,
242
+ cursorToneClass
243
+ }) {
244
+ return /* @__PURE__ */ jsx(
245
+ "span",
246
+ {
247
+ "aria-hidden": "true",
248
+ className: cn(
249
+ "ml-0.5 inline-block whitespace-pre font-mono [animation:vllnt-terminal-cursor-blink_1s_steps(1,end)_infinite]",
250
+ cursorToneClass
251
+ ),
252
+ children: cursorChar
253
+ }
254
+ );
255
+ }
256
+ const AnimatedText = React.forwardRef(
257
+ ({
258
+ className,
259
+ cursor = true,
260
+ cursorChar = "\u2588",
261
+ direction = "start",
262
+ duration = 600,
263
+ randomCharacters,
264
+ randomCharactersPreset = "matrix",
265
+ randomness = 0,
266
+ splitBy = "word",
267
+ stagger = 70,
268
+ text,
269
+ variant = "terminal",
270
+ ...props
271
+ }, ref) => {
272
+ const resolvedRandomCharacters = getResolvedRandomCharacters(
273
+ randomCharacters,
274
+ randomCharactersPreset
275
+ );
276
+ const resolvedSplitBy = variant === "reveal" ? splitBy : "character";
277
+ const segments = React.useMemo(
278
+ () => getSegments(text, resolvedSplitBy),
279
+ [resolvedSplitBy, text]
280
+ );
281
+ const segmentFrames = useAnimatedTextFrames({
282
+ direction,
283
+ randomCharacters: resolvedRandomCharacters,
284
+ randomness,
285
+ segments,
286
+ stagger,
287
+ variant
288
+ });
289
+ const showCursor = cursor && variant !== "reveal" && segmentFrames.some((frame) => !frame.isRevealed);
290
+ const cursorToneClass = getCursorToneClass(variant);
291
+ return /* @__PURE__ */ jsxs(
292
+ "p",
293
+ {
294
+ "aria-label": text,
295
+ className: cn(getContainerClasses(variant), className),
296
+ ref,
297
+ style: {
298
+ ["--vllnt-animated-text-duration"]: `${duration}ms`
299
+ },
300
+ ...props,
301
+ children: [
302
+ segmentFrames.map((segmentFrame) => /* @__PURE__ */ jsx(
303
+ "span",
304
+ {
305
+ "aria-hidden": "true",
306
+ className: getSegmentClasses(variant, segmentFrame.isRevealed),
307
+ style: segmentFrame.style,
308
+ children: segmentFrame.content
309
+ },
310
+ segmentFrame.key
311
+ )),
312
+ showCursor ? /* @__PURE__ */ jsx(
313
+ AnimatedTextCursor,
314
+ {
315
+ cursorChar,
316
+ cursorToneClass
317
+ }
318
+ ) : null
319
+ ]
320
+ }
321
+ );
322
+ }
323
+ );
324
+ AnimatedText.displayName = "AnimatedText";
325
+ export {
326
+ ANIMATED_TEXT_RANDOM_CHARACTER_PRESETS,
327
+ AnimatedText
328
+ };
@@ -0,0 +1,4 @@
1
+ import { AnimatedText } from "./animated-text";
2
+ export {
3
+ AnimatedText
4
+ };
@@ -0,0 +1,49 @@
1
+ "use client";
2
+ import { jsx, jsxs } from "react/jsx-runtime";
3
+ import { cn } from "../../lib/utils";
4
+ import { Popover, PopoverContent, PopoverTrigger } from "../popover";
5
+ const toneClasses = {
6
+ amber: "bg-amber-500/20 text-amber-950 dark:text-amber-100",
7
+ emerald: "bg-emerald-500/20 text-emerald-950 dark:text-emerald-100",
8
+ rose: "bg-rose-500/20 text-rose-950 dark:text-rose-100",
9
+ sky: "bg-sky-500/20 text-sky-950 dark:text-sky-100"
10
+ };
11
+ function Highlight({
12
+ children,
13
+ className,
14
+ tone = "amber"
15
+ }) {
16
+ return /* @__PURE__ */ jsx("mark", { className: cn("rounded px-1 py-0.5", toneClasses[tone], className), children });
17
+ }
18
+ function Annotation({
19
+ annotation,
20
+ children,
21
+ className,
22
+ label = "Annotation",
23
+ tone = "amber"
24
+ }) {
25
+ return /* @__PURE__ */ jsxs(Popover, { children: [
26
+ /* @__PURE__ */ jsx(PopoverTrigger, { asChild: true, children: /* @__PURE__ */ jsxs(
27
+ "button",
28
+ {
29
+ className: cn(
30
+ "inline-flex items-center gap-1 rounded-sm text-left focus-visible:outline-none focus-visible:ring-2 focus-visible:ring-ring",
31
+ className
32
+ ),
33
+ type: "button",
34
+ children: [
35
+ /* @__PURE__ */ jsx(Highlight, { tone, children }),
36
+ /* @__PURE__ */ jsx("span", { className: "text-[10px] font-semibold text-primary align-super", children: "+" })
37
+ ]
38
+ }
39
+ ) }),
40
+ /* @__PURE__ */ jsxs(PopoverContent, { align: "start", className: "max-w-xs space-y-2", children: [
41
+ /* @__PURE__ */ jsx("p", { className: "text-xs font-semibold uppercase tracking-[0.18em] text-muted-foreground", children: label }),
42
+ /* @__PURE__ */ jsx("div", { className: "text-sm text-muted-foreground [&>p]:mb-2", children: annotation })
43
+ ] })
44
+ ] });
45
+ }
46
+ export {
47
+ Annotation,
48
+ Highlight
49
+ };
@@ -0,0 +1,8 @@
1
+ import {
2
+ Annotation,
3
+ Highlight
4
+ } from "./annotation";
5
+ export {
6
+ Annotation,
7
+ Highlight
8
+ };
@@ -0,0 +1,82 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import * as React from "react";
3
+ import { cva } from "class-variance-authority";
4
+ import { cn } from "../../lib/utils";
5
+ import { Avatar, AvatarFallback, AvatarImage } from "../avatar";
6
+ const avatarGroupVariants = cva("flex items-center", {
7
+ defaultVariants: {
8
+ size: "md"
9
+ },
10
+ variants: {
11
+ size: {
12
+ lg: "-space-x-4",
13
+ md: "-space-x-3",
14
+ sm: "-space-x-2.5"
15
+ }
16
+ }
17
+ });
18
+ const avatarItemVariants = cva(
19
+ "ring-2 ring-background border border-border bg-muted text-muted-foreground",
20
+ {
21
+ defaultVariants: {
22
+ size: "md"
23
+ },
24
+ variants: {
25
+ size: {
26
+ lg: "h-12 w-12 text-sm",
27
+ md: "h-10 w-10 text-xs",
28
+ sm: "h-8 w-8 text-[11px]"
29
+ }
30
+ }
31
+ }
32
+ );
33
+ const overflowBadgeVariants = cva(
34
+ "inline-flex shrink-0 items-center justify-center rounded-full border border-border bg-muted font-medium text-muted-foreground ring-2 ring-background",
35
+ {
36
+ defaultVariants: {
37
+ size: "md"
38
+ },
39
+ variants: {
40
+ size: {
41
+ lg: "h-12 min-w-12 px-3 text-sm",
42
+ md: "h-10 min-w-10 px-2.5 text-xs",
43
+ sm: "h-8 min-w-8 px-2 text-[11px]"
44
+ }
45
+ }
46
+ }
47
+ );
48
+ const AvatarGroup = React.forwardRef(
49
+ ({ className, items, max, overflowLabel, size, ...props }, reference) => {
50
+ const visibleItems = max === void 0 ? items : items.slice(0, max);
51
+ const hiddenCount = max === void 0 ? 0 : Math.max(items.length - max, 0);
52
+ return /* @__PURE__ */ jsxs(
53
+ "div",
54
+ {
55
+ className: cn(avatarGroupVariants({ size }), className),
56
+ ref: reference,
57
+ ...props,
58
+ children: [
59
+ visibleItems.map((item, index) => /* @__PURE__ */ jsxs(
60
+ Avatar,
61
+ {
62
+ className: avatarItemVariants({ size }),
63
+ style: { zIndex: visibleItems.length - index },
64
+ children: [
65
+ item.src ? /* @__PURE__ */ jsx(AvatarImage, { alt: item.alt, src: item.src }) : null,
66
+ /* @__PURE__ */ jsx(AvatarFallback, { children: item.fallback })
67
+ ]
68
+ },
69
+ `${item.alt}-${index}`
70
+ )),
71
+ hiddenCount > 0 ? /* @__PURE__ */ jsx("span", { className: overflowBadgeVariants({ size }), children: overflowLabel ? overflowLabel(hiddenCount) : `+${hiddenCount}` }) : null
72
+ ]
73
+ }
74
+ );
75
+ }
76
+ );
77
+ AvatarGroup.displayName = "AvatarGroup";
78
+ export {
79
+ AvatarGroup,
80
+ avatarGroupVariants,
81
+ avatarItemVariants
82
+ };
@@ -0,0 +1,10 @@
1
+ import {
2
+ AvatarGroup,
3
+ avatarGroupVariants,
4
+ avatarItemVariants
5
+ } from "./avatar-group";
6
+ export {
7
+ AvatarGroup,
8
+ avatarGroupVariants,
9
+ avatarItemVariants
10
+ };
@@ -0,0 +1,51 @@
1
+ import { jsx } from "react/jsx-runtime";
2
+ import * as React from "react";
3
+ import { cn } from "../../lib/utils";
4
+ const BorderBeam = React.forwardRef(
5
+ ({
6
+ borderWidth = 1,
7
+ className,
8
+ colorFrom = "hsl(var(--primary) / 0.85)",
9
+ colorTo = "hsl(var(--ring) / 0.25)",
10
+ delay = 0,
11
+ duration = 6,
12
+ reverse = false,
13
+ style,
14
+ ...props
15
+ }, ref) => {
16
+ const beamStyle = {
17
+ animationDelay: `${delay}s`,
18
+ animationDirection: reverse ? "reverse" : "normal",
19
+ animationDuration: `${duration}s`,
20
+ animationIterationCount: "infinite",
21
+ animationName: "vllnt-border-beam-angle",
22
+ animationTimingFunction: "linear",
23
+ background: `conic-gradient(from var(--vllnt-border-beam-angle, 90deg), transparent 0deg, transparent 220deg, ${colorFrom} 280deg, ${colorTo} 335deg, transparent 360deg)`,
24
+ borderRadius: "inherit",
25
+ boxSizing: "border-box",
26
+ mask: "linear-gradient(#fff 0 0) content-box, linear-gradient(#fff 0 0)",
27
+ maskComposite: "exclude",
28
+ padding: `${borderWidth}px`,
29
+ WebkitMask: "linear-gradient(#fff 0 0) content-box, linear-gradient(#fff 0 0)",
30
+ WebkitMaskComposite: "xor",
31
+ ...style
32
+ };
33
+ return /* @__PURE__ */ jsx(
34
+ "span",
35
+ {
36
+ "aria-hidden": "true",
37
+ className: cn(
38
+ "pointer-events-none absolute inset-0 rounded-[inherit]",
39
+ className
40
+ ),
41
+ ref,
42
+ style: beamStyle,
43
+ ...props
44
+ }
45
+ );
46
+ }
47
+ );
48
+ BorderBeam.displayName = "BorderBeam";
49
+ export {
50
+ BorderBeam
51
+ };
@@ -0,0 +1,4 @@
1
+ import { BorderBeam } from "./border-beam";
2
+ export {
3
+ BorderBeam
4
+ };