@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,311 @@
1
+ /**
2
+ * Header Component - Mission Control Theme
3
+ *
4
+ * Top navigation bar with current context, quick actions,
5
+ * and command palette trigger.
6
+ */
7
+
8
+ import React from 'react';
9
+ import type { Agent, Project } from '../../types';
10
+ import { getAgentColor, getAgentInitials } from '../../lib/colors';
11
+ import { getAgentBreadcrumb } from '../../lib/hierarchy';
12
+ import { RepoContextHeader } from './RepoContextHeader';
13
+ import { WorkspaceStatusIndicator } from '../WorkspaceStatusIndicator';
14
+
15
+ export interface HeaderProps {
16
+ currentChannel: string;
17
+ selectedAgent?: Agent | null;
18
+ /** Connected projects for multi-repo indicator */
19
+ projects?: Project[];
20
+ /** Currently active project */
21
+ currentProject?: Project | null;
22
+ /** Recently accessed projects for quick switching */
23
+ recentProjects?: Project[];
24
+ /** Current view mode */
25
+ viewMode?: 'local' | 'fleet' | 'channels';
26
+ /** Selected channel name (for channels view) */
27
+ selectedChannelName?: string;
28
+ /** Callback when user switches project */
29
+ onProjectChange?: (project: Project) => void;
30
+ onCommandPaletteOpen?: () => void;
31
+ onSettingsClick?: () => void;
32
+ onHistoryClick?: () => void;
33
+ onNewConversationClick?: () => void;
34
+ /** Fleet view toggle */
35
+ onFleetClick?: () => void;
36
+ /** Whether fleet view is currently active */
37
+ isFleetViewActive?: boolean;
38
+ /** Trajectory viewer toggle */
39
+ onTrajectoryClick?: () => void;
40
+ /** Whether there's an active trajectory */
41
+ hasActiveTrajectory?: boolean;
42
+ /** Mobile: open sidebar handler */
43
+ onMenuClick?: () => void;
44
+ /** Show notification badge on mobile menu button */
45
+ hasUnreadNotifications?: boolean;
46
+ }
47
+
48
+ export function Header({
49
+ currentChannel,
50
+ selectedAgent,
51
+ projects = [],
52
+ currentProject,
53
+ recentProjects = [],
54
+ viewMode,
55
+ selectedChannelName,
56
+ onProjectChange,
57
+ onCommandPaletteOpen,
58
+ onSettingsClick,
59
+ onHistoryClick,
60
+ onNewConversationClick,
61
+ onFleetClick,
62
+ isFleetViewActive,
63
+ onTrajectoryClick,
64
+ hasActiveTrajectory,
65
+ onMenuClick,
66
+ hasUnreadNotifications,
67
+ }: HeaderProps) {
68
+ const colors = selectedAgent ? getAgentColor(selectedAgent.name) : null;
69
+ return (
70
+ <header className="h-[52px] bg-bg-secondary border-b border-border-subtle flex items-center justify-between px-2 sm:px-4">
71
+ {/* Mobile hamburger menu button - always visible on mobile */}
72
+ <button
73
+ className="flex md:hidden items-center justify-center w-10 h-10 sm:w-11 sm:h-11 bg-transparent border-none text-text-primary cursor-pointer rounded-lg transition-colors hover:bg-bg-hover active:bg-bg-hover relative flex-shrink-0"
74
+ onClick={onMenuClick}
75
+ aria-label="Open menu"
76
+ >
77
+ <MenuIcon />
78
+ {hasUnreadNotifications && (
79
+ <span className="absolute top-1 right-1 sm:top-1.5 sm:right-1.5 w-2.5 h-2.5 bg-error rounded-full animate-pulse shadow-[0_0_8px_rgba(239,68,68,0.6)]" />
80
+ )}
81
+ </button>
82
+
83
+ {/* Repo Context Switcher */}
84
+ {projects.length > 0 && onProjectChange && (
85
+ <div className="max-md:hidden mr-3">
86
+ <RepoContextHeader
87
+ projects={projects}
88
+ recentProjects={recentProjects}
89
+ currentProject={currentProject ?? null}
90
+ onProjectChange={onProjectChange}
91
+ />
92
+ </div>
93
+ )}
94
+
95
+ {/* Divider when repo context is shown */}
96
+ {projects.length > 0 && onProjectChange && (
97
+ <div className="w-px h-6 bg-border-subtle mr-3 max-md:hidden" />
98
+ )}
99
+
100
+ {/* Workspace Status Indicator */}
101
+ <WorkspaceStatusIndicator className="max-md:hidden mr-3" />
102
+
103
+ {/* Divider after workspace status */}
104
+ <div className="w-px h-6 bg-border-subtle mr-3 max-md:hidden" />
105
+
106
+ <div className="flex items-center gap-2 sm:gap-3 flex-1 min-w-0">
107
+ {selectedAgent && !selectedAgent.isHuman && selectedAgent.cli !== 'dashboard' ? (
108
+ <>
109
+ <div
110
+ className="w-7 h-7 sm:w-8 sm:h-8 rounded-lg flex items-center justify-center font-semibold text-[10px] sm:text-xs border-2 flex-shrink-0"
111
+ style={{
112
+ backgroundColor: colors?.primary,
113
+ borderColor: colors?.primary,
114
+ boxShadow: `0 0 12px ${colors?.primary}40`,
115
+ }}
116
+ >
117
+ <span style={{ color: colors?.text }}>
118
+ {getAgentInitials(selectedAgent.name)}
119
+ </span>
120
+ </div>
121
+ <div className="flex flex-col min-w-0">
122
+ <span className="font-display font-semibold text-sm sm:text-base text-text-primary truncate">
123
+ {selectedAgent.name}
124
+ </span>
125
+ <span className="text-text-muted text-xs font-mono hidden md:block truncate">
126
+ {getAgentBreadcrumb(selectedAgent.name)}
127
+ </span>
128
+ </div>
129
+ {selectedAgent.status && (
130
+ <span
131
+ className={`hidden sm:inline text-xs py-1 px-2.5 rounded-full font-medium ml-2 flex-shrink-0 truncate max-w-[80px] ${
132
+ selectedAgent.status === 'online'
133
+ ? 'bg-success/20 text-success'
134
+ : 'bg-bg-tertiary text-text-muted'
135
+ }`}
136
+ title={selectedAgent.status}
137
+ >
138
+ {selectedAgent.status}
139
+ </span>
140
+ )}
141
+ </>
142
+ ) : selectedAgent && (selectedAgent.isHuman || selectedAgent.cli === 'dashboard') ? (
143
+ // Human user DM - don't display channel name in header
144
+ null
145
+ ) : (
146
+ <>
147
+ <span className="text-accent-cyan text-base sm:text-lg font-mono">@</span>
148
+ <span className="font-display font-semibold text-sm sm:text-base text-text-primary truncate">{currentChannel}</span>
149
+ </>
150
+ )}
151
+ </div>
152
+
153
+ <div className="flex items-center gap-1 sm:gap-2">
154
+ <button
155
+ className="flex items-center gap-1 sm:gap-2 py-1.5 sm:py-2 px-2 sm:px-4 bg-gradient-to-r from-accent-cyan to-[#00b8d9] text-bg-deep font-semibold border-none rounded-lg text-xs sm:text-sm cursor-pointer transition-all duration-150 hover:shadow-glow-cyan hover:-translate-y-0.5"
156
+ onClick={onNewConversationClick}
157
+ title="Start new conversation (⌘N)"
158
+ >
159
+ <NewMessageIcon />
160
+ <span className="hidden sm:inline">New</span>
161
+ <span className="hidden md:inline">Message</span>
162
+ </button>
163
+
164
+ <button
165
+ className="flex items-center gap-1 sm:gap-2 py-1.5 sm:py-2 px-2 sm:px-3 bg-bg-tertiary border border-border-subtle rounded-lg text-text-secondary text-xs sm:text-sm cursor-pointer transition-all duration-150 hover:bg-bg-elevated hover:border-border-medium hover:text-text-primary"
166
+ onClick={onCommandPaletteOpen}
167
+ title="Command Palette (⌘K)"
168
+ >
169
+ <SearchIcon />
170
+ <span className="hidden md:inline">Search</span>
171
+ <kbd className="hidden md:inline bg-bg-card border border-border-subtle rounded px-1.5 py-0.5 text-xs text-text-muted font-mono">
172
+ ⌘K
173
+ </kbd>
174
+ </button>
175
+
176
+ <button
177
+ className="hidden sm:flex items-center justify-center p-1.5 sm:p-2 bg-bg-tertiary border border-border-subtle rounded-lg text-text-secondary cursor-pointer transition-all duration-150 hover:bg-bg-elevated hover:border-border-medium hover:text-accent-cyan"
178
+ onClick={onHistoryClick}
179
+ title="Message History"
180
+ >
181
+ <HistoryIcon />
182
+ </button>
183
+
184
+ {/* Fleet Overview toggle (hidden on small mobile) */}
185
+ {onFleetClick && (
186
+ <button
187
+ className={`hidden sm:flex items-center justify-center p-1.5 sm:p-2 border rounded-lg cursor-pointer transition-all duration-150 ${
188
+ isFleetViewActive
189
+ ? 'bg-accent-cyan/20 border-accent-cyan text-accent-cyan'
190
+ : 'bg-bg-tertiary border-border-subtle text-text-secondary hover:bg-bg-elevated hover:border-border-medium hover:text-accent-cyan'
191
+ }`}
192
+ onClick={onFleetClick}
193
+ title={isFleetViewActive ? 'Back to Chat' : 'Fleet Overview'}
194
+ >
195
+ <FleetIcon />
196
+ </button>
197
+ )}
198
+
199
+ {/* Trajectory Viewer toggle (hidden on mobile) */}
200
+ {onTrajectoryClick && (
201
+ <button
202
+ className={`hidden md:flex items-center justify-center p-2 border rounded-lg cursor-pointer transition-all duration-150 relative ${
203
+ hasActiveTrajectory
204
+ ? 'bg-accent-cyan/20 border-accent-cyan text-accent-cyan'
205
+ : 'bg-bg-tertiary border-border-subtle text-text-secondary hover:bg-bg-elevated hover:border-border-medium hover:text-accent-cyan'
206
+ }`}
207
+ onClick={onTrajectoryClick}
208
+ title="Trajectory Viewer"
209
+ >
210
+ <TrajectoryIcon />
211
+ {hasActiveTrajectory && (
212
+ <span className="absolute -top-1 -right-1 w-2 h-2 bg-accent-cyan rounded-full animate-pulse" />
213
+ )}
214
+ </button>
215
+ )}
216
+
217
+ <a
218
+ href="/metrics"
219
+ className="hidden sm:flex items-center justify-center p-1.5 sm:p-2 bg-bg-tertiary border border-border-subtle rounded-lg text-text-secondary cursor-pointer transition-all duration-150 hover:bg-bg-elevated hover:border-border-medium hover:text-accent-orange no-underline"
220
+ title="Fleet Metrics"
221
+ >
222
+ <MetricsIcon />
223
+ </a>
224
+
225
+ <button
226
+ className="flex items-center justify-center p-1.5 sm:p-2 bg-bg-tertiary border border-border-subtle rounded-lg text-text-secondary cursor-pointer transition-all duration-150 hover:bg-bg-elevated hover:border-border-medium hover:text-accent-purple"
227
+ onClick={onSettingsClick}
228
+ title="Settings"
229
+ >
230
+ <SettingsIcon />
231
+ </button>
232
+ </div>
233
+ </header>
234
+ );
235
+ }
236
+
237
+ function NewMessageIcon() {
238
+ return (
239
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2.5">
240
+ <path d="M12 20h9" />
241
+ <path d="M16.5 3.5a2.121 2.121 0 0 1 3 3L7 19l-4 1 1-4L16.5 3.5z" />
242
+ </svg>
243
+ );
244
+ }
245
+
246
+ function SearchIcon() {
247
+ return (
248
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
249
+ <circle cx="11" cy="11" r="8" />
250
+ <line x1="21" y1="21" x2="16.65" y2="16.65" />
251
+ </svg>
252
+ );
253
+ }
254
+
255
+ function HistoryIcon() {
256
+ return (
257
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
258
+ <circle cx="12" cy="12" r="10" />
259
+ <polyline points="12 6 12 12 16 14" />
260
+ </svg>
261
+ );
262
+ }
263
+
264
+ function MetricsIcon() {
265
+ return (
266
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
267
+ <path d="M3 3v18h18" />
268
+ <path d="M18 17V9" />
269
+ <path d="M13 17V5" />
270
+ <path d="M8 17v-3" />
271
+ </svg>
272
+ );
273
+ }
274
+
275
+ function FleetIcon() {
276
+ return (
277
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
278
+ <rect x="2" y="3" width="20" height="14" rx="2" ry="2" />
279
+ <line x1="8" y1="21" x2="16" y2="21" />
280
+ <line x1="12" y1="17" x2="12" y2="21" />
281
+ </svg>
282
+ );
283
+ }
284
+
285
+ function TrajectoryIcon() {
286
+ return (
287
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
288
+ <path d="M3 12h4l3 9 4-18 3 9h4" />
289
+ </svg>
290
+ );
291
+ }
292
+
293
+ function SettingsIcon() {
294
+ return (
295
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
296
+ <circle cx="12" cy="12" r="3" />
297
+ <path d="M19.4 15a1.65 1.65 0 0 0 .33 1.82l.06.06a2 2 0 0 1 0 2.83 2 2 0 0 1-2.83 0l-.06-.06a1.65 1.65 0 0 0-1.82-.33 1.65 1.65 0 0 0-1 1.51V21a2 2 0 0 1-2 2 2 2 0 0 1-2-2v-.09A1.65 1.65 0 0 0 9 19.4a1.65 1.65 0 0 0-1.82.33l-.06.06a2 2 0 0 1-2.83 0 2 2 0 0 1 0-2.83l.06-.06a1.65 1.65 0 0 0 .33-1.82 1.65 1.65 0 0 0-1.51-1H3a2 2 0 0 1-2-2 2 2 0 0 1 2-2h.09A1.65 1.65 0 0 0 4.6 9a1.65 1.65 0 0 0-.33-1.82l-.06-.06a2 2 0 0 1 0-2.83 2 2 0 0 1 2.83 0l.06.06a1.65 1.65 0 0 0 1.82.33H9a1.65 1.65 0 0 0 1-1.51V3a2 2 0 0 1 2-2 2 2 0 0 1 2 2v.09a1.65 1.65 0 0 0 1 1.51 1.65 1.65 0 0 0 1.82-.33l.06-.06a2 2 0 0 1 2.83 0 2 2 0 0 1 0 2.83l-.06.06a1.65 1.65 0 0 0-.33 1.82V9a1.65 1.65 0 0 0 1.51 1H21a2 2 0 0 1 2 2 2 2 0 0 1-2 2h-.09a1.65 1.65 0 0 0-1.51 1z" />
298
+ </svg>
299
+ );
300
+ }
301
+
302
+ function MenuIcon() {
303
+ return (
304
+ <svg width="24" height="24" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
305
+ <line x1="3" y1="12" x2="21" y2="12" />
306
+ <line x1="3" y1="6" x2="21" y2="6" />
307
+ <line x1="3" y1="18" x2="21" y2="18" />
308
+ </svg>
309
+ );
310
+ }
311
+
@@ -0,0 +1,361 @@
1
+ /**
2
+ * RepoContextHeader Component
3
+ *
4
+ * Slack-style repo indicator in the header with quick switcher dropdown.
5
+ * Shows current repo/project and allows switching between connected projects.
6
+ */
7
+
8
+ import React, { useState, useRef, useEffect } from 'react';
9
+ import type { Project } from '../../types';
10
+
11
+ export interface RepoContextHeaderProps {
12
+ /** All connected projects */
13
+ projects: Project[];
14
+ /** Recently accessed projects (subset of projects) */
15
+ recentProjects?: Project[];
16
+ /** Currently active project */
17
+ currentProject: Project | null;
18
+ /** Callback when user selects a project */
19
+ onProjectChange: (project: Project) => void;
20
+ }
21
+
22
+ export function RepoContextHeader({
23
+ projects,
24
+ recentProjects = [],
25
+ currentProject,
26
+ onProjectChange,
27
+ }: RepoContextHeaderProps) {
28
+ const [isOpen, setIsOpen] = useState(false);
29
+ const [focusedIndex, setFocusedIndex] = useState(-1);
30
+ const dropdownRef = useRef<HTMLDivElement>(null);
31
+ const buttonRefs = useRef<(HTMLButtonElement | null)[]>([]);
32
+
33
+ const hasMultipleProjects = projects.length > 1;
34
+
35
+ // Get IDs of recent projects for filtering
36
+ const recentIds = new Set(recentProjects.map((p) => p.id));
37
+
38
+ // Projects not in recent list
39
+ const otherProjects = projects.filter((p) => !recentIds.has(p.id));
40
+
41
+ // Combined list for keyboard navigation: recent first, then others
42
+ const allItems = [...recentProjects, ...otherProjects];
43
+
44
+ // Format project name for display (extract org/repo from path)
45
+ const formatProjectName = (project: Project | null): string => {
46
+ if (!project) return 'No project';
47
+
48
+ if (project.name) return project.name;
49
+
50
+ // Extract last two path segments for "org/repo" format
51
+ const segments = project.path.split('/').filter(Boolean);
52
+ if (segments.length >= 2) {
53
+ return `${segments[segments.length - 2]}/${segments[segments.length - 1]}`;
54
+ }
55
+ return segments[segments.length - 1] || project.id;
56
+ };
57
+
58
+ // Close on outside click
59
+ useEffect(() => {
60
+ const handleClickOutside = (event: MouseEvent) => {
61
+ if (dropdownRef.current && !dropdownRef.current.contains(event.target as Node)) {
62
+ setIsOpen(false);
63
+ }
64
+ };
65
+
66
+ document.addEventListener('mousedown', handleClickOutside);
67
+ return () => document.removeEventListener('mousedown', handleClickOutside);
68
+ }, []);
69
+
70
+ // Handle keyboard navigation
71
+ useEffect(() => {
72
+ const handleKeyDown = (event: KeyboardEvent) => {
73
+ if (!isOpen) return;
74
+
75
+ switch (event.key) {
76
+ case 'Escape':
77
+ setIsOpen(false);
78
+ setFocusedIndex(-1);
79
+ break;
80
+ case 'ArrowDown':
81
+ event.preventDefault();
82
+ setFocusedIndex((prev) => {
83
+ const next = prev < allItems.length - 1 ? prev + 1 : 0;
84
+ buttonRefs.current[next]?.focus();
85
+ return next;
86
+ });
87
+ break;
88
+ case 'ArrowUp':
89
+ event.preventDefault();
90
+ setFocusedIndex((prev) => {
91
+ const next = prev > 0 ? prev - 1 : allItems.length - 1;
92
+ buttonRefs.current[next]?.focus();
93
+ return next;
94
+ });
95
+ break;
96
+ case 'Enter':
97
+ if (focusedIndex >= 0 && focusedIndex < allItems.length) {
98
+ onProjectChange(allItems[focusedIndex]);
99
+ setIsOpen(false);
100
+ setFocusedIndex(-1);
101
+ }
102
+ break;
103
+ }
104
+ };
105
+
106
+ document.addEventListener('keydown', handleKeyDown);
107
+ return () => document.removeEventListener('keydown', handleKeyDown);
108
+ }, [isOpen, focusedIndex, allItems, onProjectChange]);
109
+
110
+ // Reset focus when dropdown opens
111
+ useEffect(() => {
112
+ if (isOpen) {
113
+ // Focus current project or first item
114
+ const currentIndex = allItems.findIndex((p) => p.id === currentProject?.id);
115
+ setFocusedIndex(currentIndex >= 0 ? currentIndex : 0);
116
+ } else {
117
+ setFocusedIndex(-1);
118
+ }
119
+ }, [isOpen, allItems, currentProject]);
120
+
121
+ // Don't render if no projects
122
+ if (projects.length === 0) return null;
123
+
124
+ return (
125
+ <div className="relative" ref={dropdownRef}>
126
+ <button
127
+ className={`flex items-center gap-2 px-3 py-1.5 rounded-lg text-sm font-medium transition-all duration-150 border ${
128
+ hasMultipleProjects
129
+ ? 'bg-bg-tertiary/80 border-border-subtle hover:bg-bg-elevated hover:border-border-medium cursor-pointer'
130
+ : 'bg-transparent border-transparent cursor-default'
131
+ }`}
132
+ onClick={() => hasMultipleProjects && setIsOpen(!isOpen)}
133
+ disabled={!hasMultipleProjects}
134
+ aria-expanded={isOpen}
135
+ aria-haspopup="listbox"
136
+ >
137
+ <RepoIcon />
138
+ <span className="text-text-primary truncate max-w-[200px]">
139
+ {formatProjectName(currentProject)}
140
+ </span>
141
+ {hasMultipleProjects && (
142
+ <ChevronIcon isOpen={isOpen} />
143
+ )}
144
+ </button>
145
+
146
+ {/* Dropdown */}
147
+ {isOpen && hasMultipleProjects && (
148
+ <div className="absolute top-[calc(100%+4px)] left-0 min-w-[280px] bg-bg-primary border border-border-subtle rounded-xl shadow-[0_10px_40px_rgba(0,0,0,0.4)] z-[1000] overflow-hidden">
149
+ {/* Header */}
150
+ <div className="px-3 py-2 border-b border-border-subtle">
151
+ <span className="text-xs font-medium text-text-muted uppercase tracking-wider">
152
+ Switch Project
153
+ </span>
154
+ </div>
155
+
156
+ <div className="max-h-[300px] overflow-y-auto" role="listbox">
157
+ {/* Recent Projects Section */}
158
+ {recentProjects.length > 0 && (
159
+ <>
160
+ <div className="px-3 py-1.5 bg-bg-tertiary/50">
161
+ <span className="text-[10px] font-medium text-text-muted uppercase tracking-wider flex items-center gap-1.5">
162
+ <ClockIcon />
163
+ Recent
164
+ </span>
165
+ </div>
166
+ <div className="py-1">
167
+ {recentProjects.map((project, index) => (
168
+ <ProjectItem
169
+ key={project.id}
170
+ project={project}
171
+ index={index}
172
+ isActive={currentProject?.id === project.id}
173
+ isFocused={focusedIndex === index}
174
+ formatProjectName={formatProjectName}
175
+ buttonRefs={buttonRefs}
176
+ onSelect={() => {
177
+ onProjectChange(project);
178
+ setIsOpen(false);
179
+ }}
180
+ />
181
+ ))}
182
+ </div>
183
+ </>
184
+ )}
185
+
186
+ {/* All Projects Section */}
187
+ {otherProjects.length > 0 && (
188
+ <>
189
+ <div className="px-3 py-1.5 bg-bg-tertiary/50">
190
+ <span className="text-[10px] font-medium text-text-muted uppercase tracking-wider">
191
+ {recentProjects.length > 0 ? 'All Projects' : 'Projects'}
192
+ </span>
193
+ </div>
194
+ <div className="py-1">
195
+ {otherProjects.map((project, index) => {
196
+ const globalIndex = recentProjects.length + index;
197
+ return (
198
+ <ProjectItem
199
+ key={project.id}
200
+ project={project}
201
+ index={globalIndex}
202
+ isActive={currentProject?.id === project.id}
203
+ isFocused={focusedIndex === globalIndex}
204
+ formatProjectName={formatProjectName}
205
+ buttonRefs={buttonRefs}
206
+ onSelect={() => {
207
+ onProjectChange(project);
208
+ setIsOpen(false);
209
+ }}
210
+ />
211
+ );
212
+ })}
213
+ </div>
214
+ </>
215
+ )}
216
+ </div>
217
+ </div>
218
+ )}
219
+ </div>
220
+ );
221
+ }
222
+
223
+ interface ProjectItemProps {
224
+ project: Project;
225
+ index: number;
226
+ isActive: boolean;
227
+ isFocused: boolean;
228
+ formatProjectName: (project: Project) => string;
229
+ buttonRefs: React.MutableRefObject<(HTMLButtonElement | null)[]>;
230
+ onSelect: () => void;
231
+ }
232
+
233
+ function ProjectItem({
234
+ project,
235
+ index,
236
+ isActive,
237
+ isFocused,
238
+ formatProjectName,
239
+ buttonRefs,
240
+ onSelect,
241
+ }: ProjectItemProps) {
242
+ const displayName = formatProjectName(project);
243
+ const agentCount = project.agents?.length || 0;
244
+
245
+ return (
246
+ <button
247
+ ref={(el) => { buttonRefs.current[index] = el; }}
248
+ className={`w-full flex items-center gap-3 px-3 py-2.5 text-left transition-colors border-none cursor-pointer ${
249
+ isActive
250
+ ? 'bg-accent-cyan/10 text-accent-cyan'
251
+ : isFocused
252
+ ? 'bg-bg-hover text-text-primary'
253
+ : 'bg-transparent text-text-primary hover:bg-bg-hover'
254
+ }`}
255
+ onClick={onSelect}
256
+ role="option"
257
+ aria-selected={isActive}
258
+ tabIndex={isFocused ? 0 : -1}
259
+ >
260
+ <RepoIcon className={isActive ? 'text-accent-cyan' : 'text-text-muted'} />
261
+
262
+ <div className="flex-1 min-w-0">
263
+ <div className="font-medium text-sm truncate">
264
+ {displayName}
265
+ </div>
266
+ <div className="text-xs text-text-muted truncate">
267
+ {project.path}
268
+ </div>
269
+ </div>
270
+
271
+ {/* Agent count badge */}
272
+ <span className={`text-xs px-2 py-0.5 rounded-full ${
273
+ isActive
274
+ ? 'bg-accent-cyan/20 text-accent-cyan'
275
+ : 'bg-bg-tertiary text-text-muted'
276
+ }`}>
277
+ {agentCount} {agentCount === 1 ? 'agent' : 'agents'}
278
+ </span>
279
+
280
+ {/* Lead indicator */}
281
+ {project.lead?.connected && (
282
+ <span className="w-2 h-2 rounded-full bg-success animate-pulse" title={`Lead: ${project.lead.name}`} />
283
+ )}
284
+
285
+ {/* Checkmark for active */}
286
+ {isActive && (
287
+ <CheckIcon />
288
+ )}
289
+ </button>
290
+ );
291
+ }
292
+
293
+ function ClockIcon() {
294
+ return (
295
+ <svg
296
+ width="10"
297
+ height="10"
298
+ viewBox="0 0 24 24"
299
+ fill="none"
300
+ stroke="currentColor"
301
+ strokeWidth="2"
302
+ strokeLinecap="round"
303
+ strokeLinejoin="round"
304
+ >
305
+ <circle cx="12" cy="12" r="10" />
306
+ <polyline points="12 6 12 12 16 14" />
307
+ </svg>
308
+ );
309
+ }
310
+
311
+ function RepoIcon({ className = 'text-text-muted' }: { className?: string }) {
312
+ return (
313
+ <svg
314
+ width="16"
315
+ height="16"
316
+ viewBox="0 0 24 24"
317
+ fill="none"
318
+ stroke="currentColor"
319
+ strokeWidth="2"
320
+ strokeLinecap="round"
321
+ strokeLinejoin="round"
322
+ className={className}
323
+ >
324
+ <path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z" />
325
+ </svg>
326
+ );
327
+ }
328
+
329
+ function ChevronIcon({ isOpen }: { isOpen: boolean }) {
330
+ return (
331
+ <svg
332
+ className={`text-text-muted transition-transform duration-150 ${isOpen ? 'rotate-180' : ''}`}
333
+ width="12"
334
+ height="12"
335
+ viewBox="0 0 24 24"
336
+ fill="none"
337
+ stroke="currentColor"
338
+ strokeWidth="2.5"
339
+ >
340
+ <polyline points="6 9 12 15 18 9" />
341
+ </svg>
342
+ );
343
+ }
344
+
345
+ function CheckIcon() {
346
+ return (
347
+ <svg
348
+ width="16"
349
+ height="16"
350
+ viewBox="0 0 24 24"
351
+ fill="none"
352
+ stroke="currentColor"
353
+ strokeWidth="2.5"
354
+ strokeLinecap="round"
355
+ strokeLinejoin="round"
356
+ className="text-accent-cyan"
357
+ >
358
+ <polyline points="20 6 9 17 4 12" />
359
+ </svg>
360
+ );
361
+ }