@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.
- package/out/404.html +1 -1
- package/out/_next/static/chunks/{118-4c8241b0218335de.js → 118-ae2b650136a5a5fc.js} +1 -1
- package/out/_next/static/chunks/407-0c82986cf79c8ecb.js +1 -0
- package/out/_next/static/chunks/app/app/[[...slug]]/{page-1e81c047cff17212.js → page-f7eca1b66fb4249b.js} +1 -1
- package/out/_next/static/chunks/app/{page-6892fe2dd07fb48b.js → page-0ee604f7070d14c0.js} +1 -1
- package/out/_next/static/css/8968d98ed4c4d33f.css +1 -0
- package/out/about.html +2 -2
- package/out/about.txt +1 -1
- package/out/app/onboarding.html +1 -1
- package/out/app/onboarding.txt +1 -1
- package/out/app.html +1 -1
- package/out/app.txt +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.html +2 -2
- package/out/blog/go-to-bed-wake-up-to-a-finished-product.txt +1 -1
- package/out/blog/let-them-cook-multi-agent-orchestration.html +2 -2
- package/out/blog/let-them-cook-multi-agent-orchestration.txt +2 -2
- package/out/blog.html +2 -2
- package/out/blog.txt +1 -1
- package/out/careers.html +2 -2
- package/out/careers.txt +1 -1
- package/out/changelog.html +2 -2
- package/out/changelog.txt +1 -1
- package/out/cloud/link.html +1 -1
- package/out/cloud/link.txt +2 -2
- package/out/complete-profile.html +2 -2
- package/out/complete-profile.txt +1 -1
- package/out/connect-repos.html +1 -1
- package/out/connect-repos.txt +1 -1
- package/out/contact.html +2 -2
- package/out/contact.txt +1 -1
- package/out/docs.html +2 -2
- package/out/docs.txt +1 -1
- package/out/history.html +1 -1
- package/out/history.txt +2 -2
- package/out/index.html +1 -1
- package/out/index.txt +2 -2
- package/out/login.html +2 -2
- package/out/login.txt +1 -1
- package/out/metrics.html +1 -1
- package/out/metrics.txt +2 -2
- package/out/pricing.html +2 -2
- package/out/pricing.txt +1 -1
- package/out/privacy.html +2 -2
- package/out/privacy.txt +1 -1
- package/out/providers/setup/claude.html +1 -1
- package/out/providers/setup/claude.txt +1 -1
- package/out/providers/setup/codex.html +1 -1
- package/out/providers/setup/codex.txt +1 -1
- package/out/providers/setup/cursor.html +1 -1
- package/out/providers/setup/cursor.txt +1 -1
- package/out/providers.html +1 -1
- package/out/providers.txt +1 -1
- package/out/security.html +2 -2
- package/out/security.txt +1 -1
- package/out/signup.html +2 -2
- package/out/signup.txt +1 -1
- package/out/terms.html +2 -2
- package/out/terms.txt +1 -1
- package/package.json +7 -1
- package/src/app/about/page.tsx +7 -0
- package/src/app/app/[[...slug]]/DashboardPageClient.tsx +853 -0
- package/src/app/app/[[...slug]]/page.tsx +23 -0
- package/src/app/app/onboarding/page.tsx +394 -0
- package/src/app/apple-icon.png +0 -0
- package/src/app/blog/go-to-bed-wake-up-to-a-finished-product/page.tsx +88 -0
- package/src/app/blog/let-them-cook-multi-agent-orchestration/page.tsx +93 -0
- package/src/app/blog/page.tsx +15 -0
- package/src/app/careers/page.tsx +7 -0
- package/src/app/changelog/page.tsx +7 -0
- package/src/app/cloud/link/page.tsx +464 -0
- package/src/app/complete-profile/page.tsx +204 -0
- package/src/app/connect-repos/page.tsx +410 -0
- package/src/app/contact/page.tsx +7 -0
- package/src/app/docs/page.tsx +7 -0
- package/src/app/favicon.png +0 -0
- package/src/app/globals.css +200 -0
- package/src/app/history/page.tsx +658 -0
- package/src/app/layout.tsx +25 -0
- package/src/app/login/page.tsx +424 -0
- package/src/app/metrics/page.tsx +781 -0
- package/src/app/page.tsx +59 -0
- package/src/app/pricing/page.tsx +7 -0
- package/src/app/privacy/page.tsx +7 -0
- package/src/app/providers/page.tsx +193 -0
- package/src/app/providers/setup/[provider]/ProviderSetupClient.tsx +197 -0
- package/src/app/providers/setup/[provider]/constants.ts +35 -0
- package/src/app/providers/setup/[provider]/page.tsx +42 -0
- package/src/app/security/page.tsx +7 -0
- package/src/app/signup/page.tsx +533 -0
- package/src/app/terms/page.tsx +7 -0
- package/src/components/ActivityFeed.tsx +216 -0
- package/src/components/AddWorkspaceModal.tsx +170 -0
- package/src/components/AgentCard.test.tsx +134 -0
- package/src/components/AgentCard.tsx +585 -0
- package/src/components/AgentList.test.tsx +147 -0
- package/src/components/AgentList.tsx +419 -0
- package/src/components/AgentLogPreview.tsx +173 -0
- package/src/components/AgentProfilePanel.tsx +569 -0
- package/src/components/App.tsx +3424 -0
- package/src/components/BillingPanel.tsx +922 -0
- package/src/components/BillingResult.tsx +447 -0
- package/src/components/BroadcastComposer.tsx +690 -0
- package/src/components/ChannelAdminPanel.tsx +773 -0
- package/src/components/ChannelBrowser.tsx +385 -0
- package/src/components/ChannelChat.tsx +261 -0
- package/src/components/ChannelSidebar.tsx +399 -0
- package/src/components/CloudSessionProvider.tsx +130 -0
- package/src/components/CommandPalette.tsx +815 -0
- package/src/components/ConfirmationDialog.tsx +133 -0
- package/src/components/ConversationHistory.tsx +518 -0
- package/src/components/CoordinatorPanel.tsx +956 -0
- package/src/components/DecisionQueue.tsx +717 -0
- package/src/components/DirectMessageView.tsx +164 -0
- package/src/components/FileAutocomplete.tsx +368 -0
- package/src/components/FleetOverview.tsx +278 -0
- package/src/components/LogViewer.tsx +310 -0
- package/src/components/LogViewerPanel.tsx +482 -0
- package/src/components/Logo.tsx +284 -0
- package/src/components/MentionAutocomplete.tsx +384 -0
- package/src/components/MessageComposer.tsx +473 -0
- package/src/components/MessageList.tsx +725 -0
- package/src/components/MessageSenderName.tsx +91 -0
- package/src/components/MessageStatusIndicator.tsx +142 -0
- package/src/components/NewConversationModal.tsx +400 -0
- package/src/components/NotificationToast.tsx +488 -0
- package/src/components/OnlineUsersIndicator.tsx +164 -0
- package/src/components/Pagination.tsx +124 -0
- package/src/components/PricingPlans.tsx +386 -0
- package/src/components/ProjectList.tsx +711 -0
- package/src/components/ProviderAuthFlow.tsx +343 -0
- package/src/components/ProviderConnectionList.tsx +375 -0
- package/src/components/ProvisioningProgress.tsx +730 -0
- package/src/components/ReactionChips.tsx +70 -0
- package/src/components/ReactionPicker.tsx +121 -0
- package/src/components/RepoAccessPanel.tsx +787 -0
- package/src/components/RepositoriesPanel.tsx +901 -0
- package/src/components/ServerCard.tsx +202 -0
- package/src/components/SessionExpiredModal.tsx +128 -0
- package/src/components/SpawnModal.test.tsx +190 -0
- package/src/components/SpawnModal.tsx +1001 -0
- package/src/components/TaskAssignmentUI.tsx +375 -0
- package/src/components/TerminalProviderSetup.tsx +517 -0
- package/src/components/ThemeProvider.tsx +159 -0
- package/src/components/ThinkingIndicator.tsx +231 -0
- package/src/components/ThreadList.tsx +198 -0
- package/src/components/ThreadPanel.tsx +405 -0
- package/src/components/TrajectoryViewer.tsx +698 -0
- package/src/components/TypingIndicator.tsx +69 -0
- package/src/components/UsageBanner.tsx +231 -0
- package/src/components/UserProfilePanel.tsx +233 -0
- package/src/components/WorkspaceContext.tsx +95 -0
- package/src/components/WorkspaceSelector.tsx +234 -0
- package/src/components/WorkspaceStatusIndicator.tsx +396 -0
- package/src/components/XTermInteractive.tsx +516 -0
- package/src/components/XTermLogViewer.tsx +719 -0
- package/src/components/channels/ChannelDialogs.tsx +1411 -0
- package/src/components/channels/ChannelHeader.tsx +317 -0
- package/src/components/channels/ChannelMessageList.tsx +463 -0
- package/src/components/channels/ChannelViewV1.tsx +146 -0
- package/src/components/channels/MessageInput.tsx +302 -0
- package/src/components/channels/SearchInput.tsx +172 -0
- package/src/components/channels/SearchResults.tsx +336 -0
- package/src/components/channels/api.test.ts +1527 -0
- package/src/components/channels/api.ts +703 -0
- package/src/components/channels/index.ts +76 -0
- package/src/components/channels/mockApi.ts +344 -0
- package/src/components/channels/types.ts +566 -0
- package/src/components/hooks/index.ts +58 -0
- package/src/components/hooks/useAgentLogs.ts +504 -0
- package/src/components/hooks/useAgents.ts +127 -0
- package/src/components/hooks/useBroadcastDedup.test.ts +371 -0
- package/src/components/hooks/useBroadcastDedup.ts +86 -0
- package/src/components/hooks/useChannelAdmin.ts +329 -0
- package/src/components/hooks/useChannelBrowser.ts +239 -0
- package/src/components/hooks/useChannelCommands.ts +138 -0
- package/src/components/hooks/useChannels.ts +367 -0
- package/src/components/hooks/useDebounce.ts +29 -0
- package/src/components/hooks/useDirectMessage.test.ts +952 -0
- package/src/components/hooks/useDirectMessage.ts +141 -0
- package/src/components/hooks/useMessages.ts +310 -0
- package/src/components/hooks/useOrchestrator.test.ts +165 -0
- package/src/components/hooks/useOrchestrator.ts +424 -0
- package/src/components/hooks/usePinnedAgents.test.ts +356 -0
- package/src/components/hooks/usePinnedAgents.ts +140 -0
- package/src/components/hooks/usePresence.test.ts +245 -0
- package/src/components/hooks/usePresence.ts +377 -0
- package/src/components/hooks/useRecentRepos.ts +130 -0
- package/src/components/hooks/useSession.ts +209 -0
- package/src/components/hooks/useThread.ts +138 -0
- package/src/components/hooks/useTrajectory.ts +265 -0
- package/src/components/hooks/useWebSocket.ts +290 -0
- package/src/components/hooks/useWorkspaceMembers.ts +132 -0
- package/src/components/hooks/useWorkspaceRepos.ts +73 -0
- package/src/components/hooks/useWorkspaceStatus.ts +237 -0
- package/src/components/index.ts +81 -0
- package/src/components/layout/Header.tsx +311 -0
- package/src/components/layout/RepoContextHeader.tsx +361 -0
- package/src/components/layout/Sidebar.archive.test.tsx +126 -0
- package/src/components/layout/Sidebar.test.tsx +691 -0
- package/src/components/layout/Sidebar.tsx +900 -0
- package/src/components/layout/index.ts +7 -0
- package/src/components/settings/BillingSettingsPanel.tsx +564 -0
- package/src/components/settings/SettingsPage.tsx +683 -0
- package/src/components/settings/TeamSettingsPanel.tsx +560 -0
- package/src/components/settings/WorkspaceSettingsPanel.tsx +1368 -0
- package/src/components/settings/index.ts +11 -0
- package/src/components/settings/types.ts +79 -0
- package/src/components/utils/messageFormatting.test.tsx +331 -0
- package/src/components/utils/messageFormatting.tsx +597 -0
- package/src/index.ts +63 -0
- package/src/landing/AboutPage.tsx +77 -0
- package/src/landing/BlogContent.tsx +187 -0
- package/src/landing/BlogPage.tsx +47 -0
- package/src/landing/CareersPage.tsx +53 -0
- package/src/landing/ChangelogPage.tsx +33 -0
- package/src/landing/ContactPage.tsx +41 -0
- package/src/landing/DocsPage.tsx +43 -0
- package/src/landing/LandingPage.tsx +702 -0
- package/src/landing/PricingPage.tsx +549 -0
- package/src/landing/PrivacyPage.tsx +117 -0
- package/src/landing/SecurityPage.tsx +42 -0
- package/src/landing/StaticPage.tsx +165 -0
- package/src/landing/TermsPage.tsx +125 -0
- package/src/landing/blogData.ts +312 -0
- package/src/landing/index.ts +18 -0
- package/src/landing/styles.css +3673 -0
- package/src/lib/agent-merge.test.ts +43 -0
- package/src/lib/agent-merge.ts +35 -0
- package/src/lib/api.ts +1294 -0
- package/src/lib/cloudApi.ts +893 -0
- package/src/lib/colors.test.ts +175 -0
- package/src/lib/colors.ts +218 -0
- package/src/lib/config.ts +109 -0
- package/src/lib/hierarchy.ts +242 -0
- package/src/lib/stuckDetection.ts +142 -0
- package/src/lib/useUrlRouting.ts +190 -0
- package/src/types/index.ts +317 -0
- package/src/types/threading.ts +7 -0
- package/out/_next/static/chunks/285-dc644487a8d6500d.js +0 -1
- package/out/_next/static/css/4c58d9cf493aa626.css +0 -1
- /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
- /package/out/_next/static/{AqelRhy1vr2nBUcU0Iqcp → IxfA6RZu4trcsEMYlkQra}/_ssgManifest.js +0 -0
- /package/out/_next/static/chunks/{528-d375bc8b46912d2c.js → 528-f5f676996d613c25.js} +0 -0
- /package/out/_next/static/chunks/app/blog/let-them-cook-multi-agent-orchestration/{page-a58308f43557b908.js → page-b194f207fbd91862.js} +0 -0
|
@@ -0,0 +1,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;
|