@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.
- 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/{dYlczDQI12PIQ3tqq3N4Y → IxfA6RZu4trcsEMYlkQra}/_buildManifest.js +0 -0
- /package/out/_next/static/{dYlczDQI12PIQ3tqq3N4Y → 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,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;
|