@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,302 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* MessageInput Component
|
|
3
|
+
*
|
|
4
|
+
* Rich text input for sending messages with:
|
|
5
|
+
* - @-mention autocomplete
|
|
6
|
+
* - Multi-line support (Shift+Enter)
|
|
7
|
+
* - Typing indicator
|
|
8
|
+
* - File attachment button (UI only)
|
|
9
|
+
*/
|
|
10
|
+
|
|
11
|
+
import React, { useState, useCallback, useRef, useEffect, useMemo } from 'react';
|
|
12
|
+
import type { MessageInputProps } from './types';
|
|
13
|
+
|
|
14
|
+
const TYPING_DEBOUNCE_MS = 1000;
|
|
15
|
+
|
|
16
|
+
export function MessageInput({
|
|
17
|
+
channelId,
|
|
18
|
+
placeholder = 'Send a message...',
|
|
19
|
+
disabled = false,
|
|
20
|
+
onSend,
|
|
21
|
+
onTyping,
|
|
22
|
+
mentionSuggestions = [],
|
|
23
|
+
}: MessageInputProps) {
|
|
24
|
+
const [value, setValue] = useState('');
|
|
25
|
+
const [showMentions, setShowMentions] = useState(false);
|
|
26
|
+
const [mentionQuery, setMentionQuery] = useState('');
|
|
27
|
+
const [selectedMentionIndex, setSelectedMentionIndex] = useState(0);
|
|
28
|
+
const [cursorPosition, setCursorPosition] = useState(0);
|
|
29
|
+
|
|
30
|
+
const textareaRef = useRef<HTMLTextAreaElement>(null);
|
|
31
|
+
const typingTimeoutRef = useRef<NodeJS.Timeout | null>(null);
|
|
32
|
+
const wasTypingRef = useRef(false);
|
|
33
|
+
|
|
34
|
+
// Filter mention suggestions based on query
|
|
35
|
+
const filteredMentions = useMemo(() => {
|
|
36
|
+
if (!mentionQuery) return mentionSuggestions.slice(0, 5);
|
|
37
|
+
const query = mentionQuery.toLowerCase();
|
|
38
|
+
return mentionSuggestions
|
|
39
|
+
.filter(name => name.toLowerCase().includes(query))
|
|
40
|
+
.slice(0, 5);
|
|
41
|
+
}, [mentionSuggestions, mentionQuery]);
|
|
42
|
+
|
|
43
|
+
// Handle typing indicator
|
|
44
|
+
const handleTyping = useCallback((isTyping: boolean) => {
|
|
45
|
+
if (!onTyping) return;
|
|
46
|
+
|
|
47
|
+
if (isTyping && !wasTypingRef.current) {
|
|
48
|
+
wasTypingRef.current = true;
|
|
49
|
+
onTyping(true);
|
|
50
|
+
}
|
|
51
|
+
|
|
52
|
+
if (typingTimeoutRef.current) {
|
|
53
|
+
clearTimeout(typingTimeoutRef.current);
|
|
54
|
+
}
|
|
55
|
+
|
|
56
|
+
if (isTyping) {
|
|
57
|
+
typingTimeoutRef.current = setTimeout(() => {
|
|
58
|
+
wasTypingRef.current = false;
|
|
59
|
+
onTyping(false);
|
|
60
|
+
}, TYPING_DEBOUNCE_MS);
|
|
61
|
+
} else {
|
|
62
|
+
wasTypingRef.current = false;
|
|
63
|
+
onTyping(false);
|
|
64
|
+
}
|
|
65
|
+
}, [onTyping]);
|
|
66
|
+
|
|
67
|
+
// Clean up typing indicator on unmount
|
|
68
|
+
useEffect(() => {
|
|
69
|
+
return () => {
|
|
70
|
+
if (typingTimeoutRef.current) {
|
|
71
|
+
clearTimeout(typingTimeoutRef.current);
|
|
72
|
+
}
|
|
73
|
+
if (wasTypingRef.current && onTyping) {
|
|
74
|
+
onTyping(false);
|
|
75
|
+
}
|
|
76
|
+
};
|
|
77
|
+
}, [onTyping]);
|
|
78
|
+
|
|
79
|
+
// Handle value change
|
|
80
|
+
const handleChange = useCallback((e: React.ChangeEvent<HTMLTextAreaElement>) => {
|
|
81
|
+
const newValue = e.target.value;
|
|
82
|
+
const newPosition = e.target.selectionStart;
|
|
83
|
+
|
|
84
|
+
setValue(newValue);
|
|
85
|
+
setCursorPosition(newPosition);
|
|
86
|
+
handleTyping(newValue.length > 0);
|
|
87
|
+
|
|
88
|
+
// Check for mention trigger
|
|
89
|
+
const textBeforeCursor = newValue.slice(0, newPosition);
|
|
90
|
+
const mentionMatch = textBeforeCursor.match(/@(\w*)$/);
|
|
91
|
+
|
|
92
|
+
if (mentionMatch) {
|
|
93
|
+
setMentionQuery(mentionMatch[1]);
|
|
94
|
+
setShowMentions(true);
|
|
95
|
+
setSelectedMentionIndex(0);
|
|
96
|
+
} else {
|
|
97
|
+
setShowMentions(false);
|
|
98
|
+
setMentionQuery('');
|
|
99
|
+
}
|
|
100
|
+
|
|
101
|
+
// Auto-resize textarea
|
|
102
|
+
const textarea = textareaRef.current;
|
|
103
|
+
if (textarea) {
|
|
104
|
+
textarea.style.height = 'auto';
|
|
105
|
+
textarea.style.height = `${Math.min(textarea.scrollHeight, 200)}px`;
|
|
106
|
+
}
|
|
107
|
+
}, [handleTyping]);
|
|
108
|
+
|
|
109
|
+
// Insert mention at cursor
|
|
110
|
+
const insertMention = useCallback((name: string) => {
|
|
111
|
+
const textBeforeCursor = value.slice(0, cursorPosition);
|
|
112
|
+
const textAfterCursor = value.slice(cursorPosition);
|
|
113
|
+
|
|
114
|
+
// Find the @ trigger position
|
|
115
|
+
const mentionMatch = textBeforeCursor.match(/@(\w*)$/);
|
|
116
|
+
if (!mentionMatch) return;
|
|
117
|
+
|
|
118
|
+
const beforeMention = textBeforeCursor.slice(0, -mentionMatch[0].length);
|
|
119
|
+
const newValue = `${beforeMention}@${name} ${textAfterCursor}`;
|
|
120
|
+
|
|
121
|
+
setValue(newValue);
|
|
122
|
+
setShowMentions(false);
|
|
123
|
+
setMentionQuery('');
|
|
124
|
+
|
|
125
|
+
// Focus and set cursor position
|
|
126
|
+
setTimeout(() => {
|
|
127
|
+
if (textareaRef.current) {
|
|
128
|
+
textareaRef.current.focus();
|
|
129
|
+
const newPosition = beforeMention.length + name.length + 2; // @ + name + space
|
|
130
|
+
textareaRef.current.setSelectionRange(newPosition, newPosition);
|
|
131
|
+
setCursorPosition(newPosition);
|
|
132
|
+
}
|
|
133
|
+
}, 0);
|
|
134
|
+
}, [value, cursorPosition]);
|
|
135
|
+
|
|
136
|
+
// Handle keyboard navigation in mention list
|
|
137
|
+
const handleKeyDown = useCallback((e: React.KeyboardEvent) => {
|
|
138
|
+
if (showMentions && filteredMentions.length > 0) {
|
|
139
|
+
switch (e.key) {
|
|
140
|
+
case 'ArrowDown':
|
|
141
|
+
e.preventDefault();
|
|
142
|
+
setSelectedMentionIndex(prev =>
|
|
143
|
+
prev < filteredMentions.length - 1 ? prev + 1 : 0
|
|
144
|
+
);
|
|
145
|
+
return;
|
|
146
|
+
case 'ArrowUp':
|
|
147
|
+
e.preventDefault();
|
|
148
|
+
setSelectedMentionIndex(prev =>
|
|
149
|
+
prev > 0 ? prev - 1 : filteredMentions.length - 1
|
|
150
|
+
);
|
|
151
|
+
return;
|
|
152
|
+
case 'Tab':
|
|
153
|
+
case 'Enter':
|
|
154
|
+
e.preventDefault();
|
|
155
|
+
insertMention(filteredMentions[selectedMentionIndex]);
|
|
156
|
+
return;
|
|
157
|
+
case 'Escape':
|
|
158
|
+
e.preventDefault();
|
|
159
|
+
setShowMentions(false);
|
|
160
|
+
return;
|
|
161
|
+
}
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
// Option/Alt+Enter: insert newline manually (browser doesn't do this by default)
|
|
165
|
+
if (e.key === 'Enter' && e.altKey) {
|
|
166
|
+
e.preventDefault();
|
|
167
|
+
const textarea = e.target as HTMLTextAreaElement;
|
|
168
|
+
const start = textarea.selectionStart;
|
|
169
|
+
const end = textarea.selectionEnd;
|
|
170
|
+
const newValue = value.slice(0, start) + '\n' + value.slice(end);
|
|
171
|
+
setValue(newValue);
|
|
172
|
+
setTimeout(() => {
|
|
173
|
+
textarea.selectionStart = textarea.selectionEnd = start + 1;
|
|
174
|
+
}, 0);
|
|
175
|
+
return;
|
|
176
|
+
}
|
|
177
|
+
|
|
178
|
+
// Send on Enter (without Shift or Option/Alt)
|
|
179
|
+
if (e.key === 'Enter' && !e.shiftKey && !e.altKey) {
|
|
180
|
+
e.preventDefault();
|
|
181
|
+
handleSend();
|
|
182
|
+
}
|
|
183
|
+
}, [showMentions, filteredMentions, selectedMentionIndex, insertMention, value]);
|
|
184
|
+
|
|
185
|
+
// Handle send
|
|
186
|
+
const handleSend = useCallback(() => {
|
|
187
|
+
const trimmed = value.trim();
|
|
188
|
+
if (!trimmed || disabled) return;
|
|
189
|
+
|
|
190
|
+
onSend(trimmed);
|
|
191
|
+
setValue('');
|
|
192
|
+
handleTyping(false);
|
|
193
|
+
|
|
194
|
+
// Reset textarea height
|
|
195
|
+
if (textareaRef.current) {
|
|
196
|
+
textareaRef.current.style.height = 'auto';
|
|
197
|
+
}
|
|
198
|
+
}, [value, disabled, onSend, handleTyping]);
|
|
199
|
+
|
|
200
|
+
return (
|
|
201
|
+
<div className="relative flex-shrink-0 border-t border-border-subtle bg-bg-primary">
|
|
202
|
+
{/* Mention autocomplete */}
|
|
203
|
+
{showMentions && filteredMentions.length > 0 && (
|
|
204
|
+
<div className="absolute bottom-full left-4 mb-1 bg-bg-elevated border border-border-subtle rounded-lg shadow-lg py-1 min-w-[200px] max-h-[200px] overflow-y-auto">
|
|
205
|
+
{filteredMentions.map((name, index) => (
|
|
206
|
+
<button
|
|
207
|
+
key={name}
|
|
208
|
+
onClick={() => insertMention(name)}
|
|
209
|
+
className={`
|
|
210
|
+
w-full flex items-center gap-2 px-3 py-1.5 text-sm text-left transition-colors
|
|
211
|
+
${index === selectedMentionIndex
|
|
212
|
+
? 'bg-accent-cyan/10 text-text-primary'
|
|
213
|
+
: 'text-text-secondary hover:bg-bg-hover'}
|
|
214
|
+
`}
|
|
215
|
+
>
|
|
216
|
+
<div className="w-6 h-6 rounded-full bg-accent-cyan/20 flex items-center justify-center text-xs font-medium text-accent-cyan">
|
|
217
|
+
{name.charAt(0).toUpperCase()}
|
|
218
|
+
</div>
|
|
219
|
+
<span>{name}</span>
|
|
220
|
+
</button>
|
|
221
|
+
))}
|
|
222
|
+
</div>
|
|
223
|
+
)}
|
|
224
|
+
|
|
225
|
+
{/* Input area */}
|
|
226
|
+
<div className="p-4">
|
|
227
|
+
<div className="flex items-end gap-3">
|
|
228
|
+
{/* Attachment button */}
|
|
229
|
+
<button
|
|
230
|
+
type="button"
|
|
231
|
+
disabled={disabled}
|
|
232
|
+
className="p-2 rounded-lg text-text-muted hover:text-text-primary hover:bg-bg-hover transition-colors disabled:opacity-50 disabled:cursor-not-allowed flex-shrink-0"
|
|
233
|
+
title="Attach file"
|
|
234
|
+
>
|
|
235
|
+
<AttachIcon className="w-5 h-5" />
|
|
236
|
+
</button>
|
|
237
|
+
|
|
238
|
+
{/* Text input */}
|
|
239
|
+
<div className="flex-1 relative">
|
|
240
|
+
<textarea
|
|
241
|
+
ref={textareaRef}
|
|
242
|
+
value={value}
|
|
243
|
+
onChange={handleChange}
|
|
244
|
+
onKeyDown={handleKeyDown}
|
|
245
|
+
placeholder={placeholder}
|
|
246
|
+
disabled={disabled}
|
|
247
|
+
rows={1}
|
|
248
|
+
className="w-full px-4 py-2.5 bg-bg-tertiary border border-border-subtle rounded-lg text-text-primary text-sm resize-none focus:outline-none focus:border-accent-cyan/50 disabled:opacity-50 disabled:cursor-not-allowed placeholder:text-text-muted"
|
|
249
|
+
style={{ maxHeight: '200px' }}
|
|
250
|
+
/>
|
|
251
|
+
</div>
|
|
252
|
+
|
|
253
|
+
{/* Send button */}
|
|
254
|
+
<button
|
|
255
|
+
type="button"
|
|
256
|
+
onClick={handleSend}
|
|
257
|
+
disabled={!value.trim() || disabled}
|
|
258
|
+
className={`
|
|
259
|
+
p-2.5 rounded-lg transition-colors flex-shrink-0
|
|
260
|
+
${value.trim() && !disabled
|
|
261
|
+
? 'bg-accent-cyan text-bg-deep hover:bg-accent-cyan/90'
|
|
262
|
+
: 'bg-bg-tertiary text-text-muted cursor-not-allowed'}
|
|
263
|
+
`}
|
|
264
|
+
title="Send message"
|
|
265
|
+
>
|
|
266
|
+
<SendIcon className="w-5 h-5" />
|
|
267
|
+
</button>
|
|
268
|
+
</div>
|
|
269
|
+
|
|
270
|
+
{/* Helper text */}
|
|
271
|
+
<p className="mt-2 text-xs text-text-muted">
|
|
272
|
+
<kbd className="px-1 py-0.5 bg-bg-tertiary rounded text-[10px]">Enter</kbd> to send,{' '}
|
|
273
|
+
<kbd className="px-1 py-0.5 bg-bg-tertiary rounded text-[10px]">Shift/Option+Enter</kbd> for new line,{' '}
|
|
274
|
+
<kbd className="px-1 py-0.5 bg-bg-tertiary rounded text-[10px]">@</kbd> to mention
|
|
275
|
+
</p>
|
|
276
|
+
</div>
|
|
277
|
+
</div>
|
|
278
|
+
);
|
|
279
|
+
}
|
|
280
|
+
|
|
281
|
+
// =============================================================================
|
|
282
|
+
// Icons
|
|
283
|
+
// =============================================================================
|
|
284
|
+
|
|
285
|
+
function AttachIcon({ className }: { className?: string }) {
|
|
286
|
+
return (
|
|
287
|
+
<svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
|
288
|
+
<path d="M21.44 11.05l-9.19 9.19a6 6 0 0 1-8.49-8.49l9.19-9.19a4 4 0 0 1 5.66 5.66l-9.2 9.19a2 2 0 0 1-2.83-2.83l8.49-8.48" />
|
|
289
|
+
</svg>
|
|
290
|
+
);
|
|
291
|
+
}
|
|
292
|
+
|
|
293
|
+
function SendIcon({ className }: { className?: string }) {
|
|
294
|
+
return (
|
|
295
|
+
<svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
|
296
|
+
<line x1="22" y1="2" x2="11" y2="13" />
|
|
297
|
+
<polygon points="22 2 15 22 11 13 2 9 22 2" />
|
|
298
|
+
</svg>
|
|
299
|
+
);
|
|
300
|
+
}
|
|
301
|
+
|
|
302
|
+
export default MessageInput;
|
|
@@ -0,0 +1,172 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* SearchInput Component
|
|
3
|
+
*
|
|
4
|
+
* Search query input with debounce for channel message search.
|
|
5
|
+
* Supports workspace-wide and channel-scoped search.
|
|
6
|
+
*/
|
|
7
|
+
|
|
8
|
+
import React, { useState, useCallback, useRef, useEffect } from 'react';
|
|
9
|
+
import type { SearchInputProps } from './types';
|
|
10
|
+
|
|
11
|
+
const DEFAULT_DEBOUNCE_MS = 300;
|
|
12
|
+
|
|
13
|
+
export function SearchInput({
|
|
14
|
+
initialQuery = '',
|
|
15
|
+
placeholder = 'Search messages...',
|
|
16
|
+
debounceMs = DEFAULT_DEBOUNCE_MS,
|
|
17
|
+
isSearching = false,
|
|
18
|
+
onSearch,
|
|
19
|
+
onClear,
|
|
20
|
+
channelId,
|
|
21
|
+
}: SearchInputProps) {
|
|
22
|
+
const [value, setValue] = useState(initialQuery);
|
|
23
|
+
const debounceRef = useRef<NodeJS.Timeout | null>(null);
|
|
24
|
+
const inputRef = useRef<HTMLInputElement>(null);
|
|
25
|
+
|
|
26
|
+
// Debounced search handler
|
|
27
|
+
const debouncedSearch = useCallback((query: string) => {
|
|
28
|
+
if (debounceRef.current) {
|
|
29
|
+
clearTimeout(debounceRef.current);
|
|
30
|
+
}
|
|
31
|
+
|
|
32
|
+
debounceRef.current = setTimeout(() => {
|
|
33
|
+
onSearch(query);
|
|
34
|
+
}, debounceMs);
|
|
35
|
+
}, [onSearch, debounceMs]);
|
|
36
|
+
|
|
37
|
+
// Handle input change
|
|
38
|
+
const handleChange = useCallback((e: React.ChangeEvent<HTMLInputElement>) => {
|
|
39
|
+
const newValue = e.target.value;
|
|
40
|
+
setValue(newValue);
|
|
41
|
+
|
|
42
|
+
if (newValue.trim()) {
|
|
43
|
+
debouncedSearch(newValue.trim());
|
|
44
|
+
} else {
|
|
45
|
+
// Clear immediately when empty
|
|
46
|
+
if (debounceRef.current) {
|
|
47
|
+
clearTimeout(debounceRef.current);
|
|
48
|
+
}
|
|
49
|
+
onClear?.();
|
|
50
|
+
}
|
|
51
|
+
}, [debouncedSearch, onClear]);
|
|
52
|
+
|
|
53
|
+
// Handle clear button
|
|
54
|
+
const handleClear = useCallback(() => {
|
|
55
|
+
setValue('');
|
|
56
|
+
if (debounceRef.current) {
|
|
57
|
+
clearTimeout(debounceRef.current);
|
|
58
|
+
}
|
|
59
|
+
onClear?.();
|
|
60
|
+
inputRef.current?.focus();
|
|
61
|
+
}, [onClear]);
|
|
62
|
+
|
|
63
|
+
// Handle keyboard shortcuts
|
|
64
|
+
const handleKeyDown = useCallback((e: React.KeyboardEvent) => {
|
|
65
|
+
if (e.key === 'Escape') {
|
|
66
|
+
handleClear();
|
|
67
|
+
} else if (e.key === 'Enter') {
|
|
68
|
+
// Trigger search immediately on Enter
|
|
69
|
+
if (debounceRef.current) {
|
|
70
|
+
clearTimeout(debounceRef.current);
|
|
71
|
+
}
|
|
72
|
+
if (value.trim()) {
|
|
73
|
+
onSearch(value.trim());
|
|
74
|
+
}
|
|
75
|
+
}
|
|
76
|
+
}, [handleClear, onSearch, value]);
|
|
77
|
+
|
|
78
|
+
// Cleanup on unmount
|
|
79
|
+
useEffect(() => {
|
|
80
|
+
return () => {
|
|
81
|
+
if (debounceRef.current) {
|
|
82
|
+
clearTimeout(debounceRef.current);
|
|
83
|
+
}
|
|
84
|
+
};
|
|
85
|
+
}, []);
|
|
86
|
+
|
|
87
|
+
return (
|
|
88
|
+
<div className="relative">
|
|
89
|
+
{/* Search icon */}
|
|
90
|
+
<div className="absolute left-3 top-1/2 -translate-y-1/2 pointer-events-none">
|
|
91
|
+
<SearchIcon className="w-4 h-4 text-text-muted" />
|
|
92
|
+
</div>
|
|
93
|
+
|
|
94
|
+
{/* Input field */}
|
|
95
|
+
<input
|
|
96
|
+
ref={inputRef}
|
|
97
|
+
type="text"
|
|
98
|
+
value={value}
|
|
99
|
+
onChange={handleChange}
|
|
100
|
+
onKeyDown={handleKeyDown}
|
|
101
|
+
placeholder={placeholder}
|
|
102
|
+
className="w-full pl-10 pr-10 py-2 bg-bg-tertiary border border-border-subtle rounded-lg text-text-primary text-sm focus:outline-none focus:border-accent-cyan/50 placeholder:text-text-muted"
|
|
103
|
+
/>
|
|
104
|
+
|
|
105
|
+
{/* Clear/Loading indicator */}
|
|
106
|
+
<div className="absolute right-3 top-1/2 -translate-y-1/2">
|
|
107
|
+
{isSearching ? (
|
|
108
|
+
<LoadingSpinner className="w-4 h-4 text-accent-cyan" />
|
|
109
|
+
) : value ? (
|
|
110
|
+
<button
|
|
111
|
+
onClick={handleClear}
|
|
112
|
+
className="p-0.5 rounded text-text-muted hover:text-text-primary transition-colors"
|
|
113
|
+
title="Clear search"
|
|
114
|
+
>
|
|
115
|
+
<CloseIcon className="w-4 h-4" />
|
|
116
|
+
</button>
|
|
117
|
+
) : null}
|
|
118
|
+
</div>
|
|
119
|
+
|
|
120
|
+
{/* Channel scope indicator */}
|
|
121
|
+
{channelId && (
|
|
122
|
+
<div className="absolute -top-6 left-0 text-xs text-text-muted">
|
|
123
|
+
Searching in #{channelId.replace('#', '')}
|
|
124
|
+
</div>
|
|
125
|
+
)}
|
|
126
|
+
</div>
|
|
127
|
+
);
|
|
128
|
+
}
|
|
129
|
+
|
|
130
|
+
// =============================================================================
|
|
131
|
+
// Icons
|
|
132
|
+
// =============================================================================
|
|
133
|
+
|
|
134
|
+
function SearchIcon({ className }: { className?: string }) {
|
|
135
|
+
return (
|
|
136
|
+
<svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
|
137
|
+
<circle cx="11" cy="11" r="8" />
|
|
138
|
+
<line x1="21" y1="21" x2="16.65" y2="16.65" />
|
|
139
|
+
</svg>
|
|
140
|
+
);
|
|
141
|
+
}
|
|
142
|
+
|
|
143
|
+
function CloseIcon({ className }: { className?: string }) {
|
|
144
|
+
return (
|
|
145
|
+
<svg className={className} viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2" strokeLinecap="round" strokeLinejoin="round">
|
|
146
|
+
<line x1="18" y1="6" x2="6" y2="18" />
|
|
147
|
+
<line x1="6" y1="6" x2="18" y2="18" />
|
|
148
|
+
</svg>
|
|
149
|
+
);
|
|
150
|
+
}
|
|
151
|
+
|
|
152
|
+
function LoadingSpinner({ className }: { className?: string }) {
|
|
153
|
+
return (
|
|
154
|
+
<svg className={`${className} animate-spin`} viewBox="0 0 24 24" fill="none">
|
|
155
|
+
<circle
|
|
156
|
+
className="opacity-25"
|
|
157
|
+
cx="12"
|
|
158
|
+
cy="12"
|
|
159
|
+
r="10"
|
|
160
|
+
stroke="currentColor"
|
|
161
|
+
strokeWidth="4"
|
|
162
|
+
/>
|
|
163
|
+
<path
|
|
164
|
+
className="opacity-75"
|
|
165
|
+
fill="currentColor"
|
|
166
|
+
d="M4 12a8 8 0 018-8V0C5.373 0 0 5.373 0 12h4zm2 5.291A7.962 7.962 0 014 12H0c0 3.042 1.135 5.824 3 7.938l3-2.647z"
|
|
167
|
+
/>
|
|
168
|
+
</svg>
|
|
169
|
+
);
|
|
170
|
+
}
|
|
171
|
+
|
|
172
|
+
export default SearchInput;
|