// 11 may 2017 #include "uipriv_unix.h" // TODO switch from GINT_TO_POINTER() and so to a fake GUINT_TO_POINTER()? struct uiOpenTypeFeatures { GHashTable *tags; }; uiOpenTypeFeatures *uiNewOpenTypeFeatures(void) { uiOpenTypeFeatures *otf; otf = uiNew(uiOpenTypeFeatures); otf->tags = g_hash_table_new(g_direct_hash, g_direct_equal); return otf; } void uiFreeOpenTypeFeatures(uiOpenTypeFeatures *otf) { g_hash_table_destroy(otf->tags); uiFree(otf); } static void cloneTags(gpointer key, gpointer value, gpointer data) { // TODO is there a G_HASH_TABLE()? g_hash_table_replace((GHashTable *) data, key, value); } uiOpenTypeFeatures *uiOpenTypeFeaturesClone(const uiOpenTypeFeatures *otf) { uiOpenTypeFeatures *out; // TODO switch the windows one to use this out = uiNewOpenTypeFeatures(); g_hash_table_foreach(otf->tags, cloneTags, out->tags); return out; } 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); } void uiOpenTypeFeaturesAdd(uiOpenTypeFeatures *otf, char a, char b, char c, char d, uint32_t value) { g_hash_table_replace(otf->tags, mkTag(a, b, c, d), GINT_TO_POINTER(value)); } void uiOpenTypeFeaturesRemove(uiOpenTypeFeatures *otf, char a, char b, char c, char d) { // will just return FALSE if the tag is not in otf->tags (half-documented as such), so we can use it safely g_hash_table_remove(otf->tags, mkTag(a, b, c, d)); } // TODO should this be before Add and Remove? // TODO better name than Get? int uiOpenTypeFeaturesGet(uiOpenTypeFeatures *otf, char a, char b, char c, char d, uint32_t *value) { 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; uiForEach ret; }; static void foreach(gpointer key, gpointer value, gpointer data) { struct otfForEach *ofe = (struct otfForEach *) data; uint32_t tag; uint8_t a, b, c, d; // we can't stop early, so just ignore the rest if we have to if (ofe->ret == uiForEachStop) return; 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); ofe->ret = (*(ofe->f))((char) a, (char) b, (char) c, (char) d, GPOINTER_TO_INT(value), ofe->data); } void uiOpenTypeFeaturesForEach(const uiOpenTypeFeatures *otf, uiOpenTypeFeaturesForEachFunc f, void *data) { struct otfForEach ofe; memset(&ofe, 0, sizeof (struct otfForEach)); ofe.f = f; ofe.data = data; g_hash_table_foreach(otf->tags, foreach, &ofe); } static gint tagcmp(gconstpointer a, gconstpointer b) { return GPOINTER_TO_INT(a) - GPOINTER_TO_INT(b); } 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 // TODO I already forgot, does g_list_sort() copy, or just change the head? return copy; } int uiOpenTypeFeaturesEqual(const uiOpenTypeFeatures *a, const uiOpenTypeFeatures *b) { GList *ak, *bk; GList *ai, *bi; guint na, nb; guint i; int equal = 0; if (a == NULL && b == NULL) return 1; if (a == NULL || b == NULL) return 0; ak = copySortedKeys(a->tags); bk = copySortedKeys(b->tags); na = g_list_length(ak); nb = g_list_length(bk); if (na != nb) { equal = 0; goto out; } // TODO use GPOINTER_TO_INT() in these? 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 av = g_hash_table_lookup(a->tags, ai->data); bv = g_hash_table_lookup(b->tags, bi->data); 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; } // see https://developer.mozilla.org/en/docs/Web/CSS/font-feature-settings // TODO make this a g_hash_table_foreach() function (which requires duplicating code)? static uiForEach toCSS(char a, char b, char c, char d, uint32_t value, void *data) { // 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); return uiForEachContinue; } gchar *otfToPangoCSSString(const uiOpenTypeFeatures *otf) { 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); return g_string_free(s, FALSE); }