/glyb

glyb is an experimental graphics API agnostic text rendering, text layout, and vector canvas library.

Primary LanguageC++MIT LicenseMIT

glyb

glyb is an experimental API agnostic text rendering and canvas library built on FreeType and HarfBuzz.

  • embedded font matching and text layout
  • simple graphics library agnostic draw lists
  • offline and online multi-threaded font-atlas
  • transparent handling of multiple atlases
  • experimental gpu-accelerated 2D canvas
  • resolution independent signed distance fields

glyb

Introduction

glyb supports simple cross-platform text rendering and vector graphics for C++ apps targeting OpenGL or Vulkan.

glyb contains high level interfaces for measuring and rendering text as well as rudimentary vector graphics. glyb outputs font atlas bitmaps, vertex arrays and index arrays which can be used with OpenGL or Vulkan.

glyb code-base is largely a proof-of-concept for antialiased text zoom, and contains experimental quality code that answers questions regarding cross platform GPU-based rendering of scalable vector graphics.

Font Atlas Management

glyb manages multiple font atlases with complete Unicode coverage using a combination of pre-generated and runtime generated atlases, performing glyph lookups and rendering missing glyphs on-demand, with the emitted draw lists containing image switch and update commands, allowing for any number of Unicode faces and glyphs to be used at one time.

glyb has online and offline font atlas glyph renderers using HarfBuzz and FreeType. glyb's font atlas uses an online 2D MAXRECTS-BSSF derived bin packing algorithm, as outlined in "A Thousand Ways to Pack the Bin - A Practical Approach to Two-Dimensional Rectangle Bin Packing, Jukka Jylänki.

Signed Distance Field Fonts

glyb includes an MSDF (multi-channel signed distance field) glyph renderer that uses the msdfgen library to create variable-size MSDF font atlases. MSDF font atlases are CPU-intensive to produce so an offline tool genatlas is included to pregenerate MSDF font atlases. The advantage of MSDF font atlases is that glyphs only need to be rendered for one size. After the atlas has been generated, text renderering becomes extremely fast.

glyb includes an online multi-threaded MSDF renderer. This allows online MSDF atlas generation with any truetype font. Rendering signed distance field font atlases from truetype contours online is typically prohibitive due to CPU usage, however, multi-threading reduces latency to acceptable for real-time use. Note: msdfgen currently employs a simple n² algorithm for scanning contours which could be improved by spatial indexing and caching polynomial roots, so it will likely be possible to generate MSDF contours in real-time in the future.

The project also contains several OpenGL examples using the library.

Project Structure

  • src
    • binpack - bin packing algorithm use by the font atlas
    • canvas - gpu-accelerated distance field based 2D canvas
    • file - simple filesystem abstraction
    • font - font manager, font face and font attributes
    • glyph - font atlas, text shaper and text renderer
    • image - image with support for PNG loading and saving
    • text - text container, text layout and text part
    • utf8 - UTF-8 <-> UTF-32 conversion
  • examples
    • fontdb - example demonstrates scanning font metadata
    • ftrender - example renders glyphs to the console
    • genatlas - example MSDF font atlas batch generator
    • glbinpack - visualization of the bin packing algorithm
    • glcanvas - example GPU-acceralated Bézier font rendering
    • glfont - example that displays sample text at multiple sizes
    • gllayout - example showing text layout with line breaks
    • glsimple - simplest possible example for OpenGL
    • gldemo - animated scalability test for font atlases
  • third_party
    • freetype - font rendering engine
    • glad - OpenGL extension loader used by examples
    • glfw - OpenGL window library used by examples
    • harfbuzz - text shaping engine
    • imgui - immediate Mode Graphical User interface for C++
    • libpng - the official PNG reference library
    • msdfgen - multi-channel signed distance field generator
    • zlib - massively spiffy yet delicately unobtrusive compression library

Dependencies

glyb: FreeType , HarfBuzz , msdfgen , zlib , brotli, bzip2 , libpng . glyb examples: GLAD , GLFW , ImGui ,

Example Code

The following code snippet shows glyb's high level text layout interface:

{
    std::vector<text_segment> segments;
    std::vector<glyph_shape> shapes;
    draw_list batch;

    font_manager_ft manager;
    text_shaper_hb shaper;
    text_renderer_ft renderer(&manager);
    text_layout layout(&manager, &shaper, &renderer);
    text_container c;

    c.append(text_span(
        "    Lorem ipsum dolor sit amet, consectetur adipiscing elit, sed "
        "do eiusmod tempor incididunt ut labore et dolore magna aliqua. "
        "Ut enim ad minim veniam, quis nostrud exercitation ullamco laboris "
        "nisi ut aliquip ex ea commodo consequat. Duis aute irure dolor in "
        "reprehenderit in voluptate velit esse cillum dolore eu fugiat nulla "
        "pariatur. Excepteur sint occaecat cupidatat non proident, sunt in "
        "culpa qui officia deserunt mollit anim id est laborum.    ",
        {{ "font-family", "Roboto" },
         { "font-style", "bold" },
         { "font-size", "36" },
         { "color", "#7f7f9f" },
         { "line-height", "48" }}));

    layout.layout(segments, &c, 50, 50, 900, 700);
    for (auto &segment : segments) {
        shapes.clear();
        shaper.shape(shapes, segment);
        renderer.render(batch, shapes, segment);
    }
}

