|
|
|
/*
|
|
|
|
* 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 <errno.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
|