use std::collections::BTreeMap;
use fontdrasil::{
orchestration::{Access, AccessBuilder, Work},
types::Axes,
variations::ModelDeltas,
};
use fontir::orchestration::WorkId as FeWorkId;
use write_fonts::{
OtRound,
tables::{
mvar::{Mvar, ValueRecord},
variations::{VariationRegion, ivs_builder::VariationStoreBuilder},
},
types::{MajorMinor, Tag},
};
use crate::{
error::Error,
orchestration::{AnyWorkId, BeWork, Context, WorkId},
};
#[derive(Debug)]
struct MvarWork {}
pub fn create_mvar_work() -> Box<BeWork> {
Box::new(MvarWork {})
}
struct MvarBuilder {
axes: Axes,
deltas: BTreeMap<Tag, Vec<(VariationRegion, i16)>>,
}
impl MvarBuilder {
fn new(axes: Axes) -> Self {
MvarBuilder {
axes,
deltas: BTreeMap::new(),
}
}
fn add_deltas(&mut self, mvar_tag: Tag, deltas: &ModelDeltas<f64>) {
if deltas.len() == 1 {
let (region, _) = deltas.first().unwrap();
assert!(region.is_default());
return;
}
let deltas: Vec<_> = deltas
.iter()
.filter_map(|(region, values)| {
if region.is_default() {
return None;
}
assert!(values.len() == 1, "{} values?!", values.len());
Some((
region.to_write_fonts_variation_region(&self.axes),
values[0].ot_round(),
))
})
.collect();
if deltas.iter().all(|(_, delta)| *delta == 0) {
return;
}
self.deltas.insert(mvar_tag, deltas);
}
fn build(self) -> Option<Mvar> {
let mut builder = VariationStoreBuilder::new(self.axes.len() as u16);
let delta_ids = self
.deltas
.into_iter()
.map(|(tag, deltas)| (tag, builder.add_deltas(deltas)))
.collect::<Vec<_>>();
let (varstore, index_map) = builder.build();
let records = delta_ids
.into_iter()
.map(|(tag, temp_id)| {
let varidx = index_map.get(temp_id).unwrap();
ValueRecord::new(
tag,
varidx.delta_set_outer_index,
varidx.delta_set_inner_index,
)
})
.collect::<Vec<_>>();
(!records.is_empty()).then(|| Mvar::new(MajorMinor::VERSION_1_0, Some(varstore), records))
}
}
impl Work<Context, AnyWorkId, Error> for MvarWork {
fn id(&self) -> AnyWorkId {
WorkId::Mvar.into()
}
fn read_access(&self) -> Access<AnyWorkId> {
AccessBuilder::new()
.variant(FeWorkId::StaticMetadata)
.variant(FeWorkId::GlobalMetrics)
.build()
}
fn exec(&self, context: &Context) -> Result<(), Error> {
let static_metadata = context.ir.static_metadata.get();
let metrics = context.ir.global_metrics.get();
let mut mvar_builder = MvarBuilder::new(static_metadata.axes.clone());
for (metric, deltas) in metrics.iter() {
if let Some(mvar_tag) = metric.mvar_tag() {
mvar_builder.add_deltas(mvar_tag, deltas);
}
}
if let Some(mvar) = mvar_builder.build() {
context.mvar.set(mvar);
}
Ok(())
}
}
#[cfg(test)]
mod tests {
use std::str::FromStr;
use fontdrasil::{coords::NormalizedLocation, types::Axis};
use fontir::ir::{GlobalMetric, GlobalMetrics, GlobalMetricsBuilder};
use write_fonts::{
dump_table,
read::{
FontData, FontRead,
tables::{mvar as read_mvar, variations::ItemVariationData},
},
types::F2Dot14,
};
use super::*;
use crate::test_util;
fn new_mvar_builder(axes: Vec<Axis>) -> MvarBuilder {
let axes = Axes::new(axes);
MvarBuilder::new(axes)
}
fn build_metrics(
metrics: &[(GlobalMetric, &[(&NormalizedLocation, f64)])],
axes: &Axes,
) -> GlobalMetrics {
metrics
.iter()
.flat_map(|(metric, values)| values.iter().map(move |value| (metric, value)))
.fold(
GlobalMetricsBuilder::new(),
|mut builder, (&metric, (pos, value))| {
builder.set(metric, (*pos).clone(), *value);
builder
},
)
.build(axes)
.unwrap()
}
fn add_deltas(builder: &mut MvarBuilder, metrics: GlobalMetrics) {
metrics.iter().for_each(|(metric, values)| {
builder.add_deltas(metric.mvar_tag().unwrap(), values);
});
}
fn delta_sets(var_data: &ItemVariationData) -> Vec<Vec<i32>> {
(0..var_data.item_count())
.map(|i| var_data.delta_set(i).collect::<Vec<_>>())
.collect()
}
#[test]
fn smoke_test() {
let regular = NormalizedLocation::for_pos(&[("wght", 0.0)]);
let bold = NormalizedLocation::for_pos(&[("wght", 1.0)]);
let axes = vec![test_util::axis("wght", 400.0, 400.0, 700.0)];
let mut builder = new_mvar_builder(axes.clone());
let metrics = build_metrics(
&[(GlobalMetric::XHeight, &[(®ular, 500.0), (&bold, 550.0)])],
&axes.into(),
);
add_deltas(&mut builder, metrics);
let Some(mvar) = builder.build() else {
panic!("no MVAR?!");
};
let bytes = dump_table(&mvar).unwrap();
let mvar = read_mvar::Mvar::read(FontData::new(&bytes)).unwrap();
assert_eq!(mvar.version(), MajorMinor::VERSION_1_0);
assert_eq!(mvar.value_records().len(), 1);
let rec = &mvar.value_records()[0];
assert_eq!(rec.value_tag(), Tag::new(b"xhgt"));
assert_eq!(rec.delta_set_outer_index(), 0);
assert_eq!(rec.delta_set_inner_index(), 0);
let Some(Ok(varstore)) = mvar.item_variation_store() else {
panic!("MVAR has no ItemVariationStore?!");
};
assert_eq!(varstore.variation_region_list().unwrap().region_count(), 1);
assert_eq!(varstore.item_variation_data_count(), 1);
let vardata = varstore.item_variation_data().get(0).unwrap().unwrap();
assert_eq!(vardata.region_indexes(), &[0]);
assert_eq!(delta_sets(&vardata), vec![vec![50]]);
}
#[test]
fn no_variations_no_mvar() {
let regular = NormalizedLocation::for_pos(&[("wght", 0.0)]);
let bold = NormalizedLocation::for_pos(&[("wght", 1.0)]);
let axes = vec![test_util::axis("wght", 400.0, 400.0, 700.0)];
let mut builder = new_mvar_builder(axes.clone());
let metrics = build_metrics(
&[
(GlobalMetric::XHeight, &[(®ular, 500.0), (&bold, 500.0)]),
(
GlobalMetric::CapHeight,
&[(®ular, 800.0), (&bold, 800.0)],
),
],
&axes.into(),
);
add_deltas(&mut builder, metrics);
assert!(builder.build().is_none());
}
struct MvarReader<'a> {
mvar: read_mvar::Mvar<'a>,
}
impl<'a> MvarReader<'a> {
fn new(mvar: read_mvar::Mvar<'a>) -> Self {
Self { mvar }
}
fn metric_delta(&self, mvar_tag: &str, coords: &[f64]) -> f64 {
let mvar_tag = Tag::from_str(mvar_tag).unwrap();
let coords: Vec<F2Dot14> = coords
.iter()
.map(|coord| F2Dot14::from_f32(*coord as _))
.collect();
self.mvar.metric_delta(mvar_tag, &coords).unwrap().to_f64()
}
}
#[test]
fn sparse_global_metrics() {
let regular = NormalizedLocation::for_pos(&[("wght", 0.0)]);
let medium = NormalizedLocation::for_pos(&[("wght", 0.5)]);
let bold = NormalizedLocation::for_pos(&[("wght", 1.0)]);
let axes = vec![test_util::axis("wght", 400.0, 400.0, 700.0)];
let mut builder = new_mvar_builder(axes.clone());
let metrics = build_metrics(
&[
(
GlobalMetric::XHeight,
&[(®ular, 500.0), (&medium, 530.0), (&bold, 550.0)],
),
(
GlobalMetric::StrikeoutSize,
&[(®ular, 50.0), (&bold, 100.0)],
),
],
&axes.into(),
);
add_deltas(&mut builder, metrics);
let Some(mvar) = builder.build() else {
panic!("no MVAR?!");
};
let bytes = dump_table(&mvar).unwrap();
let mvar = read_mvar::Mvar::read(FontData::new(&bytes)).unwrap();
assert_eq!(mvar.value_records().len(), 2);
assert_eq!(mvar.value_records()[0].value_tag(), Tag::new(b"strs"));
assert_eq!(mvar.value_records()[1].value_tag(), Tag::new(b"xhgt"));
let mvar = MvarReader::new(mvar);
assert_eq!(mvar.metric_delta("xhgt", &[0.0]), 0.0);
assert_eq!(mvar.metric_delta("xhgt", &[0.5]), 30.0); assert_eq!(mvar.metric_delta("xhgt", &[1.0]), 50.0);
assert_eq!(mvar.metric_delta("strs", &[0.0]), 0.0);
assert_eq!(mvar.metric_delta("strs", &[0.5]), 25.0); assert_eq!(mvar.metric_delta("strs", &[1.0]), 50.0);
}
}