2017-05-11 09:27:34 -05:00
|
|
|
// 11 may 2017
|
2017-05-17 22:37:16 -05:00
|
|
|
#include "uipriv_unix.h"
|
2017-05-11 09:27:34 -05:00
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
// TODO switch from GINT_TO_POINTER() and so to a fake GUINT_TO_POINTER()?
|
2017-05-17 21:56:55 -05:00
|
|
|
|
2017-05-11 09:27:34 -05:00
|
|
|
struct uiOpenTypeFeatures {
|
2017-05-17 21:56:55 -05:00
|
|
|
GHashTable *tags;
|
2017-05-11 09:27:34 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
uiOpenTypeFeatures *uiNewOpenTypeFeatures(void)
|
|
|
|
{
|
|
|
|
uiOpenTypeFeatures *otf;
|
|
|
|
|
|
|
|
otf = uiNew(uiOpenTypeFeatures);
|
2017-05-17 21:56:55 -05:00
|
|
|
otf->tags = g_hash_table_new(g_direct_hash, g_direct_equal);
|
2017-05-11 09:27:34 -05:00
|
|
|
return otf;
|
|
|
|
}
|
|
|
|
|
|
|
|
void uiFreeOpenTypeFeatures(uiOpenTypeFeatures *otf)
|
|
|
|
{
|
2017-05-17 21:56:55 -05:00
|
|
|
g_hash_table_destroy(otf->tags);
|
2017-05-11 09:27:34 -05:00
|
|
|
uiFree(otf);
|
|
|
|
}
|
|
|
|
|
2017-05-17 21:56:55 -05:00
|
|
|
static void cloneTags(gpointer key, gpointer value, gpointer data)
|
|
|
|
{
|
|
|
|
// TODO is there a G_HASH_TABLE()?
|
|
|
|
g_hash_table_replace((GHashTable *) data, key, value);
|
|
|
|
}
|
|
|
|
|
2017-05-30 08:52:38 -05:00
|
|
|
uiOpenTypeFeatures *uiOpenTypeFeaturesClone(const uiOpenTypeFeatures *otf)
|
2017-05-11 09:27:34 -05:00
|
|
|
{
|
|
|
|
uiOpenTypeFeatures *out;
|
|
|
|
|
2017-05-17 21:56:55 -05:00
|
|
|
// TODO switch the windows one to use this
|
|
|
|
out = uiNewOpenTypeFeatures();
|
|
|
|
g_hash_table_foreach(otf->tags, cloneTags, out->tags);
|
2017-05-11 09:27:34 -05:00
|
|
|
return out;
|
|
|
|
}
|
|
|
|
|
2017-05-17 21:56:55 -05:00
|
|
|
static gpointer mkTag(char a, char b, char c, char d)
|
|
|
|
{
|
|
|
|
uint32_t tag;
|
|
|
|
|
|
|
|
tag = (((uint32_t) a) & 0xFF) << 24;
|
|
|
|
tag |= (((uint32_t) b) & 0xFF) << 16;
|
|
|
|
tag |= (((uint32_t) c) & 0xFF) << 8;
|
|
|
|
tag |= ((uint32_t) d) & 0xFF;
|
|
|
|
return GINT_TO_POINTER(tag);
|
|
|
|
}
|
|
|
|
|
2017-05-11 09:27:34 -05:00
|
|
|
void uiOpenTypeFeaturesAdd(uiOpenTypeFeatures *otf, char a, char b, char c, char d, uint32_t value)
|
|
|
|
{
|
2017-05-17 21:56:55 -05:00
|
|
|
g_hash_table_replace(otf->tags, mkTag(a, b, c, d), GINT_TO_POINTER(value));
|
2017-05-11 09:27:34 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
void uiOpenTypeFeaturesRemove(uiOpenTypeFeatures *otf, char a, char b, char c, char d)
|
|
|
|
{
|
2017-06-18 19:10:42 -05:00
|
|
|
// will just return FALSE if the tag is not in otf->tags (half-documented as such), so we can use it safely
|
2017-05-17 21:56:55 -05:00
|
|
|
g_hash_table_remove(otf->tags, mkTag(a, b, c, d));
|
2017-05-11 09:27:34 -05:00
|
|
|
}
|
|
|
|
|
2017-05-17 21:56:55 -05:00
|
|
|
// TODO should this be before Add and Remove?
|
|
|
|
// TODO better name than Get?
|
2017-05-11 09:27:34 -05:00
|
|
|
int uiOpenTypeFeaturesGet(uiOpenTypeFeatures *otf, char a, char b, char c, char d, uint32_t *value)
|
|
|
|
{
|
2017-05-17 21:56:55 -05:00
|
|
|
gboolean found;
|
|
|
|
gpointer gv;
|
|
|
|
|
|
|
|
found = g_hash_table_lookup_extended(otf->tags,
|
|
|
|
mkTag(a, b, c, d),
|
|
|
|
NULL, &gv);
|
|
|
|
if (!found)
|
|
|
|
return 0;
|
|
|
|
*value = GPOINTER_TO_INT(gv);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
struct otfForEach {
|
|
|
|
uiOpenTypeFeaturesForEachFunc f;
|
|
|
|
void *data;
|
2017-06-06 11:47:07 -05:00
|
|
|
uiForEach ret;
|
2017-05-17 21:56:55 -05:00
|
|
|
};
|
|
|
|
|
|
|
|
static void foreach(gpointer key, gpointer value, gpointer data)
|
|
|
|
{
|
|
|
|
struct otfForEach *ofe = (struct otfForEach *) data;
|
|
|
|
uint32_t tag;
|
|
|
|
uint8_t a, b, c, d;
|
|
|
|
|
2017-06-06 11:47:07 -05:00
|
|
|
// we can't stop early, so just ignore the rest if we have to
|
|
|
|
if (ofe->ret == uiForEachStop)
|
|
|
|
return;
|
2017-05-17 21:56:55 -05:00
|
|
|
tag = GPOINTER_TO_INT(key);
|
|
|
|
a = (uint8_t) ((tag >> 24) & 0xFF);
|
|
|
|
b = (uint8_t) ((tag >> 16) & 0xFF);
|
|
|
|
c = (uint8_t) ((tag >> 8) & 0xFF);
|
|
|
|
d = (uint8_t) (tag & 0xFF);
|
2017-06-06 11:47:07 -05:00
|
|
|
ofe->ret = (*(ofe->f))((char) a, (char) b, (char) c, (char) d, GPOINTER_TO_INT(value), ofe->data);
|
2017-05-11 09:27:34 -05:00
|
|
|
}
|
|
|
|
|
2017-05-30 13:00:58 -05:00
|
|
|
void uiOpenTypeFeaturesForEach(const uiOpenTypeFeatures *otf, uiOpenTypeFeaturesForEachFunc f, void *data)
|
2017-05-11 09:27:34 -05:00
|
|
|
{
|
2017-05-17 21:56:55 -05:00
|
|
|
struct otfForEach ofe;
|
|
|
|
|
|
|
|
memset(&ofe, 0, sizeof (struct otfForEach));
|
|
|
|
ofe.f = f;
|
|
|
|
ofe.data = data;
|
|
|
|
g_hash_table_foreach(otf->tags, foreach, &ofe);
|
|
|
|
}
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
static gint tagcmp(gconstpointer a, gconstpointer b)
|
2017-05-17 21:56:55 -05:00
|
|
|
{
|
2017-05-17 22:37:16 -05:00
|
|
|
return GPOINTER_TO_INT(a) - GPOINTER_TO_INT(b);
|
2017-05-17 21:56:55 -05:00
|
|
|
}
|
|
|
|
|
|
|
|
static GList *copySortedKeys(GHashTable *tags)
|
|
|
|
{
|
|
|
|
GList *k, *copy;
|
|
|
|
|
|
|
|
k = g_hash_table_get_keys(tags);
|
|
|
|
copy = g_list_copy(k);
|
|
|
|
copy = g_list_sort(copy, tagcmp);
|
|
|
|
// TODO do we free k? the docs contradict themselves
|
2017-05-30 08:52:38 -05:00
|
|
|
// TODO I already forgot, does g_list_sort() copy, or just change the head?
|
2017-05-17 21:56:55 -05:00
|
|
|
return copy;
|
2017-05-11 09:27:34 -05:00
|
|
|
}
|
|
|
|
|
2017-05-30 08:52:38 -05:00
|
|
|
int uiOpenTypeFeaturesEqual(const uiOpenTypeFeatures *a, const uiOpenTypeFeatures *b)
|
2017-05-11 09:27:34 -05:00
|
|
|
{
|
2017-05-17 21:56:55 -05:00
|
|
|
GList *ak, *bk;
|
|
|
|
GList *ai, *bi;
|
|
|
|
guint na, nb;
|
|
|
|
guint i;
|
|
|
|
int equal = 0;
|
|
|
|
|
2017-06-09 18:59:48 -05:00
|
|
|
if (a == NULL && b == NULL)
|
|
|
|
return 1;
|
|
|
|
if (a == NULL || b == NULL)
|
|
|
|
return 0;
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
ak = copySortedKeys(a->tags);
|
|
|
|
bk = copySortedKeys(b->tags);
|
2017-05-17 21:56:55 -05:00
|
|
|
|
|
|
|
na = g_list_length(ak);
|
|
|
|
nb = g_list_length(bk);
|
|
|
|
if (na != nb) {
|
|
|
|
equal = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
// TODO use GPOINTER_TO_INT() in these?
|
2017-05-17 21:56:55 -05:00
|
|
|
ai = ak;
|
|
|
|
bi = bk;
|
|
|
|
for (i = 0; i < na; i++) {
|
|
|
|
gpointer av, bv;
|
|
|
|
|
|
|
|
// compare keys
|
|
|
|
// this is why we needed to sort earlier
|
|
|
|
if (ai->data != bi->data) {
|
|
|
|
equal = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
// and compare values
|
2017-05-17 22:37:16 -05:00
|
|
|
av = g_hash_table_lookup(a->tags, ai->data);
|
|
|
|
bv = g_hash_table_lookup(b->tags, bi->data);
|
2017-05-17 21:56:55 -05:00
|
|
|
if (av != bv) {
|
|
|
|
equal = 0;
|
|
|
|
goto out;
|
|
|
|
}
|
|
|
|
ai = ai->next;
|
|
|
|
bi = bi->next;
|
|
|
|
}
|
|
|
|
|
|
|
|
// all good
|
|
|
|
equal = 1;
|
|
|
|
|
|
|
|
out:
|
|
|
|
g_list_free(bk);
|
|
|
|
g_list_free(ak);
|
|
|
|
return equal;
|
|
|
|
}
|
|
|
|
|
2017-05-17 22:37:16 -05:00
|
|
|
// see https://developer.mozilla.org/en/docs/Web/CSS/font-feature-settings
|
2017-05-17 21:56:55 -05:00
|
|
|
// TODO make this a g_hash_table_foreach() function (which requires duplicating code)?
|
2017-06-06 11:47:07 -05:00
|
|
|
static uiForEach toCSS(char a, char b, char c, char d, uint32_t value, void *data)
|
2017-05-17 21:56:55 -05:00
|
|
|
{
|
|
|
|
// TODO is there a G_STRING()?
|
|
|
|
GString *s = (GString *) data;
|
|
|
|
|
|
|
|
// the last trailing comma is removed after foreach is done
|
|
|
|
g_string_append_printf(s, "\"%c%c%c%c\" %" PRIu32 ", ",
|
|
|
|
a, b, c, d, value);
|
2017-06-06 11:47:07 -05:00
|
|
|
return uiForEachContinue;
|
2017-05-11 09:27:34 -05:00
|
|
|
}
|
|
|
|
|
2017-05-30 18:07:01 -05:00
|
|
|
gchar *otfToPangoCSSString(const uiOpenTypeFeatures *otf)
|
2017-05-17 21:56:55 -05:00
|
|
|
{
|
|
|
|
GString *s;
|
|
|
|
|
|
|
|
s = g_string_new("");
|
|
|
|
uiOpenTypeFeaturesForEach(otf, toCSS, s);
|
|
|
|
if (s->len != 0)
|
|
|
|
// and remove the last comma
|
|
|
|
g_string_truncate(s, s->len - 2);
|
2017-05-30 18:07:01 -05:00
|
|
|
return g_string_free(s, FALSE);
|
2017-05-17 21:56:55 -05:00
|
|
|
}
|