/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import { Config, DetectedIde, IDEConnectionStatus, getIdeDisplayName, getIdeInstaller, } from '@google/gemini-cli-core'; import { CommandContext, SlashCommand, SlashCommandActionReturn, CommandKind, } from './types.js'; import { SettingScope } from '../../config/settings.js'; export const ideCommand = (config: Config | null): SlashCommand | null => { if (!config || !config.getIdeModeFeature()) { return null; } const ideClient = config.getIdeClient(); const currentIDE = ideClient.getCurrentIde(); if (!currentIDE || !ideClient.getDetectedIdeDisplayName()) { return { name: 'ide', description: 'manage IDE integration', kind: CommandKind.BUILT_IN, action: (): SlashCommandActionReturn => ({ type: 'message', messageType: 'error', content: `IDE integration is not supported in your current environment. To use this feature, run Gemini CLI in one of these supported IDEs: ${Object.values( DetectedIde, ) .map((ide) => getIdeDisplayName(ide)) .join(', ')}`, }) as const, }; } const ideSlashCommand: SlashCommand = { name: 'ide', description: 'manage IDE integration', kind: CommandKind.BUILT_IN, subCommands: [], }; const statusCommand: SlashCommand = { name: 'status', description: 'check status of IDE integration', kind: CommandKind.BUILT_IN, action: (_context: CommandContext): SlashCommandActionReturn => { const connection = ideClient.getConnectionStatus(); switch (connection.status) { case IDEConnectionStatus.Connected: return { type: 'message', messageType: 'info', content: `🟢 Connected to ${ideClient.getDetectedIdeDisplayName()}`, } as const; case IDEConnectionStatus.Connecting: return { type: 'message', messageType: 'info', content: `🟡 Connecting...`, } as const; default: { let content = `🔴 Disconnected`; if (connection?.details) { content += `: ${connection.details}`; } return { type: 'message', messageType: 'error', content, } as const; } } }, }; const installCommand: SlashCommand = { name: 'install', description: `install required IDE companion for ${ideClient.getDetectedIdeDisplayName()}`, kind: CommandKind.BUILT_IN, action: async (context) => { const installer = getIdeInstaller(currentIDE); if (!installer) { context.ui.addItem( { type: 'error', text: `No installer is available for ${ideClient.getDetectedIdeDisplayName()}. Please install the IDE companion manually from its marketplace.`, }, Date.now(), ); return; } context.ui.addItem( { type: 'info', text: `Installing IDE companion...`, }, Date.now(), ); const result = await installer.install(); context.ui.addItem( { type: result.success ? 'info' : 'error', text: result.message, }, Date.now(), ); }, }; const enableCommand: SlashCommand = { name: 'enable', description: 'enable IDE integration', kind: CommandKind.BUILT_IN, action: async (context: CommandContext) => { context.services.settings.setValue(SettingScope.User, 'ideMode', true); config.setIdeMode(true); config.setIdeClientConnected(); }, }; const disableCommand: SlashCommand = { name: 'disable', description: 'disable IDE integration', kind: CommandKind.BUILT_IN, action: async (context: CommandContext) => { context.services.settings.setValue(SettingScope.User, 'ideMode', false); config.setIdeMode(false); config.setIdeClientDisconnected(); }, }; const ideModeEnabled = config.getIdeMode(); if (ideModeEnabled) { ideSlashCommand.subCommands = [ disableCommand, statusCommand, installCommand, ]; } else { ideSlashCommand.subCommands = [ enableCommand, statusCommand, installCommand, ]; } return ideSlashCommand; };