@agent-relay/dashboard 2.0.81 → 2.0.82

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (244) hide show
  1. package/out/404.html +1 -1
  2. package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
  3. package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
  4. package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
  5. package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
  6. package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
  7. package/out/about.html +2 -2
  8. package/out/about.txt +1 -1
  9. package/out/app/onboarding.html +1 -1
  10. package/out/app/onboarding.txt +1 -1
  11. package/out/app.html +1 -1
  12. package/out/app.txt +2 -2
  13. package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
  14. package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
  15. package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
  16. package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
  17. package/out/blog.html +2 -2
  18. package/out/blog.txt +1 -1
  19. package/out/careers.html +2 -2
  20. package/out/careers.txt +1 -1
  21. package/out/changelog.html +2 -2
  22. package/out/changelog.txt +1 -1
  23. package/out/cloud/link.html +1 -1
  24. package/out/cloud/link.txt +2 -2
  25. package/out/complete-profile.html +2 -2
  26. package/out/complete-profile.txt +1 -1
  27. package/out/connect-repos.html +1 -1
  28. package/out/connect-repos.txt +1 -1
  29. package/out/contact.html +2 -2
  30. package/out/contact.txt +1 -1
  31. package/out/docs.html +2 -2
  32. package/out/docs.txt +1 -1
  33. package/out/history.html +1 -1
  34. package/out/history.txt +2 -2
  35. package/out/index.html +1 -1
  36. package/out/index.txt +2 -2
  37. package/out/login.html +2 -2
  38. package/out/login.txt +1 -1
  39. package/out/metrics.html +1 -1
  40. package/out/metrics.txt +2 -2
  41. package/out/pricing.html +2 -2
  42. package/out/pricing.txt +1 -1
  43. package/out/privacy.html +2 -2
  44. package/out/privacy.txt +1 -1
  45. package/out/providers/setup/claude.html +1 -1
  46. package/out/providers/setup/claude.txt +1 -1
  47. package/out/providers/setup/codex.html +1 -1
  48. package/out/providers/setup/codex.txt +1 -1
  49. package/out/providers/setup/cursor.html +1 -1
  50. package/out/providers/setup/cursor.txt +1 -1
  51. package/out/providers.html +1 -1
  52. package/out/providers.txt +1 -1
  53. package/out/security.html +2 -2
  54. package/out/security.txt +1 -1
  55. package/out/signup.html +2 -2
  56. package/out/signup.txt +1 -1
  57. package/out/terms.html +2 -2
  58. package/out/terms.txt +1 -1
  59. package/package.json +7 -1
  60. package/src/app/about/page.tsx +7 -0
  61. package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
  62. package/src/app/app/[[...slug]]/page.tsx +23 -0
  63. package/src/app/app/onboarding/page.tsx +394 -0
  64. package/src/app/apple-icon.png +0 -0
  65. package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
  66. package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
  67. package/src/app/blog/page.tsx +15 -0
  68. package/src/app/careers/page.tsx +7 -0
  69. package/src/app/changelog/page.tsx +7 -0
  70. package/src/app/cloud/link/page.tsx +464 -0
  71. package/src/app/complete-profile/page.tsx +204 -0
  72. package/src/app/connect-repos/page.tsx +410 -0
  73. package/src/app/contact/page.tsx +7 -0
  74. package/src/app/docs/page.tsx +7 -0
  75. package/src/app/favicon.png +0 -0
  76. package/src/app/globals.css +200 -0
  77. package/src/app/history/page.tsx +658 -0
  78. package/src/app/layout.tsx +25 -0
  79. package/src/app/login/page.tsx +424 -0
  80. package/src/app/metrics/page.tsx +781 -0
  81. package/src/app/page.tsx +59 -0
  82. package/src/app/pricing/page.tsx +7 -0
  83. package/src/app/privacy/page.tsx +7 -0
  84. package/src/app/providers/page.tsx +193 -0
  85. package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
  86. package/src/app/providers/setup/[provider]/constants.ts +35 -0
  87. package/src/app/providers/setup/[provider]/page.tsx +42 -0
  88. package/src/app/security/page.tsx +7 -0
  89. package/src/app/signup/page.tsx +533 -0
  90. package/src/app/terms/page.tsx +7 -0
  91. package/src/components/ActivityFeed.tsx +216 -0
  92. package/src/components/AddWorkspaceModal.tsx +170 -0
  93. package/src/components/AgentCard.test.tsx +134 -0
  94. package/src/components/AgentCard.tsx +585 -0
  95. package/src/components/AgentList.test.tsx +147 -0
  96. package/src/components/AgentList.tsx +419 -0
  97. package/src/components/AgentLogPreview.tsx +173 -0
  98. package/src/components/AgentProfilePanel.tsx +569 -0
  99. package/src/components/App.tsx +3424 -0
  100. package/src/components/BillingPanel.tsx +922 -0
  101. package/src/components/BillingResult.tsx +447 -0
  102. package/src/components/BroadcastComposer.tsx +690 -0
  103. package/src/components/ChannelAdminPanel.tsx +773 -0
  104. package/src/components/ChannelBrowser.tsx +385 -0
  105. package/src/components/ChannelChat.tsx +261 -0
  106. package/src/components/ChannelSidebar.tsx +399 -0
  107. package/src/components/CloudSessionProvider.tsx +130 -0
  108. package/src/components/CommandPalette.tsx +815 -0
  109. package/src/components/ConfirmationDialog.tsx +133 -0
  110. package/src/components/ConversationHistory.tsx +518 -0
  111. package/src/components/CoordinatorPanel.tsx +956 -0
  112. package/src/components/DecisionQueue.tsx +717 -0
  113. package/src/components/DirectMessageView.tsx +164 -0
  114. package/src/components/FileAutocomplete.tsx +368 -0
  115. package/src/components/FleetOverview.tsx +278 -0
  116. package/src/components/LogViewer.tsx +310 -0
  117. package/src/components/LogViewerPanel.tsx +482 -0
  118. package/src/components/Logo.tsx +284 -0
  119. package/src/components/MentionAutocomplete.tsx +384 -0
  120. package/src/components/MessageComposer.tsx +473 -0
  121. package/src/components/MessageList.tsx +725 -0
  122. package/src/components/MessageSenderName.tsx +91 -0
  123. package/src/components/MessageStatusIndicator.tsx +142 -0
  124. package/src/components/NewConversationModal.tsx +400 -0
  125. package/src/components/NotificationToast.tsx +488 -0
  126. package/src/components/OnlineUsersIndicator.tsx +164 -0
  127. package/src/components/Pagination.tsx +124 -0
  128. package/src/components/PricingPlans.tsx +386 -0
  129. package/src/components/ProjectList.tsx +711 -0
  130. package/src/components/ProviderAuthFlow.tsx +343 -0
  131. package/src/components/ProviderConnectionList.tsx +375 -0
  132. package/src/components/ProvisioningProgress.tsx +730 -0
  133. package/src/components/ReactionChips.tsx +70 -0
  134. package/src/components/ReactionPicker.tsx +121 -0
  135. package/src/components/RepoAccessPanel.tsx +787 -0
  136. package/src/components/RepositoriesPanel.tsx +901 -0
  137. package/src/components/ServerCard.tsx +202 -0
  138. package/src/components/SessionExpiredModal.tsx +128 -0
  139. package/src/components/SpawnModal.test.tsx +190 -0
  140. package/src/components/SpawnModal.tsx +1001 -0
  141. package/src/components/TaskAssignmentUI.tsx +375 -0
  142. package/src/components/TerminalProviderSetup.tsx +517 -0
  143. package/src/components/ThemeProvider.tsx +159 -0
  144. package/src/components/ThinkingIndicator.tsx +231 -0
  145. package/src/components/ThreadList.tsx +198 -0
  146. package/src/components/ThreadPanel.tsx +405 -0
  147. package/src/components/TrajectoryViewer.tsx +698 -0
  148. package/src/components/TypingIndicator.tsx +69 -0
  149. package/src/components/UsageBanner.tsx +231 -0
  150. package/src/components/UserProfilePanel.tsx +233 -0
  151. package/src/components/WorkspaceContext.tsx +95 -0
  152. package/src/components/WorkspaceSelector.tsx +234 -0
  153. package/src/components/WorkspaceStatusIndicator.tsx +396 -0
  154. package/src/components/XTermInteractive.tsx +516 -0
  155. package/src/components/XTermLogViewer.tsx +719 -0
  156. package/src/components/channels/ChannelDialogs.tsx +1411 -0
  157. package/src/components/channels/ChannelHeader.tsx +317 -0
  158. package/src/components/channels/ChannelMessageList.tsx +463 -0
  159. package/src/components/channels/ChannelViewV1.tsx +146 -0
  160. package/src/components/channels/MessageInput.tsx +302 -0
  161. package/src/components/channels/SearchInput.tsx +172 -0
  162. package/src/components/channels/SearchResults.tsx +336 -0
  163. package/src/components/channels/api.test.ts +1527 -0
  164. package/src/components/channels/api.ts +703 -0
  165. package/src/components/channels/index.ts +76 -0
  166. package/src/components/channels/mockApi.ts +344 -0
  167. package/src/components/channels/types.ts +566 -0
  168. package/src/components/hooks/index.ts +58 -0
  169. package/src/components/hooks/useAgentLogs.ts +504 -0
  170. package/src/components/hooks/useAgents.ts +127 -0
  171. package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
  172. package/src/components/hooks/useBroadcastDedup.ts +86 -0
  173. package/src/components/hooks/useChannelAdmin.ts +329 -0
  174. package/src/components/hooks/useChannelBrowser.ts +239 -0
  175. package/src/components/hooks/useChannelCommands.ts +138 -0
  176. package/src/components/hooks/useChannels.ts +367 -0
  177. package/src/components/hooks/useDebounce.ts +29 -0
  178. package/src/components/hooks/useDirectMessage.test.ts +952 -0
  179. package/src/components/hooks/useDirectMessage.ts +141 -0
  180. package/src/components/hooks/useMessages.ts +310 -0
  181. package/src/components/hooks/useOrchestrator.test.ts +165 -0
  182. package/src/components/hooks/useOrchestrator.ts +424 -0
  183. package/src/components/hooks/usePinnedAgents.test.ts +356 -0
  184. package/src/components/hooks/usePinnedAgents.ts +140 -0
  185. package/src/components/hooks/usePresence.test.ts +245 -0
  186. package/src/components/hooks/usePresence.ts +377 -0
  187. package/src/components/hooks/useRecentRepos.ts +130 -0
  188. package/src/components/hooks/useSession.ts +209 -0
  189. package/src/components/hooks/useThread.ts +138 -0
  190. package/src/components/hooks/useTrajectory.ts +265 -0
  191. package/src/components/hooks/useWebSocket.ts +290 -0
  192. package/src/components/hooks/useWorkspaceMembers.ts +132 -0
  193. package/src/components/hooks/useWorkspaceRepos.ts +73 -0
  194. package/src/components/hooks/useWorkspaceStatus.ts +237 -0
  195. package/src/components/index.ts +81 -0
  196. package/src/components/layout/Header.tsx +311 -0
  197. package/src/components/layout/RepoContextHeader.tsx +361 -0
  198. package/src/components/layout/Sidebar.archive.test.tsx +126 -0
  199. package/src/components/layout/Sidebar.test.tsx +691 -0
  200. package/src/components/layout/Sidebar.tsx +900 -0
  201. package/src/components/layout/index.ts +7 -0
  202. package/src/components/settings/BillingSettingsPanel.tsx +564 -0
  203. package/src/components/settings/SettingsPage.tsx +683 -0
  204. package/src/components/settings/TeamSettingsPanel.tsx +560 -0
  205. package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
  206. package/src/components/settings/index.ts +11 -0
  207. package/src/components/settings/types.ts +79 -0
  208. package/src/components/utils/messageFormatting.test.tsx +331 -0
  209. package/src/components/utils/messageFormatting.tsx +597 -0
  210. package/src/index.ts +63 -0
  211. package/src/landing/AboutPage.tsx +77 -0
  212. package/src/landing/BlogContent.tsx +187 -0
  213. package/src/landing/BlogPage.tsx +47 -0
  214. package/src/landing/CareersPage.tsx +53 -0
  215. package/src/landing/ChangelogPage.tsx +33 -0
  216. package/src/landing/ContactPage.tsx +41 -0
  217. package/src/landing/DocsPage.tsx +43 -0
  218. package/src/landing/LandingPage.tsx +702 -0
  219. package/src/landing/PricingPage.tsx +549 -0
  220. package/src/landing/PrivacyPage.tsx +117 -0
  221. package/src/landing/SecurityPage.tsx +42 -0
  222. package/src/landing/StaticPage.tsx +165 -0
  223. package/src/landing/TermsPage.tsx +125 -0
  224. package/src/landing/blogData.ts +312 -0
  225. package/src/landing/index.ts +18 -0
  226. package/src/landing/styles.css +3673 -0
  227. package/src/lib/agent-merge.test.ts +43 -0
  228. package/src/lib/agent-merge.ts +35 -0
  229. package/src/lib/api.ts +1294 -0
  230. package/src/lib/cloudApi.ts +893 -0
  231. package/src/lib/colors.test.ts +175 -0
  232. package/src/lib/colors.ts +218 -0
  233. package/src/lib/config.ts +109 -0
  234. package/src/lib/hierarchy.ts +242 -0
  235. package/src/lib/stuckDetection.ts +142 -0
  236. package/src/lib/useUrlRouting.ts +190 -0
  237. package/src/types/index.ts +317 -0
  238. package/src/types/threading.ts +7 -0
  239. package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
  240. package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
  241. /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
  242. /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
  243. /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
  244. /package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/{page-a58308f43557b908.js → page-b194f207fbd91862.js} +0 -0
