@agent-relay/dashboard 2.0.80 → 2.0.82

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 (244) hide show
  1. package/out/404.html +1 -1
  2. package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
  3. package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
  4. package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
  5. package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
  6. package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
  7. package/out/about.html +2 -2
  8. package/out/about.txt +1 -1
  9. package/out/app/onboarding.html +1 -1
  10. package/out/app/onboarding.txt +1 -1
  11. package/out/app.html +1 -1
  12. package/out/app.txt +2 -2
  13. package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
  14. package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
  15. package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
  16. package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
  17. package/out/blog.html +2 -2
  18. package/out/blog.txt +1 -1
  19. package/out/careers.html +2 -2
  20. package/out/careers.txt +1 -1
  21. package/out/changelog.html +2 -2
  22. package/out/changelog.txt +1 -1
  23. package/out/cloud/link.html +1 -1
  24. package/out/cloud/link.txt +2 -2
  25. package/out/complete-profile.html +2 -2
  26. package/out/complete-profile.txt +1 -1
  27. package/out/connect-repos.html +1 -1
  28. package/out/connect-repos.txt +1 -1
  29. package/out/contact.html +2 -2
  30. package/out/contact.txt +1 -1
  31. package/out/docs.html +2 -2
  32. package/out/docs.txt +1 -1
  33. package/out/history.html +1 -1
  34. package/out/history.txt +2 -2
  35. package/out/index.html +1 -1
  36. package/out/index.txt +2 -2
  37. package/out/login.html +2 -2
  38. package/out/login.txt +1 -1
  39. package/out/metrics.html +1 -1
  40. package/out/metrics.txt +2 -2
  41. package/out/pricing.html +2 -2
  42. package/out/pricing.txt +1 -1
  43. package/out/privacy.html +2 -2
  44. package/out/privacy.txt +1 -1
  45. package/out/providers/setup/claude.html +1 -1
  46. package/out/providers/setup/claude.txt +1 -1
  47. package/out/providers/setup/codex.html +1 -1
  48. package/out/providers/setup/codex.txt +1 -1
  49. package/out/providers/setup/cursor.html +1 -1
  50. package/out/providers/setup/cursor.txt +1 -1
  51. package/out/providers.html +1 -1
  52. package/out/providers.txt +1 -1
  53. package/out/security.html +2 -2
  54. package/out/security.txt +1 -1
  55. package/out/signup.html +2 -2
  56. package/out/signup.txt +1 -1
  57. package/out/terms.html +2 -2
  58. package/out/terms.txt +1 -1
  59. package/package.json +7 -1
  60. package/src/app/about/page.tsx +7 -0
  61. package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
  62. package/src/app/app/[[...slug]]/page.tsx +23 -0
  63. package/src/app/app/onboarding/page.tsx +394 -0
  64. package/src/app/apple-icon.png +0 -0
  65. package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
  66. package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
  67. package/src/app/blog/page.tsx +15 -0
  68. package/src/app/careers/page.tsx +7 -0
  69. package/src/app/changelog/page.tsx +7 -0
  70. package/src/app/cloud/link/page.tsx +464 -0
  71. package/src/app/complete-profile/page.tsx +204 -0
  72. package/src/app/connect-repos/page.tsx +410 -0
  73. package/src/app/contact/page.tsx +7 -0
  74. package/src/app/docs/page.tsx +7 -0
  75. package/src/app/favicon.png +0 -0
  76. package/src/app/globals.css +200 -0
  77. package/src/app/history/page.tsx +658 -0
  78. package/src/app/layout.tsx +25 -0
  79. package/src/app/login/page.tsx +424 -0
  80. package/src/app/metrics/page.tsx +781 -0
  81. package/src/app/page.tsx +59 -0
  82. package/src/app/pricing/page.tsx +7 -0
  83. package/src/app/privacy/page.tsx +7 -0
  84. package/src/app/providers/page.tsx +193 -0
  85. package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
  86. package/src/app/providers/setup/[provider]/constants.ts +35 -0
  87. package/src/app/providers/setup/[provider]/page.tsx +42 -0
  88. package/src/app/security/page.tsx +7 -0
  89. package/src/app/signup/page.tsx +533 -0
  90. package/src/app/terms/page.tsx +7 -0
  91. package/src/components/ActivityFeed.tsx +216 -0
  92. package/src/components/AddWorkspaceModal.tsx +170 -0
  93. package/src/components/AgentCard.test.tsx +134 -0
  94. package/src/components/AgentCard.tsx +585 -0
  95. package/src/components/AgentList.test.tsx +147 -0
  96. package/src/components/AgentList.tsx +419 -0
  97. package/src/components/AgentLogPreview.tsx +173 -0
  98. package/src/components/AgentProfilePanel.tsx +569 -0
  99. package/src/components/App.tsx +3424 -0
  100. package/src/components/BillingPanel.tsx +922 -0
  101. package/src/components/BillingResult.tsx +447 -0
  102. package/src/components/BroadcastComposer.tsx +690 -0
  103. package/src/components/ChannelAdminPanel.tsx +773 -0
  104. package/src/components/ChannelBrowser.tsx +385 -0
  105. package/src/components/ChannelChat.tsx +261 -0
  106. package/src/components/ChannelSidebar.tsx +399 -0
  107. package/src/components/CloudSessionProvider.tsx +130 -0
  108. package/src/components/CommandPalette.tsx +815 -0
  109. package/src/components/ConfirmationDialog.tsx +133 -0
  110. package/src/components/ConversationHistory.tsx +518 -0
  111. package/src/components/CoordinatorPanel.tsx +956 -0
  112. package/src/components/DecisionQueue.tsx +717 -0
  113. package/src/components/DirectMessageView.tsx +164 -0
  114. package/src/components/FileAutocomplete.tsx +368 -0
  115. package/src/components/FleetOverview.tsx +278 -0
  116. package/src/components/LogViewer.tsx +310 -0
  117. package/src/components/LogViewerPanel.tsx +482 -0
  118. package/src/components/Logo.tsx +284 -0
  119. package/src/components/MentionAutocomplete.tsx +384 -0
  120. package/src/components/MessageComposer.tsx +473 -0
  121. package/src/components/MessageList.tsx +725 -0
  122. package/src/components/MessageSenderName.tsx +91 -0
  123. package/src/components/MessageStatusIndicator.tsx +142 -0
  124. package/src/components/NewConversationModal.tsx +400 -0
  125. package/src/components/NotificationToast.tsx +488 -0
  126. package/src/components/OnlineUsersIndicator.tsx +164 -0
  127. package/src/components/Pagination.tsx +124 -0
  128. package/src/components/PricingPlans.tsx +386 -0
  129. package/src/components/ProjectList.tsx +711 -0
  130. package/src/components/ProviderAuthFlow.tsx +343 -0
  131. package/src/components/ProviderConnectionList.tsx +375 -0
  132. package/src/components/ProvisioningProgress.tsx +730 -0
  133. package/src/components/ReactionChips.tsx +70 -0
  134. package/src/components/ReactionPicker.tsx +121 -0
  135. package/src/components/RepoAccessPanel.tsx +787 -0
  136. package/src/components/RepositoriesPanel.tsx +901 -0
  137. package/src/components/ServerCard.tsx +202 -0
  138. package/src/components/SessionExpiredModal.tsx +128 -0
  139. package/src/components/SpawnModal.test.tsx +190 -0
  140. package/src/components/SpawnModal.tsx +1001 -0
  141. package/src/components/TaskAssignmentUI.tsx +375 -0
  142. package/src/components/TerminalProviderSetup.tsx +517 -0
  143. package/src/components/ThemeProvider.tsx +159 -0
  144. package/src/components/ThinkingIndicator.tsx +231 -0
  145. package/src/components/ThreadList.tsx +198 -0
  146. package/src/components/ThreadPanel.tsx +405 -0
  147. package/src/components/TrajectoryViewer.tsx +698 -0
  148. package/src/components/TypingIndicator.tsx +69 -0
  149. package/src/components/UsageBanner.tsx +231 -0
  150. package/src/components/UserProfilePanel.tsx +233 -0
  151. package/src/components/WorkspaceContext.tsx +95 -0
  152. package/src/components/WorkspaceSelector.tsx +234 -0
  153. package/src/components/WorkspaceStatusIndicator.tsx +396 -0
  154. package/src/components/XTermInteractive.tsx +516 -0
  155. package/src/components/XTermLogViewer.tsx +719 -0
  156. package/src/components/channels/ChannelDialogs.tsx +1411 -0
  157. package/src/components/channels/ChannelHeader.tsx +317 -0
  158. package/src/components/channels/ChannelMessageList.tsx +463 -0
  159. package/src/components/channels/ChannelViewV1.tsx +146 -0
  160. package/src/components/channels/MessageInput.tsx +302 -0
  161. package/src/components/channels/SearchInput.tsx +172 -0
  162. package/src/components/channels/SearchResults.tsx +336 -0
  163. package/src/components/channels/api.test.ts +1527 -0
  164. package/src/components/channels/api.ts +703 -0
  165. package/src/components/channels/index.ts +76 -0
  166. package/src/components/channels/mockApi.ts +344 -0
  167. package/src/components/channels/types.ts +566 -0
  168. package/src/components/hooks/index.ts +58 -0
  169. package/src/components/hooks/useAgentLogs.ts +504 -0
  170. package/src/components/hooks/useAgents.ts +127 -0
  171. package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
  172. package/src/components/hooks/useBroadcastDedup.ts +86 -0
  173. package/src/components/hooks/useChannelAdmin.ts +329 -0
  174. package/src/components/hooks/useChannelBrowser.ts +239 -0
  175. package/src/components/hooks/useChannelCommands.ts +138 -0
  176. package/src/components/hooks/useChannels.ts +367 -0
  177. package/src/components/hooks/useDebounce.ts +29 -0
  178. package/src/components/hooks/useDirectMessage.test.ts +952 -0
  179. package/src/components/hooks/useDirectMessage.ts +141 -0
  180. package/src/components/hooks/useMessages.ts +310 -0
  181. package/src/components/hooks/useOrchestrator.test.ts +165 -0
  182. package/src/components/hooks/useOrchestrator.ts +424 -0
  183. package/src/components/hooks/usePinnedAgents.test.ts +356 -0
  184. package/src/components/hooks/usePinnedAgents.ts +140 -0
  185. package/src/components/hooks/usePresence.test.ts +245 -0
  186. package/src/components/hooks/usePresence.ts +377 -0
  187. package/src/components/hooks/useRecentRepos.ts +130 -0
  188. package/src/components/hooks/useSession.ts +209 -0
  189. package/src/components/hooks/useThread.ts +138 -0
  190. package/src/components/hooks/useTrajectory.ts +265 -0
  191. package/src/components/hooks/useWebSocket.ts +290 -0
  192. package/src/components/hooks/useWorkspaceMembers.ts +132 -0
  193. package/src/components/hooks/useWorkspaceRepos.ts +73 -0
  194. package/src/components/hooks/useWorkspaceStatus.ts +237 -0
  195. package/src/components/index.ts +81 -0
  196. package/src/components/layout/Header.tsx +311 -0
  197. package/src/components/layout/RepoContextHeader.tsx +361 -0
  198. package/src/components/layout/Sidebar.archive.test.tsx +126 -0
  199. package/src/components/layout/Sidebar.test.tsx +691 -0
  200. package/src/components/layout/Sidebar.tsx +900 -0
  201. package/src/components/layout/index.ts +7 -0
  202. package/src/components/settings/BillingSettingsPanel.tsx +564 -0
  203. package/src/components/settings/SettingsPage.tsx +683 -0
  204. package/src/components/settings/TeamSettingsPanel.tsx +560 -0
  205. package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
  206. package/src/components/settings/index.ts +11 -0
  207. package/src/components/settings/types.ts +79 -0
  208. package/src/components/utils/messageFormatting.test.tsx +331 -0
  209. package/src/components/utils/messageFormatting.tsx +597 -0
  210. package/src/index.ts +63 -0
  211. package/src/landing/AboutPage.tsx +77 -0
  212. package/src/landing/BlogContent.tsx +187 -0
  213. package/src/landing/BlogPage.tsx +47 -0
  214. package/src/landing/CareersPage.tsx +53 -0
  215. package/src/landing/ChangelogPage.tsx +33 -0
  216. package/src/landing/ContactPage.tsx +41 -0
  217. package/src/landing/DocsPage.tsx +43 -0
  218. package/src/landing/LandingPage.tsx +702 -0
  219. package/src/landing/PricingPage.tsx +549 -0
  220. package/src/landing/PrivacyPage.tsx +117 -0
  221. package/src/landing/SecurityPage.tsx +42 -0
  222. package/src/landing/StaticPage.tsx +165 -0
  223. package/src/landing/TermsPage.tsx +125 -0
  224. package/src/landing/blogData.ts +312 -0
  225. package/src/landing/index.ts +18 -0
  226. package/src/landing/styles.css +3673 -0
  227. package/src/lib/agent-merge.test.ts +43 -0
  228. package/src/lib/agent-merge.ts +35 -0
  229. package/src/lib/api.ts +1294 -0
  230. package/src/lib/cloudApi.ts +893 -0
  231. package/src/lib/colors.test.ts +175 -0
  232. package/src/lib/colors.ts +218 -0
  233. package/src/lib/config.ts +109 -0
  234. package/src/lib/hierarchy.ts +242 -0
  235. package/src/lib/stuckDetection.ts +142 -0
  236. package/src/lib/useUrlRouting.ts +190 -0
  237. package/src/types/index.ts +317 -0
  238. package/src/types/threading.ts +7 -0
  239. package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
  240. package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
  241. /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
  242. /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
  243. /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
  244. /package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/{page-a58308f43557b908.js → page-b194f207fbd91862.js} +0 -0
