[hb-view/hb-shape] Add --glyphs

This makes hb-view take output of hb-shape and render it.
This commit is contained in:
Behdad Esfahbod 2022-12-23 13:13:29 -07:00
parent 30c5402e3d
commit d45f7265e9
2 changed files with 40 additions and 6 deletions

View File

@ -61,7 +61,7 @@ struct shape_consumer_t : shape_options_t
for (unsigned int n = num_iterations; n; n--) for (unsigned int n = num_iterations; n; n--)
{ {
populate_buffer (buffer, text, text_len, app.text_before, app.text_after); populate_buffer (buffer, text, text_len, app.text_before, app.text_after, app.font);
if (n == 1) if (n == 1)
output.consume_text (buffer, text, text_len, utf8_clusters); output.consume_text (buffer, text, text_len, utf8_clusters);

View File

@ -64,9 +64,21 @@ struct shape_options_t
} }
void populate_buffer (hb_buffer_t *buffer, const char *text, int text_len, void populate_buffer (hb_buffer_t *buffer, const char *text, int text_len,
const char *text_before, const char *text_after) const char *text_before, const char *text_after,
hb_font_t *font)
{ {
hb_buffer_clear_contents (buffer); hb_buffer_clear_contents (buffer);
if (glyphs)
{
hb_buffer_deserialize_glyphs (buffer,
text, text_len,
nullptr,
font,
HB_BUFFER_SERIALIZE_FORMAT_TEXT);
return;
}
if (text_before) { if (text_before) {
unsigned int len = strlen (text_before); unsigned int len = strlen (text_before);
hb_buffer_add_utf8 (buffer, text_before, len, len, 0); hb_buffer_add_utf8 (buffer, text_before, len, len, 0);
@ -93,11 +105,31 @@ struct shape_options_t
hb_bool_t shape (hb_font_t *font, hb_buffer_t *buffer, const char **error=nullptr) hb_bool_t shape (hb_font_t *font, hb_buffer_t *buffer, const char **error=nullptr)
{ {
if (!hb_shape_full (font, buffer, features, num_features, shapers)) if (glyphs)
{ {
if (error) /* Scale positions. */
*error = "Shaping failed."; int x_scale, y_scale;
goto fail; hb_font_get_scale (font, &x_scale, &y_scale);
unsigned upem = hb_face_get_upem (hb_font_get_face (font));
unsigned count;
auto *positions = hb_buffer_get_glyph_positions (buffer, &count);
for (unsigned i = 0; i < count; i++)
{
auto &pos = positions[i];
pos.x_offset = pos.x_offset * x_scale / upem;
pos.y_offset = pos.y_offset * y_scale / upem;
pos.x_advance = pos.x_advance * x_scale / upem;
pos.y_advance = pos.y_advance * y_scale / upem;
}
}
else
{
if (!hb_shape_full (font, buffer, features, num_features, shapers))
{
if (error)
*error = "Shaping failed.";
goto fail;
}
} }
if (normalize_glyphs) if (normalize_glyphs)
@ -138,6 +170,7 @@ struct shape_options_t
hb_codepoint_t not_found_glyph = 0; hb_codepoint_t not_found_glyph = 0;
hb_buffer_cluster_level_t cluster_level = HB_BUFFER_CLUSTER_LEVEL_DEFAULT; hb_buffer_cluster_level_t cluster_level = HB_BUFFER_CLUSTER_LEVEL_DEFAULT;
hb_bool_t normalize_glyphs = false; hb_bool_t normalize_glyphs = false;
hb_bool_t glyphs = false;
hb_bool_t verify = false; hb_bool_t verify = false;
hb_bool_t unsafe_to_concat = false; hb_bool_t unsafe_to_concat = false;
hb_bool_t safe_to_insert_tatweel = false; hb_bool_t safe_to_insert_tatweel = false;
@ -266,6 +299,7 @@ shape_options_t::add_options (option_parser_t *parser)
{"normalize-glyphs",0, 0, G_OPTION_ARG_NONE, &this->normalize_glyphs, "Rearrange glyph clusters in nominal order", nullptr}, {"normalize-glyphs",0, 0, G_OPTION_ARG_NONE, &this->normalize_glyphs, "Rearrange glyph clusters in nominal order", nullptr},
{"unsafe-to-concat",0, 0, G_OPTION_ARG_NONE, &this->unsafe_to_concat, "Produce unsafe-to-concat glyph flag", nullptr}, {"unsafe-to-concat",0, 0, G_OPTION_ARG_NONE, &this->unsafe_to_concat, "Produce unsafe-to-concat glyph flag", nullptr},
{"safe-to-insert-tatweel",0, 0, G_OPTION_ARG_NONE, &this->safe_to_insert_tatweel, "Produce safe-to-insert-tatweel glyph flag", nullptr}, {"safe-to-insert-tatweel",0, 0, G_OPTION_ARG_NONE, &this->safe_to_insert_tatweel, "Produce safe-to-insert-tatweel glyph flag", nullptr},
{"glyphs", 0, 0, G_OPTION_ARG_NONE, &this->glyphs, "Interpret input as glyph string", nullptr},
{"verify", 0, 0, G_OPTION_ARG_NONE, &this->verify, "Perform sanity checks on shaping results", nullptr}, {"verify", 0, 0, G_OPTION_ARG_NONE, &this->verify, "Perform sanity checks on shaping results", nullptr},
{"num-iterations", 'n', G_OPTION_FLAG_IN_MAIN, {"num-iterations", 'n', G_OPTION_FLAG_IN_MAIN,
G_OPTION_ARG_INT, &this->num_iterations, "Run shaper N times (default: 1)", "N"}, G_OPTION_ARG_INT, &this->num_iterations, "Run shaper N times (default: 1)", "N"},