@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.
- 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/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
- /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → 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,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
|
+
}
|