dumo 0.1.1

An embedded-graphics backend for Ratatui with an adjustable font size
Documentation
use core::fmt::Debug;

use embedded_graphics::draw_target::DrawTarget;
use embedded_graphics::iterator::raw::RawDataSlice;
use embedded_graphics::pixelcolor::raw::BigEndian;
use embedded_graphics::pixelcolor::{PixelColor, Rgb888};
use embedded_graphics::text::renderer::TextRenderer;
use mplusfonts::color::{Invert, Screen, WeightedAvg};
use mplusfonts::style::BitmapFontStyle;

use crate::backend::DumoBackend;
#[cfg(feature = "alloc")]
use crate::blink::{Blink, ControlBlinking};
#[cfg(feature = "alloc")]
use crate::cursor::Cursor;
#[cfg(feature = "alloc")]
use crate::wrapper::blink::BlinkWrapper;
#[cfg(feature = "alloc")]
use crate::wrapper::cursor::CursorWrapper;
use crate::wrapper::flush::FlushWrapper;

#[cfg(feature = "alloc")]
pub const BLINKING: bool = true;

#[cfg(feature = "alloc")]
macro_rules! impl_with_blink {
    ($self_ident:ident, $is_blinking:expr $(, $stop_blinking:stmt)?) => {
        /// Returns the backend with a new wrapper around it, redrawing cells to show and hide text
        /// that should blink. Every time a [`Terminal`] calls the [`Backend::draw`] method as part
        /// of the rendering process, the wrapper advances the blinking animation by one frame. The
        /// [`ControlBlinking`] trait also allows stepping the blinking animation.
        ///
        /// Backends without this wrapper display text that is set to blink as solid text.
        ///
        /// [`Backend::draw`]: ratatui_core::backend::Backend::draw
        /// [`Terminal`]: ratatui_core::terminal::Terminal
        ///
        /// # Examples
        ///
        /// ```
        /// # #[cfg(not(all(feature = "alloc", feature = "font-8x24", feature = "font-4-bits")))]
        /// # {
        /// #     compile_error!("doc-test is missing required features");
        /// # }
        /// #
        /// use dumo::DumoBackend;
        /// use dumo::blink::Blink;
        /// use dumo::fonts::*;
        /// # use embedded_graphics::mock_display::MockDisplay;
        /// # use embedded_graphics::pixelcolor::Rgb565;
        ///
        /// # let mut display: MockDisplay<Rgb565> = MockDisplay::new();
        /// let backend = DumoBackend::new(&mut display, &FONT_8X24_4_BITS)
        ///     .with_blink(Blink::with_period(16), Blink::with_period(8));
        /// ```
        pub fn with_blink(
            #[allow(unused_mut)] mut $self_ident,
            slow_blink: Blink,
            rapid_blink: Blink,
        ) -> BlinkWrapper<Self, D> {
            let is_blinking = $is_blinking;
            $($stop_blinking)?

            let mut wrapper = BlinkWrapper::new($self_ident, slow_blink, rapid_blink);
            if is_blinking {
                wrapper.start_blinking();
            }

            wrapper
        }
    };
}

#[cfg(feature = "alloc")]
macro_rules! impl_with_cursor {
    ($lifetime:lifetime, $self_ident:ident, $is_blinking:expr $(, $stop_blinking:stmt)?) => {
        /// Returns the backend with a new wrapper around it, drawing over cells to indicate cursor
        /// position. Every frame after a [`Terminal`] calls the [`Backend::show_cursor`] method is
        /// one that shows a cursor as long as it has _blinked_ on that frame. The [`Terminal`] can
        /// also move the cursor by calling [`Backend::set_cursor_position`].
        ///
        /// Backends without this wrapper do not display a cursor or track its position.
        ///
        /// [`Backend::set_cursor_position`]: ratatui_core::backend::Backend::set_cursor_position
        /// [`Backend::show_cursor`]: ratatui_core::backend::Backend::show_cursor
        /// [`Terminal`]: ratatui_core::terminal::Terminal
        ///
        /// # Examples
        ///
        /// ```
        /// # #[cfg(not(all(feature = "alloc", feature = "font-8x24", feature = "font-4-bits")))]
        /// # {
        /// #     compile_error!("doc-test is missing required features");
        /// # }
        /// #
        /// use dumo::DumoBackend;
        /// use dumo::blink::Blink;
        /// use dumo::cursor::{Colors, Cursor, Extent};
        /// use dumo::fonts::*;
        /// # use embedded_graphics::mock_display::MockDisplay;
        /// # use embedded_graphics::pixelcolor::Rgb565;
        ///
        /// # let mut display: MockDisplay<Rgb565> = MockDisplay::new();
        /// let backend = DumoBackend::new(&mut display, &FONT_8X24_4_BITS).with_cursor(
        ///     Cursor::default()
        ///         .blink(Blink::with_period(10))
        ///         .colors(Colors::InvertedReset)
        ///         .extent(Extent::Underline { height: 2 }),
        /// );
        /// ```
        pub fn with_cursor<$lifetime>(
            #[allow(unused_mut)] mut $self_ident,
            cursor: Cursor<$lifetime>,
        ) -> CursorWrapper<$lifetime, Self, D> {
            let is_blinking = $is_blinking;
            $($stop_blinking)?

            let mut wrapper = CursorWrapper::new($self_ident, cursor);
            if is_blinking {
                wrapper.start_blinking();
            }

            wrapper
        }
    };
}

