Skip to content

Commit

Permalink
docs: Revised based on comments 📚
Browse files Browse the repository at this point in the history
  • Loading branch information
kdheepak committed Oct 7, 2023
1 parent 54e3efd commit 8c961bf
Showing 1 changed file with 131 additions and 180 deletions.
311 changes: 131 additions & 180 deletions src/concepts/rendering-under-the-hood.md
Original file line number Diff line number Diff line change
@@ -1,67 +1,59 @@
# How does Ratatui work?

You may have read in previous sections that Ratatui is a immediate mode rendering library. But what
does that really mean? And how is it implemented? In this section, we will discuss how Ratatui renders `Widget`
to the screen, starting with the `Terminal`'s `draw` method and ending with your chosen backend library.
does that really mean? And how is it implemented? In this section, we will discuss how Ratatui
renders a widget to the screen, starting with the `Terminal`'s `draw` method and ending with your
chosen backend library.

In Ratatui, the primary mechanism for making something that is renderable is through the `Widget`
trait.
To render an UI in Ratatui, your application calls the [`Terminal::draw()`] method. This method
takes a [closure] which accepts an instance of a [`Frame`]. Inside the `draw` method, applications
can call [`Frame::render_widget()`] or [`Frame::render_stateful_widget()`] to render the state of a
widget within the available renderable area.

```rust
pub trait Widget {
/// Draws the current state of the widget in the given buffer. That is the only method required
/// to implement a custom widget.
fn render(self, area: Rect, buf: &mut Buffer);
}
```
The `Frame` holds a reference to a buffer which it can render widgets to using the `render_widget()`
method. At the end of the `draw` method (after the closure returns), Ratatui persists the content of
the buffer to the terminal. We will discuss more about the `Buffer` later in this page.

