@vllnt/ui 0.2.0 → 0.2.1-canary.73a93ee

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 (86) hide show
  1. package/CHANGELOG.md +12 -1
  2. package/README.md +27 -12
  3. package/dist/components/activity-log/activity-log.js +1 -0
  4. package/dist/components/anchor-port/anchor-port.js +51 -0
  5. package/dist/components/anchor-port/index.js +4 -0
  6. package/dist/components/animated-text/animated-text.js +1 -0
  7. package/dist/components/bottom-bar/bottom-bar.js +25 -0
  8. package/dist/components/bottom-bar/index.js +4 -0
  9. package/dist/components/canvas-shell/canvas-foundation-demo.js +183 -0
  10. package/dist/components/canvas-shell/canvas-shell-route-config.js +0 -0
  11. package/dist/components/canvas-shell/canvas-shell.js +261 -0
  12. package/dist/components/canvas-shell/index.js +4 -0
  13. package/dist/components/canvas-view/canvas-view.js +461 -0
  14. package/dist/components/canvas-view/index.js +6 -0
  15. package/dist/components/chart/area-chart.js +1 -0
  16. package/dist/components/chart/line-chart.js +1 -0
  17. package/dist/components/chat-dock-section/chat-dock-section.js +56 -0
  18. package/dist/components/chat-dock-section/index.js +6 -0
  19. package/dist/components/checklist/checklist.js +7 -0
  20. package/dist/components/checklist/index.js +3 -1
  21. package/dist/components/comment-pin/comment-pin.js +104 -0
  22. package/dist/components/comment-pin/index.js +6 -0
  23. package/dist/components/connector-edge/connector-edge.js +66 -0
  24. package/dist/components/connector-edge/index.js +6 -0
  25. package/dist/components/conversation-thread/conversation-thread.js +348 -0
  26. package/dist/components/conversation-thread/index.js +20 -0
  27. package/dist/components/curriculum/curriculum.js +349 -0
  28. package/dist/components/curriculum/index.js +10 -0
  29. package/dist/components/data-list/data-list.js +1 -0
  30. package/dist/components/edge-label/edge-label.js +26 -0
  31. package/dist/components/edge-label/index.js +4 -0
  32. package/dist/components/form/form.js +432 -0
  33. package/dist/components/form/index.js +20 -0
  34. package/dist/components/glass-panel/glass-panel.js +21 -0
  35. package/dist/components/glass-panel/index.js +4 -0
  36. package/dist/components/group-hull/group-hull.js +29 -0
  37. package/dist/components/group-hull/index.js +4 -0
  38. package/dist/components/index.js +176 -0
  39. package/dist/components/infinite-plane/index.js +6 -0
  40. package/dist/components/infinite-plane/infinite-plane.js +75 -0
  41. package/dist/components/left-rail/index.js +4 -0
  42. package/dist/components/left-rail/left-rail.js +25 -0
  43. package/dist/components/live-cursor/index.js +6 -0
  44. package/dist/components/live-cursor/live-cursor.js +62 -0
  45. package/dist/components/mini-map-panel/index.js +6 -0
  46. package/dist/components/mini-map-panel/mini-map-panel.js +74 -0
  47. package/dist/components/multi-select/index.js +6 -0
  48. package/dist/components/multi-select/multi-select.js +258 -0
  49. package/dist/components/object-card/index.js +6 -0
  50. package/dist/components/object-card/object-card.js +126 -0
  51. package/dist/components/object-handle/index.js +4 -0
  52. package/dist/components/object-handle/object-handle.js +38 -0
  53. package/dist/components/overview-board/index.js +8 -0
  54. package/dist/components/overview-board/overview-board.js +127 -0
  55. package/dist/components/presence-stack/index.js +6 -0
  56. package/dist/components/presence-stack/presence-stack.js +108 -0
  57. package/dist/components/presence-sync-indicator/index.js +6 -0
  58. package/dist/components/presence-sync-indicator/presence-sync-indicator.js +73 -0
  59. package/dist/components/progress-tracker/index.js +20 -0
  60. package/dist/components/progress-tracker/progress-tracker.js +527 -0
  61. package/dist/components/right-dock/index.js +4 -0
  62. package/dist/components/right-dock/right-dock.js +28 -0
  63. package/dist/components/run-timeline/index.js +6 -0
  64. package/dist/components/run-timeline/run-timeline.js +221 -0
  65. package/dist/components/segmented-control/index.js +12 -0
  66. package/dist/components/segmented-control/segmented-control.js +61 -0
  67. package/dist/components/selection-presence/index.js +6 -0
  68. package/dist/components/selection-presence/selection-presence.js +50 -0
  69. package/dist/components/spinner/unicode-spinner.js +1 -0
  70. package/dist/components/tags-input/index.js +4 -0
  71. package/dist/components/tags-input/tags-input.js +178 -0
  72. package/dist/components/thread-bubble/index.js +6 -0
  73. package/dist/components/thread-bubble/thread-bubble.js +85 -0
  74. package/dist/components/top-bar/index.js +4 -0
  75. package/dist/components/top-bar/top-bar.js +31 -0
  76. package/dist/components/usage-breakdown/usage-breakdown.js +1 -0
  77. package/dist/components/viewport-bookmarks/index.js +6 -0
  78. package/dist/components/viewport-bookmarks/viewport-bookmarks.js +116 -0
  79. package/dist/components/workspace-switcher/index.js +6 -0
  80. package/dist/components/workspace-switcher/workspace-switcher.js +61 -0
  81. package/dist/components/world-breadcrumbs/index.js +6 -0
  82. package/dist/components/world-breadcrumbs/world-breadcrumbs.js +114 -0
  83. package/dist/components/zoom-hud/index.js +4 -0
  84. package/dist/components/zoom-hud/zoom-hud.js +61 -0
  85. package/dist/index.d.ts +1468 -6
  86. package/package.json +7 -3
