swash 0.1.6

Font introspection, complex text shaping and glyph rendering.
Documentation
use super::hint::*;
use super::proxy::GlyfProxy;
use super::Point;

pub type CacheSlot = (u8, u8);

#[derive(Default)]
pub struct Cache {
    fonts: Vec<FontEntry>,
    sizes: Vec<SizeEntry>,
    stack: Vec<i32>,
    twilight: Vec<Point>,
    twilight_tags: Vec<u8>,
    epoch: u64,
    max_entries: usize,
}

impl Cache {
    pub fn new(max_entries: usize) -> Self {
        Self {
            fonts: Vec::new(),
            sizes: Vec::new(),
            stack: Vec::new(),
            twilight: Vec::new(),
            twilight_tags: Vec::new(),
            epoch: 0,
            max_entries,
        }
    }

    pub fn prepare(
        &mut self,
        font_id: u64,
        data: &[u8],
        proxy: &GlyfProxy,
        coords: &[i16],
        ppem: u16,
        scale: i32,
        mode: HinterMode,
    ) -> Option<CacheSlot> {
        self.epoch += 1;
        let epoch = self.epoch;
        let mut run_fpgm = false;
        let max_twilight = proxy.max_twilight as usize;
        self.twilight.resize(max_twilight * 3, Point::default());
        self.twilight_tags.resize(max_twilight, 0);
        self.stack.resize(proxy.max_stack as usize, 0);
        let font_entry = self.find_font(font_id);
        if !font_entry.0 {
            let max_fdefs = proxy.max_fdefs as usize;
            let max_idefs = proxy.max_idefs as usize;
            let font = &mut self.fonts[font_entry.1];
            font.font_id = font_id;
            font.epoch = epoch;
            font.definitions.clear();
            font.definitions
                .resize(max_fdefs + max_idefs, Function::default());
            font.max_fdefs = max_fdefs;
            font.cvt_len = (proxy.cvt.1 - proxy.cvt.0) as usize * 2;
            run_fpgm = true;
        } else {
            self.fonts[font_entry.1].epoch = epoch;
        }
        let size_entry = self.find_size(font_id, coords, scale, mode);
        let mut run_prep = false;
        if !size_entry.0 {
            let size = &mut self.sizes[size_entry.1];
            size.font_id = font_id;
            size.epoch = epoch;
            size.state = HinterState::default();
            size.mode = mode;
            size.scale = scale;
            size.coords.clear();
            size.coords.extend_from_slice(coords);
            let cvt_len = (proxy.cvt.1 - proxy.cvt.0) as usize * 2;
            size.store.clear();
            size.store.resize(cvt_len + proxy.max_storage as usize, 0);
            proxy.cvt(data, Some(scale), coords, &mut size.store);
            run_prep = true;
        } else {
            self.sizes[size_entry.1].epoch = epoch;
        }
        if run_fpgm | run_prep {
            let font = &mut self.fonts[font_entry.1];
            let size = &mut self.sizes[size_entry.1];
            let (cvt, store) = size.store.split_at_mut(font.cvt_len);
            let (fdefs, idefs) = font.definitions.split_at_mut(font.max_fdefs);
            let glyph = Zone::new(&mut [], &mut [], &mut [], &mut [], &[]);
            let max_twilight = self.twilight_tags.len();
            let (unscaled, rest) = self.twilight.split_at_mut(max_twilight);
            let (original, points) = rest.split_at_mut(max_twilight);
            let twilight_contours = [max_twilight as u16];
            let twilight = Zone::new(
                unscaled,
                original,
                points,
                &mut self.twilight_tags[..],
                &twilight_contours,
            );
            let mut hinter = Hinter::new(
                store,
                cvt,
                fdefs,
                idefs,
                &mut self.stack,
                twilight,
                glyph,
                coords,
                proxy.axis_count,
            );
            if run_fpgm {
                let mut state = HinterState::default();
                if !hinter.run_fpgm(&mut state, proxy.fpgm(data)) {
                    return None;
                }
            }
            if run_prep {
                size.state = HinterState::default();
                if !hinter.run_prep(
                    &mut size.state,
                    mode,
                    proxy.fpgm(data),
                    proxy.prep(data),
                    ppem,
                    scale,
                ) {
                    return None;
                }
            }
        }
        if !self.sizes[size_entry.1].state.hinting_enabled() {
            return None;
        }
        Some((font_entry.1 as u8, size_entry.1 as u8))
    }

