2017-08-11 15:47:02 +00:00
|
|
|
/* convert-emoji.c: A tool to create an Emoji data GVariant
|
|
|
|
* Copyright 2017, Red Hat, Inc.
|
|
|
|
*
|
|
|
|
* This library is free software; you can redistribute it and/or
|
|
|
|
* modify it under the terms of the GNU Lesser General Public
|
|
|
|
* License as published by the Free Software Foundation; either
|
|
|
|
* version 2 of the License, or (at your option) any later version.
|
|
|
|
*
|
|
|
|
* This library is distributed in the hope that it will be useful,
|
|
|
|
* but WITHOUT ANY WARRANTY; without even the implied warranty of
|
|
|
|
* MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the GNU
|
|
|
|
* Lesser General Public License for more details.
|
|
|
|
*
|
|
|
|
* You should have received a copy of the GNU Lesser General Public
|
|
|
|
* License along with this library. If not, see <http://www.gnu.org/licenses/>.
|
|
|
|
*/
|
|
|
|
|
|
|
|
/* Build with gcc -o convert-emoji convert-emoji.c `pkg-config --cflags --libs json-glib-1.0`
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include <json-glib/json-glib.h>
|
|
|
|
#include <string.h>
|
|
|
|
|
|
|
|
gboolean
|
2017-08-13 19:16:11 +00:00
|
|
|
parse_code (GVariantBuilder *b,
|
2017-08-19 18:05:16 +00:00
|
|
|
const char *code,
|
|
|
|
GString *name_key)
|
2017-08-11 15:47:02 +00:00
|
|
|
{
|
2017-08-13 19:16:11 +00:00
|
|
|
g_auto(GStrv) strv = NULL;
|
|
|
|
int j;
|
|
|
|
|
|
|
|
strv = g_strsplit (code, " ", -1);
|
|
|
|
for (j = 0; strv[j]; j++)
|
|
|
|
{
|
|
|
|
guint32 u;
|
|
|
|
char *end;
|
|
|
|
|
|
|
|
u = (guint32) g_ascii_strtoull (strv[j], &end, 16);
|
|
|
|
if (*end != '\0')
|
|
|
|
{
|
|
|
|
g_error ("failed to parse code: %s\n", strv[j]);
|
|
|
|
return FALSE;
|
2017-08-11 15:47:02 +00:00
|
|
|
}
|
2017-08-13 19:16:11 +00:00
|
|
|
if (0x1f3fb <= u && u <= 0x1f3ff)
|
|
|
|
g_variant_builder_add (b, "u", 0);
|
|
|
|
else
|
2017-08-19 18:05:16 +00:00
|
|
|
{
|
|
|
|
g_variant_builder_add (b, "u", u);
|
|
|
|
if (j > 0)
|
|
|
|
g_string_append_c (name_key, '-');
|
|
|
|
g_string_append_printf (name_key, "%x", u);
|
|
|
|
}
|
2017-08-13 19:16:11 +00:00
|
|
|
}
|
2017-08-11 15:47:02 +00:00
|
|
|
|
2017-08-13 19:16:11 +00:00
|
|
|
return TRUE;
|
2017-08-11 15:47:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static const char *blacklist[] = {
|
|
|
|
"child",
|
|
|
|
"adult",
|
|
|
|
"older adult",
|
|
|
|
"woman with headscarf",
|
|
|
|
"bearded person",
|
|
|
|
"breast-feeding",
|
|
|
|
"mage",
|
|
|
|
"fairy",
|
|
|
|
"vampire",
|
|
|
|
"merperson",
|
|
|
|
"merman",
|
|
|
|
"mermaid",
|
|
|
|
" elf", // avoid matching selfie
|
|
|
|
"genie",
|
|
|
|
"zombie",
|
|
|
|
"in steamy room",
|
|
|
|
"climbing",
|
|
|
|
"in lotus position",
|
|
|
|
"person in bed",
|
|
|
|
"man in suit levitating",
|
|
|
|
"horse racing",
|
|
|
|
"snowboarder",
|
|
|
|
"golfing",
|
|
|
|
"love-you gesture",
|
|
|
|
"palms up together",
|
|
|
|
NULL
|
|
|
|
};
|
|
|
|
|
|
|
|
int
|
|
|
|
main (int argc, char *argv[])
|
|
|
|
{
|
2017-08-13 19:16:11 +00:00
|
|
|
JsonParser *parser;
|
|
|
|
JsonNode *root;
|
|
|
|
JsonArray *array;
|
2017-08-19 18:05:16 +00:00
|
|
|
JsonObject *ro;
|
|
|
|
JsonNode *node;
|
|
|
|
const char *name;
|
|
|
|
JsonObjectIter iter;
|
2017-08-13 19:16:11 +00:00
|
|
|
GError *error = NULL;
|
|
|
|
guint length, i;
|
|
|
|
GVariantBuilder builder;
|
|
|
|
GVariant *v;
|
|
|
|
GString *s;
|
|
|
|
GBytes *bytes;
|
2017-08-19 18:05:16 +00:00
|
|
|
GHashTable *names;
|
|
|
|
GString *name_key;
|
|
|
|
|
|
|
|
if (argc != 4)
|
|
|
|
{
|
|
|
|
g_print ("Usage: emoji-convert INPUT INPUT1 OUTPUT\n");
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
parser = json_parser_new ();
|
2017-08-13 19:16:11 +00:00
|
|
|
|
2017-08-19 18:05:16 +00:00
|
|
|
if (!json_parser_load_from_file (parser, argv[2], &error))
|
2017-08-13 19:16:11 +00:00
|
|
|
{
|
2017-08-19 18:05:16 +00:00
|
|
|
g_error ("%s", error->message);
|
2017-08-13 19:16:11 +00:00
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
2017-08-19 18:05:16 +00:00
|
|
|
root = json_parser_get_root (parser);
|
|
|
|
ro = json_node_get_object (root);
|
|
|
|
json_object_iter_init (&iter, ro);
|
|
|
|
|
|
|
|
names = g_hash_table_new_full (g_str_hash, g_str_equal, g_free, g_free);
|
|
|
|
name_key = g_string_new ("");
|
|
|
|
|
|
|
|
while (json_object_iter_next (&iter, &name, &node))
|
|
|
|
{
|
|
|
|
JsonObject *obj = json_node_get_object (node);
|
|
|
|
const char *unicode;
|
|
|
|
const char *shortname;
|
|
|
|
|
|
|
|
unicode = json_object_get_string_member (obj, "unicode");
|
|
|
|
shortname = json_object_get_string_member (obj, "shortname");
|
|
|
|
|
|
|
|
g_hash_table_insert (names, g_strdup (unicode), g_strdup (shortname));
|
|
|
|
}
|
|
|
|
|
|
|
|
g_object_unref (parser);
|
|
|
|
|
2017-08-13 19:16:11 +00:00
|
|
|
parser = json_parser_new ();
|
|
|
|
|
|
|
|
if (!json_parser_load_from_file (parser, argv[1], &error))
|
|
|
|
{
|
|
|
|
g_error ("%s", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
|
|
|
|
|
|
|
root = json_parser_get_root (parser);
|
|
|
|
array = json_node_get_array (root);
|
|
|
|
length = json_array_get_length (array);
|
|
|
|
|
2017-08-19 18:05:16 +00:00
|
|
|
g_variant_builder_init (&builder, G_VARIANT_TYPE ("a(auss)"));
|
2017-08-13 19:16:11 +00:00
|
|
|
i = 0;
|
|
|
|
while (i < length)
|
|
|
|
{
|
|
|
|
JsonNode *node = json_array_get_element (array, i);
|
|
|
|
JsonObject *obj = json_node_get_object (node);
|
|
|
|
GVariantBuilder b1;
|
|
|
|
const char *name;
|
2017-08-19 18:05:16 +00:00
|
|
|
const char *shortname;
|
2017-08-13 19:16:11 +00:00
|
|
|
char *code;
|
|
|
|
int j;
|
|
|
|
gboolean skip;
|
|
|
|
gboolean has_variations;
|
|
|
|
|
|
|
|
i++;
|
|
|
|
|
|
|
|
g_variant_builder_init (&b1, G_VARIANT_TYPE ("au"));
|
|
|
|
|
|
|
|
name = json_object_get_string_member (obj, "name");
|
|
|
|
code = g_strdup (json_object_get_string_member (obj, "code"));
|
|
|
|
|
|
|
|
if (strcmp (name, "world map") == 0)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
skip = FALSE;
|
|
|
|
for (j = 0; blacklist[j]; j++)
|
|
|
|
{
|
|
|
|
if (strstr (name, blacklist[j]) != 0)
|
|
|
|
skip = TRUE;
|
2017-08-11 15:47:02 +00:00
|
|
|
}
|
2017-08-13 19:16:11 +00:00
|
|
|
if (skip)
|
|
|
|
continue;
|
|
|
|
|
|
|
|
has_variations = FALSE;
|
|
|
|
while (i < length)
|
|
|
|
{
|
|
|
|
JsonNode *node2 = json_array_get_element (array, i);
|
|
|
|
JsonObject *obj2 = json_node_get_object (node2);
|
|
|
|
const char *name2;
|
|
|
|
const char *code2;
|
|
|
|
|
|
|
|
name2 = json_object_get_string_member (obj2, "name");
|
|
|
|
code2 = json_object_get_string_member (obj2, "code");
|
|
|
|
|
|
|
|
if (!strstr (name2, "skin tone") || !g_str_has_prefix (name2, name))
|
|
|
|
break;
|
|
|
|
|
|
|
|
if (!has_variations)
|
|
|
|
{
|
|
|
|
has_variations = TRUE;
|
|
|
|
g_free (code);
|
|
|
|
code = g_strdup (code2);
|
|
|
|
}
|
|
|
|
i++;
|
2017-08-11 15:47:02 +00:00
|
|
|
}
|
|
|
|
|
2017-08-19 18:05:16 +00:00
|
|
|
g_string_set_size (name_key, 0);
|
|
|
|
if (!parse_code (&b1, code, name_key))
|
2017-08-13 19:16:11 +00:00
|
|
|
return 1;
|
2017-08-11 15:47:02 +00:00
|
|
|
|
2017-08-19 18:05:16 +00:00
|
|
|
shortname = g_hash_table_lookup (names, name_key->str);
|
|
|
|
|
|
|
|
g_variant_builder_add (&builder, "(auss)", &b1, name, shortname ? shortname : "");
|
2017-08-13 19:16:11 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
v = g_variant_builder_end (&builder);
|
|
|
|
bytes = g_variant_get_data_as_bytes (v);
|
2017-08-19 18:05:16 +00:00
|
|
|
if (!g_file_set_contents (argv[3], g_bytes_get_data (bytes, NULL), g_bytes_get_size (bytes), &error))
|
2017-08-13 19:16:11 +00:00
|
|
|
{
|
|
|
|
g_error ("%s", error->message);
|
|
|
|
return 1;
|
|
|
|
}
|
2017-08-11 15:47:02 +00:00
|
|
|
|
2017-08-13 19:16:11 +00:00
|
|
|
return 0;
|
2017-08-11 15:47:02 +00:00
|
|
|
}
|