mousefood 0.5.0

embedded-graphics backend for Ratatui
Documentation
//! # Framebuffer on the heap
//!
//! A framebuffer implementation for storing pixels in memory before rendering to a display.
//!
//! This module provides [`HeapBuffer`], a heap-allocated framebuffer that can be used
//! with display drivers that don't provide their own internal framebuffer. This helps
//! prevent screen tearing by allowing you to compose a complete frame in memory before
//! sending it to the display.

use alloc::{vec, vec::IntoIter, vec::Vec};

use crate::colors::{ColorTheme, TermColor, TermColorType};
use embedded_graphics::Pixel;
use embedded_graphics::draw_target::DrawTarget;
use embedded_graphics::geometry::Dimensions;
use embedded_graphics::pixelcolor::PixelColor;
use embedded_graphics::primitives::Rectangle;
use ratatui_core::style::Color;

/// A heap-allocated framebuffer for storing pixels before rendering to a display.
///
/// `HeapBuffer` provides a memory-based framebuffer that can be used with display drivers
/// that don't have their own internal framebuffer. This helps prevent screen tearing by
/// allowing you to compose a complete frame in memory before sending it to the display.
///
/// The framebuffer is generic over any pixel color type that implements [`PixelColor`]
/// and [`Copy`]. It stores pixels in a contiguous `Vec<C>` arranged in row-major order.
///
/// # Type Parameters
///
/// * `C` - The pixel color type. Must implement [`PixelColor`] and [`Copy`].
pub struct HeapBuffer<C: PixelColor + Copy> {
    data: Vec<C>,
    bounding_box: Rectangle,
}

impl<C: PixelColor + Copy> HeapBuffer<C> {
    /// Returns the color of the pixel at the given point.
    pub fn get_pixel(&self, point: embedded_graphics::geometry::Point) -> C {
        let idx = (point.y * self.bounding_box.size.width as i32 + point.x) as usize;
        self.data[idx]
    }
}

impl<C: PixelColor + for<'a> From<TermColor<'a>>> HeapBuffer<C> {
    /// Creates a new framebuffer with the specified dimensions.
    ///
    /// The framebuffer is initialized with a background color derived from [`Color::Reset`].
    ///
    /// # Arguments
    ///
    /// * `bounding_box` - A [`Rectangle`] defining the position and size of the framebuffer.
    ///   The framebuffer will have `width × height` pixels, where width and height
    ///   are taken from the rectangle's size.
    ///
    /// # Returns
    ///
    /// A new [`HeapBuffer`] instance with the specified dimensions, initialized with
    /// the default background color.
    ///
    /// # Memory Usage
    ///
    /// This method allocates `width × height × sizeof(C)` bytes of memory on the heap,
    /// where `C` is the pixel color type.
    pub fn new(bounding_box: Rectangle, color_theme: ColorTheme) -> HeapBuffer<C> {
        Self {
            data: vec![
                TermColor::new(Color::Reset, TermColorType::Background, &color_theme).into();
                (bounding_box.size.width * bounding_box.size.height) as usize
            ],
            bounding_box,
        }
    }
}

impl<C: PixelColor> IntoIterator for HeapBuffer<C> {
    type Item = C;
    type IntoIter = IntoIter<Self::Item>;

    fn into_iter(self) -> Self::IntoIter {
        self.data.into_iter()
    }
}

impl<'a, C: PixelColor + Copy> IntoIterator for &'a HeapBuffer<C> {
    type Item = C;
    type IntoIter = core::iter::Copied<core::slice::Iter<'a, C>>;

    fn into_iter(self) -> Self::IntoIter {
        self.data.iter().copied()
    }
}

impl<C: PixelColor> Dimensions for HeapBuffer<C> {
    fn bounding_box(&self) -> Rectangle {
        self.bounding_box
    }
}

impl<C: PixelColor> DrawTarget for HeapBuffer<C> {
    type Color = C;
    type Error = core::convert::Infallible;

    fn draw_iter<I>(&mut self, pixels: I) -> Result<(), Self::Error>
    where
        I: IntoIterator<Item = Pixel<Self::Color>>,
    {
        let idx_end = self.data.len() - 1;
        for Pixel(point, color) in pixels {
            let idx = point.y as usize * self.bounding_box.size.width as usize + point.x as usize;
            self.data[idx.clamp(0, idx_end)] = color;
        }
        Ok(())
    }
}

#[cfg(test)]
mod tests {
    use super::*;
    use rstest::{fixture, rstest};

    use embedded_graphics::mock_display::MockDisplay;
    use embedded_graphics::mono_font::MonoTextStyle;
    use embedded_graphics::mono_font::ascii::FONT_4X6;
    use embedded_graphics::pixelcolor::{Rgb888, RgbColor};
    use embedded_graphics::prelude::*;
    use embedded_graphics::text::Text;

    #[fixture]
    fn display() -> MockDisplay<Rgb888> {
        MockDisplay::new()
    }

    #[fixture]
    fn heap_buffer() -> HeapBuffer<Rgb888> {
        HeapBuffer::new(
            Rectangle::new(Point::zero(), Size::new(16, 8)),
            ColorTheme::default(),
        )
    }

    #[fixture]
    fn test_text<'a>() -> (Text<'a, MonoTextStyle<'a, Rgb888>>, &'a [&'a str]) {
        (
            Text::new(
                "Test",
                Point::new(0, 6),
                MonoTextStyle::new(&FONT_4X6, Rgb888::WHITE),
            ),
            &[
                "KKKKKKKKKKKKKKKK",
                "KKKKKKKKKKKKKKKK",
                "WWWKKKKKKKKKKWKK",
                "KWKKKWKKKWWKWWWK",
                "KWKKWKWKWWKKKWKK",
                "KWKKWWKKKKWKKWKK",
                "KWKKKWWKWWKKKKWK",
                "KKKKKKKKKKKKKKKK",
            ],
        )
    }

    #[rstest]
    fn test_heap_buffer(
        mut display: MockDisplay<Rgb888>,
        mut heap_buffer: HeapBuffer<Rgb888>,
        #[from(test_text)] (text, expected): (Text<MonoTextStyle<Rgb888>>, &[&str]),
    ) {
        text.draw(&mut heap_buffer).unwrap();

        display
            .fill_contiguous(&heap_buffer.bounding_box(), heap_buffer)
            .unwrap();

        display.assert_pattern(expected);
    }

    #[rstest]
    fn test_heap_buffer_as_ref(
        mut display: MockDisplay<Rgb888>,
        mut heap_buffer: HeapBuffer<Rgb888>,
        #[from(test_text)] (text, expected): (Text<MonoTextStyle<Rgb888>>, &[&str]),
    ) {
        text.draw(&mut heap_buffer).unwrap();

        display
            .fill_contiguous(&heap_buffer.bounding_box(), &heap_buffer)
            .unwrap();

        display.assert_pattern(expected);
    }
}