@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,234 @@
1
+ /**
2
+ * Workspace Selector Component
3
+ *
4
+ * Dropdown/list for switching between workspaces (repositories).
5
+ * Connects to the orchestrator API for workspace management.
6
+ */
7
+
8
+ import React, { useState, useRef, useEffect } from 'react';
9
+
10
+ export interface Workspace {
11
+ id: string;
12
+ name: string;
13
+ path: string;
14
+ status: 'active' | 'inactive' | 'error';
15
+ provider: 'claude' | 'codex' | 'gemini' | 'generic';
16
+ gitBranch?: string;
17
+ gitRemote?: string;
18
+ lastActiveAt: Date;
19
+ }
20
+
21
+ export interface WorkspaceSelectorProps {
22
+ workspaces: Workspace[];
23
+ activeWorkspaceId?: string;
24
+ onSelect: (workspace: Workspace) => void;
25
+ onAddWorkspace: () => void;
26
+ onWorkspaceSettings?: () => void;
27
+ isLoading?: boolean;
28
+ }
29
+
30
+ export function WorkspaceSelector({
31
+ workspaces,
32
+ activeWorkspaceId,
33
+ onSelect,
34
+ onAddWorkspace,
35
+ onWorkspaceSettings,
36
+ isLoading = false,
37
+ }: WorkspaceSelectorProps) {
38
+ const [isOpen, setIsOpen] = useState(false);
39
+ const dropdownRef = useRef<HTMLDivElement>(null);
40
+
41
+ const activeWorkspace = workspaces.find((w) => w.id === activeWorkspaceId);
42
+
43
+ // Close on outside click
44
+ useEffect(() => {
45
+ const handleClickOutside = (event: MouseEvent) => {
46
+ if (dropdownRef.current && !dropdownRef.current.contains(event.target as Node)) {
47
+ setIsOpen(false);
48
+ }
49
+ };
50
+
51
+ document.addEventListener('mousedown', handleClickOutside);
52
+ return () => document.removeEventListener('mousedown', handleClickOutside);
53
+ }, []);
54
+
55
+ // Close on escape
56
+ useEffect(() => {
57
+ const handleEscape = (event: KeyboardEvent) => {
58
+ if (event.key === 'Escape') {
59
+ setIsOpen(false);
60
+ }
61
+ };
62
+
63
+ document.addEventListener('keydown', handleEscape);
64
+ return () => document.removeEventListener('keydown', handleEscape);
65
+ }, []);
66
+
67
+ return (
68
+ <div className="relative w-full" ref={dropdownRef}>
69
+ <button
70
+ className="w-full flex items-center gap-2 px-3 py-2.5 bg-bg-secondary border border-border rounded-lg text-text-primary text-sm cursor-pointer transition-all hover:bg-bg-tertiary hover:border-border-medium disabled:opacity-60 disabled:cursor-not-allowed"
71
+ onClick={() => setIsOpen(!isOpen)}
72
+ disabled={isLoading}
73
+ >
74
+ {isLoading ? (
75
+ <span className="flex-1 text-left text-text-muted">Loading...</span>
76
+ ) : activeWorkspace ? (
77
+ <>
78
+ <ProviderIcon provider={activeWorkspace.provider} />
79
+ <span className="flex-1 text-left font-medium">{activeWorkspace.name}</span>
80
+ {activeWorkspace.gitBranch && (
81
+ <span className="flex items-center gap-1 text-xs text-text-muted bg-bg-hover px-1.5 py-0.5 rounded">
82
+ <BranchIcon />
83
+ {activeWorkspace.gitBranch}
84
+ </span>
85
+ )}
86
+ </>
87
+ ) : (
88
+ <span className="flex-1 text-left text-text-muted">Select workspace...</span>
89
+ )}
90
+ <ChevronIcon isOpen={isOpen} />
91
+ </button>
92
+
93
+ {isOpen && (
94
+ <div className="absolute top-[calc(100%+4px)] left-0 right-0 bg-bg-card border border-border rounded-lg shadow-modal z-[1000] overflow-hidden">
95
+ <div className="max-h-[300px] overflow-y-auto">
96
+ {workspaces.length === 0 ? (
97
+ <div className="py-6 px-4 text-center text-text-muted text-[13px] leading-relaxed">
98
+ No workspaces added yet.
99
+ <br />
100
+ Add a repository to get started.
101
+ </div>
102
+ ) : (
103
+ workspaces.map((workspace) => (
104
+ <button
105
+ key={workspace.id}
106
+ className={`w-full flex items-center gap-2.5 px-3 py-2.5 bg-transparent border-none text-text-primary text-sm cursor-pointer transition-colors text-left hover:bg-bg-hover ${
107
+ workspace.id === activeWorkspaceId ? 'bg-success-light' : ''
108
+ }`}
109
+ onClick={() => {
110
+ onSelect(workspace);
111
+ setIsOpen(false);
112
+ }}
113
+ >
114
+ <ProviderIcon provider={workspace.provider} />
115
+ <div className="flex-1 flex flex-col gap-0.5 min-w-0">
116
+ <span className="font-medium">{workspace.name}</span>
117
+ <span className="text-[11px] text-text-muted overflow-hidden text-ellipsis whitespace-nowrap">
118
+ {workspace.path}
119
+ </span>
120
+ </div>
121
+ <StatusIndicator status={workspace.status} />
122
+ </button>
123
+ ))
124
+ )}
125
+ </div>
126
+
127
+ <div className="p-2 border-t border-border space-y-1.5">
128
+ {onWorkspaceSettings && activeWorkspace && (
129
+ <button
130
+ className="w-full flex items-center justify-center gap-1.5 px-3 py-2 bg-transparent border border-border rounded-md text-text-muted text-[13px] cursor-pointer transition-all hover:bg-bg-hover hover:border-border-medium hover:text-text-primary"
131
+ onClick={() => {
132
+ onWorkspaceSettings();
133
+ setIsOpen(false);
134
+ }}
135
+ >
136
+ <SettingsIcon />
137
+ Workspace Settings
138
+ </button>
139
+ )}
140
+ <button
141
+ className="w-full flex items-center justify-center gap-1.5 px-3 py-2 bg-transparent border border-dashed border-border rounded-md text-text-muted text-[13px] cursor-pointer transition-all hover:bg-bg-hover hover:border-border-medium hover:text-text-primary"
142
+ onClick={onAddWorkspace}
143
+ >
144
+ <PlusIcon />
145
+ Add Workspace
146
+ </button>
147
+ </div>
148
+ </div>
149
+ )}
150
+ </div>
151
+ );
152
+ }
153
+
154
+ function ProviderIcon({ provider }: { provider: string }) {
155
+ const icons: Record<string, string> = {
156
+ claude: '🤖',
157
+ codex: '🧠',
158
+ gemini: '✨',
159
+ generic: '📁',
160
+ };
161
+
162
+ return (
163
+ <span className="text-base" title={provider}>
164
+ {icons[provider] || icons.generic}
165
+ </span>
166
+ );
167
+ }
168
+
169
+ function StatusIndicator({ status }: { status: string }) {
170
+ const colors: Record<string, string> = {
171
+ active: 'bg-green-500',
172
+ inactive: 'bg-gray-500',
173
+ error: 'bg-red-500',
174
+ };
175
+
176
+ return (
177
+ <span
178
+ className={`w-2 h-2 rounded-full flex-shrink-0 ${colors[status] || colors.inactive}`}
179
+ title={status}
180
+ />
181
+ );
182
+ }
183
+
184
+ function ChevronIcon({ isOpen }: { isOpen: boolean }) {
185
+ return (
186
+ <svg
187
+ className={`text-text-muted transition-transform ${isOpen ? 'rotate-180' : ''}`}
188
+ width="16"
189
+ height="16"
190
+ viewBox="0 0 24 24"
191
+ fill="none"
192
+ stroke="currentColor"
193
+ strokeWidth="2"
194
+ >
195
+ <polyline points="6 9 12 15 18 9" />
196
+ </svg>
197
+ );
198
+ }
199
+
200
+ function BranchIcon() {
201
+ return (
202
+ <svg
203
+ width="12"
204
+ height="12"
205
+ viewBox="0 0 24 24"
206
+ fill="none"
207
+ stroke="currentColor"
208
+ strokeWidth="2"
209
+ >
210
+ <line x1="6" y1="3" x2="6" y2="15" />
211
+ <circle cx="18" cy="6" r="3" />
212
+ <circle cx="6" cy="18" r="3" />
213
+ <path d="M18 9a9 9 0 0 1-9 9" />
214
+ </svg>
215
+ );
216
+ }
217
+
218
+ function PlusIcon() {
219
+ return (
220
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
221
+ <line x1="12" y1="5" x2="12" y2="19" />
222
+ <line x1="5" y1="12" x2="19" y2="12" />
223
+ </svg>
224
+ );
225
+ }
226
+
227
+ function SettingsIcon() {
228
+ return (
229
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
230
+ <circle cx="12" cy="12" r="3" />
231
+ <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" />
232
+ </svg>
233
+ );
234
+ }
@@ -0,0 +1,396 @@
1
+ /**
2
+ * Workspace Status Indicator
3
+ *
4
+ * Shows workspace status in the dashboard with visual indicators:
5
+ * - Running (green): Workspace is active and ready
6
+ * - Stopped (amber): Workspace is idle, can be woken up
7
+ * - Provisioning (cyan): Workspace is being created
8
+ * - Error (red): Workspace has an issue
9
+ * - None (gray): No workspace exists
10
+ */
11
+
12
+ import React, { useCallback, useState } from 'react';
13
+ import { cloudApi } from '../lib/cloudApi';
14
+ import { useWorkspaceStatus } from './hooks/useWorkspaceStatus';
15
+
16
+ export interface WorkspaceStatusIndicatorProps {
17
+ /** Show expanded view with details (default: false) */
18
+ expanded?: boolean;
19
+ /** Auto-wakeup when workspace is stopped (default: false) */
20
+ autoWakeup?: boolean;
21
+ /** Callback when wakeup is triggered */
22
+ onWakeup?: () => void;
23
+ /** Callback when status changes */
24
+ onStatusChange?: (status: string) => void;
25
+ /** Custom class name */
26
+ className?: string;
27
+ }
28
+
29
+ export function WorkspaceStatusIndicator({
30
+ expanded = false,
31
+ autoWakeup = false,
32
+ onWakeup,
33
+ onStatusChange,
34
+ className = '',
35
+ }: WorkspaceStatusIndicatorProps) {
36
+ const [showToast, setShowToast] = useState(false);
37
+ const [toastMessage, setToastMessage] = useState('');
38
+
39
+ const {
40
+ workspace,
41
+ exists,
42
+ isLoading,
43
+ isWakingUp,
44
+ statusMessage,
45
+ actionNeeded,
46
+ wakeup,
47
+ } = useWorkspaceStatus({
48
+ autoWakeup,
49
+ onStatusChange: (status, wasRestarted) => {
50
+ onStatusChange?.(status);
51
+ if (wasRestarted) {
52
+ setToastMessage('Workspace is starting up...');
53
+ setShowToast(true);
54
+ setTimeout(() => setShowToast(false), 5000);
55
+ } else if (status === 'running') {
56
+ setToastMessage('Workspace is ready!');
57
+ setShowToast(true);
58
+ setTimeout(() => setShowToast(false), 3000);
59
+ }
60
+ },
61
+ });
62
+
63
+ const [isRestarting, setIsRestarting] = useState(false);
64
+
65
+ const handleWakeup = useCallback(async () => {
66
+ const result = await wakeup();
67
+ if (result.success) {
68
+ onWakeup?.();
69
+ setToastMessage(result.message);
70
+ setShowToast(true);
71
+ setTimeout(() => setShowToast(false), 5000);
72
+ }
73
+ }, [wakeup, onWakeup]);
74
+
75
+ const handleRestart = useCallback(async () => {
76
+ if (!workspace) return;
77
+ setIsRestarting(true);
78
+ const result = await cloudApi.restartWorkspace(workspace.id);
79
+ if (result.success) {
80
+ setToastMessage(result.data.message);
81
+ } else {
82
+ setToastMessage(result.error);
83
+ }
84
+ setShowToast(true);
85
+ setTimeout(() => setShowToast(false), 5000);
86
+ setIsRestarting(false);
87
+ }, [workspace]);
88
+
89
+ // Get status color and icon
90
+ const getStatusConfig = () => {
91
+ if (!exists) {
92
+ return {
93
+ color: 'text-text-muted',
94
+ bgColor: 'bg-bg-tertiary',
95
+ borderColor: 'border-border-subtle',
96
+ icon: <NoWorkspaceIcon />,
97
+ label: 'No workspace',
98
+ pulseColor: null,
99
+ };
100
+ }
101
+
102
+ if (isLoading && !workspace) {
103
+ return {
104
+ color: 'text-text-muted',
105
+ bgColor: 'bg-bg-tertiary',
106
+ borderColor: 'border-border-subtle',
107
+ icon: <LoadingIcon />,
108
+ label: 'Loading...',
109
+ pulseColor: null,
110
+ };
111
+ }
112
+
113
+ if (workspace?.isRunning) {
114
+ return {
115
+ color: 'text-success',
116
+ bgColor: 'bg-success/10',
117
+ borderColor: 'border-success/30',
118
+ icon: <RunningIcon />,
119
+ label: 'Running',
120
+ pulseColor: 'bg-success',
121
+ };
122
+ }
123
+
124
+ if (workspace?.isStopped) {
125
+ return {
126
+ color: 'text-amber-400',
127
+ bgColor: 'bg-amber-400/10',
128
+ borderColor: 'border-amber-400/30',
129
+ icon: <StoppedIcon />,
130
+ label: 'Stopped',
131
+ pulseColor: null,
132
+ };
133
+ }
134
+
135
+ if (workspace?.isProvisioning || isWakingUp) {
136
+ return {
137
+ color: 'text-accent-cyan',
138
+ bgColor: 'bg-accent-cyan/10',
139
+ borderColor: 'border-accent-cyan/30',
140
+ icon: <ProvisioningIcon />,
141
+ label: isWakingUp ? 'Starting...' : 'Provisioning',
142
+ pulseColor: 'bg-accent-cyan',
143
+ };
144
+ }
145
+
146
+ if (workspace?.hasError) {
147
+ return {
148
+ color: 'text-error',
149
+ bgColor: 'bg-error/10',
150
+ borderColor: 'border-error/30',
151
+ icon: <ErrorIcon />,
152
+ label: 'Error',
153
+ pulseColor: null,
154
+ };
155
+ }
156
+
157
+ return {
158
+ color: 'text-text-muted',
159
+ bgColor: 'bg-bg-tertiary',
160
+ borderColor: 'border-border-subtle',
161
+ icon: <NoWorkspaceIcon />,
162
+ label: 'Unknown',
163
+ pulseColor: null,
164
+ };
165
+ };
166
+
167
+ const config = getStatusConfig();
168
+
169
+ // Compact indicator (for header)
170
+ if (!expanded) {
171
+ return (
172
+ <div className={`relative ${className}`}>
173
+ <div
174
+ className={`flex items-center gap-2 px-2.5 py-1.5 rounded-lg border ${config.bgColor} ${config.borderColor} cursor-default`}
175
+ title={statusMessage || config.label}
176
+ >
177
+ <span className={config.color}>{config.icon}</span>
178
+ <span
179
+ className={`text-xs font-medium ${config.color} truncate max-w-[100px]`}
180
+ title={statusMessage || config.label}
181
+ >
182
+ {config.label}
183
+ </span>
184
+ {config.pulseColor && (
185
+ <span
186
+ className={`w-2 h-2 rounded-full ${config.pulseColor} animate-pulse`}
187
+ />
188
+ )}
189
+ </div>
190
+
191
+ {/* Wakeup button for stopped state */}
192
+ {actionNeeded === 'wakeup' && !isWakingUp && (
193
+ <button
194
+ onClick={handleWakeup}
195
+ className="ml-2 px-2 py-1 text-xs font-medium text-amber-400 bg-amber-400/10 border border-amber-400/30 rounded hover:bg-amber-400/20 transition-colors"
196
+ >
197
+ Wake up
198
+ </button>
199
+ )}
200
+
201
+ {/* Toast notification */}
202
+ {showToast && (
203
+ <div className="absolute top-full mt-2 left-0 z-50 px-3 py-2 bg-bg-card border border-border-medium rounded-lg shadow-lg text-sm text-text-primary whitespace-nowrap animate-in fade-in slide-in-from-top-2">
204
+ {toastMessage}
205
+ </div>
206
+ )}
207
+ </div>
208
+ );
209
+ }
210
+
211
+ // Expanded view (for sidebar or dedicated panel)
212
+ return (
213
+ <div
214
+ className={`rounded-lg border ${config.borderColor} ${config.bgColor} p-4 ${className}`}
215
+ >
216
+ <div className="flex items-center justify-between mb-3">
217
+ <div className="flex items-center gap-2">
218
+ <span className={config.color}>{config.icon}</span>
219
+ <span className={`text-sm font-semibold ${config.color}`}>
220
+ Workspace Status
221
+ </span>
222
+ </div>
223
+ {config.pulseColor && (
224
+ <span
225
+ className={`w-2.5 h-2.5 rounded-full ${config.pulseColor} animate-pulse`}
226
+ />
227
+ )}
228
+ </div>
229
+
230
+ <div className="space-y-2">
231
+ <div className="flex items-center justify-between">
232
+ <span className="text-xs text-text-muted">Name</span>
233
+ <span className="text-sm text-text-primary font-medium truncate max-w-[150px]">
234
+ {workspace?.name || 'None'}
235
+ </span>
236
+ </div>
237
+
238
+ <div className="flex items-center justify-between">
239
+ <span className="text-xs text-text-muted">Status</span>
240
+ <span className={`text-sm font-medium ${config.color}`}>
241
+ {config.label}
242
+ </span>
243
+ </div>
244
+
245
+ {statusMessage && (
246
+ <p className="text-xs text-text-muted mt-2">{statusMessage}</p>
247
+ )}
248
+
249
+ {/* Action buttons */}
250
+ {actionNeeded === 'wakeup' && !isWakingUp && (
251
+ <button
252
+ onClick={handleWakeup}
253
+ className="w-full mt-3 px-3 py-2 text-sm font-medium text-amber-400 bg-amber-400/10 border border-amber-400/30 rounded-lg hover:bg-amber-400/20 transition-colors"
254
+ >
255
+ Wake up workspace
256
+ </button>
257
+ )}
258
+
259
+ {actionNeeded === 'check_error' && (
260
+ <div className="flex gap-2 mt-3">
261
+ <button
262
+ onClick={handleRestart}
263
+ disabled={isRestarting}
264
+ className="flex-1 px-3 py-2 text-sm font-medium text-accent-cyan bg-accent-cyan/10 border border-accent-cyan/30 rounded-lg hover:bg-accent-cyan/20 transition-colors disabled:opacity-50"
265
+ >
266
+ {isRestarting ? 'Restarting...' : 'Restart'}
267
+ </button>
268
+ <a
269
+ href="/app/settings/workspace"
270
+ className="px-3 py-2 text-sm font-medium text-center text-text-muted bg-bg-tertiary border border-border-subtle rounded-lg hover:bg-bg-hover transition-colors no-underline"
271
+ >
272
+ Settings
273
+ </a>
274
+ </div>
275
+ )}
276
+ </div>
277
+
278
+ {/* Toast notification */}
279
+ {showToast && (
280
+ <div className="mt-3 px-3 py-2 bg-bg-card border border-border-medium rounded-lg text-sm text-text-primary animate-in fade-in">
281
+ {toastMessage}
282
+ </div>
283
+ )}
284
+ </div>
285
+ );
286
+ }
287
+
288
+ // Icons
289
+ function RunningIcon() {
290
+ return (
291
+ <svg
292
+ width="14"
293
+ height="14"
294
+ viewBox="0 0 24 24"
295
+ fill="none"
296
+ stroke="currentColor"
297
+ strokeWidth="2"
298
+ strokeLinecap="round"
299
+ strokeLinejoin="round"
300
+ >
301
+ <polygon points="5 3 19 12 5 21 5 3" />
302
+ </svg>
303
+ );
304
+ }
305
+
306
+ function StoppedIcon() {
307
+ return (
308
+ <svg
309
+ width="14"
310
+ height="14"
311
+ viewBox="0 0 24 24"
312
+ fill="none"
313
+ stroke="currentColor"
314
+ strokeWidth="2"
315
+ strokeLinecap="round"
316
+ strokeLinejoin="round"
317
+ >
318
+ <rect x="6" y="6" width="12" height="12" />
319
+ </svg>
320
+ );
321
+ }
322
+
323
+ function ProvisioningIcon() {
324
+ return (
325
+ <svg
326
+ width="14"
327
+ height="14"
328
+ viewBox="0 0 24 24"
329
+ fill="none"
330
+ stroke="currentColor"
331
+ strokeWidth="2"
332
+ strokeLinecap="round"
333
+ strokeLinejoin="round"
334
+ className="animate-spin"
335
+ >
336
+ <path d="M21 12a9 9 0 1 1-6.219-8.56" />
337
+ </svg>
338
+ );
339
+ }
340
+
341
+ function ErrorIcon() {
342
+ return (
343
+ <svg
344
+ width="14"
345
+ height="14"
346
+ viewBox="0 0 24 24"
347
+ fill="none"
348
+ stroke="currentColor"
349
+ strokeWidth="2"
350
+ strokeLinecap="round"
351
+ strokeLinejoin="round"
352
+ >
353
+ <circle cx="12" cy="12" r="10" />
354
+ <line x1="15" y1="9" x2="9" y2="15" />
355
+ <line x1="9" y1="9" x2="15" y2="15" />
356
+ </svg>
357
+ );
358
+ }
359
+
360
+ function NoWorkspaceIcon() {
361
+ return (
362
+ <svg
363
+ width="14"
364
+ height="14"
365
+ viewBox="0 0 24 24"
366
+ fill="none"
367
+ stroke="currentColor"
368
+ strokeWidth="2"
369
+ strokeLinecap="round"
370
+ strokeLinejoin="round"
371
+ >
372
+ <rect x="3" y="3" width="18" height="18" rx="2" ry="2" />
373
+ <line x1="12" y1="8" x2="12" y2="16" />
374
+ <line x1="8" y1="12" x2="16" y2="12" />
375
+ </svg>
376
+ );
377
+ }
378
+
379
+ function LoadingIcon() {
380
+ return (
381
+ <svg
382
+ width="14"
383
+ height="14"
384
+ viewBox="0 0 24 24"
385
+ fill="none"
386
+ stroke="currentColor"
387
+ strokeWidth="2"
388
+ strokeLinecap="round"
389
+ strokeLinejoin="round"
390
+ className="animate-spin"
391
+ >
392
+ <circle cx="12" cy="12" r="10" strokeOpacity="0.25" />
393
+ <path d="M12 2a10 10 0 0 1 10 10" />
394
+ </svg>
395
+ );
396
+ }