@agent-relay/dashboard 2.0.80 → 2.0.82
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/out/404.html +1 -1
- package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
- package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
- package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
- package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
- package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
- package/out/about.html +2 -2
- package/out/about.txt +1 -1
- package/out/app/onboarding.html +1 -1
- package/out/app/onboarding.txt +1 -1
- package/out/app.html +1 -1
- package/out/app.txt +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
- package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
- package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
- package/out/blog.html +2 -2
- package/out/blog.txt +1 -1
- package/out/careers.html +2 -2
- package/out/careers.txt +1 -1
- package/out/changelog.html +2 -2
- package/out/changelog.txt +1 -1
- package/out/cloud/link.html +1 -1
- package/out/cloud/link.txt +2 -2
- package/out/complete-profile.html +2 -2
- package/out/complete-profile.txt +1 -1
- package/out/connect-repos.html +1 -1
- package/out/connect-repos.txt +1 -1
- package/out/contact.html +2 -2
- package/out/contact.txt +1 -1
- package/out/docs.html +2 -2
- package/out/docs.txt +1 -1
- package/out/history.html +1 -1
- package/out/history.txt +2 -2
- package/out/index.html +1 -1
- package/out/index.txt +2 -2
- package/out/login.html +2 -2
- package/out/login.txt +1 -1
- package/out/metrics.html +1 -1
- package/out/metrics.txt +2 -2
- package/out/pricing.html +2 -2
- package/out/pricing.txt +1 -1
- package/out/privacy.html +2 -2
- package/out/privacy.txt +1 -1
- package/out/providers/setup/claude.html +1 -1
- package/out/providers/setup/claude.txt +1 -1
- package/out/providers/setup/codex.html +1 -1
- package/out/providers/setup/codex.txt +1 -1
- package/out/providers/setup/cursor.html +1 -1
- package/out/providers/setup/cursor.txt +1 -1
- package/out/providers.html +1 -1
- package/out/providers.txt +1 -1
- package/out/security.html +2 -2
- package/out/security.txt +1 -1
- package/out/signup.html +2 -2
- package/out/signup.txt +1 -1
- package/out/terms.html +2 -2
- package/out/terms.txt +1 -1
- package/package.json +7 -1
- package/src/app/about/page.tsx +7 -0
- package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
- package/src/app/app/[[...slug]]/page.tsx +23 -0
- package/src/app/app/onboarding/page.tsx +394 -0
- package/src/app/apple-icon.png +0 -0
- package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
- package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
- package/src/app/blog/page.tsx +15 -0
- package/src/app/careers/page.tsx +7 -0
- package/src/app/changelog/page.tsx +7 -0
- package/src/app/cloud/link/page.tsx +464 -0
- package/src/app/complete-profile/page.tsx +204 -0
- package/src/app/connect-repos/page.tsx +410 -0
- package/src/app/contact/page.tsx +7 -0
- package/src/app/docs/page.tsx +7 -0
- package/src/app/favicon.png +0 -0
- package/src/app/globals.css +200 -0
- package/src/app/history/page.tsx +658 -0
- package/src/app/layout.tsx +25 -0
- package/src/app/login/page.tsx +424 -0
- package/src/app/metrics/page.tsx +781 -0
- package/src/app/page.tsx +59 -0
- package/src/app/pricing/page.tsx +7 -0
- package/src/app/privacy/page.tsx +7 -0
- package/src/app/providers/page.tsx +193 -0
- package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
- package/src/app/providers/setup/[provider]/constants.ts +35 -0
- package/src/app/providers/setup/[provider]/page.tsx +42 -0
- package/src/app/security/page.tsx +7 -0
- package/src/app/signup/page.tsx +533 -0
- package/src/app/terms/page.tsx +7 -0
- package/src/components/ActivityFeed.tsx +216 -0
- package/src/components/AddWorkspaceModal.tsx +170 -0
- package/src/components/AgentCard.test.tsx +134 -0
- package/src/components/AgentCard.tsx +585 -0
- package/src/components/AgentList.test.tsx +147 -0
- package/src/components/AgentList.tsx +419 -0
- package/src/components/AgentLogPreview.tsx +173 -0
- package/src/components/AgentProfilePanel.tsx +569 -0
- package/src/components/App.tsx +3424 -0
- package/src/components/BillingPanel.tsx +922 -0
- package/src/components/BillingResult.tsx +447 -0
- package/src/components/BroadcastComposer.tsx +690 -0
- package/src/components/ChannelAdminPanel.tsx +773 -0
- package/src/components/ChannelBrowser.tsx +385 -0
- package/src/components/ChannelChat.tsx +261 -0
- package/src/components/ChannelSidebar.tsx +399 -0
- package/src/components/CloudSessionProvider.tsx +130 -0
- package/src/components/CommandPalette.tsx +815 -0
- package/src/components/ConfirmationDialog.tsx +133 -0
- package/src/components/ConversationHistory.tsx +518 -0
- package/src/components/CoordinatorPanel.tsx +956 -0
- package/src/components/DecisionQueue.tsx +717 -0
- package/src/components/DirectMessageView.tsx +164 -0
- package/src/components/FileAutocomplete.tsx +368 -0
- package/src/components/FleetOverview.tsx +278 -0
- package/src/components/LogViewer.tsx +310 -0
- package/src/components/LogViewerPanel.tsx +482 -0
- package/src/components/Logo.tsx +284 -0
- package/src/components/MentionAutocomplete.tsx +384 -0
- package/src/components/MessageComposer.tsx +473 -0
- package/src/components/MessageList.tsx +725 -0
- package/src/components/MessageSenderName.tsx +91 -0
- package/src/components/MessageStatusIndicator.tsx +142 -0
- package/src/components/NewConversationModal.tsx +400 -0
- package/src/components/NotificationToast.tsx +488 -0
- package/src/components/OnlineUsersIndicator.tsx +164 -0
- package/src/components/Pagination.tsx +124 -0
- package/src/components/PricingPlans.tsx +386 -0
- package/src/components/ProjectList.tsx +711 -0
- package/src/components/ProviderAuthFlow.tsx +343 -0
- package/src/components/ProviderConnectionList.tsx +375 -0
- package/src/components/ProvisioningProgress.tsx +730 -0
- package/src/components/ReactionChips.tsx +70 -0
- package/src/components/ReactionPicker.tsx +121 -0
- package/src/components/RepoAccessPanel.tsx +787 -0
- package/src/components/RepositoriesPanel.tsx +901 -0
- package/src/components/ServerCard.tsx +202 -0
- package/src/components/SessionExpiredModal.tsx +128 -0
- package/src/components/SpawnModal.test.tsx +190 -0
- package/src/components/SpawnModal.tsx +1001 -0
- package/src/components/TaskAssignmentUI.tsx +375 -0
- package/src/components/TerminalProviderSetup.tsx +517 -0
- package/src/components/ThemeProvider.tsx +159 -0
- package/src/components/ThinkingIndicator.tsx +231 -0
- package/src/components/ThreadList.tsx +198 -0
- package/src/components/ThreadPanel.tsx +405 -0
- package/src/components/TrajectoryViewer.tsx +698 -0
- package/src/components/TypingIndicator.tsx +69 -0
- package/src/components/UsageBanner.tsx +231 -0
- package/src/components/UserProfilePanel.tsx +233 -0
- package/src/components/WorkspaceContext.tsx +95 -0
- package/src/components/WorkspaceSelector.tsx +234 -0
- package/src/components/WorkspaceStatusIndicator.tsx +396 -0
- package/src/components/XTermInteractive.tsx +516 -0
- package/src/components/XTermLogViewer.tsx +719 -0
- package/src/components/channels/ChannelDialogs.tsx +1411 -0
- package/src/components/channels/ChannelHeader.tsx +317 -0
- package/src/components/channels/ChannelMessageList.tsx +463 -0
- package/src/components/channels/ChannelViewV1.tsx +146 -0
- package/src/components/channels/MessageInput.tsx +302 -0
- package/src/components/channels/SearchInput.tsx +172 -0
- package/src/components/channels/SearchResults.tsx +336 -0
- package/src/components/channels/api.test.ts +1527 -0
- package/src/components/channels/api.ts +703 -0
- package/src/components/channels/index.ts +76 -0
- package/src/components/channels/mockApi.ts +344 -0
- package/src/components/channels/types.ts +566 -0
- package/src/components/hooks/index.ts +58 -0
- package/src/components/hooks/useAgentLogs.ts +504 -0
- package/src/components/hooks/useAgents.ts +127 -0
- package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
- package/src/components/hooks/useBroadcastDedup.ts +86 -0
- package/src/components/hooks/useChannelAdmin.ts +329 -0
- package/src/components/hooks/useChannelBrowser.ts +239 -0
- package/src/components/hooks/useChannelCommands.ts +138 -0
- package/src/components/hooks/useChannels.ts +367 -0
- package/src/components/hooks/useDebounce.ts +29 -0
- package/src/components/hooks/useDirectMessage.test.ts +952 -0
- package/src/components/hooks/useDirectMessage.ts +141 -0
- package/src/components/hooks/useMessages.ts +310 -0
- package/src/components/hooks/useOrchestrator.test.ts +165 -0
- package/src/components/hooks/useOrchestrator.ts +424 -0
- package/src/components/hooks/usePinnedAgents.test.ts +356 -0
- package/src/components/hooks/usePinnedAgents.ts +140 -0
- package/src/components/hooks/usePresence.test.ts +245 -0
- package/src/components/hooks/usePresence.ts +377 -0
- package/src/components/hooks/useRecentRepos.ts +130 -0
- package/src/components/hooks/useSession.ts +209 -0
- package/src/components/hooks/useThread.ts +138 -0
- package/src/components/hooks/useTrajectory.ts +265 -0
- package/src/components/hooks/useWebSocket.ts +290 -0
- package/src/components/hooks/useWorkspaceMembers.ts +132 -0
- package/src/components/hooks/useWorkspaceRepos.ts +73 -0
- package/src/components/hooks/useWorkspaceStatus.ts +237 -0
- package/src/components/index.ts +81 -0
- package/src/components/layout/Header.tsx +311 -0
- package/src/components/layout/RepoContextHeader.tsx +361 -0
- package/src/components/layout/Sidebar.archive.test.tsx +126 -0
- package/src/components/layout/Sidebar.test.tsx +691 -0
- package/src/components/layout/Sidebar.tsx +900 -0
- package/src/components/layout/index.ts +7 -0
- package/src/components/settings/BillingSettingsPanel.tsx +564 -0
- package/src/components/settings/SettingsPage.tsx +683 -0
- package/src/components/settings/TeamSettingsPanel.tsx +560 -0
- package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
- package/src/components/settings/index.ts +11 -0
- package/src/components/settings/types.ts +79 -0
- package/src/components/utils/messageFormatting.test.tsx +331 -0
- package/src/components/utils/messageFormatting.tsx +597 -0
- package/src/index.ts +63 -0
- package/src/landing/AboutPage.tsx +77 -0
- package/src/landing/BlogContent.tsx +187 -0
- package/src/landing/BlogPage.tsx +47 -0
- package/src/landing/CareersPage.tsx +53 -0
- package/src/landing/ChangelogPage.tsx +33 -0
- package/src/landing/ContactPage.tsx +41 -0
- package/src/landing/DocsPage.tsx +43 -0
- package/src/landing/LandingPage.tsx +702 -0
- package/src/landing/PricingPage.tsx +549 -0
- package/src/landing/PrivacyPage.tsx +117 -0
- package/src/landing/SecurityPage.tsx +42 -0
- package/src/landing/StaticPage.tsx +165 -0
- package/src/landing/TermsPage.tsx +125 -0
- package/src/landing/blogData.ts +312 -0
- package/src/landing/index.ts +18 -0
- package/src/landing/styles.css +3673 -0
- package/src/lib/agent-merge.test.ts +43 -0
- package/src/lib/agent-merge.ts +35 -0
- package/src/lib/api.ts +1294 -0
- package/src/lib/cloudApi.ts +893 -0
- package/src/lib/colors.test.ts +175 -0
- package/src/lib/colors.ts +218 -0
- package/src/lib/config.ts +109 -0
- package/src/lib/hierarchy.ts +242 -0
- package/src/lib/stuckDetection.ts +142 -0
- package/src/lib/useUrlRouting.ts +190 -0
- package/src/types/index.ts +317 -0
- package/src/types/threading.ts +7 -0
- package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
- package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
- /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
- /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
- /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
- /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
|
+
}
|