parent
43236ce345
commit
d30c1dacf5
6 changed files with 357 additions and 157 deletions
@ -0,0 +1,202 @@ |
|||||||
|
/*
|
||||||
|
* Copyright © 2023 Google, Inc. |
||||||
|
* |
||||||
|
* This is part of HarfBuzz, a text shaping library. |
||||||
|
* |
||||||
|
* Permission is hereby granted, without written agreement and without |
||||||
|
* license or royalty fees, to use, copy, modify, and distribute this |
||||||
|
* software and its documentation for any purpose, provided that the |
||||||
|
* above copyright notice and the following two paragraphs appear in |
||||||
|
* all copies of this software. |
||||||
|
* |
||||||
|
* IN NO EVENT SHALL THE COPYRIGHT HOLDER BE LIABLE TO ANY PARTY FOR |
||||||
|
* DIRECT, INDIRECT, SPECIAL, INCIDENTAL, OR CONSEQUENTIAL DAMAGES |
||||||
|
* ARISING OUT OF THE USE OF THIS SOFTWARE AND ITS DOCUMENTATION, EVEN |
||||||
|
* IF THE COPYRIGHT HOLDER HAS BEEN ADVISED OF THE POSSIBILITY OF SUCH |
||||||
|
* DAMAGE. |
||||||
|
* |
||||||
|
* THE COPYRIGHT HOLDER SPECIFICALLY DISCLAIMS ANY WARRANTIES, INCLUDING, |
||||||
|
* BUT NOT LIMITED TO, THE IMPLIED WARRANTIES OF MERCHANTABILITY AND |
||||||
|
* FITNESS FOR A PARTICULAR PURPOSE. THE SOFTWARE PROVIDED HEREUNDER IS |
||||||
|
* ON AN "AS IS" BASIS, AND THE COPYRIGHT HOLDER HAS NO OBLIGATION TO |
||||||
|
* PROVIDE MAINTENANCE, SUPPORT, UPDATES, ENHANCEMENTS, OR MODIFICATIONS. |
||||||
|
* |
||||||
|
* Google Author(s): Garret Rieger |
||||||
|
*/ |
||||||
|
#ifndef HELPER_SUBSET_HH |
||||||
|
#define HELPER_SUBSET_HH |
||||||
|
|
||||||
|
#include "glib.h" |
||||||
|
#include <math.h> |
||||||
|
#include <stdbool.h> |
||||||
|
#include "hb-subset.h" |
||||||
|
|
||||||
|
#ifndef HB_NO_VAR |
||||||
|
|
||||||
|
// Parses an axis position string and sets min, default, and max to
|
||||||
|
// the requested values. If a value should be set to it's default value
|
||||||
|
// then it will be set to NaN.
|
||||||
|
static gboolean |
||||||
|
parse_axis_position(const char* s, |
||||||
|
float* min, |
||||||
|
float* def, |
||||||
|
float* max, |
||||||
|
gboolean* drop, |
||||||
|
GError **error) |
||||||
|
{ |
||||||
|
const char* part = strpbrk(s, ":"); |
||||||
|
*drop = false; |
||||||
|
if (!part) { |
||||||
|
// Single value.
|
||||||
|
if (strcmp (s, "drop") == 0) |
||||||
|
{ |
||||||
|
*min = NAN; |
||||||
|
*def = NAN; |
||||||
|
*max = NAN; |
||||||
|
*drop = true; |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
errno = 0; |
||||||
|
char *p; |
||||||
|
float axis_value = strtof (s, &p); |
||||||
|
if (errno || s == p) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Failed parsing axis value at: '%s'", s); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
*min = axis_value; |
||||||
|
*def = axis_value; |
||||||
|
*max = axis_value; |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
float values[3]; |
||||||
|
int count = 0; |
||||||
|
for (int i = 0; i < 3; i++) { |
||||||
|
errno = 0; |
||||||
|
count++; |
||||||
|
if (!*s || part == s) { |
||||||
|
values[i] = NAN; |
||||||
|
|
||||||
|
if (part == NULL) break; |
||||||
|
s = part + 1; |
||||||
|
part = strpbrk(s, ":"); |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
char *pend; |
||||||
|
values[i] = strtof (s, &pend); |
||||||
|
if (errno || s == pend || (part && pend != part)) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Failed parsing axis value at: '%s'", s); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
if (part == NULL) break; |
||||||
|
s = pend + 1; |
||||||
|
part = strpbrk(s, ":"); |
||||||
|
} |
||||||
|
|
||||||
|
if (count == 2) { |
||||||
|
*min = values[0]; |
||||||
|
*def = NAN; |
||||||
|
*max = values[1]; |
||||||
|
return true; |
||||||
|
} else if (count == 3) { |
||||||
|
*min = values[0]; |
||||||
|
*def = values[1]; |
||||||
|
*max = values[2]; |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Failed parsing axis value at: '%s'", s); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
static gboolean |
||||||
|
parse_instancing_spec (const char *arg, |
||||||
|
hb_face_t* face, |
||||||
|
hb_subset_input_t* input, |
||||||
|
GError **error) |
||||||
|
{ |
||||||
|
char* s; |
||||||
|
while ((s = strtok((char *) arg, "="))) |
||||||
|
{ |
||||||
|
arg = NULL; |
||||||
|
unsigned len = strlen (s); |
||||||
|
if (len > 4) //Axis tags are 4 bytes.
|
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Failed parsing axis tag at: '%s'", s); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
hb_tag_t axis_tag = hb_tag_from_string (s, len); |
||||||
|
|
||||||
|
s = strtok(NULL, ", "); |
||||||
|
if (!s) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Value not specified for axis: %c%c%c%c", HB_UNTAG (axis_tag)); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
gboolean drop; |
||||||
|
float min, def, max; |
||||||
|
if (!parse_axis_position(s, &min, &def, &max, &drop, error)) |
||||||
|
return false; |
||||||
|
|
||||||
|
if (drop) |
||||||
|
{ |
||||||
|
if (!hb_subset_input_pin_axis_to_default (input, |
||||||
|
face, |
||||||
|
axis_tag)) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); |
||||||
|
return false; |
||||||
|
} |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
if (min == def && def == max) { |
||||||
|
if (!hb_subset_input_pin_axis_location (input, |
||||||
|
face, axis_tag, |
||||||
|
def)) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); |
||||||
|
return false; |
||||||
|
} |
||||||
|
continue; |
||||||
|
} |
||||||
|
|
||||||
|
#ifdef HB_EXPERIMENTAL_API |
||||||
|
if (!hb_subset_input_set_axis_range (input, |
||||||
|
face, axis_tag, |
||||||
|
min, max, def)) |
||||||
|
{ |
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Cannot pin axis: '%c%c%c%c', not present in fvar", HB_UNTAG (axis_tag)); |
||||||
|
return false; |
||||||
|
} |
||||||
|
continue; |
||||||
|
#endif |
||||||
|
|
||||||
|
g_set_error (error, G_OPTION_ERROR, G_OPTION_ERROR_BAD_VALUE, |
||||||
|
"Partial instancing is not supported."); |
||||||
|
return false; |
||||||
|
} |
||||||
|
|
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
#endif |
||||||
|
|
||||||
|
#endif |
@ -0,0 +1,98 @@ |
|||||||
|
#include <assert.h> |
||||||
|
#include <stdio.h> |
||||||
|
#include "glib.h" |
||||||
|
#include "hb-subset.h" |
||||||
|
#include "helper-subset.hh" |
||||||
|
|
||||||
|
hb_face_t* open_font(const char* path) |
||||||
|
{ |
||||||
|
hb_blob_t *blob = hb_blob_create_from_file_or_fail (path); |
||||||
|
g_assert(blob); |
||||||
|
hb_face_t* face = hb_face_create(blob, 0); |
||||||
|
hb_blob_destroy(blob); |
||||||
|
|
||||||
|
return face; |
||||||
|
} |
||||||
|
|
||||||
|
gboolean check_parsing(hb_face_t* face, const char* spec, hb_tag_t axis, float exp_min, float exp_def, float exp_max) |
||||||
|
{
|
||||||
|
printf(">> testing spec: %s\n", spec); |
||||||
|
hb_subset_input_t* input = hb_subset_input_create_or_fail(); |
||||||
|
g_assert(input); |
||||||
|
|
||||||
|
{ |
||||||
|
GError* error; |
||||||
|
char *spec_copy = g_strdup (spec); |
||||||
|
gboolean res = parse_instancing_spec(spec_copy, face, input, &error); |
||||||
|
g_free(spec_copy); |
||||||
|
if (!res) { |
||||||
|
hb_subset_input_destroy(input); |
||||||
|
return res; |
||||||
|
} |
||||||
|
} |
||||||
|
|
||||||
|
float act_min = 0.0, act_def = 0.0, act_max = 0.0; |
||||||
|
hb_bool_t res = hb_subset_input_get_axis_range(input, axis, &act_min, &act_max, &act_def); |
||||||
|
if (!res) { |
||||||
|
hb_subset_input_destroy(input); |
||||||
|
return false; |
||||||
|
|
||||||
|
} |
||||||
|
|
||||||
|
g_assert_cmpuint(exp_min, ==, act_min); |
||||||
|
g_assert_cmpuint(exp_def, ==, act_def); |
||||||
|
g_assert_cmpuint(exp_max, ==, act_max); |
||||||
|
|
||||||
|
hb_subset_input_destroy(input); |
||||||
|
return true; |
||||||
|
} |
||||||
|
|
||||||
|
static hb_tag_t wght = HB_TAG('w', 'g', 'h', 't'); |
||||||
|
static hb_tag_t xxxx = HB_TAG('x', 'x', 'x', 'x'); |
||||||
|
|
||||||
|
static void |
||||||
|
test_parse_instancing_spec (void) |
||||||
|
{ |
||||||
|
hb_face_t* face = open_font("../test/api/fonts/AdobeVFPrototype-Subset.otf"); |
||||||
|
hb_face_t* roboto = open_font("../test/api/fonts/Roboto-Variable.abc.ttf"); |
||||||
|
|
||||||
|
g_assert(check_parsing(face, "wght=300", wght, 300, 300, 300)); |
||||||
|
g_assert(check_parsing(face, "wght=100:200:300", wght, 100, 200, 300)); |
||||||
|
g_assert(check_parsing(face, "wght=:500:", wght, 0, 500, 1000)); |
||||||
|
g_assert(check_parsing(face, "wght=::700", wght, 0, 700, 700)); |
||||||
|
g_assert(check_parsing(face, "wght=200::", wght, 200, 1000, 1000)); |
||||||
|
g_assert(check_parsing(face, "wght=200:300:", wght, 200, 300, 1000)); |
||||||
|
g_assert(check_parsing(face, "wght=:300:500", wght, 0, 300, 500)); |
||||||
|
g_assert(check_parsing(face, "wght=300::700", wght, 300, 700, 700)); |
||||||
|
g_assert(check_parsing(face, "wght=300:700", wght, 300, 700, 700)); |
||||||
|
g_assert(check_parsing(face, "wght=:700", wght, 0, 700, 700)); |
||||||
|
g_assert(check_parsing(face, "wght=200:", wght, 200, 1000, 1000)); |
||||||
|
|
||||||
|
g_assert(check_parsing(face, "wght=200: xxxx=50", wght, 200, 1000, 1000)); |
||||||
|
g_assert(check_parsing(face, "wght=200: xxxx=50", xxxx, 50, 50, 50)); |
||||||
|
g_assert(check_parsing(face, "wght=200:,xxxx=50", wght, 200, 1000, 1000)); |
||||||
|
g_assert(check_parsing(face, "wght=200:,xxxx=50", xxxx, 50, 50, 50)); |
||||||
|
|
||||||
|
g_assert(check_parsing(roboto, "wght=300", wght, 300, 300, 300)); |
||||||
|
g_assert(check_parsing(roboto, "wght=100:200:300", wght, 100, 200, 300)); |
||||||
|
g_assert(check_parsing(roboto, "wght=:500:", wght, 100, 500, 900)); |
||||||
|
g_assert(check_parsing(roboto, "wght=::850", wght, 100, 400, 850)); |
||||||
|
g_assert(check_parsing(roboto, "wght=200::", wght, 200, 400, 900)); |
||||||
|
g_assert(check_parsing(roboto, "wght=200:300:", wght, 200, 300, 900)); |
||||||
|
g_assert(check_parsing(roboto, "wght=:300:500", wght, 100, 300, 500)); |
||||||
|
g_assert(check_parsing(roboto, "wght=300::700", wght, 300, 400, 700)); |
||||||
|
g_assert(check_parsing(roboto, "wght=300:700", wght, 300, 400, 700)); |
||||||
|
g_assert(check_parsing(roboto, "wght=:700", wght, 100, 400, 700)); |
||||||
|
g_assert(check_parsing(roboto, "wght=200:", wght, 200, 400, 900)); |
||||||
|
|
||||||
|
hb_face_destroy(face); |
||||||
|
} |
||||||
|
|
||||||
|
|
||||||
|
int |
||||||
|
main (int argc, char **argv) |
||||||
|
{ |
||||||
|
test_parse_instancing_spec(); |
||||||
|
|
||||||
|
return 0; |
||||||
|
} |
Loading…
Reference in new issue