@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,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
|
+
|