macro_rules! impl_with_flush {
    () => {
        /// Returns the backend with a new wrapper around it, adding the specified function item to
        /// the backend. Using this wrapper, the backend will proceed to call `flush_fn` on request
        /// from a [`Terminal`], when no further changes will be made to the draw target in a given
        /// frame; this allows for device drivers to be updated as part of `flush_fn` to push pixel
        /// information to the display.
        ///
        /// Backends without this wrapper take no action in the [`Backend::flush`] method.
        ///
        /// [`Backend::flush`]: ratatui_core::backend::Backend::flush
        /// [`Terminal`]: ratatui_core::terminal::Terminal
        ///
        /// # Examples
        ///
        /// ```
        /// # #[cfg(not(all(feature = "font-8x24", feature = "font-4-bits")))]
        /// # {
        /// #     compile_error!("doc-test is missing required features");
        /// # }
        /// #
        /// use dumo::DumoBackend;
        /// use dumo::fonts::*;
        /// # use embedded_graphics::mock_display::MockDisplay;
        /// # use embedded_graphics::pixelcolor::Rgb565;
        ///
        /// # let mut display: MockDisplay<Rgb565> = MockDisplay::new();
        /// let backend = DumoBackend::new(&mut display, &FONT_8X24_4_BITS).with_flush(|display| {
        ///     // ...
        ///
        ///     Ok(())
        /// });
        /// ```
        pub const fn with_flush<F>(self, flush_fn: F) -> FlushWrapper<Self, F, D>
        where
            F: FnMut(&mut D) -> Result<(), D::Error>,
        {
            FlushWrapper::new(self, flush_fn)
        }
    };
}

impl<'a, 'b, D, C> DumoBackend<'a, 'b, '_, '_, D, C>
where
    C: PixelColor + From<C::Raw>,
    D: DrawTarget,
    D::Color: PixelColor + Default + Invert + Screen + WeightedAvg + From<Rgb888>,
    D::Error: Debug,
    RawDataSlice<'a, C::Raw, BigEndian>: IntoIterator<Item = C::Raw>,
    BitmapFontStyle<'a, 'b, D::Color, C, 1>: TextRenderer<Color = D::Color>,
{
    #[cfg(feature = "alloc")]
    impl_with_blink!(self, BLINKING);
    #[cfg(feature = "alloc")]
    impl_with_cursor!('c, self, BLINKING);
    impl_with_flush!();
}

macro_rules! impl_wrapper {
    (
        $wrapper_ident:ident
        $([$wrapper_lifetime:lifetime])?
        $(<$wrapper_type_param_ident:ident>)?
        $(($impl_type_param_ident:ident: $($impl_type_param_bounds:tt)+))?,
        $backend_type:ty
        $([$impl_lifetime:lifetime])?,
        $($impl_item:item)*
    ) => {
        impl<'a, 'b, $($impl_lifetime,)? $($impl_type_param_ident,)? D, C>
            $wrapper_ident<$($wrapper_lifetime,)? $backend_type, $($wrapper_type_param_ident,)? D>
        where
            C: PixelColor + From<C::Raw>,
            D: DrawTarget,
            D::Color: PixelColor + Default + Invert + Screen + WeightedAvg + From<Rgb888>,
            D::Error: Debug,
            $($impl_type_param_ident: $($impl_type_param_bounds)+,)?
            RawDataSlice<'a, C::Raw, BigEndian>: IntoIterator<Item = C::Raw>,
            BitmapFontStyle<'a, 'b, D::Color, C, 1>: TextRenderer<Color = D::Color>,
        {
            $($impl_item)*
        }
    }
}