This is a utility for generating compact font atlases using MSDFgen.
The atlas generator loads a subset of glyphs from a TTF or OTF font file, generates a distance field for each of them, and tightly packs them into an atlas bitmap (example below). The finished atlas and/or its layout metadata can be exported as an Artery Font file, a plain image file, a CSV sheet or a structured JSON file.
A font atlas is typically stored in texture memory and used to draw text in real-time rendering contexts such as video games.
- See what's new in the changelog.
The atlas generator can generate the following six types of atlases.
Notes:
- Sharp corners refers to preservation of corner sharpness when upscaled.
- Soft effects refers to the support of effects that use true distance, such as glows, rounded outlines, or simplified shadows.
- Hard effects refers to the support of effects that use perpendicular distance, such as mitered outlines or thickness adjustment.
This project can be used either as a library or as a standalone console program.
Examples of how to use it as a library are available at the bottom of the page.
To start using the program right away, you may download a Windows binary in the "Releases" section.
To build the project from source, you may use the included CMake script.
In its default configuration, it requires vcpkg as the provider for third-party library dependencies.
If you set the environment variable VCPKG_ROOT to the vcpkg directory, the CMake configuration will take care of fetching all required packages from vcpkg.
Use the following command line arguments for the standalone version of the atlas generator.
- -font <fontfile.ttf/otf>(required) – sets the input font file.- Alternatively, use -varfont <fontfile.ttf/otf?var0=value0&var1=value1>to configure a variable font.
 
- Alternatively, use 
- -charset <charset.txt>– sets the character set. See the syntax specification of- charset.txt.
- -glyphset <glyphset.txt>– sets the set of input glyphs using their indices within the font file. See the syntax specification.
- -chars/- -glyphs <set string>sets the above character / glyph set in-line. See the syntax specification.
- -allglyphs– sets the set of input glyphs to all glyphs present within the font file.
- -fontscale <scale>– applies a scaling transformation to the font's glyphs. Mainly to be used to generate multiple sizes in a single atlas, otherwise use- -size.
- -fontname <name>– sets a name for the font that will be stored in certain output files as metadata.
- -and– separates multiple inputs to be combined into a single atlas.
If no character set or glyph set is provided, and -allglyphs is not used, the ASCII charset will be used.
-type <type> – see Atlas types
<type> can be one of:
- hardmask– a non-anti-aliased binary image
- softmask– an anti-aliased image
- sdf– a true signed distance field (SDF)
- psdf– a signed perpendicular distance field (PSDF)
- msdf(default) – a multi-channel signed distance field (MSDF)
- mtsdf– a combination of MSDF and true SDF in the alpha channel
-format <format>
<format> can be one of:
- png– a compressed PNG image
- bmp– an uncompressed BMP image
- tiff– an uncompressed floating-point TIFF image
- rgba– an uncompressed RGBA file
- fl32– an uncompressed floating-point FL32 file
- text– a sequence of pixel values in plain text
- textfloat– a sequence of floating-point pixel values in plain text
- bin– a sequence of pixel values encoded as raw bytes of data
- binfloat– a sequence of pixel values encoded as raw 32-bit floating-point values (little endian,- binfloatbefor big endian)
If format is not specified, it may be deduced from the extension of the -imageout argument or other clues.
Please note that all color values must be interpreted as if they were linear (not sRGB) like the alpha channel, even if the image format implies otherwise.
-dimensions <width> <height> – sets fixed atlas dimensions
Alternatively, the minimum possible dimensions may be selected automatically if a dimensions constraint is set instead:
- -pots– a power-of-two square
- -potr– a power-of-two square or rectangle (typically 2:1 aspect ratio)
- -square– any square dimensions
- -square2– square with even side length
- -square4(default) – square with side length divisible by four
By default, glyphs in the atlas have different dimensions and are bin-packed in an irregular fashion to maximize use of space.
With the -uniformgrid switch, you can instead force all glyphs to have identical dimensions and be laid out in a grid.
In that case, these additional options are available to customize the layout:
- -uniformcols <N>– sets the number of columns
- -uniformcell <width> <height>– sets the dimensions of the grid's cells
- -uniformcellconstraint <none / pots / potr / square / square2 / square4>– sets constraint for cell dimensions (see explanation of options above)
- -uniformorigin <off / on / horizontal / vertical>– sets whether the glyph's origin point should be fixed at the same position in each cell
Any non-empty subset of the following may be specified:
- -imageout <filename.*>– saves the atlas bitmap as a plain image file. Format matches- -format
- -json <filename.json>– writes the atlas's layout data as well as other metrics into a structured JSON file- JSON fields- atlassection includes the settings used to generate the atlas, including its type and dimensions. The- sizefield represents the font size in pixels per em.
- If there are multiple input fonts (-andparameter), the remaining data are grouped intovariants, each representing an input font.
- metricssection contains useful font metric values retrieved from the font. All values are in em's.
- glyphsis an array of individual glyphs identified by Unicode character index (- unicode) or glyph index (- index), depending on whether character set or glyph set mode is used.- advanceis the horizontal advance in em's.
- planeBoundsrepresents the glyph quad's bounds in em's relative to the baseline and horizontal cursor position.
- atlasBoundsrepresents the glyph's bounds in the atlas in pixels.
 