@@ -0,0 +1,216 @@
1
+ /**
2
+ * ActivityFeed Component
3
+ *
4
+ * Displays a unified timeline of workspace events including:
5
+ * - Agent spawned/released
6
+ * - Agent online/offline
7
+ * - User joined/left
8
+ * - Broadcasts
9
+ */
10
+
11
+ import React, { useMemo } from 'react';
12
+ import type { ActivityEvent, ActivityEventType } from '../types';
13
+
14
+ export interface ActivityFeedProps {
15
+ events: ActivityEvent[];
16
+ maxEvents?: number;
17
+ onEventClick?: (event: ActivityEvent) => void;
18
+ }
19
+
20
+ /**
21
+ * Get icon for activity event type
22
+ */
23
+ function getEventIcon(type: ActivityEventType): string {
24
+ switch (type) {
25
+ case 'agent_spawned':
26
+ return '🚀';
27
+ case 'agent_released':
28
+ return '🛑';
29
+ case 'agent_online':
30
+ return '🟢';
31
+ case 'agent_offline':
32
+ return '⚫';
33
+ case 'user_joined':
34
+ return '👋';
35
+ case 'user_left':
36
+ return '👋';
37
+ case 'broadcast':
38
+ return '📢';
39
+ case 'error':
40
+ return '⚠️';
41
+ default:
42
+ return '📌';
43
+ }
44
+ }
45
+
46
+ /**
47
+ * Get color class for activity event type
48
+ */
49
+ function getEventColorClass(type: ActivityEventType): string {
50
+ switch (type) {
51
+ case 'agent_spawned':
52
+ return 'text-green-400';
53
+ case 'agent_released':
54
+ return 'text-red-400';
55
+ case 'agent_online':
56
+ return 'text-green-400';
57
+ case 'agent_offline':
58
+ return 'text-gray-400';
59
+ case 'user_joined':
60
+ return 'text-cyan-400';
61
+ case 'user_left':
62
+ return 'text-gray-400';
63
+ case 'broadcast':
64
+ return 'text-yellow-400';
65
+ case 'error':
66
+ return 'text-red-500';
67
+ default:
68
+ return 'text-text-muted';
69
+ }
70
+ }
71
+
72
+ /**
73
+ * Format relative time (e.g., "2m ago", "1h ago")
74
+ */
75
+ function formatRelativeTime(timestamp: string): string {
76
+ const now = new Date();
77
+ const then = new Date(timestamp);
78
+ const diffMs = now.getTime() - then.getTime();
79
+ const diffSec = Math.floor(diffMs / 1000);
80
+ const diffMin = Math.floor(diffSec / 60);
81
+ const diffHour = Math.floor(diffMin / 60);
82
+ const diffDay = Math.floor(diffHour / 24);
83
+
84
+ if (diffSec < 60) return 'just now';
85
+ if (diffMin < 60) return `${diffMin}m ago`;
86
+ if (diffHour < 24) return `${diffHour}h ago`;
87
+ if (diffDay < 7) return `${diffDay}d ago`;
88
+ return then.toLocaleDateString();
89
+ }
90
+
91
+ /**
92
+ * Single activity event item
93
+ */
94
+ function ActivityEventItem({
95
+ event,
96
+ onClick,
97
+ }: {
98
+ event: ActivityEvent;
99
+ onClick?: () => void;
100
+ }) {
101
+ const icon = getEventIcon(event.type);
102
+ const colorClass = getEventColorClass(event.type);
103
+
104
+ return (
105
+ <div
106
+ className={`flex items-start gap-3 p-3 rounded-lg hover:bg-bg-hover transition-colors ${onClick ? 'cursor-pointer' : ''}`}
107
+ onClick={onClick}
108
+ >
109
+ {/* Avatar or Icon */}
110
+ <div className="flex-shrink-0 w-8 h-8 flex items-center justify-center">
111
+ {event.actorAvatarUrl ? (
112
+ <img
113
+ src={event.actorAvatarUrl}
114
+ alt={event.actor}
115
+ className="w-8 h-8 rounded-full"
116
+ />
117
+ ) : (
118
+ <span className="text-lg">{icon}</span>
119
+ )}
120
+ </div>
121
+
122
+ {/* Content */}
123
+ <div className="flex-1 min-w-0">
124
+ <div className="flex items-center gap-2">
125
+ <span className={`font-medium ${colorClass}`}>{event.actor}</span>
126
+ <span className="text-text-muted text-sm">{event.title}</span>
127
+ </div>
128
+ {event.description && (
129
+ <p className="text-text-muted text-sm mt-1 line-clamp-2">
130
+ {event.description}
131
+ </p>
132
+ )}
133
+ {/* Metadata badges */}
134
+ {event.metadata && Object.keys(event.metadata).length > 0 && (() => {
135
+ const cli = event.metadata.cli;
136
+ const task = event.metadata.task;
137
+ return (
138
+ <div className="flex flex-wrap gap-1 mt-2">
139
+ {cli != null && (
140
+ <span className="px-2 py-0.5 bg-bg-secondary rounded text-xs text-text-muted">
141
+ {String(cli)}
142
+ </span>
143
+ )}
144
+ {task != null && (
145
+ <span className="px-2 py-0.5 bg-bg-secondary rounded text-xs text-text-muted truncate max-w-[200px]">
146
+ {String(task)}
147
+ </span>
148
+ )}
149
+ </div>
150
+ );
151
+ })()}
152
+ </div>
153
+
154
+ {/* Timestamp */}
155
+ <div className="flex-shrink-0 text-xs text-text-muted">
156
+ {formatRelativeTime(event.timestamp)}
157
+ </div>
158
+ </div>
159
+ );
160
+ }
161
+
162
+ /**
163
+ * Empty state when no events
164
+ */
165
+ function EmptyState() {
166
+ return (
167
+ <div className="flex flex-col items-center justify-center h-full text-center p-8">
168
+ <div className="text-4xl mb-4">📋</div>
169
+ <h3 className="text-lg font-medium text-text-primary mb-2">No activity yet</h3>
170
+ <p className="text-text-muted text-sm max-w-xs">
171
+ Activity will appear here as agents spawn, users join, and broadcasts are sent.
172
+ </p>
173
+ </div>
174
+ );
175
+ }
176
+
177
+ /**
178
+ * Activity Feed - displays timeline of workspace events
179
+ */
180
+ export function ActivityFeed({ events, maxEvents = 100, onEventClick }: ActivityFeedProps) {
181
+ // Sort events by timestamp (newest first) and limit
182
+ const sortedEvents = useMemo(() => {
183
+ return [...events]
184
+ .sort((a, b) => new Date(b.timestamp).getTime() - new Date(a.timestamp).getTime())
185
+ .slice(0, maxEvents);
186
+ }, [events, maxEvents]);
187
+
188
+ if (sortedEvents.length === 0) {
189
+ return <EmptyState />;
190
+ }
191
+
192
+ return (
193
+ <div className="flex flex-col h-full">
194
+ {/* Header */}
195
+ <div className="flex-shrink-0 px-4 py-3 border-b border-border-subtle">
196
+ <h2 className="text-lg font-semibold text-text-primary">Activity</h2>
197
+ <p className="text-sm text-text-muted">
198
+ {sortedEvents.length} event{sortedEvents.length !== 1 ? 's' : ''}
199
+ </p>
200
+ </div>
201
+
202
+ {/* Event List */}
203
+ <div className="flex-1 overflow-y-auto px-2 py-2">
204
+ <div className="space-y-1">
205
+ {sortedEvents.map((event) => (
206
+ <ActivityEventItem
207
+ key={event.id}
208
+ event={event}
209
+ onClick={onEventClick ? () => onEventClick(event) : undefined}
210
+ />
211
+ ))}
212
+ </div>
213
+ </div>
214
+ </div>
215
+ );
216
+ }
@@ -0,0 +1,170 @@
1
+ /**
2
+ * Add Workspace Modal
3
+ *
4
+ * Modal dialog for adding a new workspace (repository) to the orchestrator.
5
+ */
6
+
7
+ import React, { useState, useEffect, useRef } from 'react';
8
+
9
+ export interface AddWorkspaceModalProps {
10
+ isOpen: boolean;
11
+ onClose: () => void;
12
+ onAdd: (path: string, name?: string) => Promise<void>;
13
+ isAdding?: boolean;
14
+ error?: string | null;
15
+ }
16
+
17
+ export function AddWorkspaceModal({
18
+ isOpen,
19
+ onClose,
20
+ onAdd,
21
+ isAdding = false,
22
+ error,
23
+ }: AddWorkspaceModalProps) {
24
+ const [path, setPath] = useState('');
25
+ const [name, setName] = useState('');
26
+ const [localError, setLocalError] = useState<string | null>(null);
27
+ const inputRef = useRef<HTMLInputElement>(null);
28
+
29
+ // Focus input when modal opens
30
+ useEffect(() => {
31
+ if (isOpen && inputRef.current) {
32
+ inputRef.current.focus();
33
+ }
34
+ }, [isOpen]);
35
+
36
+ // Reset form when modal closes
37
+ useEffect(() => {
38
+ if (!isOpen) {
39
+ setPath('');
40
+ setName('');
41
+ setLocalError(null);
42
+ }
43
+ }, [isOpen]);
44
+
45
+ const handleSubmit = async (e: React.FormEvent) => {
46
+ e.preventDefault();
47
+
48
+ if (!path.trim()) {
49
+ setLocalError('Path is required');
50
+ return;
51
+ }
52
+
53
+ try {
54
+ await onAdd(path.trim(), name.trim() || undefined);
55
+ onClose();
56
+ } catch (err) {
57
+ setLocalError(err instanceof Error ? err.message : 'Failed to add workspace');
58
+ }
59
+ };
60
+
61
+ const handleKeyDown = (e: React.KeyboardEvent) => {
62
+ if (e.key === 'Escape') {
63
+ onClose();
64
+ }
65
+ };
66
+
67
+ if (!isOpen) return null;
68
+
69
+ const displayError = error || localError;
70
+
71
+ return (
72
+ <div
73
+ className="fixed inset-0 bg-black/60 flex items-center justify-center z-[9999] backdrop-blur-sm"
74
+ onClick={onClose}
75
+ onKeyDown={handleKeyDown}
76
+ >
77
+ <div
78
+ className="bg-[#1a1a2e] border border-[#3a3a4e] rounded-xl p-6 min-w-[450px] max-w-[90vw] shadow-[0_20px_60px_rgba(0,0,0,0.5)]"
79
+ onClick={(e) => e.stopPropagation()}
80
+ >
81
+ <div className="flex items-center justify-between mb-6">
82
+ <h2 className="m-0 text-lg font-semibold text-[#e8e8e8]">Add Workspace</h2>
83
+ <button
84
+ className="bg-transparent border-none text-[#666] cursor-pointer p-1 flex items-center justify-center rounded transition-all hover:bg-white/10 hover:text-[#e8e8e8]"
85
+ onClick={onClose}
86
+ >
87
+ <CloseIcon />
88
+ </button>
89
+ </div>
90
+
91
+ <form onSubmit={handleSubmit}>
92
+ <div className="mb-5">
93
+ <label htmlFor="workspace-path" className="block mb-2 text-[13px] font-medium text-[#e8e8e8]">
94
+ Repository Path
95
+ </label>
96
+ <input
97
+ ref={inputRef}
98
+ id="workspace-path"
99
+ type="text"
100
+ value={path}
101
+ onChange={(e) => {
102
+ setPath(e.target.value);
103
+ setLocalError(null);
104
+ }}
105
+ placeholder="/path/to/repository"
106
+ disabled={isAdding}
107
+ autoComplete="off"
108
+ className="w-full px-3 py-2.5 bg-[#2a2a3e] border border-[#3a3a4e] rounded-md text-[#e8e8e8] text-sm outline-none transition-colors box-border focus:border-[#00c896] placeholder:text-[#666] disabled:opacity-60 disabled:cursor-not-allowed"
109
+ />
110
+ <p className="mt-1.5 text-xs text-[#666] leading-relaxed">
111
+ Enter the full path to your repository. Use ~ for home directory.
112
+ </p>
113
+ </div>
114
+
115
+ <div className="mb-5">
116
+ <label htmlFor="workspace-name" className="block mb-2 text-[13px] font-medium text-[#e8e8e8]">
117
+ Display Name (optional)
118
+ </label>
119
+ <input
120
+ id="workspace-name"
121
+ type="text"
122
+ value={name}
123
+ onChange={(e) => setName(e.target.value)}
124
+ placeholder="My Project"
125
+ disabled={isAdding}
126
+ autoComplete="off"
127
+ className="w-full px-3 py-2.5 bg-[#2a2a3e] border border-[#3a3a4e] rounded-md text-[#e8e8e8] text-sm outline-none transition-colors box-border focus:border-[#00c896] placeholder:text-[#666] disabled:opacity-60 disabled:cursor-not-allowed"
128
+ />
129
+ <p className="mt-1.5 text-xs text-[#666] leading-relaxed">
130
+ A friendly name for this workspace. Defaults to the folder name.
131
+ </p>
132
+ </div>
133
+
134
+ {displayError && (
135
+ <div className="px-3 py-2.5 bg-red-500/10 border border-red-500/30 rounded-md text-red-500 text-[13px] mb-5">
136
+ {displayError}
137
+ </div>
138
+ )}
139
+
140
+ <div className="flex gap-3 justify-end mt-6">
141
+ <button
142
+ type="button"
143
+ className="px-5 py-2.5 rounded-md text-sm font-medium cursor-pointer transition-all bg-transparent border border-[#3a3a4e] text-[#e8e8e8] hover:bg-white/5 disabled:opacity-50 disabled:cursor-not-allowed"
144
+ onClick={onClose}
145
+ disabled={isAdding}
146
+ >
147
+ Cancel
148
+ </button>
149
+ <button
150
+ type="submit"
151
+ className="px-5 py-2.5 rounded-md text-sm font-medium cursor-pointer transition-all bg-[#00c896] border-none text-[#1a1a2e] hover:bg-[#00a87d] disabled:opacity-50 disabled:cursor-not-allowed"
152
+ disabled={isAdding || !path.trim()}
153
+ >
154
+ {isAdding ? 'Adding...' : 'Add Workspace'}
155
+ </button>
156
+ </div>
157
+ </form>
158
+ </div>
159
+ </div>
160
+ );
161
+ }
162
+
163
+ function CloseIcon() {
164
+ return (
165
+ <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
166
+ <line x1="18" y1="6" x2="6" y2="18" />
167
+ <line x1="6" y1="6" x2="18" y2="18" />
168
+ </svg>
169
+ );
170
+ }
@@ -0,0 +1,134 @@
1
+ /**
2
+ * Tests for AgentCard component
3
+ *
4
+ * Covers: CLI type display, model display in compact view,
5
+ * and action button rendering.
6
+ */
7
+
8
+ // @vitest-environment jsdom
9
+ import React from 'react';
10
+ import { describe, it, expect, vi, afterEach } from 'vitest';
11
+ import { render, screen, cleanup } from '@testing-library/react';
12
+ import { AgentCard } from './AgentCard';
13
+ import type { Agent } from '../types';
14
+
15
+ afterEach(() => {
16
+ cleanup();
17
+ });
18
+
19
+ function makeAgent(name: string, overrides: Partial<Agent> = {}): Agent {
20
+ return {
21
+ name,
22
+ status: 'online',
23
+ ...overrides,
24
+ };
25
+ }
26
+
27
+ describe('AgentCard', () => {
28
+ describe('compact view - CLI type', () => {
29
+ it('shows agent CLI type when available', () => {
30
+ const agent = makeAgent('backend-api', { cli: 'claude' });
31
+ render(<AgentCard agent={agent} compact />);
32
+ expect(screen.getByText('claude')).toBeTruthy();
33
+ });
34
+
35
+ it('shows codex CLI type', () => {
36
+ const agent = makeAgent('worker-1', { cli: 'codex' });
37
+ render(<AgentCard agent={agent} compact />);
38
+ expect(screen.getByText('codex')).toBeTruthy();
39
+ });
40
+
41
+ it('does not show CLI line when cli is not set', () => {
42
+ const agent = makeAgent('backend-api');
43
+ const { container } = render(<AgentCard agent={agent} compact />);
44
+ // Should not have the CLI text span (font-mono opacity-70)
45
+ const cliSpans = container.querySelectorAll('.font-mono.opacity-70');
46
+ expect(cliSpans.length).toBe(0);
47
+ });
48
+ });
49
+
50
+ describe('compact view - model display', () => {
51
+ it('shows model from agent.model', () => {
52
+ const agent = makeAgent('backend-api', { model: 'claude-sonnet-4-5-20250929' });
53
+ render(<AgentCard agent={agent} compact />);
54
+ expect(screen.getByText('claude-sonnet-4-5-20250929')).toBeTruthy();
55
+ });
56
+
57
+ it('shows model from agent.profile.model as fallback', () => {
58
+ const agent = makeAgent('backend-api', {
59
+ profile: { model: 'gpt-5.2-codex' },
60
+ });
61
+ render(<AgentCard agent={agent} compact />);
62
+ expect(screen.getByText('gpt-5.2-codex')).toBeTruthy();
63
+ });
64
+
65
+ it('prefers agent.model over profile.model', () => {
66
+ const agent = makeAgent('backend-api', {
67
+ model: 'claude-sonnet-4-5-20250929',
68
+ profile: { model: 'old-model' },
69
+ });
70
+ render(<AgentCard agent={agent} compact />);
71
+ expect(screen.getByText('claude-sonnet-4-5-20250929')).toBeTruthy();
72
+ expect(screen.queryByText('old-model')).toBeNull();
73
+ });
74
+
75
+ it('does not show model line when no model set', () => {
76
+ const agent = makeAgent('backend-api');
77
+ render(<AgentCard agent={agent} compact />);
78
+ const modelSpan = screen.queryByTitle(/Model:/);
79
+ expect(modelSpan).toBeNull();
80
+ });
81
+ });
82
+
83
+ describe('compact view - action buttons', () => {
84
+ it('renders pin button when onPinToggle provided', () => {
85
+ const agent = makeAgent('backend-api');
86
+ render(<AgentCard agent={agent} compact onPinToggle={vi.fn()} />);
87
+ expect(screen.getByTitle('Pin to top')).toBeTruthy();
88
+ });
89
+
90
+ it('renders profile button when onProfileClick provided', () => {
91
+ const agent = makeAgent('backend-api');
92
+ render(<AgentCard agent={agent} compact onProfileClick={vi.fn()} />);
93
+ expect(screen.getByTitle('View profile')).toBeTruthy();
94
+ });
95
+
96
+ it('renders logs button when onLogsClick provided', () => {
97
+ const agent = makeAgent('backend-api');
98
+ render(<AgentCard agent={agent} compact onLogsClick={vi.fn()} />);
99
+ expect(screen.getByTitle('View logs')).toBeTruthy();
100
+ });
101
+
102
+ it('renders logs button even for non-spawned agents', () => {
103
+ const agent = makeAgent('backend-api', { isSpawned: false });
104
+ render(<AgentCard agent={agent} compact onLogsClick={vi.fn()} />);
105
+ expect(screen.getByTitle('View logs')).toBeTruthy();
106
+ });
107
+
108
+ it('renders release button for spawned agents', () => {
109
+ const agent = makeAgent('backend-api', { isSpawned: true });
110
+ render(<AgentCard agent={agent} compact onReleaseClick={vi.fn()} />);
111
+ expect(screen.getByTitle('Kill agent')).toBeTruthy();
112
+ });
113
+
114
+ it('does not render release for non-spawned agents', () => {
115
+ const agent = makeAgent('backend-api', { isSpawned: false });
116
+ render(
117
+ <AgentCard
118
+ agent={agent}
119
+ compact
120
+ onReleaseClick={vi.fn()}
121
+ />
122
+ );
123
+ expect(screen.queryByTitle('Kill agent')).toBeNull();
124
+ });
125
+ });
126
+
127
+ describe('full view - model display', () => {
128
+ it('shows model badge in full view', () => {
129
+ const agent = makeAgent('backend-api', { model: 'haiku' });
130
+ render(<AgentCard agent={agent} />);
131
+ expect(screen.getByTitle('Model: haiku')).toBeTruthy();
132
+ });
133
+ });
134
+ });