u8g2-fonts 0.2.0

A text renderer for embedded-graphics, based on u8g2.
Documentation
use embedded_graphics_core::{
    prelude::{PixelColor, Point},
    primitives::Rectangle,
};

/// The vertical rendering position of the font.
///
/// All of those texts are rendered at the green line, with different `VerticalPosition` settings:
///
/// ![Vertical text positions]()
///
/// Note that metrics like [`FontRenderer::get_ascent()`](crate::FontRenderer::get_ascent) or
/// [`FontRenderer::get_descent()`](crate::FontRenderer::get_descent)
/// are relative to [`VerticalPosition::Baseline`].
///
/// The default is [`VerticalPosition::Baseline`].
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum VerticalPosition {
    /// Anchored at the font baseline
    Baseline,
    /// Anchored at the top
    Top,
    /// Anchored at the center
    Center,
    /// Anchored at the bottom
    Bottom,
}

impl Default for VerticalPosition {
    fn default() -> Self {
        Self::Baseline
    }
}

/// The dimensions of a rendered glyph/text.
#[derive(Debug, Clone, PartialEq, Eq)]
pub struct RenderedDimensions {
    /// The relative offset where a following glyph
    /// would have to get rendered.
    pub advance: Point,
    /// The bounding box of the rendered text/glyph.
    ///
    /// Can be `None` if nothing was rendered, like for
    /// a whitespace character.
    pub bounding_box: Option<Rectangle>,
}

impl RenderedDimensions {
    /// Creates an empty [`RenderedDimensions`] object with zero advance and no bounding box.
    pub const fn empty() -> Self {
        Self {
            advance: Point::new(0, 0),
            bounding_box: None,
        }
    }
}

/// The horizontal rendering position of the font.
#[derive(Debug, Clone, Copy, PartialEq, Eq)]
pub enum HorizontalAlignment {
    /// Anchored at the left side
    Left,
    /// Anchored at the center
    Center,
    /// Anchored at the right side
    Right,
}

/// The color of the rendered text.
#[allow(clippy::derive_partial_eq_without_eq)]
#[derive(Debug, Clone, Copy, PartialEq)]
pub enum FontColor<Color>
where
    Color: PixelColor,
{
    /// Only draw the text, do not touch the background.
    Transparent(Color),
    /// Draw the text and the background.
    ///
    /// Note that not all fonts support a background color.
    WithBackground {
        /// The foreground color
        fg: Color,
        /// The background color
        bg: Color,
    },
}

impl<Color> FontColor<Color>
where
    Color: PixelColor,
{
    pub(crate) fn has_background(&self) -> bool {
        matches!(self, Self::WithBackground { .. })
    }
}

#[cfg(test)]
mod tests {
    extern crate std;
    use std::println;

    use embedded_graphics_core::{pixelcolor::Rgb888, prelude::RgbColor};

    use super::*;

    fn implements_traits<T: core::fmt::Debug + Clone + PartialEq>(obj: T) {
        println!("{:?}", obj);
    }

    #[test]
    fn verticalposition_implements_traits() {
        implements_traits(VerticalPosition::Baseline);
    }

    #[test]
    fn rendereddimensions_implements_traits() {
        implements_traits(RenderedDimensions::empty());
    }

    #[test]
    fn horizontalalignment_implements_traits() {
        implements_traits(HorizontalAlignment::Center);
    }

    #[test]
    fn fontcolor_implements_traits() {
        implements_traits(FontColor::Transparent(Rgb888::BLACK));
    }
}