Any struct (inside Ratatui or third party crates) can provide an implementation of the `Widget`
trait for said struct, making an instance of that struct renderable to the terminal.
[`Terminal::draw()`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/terminal.rs#L325-L360
[closure]: https://doc.rust-lang.org/stable/book/ch13-01-closures.html
[`Frame::render_widget()`]:
https://github.com/ratatui-org/ratatui/blob/88ae3485c2c540b4ee630ab13e613e84efa7440a/src/terminal.rs#L596
[`Frame::render_stateful_widget()`]:
https://github.com/ratatui-org/ratatui/blob/88ae3485c2c540b4ee630ab13e613e84efa7440a/src/terminal.rs#L628

For example, the `Paragraph` struct is a widget provided by Ratatui. Here's how you may use the
`Paragraph` widget:
Here is the `terminal.draw()` call for a simple "hello world" example with Ratatui.

```rust
fn main() {
let mut terminal = ratatui::terminal::Terminal::new(ratatui::backend::CrosstermBackend::new(std::io::stderr()))?;

terminal.draw(|f| {
f.render_widget(Paragraph::new("Hello World!"), f.size());
})?;
}
terminal.draw(|frame| {
frame.render_widget(Paragraph::new("Hello World!"), f.size());
});
```

In order to use Ratatui, users are expected to create an instance of the `Terminal` struct and call
the `draw` method by passing in a function. This function takes one argument of type [`Frame`]. For
example, the following code is exactly equivalent to the code above:
The closure takes an argument of type `&mut Frame`. `f.size()` returns a `Rect` that represents the
total renderable area. The `f.render_widget()` method calls a `Widget::render()` method on the
struct. This `Widget::render()` method is part of the [`Widget`] trait.

```rust
fn main() {
let mut terminal = ratatui::terminal::Terminal::new(ratatui::backend::CrosstermBackend::new(std::io::stderr()))?;

terminal.draw(ui)?;
}

fn ui(frame: &mut Frame) {
frame.render_widget(
Paragraph::new("Hello World!"),
frame.size(),
);
pub trait Widget {
/// Draws the current state of the widget in the given buffer.
fn render(self, area: Rect, buf: &mut Buffer);
}
```

This `Frame` struct contains a method called `render_widget` which takes any object that implements
the `Widget` trait. This `render_widget` method calls the `Widget::render` method on the type-erased
`Widget` struct.
[`Widget`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/widgets.rs#L107-L112

```rust
pub fn render_widget<W>(&mut self, widget: W, area: Rect)
where
W: Widget,
{
widget.render(area, self.buffer);
}
```
Any struct (inside Ratatui or third party crates) can provide an implementation of the
`Widget::render()` method, making an instance of that struct renderable to the terminal. That is the
only method required to implement a custom widget.

<!--prettier-ignore-->
In the `Paragraph` example above, `f.render_widget()` calls the
[`Widget::render()` method implemented for `Paragraph`].

[`Widget::render()` method implemented for `Paragraph`]:
https://github.com/ratatui-org/ratatui/blob/88ae3485c2c540b4ee630ab13e613e84efa7440a/src/widgets/paragraph.rs#L213-L214

As mentioned above, implementing the `Widget` trait is the primary way to make a struct renderable.
As an example, here's the full implementation for the `Clear` widget:
Expand All @@ -82,153 +74,113 @@ impl Widget for Clear {

There are 2 things that you should notice.

1. The `Widget` gets the current area as a `Rect`.
2. The `Widget` gets a mutable reference to a "`Buffer`".

The top level widget usually gets a reference to the entire size of the terminal as a `Rect`, i.e.
`f.size()`. However, nested widgets may get a smaller area. Any implementation of a `Widget` must
only draw "within the lines" of the `Rect` area.
1. The `Widget::render` gets the current area as a `Rect`.
- Any implementation of a `Widget` should only draw "within the lines" of the `Rect` area.
2. The `Widget::render` gets a mutable reference to a [`Buffer`] object.

Ratatui widgets render to an intermediate buffer before any information is rendered to the terminal.
This is in contrast to using a library like `crossterm` directly, where writing text to terminal can
occur immediately.

Every time your application calls `terminal.draw(|f| ...)`, Ratatui passes into the closure a new
instance of a [`Frame`] which contains a mutable reference to an instance of `Buffer`.

The `Buffer` represents an area that the application can draw into by manipulating its contents. A
[`Buffer`] contains a collection of [`Cell`]s to represent the rows and columns of the terminal's
display area. Widgets interact with these `Cell`s using the `Buffer` methods.

Here's a visual representation of a `Buffer` that is 12 `Cell`s wide and 4 `Cell`s tall.

```svgbob
0 1 2 3 4 5 6 7 8 9 10 11
┌─────┬─────┬─────┬─────┬─────┬─────┬─────┬─────┬─────┬─────┬─────┬─────┐
0 │ H │ e │ l │ l │ o │ │ W │ o │ r │ l │ d │ ! │
├─────┼─────┼─────┼─────┼──▲──┼─────┼─────┼─────┼─────┼─────┼─────┼─────┤
1 │ │ │ │ │ │ │ │ │ │ │ │ │ │
├─────┼─────┼─────┼─────┼──┼──┼─────┼─────┼─────┼─────┼─────┼─────┼─────┤
2 │ │ │ │ │ │ │ │ │ │ │ │ │ │
├─────┼─────┼─────┼─────┼──┼──┼─────┼─────┼─────┼─────┼─────┼─────┼─────┤
3 │ │ │ ┌─┼─────┼──┴──┼─────┼──┐ │ │ │ │ │ │
└─────┴─────┴───┼─┴─────┴─────┴─────┴──┼──┴─────┴─────┴─────┴─────┴─────┘
│ │
┌──────┴──────┐ ┌──────┴──────┐
│ symbol │ │ style │
│ │ │ │
│ “o” │ │ fg - Reset │
│ │ │ bg - Reset │
│ │ │ │
└─────────────┘ └─────────────┘
You might think that a `Widget` renders directly to the terminal but that is not the case. Ratatui
uses something called a "double-buffer" rendering technique.
Every time `terminal.draw(|f| ...)` is called, a new [`Frame`] struct is constructed that contains a
mutable reference to an instance of the `Buffer` struct:

```rust
pub struct Frame<'a> {
// --snip--
/// The buffer that is used to draw the current frame
buffer: &'a mut Buffer,
}
```

The `Buffer` represents the area in which content is going to be drawn to the terminal. The `Buffer`
struct implements a number of methods such as [`get_mut`] and [`set_string`]:
In Ratatui, a `Cell` struct is the smallest renderable unit of code. Each `Cell` tracks symbol and
style information (foreground color, background color, modifiers etc). `Cell`s are similar to a
"pixel" in a graphical UI. Terminals generally render text so that each individual cell takes up
space approximately twice as high as it is wide. A `Cell` in Ratatui should usually contain 1 wide
string content.

```rust
pub fn get_mut(&mut self, x: u16, y: u16) -> &mut Cell {
}
The `Buffer` implements methods to write text, set styles on particular areas and manipulate
individual cells. For example,

pub fn set_string<S>(&mut self, x: u16, y: u16, string: S, style: Style)
where
S: AsRef<str> {
}
```
- `buf.get_mut(0, 0)` will return a `Cell` with the symbol and style information for row = 0 and col
= 0.
- `buf.set_string(0, 0, "Hello World!", Style::default())` will render `hello world` into the
`Buffer` starting at row = 0 and col = 0 with the style set to default for all those cells.

These methods allow any implementation of the `Widget` trait to write into any parts of the
These methods allow any implementation of the `Widget` trait to write into different parts of the
`Buffer`.

A [`Buffer`] is essentially a collection of `Cell`s:

```rust
pub struct Buffer {
// --snip--
pub content: Vec<Cell>,
}
```

In Ratatui, this `Cell` struct is the smallest renderable unit of code. A `Cell` essentially has a
`String` which is the symbol that will be drawn in that `Cell` as well as `Style` information
(foreground color, background color, modifiers etc).

```rust
// pseudo code
pub struct Cell {
content: String,
style: Style,
}
```

```admonish
`Cell`s are the closest analog to a "pixel" in a terminal.
Most terminals render monospaced text 2 high and 1 wide.
A `Cell` in Ratatui should usually contain 1 wide string content.
```

Here's a simplified class diagram for reference:

```mermaid
classDiagram
class Cell {
+content: String
+style: Style
}
class Buffer {
+area: Rect
+content: Vec<Cell>
}
class Frame {
+buffer: mut Buffer
+render_widget()
}
class Widget {
<<interface>>
+render(area: Rect, buf: &mut Buffer)
}
class Paragraph
class Block
class Table
Buffer --> Cell
Frame --> Buffer
Widget <|.. Paragraph
Widget <|.. Block
Widget <|.. Table
```

Basically what this all means that any content rendered to a `Buffer` is only stored in that
particular `Buffer` struct in the form of `Cell`s that contain string and style information.
This all means that any content rendered to a `Buffer` is only stored in the `Buffer` that is
attached to the frame during the `draw` call.

```admonish note
Because any styling has to happen though the `Style` struct (i.e. methods that apply
`Style`s to `Cell`s) you can't use ansi escape sequences directly in the string content.
As a workaround, what you can do however is convert the ansi escape sequences into a
`ratatui::text::Text` struct that contains the appropriate `Style`s. There's a crate for this that
might help: <https://crates.io/crates/ansi-to-tui>. And this `Text` struct that you create can be
passed into the `Paragraph` widget to get the desired effect.
Even if this particular crate doesn't work for you this approach is probably what you want to use if
you want to render ansi escape sequences in Ratatui.
You can learn more about the text related of Ratatui features and displaying text
[here](./../how-to/render/display-text.md).
ANSI Escape sequences for color and style that are stored in the cell's string content are not
rendered as the style information is stored separately in the cell. If your text has ANSI styling
info, consider using the [`ansi-to-tui`](https://crates.io/crates/ansi-to-tui) crate to convert it
to a `Text` value before rendering. You can learn more about the text related Ratatui features and
displaying text [here](./../how-to/render/display-text.md).
```

After all the `Widget`s are rendered to the same `Buffer` struct, i.e. after
`terminal.draw(|f| ...)` returns, [`terminal.flush()`] is called. This is where the content is
actually written out to the terminal.
After the closure provided to the `draw` method finishes, the `draw` method calls
[`Terminal::flush()`]. This causes the buffer to be written to the screen. Ratatui uses a double
buffer approach so that it can use a diff to figure out which content to write to the terminal
screen efficiently.

In `flush()`, the difference between the previous and the current buffer is obtained and that is
passed it to the current backend for drawing to your actual terminal.
passed it to the current backend for drawing to your actual terminal. After `flush()`, the buffers
are swapped and the next time `terminal.draw(|f| ...)` is called a `Frame` is constructed with the
other `Buffer` struct.

```rust
pub fn flush(&mut self) -> io::Result<()> {
let previous_buffer = &self.buffers[1 - self.current];
let current_buffer = &self.buffers[self.current];
let updates = previous_buffer.diff(current_buffer);
if let Some((col, row, _)) = updates.last() {
self.last_known_cursor_pos = (*col, *row);
}
self.backend.draw(updates.into_iter())
}
```mermaid
sequenceDiagram
participant A as App
participant T as Terminal
participant B as Buffer
A ->>+ T: draw
create participant F as frame
T ->> F: new
T ->>+ A: ui
create participant W as widget
A ->> W: new
A ->>+ F: render_widget
F ->>+ W: render
opt
W ->> B: get_cell
W ->> B: set_string
W ->> B: set_line
W ->>- B: set_style
end
deactivate A
T -->> A: completed_frame
```

After `flush()`, the buffers are swapped and the next time `terminal.draw(|f| ...)` is called a
`Frame` is constructed with the other `Buffer` struct. This is why it is called a double-buffer
rendering technique.

The important thing to note is that because all widgets render to the same `Buffer` within a single
`terminal.draw(|f| ...)` call, rendering of different widgets may overwrite the same `Cell`, so the
order in which widgets are rendered is relevant. For example, in this `draw` example below,
`"content1"` will be overwritten by `"content2"` which will be overwritten by `"content3"` in the
`Buffer` struct, and Ratatui will only ever write out `"content3"` to the terminal:
`terminal.draw(|f| ...)` call, rendering of different widgets may overwrite the same `Cell`. This
means the order in which widgets are rendered will affect the final UI. For example, in this `draw`
example below, `"content1"` will be overwritten by `"content2"` which will be overwritten by
`"content3"` in the `Buffer` struct, and Ratatui will only ever write out `"content3"` to the
terminal:

```rust
terminal.draw(|f| {
Expand All @@ -238,8 +190,9 @@ terminal.draw(|f| {
})
```

This was a sneak peak into what happens under the hood in Ratatui. If you have any questions, feel
free to open issues or discussions on [the main repository](https://github.com/ratatui-org/ratatui).
In summary, the application calls `terminal.draw(|f| ...)`, and Ratatui provides a frame to the
closure. Using the frame's `render_widget` method, each widget draws to a buffer. Ratatui writes the
contents of the buffer to the terminal.

[`Cell`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/buffer.rs#L15-L26
Expand All @@ -255,9 +208,7 @@ free to open issues or discussions on [the main repository](https://github.com/r
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/widgets/block.rs#L752-L760
[`Frame`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/terminal.rs#L566-L578
[`Widget`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/widgets.rs#L107-L112
[`terminal.flush()`]:
[`Terminal::flush()`]:
https://github.com/ratatui-org/ratatui/blob/e5caf170c8c304b952cbff7499fd4da17ab154ea/src/terminal.rs#L253-L263
[`get_mut`]:
https://github.com/ratatui-org/ratatui/blob/88ae3485c2c540b4ee630ab13e613e84efa7440a/src/buffer.rs#L207-L211
Expand Down

0 comments on commit 8c961bf

Please sign in to comment.