fdsm

A pure-Rust reimplementation of multi-channel signed distance field generation.

This implementation mostly follows Victor Chlumský’s master thesis. Although it is not an exact translation of the C++ msdfgen library, it does follow it for some parts of the code.

fdsm also uses code adapted from msdfgen-rs for its tests.

Crate features

Usage

```rust,norun use fdsm::bezier::scanline::FillRule; use fdsm::generate::generatemsdf; use fdsm::render::{correctsignmsdf, rendermsdf}; use fdsm::shape::Shape; use fdsm::transform::Transform; use image::{GrayImage, RgbImage}; use nalgebra::{Affine2, Similarity2, Vector2}; use ttfparser::Face;

// First, acquire a [Shape]. This can be done by procedurally // generating one or by loading one from a font:

let face = Face::parse(notosans::REGULARTTF, 0).unwrap(); let glyphid = face.glyphindex('A').unwrap(); let mut shape = Shape::loadfromface(&face, glyphid);

// Prepare your transformation matrix and calculate the dimensions of // the resulting signed distance field. As an example, we set this up // using ‘shrinkage’ (font units per texel) and ‘range’ (number of // texels for the margin) values. // Note that since font files interpret a positive y-offset as // pointing up, the resulting distance field will be upside-down. // This can be corrected either by flipping the resulting image // vertically or by modifying the transformation matrix. We omit // this fix for simplicity.

let bbox = face.glyphboundingbox(glyph_id).unwrap();

const RANGE: f64 = 4.0; const SHRINKAGE: f64 = 16.0; let transformation = nalgebra::convert::<_, Affine2>(Similarity2::new( Vector2::new( RANGE - bbox.xmin as f64 / SHRINKAGE, RANGE - bbox.ymin as f64 / SHRINKAGE, ), 0.0, 1.0 / SHRINKAGE, )); let width = ((bbox.xmax as f64 - bbox.xmin as f64) / SHRINKAGE + 2.0 * RANGE).ceil() as u32; let height = ((bbox.ymax as f64 - bbox.ymin as f64) / SHRINKAGE + 2.0 * RANGE).ceil() as u32;

// Unlike msdfgen, the transformation is not passed into the // generate_msdf` function – the coordinates of the control points // must be expressed in terms of pixels on the distance field. To get // the correct units, we pre-transform the shape:

shape.transform(&transformation);

// We now color the edges of the shape. We also have to prepare // it for calculations:

let coloredshape = Shape::edgecoloringsimple(shape, 0.03, 69441337420); let preparedcoloredshape = coloredshape.prepare();

// Set up the resulting image and generate the distance field:

let mut msdf = RgbImage::new(width, height); generatemsdf(&preparedcoloredshape, RANGE, &mut msdf); correctsignmsdf(&mut msdf, &preparedcolored_shape, FillRule::Nonzero);

// As a test, try previewing the distance field:

let mut preview = GrayImage::new(msdf.width() * 10, msdf.height() * 10); render_msdf(&msdf, &mut preview, RANGE); ```

Roadmap

Currently, fdsm has the basic functionality of generating MSDFs and generates correct distance fields for the glyphs A to Z in Noto Sans. However, it does not have all of the features present in msdfgen.