Skip to content

Commit

Permalink
astral/text/freetype_face_generator: work arounds for when Freetype r…
Browse files Browse the repository at this point in the history
…eports a zero size for a COLR0 glyph

This issue is seen in the font from https://github.com/mozilla/twemoji-colr/.
Indeed, this issue: mozilla/twemoji-colr#50
which dates back to 2019 is still active (so over 3 and half years)
where some browsers do not render anything either for the glyphs.

This code does the work around that if the size is reported as 0 for a COLR0
glyph, then we derived the layout offsets and size entirely from the union
of the bounding boxes of the layers. Seems to work.
  • Loading branch information
krogovin authored and kevin-rogovin committed Feb 17, 2023
1 parent 153a29f commit e7786af
Showing 1 changed file with 53 additions and 6 deletions.
59 changes: 53 additions & 6 deletions src/astral/text/freetype_face_generator.cpp
Original file line number Diff line number Diff line change
Expand Up @@ -196,10 +196,10 @@ namespace
std::vector<enum astral::fill_rule_t> *out_fill_rules);

static
void
bool
grab_metrics(FT_GlyphSlot glyph,
astral::GlyphMetrics *out_metrics,
bool scale_by_64);
bool is_image_glyph);

static
bool //returns if the glyph has color, i.e. a pixel with (R, G, B) not all the same
Expand Down Expand Up @@ -377,11 +377,11 @@ add_path(FT_Outline &outline,
OutlineDecompser::extract_path(&out_paths->back(), outline, true);
}

void
bool
GlyphGeneratorFreetype::
grab_metrics(FT_GlyphSlot glyph,
astral::GlyphMetrics *out_metrics,
bool scale_by_64)
bool is_image_glyph)
{
using namespace astral;
const FT_Glyph_Metrics &metrics(glyph->metrics);
Expand All @@ -402,7 +402,7 @@ grab_metrics(FT_GlyphSlot glyph,
out_metrics->m_bb.clear();
FT_Get_Glyph(glyph, &ggg);

if (scale_by_64)
if (is_image_glyph)
{
/* When loading bitmap data, the metrics are in 26.6 units, so
* we need to divide by 64.0f
Expand All @@ -423,6 +423,7 @@ grab_metrics(FT_GlyphSlot glyph,
out_metrics->m_bb.union_point(vec2(bb.xMax, -bb.yMin));
}

return metrics.width == 0 || metrics.height == 0;
}

void
Expand All @@ -439,12 +440,13 @@ scalable_glyph_info(unsigned int thread_slot,
FT_Face face;
unsigned int num_layers = 0;
FT_UInt layer_glyph_index, layer_color_index;
bool empty_size;

ASTRALassert(thread_slot < m_faces.size());
face = m_faces[thread_slot]->face();

FT_Load_Glyph(face, glyph_index.m_value, m_load_flags);
grab_metrics(face->glyph, out_metrics, false);
empty_size = grab_metrics(face->glyph, out_metrics, false);

#ifdef COLOR_GLYPH_LAYER_SUPPORTED
{
Expand Down Expand Up @@ -501,6 +503,51 @@ scalable_glyph_info(unsigned int thread_slot,
/* load the glyph specified by layer_glyph_index */
FT_Load_Glyph(face, layer_glyph_index, m_load_flags);
add_path(face->glyph->outline, out_paths, out_fill_rules);

/* if the metrics from the base layer are empty, chances
* are the glyph is busted as seen in the font from
* https://github.com/mozilla/twemoji-colr/ (as of Dec 22, 2022).
* To work-around this, we enlarge the bounding box by the
* bounding box of each of the layer glyphs and below
* we use the bounding box to compute the various metrics
* values.
*/
if (empty_size)
{
FT_Glyph ggg;
FT_BBox ft_bb;

FT_Get_Glyph(face->glyph, &ggg);
FT_Glyph_Get_CBox(ggg, FT_GLYPH_BBOX_UNSCALED, &ft_bb);
out_metrics->m_bb.union_point(vec2(ft_bb.xMin, -ft_bb.yMax));
out_metrics->m_bb.union_point(vec2(ft_bb.xMax, -ft_bb.yMin));
}
}

if (empty_size)
{
float layout_offset;

/* The placement of glyphs is as follows:
* - max.y <---> layout_offset
* - min.y <---> max.y - size.y
*
* So:
* layout_offset = max.y
*/
layout_offset = out_metrics->m_bb.max_point().y();

out_metrics->m_size = out_metrics->m_bb.size();
out_metrics->m_horizontal_layout_offset.y() = layout_offset;
out_metrics->m_vertical_layout_offset.y() = layout_offset;

/* Some guessing here, but we will just assume that like
* the y-coordinate, the layou offset for x will match
* the bounding box.
*/
layout_offset = out_metrics->m_bb.min_point().x();
out_metrics->m_horizontal_layout_offset.x() = layout_offset;
out_metrics->m_vertical_layout_offset.x() = layout_offset;
}
}
#endif
Expand Down

0 comments on commit e7786af

Please sign in to comment.