- (backend) Add termwiz backend and example (#5)
- (block) Support placing the title on bottom (#36)
- (border) Add border! macro for easy bitflag manipulation (#11)
- (calendar) Add calendar widget (#138)
- (color) Add
FromStr
implementation forColor
(#180) - (list) Add len() to List (#24)
- (paragraph) Allow Lines to be individually aligned (#149)
- (sparkline) Finish #1 Sparkline directions PR (#134)
- (terminal) Add inline viewport (#114) [breaking]
- (test) Expose test buffer (#160)
- (text) Add
Masked
to display secure data (#168) [breaking] - (widget) Add circle widget (#159)
- (widget) Add style methods to Span, Spans, Text (#148)
- (widget) Support adding padding to Block (#20)
- (widget) Add offset() and offset_mut() for table and list state (#12)
- (canvas) Use full block for Marker::Block (#133) [breaking]
- (example) Update input in examples to only use press events (#129)
- (uncategorized) Cleanup doc example (#145)
- (reflow) Remove debug macro call (#198)
- (example) Remove redundant
vec![]
inuser_input
example (#26) - (example) Refactor paragraph example (#152)
- (style) Mark some Style fns const so they can be defined globally (#115)
- (text) Replace
Spans
withLine
(#178)
- (apps) Fix rsadsb/adsb_deku radar link (#140)
- (apps) Add tenere (#141)
- (apps) Add twitch-tui (#124)
- (apps) Add oxycards (#113)
- (apps) Re-add trippy to APPS.md (#117)
- (block) Add example for block.inner (#158)
- (changelog) Update the empty profile link in contributors (#112)
- (readme) Fix small typo in readme (#186)
- (readme) Add termwiz demo to examples (#183)
- (readme) Add acknowledgement section (#154)
- (readme) Update project description (#127)
- (uncategorized) Scrape example code from examples/* (#195)
- (apps) Update the style of application list (#184)
- (readme) Update project introduction in README.md (#153)
- (uncategorized) Clippy's variable inlining in format macros
- (buffer) Add
assert_buffer_eq!
and Debug implementation (#161) - (list) Add characterization tests for list (#167)
- (widget) Add unit tests for Paragraph (#156)
- (uncategorized) Bump MSRV to 1.65.0 (#171)
- (uncategorized) Add ci, build, and revert to allowed commit types
Thank you so much to everyone that contributed to this release!
Here is the list of contributors who has contributed to ratatui
for the first time!
- @kpcyrd
- @fujiapple852
- @BrookJeynes
- @Ziqi-Yang
- @Xithrius
- @lesleyrs
- @pythops
- @wcampbell0x2a
- @sophacles
- @Eyesonjune18
- @a-kenji
- @TimerErTim
- @Mehrbod2002
- @thomas-mauran
- @nyurik
- (style) Bold needs a bit (#104)
Thank you so much to everyone that contributed to this release!
This marks the first release of ratatui
, a community-maintained fork of tui.
The purpose of this release is to include bug fixes and small changes into the repository thus no new features are added. We have transferred all the pull requests from the original repository and worked on the low hanging ones to incorporate them in this "maintenance" release.
Here is a list of changes:
- (cd) Add continuous deployment workflow (#93)
- (ci) Add MacOS to CI (#60)
- (widget) Add
offset()
toTableState
(#10) - (widget) Add
width()
to ListItem (#17)
- (ci) Test MSRV compatibility on CI (#85)
- (ci) Bump Rust version to 1.63.0 (#80)
- (ci) Use env for the cargo-make version (#76)
- (ci) Fix deprecation warnings on CI (#58)
- (doc) Add 3rd party libraries accidentally removed at #21 (#61)
- (widget) List should not ignore empty string items (#42) [breaking]
- (uncategorized) Cassowary/layouts: add extra constraints for fixing Min(v)/Max(v) combination. (#31)
- (uncategorized) Fix user_input example double key press registered on windows
- (uncategorized) Ignore zero-width symbol on rendering
Paragraph
- (uncategorized) Fix typos (#45)
- (uncategorized) Fix typos (#47)
- (style) Make bitflags smaller (#13)
- (apps) Move 'apps using ratatui' to dedicated file (#98) (#99)
- (canvas) Add documentation for x_bounds, y_bounds (#35)
- (contributing) Specify the use of unsafe for optimization (#67)
- (github) Remove pull request template (#68)
- (readme) Update crate status badge (#102)
- (readme) Small edits before first release (#101)
- (readme) Add install instruction and update title (#100)
- (readme) Add systeroid to application list (#92)
- (readme) Add glicol-cli to showcase list (#95)
- (readme) Add oxker to application list (#74)
- (readme) Add app kubectl-watch which uses tui (#73)
- (readme) Add poketex to 'apps using tui' in README (#64)
- (readme) Update README.md (#39)
- (readme) Update README.md (#40)
- (readme) Clarify README.md fork status update
- (uncategorized) Fix: fix typos (#90)
- (uncategorized) Update to build more backends (#81)
- (uncategorized) Expand "Apps" and "Third-party" sections (#21)
- (uncategorized) Add tui-input and update xplr in README.md
- (uncategorized) Add hncli to list of applications made with tui-rs (#41)
- (uncategorized) Updated readme and contributing guide with updates about the fork (#46)
- (layout) Better safe shared layout cache (#62)
- (cargo) Update project metadata (#94)
- (ci) Integrate
typos
for checking typos (#91) - (ci) Change the target branch to main (#79)
- (ci) Re-enable clippy on CI (#59)
- (uncategorized) Integrate
committed
for checking conventional commits (#77) - (uncategorized) Update
rust-version
to 1.59 in Cargo.toml (#57) - (uncategorized) Update deps (#51)
- (uncategorized) Fix typo in layout.rs (#619)
- (uncategorized) Add apps using
tui
Thank you so much to everyone that contributed to this release!
- @orhun
- @mindoodoo
- @sayanarijit
- @Owletti
- @UncleScientist
- @rhysd
- @ckaznable
- @imuxin
- @mrjackwills
- @conradludgate
- @kianmeng
- @chaosprint
And most importantly, special thanks to Florian Dehau for creating this awesome library 💖 We look forward to building on the strong foundations that the original crate laid out.
- Bump
crossterm
to0.25
- Update
crossterm
to0.23
- Add option to
widgets::List
to repeat the highlight symbol for each line of multi-line items (#533). - Add option to control the alignment of
Axis
labels in theChart
widget (#568).
- The minimum supported rust version is now
1.56.1
.
crossterm
is now the default backend. If you are already using thecrossterm
backend, you can simplify your dependency specification inCargo.toml
:
- tui = { version = "0.16", default-features = false, features = ["crossterm"] }
+ tui = "0.17"
If you are using the termion
backend, your Cargo
is now a bit more verbose:
- tui = "0.16"
+ tui = { version = "0.17", default-features = false, features = ["termion"] }
crossterm
has also been bumped to version 0.22
.
Because of their apparent low usage, curses
and rustbox
backends have been removed.
If you are using one of them, you can import their last implementation in your own project:
- Labels of the
Canvas
widget are nowtext::Spans
. The signature ofwidgets::canvas::Context::print
has thus been updated:
- ctx.print(x, y, "Some text", Color::Yellow);
+ ctx.print(x, y, Span::styled("Some text", Style::default().fg(Color::Yellow)))
- Update
crossterm
to0.20
. - Add
From<Cow<str>>
implementation fortext::Text
(#471). - Add option to right or center align the title of a
widgets::Block
(#462).
- Apply label style in
widgets::Gauge
and avoid panics because of overflows with long labels (#494). - Avoid panics because of overflows with long axis labels in
widgets::Chart
(#512). - Fix computation of column widths in
widgets::Table
(#514). - Fix panics because of invalid offset when input changes between two frames in
widgets::List
andwidgets::Chart
(#516).
- Update
crossterm
to0.19
. - Update
rand
to0.8
. - Add a read-only view of the terminal state after the draw call (#440).
- Remove compile warning in
TestBackend::assert_buffer
(#466).
The Table
widget got a lot of improvements that should make it easier to work with:
- It should not longer panic when rendered on small areas.
Row
s are now a collection ofCell
s, themselves wrapping aText
. This means you can style the entireTable
, an entireRow
, an entireCell
and rely on the styling capabilities ofText
to get full control over the look of yourTable
.Row
s can have multiple lines.- The header is now optional and is just another
Row
always visible at the top. Row
s can have a bottom margin.- The header alignment is no longer off when an item is selected.
Taking the example of the code in examples/demo/ui.rs
, this is what you may have to change:
let failure_style = Style::default()
.fg(Color::Red)
.add_modifier(Modifier::RAPID_BLINK | Modifier::CROSSED_OUT);
- let header = ["Server", "Location", "Status"];
let rows = app.servers.iter().map(|s| {
let style = if s.status == "Up" {
up_style
} else {
failure_style
};
- Row::StyledData(vec![s.name, s.location, s.status].into_iter(), style)
+ Row::new(vec![s.name, s.location, s.status]).style(style)
});
- let table = Table::new(header.iter(), rows)
+ let table = Table::new(rows)
+ .header(
+ Row::new(vec!["Server", "Location", "Status"])
+ .style(Style::default().fg(Color::Yellow))
+ .bottom_margin(1),
+ )
.block(Block::default().title("Servers").borders(Borders::ALL))
- .header_style(Style::default().fg(Color::Yellow))
.widths(&[
Constraint::Length(15),
Constraint::Length(15),
Here, we had to:
- Change the way we construct
Row
which is no longer anenum
but astruct
. It accepts anything that can be converted to an iterator of things that can be converted to aCell
- The header is no longer a required parameter so we use
Table::header
to set it.Table::header_style
has been removed since the style can be directly set usingRow::style
. In addition, we want to preserve the old margin between the header and the rest of the rows so we add a bottom margin to the header usingRow::bottom_margin
.
You may want to look at the documentation of the different types to get a better understanding:
- Fix handling of Non Breaking Space (NBSP) in wrapped text in
Paragraph
widget.
- Add
Style::reset
to create aStyle
resetting all styling properties when applied. - Add an option to render the
Gauge
widget with unicode blocks. - Manage common project tasks with
cargo-make
rather thanmake
for easier on-boarding.
- Add
LineGauge
widget which is a more compact variant of the existingGauge
. - Bump
crossterm
to 0.18
- Take into account the borders of the
Table
widget when the widths of columns is controlled byPercentage
andRatio
constraints.
- Make it easier to work with string with multiple lines in
Text
(#361).
- Fix a style leak in
Graph
so components drawn on top of the plotted data (i.e legend and axis titles) are not affected by the style of theDataset
s (#388). - Make sure
BarChart
shows bars with the max height only when the plotted data is actually equal to the max (#383).
- Add the dot character as a new type of canvas marker (#350).
- Support more style modifiers on Windows (#368).
- Clearing the terminal through
Terminal::clear
will cause the whole UI to be redrawn (#380). - Fix incorrect output when the first diff to draw is on the second cell of the terminal (#347).
A new method has been added to Frame
called set_cursor
. It lets you specify where the cursor
should be placed after the draw call. Furthermore like any other widgets, if you do not set a cursor
position during a draw call, the cursor is automatically hidden.
For example:
fn draw_input(f: &mut Frame, app: &App) {
if app.editing {
let input_width = app.input.width() as u16;
// The cursor will be placed just after the last character of the input
f.set_cursor((input_width + 1, 0));
} else {
// We are no longer editing, the cursor does not have to be shown, set_cursor is not called and
// thus automatically hidden.
}
}
In order to make this possible, the draw closure takes in input &mut Frame
instead of mut Frame
.
It has been reported several times that the text styling capabilities were somewhat limited in many places of the crate. To solve the issue, this release includes a new set of text primitives that are now used by a majority of widgets to provide flexible text styling.
Text
is replaced by the following types:
Span
: a string with a unique style.Spans
: a string with multiple styles.Text
: a multi-lines string with multiple styles.
However, you do not always need this complexity so the crate provides From
implementations to
let you use simple strings as a default and switch to the previous primitives when you need
additional styling capabilities.
For example, the title of a Block
can be set in the following ways:
// A title with no styling
Block::default().title("My title");
// A yellow title
Block::default().title(Span::styled("My title", Style::default().fg(Color::Yellow)));
// A title where "My" is bold and "title" is a simple string
Block::default().title(vec![
Span::styled("My", Style::default().add_modifier(Modifier::BOLD)),
Span::from("title")
]);
Buffer::set_spans
andBuffer::set_span
were added.Paragraph::new
expects an input that can be converted to aText
.Block::title_style
is deprecated.Block::title
expects aSpans
.Tabs
expects a list ofSpans
.Gauge
custom label is now aSpan
.Axis
title and labels areSpans
(as a consequenceChart
no longer has generic bounds).
Previously Style
was used to represent an exhaustive set of style rules to be applied to an UI
element. It implied that whenever you wanted to change even only one property you had to provide the
complete style. For example, if you had a Block
where you wanted to have a green background and
a title in bold, you had to do the following:
let style = Style::default().bg(Color::Green);
Block::default()
.style(style)
.title("My title")
// Here we reused the style otherwise the background color would have been reset
.title_style(style.modifier(Modifier::BOLD));
In this new release, you may now write this as:
Block::default()
.style(Style::default().bg(Color::Green))
// The style is not overridden anymore, we simply add new style rule for the title.
.title(Span::styled("My title", Style::default().add_modifier(Modifier::BOLD)))
In addition, the crate now provides a method patch
to combine two styles into a new set of style
rules:
let style = Style::default().modifier(Modifier::BOLD);
let style = style.patch(Style::default().add_modifier(Modifier::ITALIC));
// style.modifier == Modifier::BOLD | Modifier::ITALIC, the modifier has been enriched not overridden
Style::modifier
has been removed in favor ofStyle::add_modifier
andStyle::remove_modifier
.Buffer::set_style
has been added.Buffer::set_background
is deprecated.BarChart::style
no longer set the style of the bars. UseBarChart::bar_style
in replacement.Gauge::style
no longer set the style of the gauge. UseGauge::gauge_style
in replacement.
The List
widget has been refactored once again to support items with variable heights and complex
styling.
List::new
expects an input that can be converted to aVec<ListItem>
whereListItem
is a wrapper around the item content to provide additional styling capabilities.ListItem
contains aText
.List::items
has been removed.
// Before
let items = vec![
"Item1",
"Item2",
"Item3"
];
List::default().items(items.iters());
// After
let items = vec![
ListItem::new("Item1"),
ListItem::new("Item2"),
ListItem::new("Item3"),
];
List::new(items);
See the examples for more advanced usages.
Paragraph::wrap
expects Wrap
instead of bool
to let users decided whether they want to trim
whitespaces when the text is wrapped.
// before
Paragraph::new(text).wrap(true)
// after
Paragraph::new(text).wrap(Wrap { trim: true }) // to have the same behavior
Paragraph::new(text).wrap(Wrap { trim: false }) // to use the new behavior
You can now scroll horizontally in Paragraph
. The argument of Paragraph::scroll
has thus be
changed from u16
to (u16, u16)
.
You can now serialize and de-serialize Style
using the optional serde
feature.
- Fix out of bounds panic in
widgets::Tabs
when the widget is rendered on small areas.
- Ignore zero-width graphemes in
Buffer::set_stringn
.
- Fix usize overflows in
widgets::Chart
when a dataset is empty.
- Fix usize overflows in
widgets::canvas::Line
drawing algorithm.
- The
List
widget now takes into account the width of thehighlight_symbol
when calculating the total width of its items. It prevents items to overflow outside of the widget area.
- Introduce stateful widgets, i.e widgets that can take advantage of keeping some state around between two draw calls (#210 goes a bit more into the details).
- Allow a
Table
row to be selected.
// State initialization
let mut state = TableState::default();
// In the terminal.draw closure
let header = ["Col1", "Col2", "Col"];
let rows = [
Row::Data(["Row11", "Row12", "Row13"].into_iter())
];
let table = Table::new(header.into_iter(), rows.into_iter());
f.render_stateful_widget(table, area, &mut state);
// In response to some event:
state.select(Some(1));
- Add a way to choose the type of border used to draw a block. You can now choose from plain, rounded, double and thick lines.
- Add a
graph_type
property on theDataset
of aChart
widget. By default it will beScatter
where the points are drawn as is. An other option isLine
where a line will be draw between each consecutive points of the dataset. - Style methods are now const, allowing you to initialize const
Style
objects. - Improve control over whether the legend in the
Chart
widget is shown or not. You can now set custom constraints usingChart::hidden_legend_constraints
. - Add
Table::header_gap
to add some space between the header and the first row. - Remove
log
from the dependencies - Add a way to use a restricted set of unicode symbols in several widgets to
improve portability in exchange of a degraded output. (see
BarChart::bar_set
,Sparkline::bar_set
andCanvas::marker
). You can check how the--enhanced-graphics
flag is used in the demos.
Widget::render
has been deleted. You should now useFrame::render_widget
to render a widget on the correspondingFrame
. This makes theWidget
implementation totally decoupled from theFrame
.
// Before
Block::default().render(&mut f, size);
// After
let block = Block::default();
f.render_widget(block, size);
Widget::draw
has been renamed toWidget::render
and the signature has been updated to reflect that widgets are consumable objects. Thus the method takesself
instead of&mut self
.
// Before
impl Widget for MyWidget {
fn draw(&mut self, area: Rect, buf: &mut Buffer) {
}
}
/// After
impl Widget for MyWidget {
fn render(self, arera: Rect, buf: &mut Buffer) {
}
}
Widget::background
has been replaced byBuffer::set_background
// Before
impl Widget for MyWidget {
fn render(self, arera: Rect, buf: &mut Buffer) {
self.background(area, buf, self.style.bg);
}
}
// After
impl Widget for MyWidget {
fn render(self, arera: Rect, buf: &mut Buffer) {
buf.set_background(area, self.style.bg);
}
}
- Update the
Shape
trait for objects that can be draw on aCanvas
widgets. Instead of returning an iterator over its points, aShape
is given aPainter
object that provides apaint
as well as aget_point
method. This gives theShape
more information about the surface it will be drawn to. In particular, this change allows theLine
shape to use a more precise and efficient drawing algorithm (Bresenham's line algorithm). SelectableList
has been deleted. You can now take advantage of the associatedListState
of theList
widget to select an item.
// Before
List::new(&["Item1", "Item2", "Item3"])
.select(Some(1))
.render(&mut f, area);
// After
// State initialization
let mut state = ListState::default();
// In the terminal.draw closure
let list = List::new(&["Item1", "Item2", "Item3"]);
f.render_stateful_widget(list, area, &mut state);
// In response to some events
state.select(Some(1));
widgets::Marker
has been moved tosymbols::Marker
- Bump crossterm to 0.14.
- Add cross symbol to the symbols list.
- Use the value of
title_style
to style the title ofAxis
.
- Use
Constraint
instead of integers to specify the widths of theTable
widget's columns. This will allow more responsive tables.
Table::new(header, row)
.widths(&[15, 15, 10])
.render(f, chunk);
becomes:
Table::new(header, row)
.widths(&[
Constraint::Length(15),
Constraint::Length(15),
Constraint::Length(10),
])
.render(f, chunk);
- Bump crossterm to 0.13.
- Use Github Actions for CI (Travis and Azure Pipelines integrations have been deleted).
- Add support for horizontal and vertical margins in
Layout
.
Text
implements PartialEq
- Avoid overflow errors in canvas
- Avoid a division by zero when all values in a barchart are equal to 0.
- Fix the inverted cursor position in the curses backend.
- Ensure that the correct terminal size is returned when using the crossterm backend.
- Avoid highlighting the separator after the selected item in the Tabs widget.
- Update crossterm backend
- Fix a panic in the Sparkline widget
- Add a new curses backend (with Windows support thanks to
pancurses
). - Add
Backend::get_cursor
andBackend::set_cursor
methods to query and set the position of the cursor. - Add more constructors to the
Crossterm
backend. - Add a demo for all backends using a shared UI and application state.
- Add
Ratio
as a new variant of layoutConstraint
. It can be used to define exact ratios constraints.
- Add support for multiple modifiers on the same
Style
by changingModifier
from an enum to a bitflags struct.
So instead of writing:
let style = Style::default().add_modifier(Modifier::Italic);
one should use:
let style = Style::default().add_modifier(Modifier::ITALIC);
// or
let style = Style::default().add_modifier(Modifier::ITALIC | Modifier::BOLD);
- Ensure correct behavior of the alternate screens with the
Crossterm
backend. - Fix out of bounds panic when two
Buffer
are merged.
- Add a new canvas shape:
Rectangle
. - Official support of
Crossterm
backend. - Make it possible to choose the divider between
Tabs
. - Add word wrapping on Paragraph.
- The gauge widget accepts a ratio (f64 between 0 and 1) in addition of a percentage.
- Upgrade to Rust 2018 edition.
- Fix rendering of double-width characters.
- Fix race condition on the size of the terminal and expose a size that is
safe to use when drawing through
Frame::size
. - Prevent unsigned int overflow on large screens.
- Add experimental test backend
show_cursor
is called whenTerminal
is dropped if the cursor is hidden.
- Remove custom
termion
backends. This is motivated by the fact thattermion
structs are meant to be combined/wrapped to provide additional functionalities to the terminal (e.g AlternateScreen, Mouse support, ...). Thus providing exclusive types do not make a lot of sense and give a false hint that additional features cannot be used together. The recommended approach is now to create your own version ofstdout
:
let stdout = io::stdout().into_raw_mode()?;
let stdout = MouseTerminal::from(stdout);
let stdout = AlternateScreen::from(stdout);
and then to create the corresponding termion
backend:
let backend = TermionBackend::new(stdout);
The resulting code is more verbose but it works with all combinations of
additional termion
features.
- Replace
Item
by a generic and flexibleText
that can be used in bothParagraph
andList
widgets. - Remove unnecessary borrows on
Style
.
- Add a basic
Crossterm
backend
- Remove
Group
and introduceLayout
in its placeTerminal
is no longer required to compute a layoutSize
has been renamedConstraint
- Widgets are rendered on a
Frame
instead of aTerminal
in order to avoid mixingdraw
andrender
calls draw
onTerminal
expects a closure where the UI is built by rendering widgets on the givenFrame
- Update
Widget
traitdraw
takes area by valuerender
takes aFrame
instead of aTerminal
- All widgets use the consumable builder pattern
SelectableList
can have no selected item and the highlight symbol is hidden in this case- Remove markup language inside
Paragraph
.Paragraph
now expects an iterator ofText
items
- Add
start_corner
option forList
- Add more text alignment options for
Paragraph
Terminal
implementsDebug
- Use
FnOnce
instead ofFnMut
in Group::render
- Add
AlternateScreenBackend
intermion
backend - Add
TermionBackend::with_stdout
in order to let an user of the library provides its own termion struct - Add tests and documentation for
Buffer::pos_of
- Remove leading whitespaces when wrapping text
- Fix
debug_assert
inBuffer::pos_of
- Pass the style of
SelectableList
to the underlyingList
- Fix missing character when wrapping text
- Fix panic when specifying layout constraints
- Add
MouseBackend
intermion
backend to handle scroll and mouse events - Add generic
Item
for items in aList
- Drop
log4rs
as a dev-dependencies in favor ofstderrlog
- Rename
TermionBackend
toRawBackend
(to distinguish it from theMouseBackend
) - Generic parameters for
List
to allow passing iterators as items - Generic parameters for
Table
to allow using iterators as rows and header - Generic parameters for
Tabs
- Rename
border
bitflags toBorders