Simple GLSL vertex shader:

in vec3 a_pos;
in vec2 a_uv0;
in vec4 a_color;
in float a_gamma;

uniform mat4 u_mvp;

out vec4 v_color;
out vec2 v_uv0;
out float v_gamma;

void main() {
    v_color = a_color;
    v_gamma = a_gamma;
    v_uv0 = a_uv0;
    gl_Position = u_mvp * vec4(a_pos.xyz, 1.0);
}

Simple GLSL fragment shader:

in vec4 v_color;
in vec2 v_uv0;
int float v_gamma;

uniform sampler2D u_tex0;

out vec4 outFragColor;

void main() {
    vec4 t_color = texture(u_tex0, v_uv0);
    outFragColor = v_color * vec4(pow(t_color.rgb, vec3(1.0/v_gamma)), t_color.a);
}

Draw Lists

The render interface is abstracted using a graphics API agnostic draw list. The draw list includes information to create all necessary textures. A single batch can use several fonts, access multiple font atlas bitmaps and include other 2D geometry such as lines, rectangles, circles, and ellipses. The batch draw commands add vertices and indicies to a pair of vertex array and index array. Client code can accumulate into one large draw batch or alternatively create many if they are invalidated at different frequencies. The following diagram shows the draw list model relationships.

model Figure 1: Draw List Model Diagram

The follow examples shows OpenGL code to render a draw list. glyb does not call any graphic APIs directly making it easy to integrate with any graphics API. There is zero graphics library API code in glyb, as the interface is exclusively via the draw list. This makes it trivial to integrate with Vulkan for example. Note: in this example, cmd_shader_gl and cmd_mode_gl functions map the draw list shader and primitive types defined in draw.h to graphics library shader program and batch types.

static void display()
{
    glClearColor(1.0f, 1.0f, 1.0f, 1.0f);
    glClear(GL_COLOR_BUFFER_BIT | GL_DEPTH_BUFFER_BIT);

    for (auto img : batch.images) {
        auto ti = tex_map.find(img.iid);
        if (ti == tex_map.end()) {
            tex_map[img.iid] = image_create_texture(img);
        } else {
            image_update_texture(tex_map[img.iid], img);
        }
    }
    glBindVertexArray(vao);
    for (auto cmd : batch.cmds) {
        glUseProgram(cmd_shader_gl(cmd.shader)->pid);
        glBindTexture(GL_TEXTURE_2D, tex_map[cmd.iid]);
        glDrawElements(cmd_mode_gl(cmd.mode), cmd.count, GL_UNSIGNED_INT,
            (void*)(cmd.offset * sizeof(uint)));
    }

    glfwSwapBuffers(window);
}

Examples

glyb contains several examples programs showing how to use its API:

  • fontdb - example demonstrates scanning font metadata
  • ftrender - example renders glyphs to the console
  • genatlas - example MSDF font atlas batch generator
  • glbinpack - visualization of the bin packing algorithm
  • glcanvas - example GPU-acceralated Bézier font rendering
  • glemoji - example that renders color emoji with alpha transparency
  • glfont - example that displays sample text at multiple sizes
  • glgraph - example showing the UI9 graphical vector toolkit
  • gllayout - example showing text layout with line breaks
  • glsimple - simplest possible example for OpenGL
  • gldemo - animated scalability test for font atlases

glfont

shows sample text font size variations.

glfont

glgraph

shows graphical toolkit using the canvas.

glgraph

glcanvas

shows usage of the GPU accelerated canvas.

glcanvas-1 glcanvas-2

glglyph

shows text selection and line editing.

glglyph

gllayout

shows rudimentary text layout support.

gllayout

glemoji

shows color emoji glyph rendering.

glemoji

Building

glyb requires cmake to build. The remaining dependencies for the library and examples are included as submodules.

Source code

git clone --recursive https://github.com/michaeljclark/glyb.git
cd glyb

Windows

To create Visual Studio 2019 project, open Visual Studio 2019 x64 Native Tools command prompt, and run:

cmake -G "Visual Studio 16 2019" -A x64 -B build
cmake --build build --config RelWithDebInfo -j

Linux

The Linux cmake build is set up by default to locate FreeType, HarfBuzz and GLFW3 system packages. The following commands will install dependencies for Ubuntu 18.04, and build the project:

sudo apt-get install libfreetype6-dev libharfbuzz-dev libglfw3-dev
cmake -B build
cmake --build build --config RelWithDebInfo -j

macOS

The macOS cmake rules will build the included FreeType, HarfBuzz and GLFW3 depdencies. The following commands will build the project:

cmake -G Xcode -B build
cmake --build build --config RelWithDebInfo

Ninja

The project can be built using the ninja build tool on Windows, Linux and macOS, by specifying the Ninja generator to cmake:

cmake -G Ninja -B build
cmake --build build -- --verbose