source dump of claude code
at main 646 lines 77 kB view raw
1import { c as _c } from "react/compiler-runtime"; 2import chalk from 'chalk'; 3import * as path from 'path'; 4import React, { useCallback, useEffect, useRef, useState } from 'react'; 5import { logEvent } from 'src/services/analytics/index.js'; 6import type { CommandResultDisplay, LocalJSXCommandContext } from '../../commands.js'; 7import { Select } from '../../components/CustomSelect/index.js'; 8import { Dialog } from '../../components/design-system/Dialog.js'; 9import { IdeAutoConnectDialog, IdeDisableAutoConnectDialog, shouldShowAutoConnectDialog, shouldShowDisableAutoConnectDialog } from '../../components/IdeAutoConnectDialog.js'; 10import { Box, Text } from '../../ink.js'; 11import { clearServerCache } from '../../services/mcp/client.js'; 12import type { ScopedMcpServerConfig } from '../../services/mcp/types.js'; 13import { useAppState, useSetAppState } from '../../state/AppState.js'; 14import { getCwd } from '../../utils/cwd.js'; 15import { execFileNoThrow } from '../../utils/execFileNoThrow.js'; 16import { type DetectedIDEInfo, detectIDEs, detectRunningIDEs, type IdeType, isJetBrainsIde, isSupportedJetBrainsTerminal, isSupportedTerminal, toIDEDisplayName } from '../../utils/ide.js'; 17import { getCurrentWorktreeSession } from '../../utils/worktree.js'; 18type IDEScreenProps = { 19 availableIDEs: DetectedIDEInfo[]; 20 unavailableIDEs: DetectedIDEInfo[]; 21 selectedIDE?: DetectedIDEInfo | null; 22 onClose: () => void; 23 onSelect: (ide?: DetectedIDEInfo) => void; 24}; 25function IDEScreen(t0) { 26 const $ = _c(39); 27 const { 28 availableIDEs, 29 unavailableIDEs, 30 selectedIDE, 31 onClose, 32 onSelect 33 } = t0; 34 let t1; 35 if ($[0] !== selectedIDE?.port) { 36 t1 = selectedIDE?.port?.toString() ?? "None"; 37 $[0] = selectedIDE?.port; 38 $[1] = t1; 39 } else { 40 t1 = $[1]; 41 } 42 const [selectedValue, setSelectedValue] = useState(t1); 43 const [showAutoConnectDialog, setShowAutoConnectDialog] = useState(false); 44 const [showDisableAutoConnectDialog, setShowDisableAutoConnectDialog] = useState(false); 45 let t2; 46 if ($[2] !== availableIDEs || $[3] !== onSelect) { 47 t2 = value => { 48 if (value !== "None" && shouldShowAutoConnectDialog()) { 49 setShowAutoConnectDialog(true); 50 } else { 51 if (value === "None" && shouldShowDisableAutoConnectDialog()) { 52 setShowDisableAutoConnectDialog(true); 53 } else { 54 onSelect(availableIDEs.find(ide => ide.port === parseInt(value))); 55 } 56 } 57 }; 58 $[2] = availableIDEs; 59 $[3] = onSelect; 60 $[4] = t2; 61 } else { 62 t2 = $[4]; 63 } 64 const handleSelectIDE = t2; 65 let t3; 66 if ($[5] !== availableIDEs) { 67 t3 = availableIDEs.reduce(_temp, {}); 68 $[5] = availableIDEs; 69 $[6] = t3; 70 } else { 71 t3 = $[6]; 72 } 73 const ideCounts = t3; 74 let t4; 75 if ($[7] !== availableIDEs || $[8] !== ideCounts) { 76 let t5; 77 if ($[10] !== ideCounts) { 78 t5 = ide_1 => { 79 const hasMultipleInstances = (ideCounts[ide_1.name] || 0) > 1; 80 const showWorkspace = hasMultipleInstances && ide_1.workspaceFolders.length > 0; 81 return { 82 label: ide_1.name, 83 value: ide_1.port.toString(), 84 description: showWorkspace ? formatWorkspaceFolders(ide_1.workspaceFolders) : undefined 85 }; 86 }; 87 $[10] = ideCounts; 88 $[11] = t5; 89 } else { 90 t5 = $[11]; 91 } 92 t4 = availableIDEs.map(t5).concat([{ 93 label: "None", 94 value: "None", 95 description: undefined 96 }]); 97 $[7] = availableIDEs; 98 $[8] = ideCounts; 99 $[9] = t4; 100 } else { 101 t4 = $[9]; 102 } 103 const options = t4; 104 if (showAutoConnectDialog) { 105 let t5; 106 if ($[12] !== handleSelectIDE || $[13] !== selectedValue) { 107 t5 = <IdeAutoConnectDialog onComplete={() => handleSelectIDE(selectedValue)} />; 108 $[12] = handleSelectIDE; 109 $[13] = selectedValue; 110 $[14] = t5; 111 } else { 112 t5 = $[14]; 113 } 114 return t5; 115 } 116 if (showDisableAutoConnectDialog) { 117 let t5; 118 if ($[15] !== onSelect) { 119 t5 = <IdeDisableAutoConnectDialog onComplete={() => { 120 onSelect(undefined); 121 }} />; 122 $[15] = onSelect; 123 $[16] = t5; 124 } else { 125 t5 = $[16]; 126 } 127 return t5; 128 } 129 let t5; 130 if ($[17] !== availableIDEs.length) { 131 t5 = availableIDEs.length === 0 && <Text dimColor={true}>{isSupportedJetBrainsTerminal() ? "No available IDEs detected. Please install the plugin and restart your IDE:\nhttps://docs.claude.com/s/claude-code-jetbrains" : "No available IDEs detected. Make sure your IDE has the Claude Code extension or plugin installed and is running."}</Text>; 132 $[17] = availableIDEs.length; 133 $[18] = t5; 134 } else { 135 t5 = $[18]; 136 } 137 let t6; 138 if ($[19] !== availableIDEs.length || $[20] !== handleSelectIDE || $[21] !== options || $[22] !== selectedValue) { 139 t6 = availableIDEs.length !== 0 && <Select defaultValue={selectedValue} defaultFocusValue={selectedValue} options={options} onChange={value_0 => { 140 setSelectedValue(value_0); 141 handleSelectIDE(value_0); 142 }} />; 143 $[19] = availableIDEs.length; 144 $[20] = handleSelectIDE; 145 $[21] = options; 146 $[22] = selectedValue; 147 $[23] = t6; 148 } else { 149 t6 = $[23]; 150 } 151 let t7; 152 if ($[24] !== availableIDEs) { 153 t7 = availableIDEs.length !== 0 && availableIDEs.some(_temp2) && <Box marginTop={1}><Text color="warning">Note: Only one Claude Code instance can be connected to VS Code at a time.</Text></Box>; 154 $[24] = availableIDEs; 155 $[25] = t7; 156 } else { 157 t7 = $[25]; 158 } 159 let t8; 160 if ($[26] !== availableIDEs.length) { 161 t8 = availableIDEs.length !== 0 && !isSupportedTerminal() && <Box marginTop={1}><Text dimColor={true}>Tip: You can enable auto-connect to IDE in /config or with the --ide flag</Text></Box>; 162 $[26] = availableIDEs.length; 163 $[27] = t8; 164 } else { 165 t8 = $[27]; 166 } 167 let t9; 168 if ($[28] !== unavailableIDEs) { 169 t9 = unavailableIDEs.length > 0 && <Box marginTop={1} flexDirection="column"><Text dimColor={true}>Found {unavailableIDEs.length} other running IDE(s). However, their workspace/project directories do not match the current cwd.</Text><Box marginTop={1} flexDirection="column">{unavailableIDEs.map(_temp3)}</Box></Box>; 170 $[28] = unavailableIDEs; 171 $[29] = t9; 172 } else { 173 t9 = $[29]; 174 } 175 let t10; 176 if ($[30] !== t5 || $[31] !== t6 || $[32] !== t7 || $[33] !== t8 || $[34] !== t9) { 177 t10 = <Box flexDirection="column">{t5}{t6}{t7}{t8}{t9}</Box>; 178 $[30] = t5; 179 $[31] = t6; 180 $[32] = t7; 181 $[33] = t8; 182 $[34] = t9; 183 $[35] = t10; 184 } else { 185 t10 = $[35]; 186 } 187 let t11; 188 if ($[36] !== onClose || $[37] !== t10) { 189 t11 = <Dialog title="Select IDE" subtitle="Connect to an IDE for integrated development features." onCancel={onClose} color="ide">{t10}</Dialog>; 190 $[36] = onClose; 191 $[37] = t10; 192 $[38] = t11; 193 } else { 194 t11 = $[38]; 195 } 196 return t11; 197} 198function _temp3(ide_3, index) { 199 return <Box key={index} paddingLeft={3}><Text dimColor={true}> {ide_3.name}: {formatWorkspaceFolders(ide_3.workspaceFolders)}</Text></Box>; 200} 201function _temp2(ide_2) { 202 return ide_2.name === "VS Code" || ide_2.name === "Visual Studio Code"; 203} 204function _temp(acc, ide_0) { 205 acc[ide_0.name] = (acc[ide_0.name] || 0) + 1; 206 return acc; 207} 208async function findCurrentIDE(availableIDEs: DetectedIDEInfo[], dynamicMcpConfig?: Record<string, ScopedMcpServerConfig>): Promise<DetectedIDEInfo | null> { 209 const currentConfig = dynamicMcpConfig?.ide; 210 if (!currentConfig || currentConfig.type !== 'sse-ide' && currentConfig.type !== 'ws-ide') { 211 return null; 212 } 213 for (const ide of availableIDEs) { 214 if (ide.url === currentConfig.url) { 215 return ide; 216 } 217 } 218 return null; 219} 220type IDEOpenSelectionProps = { 221 availableIDEs: DetectedIDEInfo[]; 222 onSelectIDE: (ide?: DetectedIDEInfo) => void; 223 onDone: (result?: string, options?: { 224 display?: CommandResultDisplay; 225 }) => void; 226}; 227function IDEOpenSelection(t0) { 228 const $ = _c(18); 229 const { 230 availableIDEs, 231 onSelectIDE, 232 onDone 233 } = t0; 234 let t1; 235 if ($[0] !== availableIDEs[0]?.port) { 236 t1 = availableIDEs[0]?.port?.toString() ?? ""; 237 $[0] = availableIDEs[0]?.port; 238 $[1] = t1; 239 } else { 240 t1 = $[1]; 241 } 242 const [selectedValue, setSelectedValue] = useState(t1); 243 let t2; 244 if ($[2] !== availableIDEs || $[3] !== onSelectIDE) { 245 t2 = value => { 246 const selectedIDE = availableIDEs.find(ide => ide.port === parseInt(value)); 247 onSelectIDE(selectedIDE); 248 }; 249 $[2] = availableIDEs; 250 $[3] = onSelectIDE; 251 $[4] = t2; 252 } else { 253 t2 = $[4]; 254 } 255 const handleSelectIDE = t2; 256 let t3; 257 if ($[5] !== availableIDEs) { 258 t3 = availableIDEs.map(_temp4); 259 $[5] = availableIDEs; 260 $[6] = t3; 261 } else { 262 t3 = $[6]; 263 } 264 const options = t3; 265 let t4; 266 if ($[7] !== onDone) { 267 t4 = function handleCancel() { 268 onDone("IDE selection cancelled", { 269 display: "system" 270 }); 271 }; 272 $[7] = onDone; 273 $[8] = t4; 274 } else { 275 t4 = $[8]; 276 } 277 const handleCancel = t4; 278 let t5; 279 if ($[9] !== handleSelectIDE) { 280 t5 = value_0 => { 281 setSelectedValue(value_0); 282 handleSelectIDE(value_0); 283 }; 284 $[9] = handleSelectIDE; 285 $[10] = t5; 286 } else { 287 t5 = $[10]; 288 } 289 let t6; 290 if ($[11] !== options || $[12] !== selectedValue || $[13] !== t5) { 291 t6 = <Select defaultValue={selectedValue} defaultFocusValue={selectedValue} options={options} onChange={t5} />; 292 $[11] = options; 293 $[12] = selectedValue; 294 $[13] = t5; 295 $[14] = t6; 296 } else { 297 t6 = $[14]; 298 } 299 let t7; 300 if ($[15] !== handleCancel || $[16] !== t6) { 301 t7 = <Dialog title="Select an IDE to open the project" onCancel={handleCancel} color="ide">{t6}</Dialog>; 302 $[15] = handleCancel; 303 $[16] = t6; 304 $[17] = t7; 305 } else { 306 t7 = $[17]; 307 } 308 return t7; 309} 310function _temp4(ide_0) { 311 return { 312 label: ide_0.name, 313 value: ide_0.port.toString() 314 }; 315} 316function RunningIDESelector(t0) { 317 const $ = _c(15); 318 const { 319 runningIDEs, 320 onSelectIDE, 321 onDone 322 } = t0; 323 const [selectedValue, setSelectedValue] = useState(runningIDEs[0] ?? ""); 324 let t1; 325 if ($[0] !== onSelectIDE) { 326 t1 = value => { 327 onSelectIDE(value as IdeType); 328 }; 329 $[0] = onSelectIDE; 330 $[1] = t1; 331 } else { 332 t1 = $[1]; 333 } 334 const handleSelectIDE = t1; 335 let t2; 336 if ($[2] !== runningIDEs) { 337 t2 = runningIDEs.map(_temp5); 338 $[2] = runningIDEs; 339 $[3] = t2; 340 } else { 341 t2 = $[3]; 342 } 343 const options = t2; 344 let t3; 345 if ($[4] !== onDone) { 346 t3 = function handleCancel() { 347 onDone("IDE selection cancelled", { 348 display: "system" 349 }); 350 }; 351 $[4] = onDone; 352 $[5] = t3; 353 } else { 354 t3 = $[5]; 355 } 356 const handleCancel = t3; 357 let t4; 358 if ($[6] !== handleSelectIDE) { 359 t4 = value_0 => { 360 setSelectedValue(value_0); 361 handleSelectIDE(value_0); 362 }; 363 $[6] = handleSelectIDE; 364 $[7] = t4; 365 } else { 366 t4 = $[7]; 367 } 368 let t5; 369 if ($[8] !== options || $[9] !== selectedValue || $[10] !== t4) { 370 t5 = <Select defaultFocusValue={selectedValue} options={options} onChange={t4} />; 371 $[8] = options; 372 $[9] = selectedValue; 373 $[10] = t4; 374 $[11] = t5; 375 } else { 376 t5 = $[11]; 377 } 378 let t6; 379 if ($[12] !== handleCancel || $[13] !== t5) { 380 t6 = <Dialog title="Select IDE to install extension" onCancel={handleCancel} color="ide">{t5}</Dialog>; 381 $[12] = handleCancel; 382 $[13] = t5; 383 $[14] = t6; 384 } else { 385 t6 = $[14]; 386 } 387 return t6; 388} 389function _temp5(ide) { 390 return { 391 label: toIDEDisplayName(ide), 392 value: ide 393 }; 394} 395function InstallOnMount(t0) { 396 const $ = _c(4); 397 const { 398 ide, 399 onInstall 400 } = t0; 401 let t1; 402 let t2; 403 if ($[0] !== ide || $[1] !== onInstall) { 404 t1 = () => { 405 onInstall(ide); 406 }; 407 t2 = [ide, onInstall]; 408 $[0] = ide; 409 $[1] = onInstall; 410 $[2] = t1; 411 $[3] = t2; 412 } else { 413 t1 = $[2]; 414 t2 = $[3]; 415 } 416 useEffect(t1, t2); 417 return null; 418} 419export async function call(onDone: (result?: string, options?: { 420 display?: CommandResultDisplay; 421}) => void, context: LocalJSXCommandContext, args: string): Promise<React.ReactNode | null> { 422 logEvent('tengu_ext_ide_command', {}); 423 const { 424 options: { 425 dynamicMcpConfig 426 }, 427 onChangeDynamicMcpConfig 428 } = context; 429 430 // Handle 'open' argument 431 if (args?.trim() === 'open') { 432 const worktreeSession = getCurrentWorktreeSession(); 433 const targetPath = worktreeSession ? worktreeSession.worktreePath : getCwd(); 434 435 // Detect available IDEs 436 const detectedIDEs = await detectIDEs(true); 437 const availableIDEs = detectedIDEs.filter(ide => ide.isValid); 438 if (availableIDEs.length === 0) { 439 onDone('No IDEs with Claude Code extension detected.'); 440 return null; 441 } 442 443 // Return IDE selection component 444 return <IDEOpenSelection availableIDEs={availableIDEs} onSelectIDE={async (selectedIDE?: DetectedIDEInfo) => { 445 if (!selectedIDE) { 446 onDone('No IDE selected.'); 447 return; 448 } 449 450 // Try to open the project in the selected IDE 451 if (selectedIDE.name.toLowerCase().includes('vscode') || selectedIDE.name.toLowerCase().includes('cursor') || selectedIDE.name.toLowerCase().includes('windsurf')) { 452 // VS Code-based IDEs 453 const { 454 code 455 } = await execFileNoThrow('code', [targetPath]); 456 if (code === 0) { 457 onDone(`Opened ${worktreeSession ? 'worktree' : 'project'} in ${chalk.bold(selectedIDE.name)}`); 458 } else { 459 onDone(`Failed to open in ${selectedIDE.name}. Try opening manually: ${targetPath}`); 460 } 461 } else if (isSupportedJetBrainsTerminal()) { 462 // JetBrains IDEs - they usually open via their CLI tools 463 onDone(`Please open the ${worktreeSession ? 'worktree' : 'project'} manually in ${chalk.bold(selectedIDE.name)}: ${targetPath}`); 464 } else { 465 onDone(`Please open the ${worktreeSession ? 'worktree' : 'project'} manually in ${chalk.bold(selectedIDE.name)}: ${targetPath}`); 466 } 467 }} onDone={() => { 468 onDone('Exited without opening IDE', { 469 display: 'system' 470 }); 471 }} />; 472 } 473 const detectedIDEs = await detectIDEs(true); 474 475 // If no IDEs with extensions detected, check for running IDEs and offer to install 476 if (detectedIDEs.length === 0 && context.onInstallIDEExtension && !isSupportedTerminal()) { 477 const runningIDEs = await detectRunningIDEs(); 478 const onInstall = (ide: IdeType) => { 479 if (context.onInstallIDEExtension) { 480 context.onInstallIDEExtension(ide); 481 // The completion message will be shown after installation 482 if (isJetBrainsIde(ide)) { 483 onDone(`Installed plugin to ${chalk.bold(toIDEDisplayName(ide))}\n` + `Please ${chalk.bold('restart your IDE')} completely for it to take effect`); 484 } else { 485 onDone(`Installed extension to ${chalk.bold(toIDEDisplayName(ide))}`); 486 } 487 } 488 }; 489 if (runningIDEs.length > 1) { 490 // Show selector when multiple IDEs are running 491 return <RunningIDESelector runningIDEs={runningIDEs} onSelectIDE={onInstall} onDone={() => { 492 onDone('No IDE selected.', { 493 display: 'system' 494 }); 495 }} />; 496 } else if (runningIDEs.length === 1) { 497 return <InstallOnMount ide={runningIDEs[0]!} onInstall={onInstall} />; 498 } 499 } 500 const availableIDEs = detectedIDEs.filter(ide => ide.isValid); 501 const unavailableIDEs = detectedIDEs.filter(ide => !ide.isValid); 502 const currentIDE = await findCurrentIDE(availableIDEs, dynamicMcpConfig); 503 return <IDECommandFlow availableIDEs={availableIDEs} unavailableIDEs={unavailableIDEs} currentIDE={currentIDE} dynamicMcpConfig={dynamicMcpConfig} onChangeDynamicMcpConfig={onChangeDynamicMcpConfig} onDone={onDone} />; 504} 505 506// Connection timeout slightly longer than the 30s MCP connection timeout 507const IDE_CONNECTION_TIMEOUT_MS = 35000; 508type IDECommandFlowProps = { 509 availableIDEs: DetectedIDEInfo[]; 510 unavailableIDEs: DetectedIDEInfo[]; 511 currentIDE: DetectedIDEInfo | null; 512 dynamicMcpConfig?: Record<string, ScopedMcpServerConfig>; 513 onChangeDynamicMcpConfig?: (config: Record<string, ScopedMcpServerConfig>) => void; 514 onDone: (result?: string, options?: { 515 display?: CommandResultDisplay; 516 }) => void; 517}; 518function IDECommandFlow({ 519 availableIDEs, 520 unavailableIDEs, 521 currentIDE, 522 dynamicMcpConfig, 523 onChangeDynamicMcpConfig, 524 onDone 525}: IDECommandFlowProps): React.ReactNode { 526 const [connectingIDE, setConnectingIDE] = useState<DetectedIDEInfo | null>(null); 527 const ideClient = useAppState(s => s.mcp.clients.find(c => c.name === 'ide')); 528 const setAppState = useSetAppState(); 529 const isFirstCheckRef = useRef(true); 530 531 // Watch for connection result 532 useEffect(() => { 533 if (!connectingIDE) return; 534 // Skip the first check — it reflects stale state from before the 535 // config change was dispatched 536 if (isFirstCheckRef.current) { 537 isFirstCheckRef.current = false; 538 return; 539 } 540 if (!ideClient || ideClient.type === 'pending') return; 541 if (ideClient.type === 'connected') { 542 onDone(`Connected to ${connectingIDE.name}.`); 543 } else if (ideClient.type === 'failed') { 544 onDone(`Failed to connect to ${connectingIDE.name}.`); 545 } 546 }, [ideClient, connectingIDE, onDone]); 547 548 // Timeout fallback 549 useEffect(() => { 550 if (!connectingIDE) return; 551 const timer = setTimeout(onDone, IDE_CONNECTION_TIMEOUT_MS, `Connection to ${connectingIDE.name} timed out.`); 552 return () => clearTimeout(timer); 553 }, [connectingIDE, onDone]); 554 const handleSelectIDE = useCallback((selectedIDE?: DetectedIDEInfo) => { 555 if (!onChangeDynamicMcpConfig) { 556 onDone('Error connecting to IDE.'); 557 return; 558 } 559 const newConfig = { 560 ...(dynamicMcpConfig || {}) 561 }; 562 if (currentIDE) { 563 delete newConfig.ide; 564 } 565 if (!selectedIDE) { 566 // Close the MCP transport and remove the client from state 567 if (ideClient && ideClient.type === 'connected' && currentIDE) { 568 // Null out onclose to prevent auto-reconnection 569 ideClient.client.onclose = () => {}; 570 void clearServerCache('ide', ideClient.config); 571 setAppState(prev => ({ 572 ...prev, 573 mcp: { 574 ...prev.mcp, 575 clients: prev.mcp.clients.filter(c_0 => c_0.name !== 'ide'), 576 tools: prev.mcp.tools.filter(t => !t.name?.startsWith('mcp__ide__')), 577 commands: prev.mcp.commands.filter(c_1 => !c_1.name?.startsWith('mcp__ide__')) 578 } 579 })); 580 } 581 onChangeDynamicMcpConfig(newConfig); 582 onDone(currentIDE ? `Disconnected from ${currentIDE.name}.` : 'No IDE selected.'); 583 return; 584 } 585 const url = selectedIDE.url; 586 newConfig.ide = { 587 type: url.startsWith('ws:') ? 'ws-ide' : 'sse-ide', 588 url: url, 589 ideName: selectedIDE.name, 590 authToken: selectedIDE.authToken, 591 ideRunningInWindows: selectedIDE.ideRunningInWindows, 592 scope: 'dynamic' as const 593 } as ScopedMcpServerConfig; 594 isFirstCheckRef.current = true; 595 setConnectingIDE(selectedIDE); 596 onChangeDynamicMcpConfig(newConfig); 597 }, [dynamicMcpConfig, currentIDE, ideClient, setAppState, onChangeDynamicMcpConfig, onDone]); 598 if (connectingIDE) { 599 return <Text dimColor>Connecting to {connectingIDE.name}</Text>; 600 } 601 return <IDEScreen availableIDEs={availableIDEs} unavailableIDEs={unavailableIDEs} selectedIDE={currentIDE} onClose={() => onDone('IDE selection cancelled', { 602 display: 'system' 603 })} onSelect={handleSelectIDE} />; 604} 605 606/** 607 * Formats workspace folders for display, stripping cwd and showing tail end of paths 608 * @param folders Array of folder paths 609 * @param maxLength Maximum total length of the formatted string 610 * @returns Formatted string with folder paths 611 */ 612export function formatWorkspaceFolders(folders: string[], maxLength: number = 100): string { 613 if (folders.length === 0) return ''; 614 const cwd = getCwd(); 615 616 // Only show first 2 workspaces 617 const foldersToShow = folders.slice(0, 2); 618 const hasMore = folders.length > 2; 619 620 // Account for ", …" if there are more folders 621 const ellipsisOverhead = hasMore ? 3 : 0; // ", …" 622 623 // Account for commas and spaces between paths (", " = 2 chars per separator) 624 const separatorOverhead = (foldersToShow.length - 1) * 2; 625 const availableLength = maxLength - separatorOverhead - ellipsisOverhead; 626 const maxLengthPerPath = Math.floor(availableLength / foldersToShow.length); 627 const cwdNFC = cwd.normalize('NFC'); 628 const formattedFolders = foldersToShow.map(folder => { 629 // Strip cwd from the beginning if present 630 // Normalize both to NFC for consistent comparison (macOS uses NFD paths) 631 const folderNFC = folder.normalize('NFC'); 632 if (folderNFC.startsWith(cwdNFC + path.sep)) { 633 folder = folderNFC.slice(cwdNFC.length + 1); 634 } 635 if (folder.length <= maxLengthPerPath) { 636 return folder; 637 } 638 return '…' + folder.slice(-(maxLengthPerPath - 1)); 639 }); 640 let result = formattedFolders.join(', '); 641 if (hasMore) { 642 result += ', …'; 643 } 644 return result; 645} 646//# sourceMappingURL=data:application/json;charset=utf-8;base64,{"version":3,"names":["chalk","path","React","useCallback","useEffect","useRef","useState","logEvent","CommandResultDisplay","LocalJSXCommandContext","Select","Dialog","IdeAutoConnectDialog","IdeDisableAutoConnectDialog","shouldShowAutoConnectDialog","shouldShowDisableAutoConnectDialog","Box","Text","clearServerCache","ScopedMcpServerConfig","useAppState","useSetAppState","getCwd","execFileNoThrow","DetectedIDEInfo","detectIDEs","detectRunningIDEs","IdeType","isJetBrainsIde","isSupportedJetBrainsTerminal","isSupportedTerminal","toIDEDisplayName","getCurrentWorktreeSession","IDEScreenProps","availableIDEs","unavailableIDEs","selectedIDE","onClose","onSelect","ide","IDEScreen","t0","$","_c","t1","port","toString","selectedValue","setSelectedValue","showAutoConnectDialog","setShowAutoConnectDialog","showDisableAutoConnectDialog","setShowDisableAutoConnectDialog","t2","value","find","parseInt","handleSelectIDE","t3","reduce","_temp","ideCounts","t4","t5","ide_1","hasMultipleInstances","name","showWorkspace","workspaceFolders","length","label","description","formatWorkspaceFolders","undefined","map","concat","options","t6","value_0","t7","some","_temp2","t8","t9","_temp3","t10","t11","ide_3","index","ide_2","acc","ide_0","findCurrentIDE","dynamicMcpConfig","Record","Promise","currentConfig","type","url","IDEOpenSelectionProps","onSelectIDE","onDone","result","display","IDEOpenSelection","_temp4","handleCancel","RunningIDESelector","runningIDEs","_temp5","InstallOnMount","onInstall","call","context","args","ReactNode","onChangeDynamicMcpConfig","trim","worktreeSession","targetPath","worktreePath","detectedIDEs","filter","isValid","toLowerCase","includes","code","bold","onInstallIDEExtension","currentIDE","IDE_CONNECTION_TIMEOUT_MS","IDECommandFlowProps","config","IDECommandFlow","connectingIDE","setConnectingIDE","ideClient","s","mcp","clients","c","setAppState","isFirstCheckRef","current","timer","setTimeout","clearTimeout","newConfig","client","onclose","prev","tools","t","startsWith","commands","ideName","authToken","ideRunningInWindows","scope","const","folders","maxLength","cwd","foldersToShow","slice","hasMore","ellipsisOverhead","separatorOverhead","availableLength","maxLengthPerPath","Math","floor","cwdNFC","normalize","formattedFolders","folder","folderNFC","sep","join"],"sources":["ide.tsx"],"sourcesContent":["import chalk from 'chalk'\nimport * as path from 'path'\nimport React, { useCallback, useEffect, useRef, useState } from 'react'\nimport { logEvent } from 'src/services/analytics/index.js'\nimport type {\n  CommandResultDisplay,\n  LocalJSXCommandContext,\n} from '../../commands.js'\nimport { Select } from '../../components/CustomSelect/index.js'\nimport { Dialog } from '../../components/design-system/Dialog.js'\nimport {\n  IdeAutoConnectDialog,\n  IdeDisableAutoConnectDialog,\n  shouldShowAutoConnectDialog,\n  shouldShowDisableAutoConnectDialog,\n} from '../../components/IdeAutoConnectDialog.js'\nimport { Box, Text } from '../../ink.js'\nimport { clearServerCache } from '../../services/mcp/client.js'\nimport type { ScopedMcpServerConfig } from '../../services/mcp/types.js'\nimport { useAppState, useSetAppState } from '../../state/AppState.js'\nimport { getCwd } from '../../utils/cwd.js'\nimport { execFileNoThrow } from '../../utils/execFileNoThrow.js'\nimport {\n  type DetectedIDEInfo,\n  detectIDEs,\n  detectRunningIDEs,\n  type IdeType,\n  isJetBrainsIde,\n  isSupportedJetBrainsTerminal,\n  isSupportedTerminal,\n  toIDEDisplayName,\n} from '../../utils/ide.js'\nimport { getCurrentWorktreeSession } from '../../utils/worktree.js'\n\ntype IDEScreenProps = {\n  availableIDEs: DetectedIDEInfo[]\n  unavailableIDEs: DetectedIDEInfo[]\n  selectedIDE?: DetectedIDEInfo | null\n  onClose: () => void\n  onSelect: (ide?: DetectedIDEInfo) => void\n}\n\nfunction IDEScreen({\n  availableIDEs,\n  unavailableIDEs,\n  selectedIDE,\n  onClose,\n  onSelect,\n}: IDEScreenProps): React.ReactNode {\n  const [selectedValue, setSelectedValue] = useState(\n    selectedIDE?.port?.toString() ?? 'None',\n  )\n  const [showAutoConnectDialog, setShowAutoConnectDialog] = useState(false)\n  const [showDisableAutoConnectDialog, setShowDisableAutoConnectDialog] =\n    useState(false)\n\n  const handleSelectIDE = useCallback(\n    (value: string) => {\n      if (value !== 'None' && shouldShowAutoConnectDialog()) {\n        setShowAutoConnectDialog(true)\n      } else if (value === 'None' && shouldShowDisableAutoConnectDialog()) {\n        setShowDisableAutoConnectDialog(true)\n      } else {\n        onSelect(availableIDEs.find(ide => ide.port === parseInt(value)))\n      }\n    },\n    [availableIDEs, onSelect],\n  )\n\n  const ideCounts = availableIDEs.reduce<Record<string, number>>((acc, ide) => {\n    acc[ide.name] = (acc[ide.name] || 0) + 1\n    return acc\n  }, {})\n\n  const options = availableIDEs\n    .map(ide => {\n      const hasMultipleInstances = (ideCounts[ide.name] || 0) > 1\n      const showWorkspace =\n        hasMultipleInstances && ide.workspaceFolders.length > 0\n\n      return {\n        label: ide.name,\n        value: ide.port.toString(),\n        description: showWorkspace\n          ? formatWorkspaceFolders(ide.workspaceFolders)\n          : undefined,\n      }\n    })\n    .concat([{ label: 'None', value: 'None', description: undefined }])\n\n  if (showAutoConnectDialog) {\n    return (\n      <IdeAutoConnectDialog onComplete={() => handleSelectIDE(selectedValue)} />\n    )\n  }\n\n  if (showDisableAutoConnectDialog) {\n    return (\n      <IdeDisableAutoConnectDialog\n        onComplete={() => {\n          // Always disconnect when user selects \"None\", regardless of their\n          // choice about disabling auto-connect\n          onSelect(undefined)\n        }}\n      />\n    )\n  }\n\n  return (\n    <Dialog\n      title=\"Select IDE\"\n      subtitle=\"Connect to an IDE for integrated development features.\"\n      onCancel={onClose}\n      color=\"ide\"\n    >\n      <Box flexDirection=\"column\">\n        {availableIDEs.length === 0 && (\n          <Text dimColor>\n            {isSupportedJetBrainsTerminal()\n              ? 'No available IDEs detected. Please install the plugin and restart your IDE:\\n' +\n                'https://docs.claude.com/s/claude-code-jetbrains'\n              : 'No available IDEs detected. Make sure your IDE has the Claude Code extension or plugin installed and is running.'}\n          </Text>\n        )}\n\n        {availableIDEs.length !== 0 && (\n          <Select\n            defaultValue={selectedValue}\n            defaultFocusValue={selectedValue}\n            options={options}\n            onChange={value => {\n              setSelectedValue(value)\n              handleSelectIDE(value)\n            }}\n          />\n        )}\n        {availableIDEs.length !== 0 &&\n          availableIDEs.some(\n            ide => ide.name === 'VS Code' || ide.name === 'Visual Studio Code',\n          ) && (\n            <Box marginTop={1}>\n              <Text color=\"warning\">\n                Note: Only one Claude Code instance can be connected to VS Code\n                at a time.\n              </Text>\n            </Box>\n          )}\n        {availableIDEs.length !== 0 && !isSupportedTerminal() && (\n          <Box marginTop={1}>\n            <Text dimColor>\n              Tip: You can enable auto-connect to IDE in /config or with the\n              --ide flag\n            </Text>\n          </Box>\n        )}\n\n        {unavailableIDEs.length > 0 && (\n          <Box marginTop={1} flexDirection=\"column\">\n            <Text dimColor>\n              Found {unavailableIDEs.length} other running IDE(s). However,\n              their workspace/project directories do not match the current cwd.\n            </Text>\n            <Box marginTop={1} flexDirection=\"column\">\n              {unavailableIDEs.map((ide, index) => (\n                <Box key={index} paddingLeft={3}>\n                  <Text dimColor>\n                    • {ide.name}: {formatWorkspaceFolders(ide.workspaceFolders)}\n                  </Text>\n                </Box>\n              ))}\n            </Box>\n          </Box>\n        )}\n      </Box>\n    </Dialog>\n  )\n}\n\nasync function findCurrentIDE(\n  availableIDEs: DetectedIDEInfo[],\n  dynamicMcpConfig?: Record<string, ScopedMcpServerConfig>,\n): Promise<DetectedIDEInfo | null> {\n  const currentConfig = dynamicMcpConfig?.ide\n  if (\n    !currentConfig ||\n    (currentConfig.type !== 'sse-ide' && currentConfig.type !== 'ws-ide')\n  ) {\n    return null\n  }\n  for (const ide of availableIDEs) {\n    if (ide.url === currentConfig.url) {\n      return ide\n    }\n  }\n  return null\n}\n\ntype IDEOpenSelectionProps = {\n  availableIDEs: DetectedIDEInfo[]\n  onSelectIDE: (ide?: DetectedIDEInfo) => void\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n}\n\nfunction IDEOpenSelection({\n  availableIDEs,\n  onSelectIDE,\n  onDone,\n}: IDEOpenSelectionProps): React.ReactNode {\n  const [selectedValue, setSelectedValue] = useState(\n    availableIDEs[0]?.port?.toString() ?? '',\n  )\n\n  const handleSelectIDE = useCallback(\n    (value: string) => {\n      const selectedIDE = availableIDEs.find(\n        ide => ide.port === parseInt(value),\n      )\n      onSelectIDE(selectedIDE)\n    },\n    [availableIDEs, onSelectIDE],\n  )\n\n  const options = availableIDEs.map(ide => ({\n    label: ide.name,\n    value: ide.port.toString(),\n  }))\n\n  function handleCancel(): void {\n    onDone('IDE selection cancelled', { display: 'system' })\n  }\n\n  return (\n    <Dialog\n      title=\"Select an IDE to open the project\"\n      onCancel={handleCancel}\n      color=\"ide\"\n    >\n      <Select\n        defaultValue={selectedValue}\n        defaultFocusValue={selectedValue}\n        options={options}\n        onChange={value => {\n          setSelectedValue(value)\n          handleSelectIDE(value)\n        }}\n      />\n    </Dialog>\n  )\n}\n\nfunction RunningIDESelector({\n  runningIDEs,\n  onSelectIDE,\n  onDone,\n}: {\n  runningIDEs: IdeType[]\n  onSelectIDE: (ide: IdeType) => void\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n}): React.ReactNode {\n  const [selectedValue, setSelectedValue] = useState(runningIDEs[0] ?? '')\n\n  const handleSelectIDE = useCallback(\n    (value: string) => {\n      onSelectIDE(value as IdeType)\n    },\n    [onSelectIDE],\n  )\n\n  const options = runningIDEs.map(ide => ({\n    label: toIDEDisplayName(ide),\n    value: ide,\n  }))\n\n  function handleCancel(): void {\n    onDone('IDE selection cancelled', { display: 'system' })\n  }\n\n  return (\n    <Dialog\n      title=\"Select IDE to install extension\"\n      onCancel={handleCancel}\n      color=\"ide\"\n    >\n      <Select\n        defaultFocusValue={selectedValue}\n        options={options}\n        onChange={value => {\n          setSelectedValue(value)\n          handleSelectIDE(value)\n        }}\n      />\n    </Dialog>\n  )\n}\n\nfunction InstallOnMount({\n  ide,\n  onInstall,\n}: {\n  ide: IdeType\n  onInstall: (ide: IdeType) => void\n}): React.ReactNode {\n  useEffect(() => {\n    onInstall(ide)\n  }, [ide, onInstall])\n  return null\n}\n\nexport async function call(\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void,\n  context: LocalJSXCommandContext,\n  args: string,\n): Promise<React.ReactNode | null> {\n  logEvent('tengu_ext_ide_command', {})\n  const {\n    options: { dynamicMcpConfig },\n    onChangeDynamicMcpConfig,\n  } = context\n\n  // Handle 'open' argument\n  if (args?.trim() === 'open') {\n    const worktreeSession = getCurrentWorktreeSession()\n    const targetPath = worktreeSession ? worktreeSession.worktreePath : getCwd()\n\n    // Detect available IDEs\n    const detectedIDEs = await detectIDEs(true)\n    const availableIDEs = detectedIDEs.filter(ide => ide.isValid)\n\n    if (availableIDEs.length === 0) {\n      onDone('No IDEs with Claude Code extension detected.')\n      return null\n    }\n\n    // Return IDE selection component\n    return (\n      <IDEOpenSelection\n        availableIDEs={availableIDEs}\n        onSelectIDE={async (selectedIDE?: DetectedIDEInfo) => {\n          if (!selectedIDE) {\n            onDone('No IDE selected.')\n            return\n          }\n\n          // Try to open the project in the selected IDE\n          if (\n            selectedIDE.name.toLowerCase().includes('vscode') ||\n            selectedIDE.name.toLowerCase().includes('cursor') ||\n            selectedIDE.name.toLowerCase().includes('windsurf')\n          ) {\n            // VS Code-based IDEs\n            const { code } = await execFileNoThrow('code', [targetPath])\n            if (code === 0) {\n              onDone(\n                `Opened ${worktreeSession ? 'worktree' : 'project'} in ${chalk.bold(selectedIDE.name)}`,\n              )\n            } else {\n              onDone(\n                `Failed to open in ${selectedIDE.name}. Try opening manually: ${targetPath}`,\n              )\n            }\n          } else if (isSupportedJetBrainsTerminal()) {\n            // JetBrains IDEs - they usually open via their CLI tools\n            onDone(\n              `Please open the ${worktreeSession ? 'worktree' : 'project'} manually in ${chalk.bold(selectedIDE.name)}: ${targetPath}`,\n            )\n          } else {\n            onDone(\n              `Please open the ${worktreeSession ? 'worktree' : 'project'} manually in ${chalk.bold(selectedIDE.name)}: ${targetPath}`,\n            )\n          }\n        }}\n        onDone={() => {\n          onDone('Exited without opening IDE', { display: 'system' })\n        }}\n      />\n    )\n  }\n\n  const detectedIDEs = await detectIDEs(true)\n\n  // If no IDEs with extensions detected, check for running IDEs and offer to install\n  if (\n    detectedIDEs.length === 0 &&\n    context.onInstallIDEExtension &&\n    !isSupportedTerminal()\n  ) {\n    const runningIDEs = await detectRunningIDEs()\n\n    const onInstall = (ide: IdeType) => {\n      if (context.onInstallIDEExtension) {\n        context.onInstallIDEExtension(ide)\n        // The completion message will be shown after installation\n        if (isJetBrainsIde(ide)) {\n          onDone(\n            `Installed plugin to ${chalk.bold(toIDEDisplayName(ide))}\\n` +\n              `Please ${chalk.bold('restart your IDE')} completely for it to take effect`,\n          )\n        } else {\n          onDone(`Installed extension to ${chalk.bold(toIDEDisplayName(ide))}`)\n        }\n      }\n    }\n\n    if (runningIDEs.length > 1) {\n      // Show selector when multiple IDEs are running\n      return (\n        <RunningIDESelector\n          runningIDEs={runningIDEs}\n          onSelectIDE={onInstall}\n          onDone={() => {\n            onDone('No IDE selected.', { display: 'system' })\n          }}\n        />\n      )\n    } else if (runningIDEs.length === 1) {\n      return <InstallOnMount ide={runningIDEs[0]!} onInstall={onInstall} />\n    }\n  }\n\n  const availableIDEs = detectedIDEs.filter(ide => ide.isValid)\n  const unavailableIDEs = detectedIDEs.filter(ide => !ide.isValid)\n\n  const currentIDE = await findCurrentIDE(availableIDEs, dynamicMcpConfig)\n\n  return (\n    <IDECommandFlow\n      availableIDEs={availableIDEs}\n      unavailableIDEs={unavailableIDEs}\n      currentIDE={currentIDE}\n      dynamicMcpConfig={dynamicMcpConfig}\n      onChangeDynamicMcpConfig={onChangeDynamicMcpConfig}\n      onDone={onDone}\n    />\n  )\n}\n\n// Connection timeout slightly longer than the 30s MCP connection timeout\nconst IDE_CONNECTION_TIMEOUT_MS = 35000\n\ntype IDECommandFlowProps = {\n  availableIDEs: DetectedIDEInfo[]\n  unavailableIDEs: DetectedIDEInfo[]\n  currentIDE: DetectedIDEInfo | null\n  dynamicMcpConfig?: Record<string, ScopedMcpServerConfig>\n  onChangeDynamicMcpConfig?: (\n    config: Record<string, ScopedMcpServerConfig>,\n  ) => void\n  onDone: (\n    result?: string,\n    options?: { display?: CommandResultDisplay },\n  ) => void\n}\n\nfunction IDECommandFlow({\n  availableIDEs,\n  unavailableIDEs,\n  currentIDE,\n  dynamicMcpConfig,\n  onChangeDynamicMcpConfig,\n  onDone,\n}: IDECommandFlowProps): React.ReactNode {\n  const [connectingIDE, setConnectingIDE] = useState<DetectedIDEInfo | null>(\n    null,\n  )\n  const ideClient = useAppState(s => s.mcp.clients.find(c => c.name === 'ide'))\n  const setAppState = useSetAppState()\n  const isFirstCheckRef = useRef(true)\n\n  // Watch for connection result\n  useEffect(() => {\n    if (!connectingIDE) return\n    // Skip the first check — it reflects stale state from before the\n    // config change was dispatched\n    if (isFirstCheckRef.current) {\n      isFirstCheckRef.current = false\n      return\n    }\n    if (!ideClient || ideClient.type === 'pending') return\n    if (ideClient.type === 'connected') {\n      onDone(`Connected to ${connectingIDE.name}.`)\n    } else if (ideClient.type === 'failed') {\n      onDone(`Failed to connect to ${connectingIDE.name}.`)\n    }\n  }, [ideClient, connectingIDE, onDone])\n\n  // Timeout fallback\n  useEffect(() => {\n    if (!connectingIDE) return\n    const timer = setTimeout(\n      onDone,\n      IDE_CONNECTION_TIMEOUT_MS,\n      `Connection to ${connectingIDE.name} timed out.`,\n    )\n    return () => clearTimeout(timer)\n  }, [connectingIDE, onDone])\n\n  const handleSelectIDE = useCallback(\n    (selectedIDE?: DetectedIDEInfo) => {\n      if (!onChangeDynamicMcpConfig) {\n        onDone('Error connecting to IDE.')\n        return\n      }\n      const newConfig = { ...(dynamicMcpConfig || {}) }\n      if (currentIDE) {\n        delete newConfig.ide\n      }\n      if (!selectedIDE) {\n        // Close the MCP transport and remove the client from state\n        if (ideClient && ideClient.type === 'connected' && currentIDE) {\n          // Null out onclose to prevent auto-reconnection\n          ideClient.client.onclose = () => {}\n          void clearServerCache('ide', ideClient.config)\n          setAppState(prev => ({\n            ...prev,\n            mcp: {\n              ...prev.mcp,\n              clients: prev.mcp.clients.filter(c => c.name !== 'ide'),\n              tools: prev.mcp.tools.filter(\n                t => !t.name?.startsWith('mcp__ide__'),\n              ),\n              commands: prev.mcp.commands.filter(\n                c => !c.name?.startsWith('mcp__ide__'),\n              ),\n            },\n          }))\n        }\n        onChangeDynamicMcpConfig(newConfig)\n        onDone(\n          currentIDE\n            ? `Disconnected from ${currentIDE.name}.`\n            : 'No IDE selected.',\n        )\n        return\n      }\n      const url = selectedIDE.url\n      newConfig.ide = {\n        type: url.startsWith('ws:') ? 'ws-ide' : 'sse-ide',\n        url: url,\n        ideName: selectedIDE.name,\n        authToken: selectedIDE.authToken,\n        ideRunningInWindows: selectedIDE.ideRunningInWindows,\n        scope: 'dynamic' as const,\n      } as ScopedMcpServerConfig\n      isFirstCheckRef.current = true\n      setConnectingIDE(selectedIDE)\n      onChangeDynamicMcpConfig(newConfig)\n    },\n    [\n      dynamicMcpConfig,\n      currentIDE,\n      ideClient,\n      setAppState,\n      onChangeDynamicMcpConfig,\n      onDone,\n    ],\n  )\n\n  if (connectingIDE) {\n    return <Text dimColor>Connecting to {connectingIDE.name}…</Text>\n  }\n\n  return (\n    <IDEScreen\n      availableIDEs={availableIDEs}\n      unavailableIDEs={unavailableIDEs}\n      selectedIDE={currentIDE}\n      onClose={() => onDone('IDE selection cancelled', { display: 'system' })}\n      onSelect={handleSelectIDE}\n    />\n  )\n}\n\n/**\n * Formats workspace folders for display, stripping cwd and showing tail end of paths\n * @param folders Array of folder paths\n * @param maxLength Maximum total length of the formatted string\n * @returns Formatted string with folder paths\n */\nexport function formatWorkspaceFolders(\n  folders: string[],\n  maxLength: number = 100,\n): string {\n  if (folders.length === 0) return ''\n\n  const cwd = getCwd()\n\n  // Only show first 2 workspaces\n  const foldersToShow = folders.slice(0, 2)\n  const hasMore = folders.length > 2\n\n  // Account for \", …\" if there are more folders\n  const ellipsisOverhead = hasMore ? 3 : 0 // \", …\"\n\n  // Account for commas and spaces between paths (\", \" = 2 chars per separator)\n  const separatorOverhead = (foldersToShow.length - 1) * 2\n  const availableLength = maxLength - separatorOverhead - ellipsisOverhead\n\n  const maxLengthPerPath = Math.floor(availableLength / foldersToShow.length)\n\n  const cwdNFC = cwd.normalize('NFC')\n  const formattedFolders = foldersToShow.map(folder => {\n    // Strip cwd from the beginning if present\n    // Normalize both to NFC for consistent comparison (macOS uses NFD paths)\n    const folderNFC = folder.normalize('NFC')\n    if (folderNFC.startsWith(cwdNFC + path.sep)) {\n      folder = folderNFC.slice(cwdNFC.length + 1)\n    }\n\n    if (folder.length <= maxLengthPerPath) {\n      return folder\n    }\n    return '…' + folder.slice(-(maxLengthPerPath - 1))\n  })\n\n  let result = formattedFolders.join(', ')\n  if (hasMore) {\n    result += ', …'\n  }\n\n  return result\n}\n"],"mappings":"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","ignoreList":[]}