@agent-relay/dashboard 2.0.81 → 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/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
  242. /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → 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,569 @@
1
+ /**
2
+ * AgentProfilePanel Component
3
+ *
4
+ * Slide-out panel showing agent profile details.
5
+ * Displays avatar, name, status, spawn prompt, persona, and other metadata.
6
+ * Similar to Slack's user profile panel.
7
+ */
8
+
9
+ import React, { useEffect, useRef, useState } from 'react';
10
+ import type { Agent, AgentSummary } from '../types';
11
+ import {
12
+ getAgentColor,
13
+ getAgentInitials,
14
+ STATUS_COLORS,
15
+ } from '../lib/colors';
16
+ import { getAgentDisplayName, getAgentBreadcrumb } from '../lib/hierarchy';
17
+
18
+ /** Provider display configuration */
19
+ const PROVIDER_CONFIG: Record<string, { label: string; color: string; icon: string }> = {
20
+ claude: { label: 'Claude', color: '#00d9ff', icon: '🤖' },
21
+ codex: { label: 'Codex', color: '#10a37f', icon: '🧠' },
22
+ gemini: { label: 'Gemini', color: '#4285f4', icon: '✨' },
23
+ droid: { label: 'Droid', color: '#ff6b35', icon: '🤖' },
24
+ opencode: { label: 'OpenCode', color: '#a855f7', icon: '💻' },
25
+ cursor: { label: 'Cursor', color: '#00b4d8', icon: '🖱️' },
26
+ other: { label: 'AI Agent', color: '#8d8d8e', icon: '🤖' },
27
+ };
28
+
29
+ /** Map CLI commands to provider names (for CLIs with different command names) */
30
+ const CLI_TO_PROVIDER_MAP: Record<string, string> = {
31
+ agent: 'cursor', // Cursor CLI installs as 'agent'
32
+ };
33
+
34
+ export interface AgentProfilePanelProps {
35
+ /** Agent to display (null to hide panel) */
36
+ agent: Agent | null;
37
+ /** Callback when panel should close */
38
+ onClose: () => void;
39
+ /** Callback when message button is clicked */
40
+ onMessage?: (agent: Agent) => void;
41
+ /** Callback when logs button is clicked */
42
+ onLogs?: (agent: Agent) => void;
43
+ /** Callback when release button is clicked */
44
+ onRelease?: (agent: Agent) => void;
45
+ /** Agent's recent work summary (optional) */
46
+ summary?: AgentSummary | null;
47
+ }
48
+
49
+ export function AgentProfilePanel({
50
+ agent,
51
+ onClose,
52
+ onMessage,
53
+ onLogs,
54
+ onRelease,
55
+ summary,
56
+ }: AgentProfilePanelProps) {
57
+ const panelRef = useRef<HTMLDivElement>(null);
58
+ const [showFullPrompt, setShowFullPrompt] = useState(false);
59
+ const [showFullPersona, setShowFullPersona] = useState(false);
60
+
61
+ // Close on Escape key
62
+ useEffect(() => {
63
+ const handleKeyDown = (e: KeyboardEvent) => {
64
+ if (e.key === 'Escape') {
65
+ onClose();
66
+ }
67
+ };
68
+
69
+ if (agent) {
70
+ window.addEventListener('keydown', handleKeyDown);
71
+ }
72
+ return () => window.removeEventListener('keydown', handleKeyDown);
73
+ }, [agent, onClose]);
74
+
75
+ // Close on outside click
76
+ const justOpenedRef = useRef(false);
77
+
78
+ useEffect(() => {
79
+ if (agent) {
80
+ justOpenedRef.current = true;
81
+ setShowFullPrompt(false);
82
+ setShowFullPersona(false);
83
+ }
84
+ }, [agent]);
85
+
86
+ useEffect(() => {
87
+ const handleClickOutside = (e: MouseEvent) => {
88
+ if (justOpenedRef.current) {
89
+ justOpenedRef.current = false;
90
+ return;
91
+ }
92
+
93
+ if (panelRef.current && !panelRef.current.contains(e.target as Node)) {
94
+ onClose();
95
+ }
96
+ };
97
+
98
+ if (agent) {
99
+ document.addEventListener('mousedown', handleClickOutside);
100
+ return () => {
101
+ document.removeEventListener('mousedown', handleClickOutside);
102
+ };
103
+ }
104
+ }, [agent, onClose]);
105
+
106
+ if (!agent) {
107
+ return null;
108
+ }
109
+
110
+ const colors = getAgentColor(agent.name);
111
+ const initials = getAgentInitials(agent.name);
112
+ const displayName = getAgentDisplayName(agent.name);
113
+ const breadcrumb = getAgentBreadcrumb(agent.name);
114
+ const statusColor = STATUS_COLORS[agent.status] || STATUS_COLORS.offline;
115
+ const isOnline = agent.status === 'online';
116
+ const profile = agent.profile;
117
+
118
+ return (
119
+ <>
120
+ {/* Backdrop */}
121
+ <div className="fixed inset-0 bg-black/50 z-40" />
122
+
123
+ {/* Panel */}
124
+ <div
125
+ ref={panelRef}
126
+ className="fixed right-0 top-0 h-full w-96 bg-[#1a1d21] border-l border-white/10 shadow-2xl z-50 flex flex-col animate-slide-in-right"
127
+ >
128
+ {/* Header */}
129
+ <div className="flex items-center justify-between p-4 border-b border-white/10">
130
+ <h2 className="text-lg font-semibold text-[#d1d2d3]">Agent Profile</h2>
131
+ <button
132
+ onClick={onClose}
133
+ className="p-1 hover:bg-white/10 rounded-md transition-colors text-[#d1d2d3]"
134
+ title="Close"
135
+ >
136
+ <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
137
+ <path d="M18 6L6 18M6 6l12 12" />
138
+ </svg>
139
+ </button>
140
+ </div>
141
+
142
+ {/* Agent Info */}
143
+ <div className="flex flex-col items-center p-6 border-b border-white/10">
144
+ {/* Large Avatar */}
145
+ <div className="relative mb-4">
146
+ <div
147
+ className="w-24 h-24 rounded-2xl flex items-center justify-center text-3xl font-bold shadow-lg"
148
+ style={{
149
+ background: `linear-gradient(135deg, ${colors.primary}, ${colors.primary}99)`,
150
+ boxShadow: isOnline ? `0 4px 20px ${colors.primary}50` : 'none',
151
+ }}
152
+ >
153
+ <span style={{ color: colors.text }}>{initials}</span>
154
+ </div>
155
+ {/* Status indicator */}
156
+ <div
157
+ className={`absolute bottom-1 right-1 w-5 h-5 rounded-full border-4 border-[#1a1d21] ${isOnline ? 'animate-pulse' : ''}`}
158
+ style={{
159
+ backgroundColor: statusColor,
160
+ boxShadow: isOnline ? `0 0 8px ${statusColor}` : 'none',
161
+ }}
162
+ />
163
+ </div>
164
+
165
+ {/* Name */}
166
+ <h3 className="text-xl font-semibold text-[#d1d2d3] mb-1">
167
+ {displayName}
168
+ </h3>
169
+
170
+ {/* Breadcrumb */}
171
+ {breadcrumb && (
172
+ <span className="text-sm text-[#8d8d8e] font-mono mb-2">
173
+ {breadcrumb}
174
+ </span>
175
+ )}
176
+
177
+ {/* Title/Role */}
178
+ {profile?.title && (
179
+ <span className="text-sm text-[#a855f7] font-medium mb-2">
180
+ {profile.title}
181
+ </span>
182
+ )}
183
+
184
+ {/* Status */}
185
+ <span
186
+ className="text-sm flex items-center gap-1.5"
187
+ style={{ color: statusColor }}
188
+ >
189
+ <div
190
+ className={`w-2 h-2 rounded-full ${isOnline ? 'animate-pulse' : ''}`}
191
+ style={{ backgroundColor: statusColor }}
192
+ />
193
+ {agent.status.charAt(0).toUpperCase() + agent.status.slice(1)}
194
+ {agent.isProcessing && ' - Thinking...'}
195
+ </span>
196
+
197
+ {/* Provider Display - Prominent like GitHub link for users */}
198
+ {agent.cli && (() => {
199
+ const cliLower = agent.cli.toLowerCase();
200
+ // Map CLI command to provider (e.g., 'agent' -> 'cursor')
201
+ const providerKey = CLI_TO_PROVIDER_MAP[cliLower] || cliLower;
202
+ const providerInfo = PROVIDER_CONFIG[providerKey] || PROVIDER_CONFIG.other;
203
+ return (
204
+ <div
205
+ className="flex items-center gap-3 mt-3 px-3 py-2.5 rounded-lg border"
206
+ style={{
207
+ borderColor: `${providerInfo.color}40`,
208
+ backgroundColor: `${providerInfo.color}10`,
209
+ }}
210
+ >
211
+ <span className="text-lg">{providerInfo.icon}</span>
212
+ <div className="flex flex-col gap-0.5">
213
+ <span
214
+ className="text-sm font-medium"
215
+ style={{ color: providerInfo.color }}
216
+ >
217
+ {providerInfo.label}
218
+ </span>
219
+ {(agent.model || profile?.model) && (
220
+ <span
221
+ className="text-[11px] font-mono px-1.5 py-0.5 rounded w-fit"
222
+ style={{
223
+ color: providerInfo.color,
224
+ backgroundColor: `${providerInfo.color}15`,
225
+ }}
226
+ >
227
+ {agent.model || profile?.model}
228
+ </span>
229
+ )}
230
+ </div>
231
+ </div>
232
+ );
233
+ })()}
234
+
235
+ {/* Tags */}
236
+ <div className="flex flex-wrap gap-2 mt-3">
237
+ {agent.isSpawned && (
238
+ <span className="text-xs bg-[#a855f7]/20 text-[#a855f7] px-2 py-1 rounded uppercase font-medium">
239
+ Spawned
240
+ </span>
241
+ )}
242
+ {agent.team && (
243
+ <span className="text-xs bg-[#00d9ff]/20 text-[#00d9ff] px-2 py-1 rounded">
244
+ {agent.team}
245
+ </span>
246
+ )}
247
+ {profile?.personaName && (
248
+ <span className="text-xs bg-[#10b981]/20 text-[#10b981] px-2 py-1 rounded">
249
+ {profile.personaName}
250
+ </span>
251
+ )}
252
+ </div>
253
+ </div>
254
+
255
+ {/* Details - Scrollable */}
256
+ <div className="flex-1 p-4 overflow-y-auto">
257
+ <div className="space-y-4">
258
+ {/* Recent Work Summary - Top of details for visibility */}
259
+ {summary && (summary.completedTasks?.length || summary.currentTask || summary.context) && (
260
+ <div className="bg-[#1e2024] border border-[#00d9ff]/20 rounded-lg p-3">
261
+ <label className="text-xs text-[#00d9ff] uppercase tracking-wide font-medium flex items-center gap-1.5 mb-2">
262
+ <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
263
+ <path d="M3 12h4l3 9 4-18 3 9h4" />
264
+ </svg>
265
+ Recent Work
266
+ </label>
267
+
268
+ {/* Current Task */}
269
+ {summary.currentTask && (
270
+ <div className="mb-2">
271
+ <span className="text-xs text-[#8d8d8e]">Working on:</span>
272
+ <p className="text-sm text-[#d1d2d3] mt-0.5 bg-[#2a2d31] p-2 rounded">
273
+ {summary.currentTask}
274
+ </p>
275
+ </div>
276
+ )}
277
+
278
+ {/* Completed Tasks */}
279
+ {summary.completedTasks && summary.completedTasks.length > 0 && (
280
+ <div className="mb-2">
281
+ <span className="text-xs text-[#8d8d8e]">Completed:</span>
282
+ <ul className="mt-1 space-y-1">
283
+ {summary.completedTasks.slice(0, 5).map((task, i) => (
284
+ <li key={i} className="text-xs text-[#d1d2d3] flex items-start gap-1.5">
285
+ <span className="text-[#10b981] mt-0.5">✓</span>
286
+ <span>{task}</span>
287
+ </li>
288
+ ))}
289
+ {summary.completedTasks.length > 5 && (
290
+ <li className="text-xs text-[#8d8d8e]">
291
+ +{summary.completedTasks.length - 5} more...
292
+ </li>
293
+ )}
294
+ </ul>
295
+ </div>
296
+ )}
297
+
298
+ {/* Context */}
299
+ {summary.context && (
300
+ <div className="mb-2">
301
+ <span className="text-xs text-[#8d8d8e]">Context:</span>
302
+ <p className="text-xs text-[#a0a0b0] mt-0.5 italic">
303
+ {summary.context}
304
+ </p>
305
+ </div>
306
+ )}
307
+
308
+ {/* Files touched */}
309
+ {summary.files && summary.files.length > 0 && (
310
+ <div>
311
+ <span className="text-xs text-[#8d8d8e]">Files:</span>
312
+ <div className="flex flex-wrap gap-1 mt-1">
313
+ {summary.files.slice(0, 6).map((file, i) => (
314
+ <span
315
+ key={i}
316
+ className="text-[10px] font-mono bg-[#2a2d31] text-[#a855f7] px-1.5 py-0.5 rounded"
317
+ title={file}
318
+ >
319
+ {file.split('/').pop()}
320
+ </span>
321
+ ))}
322
+ {summary.files.length > 6 && (
323
+ <span className="text-[10px] text-[#8d8d8e]">
324
+ +{summary.files.length - 6}
325
+ </span>
326
+ )}
327
+ </div>
328
+ </div>
329
+ )}
330
+
331
+ {/* Last updated */}
332
+ {summary.lastUpdated && (
333
+ <div className="mt-2 pt-2 border-t border-[#2a2d31]">
334
+ <span className="text-[10px] text-[#606070]">
335
+ Updated {formatRelativeTime(summary.lastUpdated)}
336
+ </span>
337
+ </div>
338
+ )}
339
+ </div>
340
+ )}
341
+
342
+ {/* Description */}
343
+ {profile?.description && (
344
+ <div>
345
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Description</label>
346
+ <p className="text-sm text-[#d1d2d3] mt-1">
347
+ {profile.description}
348
+ </p>
349
+ </div>
350
+ )}
351
+
352
+ {/* Current Task */}
353
+ {agent.currentTask && (
354
+ <div>
355
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Current Task</label>
356
+ <p className="text-sm text-[#d1d2d3] mt-1 bg-[#2a2d31] p-2 rounded">
357
+ {agent.currentTask}
358
+ </p>
359
+ </div>
360
+ )}
361
+
362
+ {/* Spawn Prompt */}
363
+ {profile?.spawnPrompt && (
364
+ <div>
365
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide flex items-center justify-between">
366
+ <span>Spawn Prompt</span>
367
+ {profile.spawnPrompt.length > 200 && (
368
+ <button
369
+ onClick={() => setShowFullPrompt(!showFullPrompt)}
370
+ className="text-[#a855f7] hover:text-[#c084fc] text-xs font-normal"
371
+ >
372
+ {showFullPrompt ? 'Show less' : 'Show more'}
373
+ </button>
374
+ )}
375
+ </label>
376
+ <pre className={`text-sm text-[#d1d2d3] mt-1 bg-[#2a2d31] p-3 rounded font-mono whitespace-pre-wrap ${!showFullPrompt && profile.spawnPrompt.length > 200 ? 'line-clamp-4' : ''}`}>
377
+ {profile.spawnPrompt}
378
+ </pre>
379
+ </div>
380
+ )}
381
+
382
+ {/* Persona Prompt */}
383
+ {profile?.personaPrompt && (
384
+ <div>
385
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide flex items-center justify-between">
386
+ <span>Agent Persona</span>
387
+ {profile.personaPrompt.length > 200 && (
388
+ <button
389
+ onClick={() => setShowFullPersona(!showFullPersona)}
390
+ className="text-[#a855f7] hover:text-[#c084fc] text-xs font-normal"
391
+ >
392
+ {showFullPersona ? 'Show less' : 'Show more'}
393
+ </button>
394
+ )}
395
+ </label>
396
+ <pre className={`text-sm text-[#d1d2d3] mt-1 bg-[#2a2d31] p-3 rounded font-mono whitespace-pre-wrap ${!showFullPersona && profile.personaPrompt.length > 200 ? 'line-clamp-4' : ''}`}>
397
+ {profile.personaPrompt}
398
+ </pre>
399
+ </div>
400
+ )}
401
+
402
+ {/* Model */}
403
+ {(agent.model || profile?.model) && (
404
+ <div>
405
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Model</label>
406
+ <p className="text-sm text-[#d1d2d3] mt-1 font-mono">
407
+ {agent.model || profile?.model}
408
+ </p>
409
+ </div>
410
+ )}
411
+
412
+ {/* Working Directory */}
413
+ {profile?.workingDirectory && (
414
+ <div>
415
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Working Directory</label>
416
+ <p className="text-sm text-[#d1d2d3] mt-1 font-mono bg-[#2a2d31] p-2 rounded truncate" title={profile.workingDirectory}>
417
+ {profile.workingDirectory}
418
+ </p>
419
+ </div>
420
+ )}
421
+
422
+ {/* Agent ID */}
423
+ {agent.agentId && (
424
+ <div>
425
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Agent ID</label>
426
+ <p className="text-sm text-[#d1d2d3] mt-1 font-mono bg-[#2a2d31] p-2 rounded">
427
+ {agent.agentId}
428
+ </p>
429
+ </div>
430
+ )}
431
+
432
+ {/* Capabilities */}
433
+ {profile?.capabilities && profile.capabilities.length > 0 && (
434
+ <div>
435
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Capabilities</label>
436
+ <div className="flex flex-wrap gap-1.5 mt-1">
437
+ {profile.capabilities.map((cap, i) => (
438
+ <span key={i} className="text-xs bg-[#2a2d31] text-[#d1d2d3] px-2 py-1 rounded">
439
+ {cap}
440
+ </span>
441
+ ))}
442
+ </div>
443
+ </div>
444
+ )}
445
+
446
+ {/* Last Seen */}
447
+ {agent.lastSeen && (
448
+ <div>
449
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Last Seen</label>
450
+ <p className="text-sm text-[#d1d2d3] mt-1">
451
+ {formatDateTime(agent.lastSeen)}
452
+ </p>
453
+ </div>
454
+ )}
455
+
456
+ {/* First Seen */}
457
+ {profile?.firstSeen && (
458
+ <div>
459
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">First Seen</label>
460
+ <p className="text-sm text-[#d1d2d3] mt-1">
461
+ {formatDateTime(profile.firstSeen)}
462
+ </p>
463
+ </div>
464
+ )}
465
+
466
+ {/* Message Count */}
467
+ {agent.messageCount !== undefined && agent.messageCount > 0 && (
468
+ <div>
469
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Messages</label>
470
+ <p className="text-sm text-[#d1d2d3] mt-1">
471
+ {agent.messageCount} messages sent
472
+ </p>
473
+ </div>
474
+ )}
475
+ </div>
476
+ </div>
477
+
478
+ {/* Actions */}
479
+ <div className="p-4 border-t border-white/10 space-y-2">
480
+ {/* Message Button */}
481
+ {onMessage && (
482
+ <button
483
+ onClick={() => {
484
+ onMessage(agent);
485
+ onClose();
486
+ }}
487
+ className="w-full flex items-center justify-center gap-2 py-2.5 bg-[#a855f7] hover:bg-[#9333ea] text-white font-medium rounded-lg transition-colors"
488
+ >
489
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
490
+ <path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" />
491
+ </svg>
492
+ Send Message
493
+ </button>
494
+ )}
495
+
496
+ {/* Logs Button */}
497
+ {agent.isSpawned && onLogs && (
498
+ <button
499
+ onClick={() => {
500
+ onLogs(agent);
501
+ onClose();
502
+ }}
503
+ className="w-full flex items-center justify-center gap-2 py-2.5 border border-[#00d9ff]/30 text-[#00d9ff] hover:bg-[#00d9ff]/10 font-medium rounded-lg transition-colors"
504
+ >
505
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
506
+ <polyline points="4 17 10 11 4 5" />
507
+ <line x1="12" y1="19" x2="20" y2="19" />
508
+ </svg>
509
+ View Logs
510
+ </button>
511
+ )}
512
+
513
+ {/* Release Button */}
514
+ {agent.isSpawned && onRelease && (
515
+ <button
516
+ onClick={() => {
517
+ if (confirm(`Are you sure you want to release ${displayName}?`)) {
518
+ onRelease(agent);
519
+ onClose();
520
+ }
521
+ }}
522
+ className="w-full flex items-center justify-center gap-2 py-2.5 border border-[#ff6b6b]/30 text-[#ff6b6b] hover:bg-[#ff6b6b]/10 font-medium rounded-lg transition-colors"
523
+ >
524
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
525
+ <circle cx="12" cy="12" r="10" />
526
+ <line x1="15" y1="9" x2="9" y2="15" />
527
+ <line x1="9" y1="9" x2="15" y2="15" />
528
+ </svg>
529
+ Release Agent
530
+ </button>
531
+ )}
532
+ </div>
533
+ </div>
534
+ </>
535
+ );
536
+ }
537
+
538
+ /**
539
+ * Format a timestamp to a readable date/time
540
+ */
541
+ function formatDateTime(timestamp: string): string {
542
+ const date = new Date(timestamp);
543
+ return date.toLocaleString([], {
544
+ month: 'short',
545
+ day: 'numeric',
546
+ year: 'numeric',
547
+ hour: '2-digit',
548
+ minute: '2-digit',
549
+ });
550
+ }
551
+
552
+ /**
553
+ * Format a timestamp as relative time (e.g., "5 min ago")
554
+ */
555
+ function formatRelativeTime(timestamp: string): string {
556
+ const date = new Date(timestamp);
557
+ const now = new Date();
558
+ const diffMs = now.getTime() - date.getTime();
559
+ const diffSec = Math.floor(diffMs / 1000);
560
+ const diffMin = Math.floor(diffSec / 60);
561
+ const diffHr = Math.floor(diffMin / 60);
562
+ const diffDay = Math.floor(diffHr / 24);
563
+
564
+ if (diffSec < 60) return 'just now';
565
+ if (diffMin < 60) return `${diffMin} min ago`;
566
+ if (diffHr < 24) return `${diffHr} hr ago`;
567
+ if (diffDay < 7) return `${diffDay} day${diffDay > 1 ? 's' : ''} ago`;
568
+ return formatDateTime(timestamp);
569
+ }