@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.
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/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
  242. /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → 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,410 @@
1
+ /**
2
+ * Connect Repos Page - GitHub App OAuth via Nango
3
+ *
4
+ * Key: Initialize Nango on page load, not on click.
5
+ * This avoids popup blockers by ensuring openConnectUI is synchronous.
6
+ */
7
+
8
+ 'use client';
9
+
10
+ import React, { useState, useEffect, useRef } from 'react';
11
+ import Nango from '@nangohq/frontend';
12
+ import { LogoIcon } from '../../components/Logo';
13
+
14
+ type ConnectState = 'checking' | 'ready' | 'connecting' | 'polling' | 'pending-approval' | 'success' | 'error';
15
+
16
+ interface ConnectedRepo {
17
+ id: string;
18
+ fullName: string;
19
+ isPrivate: boolean;
20
+ defaultBranch: string;
21
+ }
22
+
23
+ export default function ConnectReposPage() {
24
+ const [state, setState] = useState<ConnectState>('checking');
25
+ const [error, setError] = useState<string | null>(null);
26
+ const [repos, setRepos] = useState<ConnectedRepo[]>([]);
27
+ const [pendingMessage, setPendingMessage] = useState<string | null>(null);
28
+ const [statusMessage, setStatusMessage] = useState<string>('');
29
+
30
+ // Store Nango instance - initialized on mount
31
+ const nangoRef = useRef<InstanceType<typeof Nango> | null>(null);
32
+
33
+ // Check session and initialize Nango on mount
34
+ useEffect(() => {
35
+ let mounted = true;
36
+
37
+ const init = async () => {
38
+ try {
39
+ // Check if authenticated
40
+ const sessionRes = await fetch('/api/auth/session', { credentials: 'include' });
41
+ const session = await sessionRes.json();
42
+ if (!session.authenticated) {
43
+ window.location.href = '/login';
44
+ return;
45
+ }
46
+
47
+ if (!mounted) return;
48
+
49
+ // Get Nango session token for repo connection
50
+ const nangoRes = await fetch('/api/auth/nango/repo-session', {
51
+ credentials: 'include',
52
+ });
53
+ const nangoData = await nangoRes.json();
54
+
55
+ if (!mounted) return;
56
+
57
+ if (!nangoRes.ok || !nangoData.sessionToken) {
58
+ if (nangoData?.sessionExpired || nangoData?.code === 'SESSION_EXPIRED') {
59
+ window.location.href = '/login';
60
+ return;
61
+ }
62
+ setError('Failed to initialize. Please refresh the page.');
63
+ setState('error');
64
+ return;
65
+ }
66
+
67
+ // Create Nango instance NOW, not on click
68
+ nangoRef.current = new Nango({ connectSessionToken: nangoData.sessionToken });
69
+ setState('ready');
70
+ } catch {
71
+ if (mounted) {
72
+ window.location.href = '/login';
73
+ }
74
+ }
75
+ };
76
+
77
+ init();
78
+ return () => { mounted = false; };
79
+ }, []);
80
+
81
+ const checkRepoStatus = async (connectionId: string): Promise<{
82
+ ready: boolean;
83
+ pendingApproval?: boolean;
84
+ message?: string;
85
+ repos?: ConnectedRepo[];
86
+ }> => {
87
+ const response = await fetch(`/api/auth/nango/repo-status/${connectionId}`, {
88
+ credentials: 'include',
89
+ });
90
+ if (!response.ok) {
91
+ throw new Error('Status not ready');
92
+ }
93
+ return response.json();
94
+ };
95
+
96
+ const handleAuthSuccess = async (connectionId: string) => {
97
+ try {
98
+ setState('polling');
99
+ setStatusMessage('Syncing repositories...');
100
+
101
+ const pollStartTime = Date.now();
102
+ const maxPollTime = 5 * 60 * 1000;
103
+ const pollInterval = 2000;
104
+
105
+ const pollForRepos = async (): Promise<void> => {
106
+ const elapsed = Date.now() - pollStartTime;
107
+
108
+ if (elapsed > maxPollTime) {
109
+ throw new Error('Connection timed out. Please try again.');
110
+ }
111
+
112
+ try {
113
+ const result = await checkRepoStatus(connectionId);
114
+ if (result.pendingApproval) {
115
+ setState('pending-approval');
116
+ setPendingMessage(result.message || 'Waiting for organization admin approval');
117
+ return;
118
+ } else if (result.ready && result.repos) {
119
+ setRepos(result.repos);
120
+ setState('success');
121
+ return;
122
+ }
123
+
124
+ await new Promise(resolve => setTimeout(resolve, pollInterval));
125
+ return pollForRepos();
126
+ } catch {
127
+ await new Promise(resolve => setTimeout(resolve, pollInterval));
128
+ return pollForRepos();
129
+ }
130
+ };
131
+
132
+ await pollForRepos();
133
+ } catch (err) {
134
+ console.error('[AUTH] Error:', err);
135
+ setError(err instanceof Error ? err.message : 'Connection failed');
136
+ setState('error');
137
+ setStatusMessage('');
138
+ }
139
+ };
140
+
141
+ // Use nango.auth() instead of openConnectUI to avoid popup blocker issues
142
+ const handleConnect = async () => {
143
+ if (!nangoRef.current) {
144
+ setError('Not ready. Please refresh the page.');
145
+ return;
146
+ }
147
+
148
+ setState('connecting');
149
+ setError(null);
150
+ setStatusMessage('Connecting to GitHub...');
151
+
152
+ try {
153
+ // Use github-app-oauth for GitHub App installation
154
+ const result = await nangoRef.current.auth('github-app-oauth');
155
+ if (result && 'connectionId' in result) {
156
+ await handleAuthSuccess(result.connectionId);
157
+ } else {
158
+ throw new Error('No connection ID returned');
159
+ }
160
+ } catch (err: unknown) {
161
+ const error = err as Error & { type?: string };
162
+ console.error('GitHub App auth error:', error);
163
+
164
+ // Don't show error for user-cancelled auth
165
+ if (error.type === 'user_cancelled' || error.message?.includes('closed')) {
166
+ setStatusMessage('');
167
+ // Re-initialize for next attempt
168
+ fetch('/api/auth/nango/repo-session', { credentials: 'include' })
169
+ .then(res => res.json())
170
+ .then(data => {
171
+ if (data.sessionToken) {
172
+ nangoRef.current = new Nango({ connectSessionToken: data.sessionToken });
173
+ setState('ready');
174
+ }
175
+ });
176
+ return;
177
+ }
178
+
179
+ setError(error.message || 'Connection failed');
180
+ setState('error');
181
+ setStatusMessage('');
182
+ }
183
+ };
184
+
185
+ const handleRetry = async () => {
186
+ setError(null);
187
+ setRepos([]);
188
+ setPendingMessage(null);
189
+ setStatusMessage('');
190
+ setState('checking');
191
+
192
+ // Re-initialize Nango for the retry
193
+ try {
194
+ const nangoRes = await fetch('/api/auth/nango/repo-session', {
195
+ credentials: 'include',
196
+ });
197
+ const nangoData = await nangoRes.json();
198
+
199
+ if (!nangoRes.ok || !nangoData.sessionToken) {
200
+ if (nangoData?.sessionExpired || nangoData?.code === 'SESSION_EXPIRED') {
201
+ window.location.href = '/login';
202
+ return;
203
+ }
204
+ setError('Failed to initialize. Please refresh the page.');
205
+ setState('error');
206
+ return;
207
+ }
208
+
209
+ nangoRef.current = new Nango({ connectSessionToken: nangoData.sessionToken });
210
+ setState('ready');
211
+ } catch {
212
+ setError('Failed to initialize. Please refresh the page.');
213
+ setState('error');
214
+ }
215
+ };
216
+
217
+ const handleContinue = () => {
218
+ window.location.href = '/app';
219
+ };
220
+
221
+ if (state === 'checking') {
222
+ return (
223
+ <div className="min-h-screen bg-gradient-to-br from-[#0a0a0f] via-[#0d1117] to-[#0a0a0f] flex items-center justify-center">
224
+ <div className="text-center">
225
+ <svg className="w-8 h-8 text-accent-cyan animate-spin mx-auto" fill="none" viewBox="0 0 24 24">
226
+ <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
227
+ <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
228
+ </svg>
229
+ <p className="mt-4 text-text-muted">Loading...</p>
230
+ </div>
231
+ </div>
232
+ );
233
+ }
234
+
235
+ const isConnecting = state === 'connecting' || state === 'polling';
236
+ const isReady = state === 'ready';
237
+
238
+ return (
239
+ <div className="min-h-screen bg-gradient-to-br from-[#0a0a0f] via-[#0d1117] to-[#0a0a0f] flex flex-col items-center justify-center p-4">
240
+ {/* Background grid */}
241
+ <div className="fixed inset-0 opacity-10">
242
+ <div
243
+ className="absolute inset-0"
244
+ style={{
245
+ backgroundImage: `linear-gradient(rgba(0, 217, 255, 0.1) 1px, transparent 1px),
246
+ linear-gradient(90deg, rgba(0, 217, 255, 0.1) 1px, transparent 1px)`,
247
+ backgroundSize: '50px 50px',
248
+ }}
249
+ />
250
+ </div>
251
+
252
+ {/* Content */}
253
+ <div className="relative z-10 w-full max-w-lg">
254
+ {/* Logo */}
255
+ <div className="flex flex-col items-center mb-8">
256
+ <LogoIcon size={48} withGlow={true} />
257
+ <h1 className="mt-4 text-2xl font-bold text-white">Connect Repositories</h1>
258
+ <p className="mt-2 text-text-muted text-center">
259
+ Connect your GitHub repositories to enable agent access
260
+ </p>
261
+ </div>
262
+
263
+ {/* Connect Card */}
264
+ <div className="bg-bg-primary/80 backdrop-blur-sm border border-border-subtle rounded-2xl p-8 shadow-xl">
265
+ {state === 'success' ? (
266
+ <div>
267
+ <div className="w-16 h-16 mx-auto mb-4 bg-success/20 rounded-full flex items-center justify-center">
268
+ <svg className="w-8 h-8 text-success" fill="none" viewBox="0 0 24 24" stroke="currentColor">
269
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
270
+ </svg>
271
+ </div>
272
+ <h2 className="text-xl font-semibold text-white mb-4 text-center">Repositories Connected!</h2>
273
+
274
+ <div className="max-h-60 overflow-y-auto mb-6 space-y-2">
275
+ {repos.map((repo) => (
276
+ <div key={repo.id} className="flex items-center gap-3 p-3 bg-bg-tertiary rounded-lg">
277
+ <svg className="w-5 h-5 text-text-muted flex-shrink-0" fill="currentColor" viewBox="0 0 16 16">
278
+ <path d="M2 2.5A2.5 2.5 0 014.5 0h8.75a.75.75 0 01.75.75v12.5a.75.75 0 01-.75.75h-2.5a.75.75 0 110-1.5h1.75v-2h-8a1 1 0 00-.714 1.7.75.75 0 01-1.072 1.05A2.495 2.495 0 012 11.5v-9zm10.5-1V9h-8c-.356 0-.694.074-1 .208V2.5a1 1 0 011-1h8z" />
279
+ </svg>
280
+ <div className="flex-1 min-w-0">
281
+ <p className="text-white font-medium truncate">{repo.fullName}</p>
282
+ <p className="text-text-muted text-sm">{repo.isPrivate ? 'Private' : 'Public'}</p>
283
+ </div>
284
+ </div>
285
+ ))}
286
+ </div>
287
+
288
+ <button
289
+ onClick={handleContinue}
290
+ className="w-full py-3 px-4 bg-gradient-to-r from-accent-cyan to-[#00b8d9] text-bg-deep font-semibold rounded-xl hover:shadow-glow-cyan transition-all"
291
+ >
292
+ Continue to Dashboard
293
+ </button>
294
+ </div>
295
+ ) : state === 'pending-approval' ? (
296
+ <div className="text-center">
297
+ <div className="w-16 h-16 mx-auto mb-4 bg-warning/20 rounded-full flex items-center justify-center">
298
+ <svg className="w-8 h-8 text-warning" fill="none" viewBox="0 0 24 24" stroke="currentColor">
299
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M12 8v4l3 3m6-3a9 9 0 11-18 0 9 9 0 0118 0z" />
300
+ </svg>
301
+ </div>
302
+ <h2 className="text-xl font-semibold text-white mb-2">Approval Pending</h2>
303
+ <p className="text-text-muted mb-6">{pendingMessage}</p>
304
+ <p className="text-sm text-text-muted mb-6">
305
+ An organization admin needs to approve the GitHub App installation.
306
+ </p>
307
+ <div className="flex gap-3">
308
+ <button onClick={handleRetry} className="flex-1 py-3 px-4 bg-bg-tertiary border border-border-subtle rounded-xl text-white font-medium hover:bg-bg-hover transition-colors">
309
+ Try Again
310
+ </button>
311
+ <button onClick={handleContinue} className="flex-1 py-3 px-4 bg-bg-tertiary border border-border-subtle rounded-xl text-white font-medium hover:bg-bg-hover transition-colors">
312
+ Skip for Now
313
+ </button>
314
+ </div>
315
+ </div>
316
+ ) : state === 'error' ? (
317
+ <div className="text-center">
318
+ <div className="w-16 h-16 mx-auto mb-4 bg-error/20 rounded-full flex items-center justify-center">
319
+ <svg className="w-8 h-8 text-error" fill="none" viewBox="0 0 24 24" stroke="currentColor">
320
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M6 18L18 6M6 6l12 12" />
321
+ </svg>
322
+ </div>
323
+ <h2 className="text-xl font-semibold text-white mb-2">Connection Failed</h2>
324
+ <p className="text-text-muted mb-6">{error}</p>
325
+ <button onClick={handleRetry} className="w-full py-3 px-4 bg-bg-tertiary border border-border-subtle rounded-xl text-white font-medium hover:bg-bg-hover transition-colors">
326
+ Try Again
327
+ </button>
328
+ </div>
329
+ ) : state === 'polling' ? (
330
+ <div className="text-center">
331
+ <div className="w-16 h-16 mx-auto mb-4 flex items-center justify-center">
332
+ <svg className="w-8 h-8 text-accent-cyan animate-spin" fill="none" viewBox="0 0 24 24">
333
+ <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
334
+ <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
335
+ </svg>
336
+ </div>
337
+ <h2 className="text-xl font-semibold text-white mb-2">Syncing Repositories</h2>
338
+ <p className="text-text-muted">{statusMessage || 'Fetching your repositories...'}</p>
339
+ </div>
340
+ ) : (
341
+ <div>
342
+ {error && (
343
+ <div className="mb-4 p-3 bg-error/10 border border-error/20 rounded-lg">
344
+ <p className="text-error text-sm">{error}</p>
345
+ </div>
346
+ )}
347
+
348
+ <div className="mb-6 p-4 bg-bg-tertiary rounded-lg border border-border-subtle">
349
+ <h3 className="font-medium text-white mb-2">What this enables:</h3>
350
+ <ul className="space-y-2 text-sm text-text-muted">
351
+ <li className="flex items-start gap-2">
352
+ <svg className="w-4 h-4 text-success mt-0.5 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor">
353
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
354
+ </svg>
355
+ <span>Agents can clone and work on your repositories</span>
356
+ </li>
357
+ <li className="flex items-start gap-2">
358
+ <svg className="w-4 h-4 text-success mt-0.5 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor">
359
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
360
+ </svg>
361
+ <span>Create issues, PRs, and comments on your behalf</span>
362
+ </li>
363
+ <li className="flex items-start gap-2">
364
+ <svg className="w-4 h-4 text-success mt-0.5 flex-shrink-0" fill="none" viewBox="0 0 24 24" stroke="currentColor">
365
+ <path strokeLinecap="round" strokeLinejoin="round" strokeWidth={2} d="M5 13l4 4L19 7" />
366
+ </svg>
367
+ <span>Secure token management - we never store your password</span>
368
+ </li>
369
+ </ul>
370
+ </div>
371
+
372
+ <button
373
+ onClick={handleConnect}
374
+ disabled={!isReady || isConnecting}
375
+ className="w-full py-4 px-6 bg-[#24292e] hover:bg-[#2f363d] border border-[#444d56] rounded-xl text-white font-medium flex items-center justify-center gap-3 transition-colors disabled:opacity-50 disabled:cursor-not-allowed"
376
+ >
377
+ {isConnecting ? (
378
+ <>
379
+ <svg className="w-5 h-5 animate-spin" fill="none" viewBox="0 0 24 24">
380
+ <circle className="opacity-25" cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="4" />
381
+ <path className="opacity-75" fill="currentColor" d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4z" />
382
+ </svg>
383
+ <span>{statusMessage || 'Connecting...'}</span>
384
+ </>
385
+ ) : (
386
+ <>
387
+ <svg className="w-6 h-6" fill="currentColor" viewBox="0 0 24 24">
388
+ <path d="M12 0C5.37 0 0 5.37 0 12c0 5.31 3.435 9.795 8.205 11.385.6.105.825-.255.825-.57 0-.285-.015-1.23-.015-2.235-3.015.555-3.795-.735-4.035-1.41-.135-.345-.72-1.41-1.23-1.695-.42-.225-1.02-.78-.015-.795.945-.015 1.62.87 1.845 1.23 1.08 1.815 2.805 1.305 3.495.99.105-.78.42-1.305.765-1.605-2.67-.3-5.46-1.335-5.46-5.925 0-1.305.465-2.385 1.23-3.225-.12-.3-.54-1.53.12-3.18 0 0 1.005-.315 3.3 1.23.96-.27 1.98-.405 3-.405s2.04.135 3 .405c2.295-1.56 3.3-1.23 3.3-1.23.66 1.65.24 2.88.12 3.18.765.84 1.23 1.905 1.23 3.225 0 4.605-2.805 5.625-5.475 5.925.435.375.81 1.095.81 2.22 0 1.605-.015 2.895-.015 3.3 0 .315.225.69.825.57A12.02 12.02 0 0024 12c0-6.63-5.37-12-12-12z" />
389
+ </svg>
390
+ <span>Connect GitHub Repositories</span>
391
+ </>
392
+ )}
393
+ </button>
394
+
395
+ <button onClick={handleContinue} className="w-full mt-3 py-3 px-4 text-text-muted hover:text-white transition-colors text-sm">
396
+ Skip for now
397
+ </button>
398
+ </div>
399
+ )}
400
+ </div>
401
+
402
+ <div className="mt-6 text-center">
403
+ <a href="/app" className="text-text-muted hover:text-white transition-colors">
404
+ Back to dashboard
405
+ </a>
406
+ </div>
407
+ </div>
408
+ </div>
409
+ );
410
+ }
@@ -0,0 +1,7 @@
1
+ 'use client';
2
+
3
+ import { ContactPage } from '../../landing';
4
+
5
+ export default function ContactRoute() {
6
+ return <ContactPage />;
7
+ }
@@ -0,0 +1,7 @@
1
+ 'use client';
2
+
3
+ import { DocsPage } from '../../landing';
4
+
5
+ export default function DocsRoute() {
6
+ return <DocsPage />;
7
+ }
Binary file
@@ -0,0 +1,200 @@
1
+ /**
2
+ * Dashboard V2 - Global Styles with Tailwind CSS v4
3
+ * Mission Control Theme
4
+ *
5
+ * Theme colors are defined as CSS variables and referenced by Tailwind.
6
+ * This enables automatic theme switching without !important overrides.
7
+ */
8
+
9
+ @import url('https://fonts.googleapis.com/css2?family=Outfit:wght@400;500;600;700;800&family=IBM+Plex+Mono:wght@400;500;600&family=Inter:wght@400;500;600&display=swap');
10
+ @import "tailwindcss";
11
+ @import "@xterm/xterm/css/xterm.css";
12
+ @config "../../tailwind.config.cjs";
13
+
14
+ /* ===== Theme System - CSS Variables ===== */
15
+
16
+ /**
17
+ * Dark Theme (Default) - Mission Control Deep Space
18
+ * These values are referenced by Tailwind via var(--color-*)
19
+ */
20
+ :root,
21
+ .theme-dark {
22
+ color-scheme: dark;
23
+
24
+ /* Background Colors */
25
+ --color-bg-deep: #0a0a0f;
26
+ --color-bg-primary: #0d0d14;
27
+ --color-bg-secondary: #12121c;
28
+ --color-bg-tertiary: #181824;
29
+ --color-bg-card: #1a1a28;
30
+ --color-bg-elevated: #202030;
31
+ --color-bg-hover: rgba(255, 255, 255, 0.04);
32
+ --color-bg-active: rgba(255, 255, 255, 0.08);
33
+
34
+ /* Text Colors */
35
+ --color-text-primary: #f0f0f5;
36
+ --color-text-secondary: #a0a0b0;
37
+ --color-text-muted: #606070;
38
+ --color-text-dim: #404050;
39
+ --color-text-inverse: #0a0a0f;
40
+
41
+ /* Border Colors */
42
+ --color-border-default: rgba(255, 255, 255, 0.1);
43
+ --color-border-subtle: rgba(255, 255, 255, 0.06);
44
+ --color-border-light: rgba(255, 255, 255, 0.1);
45
+ --color-border-medium: rgba(255, 255, 255, 0.15);
46
+
47
+ /* Accent Colors - Neon */
48
+ --color-accent-cyan: #00d9ff;
49
+ --color-accent-orange: #ff6b35;
50
+ --color-accent-purple: #a855f7;
51
+ --color-accent-green: #00ffc8;
52
+ --color-accent-hover: #00b8d9;
53
+ --color-accent-light: rgba(0, 217, 255, 0.15);
54
+
55
+ /* Status Colors */
56
+ --color-success: #00ffc8;
57
+ --color-success-light: rgba(0, 255, 200, 0.15);
58
+ --color-warning: #ff6b35;
59
+ --color-warning-light: rgba(255, 107, 53, 0.15);
60
+ --color-error: #ff4757;
61
+ --color-error-light: rgba(255, 71, 87, 0.15);
62
+
63
+ /* Sidebar Colors */
64
+ --color-sidebar-bg: #0d0d14;
65
+ --color-sidebar-border: rgba(255, 255, 255, 0.08);
66
+ --color-sidebar-hover: rgba(255, 255, 255, 0.06);
67
+
68
+ /* Scrollbar */
69
+ --scrollbar-thumb: rgba(255, 255, 255, 0.1);
70
+ --scrollbar-thumb-hover: rgba(255, 255, 255, 0.2);
71
+
72
+ /* Selection */
73
+ --selection-bg: rgba(0, 217, 255, 0.15);
74
+ --selection-color: #00d9ff;
75
+
76
+ /* Focus */
77
+ --focus-color: #00d9ff;
78
+ }
79
+
80
+ /**
81
+ * Light Theme
82
+ * Overrides the dark theme CSS variables
83
+ */
84
+ .theme-light {
85
+ color-scheme: light;
86
+
87
+ /* Background Colors */
88
+ --color-bg-deep: #f8f9fa;
89
+ --color-bg-primary: #f0f2f5;
90
+ --color-bg-secondary: #ffffff;
91
+ --color-bg-tertiary: #f5f6f8;
92
+ --color-bg-card: #ffffff;
93
+ --color-bg-elevated: #ffffff;
94
+ --color-bg-hover: rgba(0, 0, 0, 0.04);
95
+ --color-bg-active: rgba(0, 0, 0, 0.08);
96
+
97
+ /* Text Colors */
98
+ --color-text-primary: #1a1a2e;
99
+ --color-text-secondary: #4a4a5a;
100
+ --color-text-muted: #6a6a7a;
101
+ --color-text-dim: #8a8a9a;
102
+ --color-text-inverse: #ffffff;
103
+
104
+ /* Border Colors */
105
+ --color-border-default: rgba(0, 0, 0, 0.1);
106
+ --color-border-subtle: rgba(0, 0, 0, 0.06);
107
+ --color-border-light: rgba(0, 0, 0, 0.1);
108
+ --color-border-medium: rgba(0, 0, 0, 0.15);
109
+
110
+ /* Accent Colors - Adjusted for light backgrounds */
111
+ --color-accent-cyan: #0099cc;
112
+ --color-accent-orange: #e55a2b;
113
+ --color-accent-purple: #8b44d9;
114
+ --color-accent-green: #00b894;
115
+ --color-accent-hover: #007aa3;
116
+ --color-accent-light: rgba(0, 153, 204, 0.12);
117
+
118
+ /* Status Colors */
119
+ --color-success: #00b894;
120
+ --color-success-light: rgba(0, 184, 148, 0.12);
121
+ --color-warning: #e55a2b;
122
+ --color-warning-light: rgba(229, 90, 43, 0.12);
123
+ --color-error: #e53935;
124
+ --color-error-light: rgba(229, 57, 53, 0.12);
125
+
126
+ /* Sidebar Colors */
127
+ --color-sidebar-bg: #f0f2f5;
128
+ --color-sidebar-border: rgba(0, 0, 0, 0.08);
129
+ --color-sidebar-hover: rgba(0, 0, 0, 0.06);
130
+
131
+ /* Scrollbar */
132
+ --scrollbar-thumb: rgba(0, 0, 0, 0.15);
133
+ --scrollbar-thumb-hover: rgba(0, 0, 0, 0.25);
134
+
135
+ /* Selection */
136
+ --selection-bg: rgba(0, 153, 204, 0.2);
137
+ --selection-color: #0099cc;
138
+
139
+ /* Focus */
140
+ --focus-color: #0099cc;
141
+ }
142
+
143
+ /* ===== Base Styles ===== */
144
+
145
+ html,
146
+ body {
147
+ height: 100%;
148
+ -webkit-font-smoothing: antialiased;
149
+ -moz-osx-font-smoothing: grayscale;
150
+ }
151
+
152
+ /* Focus styles */
153
+ :focus-visible {
154
+ outline: 2px solid var(--focus-color);
155
+ outline-offset: 2px;
156
+ }
157
+
158
+ /* Scrollbar styles */
159
+ ::-webkit-scrollbar {
160
+ width: 6px;
161
+ height: 6px;
162
+ }
163
+
164
+ ::-webkit-scrollbar-track {
165
+ background: transparent;
166
+ }
167
+
168
+ ::-webkit-scrollbar-thumb {
169
+ background: var(--scrollbar-thumb);
170
+ border-radius: 3px;
171
+ }
172
+
173
+ ::-webkit-scrollbar-thumb:hover {
174
+ background: var(--scrollbar-thumb-hover);
175
+ }
176
+
177
+ /* Selection */
178
+ ::selection {
179
+ background: var(--selection-bg);
180
+ color: var(--selection-color);
181
+ }
182
+
183
+ /* Reduced motion preference */
184
+ @media (prefers-reduced-motion: reduce) {
185
+ *,
186
+ *::before,
187
+ *::after {
188
+ animation-duration: 0.01ms !important;
189
+ animation-iteration-count: 1 !important;
190
+ transition-duration: 0.01ms !important;
191
+ }
192
+ }
193
+
194
+ /* Smooth theme transition */
195
+ html.theme-transitioning,
196
+ html.theme-transitioning *,
197
+ html.theme-transitioning *::before,
198
+ html.theme-transitioning *::after {
199
+ transition: background-color 0.2s ease, border-color 0.2s ease, color 0.1s ease !important;
200
+ }