@@ -0,0 +1,69 @@
1
+ /**
2
+ * TypingIndicator Component
3
+ *
4
+ * Shows animated typing indicator when other users are typing.
5
+ * Displays user avatars and "X is typing..." text.
6
+ */
7
+
8
+ import React from 'react';
9
+ import type { TypingIndicator as TypingIndicatorType } from './hooks/usePresence';
10
+
11
+ export interface TypingIndicatorProps {
12
+ /** List of users currently typing */
13
+ typingUsers: TypingIndicatorType[];
14
+ }
15
+
16
+ export function TypingIndicator({ typingUsers }: TypingIndicatorProps) {
17
+ if (typingUsers.length === 0) {
18
+ return null;
19
+ }
20
+
21
+ // Format the typing text
22
+ const formatTypingText = () => {
23
+ if (typingUsers.length === 1) {
24
+ return `${typingUsers[0].username} is typing`;
25
+ } else if (typingUsers.length === 2) {
26
+ return `${typingUsers[0].username} and ${typingUsers[1].username} are typing`;
27
+ } else {
28
+ return `${typingUsers[0].username} and ${typingUsers.length - 1} others are typing`;
29
+ }
30
+ };
31
+
32
+ return (
33
+ <div className="flex items-center gap-2 px-4 py-2 text-sm text-[#8d8d8e]">
34
+ {/* Avatars */}
35
+ <div className="flex -space-x-1.5">
36
+ {typingUsers.slice(0, 3).map((user) => (
37
+ <div
38
+ key={user.username}
39
+ className="relative"
40
+ title={user.username}
41
+ >
42
+ {user.avatarUrl ? (
43
+ <img
44
+ src={user.avatarUrl}
45
+ alt={user.username}
46
+ className="w-5 h-5 rounded-full border border-[#1a1d21]"
47
+ />
48
+ ) : (
49
+ <div className="w-5 h-5 rounded-full bg-[#a855f7] border border-[#1a1d21] flex items-center justify-center text-[9px] text-white font-medium">
50
+ {user.username.charAt(0).toUpperCase()}
51
+ </div>
52
+ )}
53
+ </div>
54
+ ))}
55
+ </div>
56
+
57
+ {/* Typing text */}
58
+ <span className="flex items-center gap-1">
59
+ {formatTypingText()}
60
+ {/* Animated dots */}
61
+ <span className="flex gap-0.5">
62
+ <span className="w-1 h-1 bg-[#8d8d8e] rounded-full animate-bounce" style={{ animationDelay: '0ms' }} />
63
+ <span className="w-1 h-1 bg-[#8d8d8e] rounded-full animate-bounce" style={{ animationDelay: '150ms' }} />
64
+ <span className="w-1 h-1 bg-[#8d8d8e] rounded-full animate-bounce" style={{ animationDelay: '300ms' }} />
65
+ </span>
66
+ </span>
67
+ </div>
68
+ );
69
+ }
@@ -0,0 +1,231 @@
1
+ /**
2
+ * Usage Banner Component
3
+ *
4
+ * Displays remaining compute hours for free tier users.
5
+ * Shows warning when approaching limit and upgrade CTA when exceeded.
6
+ */
7
+
8
+ import React, { useEffect, useState } from 'react';
9
+
10
+ interface UsageData {
11
+ plan: 'free' | 'pro' | 'team' | 'enterprise';
12
+ limits: {
13
+ computeHoursPerMonth: number;
14
+ };
15
+ usage: {
16
+ computeHoursThisMonth: number;
17
+ };
18
+ percentUsed: {
19
+ computeHours: number;
20
+ };
21
+ introBonus?: {
22
+ isActive: boolean;
23
+ daysRemaining: number;
24
+ totalDays: number;
25
+ expiresAt: string | null;
26
+ resources: {
27
+ cpus: number;
28
+ memoryGb: number;
29
+ description: string;
30
+ };
31
+ };
32
+ }
33
+
34
+ export interface UsageBannerProps {
35
+ /** API base URL (default: '') */
36
+ apiBaseUrl?: string;
37
+ /** Callback when upgrade is clicked */
38
+ onUpgradeClick?: () => void;
39
+ }
40
+
41
+ export function UsageBanner({ apiBaseUrl = '', onUpgradeClick }: UsageBannerProps) {
42
+ const [usage, setUsage] = useState<UsageData | null>(null);
43
+ const [loading, setLoading] = useState(true);
44
+ const [error, setError] = useState<string | null>(null);
45
+ const [dismissed, setDismissed] = useState(false);
46
+
47
+ useEffect(() => {
48
+ async function fetchUsage() {
49
+ try {
50
+ const response = await fetch(`${apiBaseUrl}/api/usage`, {
51
+ credentials: 'include',
52
+ });
53
+
54
+ if (!response.ok) {
55
+ if (response.status === 401) {
56
+ // Not logged in, don't show banner
57
+ setLoading(false);
58
+ return;
59
+ }
60
+ throw new Error('Failed to fetch usage');
61
+ }
62
+
63
+ const data = await response.json();
64
+ setUsage(data);
65
+ } catch (err) {
66
+ setError(err instanceof Error ? err.message : 'Unknown error');
67
+ } finally {
68
+ setLoading(false);
69
+ }
70
+ }
71
+
72
+ fetchUsage();
73
+
74
+ // Refresh every 5 minutes
75
+ const interval = setInterval(fetchUsage, 5 * 60 * 1000);
76
+ return () => clearInterval(interval);
77
+ }, [apiBaseUrl]);
78
+
79
+ // Don't show for non-free plans
80
+ if (loading || error || !usage || usage.plan !== 'free' || dismissed) {
81
+ return null;
82
+ }
83
+
84
+ const { computeHoursThisMonth } = usage.usage;
85
+ const { computeHoursPerMonth } = usage.limits;
86
+ const percentUsed = usage.percentUsed.computeHours;
87
+ const remaining = Math.max(0, computeHoursPerMonth - computeHoursThisMonth);
88
+ const isExceeded = remaining <= 0;
89
+ const isWarning = percentUsed >= 80 && !isExceeded;
90
+
91
+ // Intro bonus status
92
+ const introBonus = usage.introBonus;
93
+ const hasActiveIntro = introBonus?.isActive && introBonus.daysRemaining > 0;
94
+
95
+ // Get current month name
96
+ const monthName = new Date().toLocaleDateString('en-US', { month: 'long' });
97
+
98
+ // Determine banner style
99
+ let bgClass = 'bg-bg-tertiary border-border-subtle';
100
+ let textClass = 'text-text-secondary';
101
+ let iconColor = 'text-accent-cyan';
102
+
103
+ if (isExceeded) {
104
+ bgClass = 'bg-error/10 border-error/30';
105
+ textClass = 'text-error';
106
+ iconColor = 'text-error';
107
+ } else if (isWarning) {
108
+ bgClass = 'bg-warning/10 border-warning/30';
109
+ textClass = 'text-warning';
110
+ iconColor = 'text-warning';
111
+ } else if (hasActiveIntro) {
112
+ // Special styling for intro bonus - use brand cyan
113
+ bgClass = 'bg-accent-cyan/10 border-accent-cyan/30';
114
+ textClass = 'text-accent-cyan';
115
+ iconColor = 'text-accent-cyan';
116
+ }
117
+
118
+ return (
119
+ <div className={`flex items-center justify-between px-4 py-2 border-b ${bgClass}`}>
120
+ <div className="flex items-center gap-3">
121
+ {hasActiveIntro ? <RocketIcon className={iconColor} /> : <ClockIcon className={iconColor} />}
122
+ <span className={`text-sm ${textClass}`}>
123
+ {isExceeded ? (
124
+ <>
125
+ <strong>Compute limit reached</strong> — Your free tier compute hours for {monthName} have been used.
126
+ Workspaces are paused until next month.
127
+ </>
128
+ ) : isWarning ? (
129
+ <>
130
+ <strong>{remaining.toFixed(1)}h remaining</strong> — You&apos;ve used {percentUsed}% of your
131
+ free tier compute hours for {monthName}.
132
+ </>
133
+ ) : hasActiveIntro ? (
134
+ <>
135
+ <strong>Intro Bonus Active</strong> — {introBonus!.resources.cpus} CPU / {introBonus!.resources.memoryGb}GB RAM.{' '}
136
+ <span className="text-text-secondary">
137
+ {introBonus!.daysRemaining} day{introBonus!.daysRemaining !== 1 ? 's' : ''} remaining before auto-resize to 1 CPU / 2GB.
138
+ </span>
139
+ </>
140
+ ) : (
141
+ <>
142
+ <strong>{remaining.toFixed(1)} of {computeHoursPerMonth}h</strong> compute hours remaining
143
+ this month
144
+ </>
145
+ )}
146
+ </span>
147
+ </div>
148
+
149
+ <div className="flex items-center gap-2">
150
+ {(isExceeded || isWarning || hasActiveIntro) && (
151
+ <button
152
+ onClick={onUpgradeClick || (() => window.location.href = '/pricing')}
153
+ className="px-3 py-1.5 bg-gradient-to-r from-accent-cyan to-[#00b8d9] text-bg-deep font-semibold border-none rounded-md text-xs cursor-pointer transition-all duration-150 hover:shadow-glow-cyan hover:-translate-y-0.5"
154
+ >
155
+ {hasActiveIntro ? 'Keep Pro Resources' : 'Upgrade Plan'}
156
+ </button>
157
+ )}
158
+
159
+ {!isExceeded && (
160
+ <button
161
+ onClick={() => setDismissed(true)}
162
+ className="p-1 text-text-muted hover:text-text-primary transition-colors"
163
+ aria-label="Dismiss"
164
+ >
165
+ <CloseIcon />
166
+ </button>
167
+ )}
168
+ </div>
169
+ </div>
170
+ );
171
+ }
172
+
173
+ function ClockIcon({ className }: { className?: string }) {
174
+ return (
175
+ <svg
176
+ width="16"
177
+ height="16"
178
+ viewBox="0 0 24 24"
179
+ fill="none"
180
+ stroke="currentColor"
181
+ strokeWidth="2"
182
+ strokeLinecap="round"
183
+ strokeLinejoin="round"
184
+ className={className}
185
+ >
186
+ <circle cx="12" cy="12" r="10" />
187
+ <polyline points="12 6 12 12 16 14" />
188
+ </svg>
189
+ );
190
+ }
191
+
192
+ function CloseIcon() {
193
+ return (
194
+ <svg
195
+ width="14"
196
+ height="14"
197
+ viewBox="0 0 24 24"
198
+ fill="none"
199
+ stroke="currentColor"
200
+ strokeWidth="2"
201
+ strokeLinecap="round"
202
+ strokeLinejoin="round"
203
+ >
204
+ <line x1="18" y1="6" x2="6" y2="18" />
205
+ <line x1="6" y1="6" x2="18" y2="18" />
206
+ </svg>
207
+ );
208
+ }
209
+
210
+ function RocketIcon({ className }: { className?: string }) {
211
+ return (
212
+ <svg
213
+ width="16"
214
+ height="16"
215
+ viewBox="0 0 24 24"
216
+ fill="none"
217
+ stroke="currentColor"
218
+ strokeWidth="2"
219
+ strokeLinecap="round"
220
+ strokeLinejoin="round"
221
+ className={className}
222
+ >
223
+ <path d="M4.5 16.5c-1.5 1.26-2 5-2 5s3.74-.5 5-2c.71-.84.7-2.13-.09-2.91a2.18 2.18 0 0 0-2.91-.09z" />
224
+ <path d="m12 15-3-3a22 22 0 0 1 2-3.95A12.88 12.88 0 0 1 22 2c0 2.72-.78 7.5-6 11a22.35 22.35 0 0 1-4 2z" />
225
+ <path d="M9 12H4s.55-3.03 2-4c1.62-1.08 5 0 5 0" />
226
+ <path d="M12 15v5s3.03-.55 4-2c1.08-1.62 0-5 0-5" />
227
+ </svg>
228
+ );
229
+ }
230
+
231
+ export default UsageBanner;
@@ -0,0 +1,233 @@
1
+ /**
2
+ * UserProfilePanel Component
3
+ *
4
+ * Slide-out panel showing user profile details.
5
+ * Displays avatar, username, GitHub link, and action buttons.
6
+ */
7
+
8
+ import React, { useEffect, useRef } from 'react';
9
+ import type { UserPresence } from './hooks/usePresence';
10
+
11
+ export interface UserProfilePanelProps {
12
+ /** User to display (null to hide panel) */
13
+ user: UserPresence | null;
14
+ /** Callback when panel should close */
15
+ onClose: () => void;
16
+ /** Callback when mention button is clicked */
17
+ onMention?: (username: string) => void;
18
+ /** Callback when send message button is clicked */
19
+ onSendMessage?: (user: UserPresence) => void;
20
+ }
21
+
22
+ export function UserProfilePanel({ user, onClose, onMention, onSendMessage }: UserProfilePanelProps) {
23
+ const panelRef = useRef<HTMLDivElement>(null);
24
+
25
+ // Close on Escape key
26
+ useEffect(() => {
27
+ const handleKeyDown = (e: KeyboardEvent) => {
28
+ if (e.key === 'Escape') {
29
+ onClose();
30
+ }
31
+ };
32
+
33
+ if (user) {
34
+ window.addEventListener('keydown', handleKeyDown);
35
+ }
36
+ return () => window.removeEventListener('keydown', handleKeyDown);
37
+ }, [user, onClose]);
38
+
39
+ // Close on outside click
40
+ // Use a ref to track if the panel just opened to avoid closing on the same click
41
+ const justOpenedRef = useRef(false);
42
+
43
+ useEffect(() => {
44
+ if (user) {
45
+ // Mark as just opened
46
+ justOpenedRef.current = true;
47
+ }
48
+ }, [user]);
49
+
50
+ useEffect(() => {
51
+ const handleClickOutside = (e: MouseEvent) => {
52
+ // Skip if panel just opened (same event loop tick that opened it)
53
+ if (justOpenedRef.current) {
54
+ justOpenedRef.current = false;
55
+ return;
56
+ }
57
+
58
+ if (panelRef.current && !panelRef.current.contains(e.target as Node)) {
59
+ onClose();
60
+ }
61
+ };
62
+
63
+ if (user) {
64
+ document.addEventListener('mousedown', handleClickOutside);
65
+ return () => {
66
+ document.removeEventListener('mousedown', handleClickOutside);
67
+ };
68
+ }
69
+ }, [user, onClose]);
70
+
71
+ if (!user) {
72
+ return null;
73
+ }
74
+
75
+ const githubUrl = `https://github.com/${user.username}`;
76
+
77
+ return (
78
+ <>
79
+ {/* Backdrop */}
80
+ <div className="fixed inset-0 bg-black/50 z-40" />
81
+
82
+ {/* Panel */}
83
+ <div
84
+ ref={panelRef}
85
+ className="fixed right-0 top-0 h-full w-80 bg-[#1a1d21] border-l border-white/10 shadow-2xl z-50 flex flex-col animate-slide-in-right"
86
+ >
87
+ {/* Header */}
88
+ <div className="flex items-center justify-between p-4 border-b border-white/10">
89
+ <h2 className="text-lg font-semibold text-[#d1d2d3]">Profile</h2>
90
+ <button
91
+ onClick={onClose}
92
+ className="p-1 hover:bg-white/10 rounded-md transition-colors"
93
+ title="Close"
94
+ >
95
+ <svg width="20" height="20" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
96
+ <path d="M18 6L6 18M6 6l12 12" />
97
+ </svg>
98
+ </button>
99
+ </div>
100
+
101
+ {/* User Info */}
102
+ <div className="flex flex-col items-center p-6 border-b border-white/10">
103
+ {/* Large Avatar */}
104
+ <div className="relative mb-4">
105
+ {user.avatarUrl ? (
106
+ <img
107
+ src={user.avatarUrl}
108
+ alt={user.username}
109
+ className="w-24 h-24 rounded-full object-cover border-4 border-[#a855f7]/30"
110
+ />
111
+ ) : (
112
+ <div className="w-24 h-24 rounded-full bg-[#a855f7] flex items-center justify-center text-3xl text-white font-bold border-4 border-[#a855f7]/30">
113
+ {user.username.charAt(0).toUpperCase()}
114
+ </div>
115
+ )}
116
+ {/* Online indicator */}
117
+ <div className="absolute bottom-1 right-1 w-5 h-5 bg-green-500 rounded-full border-4 border-[#1a1d21]" />
118
+ </div>
119
+
120
+ {/* Username */}
121
+ <h3 className="text-xl font-semibold text-[#d1d2d3] mb-1">
122
+ {user.username}
123
+ </h3>
124
+
125
+ {/* Status */}
126
+ <span className="text-sm text-green-400 flex items-center gap-1.5">
127
+ <div className="w-2 h-2 bg-green-500 rounded-full" />
128
+ Online
129
+ </span>
130
+ </div>
131
+
132
+ {/* Details */}
133
+ <div className="flex-1 p-4 overflow-y-auto">
134
+ <div className="space-y-4">
135
+ {/* Online Since */}
136
+ <div>
137
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Online Since</label>
138
+ <p className="text-sm text-[#d1d2d3] mt-1">
139
+ {formatDateTime(user.connectedAt)}
140
+ </p>
141
+ </div>
142
+
143
+ {/* Last Active */}
144
+ <div>
145
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">Last Active</label>
146
+ <p className="text-sm text-[#d1d2d3] mt-1">
147
+ {formatDateTime(user.lastSeen)}
148
+ </p>
149
+ </div>
150
+
151
+ {/* GitHub Link */}
152
+ <div>
153
+ <label className="text-xs text-[#8d8d8e] uppercase tracking-wide">GitHub</label>
154
+ <a
155
+ href={githubUrl}
156
+ target="_blank"
157
+ rel="noopener noreferrer"
158
+ className="flex items-center gap-2 mt-1 text-sm text-[#a855f7] hover:text-[#c084fc] transition-colors"
159
+ >
160
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor">
161
+ <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" />
162
+ </svg>
163
+ @{user.username}
164
+ <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
165
+ <path d="M18 13v6a2 2 0 01-2 2H5a2 2 0 01-2-2V8a2 2 0 012-2h6M15 3h6v6M10 14L21 3" />
166
+ </svg>
167
+ </a>
168
+ </div>
169
+ </div>
170
+ </div>
171
+
172
+ {/* Actions */}
173
+ <div className="p-4 border-t border-white/10 space-y-2">
174
+ {/* Send Message Button */}
175
+ <button
176
+ onClick={() => {
177
+ onSendMessage?.(user);
178
+ onClose();
179
+ }}
180
+ className="w-full flex items-center justify-center gap-2 py-2.5 bg-[#00d4aa] hover:bg-[#00bfa0] text-[#0a0e14] font-medium rounded-lg transition-colors"
181
+ >
182
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
183
+ <path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" />
184
+ </svg>
185
+ Send Message
186
+ </button>
187
+
188
+ {/* Mention Button */}
189
+ <button
190
+ onClick={() => {
191
+ onMention?.(user.username);
192
+ onClose();
193
+ }}
194
+ className="w-full flex items-center justify-center gap-2 py-2.5 bg-[#a855f7] hover:bg-[#9333ea] text-white font-medium rounded-lg transition-colors"
195
+ >
196
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
197
+ <circle cx="12" cy="12" r="4" />
198
+ <path d="M16 8v5a3 3 0 006 0v-1a10 10 0 10-3.92 7.94" />
199
+ </svg>
200
+ Mention @{user.username}
201
+ </button>
202
+
203
+ {/* View on GitHub */}
204
+ <a
205
+ href={githubUrl}
206
+ target="_blank"
207
+ rel="noopener noreferrer"
208
+ className="w-full flex items-center justify-center gap-2 py-2.5 border border-white/20 text-[#d1d2d3] hover:bg-white/5 font-medium rounded-lg transition-colors"
209
+ >
210
+ <svg width="16" height="16" viewBox="0 0 24 24" fill="currentColor">
211
+ <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" />
212
+ </svg>
213
+ View on GitHub
214
+ </a>
215
+ </div>
216
+ </div>
217
+ </>
218
+ );
219
+ }
220
+
221
+ /**
222
+ * Format a timestamp to a readable date/time
223
+ */
224
+ function formatDateTime(timestamp: string): string {
225
+ const date = new Date(timestamp);
226
+ return date.toLocaleString([], {
227
+ month: 'short',
228
+ day: 'numeric',
229
+ year: 'numeric',
230
+ hour: '2-digit',
231
+ minute: '2-digit',
232
+ });
233
+ }
@@ -0,0 +1,95 @@
1
+ /**
2
+ * Workspace Context
3
+ *
4
+ * Provides the current workspace's base URL for WebSocket connections.
5
+ * Used by LogViewer and other components that need to connect to workspace-specific endpoints.
6
+ */
7
+
8
+ import React, { createContext, useContext, useMemo } from 'react';
9
+ import { getWebSocketUrl } from '../lib/config';
10
+
11
+ interface WorkspaceContextValue {
12
+ /** Base WebSocket URL for the workspace (e.g., wss://workspace-abc.agentrelay.dev) */
13
+ wsBaseUrl: string | null;
14
+ /** Whether we're in cloud mode (workspace URL is different from page host) */
15
+ isCloudMode: boolean;
16
+ }
17
+
18
+ const WorkspaceContext = createContext<WorkspaceContextValue>({
19
+ wsBaseUrl: null,
20
+ isCloudMode: false,
21
+ });
22
+
23
+ export interface WorkspaceProviderProps {
24
+ children: React.ReactNode;
25
+ /** The workspace WebSocket URL (e.g., wss://workspace-abc.agentrelay.dev/ws) */
26
+ wsUrl?: string;
27
+ }
28
+
29
+ /**
30
+ * Extract base URL from a WebSocket URL
31
+ * e.g., wss://workspace-abc.agentrelay.dev/ws -> wss://workspace-abc.agentrelay.dev
32
+ */
33
+ function getBaseUrl(wsUrl: string): string {
34
+ try {
35
+ const url = new URL(wsUrl);
36
+ return `${url.protocol}//${url.host}`;
37
+ } catch {
38
+ return wsUrl;
39
+ }
40
+ }
41
+
42
+ export function WorkspaceProvider({ children, wsUrl }: WorkspaceProviderProps) {
43
+ const value = useMemo(() => {
44
+ if (!wsUrl) {
45
+ return { wsBaseUrl: null, isCloudMode: false };
46
+ }
47
+
48
+ const wsBaseUrl = getBaseUrl(wsUrl);
49
+
50
+ // Check if we're in cloud mode by comparing the workspace URL host with the current page host
51
+ let isCloudMode = false;
52
+ if (typeof window !== 'undefined') {
53
+ try {
54
+ const wsHost = new URL(wsUrl).host;
55
+ isCloudMode = wsHost !== window.location.host;
56
+ } catch {
57
+ // Ignore parse errors
58
+ }
59
+ }
60
+
61
+ return { wsBaseUrl, isCloudMode };
62
+ }, [wsUrl]);
63
+
64
+ return (
65
+ <WorkspaceContext.Provider value={value}>
66
+ {children}
67
+ </WorkspaceContext.Provider>
68
+ );
69
+ }
70
+
71
+ /**
72
+ * Hook to access the workspace context
73
+ */
74
+ export function useWorkspace(): WorkspaceContextValue {
75
+ return useContext(WorkspaceContext);
76
+ }
77
+
78
+ /**
79
+ * Get the WebSocket URL for a specific path within the workspace
80
+ * Falls back to centralized config if not in a workspace context
81
+ */
82
+ export function useWorkspaceWsUrl(path: string): string {
83
+ const { wsBaseUrl } = useWorkspace();
84
+
85
+ return useMemo(() => {
86
+ if (wsBaseUrl) {
87
+ return `${wsBaseUrl}${path}`;
88
+ }
89
+
90
+ // Fallback to centralized config
91
+ return getWebSocketUrl(path);
92
+ }, [wsBaseUrl, path]);
93
+ }
94
+
95
+ export default WorkspaceContext;