-
Notifications
You must be signed in to change notification settings - Fork 3
/
Copy pathfind.py
217 lines (188 loc) · 8.85 KB
/
find.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
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
###############################################################################
###############################################################################
## ##
## THATCHORD BY TOM CONTI-LESLIE find.py ##
## ##
## This script takes a list of notes and some technical parameters, and ##
## outputs a list of lists of frets which correspond to all ways of ##
## playing the chord. It's where the magic happens! ##
## ##
## ##
## License: CC BY-SA 4.0 ##
## ##
## Contact: tom (dot) contileslie (at) gmail (dot) com ##
## ##
###############################################################################
###############################################################################
# import error messages
from errors import err
# import ranking functions
import rank
def smart_increment(maxes, current, remaining):
# How many notes is our chord missing? I.e. how big is the remaining set?
# If there is a gap of k notes, then we need to change at least the
# first k counters.
k = len(remaining)
# if n is 0 or 1 then we don't need to skip anything; just increment
# normally.
k = max(k - 1, 0)
for i in range(k):
current[i] = 0
for i in range(k, len(maxes)):
if not current[i] == maxes[i] - 1:
current[i] += 1
break
current[i] = 0
# the counter i will now be set to the number of strings changed, minus 1
return i + 1
def insert(options, frets, index, r):
"""
N.B. modifies options in place.
Inserts frets into options if the rank of frets is lower than some of the
current options. Pushes the worse options out of the list.
r is the calculated rank of the option 'frets'.
"""
tup = (frets.copy(), r)
# start at worse end of list, move all the way to where r is in order
l = len(options)
i = l
while i > 0 and r < options[i - 1][1]:
i -= 1
if l < index:
# in this case, options has not yet reached full size. Just add the opt
options.insert(i, tup)
elif i < l:
# in this case, options is full. Add only if our option beats at least
# one thing.
options.insert(i, tup)
# push out the worst retained option.
options.pop()
def find(chord, nmute = 0, important = 0, index = 1, nfrets = 12,
# Below are ranking args (some are also used for finding)
tuning = [], order = [], ranks = [], stringstarts = [],
# fret specification
fretspec = 0,
# args to activate for testing
keep_full_list = False):
"""
This function is called in the main file, thatchord.py.
In this function, all possible fret positions are considered on each string
and are compiled into 'valids'. All positions which give any note
in the chord are maintained. This has the disadvantage of considering, e.g.
4 copies of C as a valid configuration for the chord C major.
The second part of this function tests every possible configuration and
calculates their ranks. It maintains a short list of the best options
and then outputs the 'index'th best option found (default 0, i.e. best opt)
To avoid recalculating ranks, options are stored as tuples.
chord is a list of notes.
tuning is a list of notes with as many entries as strings.
nfrets is the number of frets of the instrument.
important, if nonzero, is the number of notes from the requested chord that
suffice to "define" the chord.
"""
# start by cutting off the chord so that we don't have more distinct notes
# than strings to play!
if len(chord) > len(tuning):
chord = chord[:len(tuning)]
# define some basic variables. If 0 important notes, we assume the whole
# chord is needed.
n = len(tuning)
valids = []
if important == 0:
important = len(chord)
# if there are more important notes than strings, cutoff at len(tuning)
important = min(important, len(tuning))
# create the set of notes we absolutely need in the chord
chordset = set(chord[:important])
# start by finding all valid positions.
for i in range(n):
valids.append([])
if i < nmute:
valids[i].append(-1)
for j in range(max(stringstarts[i], fretspec), nfrets + 1):
if (tuning[i] + j) % 12 in chord:
valids[i].append(j)
# we now have a list of possible frets for each string. Iterate through
# each combination and filter out the ones that are not satisfactory.
maxes = [len(l) for l in valids]
# check we have valid options for each string
if 0 in maxes:
err(5)
# all is good to go: initiate at first possibility and make list of valid
# options.
current = [0] * n
options = []
# initiate our first attempt as the lowest value on all strings.
attempt = [valids[i][0] for i in range(n)]
# populate list of note multiplicities. mults[i] is equal to the number of
# distinct strings playing i.
mults = [0] * 12
for i in range(n):
mults[(tuning[i] + attempt[i]) % 12] += 1
# create set of remaining notes: notes that are not played in the current
# attempt but needed in the chord.
remaining = set()
for note in chordset:
if mults[note] == 0:
remaining.add(note)
# want to iterate until we see 000..0 again
first_value = [0] * n
first_time = True
# FOR TESTING PURPOSES - function attribute
find.count = 0
if keep_full_list:
find.full_list = []
while first_time or current != first_value:
first_time = False
# attempt = [valids[s][current[s]] for s in range(n)]
# we will assess whether mutes are valid, and then whether sufficiently
# many notes from the required chord have been hit.
# First, see all muted strings. We already know they are < than nmute.
muted = [i for i in range(n) if attempt[i] == -1]
bool_mute = True
if not len(muted) == 0:
if not len(muted) == max(muted) + 1:
bool_mute = False
# Second, check that the attempt covers the important notes.
# This is the case iff our remaining set is empty.
bool_impo = len(remaining) == 0
# if both conditions are satisfied, store the option.
if bool_mute and bool_impo:
r = rank.rank(attempt, chord, tuning, order, ranks, stringstarts)
insert(options, attempt, index, r)
# FOR TESTING PURPOSES
find.count += 1
if keep_full_list:
find.full_list.append(attempt.copy())
# intelligently increment the current attempt to the next possibility.
updated = smart_increment(maxes, current, remaining)
# we may have changed the values of several strings.
# smart_increment has told us how many, and we can update the
# remaining set by removing notes that are no longer played now that
# we have changed our fret positions.
for i in range(updated):
# only remove a note if it was not muted.
if attempt[i] != -1:
old_note = (attempt[i] + tuning[i]) % 12
mults[old_note] -= 1
if mults[old_note] == 0 and old_note in chordset:
# we have lost all copies of this old note. It is now remaining
remaining.add(old_note)
# update that entry of the attempt to the new value
attempt[i] = valids[i][current[i]]
# now calculate the new notes that are being played as a result of
# the incrementation
for i in range(updated):
# only add notes if the string is not muted
if attempt[i] != -1:
new_note = (attempt[i] + tuning[i]) % 12
mults[new_note] += 1
if mults[new_note] == 1 and new_note in chordset:
# then we have just re-introduced this note into our set
# of currently played notes. It is no longer remaining.
remaining.remove(new_note)
# return the worst option in the list of options, which is at the index
# requested (default is for options to have 1 entry).
if options == []:
err(16)
return options[-1][0]