@@ -0,0 +1,66 @@
1
+ import { jsx, jsxs } from "react/jsx-runtime";
2
+ import { forwardRef } from "react";
3
+ import { cn } from "../../lib/utils";
4
+ import { EdgeLabel } from "../edge-label";
5
+ const strokeClasses = {
6
+ active: "stroke-sky-500",
7
+ blocked: "stroke-amber-500",
8
+ idle: "stroke-muted-foreground/60"
9
+ };
10
+ const ConnectorEdge = forwardRef(
11
+ ({ className, end, label, start, state = "idle", ...props }, ref) => {
12
+ const width = Math.max(Math.abs(end.x - start.x), 32);
13
+ const height = Math.max(Math.abs(end.y - start.y), 32);
14
+ const midX = width / 2;
15
+ const startX = start.x <= end.x ? 4 : width - 4;
16
+ const endX = start.x <= end.x ? width - 4 : 4;
17
+ const startY = start.y <= end.y ? 4 : height - 4;
18
+ const endY = start.y <= end.y ? height - 4 : 4;
19
+ const path = `M ${startX} ${startY} C ${midX} ${startY}, ${midX} ${endY}, ${endX} ${endY}`;
20
+ const style = {
21
+ height,
22
+ width
23
+ };
24
+ return /* @__PURE__ */ jsxs(
25
+ "div",
26
+ {
27
+ className: cn("relative inline-flex", className),
28
+ ref,
29
+ style,
30
+ ...props,
31
+ children: [
32
+ /* @__PURE__ */ jsx(
33
+ "svg",
34
+ {
35
+ className: "overflow-visible",
36
+ height,
37
+ viewBox: `0 0 ${width} ${height}`,
38
+ width,
39
+ children: /* @__PURE__ */ jsx(
40
+ "path",
41
+ {
42
+ className: cn("fill-none stroke-[2.5]", strokeClasses[state]),
43
+ d: path,
44
+ strokeDasharray: state === "blocked" ? "4 4" : void 0,
45
+ strokeLinecap: "round"
46
+ }
47
+ )
48
+ }
49
+ ),
50
+ label ? /* @__PURE__ */ jsx(
51
+ EdgeLabel,
52
+ {
53
+ className: "absolute left-1/2 top-1/2 -translate-x-1/2 -translate-y-1/2",
54
+ emphasis: state === "active" ? "active" : "subtle",
55
+ children: label
56
+ }
57
+ ) : null
58
+ ]
59
+ }
60
+ );
61
+ }
62
+ );
63
+ ConnectorEdge.displayName = "ConnectorEdge";
64
+ export {
65
+ ConnectorEdge
66
+ };
@@ -0,0 +1,6 @@
1
+ import {
2
+ ConnectorEdge
3
+ } from "./connector-edge";
4
+ export {
5
+ ConnectorEdge
6
+ };
@@ -0,0 +1,348 @@
1
+ "use client";
2
+ import { Fragment, jsx, jsxs } from "react/jsx-runtime";
3
+ import {
4
+ createContext,
5
+ forwardRef,
6
+ useCallback,
7
+ useContext,
8
+ useEffect,
9
+ useMemo,
10
+ useRef,
11
+ useState
12
+ } from "react";
13
+ import { ArrowDown, RefreshCw, ThumbsDown, ThumbsUp } from "lucide-react";
14
+ import { cn } from "../../lib/utils";
15
+ import { ThinkingBlock } from "../thinking-block";
16
+ const ConversationThreadContext = createContext(null);
17
+ function useConversationThreadContext() {
18
+ const ctx = useContext(ConversationThreadContext);
19
+ if (!ctx) {
20
+ throw new Error(
21
+ "ConversationThread compound components must be used within <ConversationThread>"
22
+ );
23
+ }
24
+ return ctx;
25
+ }
26
+ function MessageActions({ messageId }) {
27
+ const { onFeedback, onRetry } = useConversationThreadContext();
28
+ return /* @__PURE__ */ jsxs("div", { className: "mt-2 flex items-center gap-1", children: [
29
+ onRetry ? /* @__PURE__ */ jsx(
30
+ "button",
31
+ {
32
+ "aria-label": "Retry message",
33
+ className: "rounded p-1 text-muted-foreground transition-colors hover:bg-muted hover:text-foreground",
34
+ onClick: () => {
35
+ onRetry(messageId);
36
+ },
37
+ type: "button",
38
+ children: /* @__PURE__ */ jsx(RefreshCw, { className: "h-3 w-3" })
39
+ }
40
+ ) : null,
41
+ onFeedback ? /* @__PURE__ */ jsxs(Fragment, { children: [
42
+ /* @__PURE__ */ jsx(
43
+ "button",
44
+ {
45
+ "aria-label": "Positive feedback",
46
+ className: "rounded p-1 text-muted-foreground transition-colors hover:bg-muted hover:text-foreground",
47
+ onClick: () => {
48
+ onFeedback(messageId, "positive");
49
+ },
50
+ type: "button",
51
+ children: /* @__PURE__ */ jsx(ThumbsUp, { className: "h-3 w-3" })
52
+ }
53
+ ),
54
+ /* @__PURE__ */ jsx(
55
+ "button",
56
+ {
57
+ "aria-label": "Negative feedback",
58
+ className: "rounded p-1 text-muted-foreground transition-colors hover:bg-muted hover:text-foreground",
59
+ onClick: () => {
60
+ onFeedback(messageId, "negative");
61
+ },
62
+ type: "button",
63
+ children: /* @__PURE__ */ jsx(ThumbsDown, { className: "h-3 w-3" })
64
+ }
65
+ )
66
+ ] }) : null
67
+ ] });
68
+ }
69
+ function MessageItem({ message }) {
70
+ const isUser = message.role === "user";
71
+ return /* @__PURE__ */ jsx(
72
+ "div",
73
+ {
74
+ className: cn(
75
+ "mb-4 flex gap-3",
76
+ isUser ? "justify-end" : "justify-start"
77
+ ),
78
+ children: /* @__PURE__ */ jsxs(
79
+ "div",
80
+ {
81
+ className: cn(
82
+ "max-w-[80%] rounded-2xl px-4 py-3 text-sm",
83
+ isUser ? "rounded-br-sm bg-primary text-primary-foreground" : "rounded-bl-sm bg-muted text-foreground"
84
+ ),
85
+ children: [
86
+ !isUser && message.thinking ? /* @__PURE__ */ jsx(
87
+ ThinkingBlock,
88
+ {
89
+ isStreaming: message.isStreaming,
90
+ thinking: message.thinking
91
+ }
92
+ ) : null,
93
+ message.toolCalls && message.toolCalls.length > 0 ? /* @__PURE__ */ jsx(
94
+ "ul",
95
+ {
96
+ "aria-label": "Tool calls",
97
+ className: "mb-2 flex flex-col gap-1 text-xs text-muted-foreground",
98
+ children: message.toolCalls.map((toolCall) => /* @__PURE__ */ jsx("li", { className: "font-mono", children: toolCall.name }, toolCall.id))
99
+ }
100
+ ) : null,
101
+ /* @__PURE__ */ jsx("p", { className: "whitespace-pre-wrap leading-relaxed", children: message.content }),
102
+ isUser ? null : /* @__PURE__ */ jsx(MessageActions, { messageId: message.id })
103
+ ]
104
+ }
105
+ )
106
+ }
107
+ );
108
+ }
109
+ function useConversationScroll(messages, isStreaming) {
110
+ const scrollContainerRef = useRef(null);
111
+ const messagesEndRef = useRef(null);
112
+ const isAtBottomRef = useRef(true);
113
+ const [isAtBottom, setIsAtBottom] = useState(true);
114
+ const scrollToBottom = useCallback(() => {
115
+ const element = messagesEndRef.current;
116
+ if (element && typeof element.scrollIntoView === "function") {
117
+ element.scrollIntoView({ behavior: "smooth" });
118
+ }
119
+ }, []);
120
+ const scrollToBottomInstant = useCallback(() => {
121
+ const element = messagesEndRef.current;
122
+ if (element && typeof element.scrollIntoView === "function") {
123
+ element.scrollIntoView({ behavior: "instant" });
124
+ }
125
+ }, []);
126
+ const handleScroll = useCallback(() => {
127
+ const container = scrollContainerRef.current;
128
+ if (!container) return;
129
+ const { clientHeight, scrollHeight, scrollTop } = container;
130
+ const nearBottom = scrollHeight - scrollTop - clientHeight <= 100;
131
+ isAtBottomRef.current = nearBottom;
132
+ setIsAtBottom(nearBottom);
133
+ }, []);
134
+ useEffect(() => {
135
+ if (!isAtBottomRef.current) return;
136
+ scrollToBottomInstant();
137
+ }, [messages, scrollToBottomInstant]);
138
+ useEffect(() => {
139
+ if (!isStreaming || !isAtBottomRef.current) return;
140
+ scrollToBottomInstant();
141
+ }, [isStreaming, scrollToBottomInstant]);
142
+ return {
143
+ handleScroll,
144
+ isAtBottom,
145
+ messagesEndRef,
146
+ scrollContainerRef,
147
+ scrollToBottom
148
+ };
149
+ }
150
+ const ConversationThread = forwardRef(
151
+ ({
152
+ children,
153
+ className,
154
+ isStreaming = false,
155
+ messages,
156
+ onFeedback,
157
+ onRetry,
158
+ onSend
159
+ }, reference) => {
160
+ const {
161
+ handleScroll,
162
+ isAtBottom,
163
+ messagesEndRef,
164
+ scrollContainerRef,
165
+ scrollToBottom
166
+ } = useConversationScroll(messages, isStreaming);
167
+ const contextValue = useMemo(
168
+ () => ({
169
+ handleScroll,
170
+ isAtBottom,
171
+ isStreaming,
172
+ messages,
173
+ messagesEndRef,
174
+ onFeedback,
175
+ onRetry,
176
+ onSend,
177
+ scrollContainerRef,
178
+ scrollToBottom
179
+ }),
180
+ [
181
+ handleScroll,
182
+ isAtBottom,
183
+ isStreaming,
184
+ messages,
185
+ messagesEndRef,
186
+ onFeedback,
187
+ onRetry,
188
+ onSend,
189
+ scrollContainerRef,
190
+ scrollToBottom
191
+ ]
192
+ );
193
+ return /* @__PURE__ */ jsx(ConversationThreadContext.Provider, { value: contextValue, children: /* @__PURE__ */ jsx(
194
+ "div",
195
+ {
196
+ className: cn("flex h-full flex-col overflow-hidden", className),
197
+ ref: reference,
198
+ children
199
+ }
200
+ ) });
201
+ }
202
+ );
203
+ ConversationThread.displayName = "ConversationThread";
204
+ const ConversationHeader = forwardRef(({ children, className }, reference) => {
205
+ return /* @__PURE__ */ jsx(
206
+ "div",
207
+ {
208
+ className: cn("flex shrink-0 items-center border-b px-4 py-3", className),
209
+ ref: reference,
210
+ children
211
+ }
212
+ );
213
+ });
214
+ ConversationHeader.displayName = "ConversationHeader";
215
+ const ConversationTitle = forwardRef(({ children, className }, reference) => {
216
+ return /* @__PURE__ */ jsx(
217
+ "h2",
218
+ {
219
+ className: cn("text-sm font-semibold leading-none", className),
220
+ ref: reference,
221
+ children
222
+ }
223
+ );
224
+ });
225
+ ConversationTitle.displayName = "ConversationTitle";
226
+ const ConversationMessages = forwardRef(({ children, className }, reference) => {
227
+ const { handleScroll, messages, messagesEndRef, scrollContainerRef } = useConversationThreadContext();
228
+ return /* @__PURE__ */ jsxs("div", { className: cn("relative min-h-0 flex-1", className), ref: reference, children: [
229
+ /* @__PURE__ */ jsx(
230
+ "div",
231
+ {
232
+ "aria-label": "Conversation messages",
233
+ "aria-live": "polite",
234
+ className: "absolute inset-0 overflow-y-auto",
235
+ onScroll: handleScroll,
236
+ ref: scrollContainerRef,
237
+ role: "log",
238
+ children: /* @__PURE__ */ jsxs("div", { className: "flex flex-col p-4", children: [
239
+ messages.map((message) => /* @__PURE__ */ jsx(MessageItem, { message }, message.id)),
240
+ /* @__PURE__ */ jsx("div", { "aria-hidden": "true", ref: messagesEndRef })
241
+ ] })
242
+ }
243
+ ),
244
+ children
245
+ ] });
246
+ });
247
+ ConversationMessages.displayName = "ConversationMessages";
248
+ const ConversationEmpty = forwardRef(({ children, className }, reference) => {
249
+ const { messages } = useConversationThreadContext();
250
+ if (messages.length > 0) return null;
251
+ return /* @__PURE__ */ jsx(
252
+ "div",
253
+ {
254
+ className: cn(
255
+ "pointer-events-none absolute inset-0 flex flex-col items-center justify-center gap-4 p-8",
256
+ className
257
+ ),
258
+ ref: reference,
259
+ children: /* @__PURE__ */ jsx("div", { className: "pointer-events-auto flex flex-col items-center gap-4", children })
260
+ }
261
+ );
262
+ });
263
+ ConversationEmpty.displayName = "ConversationEmpty";
264
+ const ConversationSuggestions = forwardRef(({ className, suggestions = [] }, reference) => {
265
+ const { onSend } = useConversationThreadContext();
266
+ return /* @__PURE__ */ jsx(
267
+ "div",
268
+ {
269
+ className: cn("flex flex-wrap justify-center gap-2", className),
270
+ ref: reference,
271
+ children: suggestions.map((suggestion) => /* @__PURE__ */ jsx(
272
+ "button",
273
+ {
274
+ className: "rounded-full border bg-background px-4 py-2 text-sm transition-colors hover:bg-muted",
275
+ onClick: () => onSend?.(suggestion),
276
+ type: "button",
277
+ children: suggestion
278
+ },
279
+ suggestion
280
+ ))
281
+ }
282
+ );
283
+ });
284
+ ConversationSuggestions.displayName = "ConversationSuggestions";
285
+ const ConversationScrollButton = forwardRef(({ className }, reference) => {
286
+ const { isAtBottom, scrollToBottom } = useConversationThreadContext();
287
+ if (isAtBottom) return null;
288
+ return /* @__PURE__ */ jsx(
289
+ "button",
290
+ {
291
+ "aria-label": "Scroll to bottom",
292
+ className: cn(
293
+ "absolute bottom-4 right-4 flex h-8 w-8 items-center justify-center rounded-full border bg-background shadow-md transition-colors hover:bg-muted",
294
+ className
295
+ ),
296
+ onClick: scrollToBottom,
297
+ ref: reference,
298
+ type: "button",
299
+ children: /* @__PURE__ */ jsx(ArrowDown, { className: "h-4 w-4" })
300
+ }
301
+ );
302
+ });
303
+ ConversationScrollButton.displayName = "ConversationScrollButton";
304
+ const ConversationLoading = forwardRef(({ className }, reference) => {
305
+ const { isStreaming, messages } = useConversationThreadContext();
306
+ const lastMessage = messages.at(-1);
307
+ if (!isStreaming || lastMessage?.role !== "assistant") return null;
308
+ return /* @__PURE__ */ jsxs(
309
+ "div",
310
+ {
311
+ "aria-label": "Assistant is typing",
312
+ className: cn(
313
+ "absolute bottom-4 left-4 flex items-center gap-1",
314
+ className
315
+ ),
316
+ ref: reference,
317
+ role: "status",
318
+ children: [
319
+ /* @__PURE__ */ jsx(
320
+ "span",
321
+ {
322
+ className: "h-2 w-2 animate-bounce rounded-full bg-muted-foreground",
323
+ style: { animationDelay: "-0.3s" }
324
+ }
325
+ ),
326
+ /* @__PURE__ */ jsx(
327
+ "span",
328
+ {
329
+ className: "h-2 w-2 animate-bounce rounded-full bg-muted-foreground",
330
+ style: { animationDelay: "-0.15s" }
331
+ }
332
+ ),
333
+ /* @__PURE__ */ jsx("span", { className: "h-2 w-2 animate-bounce rounded-full bg-muted-foreground" })
334
+ ]
335
+ }
336
+ );
337
+ });
338
+ ConversationLoading.displayName = "ConversationLoading";
339
+ export {
340
+ ConversationEmpty,
341
+ ConversationHeader,
342
+ ConversationLoading,
343
+ ConversationMessages,
344
+ ConversationScrollButton,
345
+ ConversationSuggestions,
346
+ ConversationThread,
347
+ ConversationTitle
348
+ };
@@ -0,0 +1,20 @@
1
+ import {
2
+ ConversationEmpty,
3
+ ConversationHeader,
4
+ ConversationLoading,
5
+ ConversationMessages,
6
+ ConversationScrollButton,
7
+ ConversationSuggestions,
8
+ ConversationThread,
9
+ ConversationTitle
10
+ } from "./conversation-thread";
11
+ export {
12
+ ConversationEmpty,
13
+ ConversationHeader,
14
+ ConversationLoading,
15
+ ConversationMessages,
16
+ ConversationScrollButton,
17
+ ConversationSuggestions,
18
+ ConversationThread,
19
+ ConversationTitle
20
+ };