/ commands / ide / ide.tsx
ide.tsx
  1  import { c as _c } from "react/compiler-runtime";
  2  import chalk from 'chalk';
  3  import * as path from 'path';
  4  import React, { useCallback, useEffect, useRef, useState } from 'react';
  5  import { logEvent } from 'src/services/analytics/index.js';
  6  import type { CommandResultDisplay, LocalJSXCommandContext } from '../../commands.js';
  7  import { Select } from '../../components/CustomSelect/index.js';
  8  import { Dialog } from '../../components/design-system/Dialog.js';
  9  import { IdeAutoConnectDialog, IdeDisableAutoConnectDialog, shouldShowAutoConnectDialog, shouldShowDisableAutoConnectDialog } from '../../components/IdeAutoConnectDialog.js';
 10  import { Box, Text } from '../../ink.js';
 11  import { clearServerCache } from '../../services/mcp/client.js';
 12  import type { ScopedMcpServerConfig } from '../../services/mcp/types.js';
 13  import { useAppState, useSetAppState } from '../../state/AppState.js';
 14  import { getCwd } from '../../utils/cwd.js';
 15  import { execFileNoThrow } from '../../utils/execFileNoThrow.js';
 16  import { type DetectedIDEInfo, detectIDEs, detectRunningIDEs, type IdeType, isJetBrainsIde, isSupportedJetBrainsTerminal, isSupportedTerminal, toIDEDisplayName } from '../../utils/ide.js';
 17  import { getCurrentWorktreeSession } from '../../utils/worktree.js';
 18  type IDEScreenProps = {
 19    availableIDEs: DetectedIDEInfo[];
 20    unavailableIDEs: DetectedIDEInfo[];
 21    selectedIDE?: DetectedIDEInfo | null;
 22    onClose: () => void;
 23    onSelect: (ide?: DetectedIDEInfo) => void;
 24  };
 25  function 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  }
198  function _temp3(ide_3, index) {
199    return <Box key={index} paddingLeft={3}><Text dimColor={true}>• {ide_3.name}: {formatWorkspaceFolders(ide_3.workspaceFolders)}</Text></Box>;
200  }
201  function _temp2(ide_2) {
202    return ide_2.name === "VS Code" || ide_2.name === "Visual Studio Code";
203  }
204  function _temp(acc, ide_0) {
205    acc[ide_0.name] = (acc[ide_0.name] || 0) + 1;
206    return acc;
207  }
208  async 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  }
220  type IDEOpenSelectionProps = {
221    availableIDEs: DetectedIDEInfo[];
222    onSelectIDE: (ide?: DetectedIDEInfo) => void;
223    onDone: (result?: string, options?: {
224      display?: CommandResultDisplay;
225    }) => void;
226  };
227  function 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  }
310  function _temp4(ide_0) {
311    return {
312      label: ide_0.name,
313      value: ide_0.port.toString()
314    };
315  }
316  function 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  }
389  function _temp5(ide) {
390    return {
391      label: toIDEDisplayName(ide),
392      value: ide
393    };
394  }
395  function 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  }
419  export 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
507  const IDE_CONNECTION_TIMEOUT_MS = 35000;
508  type 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  };
518  function 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   */
612  export 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":[]}