tmux-control turns Emacs into a control-mode client for a tmux pane —
the iTerm2 tmux-integration
idea, but in Emacs.
A live tmux session in Emacs. Each window is a tab in the header line,
flipped with one key (C-c C-n); a dot marks a background window that
produced output — the "which window wants me?" signal — and clears when you
visit it. Every pane is just an Emacs buffer you can search and copy from.
Other ways to pair Emacs with tmux either send it commands
(emamux), navigate between
Emacs windows and tmux panes when Emacs itself runs inside tmux
(tmux-pane), or run tmux
inside an Emacs terminal buffer (vterm, eat, ansi-term) — a terminal in
a terminal, with tmux's own status bar and prefix keys. None of them render
tmux's own panes as Emacs buffers. tmux-control does: it speaks tmux's
control-mode protocol (tmux -C, the same one iTerm2's native integration
uses), so each live pane becomes an Emacs buffer rendered through
Eat — no nested terminal, no tmux
chrome, just a buffer you navigate, search, and copy from. The session lives
on a persistent, possibly remote server and outlives Emacs: detach,
restart, or reconnect from another machine and the pane is still there.
- Emacs 29.1+
- Eat 0.9.4+ — the terminal renderer
and a hard dependency (
straight/package.elpulls it in automatically). - tmux 3.x on the target host, local or remote over SSH (flow control needs 3.2+).
- macOS or Linux.
(use-package tmux-control
:straight (tmux-control :type git :host github :repo "csheaff/tmux-control")
:custom
;; Connection defaults for `M-x tmux-control-connect' — these are examples;
;; set them to your own host / socket / session.
(tmux-control-default-host "dev") ; an SSH host alias, or nil for local
(tmux-control-default-socket-name "main")
(tmux-control-default-session "emacs"))Then M-x tmux-control-connect. The session prompt completes over the
sessions that already exist on the chosen host and socket; selecting one
attaches, typing a new name creates it.
- Live view of a tmux window, rendered through Eat — a normal Emacs buffer you move, search, and copy in, with no nested terminal and no tmux chrome.
- Windows as tabs in a header-line tab bar, with an activity dot on
background windows; flip them like browser tabs (
C-c C-n/C-c C-p). - Switch between sessions on the host in place (
C-c C-s) — each tmux session is its own buffer with its own scrollback and tabs — with a dot flagging another session that wants you, or tile every session at once in a live grid (C-c C-f, experimental). - Scrollback as ordinary Emacs text (
C-c C-e), auto-collapsing repeated full-screen redraws. - Tiled view (
C-c C-t, experimental) — every pane of a window at once, split to match tmux's layout. - Persistent & remote — the session lives on the server and outlives Emacs; detach, restart, reconnect from another machine.
- Stays responsive under a flood of output (optional flow control).
Each connected tmux session is its own buffer; a header-line dot flags the ones
with new output, and C-c C-f tiles them all into one live grid (experimental).
→ Full command and key reference, the tiled view, and tuning live in docs/guide.md.
The single-pane client is stable and in daily use: attach (local or remote over SSH), live render through Eat, input, resize, window tabs, scrollback with redraw-compaction, optional flow control. Mouse handling and broader edge-case hardening are still in progress.
The tiled view (C-c C-t) is experimental — whole-frame only, with a
few known limits — but renders every pane cell-for-cell
with live per-pane I/O and automatic re-tiling.
Some CLI coding-agent tools run a session per agent in tmux — one pane per agent, or one window per agent. Because tmux-control renders the tmux session itself, those layouts show up as tiled buffers and window tabs (with the activity dot flagging which one wants you), with no special handling. If that's of interest, see docs/agents.md; if it isn't, you can ignore it entirely — none of the above depends on it.
make test # pure-logic unit tests (no tmux server required)eat must be on the load path; pass EAT_DIR=/path/to/eat if it isn't at the
straight.el default. The live render-fidelity suites (make test-integration
and a GUI tiling oracle) are described in the
guide.
GPL-3.0-or-later. See LICENSE.

