/** * @license * Copyright 2025 Google LLC * SPDX-License-Identifier: Apache-2.0 */ import * as fs from 'node:fs'; import { debugLogger, enableKittyKeyboardProtocol, disableKittyKeyboardProtocol, enableModifyOtherKeys, disableModifyOtherKeys, enableBracketedPasteMode, disableBracketedPasteMode, } from '@google/gemini-cli-core'; import { parseColor } from '../themes/color-utils.js'; export type TerminalBackgroundColor = string | undefined; export class TerminalCapabilityManager { private static instance: TerminalCapabilityManager | undefined; private static readonly KITTY_QUERY = '\x1b[?u'; private static readonly OSC_11_QUERY = '\x1b]11;?\x1b\\'; private static readonly TERMINAL_NAME_QUERY = '\x1b[>q'; private static readonly DEVICE_ATTRIBUTES_QUERY = '\x1b[c'; private static readonly MODIFY_OTHER_KEYS_QUERY = '\x1b[>4;?m'; // Kitty keyboard flags: CSI ? flags u // eslint-disable-next-line no-control-regex private static readonly KITTY_REGEX = /\x1b\[\?(\d+)u/; // Terminal Name/Version response: DCS > | text ST (or BEL) // eslint-disable-next-line no-control-regex private static readonly TERMINAL_NAME_REGEX = /\x1bP>\|(.+?)(\x1b\\|\x07)/; // Primary Device Attributes: CSI ? ID ; ... c // eslint-disable-next-line no-control-regex private static readonly DEVICE_ATTRIBUTES_REGEX = /\x1b\[\?(\d+)(;\d+)*c/; // OSC 11 response: OSC 11 ; rgb:rrrr/gggg/bbbb ST (or BEL) static readonly OSC_11_REGEX = // eslint-disable-next-line no-control-regex /\x1b\]11;rgb:([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})\/([0-9a-fA-F]{1,4})(\x1b\\|\x07)?/; // modifyOtherKeys response: CSI > 4 ; level m // eslint-disable-next-line no-control-regex private static readonly MODIFY_OTHER_KEYS_REGEX = /\x1b\[>4;(\d+)m/; private detectionComplete = false; private terminalBackgroundColor: TerminalBackgroundColor; private kittySupported = false; private kittyEnabled = false; private modifyOtherKeysSupported = false; private terminalName: string | undefined; private constructor() {} static getInstance(): TerminalCapabilityManager { if (!this.instance) { this.instance = new TerminalCapabilityManager(); } return this.instance; } static resetInstanceForTesting(): void { this.instance = undefined; } /** * Detects terminal capabilities (Kitty protocol support, terminal name, * background color). * This should be called once at app startup. */ async detectCapabilities(): Promise { if (this.detectionComplete) return; if (!process.stdin.isTTY || !process.stdout.isTTY) { this.detectionComplete = true; return; } const cleanupOnExit = () => { // don't bother catching errors since if one write // fails, the other probably will too disableKittyKeyboardProtocol(); disableModifyOtherKeys(); disableBracketedPasteMode(); }; process.on('exit', cleanupOnExit); process.on('SIGTERM', cleanupOnExit); process.on('SIGINT', cleanupOnExit); return new Promise((resolve) => { const originalRawMode = process.stdin.isRaw; if (!originalRawMode) { process.stdin.setRawMode(true); } let buffer = ''; let kittyKeyboardReceived = false; let terminalNameReceived = false; let deviceAttributesReceived = false; let bgReceived = false; let modifyOtherKeysReceived = false; // eslint-disable-next-line prefer-const let timeoutId: NodeJS.Timeout; const cleanup = () => { if (timeoutId) { clearTimeout(timeoutId); } process.stdin.removeListener('data', onData); if (!originalRawMode) { process.stdin.setRawMode(false); } this.detectionComplete = true; this.enableSupportedModes(); resolve(); }; // A somewhat long timeout is acceptable as all terminals should respond // to the device attributes query used as a sentinel. timeoutId = setTimeout(cleanup, 1000); const onData = (data: Buffer) => { buffer += data.toString(); // Check OSC 11 if (!bgReceived) { const match = buffer.match(TerminalCapabilityManager.OSC_11_REGEX); if (match) { bgReceived = true; this.terminalBackgroundColor = parseColor( match[1], match[2], match[3], ); debugLogger.log( `Detected terminal background color: ${this.terminalBackgroundColor}`, ); } } if ( !kittyKeyboardReceived && TerminalCapabilityManager.KITTY_REGEX.test(buffer) ) { kittyKeyboardReceived = true; this.kittySupported = true; } // check for modifyOtherKeys support if (!modifyOtherKeysReceived) { const match = buffer.match( TerminalCapabilityManager.MODIFY_OTHER_KEYS_REGEX, ); if (match) { modifyOtherKeysReceived = true; const level = parseInt(match[1], 10); this.modifyOtherKeysSupported = level >= 2; debugLogger.log( `Detected modifyOtherKeys support: ${this.modifyOtherKeysSupported} (level ${level})`, ); } } // Check for Terminal Name/Version response. if (!terminalNameReceived) { const match = buffer.match( TerminalCapabilityManager.TERMINAL_NAME_REGEX, ); if (match) { terminalNameReceived = true; this.terminalName = match[1]; debugLogger.log(`Detected terminal name: ${this.terminalName}`); } } // We use the Primary Device Attributes response as a sentinel to know // that the terminal has processed all our queries. Since we send it // last, receiving it means we can stop waiting. if (!deviceAttributesReceived) { const match = buffer.match( TerminalCapabilityManager.DEVICE_ATTRIBUTES_REGEX, ); if (match) { deviceAttributesReceived = true; cleanup(); } } }; process.stdin.on('data', onData); try { fs.writeSync( process.stdout.fd, TerminalCapabilityManager.KITTY_QUERY + TerminalCapabilityManager.OSC_11_QUERY + TerminalCapabilityManager.TERMINAL_NAME_QUERY + TerminalCapabilityManager.MODIFY_OTHER_KEYS_QUERY + TerminalCapabilityManager.DEVICE_ATTRIBUTES_QUERY, ); } catch (e) { debugLogger.warn('Failed to write terminal capability queries:', e); cleanup(); } }); } enableSupportedModes() { try { if (this.kittySupported) { enableKittyKeyboardProtocol(); this.kittyEnabled = true; } else if (this.modifyOtherKeysSupported) { enableModifyOtherKeys(); } // Always enable bracketed paste since it'll be ignored if unsupported. enableBracketedPasteMode(); } catch (e) { debugLogger.warn('Failed to enable keyboard protocols:', e); } } getTerminalBackgroundColor(): TerminalBackgroundColor { return this.terminalBackgroundColor; } getTerminalName(): string | undefined { return this.terminalName; } isKittyProtocolEnabled(): boolean { return this.kittyEnabled; } } export const terminalCapabilityManager = TerminalCapabilityManager.getInstance();