π File detail
keybindings/defaultBindings.ts
π§© .tsπ 341 linesπΎ 11,635 bytesπ text
β Back to All Filesπ― Use case
This file lives under βkeybindings/β, which covers keyboard shortcuts and binding tables. On the API surface it exposes DEFAULT_BINDINGS β mainly types, interfaces, or factory objects. Dependencies touch bun:bundle and src. It composes internal code from utils and types (relative imports).
Generated from folder role, exports, dependency roots, and inline comments β not hand-reviewed for every path.
π§ Inline summary
import { feature } from 'bun:bundle' import { satisfies } from 'src/utils/semver.js' import { isRunningWithBun } from '../utils/bundledMode.js' import { getPlatform } from '../utils/platform.js' import type { KeybindingBlock } from './types.js'
π€ Exports (heuristic)
DEFAULT_BINDINGS
π External import roots
Package roots from from "β¦" (relative paths omitted).
bun:bundlesrc
π₯οΈ Source preview
import { feature } from 'bun:bundle'
import { satisfies } from 'src/utils/semver.js'
import { isRunningWithBun } from '../utils/bundledMode.js'
import { getPlatform } from '../utils/platform.js'
import type { KeybindingBlock } from './types.js'
/**
* Default keybindings that match current Claude Code behavior.
* These are loaded first, then user keybindings.json overrides them.
*/
// Platform-specific image paste shortcut:
// - Windows: alt+v (ctrl+v is system paste)
// - Other platforms: ctrl+v
const IMAGE_PASTE_KEY = getPlatform() === 'windows' ? 'alt+v' : 'ctrl+v'
// Modifier-only chords (like shift+tab) may fail on Windows Terminal without VT mode
// See: https://github.com/microsoft/terminal/issues/879#issuecomment-618801651
// Node enabled VT mode in 24.2.0 / 22.17.0: https://github.com/nodejs/node/pull/58358
// Bun enabled VT mode in 1.2.23: https://github.com/oven-sh/bun/pull/21161
const SUPPORTS_TERMINAL_VT_MODE =
getPlatform() !== 'windows' ||
(isRunningWithBun()
? satisfies(process.versions.bun, '>=1.2.23')
: satisfies(process.versions.node, '>=22.17.0 <23.0.0 || >=24.2.0'))
// Platform-specific mode cycle shortcut:
// - Windows without VT mode: meta+m (shift+tab doesn't work reliably)
// - Other platforms: shift+tab
const MODE_CYCLE_KEY = SUPPORTS_TERMINAL_VT_MODE ? 'shift+tab' : 'meta+m'
export const DEFAULT_BINDINGS: KeybindingBlock[] = [
{
context: 'Global',
bindings: {
// ctrl+c and ctrl+d use special time-based double-press handling.
// They ARE defined here so the resolver can find them, but they
// CANNOT be rebound by users - validation in reservedShortcuts.ts
// will show an error if users try to override these keys.
'ctrl+c': 'app:interrupt',
'ctrl+d': 'app:exit',
'ctrl+l': 'app:redraw',
'ctrl+t': 'app:toggleTodos',
'ctrl+o': 'app:toggleTranscript',
...(feature('KAIROS') || feature('KAIROS_BRIEF')
? { 'ctrl+shift+b': 'app:toggleBrief' as const }
: {}),
'ctrl+shift+o': 'app:toggleTeammatePreview',
'ctrl+r': 'history:search',
// File navigation. cmd+ bindings only fire on kitty-protocol terminals;
// ctrl+shift is the portable fallback.
...(feature('QUICK_SEARCH')
? {
'ctrl+shift+f': 'app:globalSearch' as const,
'cmd+shift+f': 'app:globalSearch' as const,
'ctrl+shift+p': 'app:quickOpen' as const,
'cmd+shift+p': 'app:quickOpen' as const,
}
: {}),
...(feature('TERMINAL_PANEL') ? { 'meta+j': 'app:toggleTerminal' } : {}),
},
},
{
context: 'Chat',
bindings: {
escape: 'chat:cancel',
// ctrl+x chord prefix avoids shadowing readline editing keys (ctrl+a/b/e/f/...).
'ctrl+x ctrl+k': 'chat:killAgents',
[MODE_CYCLE_KEY]: 'chat:cycleMode',
'meta+p': 'chat:modelPicker',
'meta+o': 'chat:fastMode',
'meta+t': 'chat:thinkingToggle',
enter: 'chat:submit',
up: 'history:previous',
down: 'history:next',
// Editing shortcuts (defined here, migration in progress)
// Undo has two bindings to support different terminal behaviors:
// - ctrl+_ for legacy terminals (send \x1f control char)
// - ctrl+shift+- for Kitty protocol (sends physical key with modifiers)
'ctrl+_': 'chat:undo',
'ctrl+shift+-': 'chat:undo',
// ctrl+x ctrl+e is the readline-native edit-and-execute-command binding.
'ctrl+x ctrl+e': 'chat:externalEditor',
'ctrl+g': 'chat:externalEditor',
'ctrl+s': 'chat:stash',
// Image paste shortcut (platform-specific key defined above)
[IMAGE_PASTE_KEY]: 'chat:imagePaste',
...(feature('MESSAGE_ACTIONS')
? { 'shift+up': 'chat:messageActions' as const }
: {}),
// Voice activation (hold-to-talk). Registered so getShortcutDisplay
// finds it without hitting the fallback analytics log. To rebind,
// add a voice:pushToTalk entry (last wins); to disable, use /voice
// β null-unbinding space hits a pre-existing useKeybinding.ts trap
// where 'unbound' swallows the event (space dead for typing).
...(feature('VOICE_MODE') ? { space: 'voice:pushToTalk' } : {}),
},
},
{
context: 'Autocomplete',
bindings: {
tab: 'autocomplete:accept',
escape: 'autocomplete:dismiss',
up: 'autocomplete:previous',
down: 'autocomplete:next',
},
},
{
context: 'Settings',
bindings: {
// Settings menu uses escape only (not 'n') to dismiss
escape: 'confirm:no',
// Config panel list navigation (reuses Select actions)
up: 'select:previous',
down: 'select:next',
k: 'select:previous',
j: 'select:next',
'ctrl+p': 'select:previous',
'ctrl+n': 'select:next',
// Toggle/activate the selected setting (space only β enter saves & closes)
space: 'select:accept',
// Save and close the config panel
enter: 'settings:close',
// Enter search mode
'/': 'settings:search',
// Retry loading usage data (only active on error)
r: 'settings:retry',
},
},
{
context: 'Confirmation',
bindings: {
y: 'confirm:yes',
n: 'confirm:no',
enter: 'confirm:yes',
escape: 'confirm:no',
// Navigation for dialogs with lists
up: 'confirm:previous',
down: 'confirm:next',
tab: 'confirm:nextField',
space: 'confirm:toggle',
// Cycle modes (used in file permission dialogs and teams dialog)
'shift+tab': 'confirm:cycleMode',
// Toggle permission explanation in permission dialogs
'ctrl+e': 'confirm:toggleExplanation',
// Toggle permission debug info
'ctrl+d': 'permission:toggleDebug',
},
},
{
context: 'Tabs',
bindings: {
// Tab cycling navigation
tab: 'tabs:next',
'shift+tab': 'tabs:previous',
right: 'tabs:next',
left: 'tabs:previous',
},
},
{
context: 'Transcript',
bindings: {
'ctrl+e': 'transcript:toggleShowAll',
'ctrl+c': 'transcript:exit',
escape: 'transcript:exit',
// q β pager convention (less, tmux copy-mode). Transcript is a modal
// reading view with no prompt, so q-as-literal-char has no owner.
q: 'transcript:exit',
},
},
{
context: 'HistorySearch',
bindings: {
'ctrl+r': 'historySearch:next',
escape: 'historySearch:accept',
tab: 'historySearch:accept',
'ctrl+c': 'historySearch:cancel',
enter: 'historySearch:execute',
},
},
{
context: 'Task',
bindings: {
// Background running foreground tasks (bash commands, agents)
// In tmux, users must press ctrl+b twice (tmux prefix escape)
'ctrl+b': 'task:background',
},
},
{
context: 'ThemePicker',
bindings: {
'ctrl+t': 'theme:toggleSyntaxHighlighting',
},
},
{
context: 'Scroll',
bindings: {
pageup: 'scroll:pageUp',
pagedown: 'scroll:pageDown',
wheelup: 'scroll:lineUp',
wheeldown: 'scroll:lineDown',
'ctrl+home': 'scroll:top',
'ctrl+end': 'scroll:bottom',
// Selection copy. ctrl+shift+c is standard terminal copy.
// cmd+c only fires on terminals using the kitty keyboard
// protocol (kitty/WezTerm/ghostty/iTerm2) where the super
// modifier actually reaches the pty β inert elsewhere.
// Esc-to-clear and contextual ctrl+c are handled via raw
// useInput so they can conditionally propagate.
'ctrl+shift+c': 'selection:copy',
'cmd+c': 'selection:copy',
},
},
{
context: 'Help',
bindings: {
escape: 'help:dismiss',
},
},
// Attachment navigation (select dialog image attachments)
{
context: 'Attachments',
bindings: {
right: 'attachments:next',
left: 'attachments:previous',
backspace: 'attachments:remove',
delete: 'attachments:remove',
down: 'attachments:exit',
escape: 'attachments:exit',
},
},
// Footer indicator navigation (tasks, teams, diff, loop)
{
context: 'Footer',
bindings: {
up: 'footer:up',
'ctrl+p': 'footer:up',
down: 'footer:down',
'ctrl+n': 'footer:down',
right: 'footer:next',
left: 'footer:previous',
enter: 'footer:openSelected',
escape: 'footer:clearSelection',
},
},
// Message selector (rewind dialog) navigation
{
context: 'MessageSelector',
bindings: {
up: 'messageSelector:up',
down: 'messageSelector:down',
k: 'messageSelector:up',
j: 'messageSelector:down',
'ctrl+p': 'messageSelector:up',
'ctrl+n': 'messageSelector:down',
'ctrl+up': 'messageSelector:top',
'shift+up': 'messageSelector:top',
'meta+up': 'messageSelector:top',
'shift+k': 'messageSelector:top',
'ctrl+down': 'messageSelector:bottom',
'shift+down': 'messageSelector:bottom',
'meta+down': 'messageSelector:bottom',
'shift+j': 'messageSelector:bottom',
enter: 'messageSelector:select',
},
},
// PromptInput unmounts while cursor active β no key conflict.
...(feature('MESSAGE_ACTIONS')
? [
{
context: 'MessageActions' as const,
bindings: {
up: 'messageActions:prev' as const,
down: 'messageActions:next' as const,
k: 'messageActions:prev' as const,
j: 'messageActions:next' as const,
// meta = cmd on macOS; super for kitty keyboard-protocol β bind both.
'meta+up': 'messageActions:top' as const,
'meta+down': 'messageActions:bottom' as const,
'super+up': 'messageActions:top' as const,
'super+down': 'messageActions:bottom' as const,
// Mouse selection extends on shift+arrow (ScrollKeybindingHandler:573) when present β
// correct layered UX: esc clears selection, then shift+β jumps.
'shift+up': 'messageActions:prevUser' as const,
'shift+down': 'messageActions:nextUser' as const,
escape: 'messageActions:escape' as const,
'ctrl+c': 'messageActions:ctrlc' as const,
// Mirror MESSAGE_ACTIONS. Not imported β would pull React/ink into this config module.
enter: 'messageActions:enter' as const,
c: 'messageActions:c' as const,
p: 'messageActions:p' as const,
},
},
]
: []),
// Diff dialog navigation
{
context: 'DiffDialog',
bindings: {
escape: 'diff:dismiss',
left: 'diff:previousSource',
right: 'diff:nextSource',
up: 'diff:previousFile',
down: 'diff:nextFile',
enter: 'diff:viewDetails',
// Note: diff:back is handled by left arrow in detail mode
},
},
// Model picker effort cycling (ant-only)
{
context: 'ModelPicker',
bindings: {
left: 'modelPicker:decreaseEffort',
right: 'modelPicker:increaseEffort',
},
},
// Select component navigation (used by /model, /resume, permission prompts, etc.)
{
context: 'Select',
bindings: {
up: 'select:previous',
down: 'select:next',
j: 'select:next',
k: 'select:previous',
'ctrl+n': 'select:next',
'ctrl+p': 'select:previous',
enter: 'select:accept',
escape: 'select:cancel',
},
},
// Plugin dialog actions (manage, browse, discover plugins)
// Navigation (select:*) uses the Select context above
{
context: 'Plugin',
bindings: {
space: 'plugin:toggle',
i: 'plugin:install',
},
},
]