Display images in terminal Emacs (emacs -nw) using the Kitty graphics protocol.
I got the opportunity to subscribe to the highest Claude level. I’m allowed to use it outside of work. I was able to extend Emacs in terminal mode with the kitty protocol. Out of curiosity I wanted to know if a thought I had for months was possible or not, which was basically to extend libvterm with the kitty image protocol and then utilize vterm to display those images inside Emacs Terminal.
Besides, I don’t even have the guts to release it, even though my intention of such a release would be first and foremost for the people who have been asking for such a feature specifically for years, or for other developers to use it as some kind of reference. So idk, while I totally agree that the current AI slop is getting to be exhausting, we as humans should judge projects on a project by project basis.
kitty-graphics.el renders images directly in terminal Emacs using Kitty’s
graphics protocol with direct placements. Images are transmitted once to the
terminal, then positioned at overlay screen coordinates after each redisplay.
They scroll with text, survive buffer switches, and work in split windows —
they’re managed as overlays in Emacs’s display engine.
Integrations are provided for:
- org-mode — inline images via
C-c C-x C-v(bothorg-toggle-inline-imagesandorg-link-previewfor org 10.0+) - org LaTeX preview — render LaTeX fragments as images via
C-c C-x C-l - doc-view — PDF/DVI/PS viewing with page navigation and zoom
- image-mode — terminal-aware image file viewing
- shr/eww — inline images in HTML rendering (eww, mu4e, gnus)
- dired — image preview in side window (
Pkey) - dirvish — native preview dispatcher for image files
- Emacs >= 27.1
- Kitty >= 0.20.0 (or WezTerm/Ghostty with direct placement support)
- ImageMagick (
magick/convert/identify) for non-PNG formats and image sizing - For LaTeX preview: a TeX distribution with
dvipng(e.g.texlive) - For doc-view:
ghostscript(for PDF),dvipng(for DVI) — same tools as GUI Emacs - Launch Emacs with
TERM=xterm-256color(Emacs often can’t find thexterm-kittyterminfo)TERM=xterm-256color emacsclient -nw
(use-package kitty-graphics
:straight (:local-repo "~/projects/kitty-graphics")
:if (and (not (display-graphic-p)) (getenv "KITTY_PID"))
:config
(kitty-graphics-mode 1))(add-to-list 'load-path "~/projects/kitty-graphics")
(require 'kitty-graphics)
(when (and (not (display-graphic-p)) (getenv "KITTY_PID"))
(kitty-graphics-mode 1))Enable the global minor mode:
(kitty-graphics-mode 1)Then:
- In org-mode: toggle inline images with
C-c C-x C-v - In org-mode: preview LaTeX fragments with
C-c C-x C-l(requires a LaTeX installation anddvipng) - Open a PDF:
doc-view-moderenders pages via Kitty (n/pto navigate,+/-/0to zoom) - In dired: press
Pon an image file for a side-window preview - In dirvish: image previews work automatically (no extra config)
- Open an image file:
image-modedisplays it via Kitty - In eww/mu4e/gnus: HTML images render inline
| Command | Description |
|---|---|
kitty-graphics-mode | Toggle the global minor mode |
kitty-gfx-display-image | Display a single image at point |
kitty-gfx-remove-images | Remove images in region or buffer |
kitty-gfx-clear-all | Remove all images from all buffers |
kitty-gfx-dired-preview | Preview image at point in dired |
| Variable | Default | Description |
|---|---|---|
kitty-gfx-max-width | 120 | Maximum inline image width in columns |
kitty-gfx-max-height | 40 | Maximum inline image height in rows |
kitty-gfx-render-delay | 0.016 | Debounce delay for re-rendering (s) |
kitty-gfx-debug | nil | Log debug info to /tmp/kitty-gfx.log |
These defaults control inline image sizing (org-mode, eww, dired previews).
Doc-view ignores these and fills the window, with + / - / 0 for zoom.
- On startup, cell pixel size is queried via
CSI 16 t(XTWINOPS) for accurate image scaling. Falls back to 8x16 if the query times out. - Image data is transmitted once via APC escape sequences (
a=t, store only) - Overlays with a
displayproperty reserve blank space in Emacs buffers - After each redisplay, direct placements (
a=p) are emitted viasend-string-to-terminalat the overlay’s screen coordinates - All placement output is wrapped in synchronized output (
BSU/ESU,DEC mode 2026) to prevent partial rendering and flicker - Each placement uses a unique placement ID (
p=PID) so re-placements at new positions replace the old rendering without accumulation - Position caching skips redundant re-placements when nothing moved
- Placements are deleted when overlays scroll out of view
PNG is sent directly to the terminal. Other formats (JPEG, GIF, WebP, SVG, TIFF, BMP) are converted to PNG via ImageMagick before transmission.
- No tmux support
- Primary focus on kitty terminal. While other terminals should work too, I wont give any guarantee
- Animated GIFs display only the first frame
- Without ImageMagick, only PNG files work and image sizing falls back to a fixed default
- Doc-view page may briefly flash at wrong position before centering on initial load
GPL-2.0-or-later

