@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,463 @@
1
+ /**
2
+ * ChannelMessageList Component
3
+ *
4
+ * Displays messages in a channel with:
5
+ * - Unread separator
6
+ * - Date dividers
7
+ * - Auto-scroll to new messages
8
+ * - Infinite scroll for history
9
+ */
10
+
11
+ import React, { useRef, useEffect, useCallback, useState, useMemo } from 'react';
12
+ import type { ChannelMessage, ChannelMessageListProps, UnreadState } from './types';
13
+ import { formatMessageBody } from '../utils/messageFormatting';
14
+
15
+ export function ChannelMessageList({
16
+ messages,
17
+ unreadState,
18
+ currentUser,
19
+ isLoadingMore = false,
20
+ hasMore = false,
21
+ onLoadMore,
22
+ onThreadClick,
23
+ onMemberClick,
24
+ }: ChannelMessageListProps) {
25
+ const containerRef = useRef<HTMLDivElement>(null);
26
+ const bottomRef = useRef<HTMLDivElement>(null);
27
+ const [isNearBottom, setIsNearBottom] = useState(true);
28
+ const [showScrollToBottom, setShowScrollToBottom] = useState(false);
29
+
30
+ // Group messages by date
31
+ const groupedMessages = useMemo(() => {
32
+ const groups: { date: string; messages: ChannelMessage[] }[] = [];
33
+ let currentDate = '';
34
+
35
+ messages.forEach(message => {
36
+ const messageDate = formatDateKey(message.timestamp);
37
+ if (messageDate !== currentDate) {
38
+ currentDate = messageDate;
39
+ groups.push({ date: messageDate, messages: [message] });
40
+ } else {
41
+ groups[groups.length - 1].messages.push(message);
42
+ }
43
+ });
44
+
45
+ return groups;
46
+ }, [messages]);
47
+
48
+ // Handle scroll events
49
+ const handleScroll = useCallback(() => {
50
+ const container = containerRef.current;
51
+ if (!container) return;
52
+
53
+ const { scrollTop, scrollHeight, clientHeight } = container;
54
+ const distanceFromBottom = scrollHeight - scrollTop - clientHeight;
55
+ const nearBottom = distanceFromBottom < 100;
56
+
57
+ setIsNearBottom(nearBottom);
58
+ setShowScrollToBottom(!nearBottom && distanceFromBottom > 500);
59
+
60
+ // Load more when scrolling near top
61
+ if (scrollTop < 100 && hasMore && !isLoadingMore && onLoadMore) {
62
+ onLoadMore();
63
+ }
64
+ }, [hasMore, isLoadingMore, onLoadMore]);
65
+
66
+ // Scroll to bottom when new messages arrive (if near bottom)
67
+ useEffect(() => {
68
+ if (isNearBottom && bottomRef.current) {
69
+ bottomRef.current.scrollIntoView({ behavior: 'smooth' });
70
+ }
71
+ }, [messages.length, isNearBottom]);
72
+
73
+ // Initial scroll to bottom
74
+ useEffect(() => {
75
+ if (bottomRef.current) {
76
+ bottomRef.current.scrollIntoView();
77
+ }
78
+ }, []);
79
+
80
+ const scrollToBottom = useCallback(() => {
81
+ bottomRef.current?.scrollIntoView({ behavior: 'smooth' });
82
+ }, []);
83
+
84
+ return (
85
+ <div className="relative flex-1 overflow-hidden">
86
+ <div
87
+ ref={containerRef}
88
+ onScroll={handleScroll}
89
+ className="h-full overflow-y-auto px-4 py-2"
90
+ >
91
+ {/* Loading more indicator */}
92
+ {isLoadingMore && (
93
+ <div className="flex justify-center py-4">
94
+ <LoadingSpinner />
95
+ </div>
96
+ )}
97
+
98
+ {/* Load more button */}
99
+ {hasMore && !isLoadingMore && (
100
+ <div className="flex justify-center py-4">
101
+ <button
102
+ onClick={onLoadMore}
103
+ className="text-sm text-accent-cyan hover:underline"
104
+ >
105
+ Load earlier messages
106
+ </button>
107
+ </div>
108
+ )}
109
+
110
+ {/* Empty state */}
111
+ {messages.length === 0 && !isLoadingMore && (
112
+ <div className="flex flex-col items-center justify-center h-full text-center py-12">
113
+ <div className="text-4xl mb-3">
114
+ <MessageIcon className="w-12 h-12 text-text-muted" />
115
+ </div>
116
+ <h3 className="text-lg font-medium text-text-primary mb-1">
117
+ No messages yet
118
+ </h3>
119
+ <p className="text-sm text-text-muted">
120
+ Be the first to send a message in this channel
121
+ </p>
122
+ </div>
123
+ )}
124
+
125
+ {/* Message groups */}
126
+ {groupedMessages.map(({ date, messages: dateMessages }) => (
127
+ <div key={date}>
128
+ {/* Date divider */}
129
+ <DateDivider date={date} />
130
+
131
+ {/* Messages for this date */}
132
+ {dateMessages.map((message, index) => {
133
+ const isFirstUnread = unreadState?.firstUnreadMessageId === message.id;
134
+ const showUnreadSeparator = isFirstUnread && unreadState && unreadState.count > 0;
135
+
136
+ return (
137
+ <React.Fragment key={message.id}>
138
+ {/* Unread separator */}
139
+ {showUnreadSeparator && (
140
+ <UnreadSeparator count={unreadState.count} />
141
+ )}
142
+
143
+ {/* Message */}
144
+ <MessageItem
145
+ message={message}
146
+ isOwn={message.from === currentUser}
147
+ onThreadClick={onThreadClick}
148
+ onMemberClick={onMemberClick}
149
+ showAvatar={shouldShowAvatar(dateMessages, index)}
150
+ />
151
+ </React.Fragment>
152
+ );
153
+ })}
154
+ </div>
155
+ ))}
156
+
157
+ {/* Scroll anchor */}
158
+ <div ref={bottomRef} />
159
+ </div>
160
+
161
+ {/* Scroll to bottom button */}
162
+ {showScrollToBottom && (
163
+ <button
164
+ onClick={scrollToBottom}
165
+ className="absolute bottom-4 right-4 p-3 bg-bg-elevated border border-border-subtle rounded-full shadow-lg hover:bg-bg-hover transition-colors"
166
+ title="Scroll to bottom"
167
+ >
168
+ <ChevronDownIcon className="w-5 h-5 text-text-primary" />
169
+ </button>
170
+ )}
171
+ </div>
172
+ );
173
+ }
174
+
175
+ // =============================================================================
176
+ // Sub-components
177
+ // =============================================================================
178
+
179
+ interface MessageItemProps {
180
+ message: ChannelMessage;
181
+ isOwn: boolean;
182
+ onThreadClick?: (messageId: string) => void;
183
+ onMemberClick?: (memberId: string, entityType: 'user' | 'agent') => void;
184
+ showAvatar: boolean;
185
+ }
186
+
187
+ function MessageItem({
188
+ message,
189
+ isOwn,
190
+ onThreadClick,
191
+ onMemberClick,
192
+ showAvatar,
193
+ }: MessageItemProps) {
194
+ const hasThread = message.threadSummary && message.threadSummary.replyCount > 0;
195
+
196
+ return (
197
+ <div className={`group relative py-1 ${showAvatar ? 'mt-3' : ''}`}>
198
+ <div className="flex gap-3">
199
+ {/* Avatar column */}
200
+ <div className="w-9 flex-shrink-0">
201
+ {showAvatar && (
202
+ <Avatar
203
+ name={message.from}
204
+ avatarUrl={message.fromAvatarUrl}
205
+ entityType={message.fromEntityType}
206
+ />
207
+ )}
208
+ </div>
209
+
210
+ {/* Content column */}
211
+ <div className="flex-1 min-w-0">
212
+ {/* Header (only show with avatar) */}
213
+ {showAvatar && (
214
+ <div className="flex items-center gap-2 mb-0.5">
215
+ <button
216
+ type="button"
217
+ onClick={() => {
218
+ if (!isOwn && onMemberClick) {
219
+ onMemberClick(message.from, message.fromEntityType || 'agent');
220
+ }
221
+ }}
222
+ disabled={isOwn || !onMemberClick}
223
+ className={`text-sm font-semibold ${
224
+ isOwn ? 'text-accent-cyan cursor-default' : 'text-text-primary hover:underline cursor-pointer'
225
+ } disabled:cursor-default disabled:hover:no-underline`}
226
+ >
227
+ {message.from}
228
+ </button>
229
+ <span className="text-xs text-text-muted">
230
+ {formatTime(message.timestamp)}
231
+ </span>
232
+ {message.editedAt && (
233
+ <span className="text-xs text-text-muted">(edited)</span>
234
+ )}
235
+
236
+ {/* Thread button */}
237
+ <button
238
+ className={`
239
+ inline-flex items-center gap-1.5 p-1.5 rounded-lg transition-all duration-150 cursor-pointer border-none
240
+ ${hasThread || message.threadId
241
+ ? 'text-accent-cyan bg-accent-cyan/10 hover:bg-accent-cyan/20'
242
+ : 'text-text-muted bg-transparent opacity-0 group-hover:opacity-100 hover:text-accent-cyan hover:bg-accent-cyan/10'}
243
+ `}
244
+ onClick={() => onThreadClick?.(message.threadId || message.id)}
245
+ title={message.threadId ? `View thread` : (hasThread ? `${message.threadSummary!.replyCount} ${message.threadSummary!.replyCount === 1 ? 'reply' : 'replies'}` : 'Reply in thread')}
246
+ >
247
+ <ThreadIcon className="w-3.5 h-3.5" />
248
+ {hasThread && (
249
+ <span className="text-xs font-medium">{message.threadSummary!.replyCount}</span>
250
+ )}
251
+ </button>
252
+ </div>
253
+ )}
254
+
255
+ {/* Message content */}
256
+ <div className="text-sm text-text-primary whitespace-pre-wrap break-words">
257
+ {formatMessageBody(message.content, { mentions: message.mentions })}
258
+ </div>
259
+
260
+ {/* Attachments */}
261
+ {message.attachments && message.attachments.length > 0 && (
262
+ <div className="mt-2 flex flex-wrap gap-2">
263
+ {message.attachments.map(attachment => (
264
+ <AttachmentPreview key={attachment.id} attachment={attachment} />
265
+ ))}
266
+ </div>
267
+ )}
268
+ </div>
269
+ </div>
270
+ </div>
271
+ );
272
+ }
273
+
274
+ function Avatar({
275
+ name,
276
+ avatarUrl,
277
+ entityType,
278
+ }: {
279
+ name: string;
280
+ avatarUrl?: string;
281
+ entityType: 'agent' | 'user';
282
+ }) {
283
+ if (avatarUrl) {
284
+ return (
285
+ <img
286
+ src={avatarUrl}
287
+ alt={name}
288
+ className="w-9 h-9 rounded-full object-cover"
289
+ />
290
+ );
291
+ }
292
+
293
+ return (
294
+ <div className={`
295
+ w-9 h-9 rounded-full flex items-center justify-center text-sm font-medium
296
+ ${entityType === 'user'
297
+ ? 'bg-purple-500/30 text-purple-300'
298
+ : 'bg-accent-cyan/30 text-accent-cyan'}
299
+ `}>
300
+ {name.charAt(0).toUpperCase()}
301
+ </div>
302
+ );
303
+ }
304
+
305
+ function AttachmentPreview({ attachment }: { attachment: NonNullable<ChannelMessage['attachments']>[0] }) {
306
+ const isImage = attachment.mimeType.startsWith('image/');
307
+
308
+ if (isImage) {
309
+ return (
310
+ <a
311
+ href={attachment.url}
312
+ target="_blank"
313
+ rel="noopener noreferrer"
314
+ className="block max-w-xs rounded-lg overflow-hidden border border-border-subtle hover:border-accent-cyan/30 transition-colors"
315
+ >
316
+ <img
317
+ src={attachment.thumbnailUrl || attachment.url}
318
+ alt={attachment.filename}
319
+ className="max-w-full max-h-48 object-cover"
320
+ />
321
+ </a>
322
+ );
323
+ }
324
+
325
+ return (
326
+ <a
327
+ href={attachment.url}
328
+ target="_blank"
329
+ rel="noopener noreferrer"
330
+ className="flex items-center gap-2 px-3 py-2 bg-bg-tertiary rounded-lg border border-border-subtle hover:border-accent-cyan/30 transition-colors"
331
+ >
332
+ <FileIcon className="w-5 h-5 text-text-muted" />
333
+ <div className="min-w-0">
334
+ <p className="text-sm text-text-primary truncate">{attachment.filename}</p>
335
+ <p className="text-xs text-text-muted">{formatFileSize(attachment.size)}</p>
336
+ </div>
337
+ </a>
338
+ );
339
+ }
340
+
341
+ function DateDivider({ date }: { date: string }) {
342
+ return (
343
+ <div className="flex items-center gap-3 py-3">
344
+ <div className="flex-1 h-px bg-border-subtle" />
345
+ <span className="text-xs font-medium text-text-muted px-2">
346
+ {formatDateDisplay(date)}
347
+ </span>
348
+ <div className="flex-1 h-px bg-border-subtle" />
349
+ </div>
350
+ );
351
+ }
352
+
353
+ function UnreadSeparator({ count }: { count: number }) {
354
+ return (
355
+ <div className="flex items-center gap-3 py-2 my-2">
356
+ <div className="flex-1 h-px bg-red-500/50" />
357
+ <span className="text-xs font-semibold text-red-400 px-2 flex items-center gap-1">
358
+ <span className="w-2 h-2 bg-red-500 rounded-full" />
359
+ {count} new {count === 1 ? 'message' : 'messages'}
360
+ </span>
361
+ <div className="flex-1 h-px bg-red-500/50" />
362
+ </div>
363
+ );
364
+ }
365
+
366
+ function LoadingSpinner() {
367
+ return (
368
+ <div className="w-5 h-5 border-2 border-accent-cyan/30 border-t-accent-cyan rounded-full animate-spin" />
369
+ );
370
+ }
371
+
372
+ // =============================================================================
373
+ // Helper functions
374
+ // =============================================================================
375
+
376
+ function shouldShowAvatar(messages: ChannelMessage[], index: number): boolean {
377
+ if (index === 0) return true;
378
+ const current = messages[index];
379
+ const previous = messages[index - 1];
380
+
381
+ // Show avatar if different sender
382
+ if (current.from !== previous.from) return true;
383
+
384
+ // Show avatar if more than 5 minutes since last message
385
+ const currentTime = new Date(current.timestamp).getTime();
386
+ const previousTime = new Date(previous.timestamp).getTime();
387
+ return currentTime - previousTime > 5 * 60 * 1000;
388
+ }
389
+
390
+ function formatDateKey(isoString: string): string {
391
+ return new Date(isoString).toDateString();
392
+ }
393
+
394
+ function formatDateDisplay(dateKey: string): string {
395
+ const date = new Date(dateKey);
396
+ const today = new Date();
397
+ const yesterday = new Date(today);
398
+ yesterday.setDate(yesterday.getDate() - 1);
399
+
400
+ if (date.toDateString() === today.toDateString()) {
401
+ return 'Today';
402
+ }
403
+ if (date.toDateString() === yesterday.toDateString()) {
404
+ return 'Yesterday';
405
+ }
406
+ return date.toLocaleDateString(undefined, {
407
+ weekday: 'long',
408
+ month: 'long',
409
+ day: 'numeric',
410
+ });
411
+ }
412
+
413
+ function formatTime(isoString: string): string {
414
+ return new Date(isoString).toLocaleTimeString(undefined, {
415
+ hour: '2-digit',
416
+ minute: '2-digit',
417
+ });
418
+ }
419
+
420
+ function formatFileSize(bytes: number): string {
421
+ if (bytes < 1024) return `${bytes} B`;
422
+ if (bytes < 1024 * 1024) return `${(bytes / 1024).toFixed(1)} KB`;
423
+ return `${(bytes / (1024 * 1024)).toFixed(1)} MB`;
424
+ }
425
+
426
+ // =============================================================================
427
+ // Icons
428
+ // =============================================================================
429
+
430
+ function MessageIcon({ className }: { className?: string }) {
431
+ return (
432
+ <svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="1.5" strokeLinecap="round" strokeLinejoin="round">
433
+ <path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" />
434
+ </svg>
435
+ );
436
+ }
437
+
438
+ function ThreadIcon({ className }: { className?: string }) {
439
+ return (
440
+ <svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
441
+ <path d="M21 15a2 2 0 0 1-2 2H7l-4 4V5a2 2 0 0 1 2-2h14a2 2 0 0 1 2 2z" />
442
+ </svg>
443
+ );
444
+ }
445
+
446
+ function ChevronDownIcon({ className }: { className?: string }) {
447
+ return (
448
+ <svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
449
+ <polyline points="6 9 12 15 18 9" />
450
+ </svg>
451
+ );
452
+ }
453
+
454
+ function FileIcon({ className }: { className?: string }) {
455
+ return (
456
+ <svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
457
+ <path d="M14 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V8z" />
458
+ <polyline points="14 2 14 8 20 8" />
459
+ </svg>
460
+ );
461
+ }
462
+
463
+ export default ChannelMessageList;
@@ -0,0 +1,146 @@
1
+ /**
2
+ * ChannelViewV1 Component
3
+ *
4
+ * Composed channel view that combines:
5
+ * - ChannelHeader
6
+ * - ChannelMessageList
7
+ * - MessageInput
8
+ *
9
+ * This is the main view component for displaying a channel's content.
10
+ */
11
+
12
+ import React, { useCallback, useMemo } from 'react';
13
+ import { ChannelHeader } from './ChannelHeader';
14
+ import { ChannelMessageList } from './ChannelMessageList';
15
+ import { MessageInput } from './MessageInput';
16
+ import type {
17
+ Channel,
18
+ ChannelMember,
19
+ ChannelMessage,
20
+ UnreadState,
21
+ } from './types';
22
+
23
+ export interface ChannelViewV1Props {
24
+ /** Current channel to display */
25
+ channel: Channel;
26
+ /** Channel members */
27
+ members?: ChannelMember[];
28
+ /** Messages in the channel */
29
+ messages: ChannelMessage[];
30
+ /** Unread state for the channel */
31
+ unreadState?: UnreadState;
32
+ /** Current user's name */
33
+ currentUser: string;
34
+ /** Whether user can edit the channel */
35
+ canEditChannel?: boolean;
36
+ /** Whether loading more messages */
37
+ isLoadingMore?: boolean;
38
+ /** Whether there are more messages to load */
39
+ hasMoreMessages?: boolean;
40
+ /** Available users/agents for @-mentions */
41
+ mentionSuggestions?: string[];
42
+ /** Callback to load more messages */
43
+ onLoadMore?: () => void;
44
+ /** Callback to send a message */
45
+ onSendMessage: (content: string) => void;
46
+ /** Callback when editing channel settings */
47
+ onEditChannel?: () => void;
48
+ /** Callback to show member list */
49
+ onShowMembers?: () => void;
50
+ /** Callback to show pinned messages */
51
+ onShowPinned?: () => void;
52
+ /** Callback to search in channel */
53
+ onSearch?: () => void;
54
+ /** Callback when clicking thread button */
55
+ onThreadClick?: (messageId: string) => void;
56
+ /** Callback when typing status changes */
57
+ onTyping?: (isTyping: boolean) => void;
58
+ /** Callback to mark messages as read */
59
+ onMarkRead?: (upToTimestamp: string) => void;
60
+ /** Callback when clicking on a member name (for DM navigation) */
61
+ onMemberClick?: (memberId: string, entityType: 'user' | 'agent') => void;
62
+ }
63
+
64
+ export function ChannelViewV1({
65
+ channel,
66
+ members = [],
67
+ messages,
68
+ unreadState,
69
+ currentUser,
70
+ canEditChannel = false,
71
+ isLoadingMore = false,
72
+ hasMoreMessages = false,
73
+ mentionSuggestions = [],
74
+ onLoadMore,
75
+ onSendMessage,
76
+ onEditChannel,
77
+ onShowMembers,
78
+ onShowPinned,
79
+ onSearch,
80
+ onThreadClick,
81
+ onTyping,
82
+ onMarkRead,
83
+ onMemberClick,
84
+ }: ChannelViewV1Props) {
85
+ // Handle send
86
+ const handleSend = useCallback((content: string) => {
87
+ onSendMessage(content);
88
+ }, [onSendMessage]);
89
+
90
+ // Get placeholder text based on channel type
91
+ const inputPlaceholder = useMemo(() => {
92
+ if (channel.isDm) {
93
+ return `Message ${channel.name}`;
94
+ }
95
+ return `Message #${channel.name}`;
96
+ }, [channel]);
97
+
98
+ // Check if channel is archived (disable input)
99
+ const isArchived = channel.status === 'archived';
100
+
101
+ return (
102
+ <div className="flex flex-col h-full bg-bg-primary">
103
+ {/* Header */}
104
+ <ChannelHeader
105
+ channel={channel}
106
+ members={members}
107
+ canEdit={canEditChannel}
108
+ onEditChannel={onEditChannel}
109
+ onShowMembers={onShowMembers}
110
+ onShowPinned={onShowPinned}
111
+ onSearch={onSearch}
112
+ />
113
+
114
+ {/* Message List */}
115
+ <ChannelMessageList
116
+ messages={messages}
117
+ unreadState={unreadState}
118
+ currentUser={currentUser}
119
+ isLoadingMore={isLoadingMore}
120
+ hasMore={hasMoreMessages}
121
+ onLoadMore={onLoadMore}
122
+ onThreadClick={onThreadClick}
123
+ onMemberClick={onMemberClick}
124
+ />
125
+
126
+ {/* Message Input */}
127
+ {isArchived ? (
128
+ <div className="px-4 py-3 bg-bg-secondary border-t border-border-subtle text-center">
129
+ <p className="text-sm text-text-muted">
130
+ This channel is archived. Unarchive it to send messages.
131
+ </p>
132
+ </div>
133
+ ) : (
134
+ <MessageInput
135
+ channelId={channel.id}
136
+ placeholder={inputPlaceholder}
137
+ onSend={handleSend}
138
+ onTyping={onTyping}
139
+ mentionSuggestions={mentionSuggestions}
140
+ />
141
+ )}
142
+ </div>
143
+ );
144
+ }
145
+
146
+ export default ChannelViewV1;