-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathgenmnemo.py
executable file
·103 lines (91 loc) · 3.01 KB
/
genmnemo.py
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
#!/usr/bin/python3
# SPDX-License-Identifier: AGPL-3.0-only
import os
import sys
import subprocess
import time
def clean_name(s: str) -> str:
cs = []
for c in list(s):
if c == '♭':
cs.append('b')
elif c == '♯':
cs.append('s')
elif c == '/':
cs.append('_')
else:
cs.append(c)
return ''.join(cs)
chords: dict[str, tuple[str, str]] = {}
print("Reading space-separated values from stdin...")
while line := sys.stdin.readline():
if line.endswith("\n"):
line = line[:-1]
if line == '':
continue
name, frets, fingers = line.split(" ")
if name in chords.keys():
print(f"Duplicate chord: {name}")
exit(1)
chords[name] = frets, fingers
if len(chords.keys()) == 0:
print("Chords are a space-separated value, read from stdin")
print(r' e.g., printf "Am X,0,2,2,1,0 X,0,2,3,1,0\n" | ./genmnemo.py')
exit(1)
# A mnemosyne v2 file is a ZIP container consisting of:
# - cards.xml, which is both complicated and important
# - METADATA, a colon-delimited set of magic keys
# - any linked media files
#
# To misquote @dril:
#
# who the fuck is scraeming "STOP BLACKBOXING PROGRAMS YOU HAVE SOURCE FOR"
# at my house. show yourself, coward. i will never stop blackboxing
files = ["cards.xml", "METADATA"]
with open("METADATA", "w") as f:
f.write(f"""\
card_set_name:Guitar chords
author_name:https://github.com/frozencemetery/chordboard
author_email:
tags:guitar chords
date:{time.strftime('%a %b %d %Y')}
revision:1
notes:
""")
imgs = []
cards = []
links = []
for cname, (frets, fingers) in chords.items():
dest = f"{clean_name(cname)}.png"
files.append(dest)
subprocess.check_call(["./main.py", frets, fingers, "-o", dest, "-x3"])
imgs.append(f'<log type="13"><fname>{dest}</fname></log>')
cards.append(f'<log type="16" o_id="{cname}">'
f'<f>{cname}</f>'
f'<b><img src="{dest}"></b>'
f'</log>')
links.append(f'<log type="6" o_id="{cname}1" card_t="2" fact="{cname}" '
f'fact_v="2.1" tags="gchords" gr="-1" e="2.5" ac_rp="0" '
f'rt_rp="0" lps="0" ac_rp_l="0" rt_rp_l="0" l_rp="-1" '
f'n_rp="-1"></log>')
links.append(f'<log type="6" o_id="{cname}2" card_t="2" fact="{cname}" '
f'fact_v="2.2" tags="gchords" gr="-1" e="2.5" ac_rp="0" '
f'rt_rp="0" lps="0" ac_rp_l="0" rt_rp_l="0" l_rp="-1" '
f'n_rp="-1"></log>')
data = imgs + cards + links
flat_data = "\n".join(data)
with open("cards.xml", "w") as f:
f.write(f'<openSM2sync number_of_entries="{len(data) + 1}">'
f'<log type="10" o_id="gchords"><name>guitar chords</name></log>'
f'\n{flat_data}\n'
f'</openSM2sync>\n')
try:
os.unlink("chords.cards")
except FileNotFoundError:
pass
subprocess.check_call(["zip", "chords.cards"] + files)
for fname in files:
try:
os.unlink(fname)
except FileNotFoundError:
pass