5 releases

0.1.4 Sep 15, 2024
0.1.3 Sep 11, 2024
0.1.2 Aug 25, 2024
0.1.1 Aug 17, 2024
0.1.0 Aug 17, 2024

#314 in Graphics APIs

30 downloads per month

MIT/Apache

1MB
5K SLoC

ratatui-wgpu

Crate Badge Deps.rs Badge Docs Badge License Badge

A wgpu based rendering backend for ratatui.

https://github.com/user-attachments/assets/eb99fd5c-7f07-4a2f-ab11-3e7f6557c6f7

This started out as a custom rendering backend for a game I'm developing, and I thought I'd make it available to the broader community as an alternative rendering target for TUI applications. One of its primary goals is to support serving TUI applications on the web.

Alternatives

  • egui_ratatui uses an egui widget as its backend, allowing it to run anywhere egui can run (including the web).
    • Advantages: Egui is significantly more mature than this library and brings with it a host of widgets and builtin accessibility support.
    • Disadvantages: You can't run custom shader code against its output.

Goals

The crate has the following goals in order of descending priority.

  1. Allow custom shader code to be run against rendered text.
  2. Target WASM.
    • The hello_web example demonstrates its usage for web. hello_webworker shows how to use this backend to render from a worker thread.
    • You will likely want to enable the web feature if you intend to support Firefox.
  3. Correct text rendering (including shaping, mixed bidi, and combining sequences).
    • For color fonts (e.g. emojis) only colr v0 & v1 outlines and raster images are supported.
  4. Reasonable performance.

Non-goals

  1. Builtin-accessibility support.
    • I'm willing to make concessions to enable accessibility (the native version of my game uses accesskit), but integrating directly with accessibility libraries is outside the scope of this library.

Known Limitations

  1. No cursor rendering.

    • The location of the cursor is tracked, and operations using it should behave as expected, but the cursor is not rendered to the screen.
  2. Attempting to render more unique (utf8 character * BOLD|ITALIC) characters than can fit in the cache in a single draw call will cause incorrect rendering. This is ~3750 characters at the default font size with most fonts. If you need more than this, file a bug and I'll do the work to make rendering handle an unbounded number of unique characters.

    To put that in perspective, rendering every printable ascii character in every combination of styles would take (95 * 4) 380 cache entries or ~10% of the cache.

Changelog

1.3 -> 1.4

  • Support colr v1.
    • Drop skrifa.
  • Deprecate with_dimensions and replace with with_width_and_height.
    • The order of arguments (height, width) to with_dimensions was confusing. The new function takes in a struct which explicitly specifies width & height.

1.2 -> 1.3

  • Support colr v0 and png/bitmap images in fonts, allowing emoji rendering.
    • Switched tiny_skia -> raqote.
    • Added skrifa.
    • Added unicode-properties.
    • Added png.
  • Fix an issue with incorrect handling of srgb conversion in the default shader.

1.1 -> 1.2

  • Added support for rtl and mixed bidi.
    • Switched swash -> rustybuzz.
  • Added support for combining characters.
  • Added webworker example.
  • Added colors example.
  • Added dependency on ahash.
  • Dropped dependency on coolors.
  • Dropped dependency on palette.
  • Dropped dependency on guillotiere.

Dependencies

This crate attempts to be reasonable with its usage of external dependencies, although it is definitely not minimal.

  1. ratatui & wgpu: This crate's core purpose is to create a backend for the former using the latter, so these are both necessary.
  2. ahash (optional, default): Inserting into map & set tracking structures during the core rendering loop takes up a significant portion of the render time. Using ahash improves overall performance by ~3% for the colors example in my profiling (12% of execution time vs 15%).
  3. bitvec: During rendering, I need to efficiently track dirty cells which need their background/contents repainted. Bitvec is an efficient structure for tracking which cells are dirty.
  4. bytemuck: Working directly with byte slices is needed to interface with the shader code, and this nicely encapsulates work that would otherwise be unsafe.
  5. indexmap: Used internally to implement an lru heap which has O(1) lookup for entries and to order glyphs in target cells for rendering. This could be replaced with a HashMap<Key, usize> + Vec<Value>, but doing so would require a lot of tedious & error prone book keeping when bubbling entries down the heap.
  6. log: Integrating with standard logging infrastructure is very useful. This might be replaced with tracing, but I'm not going to go without some sort of logging.
  7. png (optional, default): Some fonts embed png images as raster graphics for characters. The png crate is used to decode these images if they are present.
  8. raqote: I don't want to implement path stroking & filling by hand and this library supports all the gradient modes required to render from a font's COLR table.
  9. rustybuzz: Text shaping is hard and way out of scope for this library. There will always be an external dependency on some other library to do this for me. Rustybuzz happens to be (imo) the current best choice.
  10. thiserror: I don't want to write the Error trait by hand. I might consider removing this if doing so doesn't turn out to be so bad.
  11. unicode-bidi: I don't want to implement the unicode bidi algorithm by hand, and even if I did, most of the code would be based on a implementation like this anyways. This performs well enough even though cells have to be concatenated into a single string for processing. There are smarter ways to to this processing I'm sure, but I'll optimize when I need to.
  12. unicode-properties: I need to check if a character is an emoji in order to know how to handle foreground colors and bold/italic styles.
  13. unicode-width: I need to access the width of characters to figure out row layout and implementing this myself seems silly. This is already pulled in by ratatui, so it doesn't really increase the size of the dependency tree.
  14. web-time: Used for crossplatform (web & native) time support in order to handle text blinking.

Dependencies

~14–45MB
~705K SLoC