38 releases (14 stable)
4.2.0 | Dec 31, 2024 |
---|---|
3.0.0 | Nov 1, 2024 |
2.0.1 | Oct 7, 2024 |
1.0.5 | Jul 20, 2024 |
0.4.3 | Nov 24, 2023 |
#11 in Graphics APIs
3,858 downloads per month
Used in 29 crates
(26 directly)
6MB
2K
SLoC
Ratatui-image
Showcase:
Image widgets with multiple graphics protocol backends for ratatui
ratatui is an immediate-mode TUI library. ratatui-image tackles 3 general problems when rendering images with an immediate-mode TUI:
Query the terminal for available graphics protocols
Some terminals may implement one or more graphics protocols, such as Sixels, or the iTerm2 or Kitty graphics protocols. Guess by env vars. If that fails, query the terminal with some control sequences. Fallback to "halfblocks" which uses some unicode half-block characters with fore- and background colors.
Query the terminal for the font-size in pixels.
If there is an actual graphics protocol available, it is necessary to know the font-size to be able to map the image pixels to character cell area. The image can be resized, fit, or cropped to an area. Query the terminal for the window and columns/rows sizes, and derive the font-size.
Render the image by the means of the guessed protocol.
Some protocols, like Sixels, are essentially "immediate-mode", but we still need to avoid the TUI from overwriting the image area, even with blank characters. Other protocols, like Kitty, are essentially stateful, but at least provide a way to re-render an image that has been loaded, at a different or same position. Since we have the font-size in pixels, we can precisely map the characters/cells/rows-columns that will be covered by the image and skip drawing over the image.
Quick start
use ratatui::{backend::TestBackend, Terminal, Frame};
use ratatui_image::{picker::Picker, StatefulImage, protocol::StatefulProtocol};
struct App {
// We need to hold the render state.
image: StatefulProtocol,
}
fn main() -> Result<(), Box<dyn std::error::Error>> {
let backend = TestBackend::new(80, 30);
let mut terminal = Terminal::new(backend)?;
// Should use Picker::from_query_stdio() to get the font size and protocol,
// but we can't put that here because that would break doctests!
let mut picker = Picker::from_fontsize((8, 12));
// Load an image with the image crate.
let dyn_img = image::io::Reader::open("./assets/Ada.png")?.decode()?;
// Create the Protocol which will be used by the widget.
let image = picker.new_resize_protocol(dyn_img);
let mut app = App { image };
// This would be your typical `loop {` in a real app:
terminal.draw(|f| ui(f, &mut app))?;
Ok(())
}
fn ui(f: &mut Frame<'_>, app: &mut App) {
// The image widget.
let image = StatefulImage::default();
// Render with the protocol state.
f.render_stateful_widget(image, f.area(), &mut app.image);
}
The picker::Picker helper is there to do all this font-size and graphics-protocol guessing, and also to map character-cell-size to pixel size so that we can e.g. "fit" an image inside a desired columns+rows bound, and so on.
Widget choice
- The [Image] widget does not adapt to rendering area (except not drawing at all if space is insufficient), may be a bit more bug prone (overdrawing or artifacts), and is not friendly with some of the protocols (e.g. the Kitty graphics protocol, which is stateful). Its big upside is that it is stateless (in terms of ratatui, i.e. immediate-mode), and thus can never block the rendering thread/task. A lot of ratatui apps only use stateless widgets.
- The [StatefulImage] widget adapts to its render area, is more robust against overdraw bugs and
artifacts, and plays nicer with some of the graphics protocols.
The resizing and encoding is blocking by default, but it is possible to offload this to another
thread or async task (see
examples/async.rs
). It must be rendered withrender_stateful_widget
(i.e. with some mutable state).
Examples
examples/demo.rs
is a fully fledged demo.examples/async.rs
shows how to offload resize and encoding to another thread, to avoid blocking the UI thread.
The lib also includes a binary that renders an image file, but it is focused on testing.
Features
crossterm
ortermion
should match your ratatui backend.termwiz
is available, but not working correctly with ratatu-image.serde
for#[derive]
s on picker::ProtocolType for convenience, because it might be useful to save it in some user configuration.image-defaults
(default) just enablesimage/defaults
(image
hasdefault-features = false
). To only support a selection of image formats and cut down dependencies, disable this feature, addimage
to your crate, and enable its features/formats as desired. See https://doc.rust-lang.org/cargo/reference/features.html#feature-unification.
Compatibility matrix
Compatibility and QA:
Terminal | Protocol | OK | Notes |
---|---|---|---|
Xterm | Sixel |
✔️ | Run with -ti 340 to make sure sixel support is enabled. |
Foot | Sixel |
✔️ | Wayland. |
Kitty | Kitty |
✔️ | Reference for the Kitty protocol. |
Wezterm | iTerm2 |
✔️ | Also would support Sixel and Kitty , but only iTerm2 actually works bug-free. |
Ghostty | Kitty |
✔️ | Implements Kitty with unicode placeholders. |
Rio | iTerm2 |
✔️ | Also supports Sixel but has glitches. |
Alacritty | Sixel |
❌ | There is a sixel fork, but it's probably never getting merged, and does not clear graphics. |
iTerm2 | iTerm2 |
❔ | Feedback from mac users wanted. |
Konsole | Sixel |
❌ | Possibly fixed in 24.12 |
Contour | Sixel |
❌ | Does not clear graphics. |
ctx | Sixel |
❌ | Buggy. |
A basic screenshot test is run with xterm on Xvfb in the CI (or cargo make screenshot-xvfb && cargo make screenshot-diff
).
Halfblocks should work in all terminals, even if the font size could not be detected, with a 4:8 pixel ratio.
Known issues
Summary | Link |
---|---|
Termwiz backend does not work at all | #1 |
Sixel image rendered on the last line of terminal causes a scroll | #57 |
Terminals may or may not take DPI scale into account | #59 (closed) |
Projects that use ratatui-image
- mdfried A markdown viewer that renders headers bigger (as images), and regular images too.
- iamb A matrix client with vim keybindings.
- joshuto A terminal file manager that can preview images.
- Aerostream A Bluesky client using EventStream.
Many more, see and (note that github includes a huge number of unrelated dotfile repos).
Comparison
- viuer Renders graphics in different terminals/protocols, but "dumps" the image, making it difficult to work for TUI programs. The terminal protocol guessing code has been adapted to rustix, thus the author of viuer is included in the copyright notice.
- yazi Not a library but a terminal file manager that implementes many graphics protocols and lets you preview images in the filesystem.
- Überzug++ CLI utility that draws images on terminals by using X11/wayland child windows, sixels, kitty, and/or iterm2 protocols (any means necessary). There exists several wrapper or bindings crates. More battle-tested but essentially stateful, which makes it hard to use with immediate-mode.
Contributing
PRs and issues/discussions welcome!
You can run an aproximation of the CI with cargo make ci
.
I must manually approve CI runs for new PRs to prevent github-action attacks.
The demo is useful to test that everything works correctly.
PRs should be rebased on master
and not contain any merge commits.
License: MIT
Dependencies
~8–50MB
~778K SLoC