@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,329 @@
1
+ /**
2
+ * useChannelAdmin Hook
3
+ *
4
+ * Manages channel administration: settings, members, and permissions.
5
+ * Used by ChannelAdminPanel for admin-only operations.
6
+ */
7
+
8
+ import { useState, useEffect, useCallback, useMemo } from 'react';
9
+ import { api } from '../../lib/api';
10
+
11
+ export interface ChannelMemberInfo {
12
+ id: string;
13
+ name: string;
14
+ displayName?: string;
15
+ avatarUrl?: string;
16
+ role: 'admin' | 'member';
17
+ joinedAt: string;
18
+ isAgent: boolean;
19
+ }
20
+
21
+ export interface ChannelSettings {
22
+ id: string;
23
+ name: string;
24
+ description?: string;
25
+ topic?: string;
26
+ isPrivate: boolean;
27
+ createdAt: string;
28
+ creatorId: string;
29
+ admins: string[];
30
+ }
31
+
32
+ export interface UseChannelAdminOptions {
33
+ /** Channel ID to manage */
34
+ channelId: string;
35
+ /** Current user ID for permission checks */
36
+ currentUserId?: string;
37
+ /** Page size for member pagination (default: 20) */
38
+ pageSize?: number;
39
+ /** Auto-fetch on mount (default: true) */
40
+ autoFetch?: boolean;
41
+ }
42
+
43
+ export interface UseChannelAdminReturn {
44
+ /** Channel settings */
45
+ settings: ChannelSettings | null;
46
+ /** List of members for current page */
47
+ members: ChannelMemberInfo[];
48
+ /** Loading states */
49
+ isLoadingSettings: boolean;
50
+ isLoadingMembers: boolean;
51
+ /** Error messages */
52
+ settingsError: string | null;
53
+ membersError: string | null;
54
+ /** Whether current user is admin */
55
+ isAdmin: boolean;
56
+ /** Member pagination */
57
+ memberPage: number;
58
+ memberTotalPages: number;
59
+ memberTotalCount: number;
60
+ goToMemberPage: (page: number) => void;
61
+ /** Member search */
62
+ memberSearchQuery: string;
63
+ setMemberSearchQuery: (query: string) => void;
64
+ /** Update channel settings */
65
+ updateSettings: (updates: Partial<Pick<ChannelSettings, 'description' | 'topic'>>) => Promise<void>;
66
+ /** Remove a member from channel */
67
+ removeMember: (memberId: string) => Promise<void>;
68
+ /** Assign an agent to the channel */
69
+ assignAgent: (agentName: string) => Promise<void>;
70
+ /** Promote/demote member to/from admin */
71
+ setMemberRole: (memberId: string, role: 'admin' | 'member') => Promise<void>;
72
+ /** Refresh data */
73
+ refreshSettings: () => void;
74
+ refreshMembers: () => void;
75
+ }
76
+
77
+ export function useChannelAdmin(
78
+ options: UseChannelAdminOptions
79
+ ): UseChannelAdminReturn {
80
+ const {
81
+ channelId,
82
+ currentUserId,
83
+ pageSize = 20,
84
+ autoFetch = true,
85
+ } = options;
86
+
87
+ // Settings state
88
+ const [settings, setSettings] = useState<ChannelSettings | null>(null);
89
+ const [isLoadingSettings, setIsLoadingSettings] = useState(false);
90
+ const [settingsError, setSettingsError] = useState<string | null>(null);
91
+
92
+ // Members state
93
+ const [members, setMembers] = useState<ChannelMemberInfo[]>([]);
94
+ const [isLoadingMembers, setIsLoadingMembers] = useState(false);
95
+ const [membersError, setMembersError] = useState<string | null>(null);
96
+ const [memberPage, setMemberPage] = useState(1);
97
+ const [memberTotalCount, setMemberTotalCount] = useState(0);
98
+ const [memberSearchQuery, setMemberSearchQuery] = useState('');
99
+
100
+ // Calculate admin status
101
+ const isAdmin = useMemo(() => {
102
+ if (!currentUserId || !settings) return false;
103
+ return settings.creatorId === currentUserId || settings.admins.includes(currentUserId);
104
+ }, [currentUserId, settings]);
105
+
106
+ // Calculate total pages
107
+ const memberTotalPages = useMemo(() => {
108
+ return Math.max(1, Math.ceil(memberTotalCount / pageSize));
109
+ }, [memberTotalCount, pageSize]);
110
+
111
+ // Fetch channel settings
112
+ const fetchSettings = useCallback(async () => {
113
+ setIsLoadingSettings(true);
114
+ setSettingsError(null);
115
+
116
+ try {
117
+ const result = await api.get<{
118
+ channel: ChannelSettings;
119
+ currentUserRole: 'admin' | 'member' | null;
120
+ }>(`/api/channels/${channelId}`);
121
+
122
+ if (result.channel) {
123
+ setSettings(result.channel);
124
+ }
125
+ } catch (err) {
126
+ const message = err instanceof Error ? err.message : 'Failed to fetch channel settings';
127
+ setSettingsError(message);
128
+ console.error('[useChannelAdmin] Settings fetch error:', err);
129
+ } finally {
130
+ setIsLoadingSettings(false);
131
+ }
132
+ }, [channelId]);
133
+
134
+ // Fetch channel members
135
+ const fetchMembers = useCallback(async (page: number, search: string) => {
136
+ setIsLoadingMembers(true);
137
+ setMembersError(null);
138
+
139
+ try {
140
+ const params = new URLSearchParams({
141
+ page: page.toString(),
142
+ limit: pageSize.toString(),
143
+ });
144
+
145
+ if (search.trim()) {
146
+ params.set('search', search.trim());
147
+ }
148
+
149
+ const result = await api.get<{
150
+ members: ChannelMemberInfo[];
151
+ pagination: {
152
+ page: number;
153
+ limit: number;
154
+ total: number;
155
+ totalPages: number;
156
+ };
157
+ }>(`/api/channels/${channelId}/members?${params.toString()}`);
158
+
159
+ if (result.members) {
160
+ setMembers(result.members);
161
+ setMemberTotalCount(result.pagination?.total || result.members.length);
162
+ }
163
+ } catch (err) {
164
+ const message = err instanceof Error ? err.message : 'Failed to fetch members';
165
+ setMembersError(message);
166
+ console.error('[useChannelAdmin] Members fetch error:', err);
167
+ } finally {
168
+ setIsLoadingMembers(false);
169
+ }
170
+ }, [channelId, pageSize]);
171
+
172
+ // Auto-fetch on mount or channelId change
173
+ useEffect(() => {
174
+ if (autoFetch && channelId) {
175
+ fetchSettings();
176
+ fetchMembers(1, '');
177
+ }
178
+ }, [autoFetch, channelId, fetchSettings, fetchMembers]);
179
+
180
+ // Fetch members when page or search changes
181
+ useEffect(() => {
182
+ if (autoFetch && channelId) {
183
+ fetchMembers(memberPage, memberSearchQuery);
184
+ }
185
+ }, [memberPage, memberSearchQuery, autoFetch, channelId, fetchMembers]);
186
+
187
+ // Reset to page 1 when search changes
188
+ useEffect(() => {
189
+ setMemberPage(1);
190
+ }, [memberSearchQuery]);
191
+
192
+ // Go to specific member page
193
+ const goToMemberPage = useCallback((page: number) => {
194
+ const validPage = Math.max(1, Math.min(page, memberTotalPages));
195
+ setMemberPage(validPage);
196
+ }, [memberTotalPages]);
197
+
198
+ // Update channel settings
199
+ const updateSettings = useCallback(async (updates: Partial<Pick<ChannelSettings, 'description' | 'topic'>>) => {
200
+ if (!isAdmin) {
201
+ throw new Error('Permission denied: Admin access required');
202
+ }
203
+
204
+ try {
205
+ await api.patch(`/api/channels/${channelId}`, updates);
206
+
207
+ // Optimistically update local state
208
+ setSettings((prev) => prev ? { ...prev, ...updates } : null);
209
+ } catch (err) {
210
+ const message = err instanceof Error ? err.message : 'Failed to update settings';
211
+ setSettingsError(message);
212
+ throw err;
213
+ }
214
+ }, [channelId, isAdmin]);
215
+
216
+ // Remove a member
217
+ const removeMember = useCallback(async (memberId: string) => {
218
+ if (!isAdmin) {
219
+ throw new Error('Permission denied: Admin access required');
220
+ }
221
+
222
+ try {
223
+ await api.delete(`/api/channels/${channelId}/members/${memberId}`);
224
+
225
+ // Optimistically update local state
226
+ setMembers((prev) => prev.filter((m) => m.id !== memberId));
227
+ setMemberTotalCount((prev) => Math.max(0, prev - 1));
228
+ } catch (err) {
229
+ const message = err instanceof Error ? err.message : 'Failed to remove member';
230
+ setMembersError(message);
231
+ throw err;
232
+ }
233
+ }, [channelId, isAdmin]);
234
+
235
+ // Assign an agent to the channel
236
+ const assignAgent = useCallback(async (agentName: string) => {
237
+ if (!isAdmin) {
238
+ throw new Error('Permission denied: Admin access required');
239
+ }
240
+
241
+ try {
242
+ const result = await api.post<{ member: ChannelMemberInfo }>(`/api/channels/${channelId}/agents`, {
243
+ agentName,
244
+ });
245
+
246
+ // Add to members list if returned
247
+ if (result.member) {
248
+ setMembers((prev) => [...prev, result.member]);
249
+ setMemberTotalCount((prev) => prev + 1);
250
+ }
251
+ } catch (err) {
252
+ const message = err instanceof Error ? err.message : 'Failed to assign agent';
253
+ setMembersError(message);
254
+ throw err;
255
+ }
256
+ }, [channelId, isAdmin]);
257
+
258
+ // Set member role
259
+ const setMemberRole = useCallback(async (memberId: string, role: 'admin' | 'member') => {
260
+ if (!isAdmin) {
261
+ throw new Error('Permission denied: Admin access required');
262
+ }
263
+
264
+ try {
265
+ await api.patch(`/api/channels/${channelId}/members/${memberId}`, { role });
266
+
267
+ // Optimistically update local state
268
+ setMembers((prev) =>
269
+ prev.map((m) => m.id === memberId ? { ...m, role } : m)
270
+ );
271
+
272
+ // Update admins list in settings
273
+ if (role === 'admin') {
274
+ setSettings((prev) => {
275
+ if (!prev) return null;
276
+ const member = members.find((m) => m.id === memberId);
277
+ if (member && !prev.admins.includes(member.name)) {
278
+ return { ...prev, admins: [...prev.admins, member.name] };
279
+ }
280
+ return prev;
281
+ });
282
+ } else {
283
+ setSettings((prev) => {
284
+ if (!prev) return null;
285
+ const member = members.find((m) => m.id === memberId);
286
+ if (member) {
287
+ return { ...prev, admins: prev.admins.filter((a) => a !== member.name) };
288
+ }
289
+ return prev;
290
+ });
291
+ }
292
+ } catch (err) {
293
+ const message = err instanceof Error ? err.message : 'Failed to update member role';
294
+ setMembersError(message);
295
+ throw err;
296
+ }
297
+ }, [channelId, isAdmin, members]);
298
+
299
+ // Refresh functions
300
+ const refreshSettings = useCallback(() => {
301
+ fetchSettings();
302
+ }, [fetchSettings]);
303
+
304
+ const refreshMembers = useCallback(() => {
305
+ fetchMembers(memberPage, memberSearchQuery);
306
+ }, [fetchMembers, memberPage, memberSearchQuery]);
307
+
308
+ return {
309
+ settings,
310
+ members,
311
+ isLoadingSettings,
312
+ isLoadingMembers,
313
+ settingsError,
314
+ membersError,
315
+ isAdmin,
316
+ memberPage,
317
+ memberTotalPages,
318
+ memberTotalCount,
319
+ goToMemberPage,
320
+ memberSearchQuery,
321
+ setMemberSearchQuery,
322
+ updateSettings,
323
+ removeMember,
324
+ assignAgent,
325
+ setMemberRole,
326
+ refreshSettings,
327
+ refreshMembers,
328
+ };
329
+ }
@@ -0,0 +1,239 @@
1
+ /**
2
+ * useChannelBrowser Hook
3
+ *
4
+ * Manages browsing, searching, and joining channels.
5
+ * Includes debounced search and pagination support.
6
+ */
7
+
8
+ import { useState, useEffect, useCallback, useMemo } from 'react';
9
+ import { useDebounce } from './useDebounce';
10
+ import { api } from '../../lib/api';
11
+
12
+ export interface BrowseChannel {
13
+ id: string;
14
+ name: string;
15
+ description?: string;
16
+ memberCount: number;
17
+ isJoined: boolean;
18
+ isPrivate: boolean;
19
+ createdAt: string;
20
+ }
21
+
22
+ export interface UseChannelBrowserOptions {
23
+ /** Workspace ID (required for API calls) */
24
+ workspaceId: string;
25
+ /** Initial page size (default: 20) */
26
+ pageSize?: number;
27
+ /** Search debounce delay in ms (default: 300) */
28
+ debounceDelay?: number;
29
+ /** Auto-fetch on mount (default: true) */
30
+ autoFetch?: boolean;
31
+ }
32
+
33
+ export interface UseChannelBrowserReturn {
34
+ /** List of channels for current page */
35
+ channels: BrowseChannel[];
36
+ /** Loading state */
37
+ isLoading: boolean;
38
+ /** Error message if any */
39
+ error: string | null;
40
+ /** Current search query */
41
+ searchQuery: string;
42
+ /** Update search query */
43
+ setSearchQuery: (query: string) => void;
44
+ /** Current page (1-indexed) */
45
+ currentPage: number;
46
+ /** Total number of pages */
47
+ totalPages: number;
48
+ /** Total count of channels matching search */
49
+ totalCount: number;
50
+ /** Navigate to a specific page */
51
+ goToPage: (page: number) => void;
52
+ /** Join a channel */
53
+ joinChannel: (channelId: string) => Promise<void>;
54
+ /** Leave a channel */
55
+ leaveChannel: (channelId: string) => Promise<void>;
56
+ /** Refresh the channel list */
57
+ refresh: () => void;
58
+ }
59
+
60
+ export function useChannelBrowser(
61
+ options: UseChannelBrowserOptions
62
+ ): UseChannelBrowserReturn {
63
+ const {
64
+ workspaceId,
65
+ pageSize = 20,
66
+ debounceDelay = 300,
67
+ autoFetch = true,
68
+ } = options;
69
+
70
+ const [channels, setChannels] = useState<BrowseChannel[]>([]);
71
+ const [isLoading, setIsLoading] = useState(false);
72
+ const [error, setError] = useState<string | null>(null);
73
+ const [searchQuery, setSearchQuery] = useState('');
74
+ const [currentPage, setCurrentPage] = useState(1);
75
+ const [totalCount, setTotalCount] = useState(0);
76
+
77
+ // Debounce search query
78
+ const debouncedSearchQuery = useDebounce(searchQuery, debounceDelay);
79
+
80
+ // Calculate total pages
81
+ const totalPages = useMemo(() => {
82
+ return Math.max(1, Math.ceil(totalCount / pageSize));
83
+ }, [totalCount, pageSize]);
84
+
85
+ // Fetch channels from API (workspace-scoped)
86
+ const fetchChannels = useCallback(async (page: number, search: string) => {
87
+ if (!workspaceId) {
88
+ setError('Workspace ID is required');
89
+ return;
90
+ }
91
+
92
+ setIsLoading(true);
93
+ setError(null);
94
+
95
+ try {
96
+ // Build query params
97
+ const params = new URLSearchParams({
98
+ page: page.toString(),
99
+ limit: pageSize.toString(),
100
+ });
101
+
102
+ if (search.trim()) {
103
+ params.set('search', search.trim());
104
+ }
105
+
106
+ // Use workspace-scoped endpoint
107
+ const result = await api.get<{
108
+ channels: Array<{
109
+ id: string;
110
+ name: string;
111
+ description?: string;
112
+ memberCount?: number;
113
+ isPrivate?: boolean;
114
+ createdAt: string;
115
+ // Backend may use different field names
116
+ isMember?: boolean;
117
+ }>;
118
+ archivedChannels?: unknown[];
119
+ pagination?: {
120
+ page: number;
121
+ limit: number;
122
+ total: number;
123
+ totalPages: number;
124
+ };
125
+ }>(`/api/workspaces/${workspaceId}/channels?${params.toString()}`);
126
+
127
+ if (result.channels) {
128
+ // Map backend response to BrowseChannel format
129
+ const mappedChannels: BrowseChannel[] = result.channels.map((ch) => ({
130
+ id: ch.id,
131
+ name: ch.name,
132
+ description: ch.description,
133
+ memberCount: ch.memberCount || 0,
134
+ isJoined: ch.isMember ?? false,
135
+ isPrivate: ch.isPrivate ?? false,
136
+ createdAt: ch.createdAt,
137
+ }));
138
+ setChannels(mappedChannels);
139
+ setTotalCount(result.pagination?.total || result.channels.length);
140
+ } else {
141
+ // API might return different structure - handle gracefully
142
+ setChannels([]);
143
+ setTotalCount(0);
144
+ }
145
+ } catch (err) {
146
+ const message = err instanceof Error ? err.message : 'Failed to fetch channels';
147
+ setError(message);
148
+ console.error('[useChannelBrowser] Fetch error:', err);
149
+ } finally {
150
+ setIsLoading(false);
151
+ }
152
+ }, [workspaceId, pageSize]);
153
+
154
+ // Fetch when search or page changes
155
+ useEffect(() => {
156
+ if (autoFetch) {
157
+ fetchChannels(currentPage, debouncedSearchQuery);
158
+ }
159
+ }, [currentPage, debouncedSearchQuery, autoFetch, fetchChannels]);
160
+
161
+ // Reset to page 1 when search changes
162
+ useEffect(() => {
163
+ setCurrentPage(1);
164
+ }, [debouncedSearchQuery]);
165
+
166
+ // Go to specific page
167
+ const goToPage = useCallback((page: number) => {
168
+ const validPage = Math.max(1, Math.min(page, totalPages));
169
+ setCurrentPage(validPage);
170
+ }, [totalPages]);
171
+
172
+ // Join a channel (workspace-scoped)
173
+ const joinChannel = useCallback(async (channelId: string) => {
174
+ if (!workspaceId) {
175
+ throw new Error('Workspace ID is required');
176
+ }
177
+
178
+ try {
179
+ await api.post(`/api/workspaces/${workspaceId}/channels/${encodeURIComponent(channelId)}/join`);
180
+
181
+ // Optimistically update local state
182
+ setChannels((prev) =>
183
+ prev.map((ch) =>
184
+ ch.id === channelId
185
+ ? { ...ch, isJoined: true, memberCount: ch.memberCount + 1 }
186
+ : ch
187
+ )
188
+ );
189
+ } catch (err) {
190
+ const message = err instanceof Error ? err.message : 'Failed to join channel';
191
+ setError(message);
192
+ throw err;
193
+ }
194
+ }, [workspaceId]);
195
+
196
+ // Leave a channel (workspace-scoped)
197
+ const leaveChannel = useCallback(async (channelId: string) => {
198
+ if (!workspaceId) {
199
+ throw new Error('Workspace ID is required');
200
+ }
201
+
202
+ try {
203
+ await api.post(`/api/workspaces/${workspaceId}/channels/${encodeURIComponent(channelId)}/leave`);
204
+
205
+ // Optimistically update local state
206
+ setChannels((prev) =>
207
+ prev.map((ch) =>
208
+ ch.id === channelId
209
+ ? { ...ch, isJoined: false, memberCount: Math.max(0, ch.memberCount - 1) }
210
+ : ch
211
+ )
212
+ );
213
+ } catch (err) {
214
+ const message = err instanceof Error ? err.message : 'Failed to leave channel';
215
+ setError(message);
216
+ throw err;
217
+ }
218
+ }, [workspaceId]);
219
+
220
+ // Refresh current view
221
+ const refresh = useCallback(() => {
222
+ fetchChannels(currentPage, debouncedSearchQuery);
223
+ }, [fetchChannels, currentPage, debouncedSearchQuery]);
224
+
225
+ return {
226
+ channels,
227
+ isLoading,
228
+ error,
229
+ searchQuery,
230
+ setSearchQuery,
231
+ currentPage,
232
+ totalPages,
233
+ totalCount,
234
+ goToPage,
235
+ joinChannel,
236
+ leaveChannel,
237
+ refresh,
238
+ };
239
+ }
@@ -0,0 +1,138 @@
1
+ /**
2
+ * useChannelCommands Hook
3
+ *
4
+ * Provides channel-related commands for the CommandPalette.
5
+ * Integrates /create-channel, /join-channel, /leave-channel, /channels commands.
6
+ */
7
+
8
+ import { useMemo, useCallback, createElement } from 'react';
9
+ import type { Command } from '../CommandPalette';
10
+ import { api } from '../../lib/api';
11
+
12
+ export interface ChannelInfo {
13
+ id: string;
14
+ name: string;
15
+ memberCount?: number;
16
+ }
17
+
18
+ export interface UseChannelCommandsOptions {
19
+ /** List of channels user has joined */
20
+ joinedChannels: string[];
21
+ /** Callback when user wants to browse channels */
22
+ onBrowseChannels: () => void;
23
+ /** Callback when user wants to create a channel */
24
+ onCreateChannel: () => void;
25
+ /** Callback when a channel is joined */
26
+ onChannelJoined?: (channelName: string) => void;
27
+ /** Callback when a channel is left */
28
+ onChannelLeft?: (channelName: string) => void;
29
+ }
30
+
31
+ export interface UseChannelCommandsReturn {
32
+ /** Commands to add to CommandPalette */
33
+ commands: Command[];
34
+ /** Autocomplete suggestions for channel names */
35
+ getChannelSuggestions: (query: string) => Promise<ChannelInfo[]>;
36
+ }
37
+
38
+ export function useChannelCommands(
39
+ options: UseChannelCommandsOptions
40
+ ): UseChannelCommandsReturn {
41
+ const {
42
+ joinedChannels,
43
+ onBrowseChannels,
44
+ onCreateChannel,
45
+ onChannelJoined,
46
+ onChannelLeft,
47
+ } = options;
48
+
49
+ // Get channel suggestions for autocomplete
50
+ const getChannelSuggestions = useCallback(async (query: string): Promise<ChannelInfo[]> => {
51
+ try {
52
+ const params = new URLSearchParams({
53
+ search: query,
54
+ limit: '10',
55
+ });
56
+
57
+ const result = await api.get<{
58
+ channels: Array<{ id: string; name: string; memberCount: number }>;
59
+ }>(`/api/channels/browse?${params.toString()}`);
60
+
61
+ return result.channels || [];
62
+ } catch (err) {
63
+ console.error('[useChannelCommands] Failed to get suggestions:', err);
64
+ return [];
65
+ }
66
+ }, []);
67
+
68
+ // Join channel action (reserved for future use)
69
+ const _joinChannel = useCallback(async (channelName: string) => {
70
+ try {
71
+ // Normalize channel name
72
+ const normalized = channelName.startsWith('#') ? channelName.slice(1) : channelName;
73
+
74
+ await api.post(`/api/channels/${normalized}/join`);
75
+ onChannelJoined?.(`#${normalized}`);
76
+ } catch (err) {
77
+ console.error('[useChannelCommands] Failed to join channel:', err);
78
+ }
79
+ }, [onChannelJoined]);
80
+
81
+ // Leave channel action
82
+ const leaveChannel = useCallback(async (channelName: string) => {
83
+ try {
84
+ // Normalize channel name
85
+ const normalized = channelName.startsWith('#') ? channelName.slice(1) : channelName;
86
+
87
+ await api.post(`/api/channels/${normalized}/leave`);
88
+ onChannelLeft?.(`#${normalized}`);
89
+ } catch (err) {
90
+ console.error('[useChannelCommands] Failed to leave channel:', err);
91
+ }
92
+ }, [onChannelLeft]);
93
+
94
+ // Build commands
95
+ const commands = useMemo((): Command[] => {
96
+ const channelCommands: Command[] = [
97
+ {
98
+ id: 'browse-channels',
99
+ label: 'Browse Channels',
100
+ description: 'Discover and join public channels',
101
+ category: 'channels',
102
+ icon: createElement('span', { className: 'text-sm' }, '#'),
103
+ shortcut: '/channels',
104
+ action: onBrowseChannels,
105
+ },
106
+ {
107
+ id: 'create-channel',
108
+ label: 'Create Channel',
109
+ description: 'Start a new channel',
110
+ category: 'channels',
111
+ icon: createElement('span', { className: 'text-sm' }, '+'),
112
+ shortcut: '/create-channel',
113
+ action: onCreateChannel,
114
+ },
115
+ ];
116
+
117
+ // Add leave commands for joined channels
118
+ for (const channel of joinedChannels) {
119
+ const displayName = channel.startsWith('#') ? channel : `#${channel}`;
120
+ channelCommands.push({
121
+ id: `leave-${channel}`,
122
+ label: `Leave ${displayName}`,
123
+ description: 'Leave this channel',
124
+ category: 'channels',
125
+ icon: createElement('span', { className: 'text-sm' }, '⊗'),
126
+ action: () => leaveChannel(channel),
127
+ });
128
+ }
129
+
130
+ return channelCommands;
131
+ }, [joinedChannels, onBrowseChannels, onCreateChannel, leaveChannel]);
132
+
133
+ return {
134
+ commands,
135
+ getChannelSuggestions,
136
+ };
137
+ }
138
+