    pub fn hint(
        &mut self,
        data: &[u8],
        proxy: &GlyfProxy,
        coords: &[i16],
        slot: CacheSlot,
        unscaled: &mut [Point],
        original: &mut [Point],
        scaled: &mut [Point],
        tags: &mut [u8],
        contours: &mut [u16],
        phantom: &mut [Point],
        point_base: usize,
        contour_base: usize,
        ins: &[u8],
        is_composite: bool,
    ) {
        let font = &mut self.fonts[slot.0 as usize];
        let size = &mut self.sizes[slot.1 as usize];
        if is_composite && point_base != 0 {
            for c in &mut contours[contour_base..] {
                *c -= point_base as u16;
            }
        }
        let glyph = Zone::new(
            unscaled,
            original,
            &mut scaled[point_base..],
            &mut tags[point_base..],
            &contours[contour_base..],
        );
        let twilight_len = self.twilight_tags.len();
        let twilight_contours = [twilight_len as u16];
        let (twilight_original, rest) = self.twilight.split_at_mut(twilight_len);
        let (twilight_unscaled, twilight_points) = rest.split_at_mut(twilight_len);
        let twilight = Zone::new(
            twilight_unscaled,
            twilight_original,
            twilight_points,
            &mut self.twilight_tags[..],
            &twilight_contours,
        );
        let (cvt, store) = size.store.split_at_mut(font.cvt_len);
        let (fdefs, idefs) = font.definitions.split_at_mut(font.max_fdefs);
        let mut hinter = Hinter::new(
            store,
            cvt,
            fdefs,
            idefs,
            &mut self.stack[..],
            twilight,
            glyph,
            coords,
            proxy.axis_count,
        );
        hinter.run(
            &mut size.state,
            proxy.fpgm(data),
            proxy.prep(data),
            ins,
            is_composite,
        );
        if !size.state.compat_enabled() {
            for (i, p) in (&scaled[scaled.len() - 4..]).iter().enumerate() {
                phantom[i] = *p;
            }
        }
        if is_composite && point_base != 0 {
            for c in &mut contours[contour_base..] {
                *c += point_base as u16;
            }
        }
    }

    fn find_font(&mut self, font_id: u64) -> (bool, usize) {
        let mut lowest_epoch = self.epoch;
        let mut lowest_index = 0;
        for (i, font) in self.fonts.iter().enumerate() {
            if font.font_id == font_id {
                return (true, i);
            }
            if font.epoch < lowest_epoch {
                lowest_epoch = font.epoch;
                lowest_index = i;
            }
        }
        if self.fonts.len() < self.max_entries {
            lowest_index = self.fonts.len();
            self.fonts.push(FontEntry::default());
        }
        (false, lowest_index)
    }

    fn find_size(
        &mut self,
        font_id: u64,
        coords: &[i16],
        scale: i32,
        mode: HinterMode,
    ) -> (bool, usize) {
        let mut lowest_epoch = self.epoch;
        let mut lowest_index = 0;
        let vary = !coords.is_empty();
        for (i, size) in self.sizes.iter().enumerate() {
            if size.epoch < lowest_epoch {
                lowest_epoch = size.epoch;
                lowest_index = i;
            }
            if size.font_id == font_id
                && size.scale == scale
                && size.mode == mode
                && (!vary || (coords == &size.coords[..]))
            {
                return (true, i);
            }
        }
        if self.sizes.len() < self.max_entries {
            lowest_index = self.sizes.len();
            self.sizes.push(SizeEntry::default());
        }
        (false, lowest_index)
    }
}

#[derive(Default)]
struct FontEntry {
    font_id: u64,
    definitions: Vec<Function>,
    max_fdefs: usize,
    cvt_len: usize,
    epoch: u64,
}

#[derive(Default)]
struct SizeEntry {
    font_id: u64,
    state: HinterState,
    mode: HinterMode,
    coords: Vec<i16>,
    scale: i32,
    store: Vec<i32>,
    epoch: u64,
}