2017-04-23 20:31:08 +02:00
|
|
|
#!/usr/bin/env python3
|
|
|
|
|
|
|
|
import sys
|
2020-01-24 04:18:14 +01:00
|
|
|
import re
|
2022-04-21 19:25:39 +02:00
|
|
|
from unidecode import unidecode
|
2017-04-23 20:31:08 +02:00
|
|
|
from jinja2 import Template
|
|
|
|
|
|
|
|
|
|
|
|
class Emoji(object):
|
2022-04-22 18:49:57 +02:00
|
|
|
def __init__(self, code, shortname, unicodename):
|
2022-01-01 06:16:37 +01:00
|
|
|
self.code = ''.join(['\\U'+c.rjust(8, '0') for c in code.strip().split(' ')])
|
2017-04-23 20:31:08 +02:00
|
|
|
self.shortname = shortname
|
2022-04-22 18:49:57 +02:00
|
|
|
self.unicodename = unicodename
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2020-05-13 06:35:26 +02:00
|
|
|
def generate_qml_list(**kwargs):
|
|
|
|
tmpl = Template('''
|
|
|
|
const QVector<Emoji> emoji::Provider::emoji = {
|
|
|
|
{%- for c in kwargs.items() %}
|
|
|
|
// {{ c[0].capitalize() }}
|
|
|
|
{%- for e in c[1] %}
|
2022-04-22 18:49:57 +02:00
|
|
|
Emoji{QStringLiteral(u"{{ e.code }}"), QStringLiteral(u"{{ e.shortname }}"), QStringLiteral(u"{{ e.unicodename }}"), emoji::Emoji::Category::{{ c[0].capitalize() }}},
|
2020-05-13 06:35:26 +02:00
|
|
|
{%- endfor %}
|
|
|
|
{%- endfor %}
|
|
|
|
};
|
|
|
|
''')
|
|
|
|
d = dict(kwargs=kwargs)
|
|
|
|
print(tmpl.render(d))
|
2022-04-24 18:14:23 +02:00
|
|
|
# FIXME: Stop this madness
|
|
|
|
def humanize_keypad(num):
|
|
|
|
match num:
|
|
|
|
case "0":
|
|
|
|
return "zero"
|
|
|
|
case "1":
|
|
|
|
return "one"
|
|
|
|
case "2":
|
|
|
|
return "two"
|
|
|
|
case "3":
|
|
|
|
return "three"
|
|
|
|
case "4":
|
|
|
|
return "four"
|
|
|
|
case "5":
|
|
|
|
return "five"
|
|
|
|
case "6":
|
|
|
|
return "six"
|
|
|
|
case "7":
|
|
|
|
return "seven"
|
|
|
|
case "8":
|
|
|
|
return "eight"
|
|
|
|
case "9":
|
|
|
|
return "nine"
|
|
|
|
case "10":
|
|
|
|
return "ten"
|
|
|
|
case "*":
|
|
|
|
return "asterisk"
|
|
|
|
case "#":
|
|
|
|
return "hash"
|
|
|
|
case _:
|
|
|
|
return None
|
2017-04-23 20:31:08 +02:00
|
|
|
if __name__ == '__main__':
|
2022-04-21 19:25:39 +02:00
|
|
|
if len(sys.argv) < 3:
|
|
|
|
print('usage: emoji_codegen.py /path/to/emoji-test.txt /path/to/shortcodes.txt')
|
2017-04-23 20:31:08 +02:00
|
|
|
sys.exit(1)
|
|
|
|
|
|
|
|
filename = sys.argv[1]
|
2022-04-21 19:25:39 +02:00
|
|
|
shortcodefilename = sys.argv[2]
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2020-01-24 04:18:14 +01:00
|
|
|
people = []
|
|
|
|
nature = []
|
|
|
|
food = []
|
|
|
|
activity = []
|
|
|
|
travel = []
|
|
|
|
objects = []
|
|
|
|
symbols = []
|
|
|
|
flags = []
|
|
|
|
|
|
|
|
categories = {
|
|
|
|
'Smileys & Emotion': people,
|
|
|
|
'People & Body': people,
|
|
|
|
'Animals & Nature': nature,
|
|
|
|
'Food & Drink': food,
|
|
|
|
'Travel & Places': travel,
|
|
|
|
'Activities': activity,
|
|
|
|
'Objects': objects,
|
|
|
|
'Symbols': symbols,
|
2022-04-24 18:14:23 +02:00
|
|
|
'Flags': flags,
|
|
|
|
'Component': symbols
|
2020-01-24 04:18:14 +01:00
|
|
|
}
|
2022-04-21 19:25:39 +02:00
|
|
|
shortcodeDict = {}
|
|
|
|
# for my sanity - this strips newlines
|
|
|
|
for line in open(shortcodefilename, 'r', encoding="utf8"):
|
|
|
|
longname, shortname = line.strip().split(':')
|
|
|
|
shortcodeDict[longname] = shortname
|
2020-01-24 04:18:14 +01:00
|
|
|
current_category = ''
|
2021-09-25 08:19:44 +02:00
|
|
|
for line in open(filename, 'r', encoding="utf8"):
|
2020-01-24 04:18:14 +01:00
|
|
|
if line.startswith('# group:'):
|
|
|
|
current_category = line.split(':', 1)[1].strip()
|
|
|
|
|
|
|
|
if not line or line.startswith('#'):
|
|
|
|
continue
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2020-01-24 04:18:14 +01:00
|
|
|
segments = re.split(r'\s+[#;] ', line.strip())
|
|
|
|
if len(segments) != 3:
|
|
|
|
continue
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2020-01-24 04:18:14 +01:00
|
|
|
code, qualification, charAndName = segments
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2022-01-01 06:16:37 +01:00
|
|
|
# skip unqualified versions of same unicode
|
2022-04-24 18:14:23 +02:00
|
|
|
if qualification != 'fully-qualified' and qualification != 'component' :
|
2020-01-24 04:18:14 +01:00
|
|
|
continue
|
2022-04-24 18:14:23 +02:00
|
|
|
|
2017-04-23 20:31:08 +02:00
|
|
|
|
2022-04-23 19:22:42 +02:00
|
|
|
char, name = re.match(r'^(\S+) E\d+\.\d+ (.*)$', charAndName).groups()
|
2022-04-22 18:49:57 +02:00
|
|
|
shortname = name
|
2022-04-24 18:14:23 +02:00
|
|
|
|
|
|
|
# discard skin tone variants for sanity
|
|
|
|
# __contains__ is so stupid i hate prototype languages
|
|
|
|
if name.__contains__("skin tone") and qualification != 'component':
|
|
|
|
continue
|
|
|
|
if qualification == 'component' and not name.__contains__("skin tone"):
|
|
|
|
continue
|
2022-04-22 16:46:43 +02:00
|
|
|
#TODO: Handle skintone modifiers in a sane way
|
2022-04-22 18:49:57 +02:00
|
|
|
if shortname in shortcodeDict:
|
|
|
|
shortname = shortcodeDict[shortname]
|
2022-04-24 18:14:23 +02:00
|
|
|
else:
|
|
|
|
shortname = shortname.lower()
|
|
|
|
if shortname.endswith(' (blood type)'):
|
|
|
|
shortname = shortname[:-13]
|
|
|
|
if shortname.endswith(': red hair'):
|
|
|
|
shortname = "red_haired_" + shortname[:-10]
|
|
|
|
if shortname.endswith(': curly hair'):
|
|
|
|
shortname = "curly_haired_" + shortname[:-12]
|
|
|
|
if shortname.endswith(': white hair'):
|
|
|
|
shortname = "white_haried_" + shortname[:-12]
|
|
|
|
if shortname.endswith(': bald'):
|
|
|
|
shortname = "bald_" + shortname[:-6]
|
|
|
|
if shortname.endswith(': beard'):
|
|
|
|
shortname = "bearded_" + shortname[:-7]
|
2022-04-22 18:49:57 +02:00
|
|
|
if shortname.endswith(' face'):
|
|
|
|
shortname = shortname[:-5]
|
2022-04-24 18:14:23 +02:00
|
|
|
if shortname.endswith(' button'):
|
2022-04-22 18:49:57 +02:00
|
|
|
shortname = shortname[:-7]
|
2022-04-24 18:14:23 +02:00
|
|
|
if shortname.endswith(' banknote'):
|
|
|
|
shortname = shortname[:-9]
|
|
|
|
keycapmtch = re.match(r'^keycap: (.+)$', shortname)
|
|
|
|
if keycapmtch:
|
|
|
|
keycapthing, = keycapmtch.groups()
|
|
|
|
type(keycapthing)
|
|
|
|
num_name = humanize_keypad(keycapthing)
|
|
|
|
if num_name:
|
|
|
|
shortname = num_name
|
|
|
|
else:
|
|
|
|
raise Exception("incomplete keycap " + keycapthing + ", fix ur code")
|
|
|
|
|
|
|
|
# FIXME: Is there a better way to do this?
|
|
|
|
matchobj = re.match(r'^flag: (.*)$', shortname)
|
|
|
|
if matchobj:
|
|
|
|
country, = matchobj.groups()
|
|
|
|
shortname = country + " flag"
|
|
|
|
shortname = shortname.replace("u.s.", "us")
|
|
|
|
shortname = shortname.replace("&", "and")
|
2022-04-22 18:49:57 +02:00
|
|
|
shortname = shortname.replace("-", "_")
|
2022-04-24 18:14:23 +02:00
|
|
|
shortname, = re.match(r'^_*(.+)_*$', shortname).groups()
|
|
|
|
shortname = re.sub(r'\W', '_', shortname)
|
2022-04-22 18:49:57 +02:00
|
|
|
shortname = re.sub(r'_{2,}', '_', shortname)
|
|
|
|
shortname = unidecode(shortname)
|
|
|
|
categories[current_category].append(Emoji(code, shortname, name))
|
2017-04-23 20:31:08 +02:00
|
|
|
|
|
|
|
# Use xclip to pipe the output to clipboard.
|
|
|
|
# e.g ./codegen.py emoji.json | xclip -sel clip
|
2022-04-21 19:25:39 +02:00
|
|
|
# alternatively - delete the var from src/emoji/Provider.cpp, and do ./codegen.py emojis shortcodes >> src/emoji/Provider.cpp
|
2021-01-23 23:25:52 +01:00
|
|
|
generate_qml_list(people=people, nature=nature, food=food, activity=activity, travel=travel, objects=objects, symbols=symbols, flags=flags)
|