- If available, kerninglists all kerning pairs and their advance adjustment (which needs to be added to the base advance of the first glyph in the pair).
 
- -csv <filename.csv>– writes the glyph layout data into a simple CSV file- CSV columns- If there are multiple input fonts (-andparameter), the first column is the font index, otherwise it is skipped.
- Character Unicode value or glyph index, depending on whether character set or glyph set mode is used.
- Horizontal advance in em's.
- The next 4 columns are the glyph quad's bounds in em's relative to the baseline and cursor. Depending on the -yoriginsetting, this is either left, bottom, right, top (bottom-up Y) or left, top, right, bottom (top-down Y).
- The last 4 columns the the glyph's bounds in the atlas in pixels. Depending on the -yoriginsetting, this is either left, bottom, right, top (bottom-up Y) or left, top, right, bottom (top-down Y).
 
- If there are multiple input fonts (
- -arfont <filename.arfont>– saves the atlas and its layout data as an Artery Font file
- -shadronpreview <filename.shadron> <sample text>– generates a Shadron script that uses the generated atlas to draw a sample text as a preview
- -size <em size>– sets the size of the glyphs in the atlas in pixels per em
- -minsize <em size>– sets the minimum size. The largest possible size that fits the same atlas dimensions will be used
- -emrange <em range>– sets the distance field range in em's
- -pxrange <pixel range>(default = 2) – sets the distance field range in output pixels
- -aemrange/- -apxrange <outermost distance> <innermost distance>– sets the distance field range asymmetrically by specifying the minimum and maximum representable signed distances (outside distances are negative!)
- -pxalign <off / on / horizontal / vertical>(default = vertical) – enables or disables alignment of glyph's origin point with the pixel grid
- -empadding/- -pxpadding <width>– sets additional padding within each glyph's box (in em's / pixels)
- -outerempadding/- -outerpxpadding <width>– sets additional padding around each glyph's box
- -aempadding/- -apxpadding/- -aouterempadding/- -aouterpxpadding <left> <bottom> <right> <top>– sets additional padding (see above) asymmetrically with a separate width value for each side
- -angle <angle>– sets the minimum angle between adjacent edges to be considered a corner. Append D for degrees (- msdf/- mtsdfonly)
- -coloringstrategy <simple / inktrap / distance>– selects the edge coloring heuristic (- msdf/- mtsdfonly)
- -errorcorrection <mode>– selects the error correction algorithm. Use- helpas mode for more information (- msdf/- mtsdfonly)
- -miterlimit <value>– sets the miter limit that limits the extension of each glyph's bounding box due to very sharp corners (- psdf/- msdf/- mtsdfonly)
- -overlap– switches to distance field generator with support for overlapping contours
- -nopreprocess– disables path preprocessing which resolves self-intersections and overlapping contours
- -scanline– performs an additional scanline pass to fix the signs of the distances
- -seed <N>– sets the initial seed for the edge coloring heuristic
- -threads <N>– sets the number of threads for the parallel computation (0 = auto)
- -yorigin <bottom / top>– specifies the direction of the Y-axis in output coordinates. The default is bottom-up.
Use -help for an exhaustive list of options.
The character set file is a text file with UTF-8 or ASCII encoding. The characters can be denoted in the following ways:
- Single character: 'A'(UTF-8 encoded),65(decimal Unicode),0x41(hexadecimal Unicode)
- Range of characters: ['A', 'Z'],[65, 90],[0x41, 0x5a]
- String of characters: "ABCDEFGHIJKLMNOPQRSTUVWXYZ"(UTF-8 encoded)
The entries should be separated by commas or whitespace.
In between quotation marks, backslash is used as the escape character (e.g. '\'', '\\', "!\"#").
The order in which characters appear is not taken into consideration.
Additionally, the include directive can be used to include other charset files and combine character sets in a hierarchical way. It must be written on a separate line:
@include "base-charset.txt"
The syntax of the glyph set specification is mostly the same as that of a character set, but only numeric values (decimal and hexadecimal) are allowed.
Here are commented snippets of code that demonstrate how the project can be used as a library.
#include <msdf-atlas-gen/msdf-atlas-gen.h>
using namespace msdf_atlas;
bool generateAtlas(const char *fontFilename) {
    bool success = false;
    // Initialize instance of FreeType library
    if (msdfgen::FreetypeHandle *ft = msdfgen::initializeFreetype()) {
        // Load font file
        if (msdfgen::FontHandle *font = msdfgen::loadFont(ft, fontFilename)) {
            // Storage for glyph geometry and their coordinates in the atlas
            std::vector<GlyphGeometry> glyphs;
            // FontGeometry is a helper class that loads a set of glyphs from a single font.
            // It can also be used to get additional font metrics, kerning information, etc.
            FontGeometry fontGeometry(&glyphs);
            // Load a set of character glyphs:
            // The second argument can be ignored unless you mix different font sizes in one atlas.
            // In the last argument, you can specify a charset other than ASCII.
            // To load specific glyph indices, use loadGlyphs instead.
            fontGeometry.loadCharset(font, 1.0, Charset::ASCII);
            // Apply MSDF edge coloring. See edge-coloring.h for other coloring strategies.
            const double maxCornerAngle = 3.0;
            for (GlyphGeometry &glyph : glyphs)
                glyph.edgeColoring(&msdfgen::edgeColoringInkTrap, maxCornerAngle, 0);
            // TightAtlasPacker class computes the layout of the atlas.
            TightAtlasPacker packer;
            // Set atlas parameters:
            // setDimensions or setDimensionsConstraint to find the best value
            packer.setDimensionsConstraint(TightAtlasPacker::DimensionsConstraint::SQUARE);
            // setScale for a fixed size or setMinimumScale to use the largest that fits
            packer.setMinimumScale(24.0);
            // setPixelRange or setUnitRange
            packer.setPixelRange(2.0);
            packer.setMiterLimit(1.0);
            // Compute atlas layout - pack glyphs
            packer.pack(glyphs.data(), glyphs.size());
            // Get final atlas dimensions
            int width = 0, height = 0;
            packer.getDimensions(width, height);
            // The ImmediateAtlasGenerator class facilitates the generation of the atlas bitmap.
            ImmediateAtlasGenerator<
                float, // pixel type of buffer for individual glyphs depends on generator function
                3, // number of atlas color channels
                msdfGenerator, // function to generate bitmaps for individual glyphs
                BitmapAtlasStorage<byte, 3> // class that stores the atlas bitmap
                // For example, a custom atlas storage class that stores it in VRAM can be used.
            > generator(width, height);
            // GeneratorAttributes can be modified to change the generator's default settings.
            GeneratorAttributes attributes;
            generator.setAttributes(attributes);
            generator.setThreadCount(4);
            // Generate atlas bitmap
            generator.generate(glyphs.data(), glyphs.size());
            // The atlas bitmap can now be retrieved via atlasStorage as a BitmapConstRef.
            // The glyphs array (or fontGeometry) contains positioning data for typesetting text.
            success = my_project::submitAtlasBitmapAndLayout(generator.atlasStorage(), glyphs);
            // Cleanup
            msdfgen::destroyFont(font);
        }
        msdfgen::deinitializeFreetype(ft);
    }
    return success;
}The DynamicAtlas class allows you to add glyphs to the atlas "on-the-fly" as they are needed. In this example, the ImmediateAtlasGenerator is used as the underlying atlas generator, which however isn't ideal for this purpose because it may launch new threads every time. In practice, you would typically define your own atlas generator class that properly handles your specific performance and synchronization requirements.
Acquiring the GlyphGeometry objects can be adapted from the previous example.
#include <msdf-atlas-gen/msdf-atlas-gen.h>
using namespace msdf_atlas;
using MyDynamicAtlas = DynamicAtlas<ImmediateAtlasGenerator<float, 3, msdfGenerator, BitmapAtlasStorage<byte, 3>>>;
const double pixelRange = 2.0;
const double glyphScale = 32.0;
const double miterLimit = 1.0;
const double maxCornerAngle = 3.0;
MyDynamicAtlas atlas;
void addGlyphsToAtlas(GlyphGeometry *glyphs, int count) {
    for (int i = 0; i < count; ++i) {
        // Apply MSDF edge coloring. See edge-coloring.h for other coloring strategies.
        glyphs[i].edgeColoring(&msdfgen::edgeColoringInkTrap, maxCornerAngle, 0);
        // Finalize glyph box size based on the parameters
        glyphs[i].wrapBox(glyphScale, pixelRange/glyphScale, miterLimit);
    }
    // Add glyphs to atlas - invokes the underlying atlas generator
    // Adding multiple glyphs at once may improve packing efficiency.
    MyDynamicAtlas::ChangeFlags change = atlas.add(glyphs, count);
    if (change&MyDynamicAtlas::RESIZED) {
        // Atlas has been enlarged - can be handled here or directly in custom generator class
    }
    // Glyph positioning data is now stored in glyphs.
}The atlas storage (and its bitmap) can be accessed as dynamicAtlas.atlasGenerator().atlasStorage().






