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