203 lines
8.4 KiB
TypeScript
203 lines
8.4 KiB
TypeScript
import { spawnSync } from 'child_process'
|
|
import { getIsInteractive } from '../bootstrap/state.js'
|
|
import { logForDebugging } from './debug.js'
|
|
import { isEnvDefinedFalsy, isEnvTruthy } from './envUtils.js'
|
|
import { execFileNoThrow } from './execFileNoThrow.js'
|
|
|
|
let loggedTmuxCcDisable = false
|
|
let checkedTmuxMouseHint = false
|
|
|
|
/**
|
|
* Cached result from `tmux display-message -p '#{client_control_mode}'`.
|
|
* undefined = not yet queried (or probe failed) — env heuristic stays authoritative.
|
|
*/
|
|
let tmuxControlModeProbed: boolean | undefined
|
|
|
|
/**
|
|
* Env-var heuristic for iTerm2's tmux integration mode (`tmux -CC` / `tmux -2CC`).
|
|
*
|
|
* In `-CC` mode, iTerm2 renders tmux panes as native splits — tmux runs
|
|
* as a server (TMUX is set) but iTerm2 is the actual terminal emulator
|
|
* for each pane, so TERM_PROGRAM stays `iTerm.app` and TERM is iTerm2's
|
|
* default (xterm-*). Contrast with regular tmux-inside-iTerm2, where tmux
|
|
* overwrites TERM_PROGRAM to `tmux` and sets TERM to screen-* or tmux-*.
|
|
*
|
|
* This heuristic has known holes (SSH often doesn't propagate TERM_PROGRAM;
|
|
* .tmux.conf can override TERM) — probeTmuxControlModeSync() is the
|
|
* authoritative backstop. Kept as a zero-subprocess fast path.
|
|
*/
|
|
function isTmuxControlModeEnvHeuristic(): boolean {
|
|
if (!process.env.TMUX) return false
|
|
if (process.env.TERM_PROGRAM !== 'iTerm.app') return false
|
|
// Belt-and-suspenders: in regular tmux TERM is screen-* or tmux-*;
|
|
// in -CC mode iTerm2 sets its own TERM (xterm-*).
|
|
const term = process.env.TERM ?? ''
|
|
return !term.startsWith('screen') && !term.startsWith('tmux')
|
|
}
|
|
|
|
/**
|
|
* Sync one-shot probe: asks tmux directly whether this client is in control
|
|
* mode via `#{client_control_mode}`. Runs on first isTmuxControlMode() call
|
|
* when the env heuristic can't decide; result is cached.
|
|
*
|
|
* Sync (spawnSync) because the answer gates whether we enter fullscreen — an
|
|
* async probe raced against React render and lost: coder-tmux (ssh → tmux -CC
|
|
* on a remote box) doesn't propagate TERM_PROGRAM, so the env heuristic missed,
|
|
* and by the time the async probe resolved we'd already entered alt-screen with
|
|
* mouse tracking enabled. Mouse wheel is dead in iTerm2's -CC integration, so
|
|
* users couldn't scroll at all.
|
|
*
|
|
* Cost: one ~5ms subprocess, only when $TMUX is set AND $TERM_PROGRAM is unset
|
|
* (the SSH-into-tmux case). Local iTerm2 -CC and non-tmux paths skip the spawn.
|
|
*
|
|
* The TMUX env check MUST come first — without it, display-message would
|
|
* query whatever tmux server happens to be running rather than our client.
|
|
*/
|
|
function probeTmuxControlModeSync(): void {
|
|
// Seed cache with heuristic result so early returns below don't leave it
|
|
// undefined — isTmuxControlMode() is called 15+ times per render, and an
|
|
// undefined cache would re-enter this function (re-spawning tmux in the
|
|
// failure case) on every call.
|
|
tmuxControlModeProbed = isTmuxControlModeEnvHeuristic()
|
|
if (tmuxControlModeProbed) return
|
|
if (!process.env.TMUX) return
|
|
// Only probe when iTerm might be involved: TERM_PROGRAM is iTerm.app
|
|
// (covered above) or not set (SSH often doesn't propagate it). When
|
|
// TERM_PROGRAM is explicitly a non-iTerm terminal, skip — tmux -CC is
|
|
// an iTerm-only feature, so the subprocess would be wasted.
|
|
if (process.env.TERM_PROGRAM) return
|
|
let result
|
|
try {
|
|
result = spawnSync(
|
|
'tmux',
|
|
['display-message', '-p', '#{client_control_mode}'],
|
|
{ encoding: 'utf8', timeout: 2000 },
|
|
)
|
|
} catch {
|
|
// spawnSync can throw on some platforms (e.g. ENOENT on Windows if tmux
|
|
// is absent and the runtime surfaces it as an exception rather than in
|
|
// result.error). Treat the same as a non-zero exit.
|
|
return
|
|
}
|
|
// Non-zero exit / spawn error: tmux too old (format var added in 2.4) or
|
|
// unavailable. Keep the heuristic result cached.
|
|
if (result.status !== 0) return
|
|
tmuxControlModeProbed = result.stdout.trim() === '1'
|
|
}
|
|
|
|
/**
|
|
* True when running under `tmux -CC` (iTerm2 integration mode).
|
|
*
|
|
* The alt-screen / mouse-tracking path in fullscreen mode is unrecoverable
|
|
* in -CC mode (double-click corrupts terminal state; mouse wheel is dead),
|
|
* so callers auto-disable fullscreen.
|
|
*
|
|
* Lazily probes tmux on first call when the env heuristic can't decide.
|
|
*/
|
|
export function isTmuxControlMode(): boolean {
|
|
if (tmuxControlModeProbed === undefined) probeTmuxControlModeSync()
|
|
return tmuxControlModeProbed ?? false
|
|
}
|
|
|
|
export function _resetTmuxControlModeProbeForTesting(): void {
|
|
tmuxControlModeProbed = undefined
|
|
loggedTmuxCcDisable = false
|
|
}
|
|
|
|
/**
|
|
* Runtime env-var check only. Ants default to on (CLAUDE_CODE_NO_FLICKER=0
|
|
* to opt out); external users default to off (CLAUDE_CODE_NO_FLICKER=1 to
|
|
* opt in).
|
|
*/
|
|
export function isFullscreenEnvEnabled(): boolean {
|
|
// Explicit user opt-out always wins.
|
|
if (isEnvDefinedFalsy(process.env.CLAUDE_CODE_NO_FLICKER)) return false
|
|
// Explicit opt-in overrides auto-detection (escape hatch).
|
|
if (isEnvTruthy(process.env.CLAUDE_CODE_NO_FLICKER)) return true
|
|
// Auto-disable under tmux -CC: alt-screen + mouse tracking corrupts
|
|
// terminal state on double-click and mouse wheel is dead.
|
|
if (isTmuxControlMode()) {
|
|
if (!loggedTmuxCcDisable) {
|
|
loggedTmuxCcDisable = true
|
|
logForDebugging(
|
|
'fullscreen disabled: tmux -CC (iTerm2 integration mode) detected · set CLAUDE_CODE_NO_FLICKER=1 to override',
|
|
)
|
|
}
|
|
return false
|
|
}
|
|
return process.env.USER_TYPE === 'ant'
|
|
}
|
|
|
|
/**
|
|
* Whether fullscreen mode should enable SGR mouse tracking (DEC 1000/1002/1006).
|
|
* Set CLAUDE_CODE_DISABLE_MOUSE=1 to keep alt-screen + virtualized scroll
|
|
* (keyboard PgUp/PgDn/Ctrl+Home/End still work) but skip mouse capture,
|
|
* so tmux/kitty/terminal-native copy-on-select keeps working.
|
|
*
|
|
* Compare with CLAUDE_CODE_NO_FLICKER=0 which is all-or-nothing — it also
|
|
* disables alt-screen and virtualized scrollback.
|
|
*/
|
|
export function isMouseTrackingEnabled(): boolean {
|
|
return !isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_MOUSE)
|
|
}
|
|
|
|
/**
|
|
* Whether mouse click handling is disabled (clicks/drags ignored, wheel still
|
|
* works). Set CLAUDE_CODE_DISABLE_MOUSE_CLICKS=1 to prevent accidental clicks
|
|
* from triggering cursor positioning, text selection, or message expansion.
|
|
*
|
|
* Fullscreen-specific — only reachable when CLAUDE_CODE_NO_FLICKER is active.
|
|
*/
|
|
export function isMouseClicksDisabled(): boolean {
|
|
return isEnvTruthy(process.env.CLAUDE_CODE_DISABLE_MOUSE_CLICKS)
|
|
}
|
|
|
|
/**
|
|
* True when the fullscreen alt-screen layout is actually rendering —
|
|
* requires an interactive REPL session AND the env var not explicitly
|
|
* set falsy. Headless paths (--print, SDK, in-process teammates) never
|
|
* enter fullscreen, so features that depend on alt-screen re-rendering
|
|
* should gate on this.
|
|
*/
|
|
export function isFullscreenActive(): boolean {
|
|
return getIsInteractive() && isFullscreenEnvEnabled()
|
|
}
|
|
|
|
/**
|
|
* One-time hint for tmux users in fullscreen with `mouse off`.
|
|
*
|
|
* tmux's `mouse` option is session-scoped by design — there is no
|
|
* pane-level equivalent. We used to `tmux set mouse on` when entering
|
|
* alt-screen so wheel scrolling worked, but that changed mouse behavior
|
|
* for every sibling pane (vim, less, shell) and leaked on kill-pane or
|
|
* when multiple CC instances raced on restore. Now we leave tmux state
|
|
* alone — same as vim/less/htop — and just tell the user their options.
|
|
*
|
|
* Fire-and-forget from REPL startup. Returns the hint text once per
|
|
* session if TMUX is set, fullscreen is active, and tmux's current
|
|
* `mouse` option is off; null otherwise.
|
|
*/
|
|
export async function maybeGetTmuxMouseHint(): Promise<string | null> {
|
|
if (!process.env.TMUX) return null
|
|
// tmux -CC auto-disables fullscreen above, but belt-and-suspenders.
|
|
if (!isFullscreenActive() || isTmuxControlMode()) return null
|
|
if (checkedTmuxMouseHint) return null
|
|
checkedTmuxMouseHint = true
|
|
// -A includes inherited values: `show -v mouse` returns empty when the
|
|
// option is set globally (`set -g mouse on` in .tmux.conf) but not at
|
|
// session level — which is the common case. -A gives the effective value.
|
|
const { stdout, code } = await execFileNoThrow(
|
|
'tmux',
|
|
['show', '-Av', 'mouse'],
|
|
{ useCwd: false, timeout: 2000 },
|
|
)
|
|
if (code !== 0 || stdout.trim() === 'on') return null
|
|
return "tmux detected · scroll with PgUp/PgDn · or add 'set -g mouse on' to ~/.tmux.conf for wheel scroll"
|
|
}
|
|
|
|
/** Test-only: reset module-level once-per-session flags. */
|
|
export function _resetForTesting(): void {
|
|
loggedTmuxCcDisable = false
|
|
checkedTmuxMouseHint = false
|
|
}
|