1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
use std::{
    fmt::Debug,
    sync::{
        atomic::{AtomicU64, Ordering},
        Arc,
    },
};

use easygpu::prelude::*;
use figures::Figure;
use lazy_static::lazy_static;
use rusttype::{gpu_cache, Scale};

use crate::math::Pixels;

lazy_static! {
    static ref GLOBAL_ID_CELL: AtomicU64 = AtomicU64::new(0);
}

/// Embeds a font into your executable.
#[macro_export]
macro_rules! include_font {
    ($path:expr) => {{
        let bytes = std::include_bytes!($path);
        Font::try_from_bytes(bytes as &[u8]).expect("Error loading bundled font")
    }};
}

/// Font provides TrueType Font rendering
#[derive(Clone, Debug)]
pub struct Font {
    pub(crate) handle: Arc<FontData>,
}

impl Font {
    /// Try to load a font from `bytes`.
    #[must_use]
    pub fn try_from_bytes(bytes: &'static [u8]) -> Option<Self> {
        let font = rusttype::Font::try_from_bytes(bytes)?;
        let id = GLOBAL_ID_CELL.fetch_add(1, Ordering::SeqCst);
        Some(Self {
            handle: Arc::new(FontData { id, font }),
        })
    }

    /// Returns the unique ID of the font. This ID depends on load order, and is
    /// not from the font data.
    #[must_use]
    pub fn id(&self) -> u64 {
        self.handle.id
    }

    /// Measures the vertical metrics for a given size.
    #[must_use]
    pub fn metrics(&self, size: Figure<f32, Pixels>) -> rusttype::VMetrics {
        self.handle
            .font
            .v_metrics(rusttype::Scale::uniform(size.get()))
    }

    /// Returns the name of the font's family, if available.
    #[must_use]
    pub fn family(&self) -> Option<String> {
        match &self.handle.font {
            rusttype::Font::Ref(f) => f.family_name(),
            rusttype::Font::Owned(_) => None,
        }
    }

    #[must_use]
    pub(crate) fn glyph(&self, c: char) -> rusttype::Glyph<'static> {
        self.handle.font.glyph(c)
    }

    #[must_use]
    pub(crate) fn pair_kerning(
        &self,
        size: f32,
        a: rusttype::GlyphId,
        b: rusttype::GlyphId,
    ) -> f32 {
        self.handle.font.pair_kerning(Scale::uniform(size), a, b)
    }
}

#[derive(Debug)]
pub struct FontData {
    pub(crate) id: u64,
    pub(crate) font: rusttype::Font<'static>,
}

pub struct LoadedFont {
    pub font: Font,
    pub cache: gpu_cache::Cache<'static>,
    pub(crate) binding: Option<BindingGroup>,
    pub(crate) texture: Option<Texture>,
}

impl Debug for LoadedFont {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        f.debug_struct("LoadedFont")
            .field("font", &self.font)
            .field("binding", &self.binding)
            .field("texture", &self.texture)
            .finish()
    }
}

impl LoadedFont {
    pub fn new(font: &Font) -> Self {
        Self {
            font: font.clone(),
            cache: gpu_cache::Cache::builder().dimensions(512, 512).build(),
            binding: None,
            texture: None,
        }
    }
}