@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,164 @@
1
+ /**
2
+ * DirectMessageView Component
3
+ *
4
+ * Handles direct message conversations with humans and optional agent participants.
5
+ * Manages agent invitations, message deduplication, and group DM functionality.
6
+ */
7
+
8
+ import React, { useMemo, useCallback } from 'react';
9
+ import type { Agent, Message } from '../types';
10
+
11
+ export interface DirectMessageViewProps {
12
+ /** The human user being DM'd */
13
+ currentHuman: { name: string; isHuman: boolean } | null;
14
+ /** All messages */
15
+ messages: Message[];
16
+ /** All agents */
17
+ agents: Agent[];
18
+ /** Currently selected agents for this DM */
19
+ selectedAgents: string[];
20
+ /** Agents removed from this DM */
21
+ removedAgents: string[];
22
+ /** Callback to toggle agent participation */
23
+ onAgentToggle: (agentName: string) => void;
24
+ /** Children to render (message list, composer, etc.) */
25
+ children: (props: {
26
+ visibleMessages: Message[];
27
+ participantAgents: string[];
28
+ }) => React.ReactNode;
29
+ }
30
+
31
+ export function DirectMessageView({
32
+ currentHuman,
33
+ messages,
34
+ agents,
35
+ selectedAgents,
36
+ removedAgents,
37
+ onAgentToggle,
38
+ children,
39
+ }: DirectMessageViewProps) {
40
+ const agentNameSet = useMemo(() => new Set(agents.map((a) => a.name)), [agents]);
41
+
42
+ // Derive agents participating in this conversation from message history
43
+ const dmParticipantAgents = useMemo(() => {
44
+ if (!currentHuman) return [];
45
+ const humanName = currentHuman.name;
46
+ const derived = new Set<string>();
47
+
48
+ for (const msg of messages) {
49
+ const { from, to } = msg;
50
+ if (!from || !to) continue;
51
+ if (from === humanName && agentNameSet.has(to)) derived.add(to);
52
+ if (to === humanName && agentNameSet.has(from)) derived.add(from);
53
+ if (selectedAgents.includes(from) && agentNameSet.has(to)) derived.add(to);
54
+ if (selectedAgents.includes(to) && agentNameSet.has(from)) derived.add(from);
55
+ }
56
+
57
+ const participants = new Set<string>([...selectedAgents, ...derived]);
58
+ removedAgents.forEach((a) => participants.delete(a));
59
+ return Array.from(participants);
60
+ }, [agentNameSet, currentHuman, messages, removedAgents, selectedAgents]);
61
+
62
+ // Filter messages for this DM conversation
63
+ const visibleMessages = useMemo(() => {
64
+ if (!currentHuman) return messages;
65
+ const participants = new Set<string>([currentHuman.name, ...dmParticipantAgents]);
66
+ return messages.filter(
67
+ (msg) => msg.from && msg.to && participants.has(msg.from) && participants.has(msg.to)
68
+ );
69
+ }, [currentHuman, dmParticipantAgents, messages]);
70
+
71
+ // Deduplicate DM messages (merge duplicates sent to multiple participants)
72
+ const dedupedVisibleMessages = useMemo(() => {
73
+ if (!currentHuman) return visibleMessages;
74
+
75
+ const normalizeBody = (content?: string) => (content ?? '').trim().replace(/\s+/g, ' ');
76
+ const rank = (msg: Message) => (msg.status === 'sending' ? 1 : 0);
77
+ const choose = (current: Message, incoming: Message) => {
78
+ const currentRank = rank(current);
79
+ const incomingRank = rank(incoming);
80
+ const currentTs = new Date(current.timestamp).getTime();
81
+ const incomingTs = new Date(incoming.timestamp).getTime();
82
+ if (incomingRank < currentRank) return incoming;
83
+ if (incomingRank > currentRank) return current;
84
+ return incomingTs >= currentTs ? incoming : current;
85
+ };
86
+
87
+ const sorted = [...visibleMessages].sort(
88
+ (a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime()
89
+ );
90
+
91
+ const byId = new Map<string, Message>();
92
+ const byFuzzy = new Map<string, Message>();
93
+
94
+ for (const msg of sorted) {
95
+ if (msg.id) {
96
+ const existing = byId.get(msg.id);
97
+ byId.set(msg.id, existing ? choose(existing, msg) : msg);
98
+ continue;
99
+ }
100
+
101
+ const sender = msg.from?.toLowerCase() ?? '';
102
+ const bucket = Math.floor(new Date(msg.timestamp).getTime() / 5000);
103
+ const key = `${sender}|${bucket}|${normalizeBody(msg.content)}`;
104
+ const existing = byFuzzy.get(key);
105
+ byFuzzy.set(key, existing ? choose(existing, msg) : msg);
106
+ }
107
+
108
+ const merged = [...byId.values(), ...byFuzzy.values()];
109
+
110
+ // Final pass: deduplicate by sender + recipient + content (no time bucket)
111
+ const finalDedup = new Map<string, Message>();
112
+ for (const msg of merged) {
113
+ const sender = msg.from?.toLowerCase() ?? '';
114
+ const recipient = msg.to?.toLowerCase() ?? '';
115
+ const key = `${sender}|${recipient}|${normalizeBody(msg.content)}`;
116
+ const existing = finalDedup.get(key);
117
+ finalDedup.set(key, existing ? choose(existing, msg) : msg);
118
+ }
119
+
120
+ return Array.from(finalDedup.values()).sort(
121
+ (a, b) => new Date(a.timestamp).getTime() - new Date(b.timestamp).getTime()
122
+ );
123
+ }, [currentHuman, visibleMessages]);
124
+
125
+ if (!currentHuman) return null;
126
+
127
+ return (
128
+ <>
129
+ {/* DM Header with Agent Invites */}
130
+ <div className="px-4 py-2 border-b border-border-subtle bg-bg-secondary flex flex-col gap-2 sticky top-0 z-10">
131
+ <div className="text-xs text-text-muted">
132
+ DM with <span className="font-semibold text-text-primary">{currentHuman.name}</span>. Invite agents:
133
+ </div>
134
+ <div className="flex flex-wrap gap-2">
135
+ {agents
136
+ .filter((a) => !a.isHuman)
137
+ .map((agent) => {
138
+ const isSelected = selectedAgents.includes(agent.name);
139
+ return (
140
+ <button
141
+ key={agent.name}
142
+ onClick={() => onAgentToggle(agent.name)}
143
+ className={`px-3 py-1.5 text-xs font-medium rounded-lg transition-colors ${
144
+ isSelected
145
+ ? 'bg-accent-cyan text-bg-deep'
146
+ : 'bg-bg-tertiary text-text-secondary hover:bg-bg-tertiary/80'
147
+ }`}
148
+ title={agent.name}
149
+ >
150
+ {isSelected ? '✓ ' : ''}{agent.name}
151
+ </button>
152
+ );
153
+ })}
154
+ </div>
155
+ </div>
156
+
157
+ {/* Render children with deduped messages */}
158
+ {children({
159
+ visibleMessages: dedupedVisibleMessages,
160
+ participantAgents: dmParticipantAgents,
161
+ })}
162
+ </>
163
+ );
164
+ }
@@ -0,0 +1,368 @@
1
+ /**
2
+ * FileAutocomplete Component
3
+ *
4
+ * Provides @-file autocomplete for the message composer.
5
+ * Shows a dropdown list of files when typing @ followed by a path pattern.
6
+ * Triggered by @path/to/file or @filename patterns containing / or .
7
+ */
8
+
9
+ import React, { useState, useCallback, useEffect, useRef, useMemo } from 'react';
10
+ import { getApiUrl } from '../lib/api';
11
+
12
+ export interface FileAutocompleteProps {
13
+ /** Current input value */
14
+ inputValue: string;
15
+ /** Cursor position in input */
16
+ cursorPosition: number;
17
+ /** Called when a file is selected */
18
+ onSelect: (filePath: string, newValue: string) => void;
19
+ /** Called when autocomplete should be hidden */
20
+ onClose: () => void;
21
+ /** Whether the autocomplete is visible */
22
+ isVisible: boolean;
23
+ /** API base URL for fetching files */
24
+ apiBase?: string;
25
+ }
26
+
27
+ interface FileOption {
28
+ path: string;
29
+ name: string;
30
+ isDirectory: boolean;
31
+ }
32
+
33
+ /**
34
+ * Check if the input has an @-file path being typed at the cursor position.
35
+ * Returns the query if it looks like a file path.
36
+ * Works for @ at any position in the text, not just the start.
37
+ *
38
+ * Trigger conditions (to avoid conflict with agent mentions):
39
+ * - Contains `/` (path separator) - e.g., @src/components
40
+ * - Contains `.` with more characters after (file extension) - e.g., @package.json
41
+ * - Starts with `./` or `../` (relative path) - e.g., @./src
42
+ *
43
+ * Does NOT trigger for:
44
+ * - Simple names like @Alice (could be an agent name)
45
+ * - Names with trailing dot like @config. (user still typing)
46
+ */
47
+ export function getFileQuery(value: string, cursorPos: number): string | null {
48
+ // Search backwards from cursor to find @
49
+ const textBeforeCursor = value.substring(0, cursorPos);
50
+
51
+ // Find the last @ before cursor that starts a mention
52
+ // A mention starts after whitespace, at start of string, or after certain punctuation
53
+ const mentionMatch = textBeforeCursor.match(/(?:^|[\s(])@(\S*)$/);
54
+ if (mentionMatch) {
55
+ const query = mentionMatch[1];
56
+
57
+ // Trigger file autocomplete only for unambiguous file patterns:
58
+ // 1. Contains path separator: @src/components, @./file
59
+ if (query.includes('/')) {
60
+ return query;
61
+ }
62
+
63
+ // 2. Has file extension (dot followed by 1-10 chars): @file.ts, @package.json
64
+ // But not just a trailing dot (user still typing)
65
+ if (/\.[a-zA-Z0-9]{1,10}$/.test(query)) {
66
+ return query;
67
+ }
68
+ }
69
+ return null;
70
+ }
71
+
72
+ /**
73
+ * Complete a file path in the input value at the cursor position.
74
+ */
75
+ export function completeFileInValue(
76
+ value: string,
77
+ filePath: string,
78
+ cursorPos: number
79
+ ): string {
80
+ const textBeforeCursor = value.substring(0, cursorPos);
81
+ const textAfterCursor = value.substring(cursorPos);
82
+
83
+ // Find the @ and partial text before cursor
84
+ const mentionMatch = textBeforeCursor.match(/(?:^|[\s(])@(\S*)$/);
85
+ if (mentionMatch) {
86
+ // Calculate where the @ starts (accounting for whitespace/punctuation before it)
87
+ const matchStart = mentionMatch.index || 0;
88
+ const prefixChar = mentionMatch[0].charAt(0);
89
+ const atStart = prefixChar === '@' ? matchStart : matchStart + 1;
90
+
91
+ // Build the new value
92
+ const beforeMention = value.substring(0, atStart);
93
+ const completedFile = `@${filePath} `;
94
+ return beforeMention + completedFile + textAfterCursor;
95
+ }
96
+ return value;
97
+ }
98
+
99
+ // Cache for file search results
100
+ const fileCache = new Map<string, { files: FileOption[]; timestamp: number }>();
101
+ const CACHE_TTL_MS = 30000; // 30 seconds cache
102
+
103
+ export function FileAutocomplete({
104
+ inputValue,
105
+ cursorPosition,
106
+ onSelect,
107
+ onClose,
108
+ isVisible,
109
+ apiBase = '',
110
+ }: FileAutocompleteProps) {
111
+ const [selectedIndex, setSelectedIndex] = useState(0);
112
+ const [files, setFiles] = useState<FileOption[]>([]);
113
+ const [isLoading, setIsLoading] = useState(false);
114
+ const [error, setError] = useState<string | null>(null);
115
+ const listRef = useRef<HTMLDivElement>(null);
116
+ const abortControllerRef = useRef<AbortController | null>(null);
117
+
118
+ // Get the current file query
119
+ const query = useMemo(
120
+ () => getFileQuery(inputValue, cursorPosition),
121
+ [inputValue, cursorPosition]
122
+ );
123
+
124
+ // Fetch files when query changes
125
+ useEffect(() => {
126
+ if (!isVisible || query === null) {
127
+ setFiles([]);
128
+ return;
129
+ }
130
+
131
+ // Check cache first
132
+ const cacheKey = query || '__root__';
133
+ const cached = fileCache.get(cacheKey);
134
+ if (cached && Date.now() - cached.timestamp < CACHE_TTL_MS) {
135
+ setFiles(cached.files);
136
+ setError(null);
137
+ return;
138
+ }
139
+
140
+ // Cancel previous request
141
+ if (abortControllerRef.current) {
142
+ abortControllerRef.current.abort();
143
+ }
144
+
145
+ const controller = new AbortController();
146
+ abortControllerRef.current = controller;
147
+
148
+ const fetchFiles = async () => {
149
+ setIsLoading(true);
150
+ setError(null);
151
+
152
+ try {
153
+ const searchQuery = query || '';
154
+ // Use getApiUrl for cloud mode support (routes through workspace proxy)
155
+ const url = apiBase
156
+ ? `${apiBase}/api/files?q=${encodeURIComponent(searchQuery)}&limit=15`
157
+ : getApiUrl(`/api/files?q=${encodeURIComponent(searchQuery)}&limit=15`);
158
+ const response = await fetch(url, { signal: controller.signal });
159
+
160
+ if (!response.ok) {
161
+ throw new Error('Failed to fetch files');
162
+ }
163
+
164
+ const data = await response.json();
165
+ const fileList: FileOption[] = (data.files || []).map((f: { path: string; name: string; isDirectory?: boolean }) => ({
166
+ path: f.path,
167
+ name: f.name,
168
+ isDirectory: f.isDirectory || false,
169
+ }));
170
+
171
+ // Update cache
172
+ fileCache.set(cacheKey, { files: fileList, timestamp: Date.now() });
173
+ setFiles(fileList);
174
+ } catch (err: unknown) {
175
+ if (err instanceof Error && err.name === 'AbortError') return;
176
+ setError('Failed to load files');
177
+ setFiles([]);
178
+ } finally {
179
+ setIsLoading(false);
180
+ }
181
+ };
182
+
183
+ // Debounce the search
184
+ const timeoutId = setTimeout(fetchFiles, 150);
185
+ return () => {
186
+ clearTimeout(timeoutId);
187
+ controller.abort();
188
+ };
189
+ }, [isVisible, query, apiBase]);
190
+
191
+ // Reset selection when files change
192
+ useEffect(() => {
193
+ setSelectedIndex(0);
194
+ }, [files.length]);
195
+
196
+ // Scroll selected item into view
197
+ useEffect(() => {
198
+ if (!listRef.current) return;
199
+ const selected = listRef.current.querySelector('[data-selected="true"]');
200
+ if (selected) {
201
+ selected.scrollIntoView({ block: 'nearest' });
202
+ }
203
+ }, [selectedIndex]);
204
+
205
+ // Handle keyboard navigation
206
+ const handleKeyDown = useCallback(
207
+ (e: KeyboardEvent) => {
208
+ if (!isVisible || files.length === 0) return;
209
+
210
+ switch (e.key) {
211
+ case 'ArrowDown':
212
+ e.preventDefault();
213
+ setSelectedIndex((prev) => (prev + 1) % files.length);
214
+ break;
215
+ case 'ArrowUp':
216
+ e.preventDefault();
217
+ setSelectedIndex((prev) => (prev - 1 + files.length) % files.length);
218
+ break;
219
+ case 'Enter':
220
+ case 'Tab':
221
+ e.preventDefault();
222
+ const selected = files[selectedIndex];
223
+ if (selected) {
224
+ const newValue = completeFileInValue(inputValue, selected.path, cursorPosition);
225
+ onSelect(selected.path, newValue);
226
+ }
227
+ break;
228
+ case 'Escape':
229
+ e.preventDefault();
230
+ onClose();
231
+ break;
232
+ }
233
+ },
234
+ [isVisible, files, selectedIndex, inputValue, cursorPosition, onSelect, onClose]
235
+ );
236
+
237
+ // Register keyboard listener
238
+ useEffect(() => {
239
+ if (isVisible) {
240
+ window.addEventListener('keydown', handleKeyDown);
241
+ return () => window.removeEventListener('keydown', handleKeyDown);
242
+ }
243
+ }, [isVisible, handleKeyDown]);
244
+
245
+ // Handle click on option
246
+ const handleClick = useCallback(
247
+ (file: FileOption) => {
248
+ const newValue = completeFileInValue(inputValue, file.path, cursorPosition);
249
+ onSelect(file.path, newValue);
250
+ },
251
+ [inputValue, cursorPosition, onSelect]
252
+ );
253
+
254
+ if (!isVisible || (files.length === 0 && !isLoading && !error)) {
255
+ return null;
256
+ }
257
+
258
+ return (
259
+ <div
260
+ className="absolute bottom-full left-0 right-0 max-h-[240px] overflow-y-auto bg-bg-card border border-border rounded-lg shadow-modal z-[100] mb-1"
261
+ ref={listRef}
262
+ >
263
+ {/* Header */}
264
+ <div className="px-3 py-1.5 text-[10px] uppercase tracking-wider text-text-muted border-b border-border-subtle flex items-center gap-2">
265
+ <svg width="12" height="12" viewBox="0 0 24 24" fill="none" stroke="currentColor" strokeWidth="2">
266
+ <path d="M13 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V9z" />
267
+ <polyline points="13 2 13 9 20 9" />
268
+ </svg>
269
+ Files {query && <span className="text-text-dim">matching "{query}"</span>}
270
+ </div>
271
+
272
+ {/* Loading state */}
273
+ {isLoading && files.length === 0 && (
274
+ <div className="px-3 py-4 text-sm text-text-muted text-center">
275
+ <svg className="animate-spin mx-auto mb-2" width="16" height="16" viewBox="0 0 24 24">
276
+ <circle cx="12" cy="12" r="10" stroke="currentColor" strokeWidth="2" fill="none" strokeDasharray="32" strokeLinecap="round" />
277
+ </svg>
278
+ Searching files...
279
+ </div>
280
+ )}
281
+
282
+ {/* Error state */}
283
+ {error && (
284
+ <div className="px-3 py-4 text-sm text-red-400 text-center">
285
+ {error}
286
+ </div>
287
+ )}
288
+
289
+ {/* File list */}
290
+ {files.map((file, index) => (
291
+ <div
292
+ key={file.path}
293
+ data-selected={index === selectedIndex}
294
+ className={`flex items-center gap-2.5 py-2 px-3 cursor-pointer transition-colors duration-150 ${
295
+ index === selectedIndex ? 'bg-bg-hover' : 'hover:bg-bg-hover'
296
+ }`}
297
+ onClick={() => handleClick(file)}
298
+ onMouseEnter={() => setSelectedIndex(index)}
299
+ >
300
+ {/* File/Folder icon */}
301
+ <div
302
+ className="w-7 h-7 rounded-md flex items-center justify-center text-text-muted"
303
+ style={{ background: file.isDirectory ? 'rgba(251, 191, 36, 0.15)' : 'rgba(96, 165, 250, 0.15)' }}
304
+ >
305
+ {file.isDirectory ? (
306
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#fbbf24" strokeWidth="2">
307
+ <path d="M22 19a2 2 0 0 1-2 2H4a2 2 0 0 1-2-2V5a2 2 0 0 1 2-2h5l2 3h9a2 2 0 0 1 2 2z" />
308
+ </svg>
309
+ ) : (
310
+ <svg width="14" height="14" viewBox="0 0 24 24" fill="none" stroke="#60a5fa" strokeWidth="2">
311
+ <path d="M13 2H6a2 2 0 0 0-2 2v16a2 2 0 0 0 2 2h12a2 2 0 0 0 2-2V9z" />
312
+ <polyline points="13 2 13 9 20 9" />
313
+ </svg>
314
+ )}
315
+ </div>
316
+ <div className="flex flex-col gap-0.5 min-w-0 flex-1">
317
+ <span className="text-sm font-medium text-text-primary truncate">{file.name}</span>
318
+ <span className="text-xs text-text-dim truncate">{file.path}</span>
319
+ </div>
320
+ </div>
321
+ ))}
322
+
323
+ {/* Empty state (no results after search) */}
324
+ {!isLoading && !error && files.length === 0 && query && (
325
+ <div className="px-3 py-4 text-sm text-text-muted text-center">
326
+ No files found matching "{query}"
327
+ </div>
328
+ )}
329
+ </div>
330
+ );
331
+ }
332
+
333
+ /**
334
+ * Hook to manage file autocomplete state
335
+ */
336
+ export function useFileAutocomplete() {
337
+ const [isVisible, setIsVisible] = useState(false);
338
+ const [inputValue, setInputValue] = useState('');
339
+ const [cursorPosition, setCursorPosition] = useState(0);
340
+
341
+ const handleInputChange = useCallback((value: string, cursorPos: number) => {
342
+ setInputValue(value);
343
+ setCursorPosition(cursorPos);
344
+
345
+ // Show autocomplete if typing @file pattern
346
+ const query = getFileQuery(value, cursorPos);
347
+ setIsVisible(query !== null);
348
+ }, []);
349
+
350
+ const handleSelect = useCallback((filePath: string, newValue: string) => {
351
+ setInputValue(newValue);
352
+ setCursorPosition(newValue.indexOf(' ') + 1);
353
+ setIsVisible(false);
354
+ }, []);
355
+
356
+ const handleClose = useCallback(() => {
357
+ setIsVisible(false);
358
+ }, []);
359
+
360
+ return {
361
+ isVisible,
362
+ inputValue,
363
+ cursorPosition,
364
+ setInputValue: handleInputChange,
365
+ handleSelect,
366
+ handleClose,
367
+ };
368
+ }