@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,399 @@
1
+ /**
2
+ * ChannelSidebar Component
3
+ *
4
+ * Displays joined channels and allows channel management.
5
+ * Slack-like interface for channel navigation.
6
+ */
7
+
8
+ import React, { useState, useCallback } from 'react';
9
+ import type { ChannelMessage } from './hooks/useChannels';
10
+
11
+ export interface ChannelSidebarProps {
12
+ /** List of joined channels */
13
+ channels: string[];
14
+ /** Currently selected channel */
15
+ selectedChannel?: string;
16
+ /** Callback when channel is selected */
17
+ onSelectChannel: (channel: string) => void;
18
+ /** Callback to join a channel */
19
+ onJoinChannel: (channel: string) => void;
20
+ /** Callback to leave a channel */
21
+ onLeaveChannel: (channel: string) => void;
22
+ /** Unread message counts per channel */
23
+ unreadCounts?: Record<string, number>;
24
+ /** Whether connected to server */
25
+ isConnected?: boolean;
26
+ }
27
+
28
+ const DEFAULT_CHANNELS = ['#general', '#random', '#help'];
29
+
30
+ export function ChannelSidebar({
31
+ channels,
32
+ selectedChannel,
33
+ onSelectChannel,
34
+ onJoinChannel,
35
+ onLeaveChannel,
36
+ unreadCounts = {},
37
+ isConnected = true,
38
+ }: ChannelSidebarProps) {
39
+ const [showJoinModal, setShowJoinModal] = useState(false);
40
+ const [newChannelName, setNewChannelName] = useState('');
41
+
42
+ const handleJoinChannel = useCallback(() => {
43
+ if (!newChannelName.trim()) return;
44
+
45
+ let channelName = newChannelName.trim();
46
+ // Auto-prefix with # if missing
47
+ if (!channelName.startsWith('#') && !channelName.startsWith('dm:')) {
48
+ channelName = `#${channelName}`;
49
+ }
50
+
51
+ onJoinChannel(channelName);
52
+ setNewChannelName('');
53
+ setShowJoinModal(false);
54
+ }, [newChannelName, onJoinChannel]);
55
+
56
+ const handleLeaveChannel = useCallback((e: React.MouseEvent, channel: string) => {
57
+ e.stopPropagation();
58
+ onLeaveChannel(channel);
59
+ }, [onLeaveChannel]);
60
+
61
+ // Separate public channels and DMs
62
+ const publicChannels = channels.filter(c => c.startsWith('#'));
63
+ const dmChannels = channels.filter(c => c.startsWith('dm:'));
64
+
65
+ return (
66
+ <div className="channel-sidebar" style={{
67
+ width: '240px',
68
+ backgroundColor: 'var(--bg-secondary, #1e1e2e)',
69
+ borderRight: '1px solid var(--border-color, #313244)',
70
+ display: 'flex',
71
+ flexDirection: 'column',
72
+ height: '100%',
73
+ }}>
74
+ {/* Header */}
75
+ <div style={{
76
+ padding: '16px',
77
+ borderBottom: '1px solid var(--border-color, #313244)',
78
+ display: 'flex',
79
+ alignItems: 'center',
80
+ justifyContent: 'space-between',
81
+ }}>
82
+ <div style={{ display: 'flex', alignItems: 'center', gap: '8px' }}>
83
+ <span style={{ fontSize: '14px', fontWeight: 600, color: 'var(--text-primary, #cdd6f4)' }}>
84
+ Channels
85
+ </span>
86
+ <span style={{
87
+ width: '8px',
88
+ height: '8px',
89
+ borderRadius: '50%',
90
+ backgroundColor: isConnected ? '#a6e3a1' : '#f38ba8',
91
+ }} />
92
+ </div>
93
+ <button
94
+ onClick={() => setShowJoinModal(true)}
95
+ style={{
96
+ background: 'none',
97
+ border: 'none',
98
+ color: 'var(--text-secondary, #a6adc8)',
99
+ cursor: 'pointer',
100
+ padding: '4px 8px',
101
+ borderRadius: '4px',
102
+ fontSize: '16px',
103
+ }}
104
+ title="Join or create channel"
105
+ >
106
+ +
107
+ </button>
108
+ </div>
109
+
110
+ {/* Channel List */}
111
+ <div style={{ flex: 1, overflow: 'auto', padding: '8px 0' }}>
112
+ {/* Public Channels */}
113
+ <div style={{ marginBottom: '16px' }}>
114
+ <div style={{
115
+ padding: '4px 16px',
116
+ fontSize: '11px',
117
+ fontWeight: 600,
118
+ color: 'var(--text-secondary, #a6adc8)',
119
+ textTransform: 'uppercase',
120
+ letterSpacing: '0.5px',
121
+ }}>
122
+ Channels
123
+ </div>
124
+ {publicChannels.length === 0 ? (
125
+ <div style={{
126
+ padding: '8px 16px',
127
+ fontSize: '13px',
128
+ color: 'var(--text-muted, #6c7086)',
129
+ fontStyle: 'italic',
130
+ }}>
131
+ No channels joined
132
+ </div>
133
+ ) : (
134
+ publicChannels.map(channel => (
135
+ <ChannelItem
136
+ key={channel}
137
+ channel={channel}
138
+ isSelected={selectedChannel === channel}
139
+ unreadCount={unreadCounts[channel] || 0}
140
+ onSelect={() => onSelectChannel(channel)}
141
+ onLeave={(e) => handleLeaveChannel(e, channel)}
142
+ />
143
+ ))
144
+ )}
145
+ </div>
146
+
147
+ {/* Direct Messages */}
148
+ <div>
149
+ <div style={{
150
+ padding: '4px 16px',
151
+ fontSize: '11px',
152
+ fontWeight: 600,
153
+ color: 'var(--text-secondary, #a6adc8)',
154
+ textTransform: 'uppercase',
155
+ letterSpacing: '0.5px',
156
+ }}>
157
+ Direct Messages
158
+ </div>
159
+ {dmChannels.length === 0 ? (
160
+ <div style={{
161
+ padding: '8px 16px',
162
+ fontSize: '13px',
163
+ color: 'var(--text-muted, #6c7086)',
164
+ fontStyle: 'italic',
165
+ }}>
166
+ No conversations
167
+ </div>
168
+ ) : (
169
+ dmChannels.map(channel => (
170
+ <ChannelItem
171
+ key={channel}
172
+ channel={channel}
173
+ displayName={formatDmName(channel)}
174
+ isSelected={selectedChannel === channel}
175
+ unreadCount={unreadCounts[channel] || 0}
176
+ onSelect={() => onSelectChannel(channel)}
177
+ onLeave={(e) => handleLeaveChannel(e, channel)}
178
+ isDm
179
+ />
180
+ ))
181
+ )}
182
+ </div>
183
+ </div>
184
+
185
+ {/* Quick Join Suggestions */}
186
+ {channels.length === 0 && (
187
+ <div style={{
188
+ padding: '16px',
189
+ borderTop: '1px solid var(--border-color, #313244)',
190
+ }}>
191
+ <div style={{
192
+ fontSize: '12px',
193
+ color: 'var(--text-secondary, #a6adc8)',
194
+ marginBottom: '8px',
195
+ }}>
196
+ Suggested channels:
197
+ </div>
198
+ <div style={{ display: 'flex', flexWrap: 'wrap', gap: '4px' }}>
199
+ {DEFAULT_CHANNELS.map(channel => (
200
+ <button
201
+ key={channel}
202
+ onClick={() => onJoinChannel(channel)}
203
+ style={{
204
+ padding: '4px 8px',
205
+ fontSize: '12px',
206
+ backgroundColor: 'var(--bg-tertiary, #313244)',
207
+ border: 'none',
208
+ borderRadius: '4px',
209
+ color: 'var(--text-primary, #cdd6f4)',
210
+ cursor: 'pointer',
211
+ }}
212
+ >
213
+ {channel}
214
+ </button>
215
+ ))}
216
+ </div>
217
+ </div>
218
+ )}
219
+
220
+ {/* Join Modal */}
221
+ {showJoinModal && (
222
+ <div style={{
223
+ position: 'fixed',
224
+ top: 0,
225
+ left: 0,
226
+ right: 0,
227
+ bottom: 0,
228
+ backgroundColor: 'rgba(0, 0, 0, 0.5)',
229
+ display: 'flex',
230
+ alignItems: 'center',
231
+ justifyContent: 'center',
232
+ zIndex: 1000,
233
+ }} onClick={() => setShowJoinModal(false)}>
234
+ <div style={{
235
+ backgroundColor: 'var(--bg-secondary, #1e1e2e)',
236
+ borderRadius: '8px',
237
+ padding: '24px',
238
+ width: '320px',
239
+ boxShadow: '0 4px 24px rgba(0,0,0,0.3)',
240
+ }} onClick={e => e.stopPropagation()}>
241
+ <h3 style={{
242
+ margin: '0 0 16px 0',
243
+ fontSize: '16px',
244
+ fontWeight: 600,
245
+ color: 'var(--text-primary, #cdd6f4)',
246
+ }}>
247
+ Join Channel
248
+ </h3>
249
+ <input
250
+ type="text"
251
+ value={newChannelName}
252
+ onChange={(e) => setNewChannelName(e.target.value)}
253
+ placeholder="Channel name (e.g., general)"
254
+ style={{
255
+ width: '100%',
256
+ padding: '10px 12px',
257
+ backgroundColor: 'var(--bg-primary, #11111b)',
258
+ border: '1px solid var(--border-color, #313244)',
259
+ borderRadius: '6px',
260
+ color: 'var(--text-primary, #cdd6f4)',
261
+ fontSize: '14px',
262
+ marginBottom: '16px',
263
+ outline: 'none',
264
+ }}
265
+ onKeyDown={(e) => e.key === 'Enter' && handleJoinChannel()}
266
+ autoFocus
267
+ />
268
+ <div style={{ display: 'flex', gap: '8px', justifyContent: 'flex-end' }}>
269
+ <button
270
+ onClick={() => setShowJoinModal(false)}
271
+ style={{
272
+ padding: '8px 16px',
273
+ backgroundColor: 'transparent',
274
+ border: '1px solid var(--border-color, #313244)',
275
+ borderRadius: '6px',
276
+ color: 'var(--text-primary, #cdd6f4)',
277
+ cursor: 'pointer',
278
+ fontSize: '14px',
279
+ }}
280
+ >
281
+ Cancel
282
+ </button>
283
+ <button
284
+ onClick={handleJoinChannel}
285
+ disabled={!newChannelName.trim()}
286
+ style={{
287
+ padding: '8px 16px',
288
+ backgroundColor: 'var(--accent-color, #89b4fa)',
289
+ border: 'none',
290
+ borderRadius: '6px',
291
+ color: '#11111b',
292
+ cursor: 'pointer',
293
+ fontSize: '14px',
294
+ fontWeight: 500,
295
+ opacity: newChannelName.trim() ? 1 : 0.5,
296
+ }}
297
+ >
298
+ Join
299
+ </button>
300
+ </div>
301
+ </div>
302
+ </div>
303
+ )}
304
+ </div>
305
+ );
306
+ }
307
+
308
+ interface ChannelItemProps {
309
+ channel: string;
310
+ displayName?: string;
311
+ isSelected: boolean;
312
+ unreadCount: number;
313
+ onSelect: () => void;
314
+ onLeave: (e: React.MouseEvent) => void;
315
+ isDm?: boolean;
316
+ }
317
+
318
+ function ChannelItem({ channel, displayName, isSelected, unreadCount, onSelect, onLeave, isDm }: ChannelItemProps) {
319
+ const [showLeave, setShowLeave] = useState(false);
320
+
321
+ return (
322
+ <div
323
+ onClick={onSelect}
324
+ onMouseEnter={() => setShowLeave(true)}
325
+ onMouseLeave={() => setShowLeave(false)}
326
+ style={{
327
+ padding: '6px 16px',
328
+ display: 'flex',
329
+ alignItems: 'center',
330
+ justifyContent: 'space-between',
331
+ cursor: 'pointer',
332
+ backgroundColor: isSelected ? 'var(--bg-tertiary, #313244)' : 'transparent',
333
+ color: unreadCount > 0
334
+ ? 'var(--text-primary, #cdd6f4)'
335
+ : 'var(--text-secondary, #a6adc8)',
336
+ fontWeight: unreadCount > 0 ? 600 : 400,
337
+ fontSize: '14px',
338
+ }}
339
+ >
340
+ <div style={{ display: 'flex', alignItems: 'center', gap: '8px', flex: 1, minWidth: 0 }}>
341
+ <span style={{ flexShrink: 0 }}>
342
+ {isDm ? '@' : '#'}
343
+ </span>
344
+ <span style={{
345
+ overflow: 'hidden',
346
+ textOverflow: 'ellipsis',
347
+ whiteSpace: 'nowrap',
348
+ }}>
349
+ {displayName || channel.replace(/^#/, '')}
350
+ </span>
351
+ </div>
352
+ <div style={{ display: 'flex', alignItems: 'center', gap: '8px' }}>
353
+ {unreadCount > 0 && (
354
+ <span style={{
355
+ backgroundColor: 'var(--accent-color, #89b4fa)',
356
+ color: '#11111b',
357
+ fontSize: '11px',
358
+ fontWeight: 600,
359
+ padding: '2px 6px',
360
+ borderRadius: '10px',
361
+ minWidth: '18px',
362
+ textAlign: 'center',
363
+ }}>
364
+ {unreadCount > 99 ? '99+' : unreadCount}
365
+ </span>
366
+ )}
367
+ {showLeave && (
368
+ <button
369
+ onClick={onLeave}
370
+ style={{
371
+ background: 'none',
372
+ border: 'none',
373
+ color: 'var(--text-muted, #6c7086)',
374
+ cursor: 'pointer',
375
+ padding: '2px',
376
+ fontSize: '12px',
377
+ lineHeight: 1,
378
+ }}
379
+ title="Leave channel"
380
+ >
381
+ x
382
+ </button>
383
+ )}
384
+ </div>
385
+ </div>
386
+ );
387
+ }
388
+
389
+ /**
390
+ * Format DM channel name for display.
391
+ * dm:alice:bob -> "alice, bob" (excluding current user if known)
392
+ */
393
+ function formatDmName(channel: string): string {
394
+ if (!channel.startsWith('dm:')) return channel;
395
+ const parts = channel.split(':').slice(1);
396
+ return parts.join(', ');
397
+ }
398
+
399
+ export default ChannelSidebar;
@@ -0,0 +1,130 @@
1
+ /**
2
+ * Cloud Session Provider
3
+ *
4
+ * Wraps the dashboard app to provide cloud session management.
5
+ * Automatically detects session expiration and prompts re-login.
6
+ *
7
+ * Usage:
8
+ * ```tsx
9
+ * <CloudSessionProvider>
10
+ * <App />
11
+ * </CloudSessionProvider>
12
+ * ```
13
+ */
14
+
15
+ import React, { createContext, useContext, useCallback } from 'react';
16
+ import { useSession, type UseSessionReturn, type SessionError } from './hooks/useSession';
17
+ import { SessionExpiredModal } from './SessionExpiredModal';
18
+
19
+ // Context type
20
+ interface CloudSessionContextValue extends UseSessionReturn {
21
+ /** Whether this is a cloud-hosted dashboard */
22
+ isCloudMode: boolean;
23
+ }
24
+
25
+ // Create context with undefined default
26
+ const CloudSessionContext = createContext<CloudSessionContextValue | undefined>(undefined);
27
+
28
+ export interface CloudSessionProviderProps {
29
+ /** Child components */
30
+ children: React.ReactNode;
31
+ /** Whether this dashboard is running in cloud mode (default: auto-detect) */
32
+ cloudMode?: boolean;
33
+ /** Session check interval in ms (default: 60000) */
34
+ checkInterval?: number;
35
+ /** Callback when session expires */
36
+ onSessionExpired?: (error: SessionError) => void;
37
+ }
38
+
39
+ /**
40
+ * Auto-detect if running in cloud mode
41
+ * Cloud mode is detected by checking for cloud-specific environment markers
42
+ */
43
+ function detectCloudMode(): boolean {
44
+ if (typeof window === 'undefined') return false;
45
+
46
+ // Check for cloud URL patterns
47
+ const hostname = window.location.hostname;
48
+ if (hostname.includes('agent-relay.com')) return true;
49
+ if (hostname.includes('agentrelay.cloud')) return true;
50
+
51
+ // Check for cloud mode flag in meta tags
52
+ const cloudMeta = document.querySelector('meta[name="agent-relay-cloud"]');
53
+ if (cloudMeta?.getAttribute('content') === 'true') return true;
54
+
55
+ // Check for cloud mode in local storage (for development)
56
+ if (localStorage.getItem('agent-relay-cloud-mode') === 'true') return true;
57
+
58
+ return false;
59
+ }
60
+
61
+ export function CloudSessionProvider({
62
+ children,
63
+ cloudMode,
64
+ checkInterval = 60000,
65
+ onSessionExpired,
66
+ }: CloudSessionProviderProps) {
67
+ const isCloudMode = cloudMode ?? detectCloudMode();
68
+
69
+ // Use session hook only in cloud mode
70
+ const session = useSession({
71
+ checkOnMount: isCloudMode,
72
+ checkInterval: isCloudMode ? checkInterval : 0,
73
+ onExpired: onSessionExpired,
74
+ });
75
+
76
+ // Handle login redirect
77
+ const handleLogin = useCallback(() => {
78
+ session.redirectToLogin();
79
+ }, [session]);
80
+
81
+ // Handle modal dismiss (optional - keeps modal closable for some use cases)
82
+ const handleDismiss = useCallback(() => {
83
+ session.clearExpired();
84
+ }, [session]);
85
+
86
+ // Context value
87
+ const contextValue: CloudSessionContextValue = {
88
+ ...session,
89
+ isCloudMode,
90
+ };
91
+
92
+ return (
93
+ <CloudSessionContext.Provider value={contextValue}>
94
+ {children}
95
+
96
+ {/* Session Expired Modal - only shown in cloud mode */}
97
+ {isCloudMode && (
98
+ <SessionExpiredModal
99
+ isOpen={session.isExpired}
100
+ error={session.error}
101
+ onLogin={handleLogin}
102
+ onDismiss={handleDismiss}
103
+ />
104
+ )}
105
+ </CloudSessionContext.Provider>
106
+ );
107
+ }
108
+
109
+ /**
110
+ * Hook to access cloud session context
111
+ *
112
+ * @throws Error if used outside of CloudSessionProvider
113
+ */
114
+ export function useCloudSession(): CloudSessionContextValue {
115
+ const context = useContext(CloudSessionContext);
116
+ if (!context) {
117
+ throw new Error('useCloudSession must be used within a CloudSessionProvider');
118
+ }
119
+ return context;
120
+ }
121
+
122
+ /**
123
+ * Hook to optionally access cloud session context
124
+ * Returns undefined if not within a CloudSessionProvider
125
+ */
126
+ export function useCloudSessionOptional(): CloudSessionContextValue | undefined {
127
+ return useContext(CloudSessionContext);
128
+ }
129
+
130
+ export default CloudSessionProvider;