Upload TMIDIX.py
Browse files
TMIDIX.py
CHANGED
@@ -14,14 +14,14 @@ r'''############################################################################
|
|
14 |
#
|
15 |
# Project Los Angeles
|
16 |
#
|
17 |
-
# Tegridy Code
|
18 |
#
|
19 |
# https://github.com/Tegridy-Code/Project-Los-Angeles
|
20 |
#
|
21 |
#
|
22 |
###################################################################################
|
23 |
###################################################################################
|
24 |
-
# Copyright
|
25 |
#
|
26 |
# Licensed under the Apache License, Version 2.0 (the "License");
|
27 |
# you may not use this file except in compliance with the License.
|
@@ -56,6 +56,8 @@ VersionDate = '20201120'
|
|
56 |
|
57 |
_previous_warning = '' # 5.4
|
58 |
_previous_times = 0 # 5.4
|
|
|
|
|
59 |
#------------------------------- Encoding stuff --------------------------
|
60 |
|
61 |
def opus2midi(opus=[], text_encoding='ISO-8859-1'):
|
@@ -845,10 +847,11 @@ def _unshift_ber_int(ba):
|
|
845 |
r'''Given a bytearray, returns a tuple of (the ber-integer at the
|
846 |
start, and the remainder of the bytearray).
|
847 |
'''
|
848 |
-
if not len(ba):
|
849 |
_warn('_unshift_ber_int: no integer found')
|
850 |
return ((0, b""))
|
851 |
-
byte = ba
|
|
|
852 |
integer = 0
|
853 |
while True:
|
854 |
integer += (byte & 0x7F)
|
@@ -857,13 +860,17 @@ start, and the remainder of the bytearray).
|
|
857 |
if not len(ba):
|
858 |
_warn('_unshift_ber_int: no end-of-integer found')
|
859 |
return ((0, ba))
|
860 |
-
byte = ba
|
|
|
861 |
integer <<= 7
|
862 |
|
|
|
863 |
def _clean_up_warnings(): # 5.4
|
864 |
# Call this before returning from any publicly callable function
|
865 |
# whenever there's a possibility that a warning might have been printed
|
866 |
# by the function, or by any private functions it might have called.
|
|
|
|
|
867 |
global _previous_times
|
868 |
global _previous_warning
|
869 |
if _previous_times > 1:
|
@@ -876,27 +883,32 @@ def _clean_up_warnings(): # 5.4
|
|
876 |
_previous_times = 0
|
877 |
_previous_warning = ''
|
878 |
|
|
|
879 |
def _warn(s=''):
|
|
|
|
|
880 |
global _previous_times
|
881 |
global _previous_warning
|
882 |
if s == _previous_warning: # 5.4
|
883 |
_previous_times = _previous_times + 1
|
884 |
else:
|
885 |
_clean_up_warnings()
|
886 |
-
sys.stderr.write(str(s)+"\n")
|
887 |
_previous_warning = s
|
888 |
|
|
|
889 |
def _some_text_event(which_kind=0x01, text=b'some_text', text_encoding='ISO-8859-1'):
|
890 |
-
if str(type(text)).find("'str'") >= 0:
|
891 |
data = bytes(text, encoding=text_encoding)
|
892 |
else:
|
893 |
data = bytes(text)
|
894 |
-
return b'\xFF'+bytes((which_kind,))+_ber_compressed_int(len(data))+data
|
|
|
895 |
|
896 |
def _consistentise_ticks(scores): # 3.6
|
897 |
# used by mix_scores, merge_scores, concatenate_scores
|
898 |
if len(scores) == 1:
|
899 |
-
|
900 |
are_consistent = True
|
901 |
ticks = scores[0][0]
|
902 |
iscore = 1
|
@@ -917,9 +929,8 @@ def _consistentise_ticks(scores): # 3.6
|
|
917 |
|
918 |
|
919 |
###########################################################################
|
920 |
-
|
921 |
def _decode(trackdata=b'', exclude=None, include=None,
|
922 |
-
|
923 |
r'''Decodes MIDI track data into an opus-style list of events.
|
924 |
The options:
|
925 |
'exclude' is a list of event types which will be ignored SHOULD BE A SET
|
@@ -939,24 +950,24 @@ The options:
|
|
939 |
exclude = set(exclude)
|
940 |
|
941 |
# Pointer = 0; not used here; we eat through the bytearray instead.
|
942 |
-
event_code = -1;
|
943 |
event_count = 0;
|
944 |
events = []
|
945 |
|
946 |
-
while(len(trackdata)):
|
947 |
# loop while there's anything to analyze ...
|
948 |
-
eot = False
|
949 |
event_count += 1
|
950 |
|
951 |
E = []
|
952 |
# E for events - we'll feed it to the event registrar at the end.
|
953 |
|
954 |
# Slice off the delta time code, and analyze it
|
955 |
-
[time,
|
956 |
|
957 |
# Now let's see what we can make of the command
|
958 |
-
first_byte = trackdata
|
959 |
-
|
960 |
if (first_byte < 0xF0): # It's a MIDI event
|
961 |
if (first_byte & 0x80):
|
962 |
event_code = first_byte
|
@@ -970,17 +981,19 @@ The options:
|
|
970 |
command = event_code & 0xF0
|
971 |
channel = event_code & 0x0F
|
972 |
|
973 |
-
if (command == 0xF6): #
|
974 |
pass
|
975 |
-
elif (command == 0xC0 or command == 0xD0): #
|
976 |
-
parameter = trackdata
|
977 |
-
|
978 |
-
|
|
|
|
|
979 |
|
980 |
#################################################################
|
981 |
# MIDI events
|
982 |
|
983 |
-
if (command
|
984 |
if 'note_off' in exclude:
|
985 |
continue
|
986 |
E = ['note_off', time, channel, parameter[0], parameter[1]]
|
@@ -991,11 +1004,11 @@ The options:
|
|
991 |
elif (command == 0xA0):
|
992 |
if 'key_after_touch' in exclude:
|
993 |
continue
|
994 |
-
E = ['key_after_touch',time,channel,parameter[0],parameter[1]]
|
995 |
elif (command == 0xB0):
|
996 |
if 'control_change' in exclude:
|
997 |
continue
|
998 |
-
E = ['control_change',time,channel,parameter[0],parameter[1]]
|
999 |
elif (command == 0xC0):
|
1000 |
if 'patch_change' in exclude:
|
1001 |
continue
|
@@ -1008,93 +1021,94 @@ The options:
|
|
1008 |
if 'pitch_wheel_change' in exclude:
|
1009 |
continue
|
1010 |
E = ['pitch_wheel_change', time, channel,
|
1011 |
-
|
1012 |
else:
|
1013 |
-
_warn("Shouldn't get here; command="+hex(command))
|
1014 |
|
1015 |
elif (first_byte == 0xFF): # It's a Meta-Event! ##################
|
1016 |
-
#[command, length, remainder] =
|
1017 |
# unpack("xCwa*", substr(trackdata, $Pointer, 6));
|
1018 |
-
#Pointer += 6 - len(remainder);
|
1019 |
# # Move past JUST the length-encoded.
|
1020 |
-
command = trackdata
|
|
|
1021 |
[length, trackdata] = _unshift_ber_int(trackdata)
|
1022 |
-
if (command
|
1023 |
-
|
1024 |
-
|
1025 |
-
|
1026 |
-
|
1027 |
-
|
1028 |
-
|
1029 |
-
elif command >= 0x01 and command <= 0x0f:
|
1030 |
# 6.2 take it in bytes; let the user get the right encoding.
|
1031 |
# text_str = trackdata[0:length].decode('ascii','ignore')
|
1032 |
# text_str = trackdata[0:length].decode('ISO-8859-1')
|
1033 |
# 6.4 take it in bytes; let the user get the right encoding.
|
1034 |
-
text_data = bytes(trackdata[0:length])
|
1035 |
# Defined text events
|
1036 |
if (command == 0x01):
|
1037 |
-
|
1038 |
elif (command == 0x02):
|
1039 |
-
|
1040 |
elif (command == 0x03):
|
1041 |
-
|
1042 |
elif (command == 0x04):
|
1043 |
-
|
1044 |
elif (command == 0x05):
|
1045 |
-
|
1046 |
elif (command == 0x06):
|
1047 |
-
|
1048 |
elif (command == 0x07):
|
1049 |
-
|
1050 |
# Reserved but apparently unassigned text events
|
1051 |
elif (command == 0x08):
|
1052 |
-
|
1053 |
elif (command == 0x09):
|
1054 |
-
|
1055 |
elif (command == 0x0a):
|
1056 |
-
|
1057 |
elif (command == 0x0b):
|
1058 |
-
|
1059 |
elif (command == 0x0c):
|
1060 |
-
|
1061 |
elif (command == 0x0d):
|
1062 |
-
|
1063 |
elif (command == 0x0e):
|
1064 |
-
|
1065 |
elif (command == 0x0f):
|
1066 |
-
|
1067 |
|
1068 |
# Now the sticky events -------------------------------------
|
1069 |
elif (command == 0x2F):
|
1070 |
-
|
1071 |
-
|
1072 |
-
|
1073 |
-
elif (command == 0x51):
|
1074 |
-
|
1075 |
-
|
1076 |
-
|
1077 |
-
|
1078 |
elif (command == 0x54):
|
1079 |
-
|
1080 |
-
|
1081 |
-
|
1082 |
elif (command == 0x58):
|
1083 |
-
|
1084 |
-
|
1085 |
-
|
1086 |
elif (command == 0x59):
|
1087 |
-
|
1088 |
-
|
1089 |
-
|
1090 |
-
elif (command == 0x7F):
|
1091 |
-
|
1092 |
else:
|
1093 |
-
|
1094 |
-
|
1095 |
-
|
1096 |
-
|
1097 |
-
|
1098 |
|
1099 |
# Pointer += length; # Now move Pointer
|
1100 |
trackdata = trackdata[length:]
|
@@ -1111,7 +1125,7 @@ The options:
|
|
1111 |
# is omitted if this is a non-final block in a multiblock sysex;
|
1112 |
# but the F7 (if there) is counted in the message's declared
|
1113 |
# length, so we don't have to think about it anyway.)
|
1114 |
-
#command = trackdata.pop(0)
|
1115 |
[length, trackdata] = _unshift_ber_int(trackdata)
|
1116 |
if first_byte == 0xF0:
|
1117 |
# 20091008 added ISO-8859-1 to get an 8-bit str
|
@@ -1135,32 +1149,32 @@ The options:
|
|
1135 |
# from the MIDI file spec. So, I'm going to assume that
|
1136 |
# they CAN, in practice, occur. I don't know whether it's
|
1137 |
# proper for you to actually emit these into a MIDI file.
|
1138 |
-
|
1139 |
-
elif (first_byte == 0xF2):
|
1140 |
# <song position msg> ::= F2 <data pair>
|
1141 |
E = ['song_position', time, _read_14_bit(trackdata[:2])]
|
1142 |
trackdata = trackdata[2:]
|
1143 |
|
1144 |
-
elif (first_byte == 0xF3):
|
1145 |
# E = ['song_select', time, struct.unpack('>B',trackdata.pop(0))[0]]
|
1146 |
E = ['song_select', time, trackdata[0]]
|
1147 |
trackdata = trackdata[1:]
|
1148 |
# DTime, Thing (what?! song number? whatever ...)
|
1149 |
|
1150 |
-
elif (first_byte == 0xF6):
|
1151 |
E = ['tune_request', time]
|
1152 |
# What would a tune request be doing in a MIDI /file/?
|
1153 |
|
1154 |
-
|
1155 |
-
|
1156 |
-
|
1157 |
-
|
1158 |
-
|
1159 |
-
|
1160 |
-
|
1161 |
-
|
1162 |
-
|
1163 |
-
|
1164 |
|
1165 |
r'''
|
1166 |
elif (first_byte > 0xF0) { # Some unknown kinda F-series event ####
|
@@ -1175,31 +1189,30 @@ The options:
|
|
1175 |
elif first_byte > 0xF0: # Some unknown F-series event
|
1176 |
# Here we only produce a one-byte piece of raw data.
|
1177 |
# E = ['raw_data', time, bytest(trackdata[0])] # 6.4
|
1178 |
-
E = ['raw_data', time, trackdata[0]]
|
1179 |
trackdata = trackdata[1:]
|
1180 |
else: # Fallthru.
|
1181 |
-
_warn("Aborting track. Command-byte first_byte="+hex(first_byte))
|
1182 |
break
|
1183 |
# End of the big if-group
|
1184 |
|
1185 |
-
|
1186 |
######################################################################
|
1187 |
# THE EVENT REGISTRAR...
|
1188 |
-
if E and
|
1189 |
# This is the code for exceptional handling of the EOT event.
|
1190 |
eot = True
|
1191 |
if not no_eot_magic:
|
1192 |
if E[1] > 0: # a null text-event to carry the delta-time
|
1193 |
E = ['text_event', E[1], '']
|
1194 |
else:
|
1195 |
-
E = []
|
1196 |
-
|
1197 |
if E and not (E[0] in exclude):
|
1198 |
-
#if ( $exclusive_event_callback ):
|
1199 |
# &{ $exclusive_event_callback }( @E );
|
1200 |
-
#else:
|
1201 |
# &{ $event_callback }( @E ) if $event_callback;
|
1202 |
-
|
1203 |
if eot:
|
1204 |
break
|
1205 |
|
@@ -1819,6 +1832,7 @@ def plot_ms_SONG(ms_song,
|
|
1819 |
plt.close()
|
1820 |
|
1821 |
if return_plt:
|
|
|
1822 |
return fig
|
1823 |
|
1824 |
plt.show()
|
@@ -4595,12 +4609,18 @@ def ascii_text_words_counter(ascii_text):
|
|
4595 |
|
4596 |
###################################################################################
|
4597 |
|
4598 |
-
def check_and_fix_tones_chord(tones_chord):
|
4599 |
|
4600 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord), 0, -1) for comb in combinations(tones_chord, i)]
|
4601 |
|
|
|
|
|
|
|
|
|
|
|
|
|
4602 |
for c in tones_chord_combs:
|
4603 |
-
if c in
|
4604 |
checked_tones_chord = c
|
4605 |
break
|
4606 |
|
@@ -4613,12 +4633,18 @@ def find_closest_tone(tones, tone):
|
|
4613 |
|
4614 |
###################################################################################
|
4615 |
|
4616 |
-
def advanced_check_and_fix_tones_chord(tones_chord, high_pitch=0):
|
4617 |
|
4618 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord), 0, -1) for comb in combinations(tones_chord, i)]
|
4619 |
|
|
|
|
|
|
|
|
|
|
|
|
|
4620 |
for c in tones_chord_combs:
|
4621 |
-
if c in
|
4622 |
tchord = c
|
4623 |
|
4624 |
if 0 < high_pitch < 128 and len(tchord) == 1:
|
@@ -4664,7 +4690,8 @@ def augment_enhanced_score_notes(enhanced_score_notes,
|
|
4664 |
pitch_shift=0,
|
4665 |
ceil_timings=False,
|
4666 |
round_timings=False,
|
4667 |
-
legacy_timings=
|
|
|
4668 |
):
|
4669 |
|
4670 |
esn = copy.deepcopy(enhanced_score_notes)
|
@@ -4714,6 +4741,9 @@ def augment_enhanced_score_notes(enhanced_score_notes,
|
|
4714 |
esn.sort(key=lambda x: x[6])
|
4715 |
esn.sort(key=lambda x: x[4], reverse=True)
|
4716 |
esn.sort(key=lambda x: x[1])
|
|
|
|
|
|
|
4717 |
|
4718 |
return esn
|
4719 |
|
@@ -5153,8 +5183,9 @@ def advanced_check_and_fix_chords_in_chordified_score(chordified_score,
|
|
5153 |
pitches_index=4,
|
5154 |
patches_index=6,
|
5155 |
use_filtered_chords=False,
|
5156 |
-
use_full_chords=
|
5157 |
remove_duplicate_pitches=True,
|
|
|
5158 |
fix_bad_pitches=False,
|
5159 |
skip_drums=False
|
5160 |
):
|
@@ -5208,22 +5239,64 @@ def advanced_check_and_fix_chords_in_chordified_score(chordified_score,
|
|
5208 |
tones_counts = Counter([p % 12 for p in pitches_chord]).most_common()
|
5209 |
|
5210 |
if tones_counts[0][1] > 1:
|
5211 |
-
|
|
|
5212 |
|
5213 |
elif tones_counts[1][1] > 1:
|
5214 |
-
|
|
|
5215 |
|
5216 |
else:
|
5217 |
-
|
|
|
5218 |
|
5219 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
5220 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord)-1, 0, -1) for comb in combinations(tones_chord, i)]
|
5221 |
|
5222 |
for co in tones_chord_combs:
|
5223 |
if co in CHORDS:
|
5224 |
-
tones_chord = co
|
5225 |
break
|
5226 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
5227 |
if len(tones_chord) == 1:
|
5228 |
tones_chord = [pitches_chord[0] % 12]
|
5229 |
|
@@ -7028,13 +7101,25 @@ def escore_notes_averages(escore_notes,
|
|
7028 |
else:
|
7029 |
durs = [e[durs_index] for e in escore_notes if e[chans_index] != 9]
|
7030 |
|
|
|
|
|
|
|
|
|
|
|
|
|
7031 |
if return_ptcs_and_vels:
|
7032 |
if average_drums:
|
7033 |
ptcs = [e[ptcs_index] for e in escore_notes]
|
7034 |
vels = [e[vels_index] for e in escore_notes]
|
7035 |
else:
|
7036 |
ptcs = [e[ptcs_index] for e in escore_notes if e[chans_index] != 9]
|
7037 |
-
vels = [e[vels_index] for e in escore_notes if e[chans_index] != 9]
|
|
|
|
|
|
|
|
|
|
|
|
|
7038 |
|
7039 |
return [sum(times) / len(times), sum(durs) / len(durs), sum(ptcs) / len(ptcs), sum(vels) / len(vels)]
|
7040 |
|
@@ -7814,19 +7899,21 @@ def solo_piano_escore_notes(escore_notes,
|
|
7814 |
chord = []
|
7815 |
|
7816 |
for cc in c:
|
7817 |
-
if cc[pitches_index] not in seen:
|
7818 |
|
7819 |
-
|
|
|
|
|
7820 |
cc[channels_index] = 0
|
7821 |
cc[patches_index] = 0
|
7822 |
-
|
7823 |
chord.append(cc)
|
7824 |
seen.append(cc[pitches_index])
|
7825 |
-
|
7826 |
-
|
7827 |
-
|
|
|
7828 |
chord.append(cc)
|
7829 |
-
seen.append(cc[pitches_index])
|
7830 |
|
7831 |
sp_escore_notes.append(chord)
|
7832 |
|
@@ -8858,7 +8945,1360 @@ def escore_notes_to_parsons_code(escore_notes,
|
|
8858 |
def all_consequtive(list_of_values):
|
8859 |
return all(b > a for a, b in zip(list_of_values[:-1], list_of_values[1:]))
|
8860 |
|
8861 |
-
###################################################################################
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
8862 |
#
|
8863 |
# This is the end of the TMIDI X Python module
|
8864 |
#
|
|
|
14 |
#
|
15 |
# Project Los Angeles
|
16 |
#
|
17 |
+
# Tegridy Code 2024
|
18 |
#
|
19 |
# https://github.com/Tegridy-Code/Project-Los-Angeles
|
20 |
#
|
21 |
#
|
22 |
###################################################################################
|
23 |
###################################################################################
|
24 |
+
# Copyright 2024 Project Los Angeles / Tegridy Code
|
25 |
#
|
26 |
# Licensed under the Apache License, Version 2.0 (the "License");
|
27 |
# you may not use this file except in compliance with the License.
|
|
|
56 |
|
57 |
_previous_warning = '' # 5.4
|
58 |
_previous_times = 0 # 5.4
|
59 |
+
_no_warning = False
|
60 |
+
|
61 |
#------------------------------- Encoding stuff --------------------------
|
62 |
|
63 |
def opus2midi(opus=[], text_encoding='ISO-8859-1'):
|
|
|
847 |
r'''Given a bytearray, returns a tuple of (the ber-integer at the
|
848 |
start, and the remainder of the bytearray).
|
849 |
'''
|
850 |
+
if not len(ba): # 6.7
|
851 |
_warn('_unshift_ber_int: no integer found')
|
852 |
return ((0, b""))
|
853 |
+
byte = ba[0]
|
854 |
+
ba = ba[1:]
|
855 |
integer = 0
|
856 |
while True:
|
857 |
integer += (byte & 0x7F)
|
|
|
860 |
if not len(ba):
|
861 |
_warn('_unshift_ber_int: no end-of-integer found')
|
862 |
return ((0, ba))
|
863 |
+
byte = ba[0]
|
864 |
+
ba = ba[1:]
|
865 |
integer <<= 7
|
866 |
|
867 |
+
|
868 |
def _clean_up_warnings(): # 5.4
|
869 |
# Call this before returning from any publicly callable function
|
870 |
# whenever there's a possibility that a warning might have been printed
|
871 |
# by the function, or by any private functions it might have called.
|
872 |
+
if _no_warning:
|
873 |
+
return
|
874 |
global _previous_times
|
875 |
global _previous_warning
|
876 |
if _previous_times > 1:
|
|
|
883 |
_previous_times = 0
|
884 |
_previous_warning = ''
|
885 |
|
886 |
+
|
887 |
def _warn(s=''):
|
888 |
+
if _no_warning:
|
889 |
+
return
|
890 |
global _previous_times
|
891 |
global _previous_warning
|
892 |
if s == _previous_warning: # 5.4
|
893 |
_previous_times = _previous_times + 1
|
894 |
else:
|
895 |
_clean_up_warnings()
|
896 |
+
sys.stderr.write(str(s) + "\n")
|
897 |
_previous_warning = s
|
898 |
|
899 |
+
|
900 |
def _some_text_event(which_kind=0x01, text=b'some_text', text_encoding='ISO-8859-1'):
|
901 |
+
if str(type(text)).find("'str'") >= 0: # 6.4 test for back-compatibility
|
902 |
data = bytes(text, encoding=text_encoding)
|
903 |
else:
|
904 |
data = bytes(text)
|
905 |
+
return b'\xFF' + bytes((which_kind,)) + _ber_compressed_int(len(data)) + data
|
906 |
+
|
907 |
|
908 |
def _consistentise_ticks(scores): # 3.6
|
909 |
# used by mix_scores, merge_scores, concatenate_scores
|
910 |
if len(scores) == 1:
|
911 |
+
return copy.deepcopy(scores)
|
912 |
are_consistent = True
|
913 |
ticks = scores[0][0]
|
914 |
iscore = 1
|
|
|
929 |
|
930 |
|
931 |
###########################################################################
|
|
|
932 |
def _decode(trackdata=b'', exclude=None, include=None,
|
933 |
+
event_callback=None, exclusive_event_callback=None, no_eot_magic=False):
|
934 |
r'''Decodes MIDI track data into an opus-style list of events.
|
935 |
The options:
|
936 |
'exclude' is a list of event types which will be ignored SHOULD BE A SET
|
|
|
950 |
exclude = set(exclude)
|
951 |
|
952 |
# Pointer = 0; not used here; we eat through the bytearray instead.
|
953 |
+
event_code = -1; # used for running status
|
954 |
event_count = 0;
|
955 |
events = []
|
956 |
|
957 |
+
while (len(trackdata)):
|
958 |
# loop while there's anything to analyze ...
|
959 |
+
eot = False # When True, the event registrar aborts this loop
|
960 |
event_count += 1
|
961 |
|
962 |
E = []
|
963 |
# E for events - we'll feed it to the event registrar at the end.
|
964 |
|
965 |
# Slice off the delta time code, and analyze it
|
966 |
+
[time, trackdata] = _unshift_ber_int(trackdata)
|
967 |
|
968 |
# Now let's see what we can make of the command
|
969 |
+
first_byte = trackdata[0] & 0xFF
|
970 |
+
trackdata = trackdata[1:]
|
971 |
if (first_byte < 0xF0): # It's a MIDI event
|
972 |
if (first_byte & 0x80):
|
973 |
event_code = first_byte
|
|
|
981 |
command = event_code & 0xF0
|
982 |
channel = event_code & 0x0F
|
983 |
|
984 |
+
if (command == 0xF6): # 0-byte argument
|
985 |
pass
|
986 |
+
elif (command == 0xC0 or command == 0xD0): # 1-byte argument
|
987 |
+
parameter = trackdata[0] # could be B
|
988 |
+
trackdata = trackdata[1:]
|
989 |
+
else: # 2-byte argument could be BB or 14-bit
|
990 |
+
parameter = (trackdata[0], trackdata[1])
|
991 |
+
trackdata = trackdata[2:]
|
992 |
|
993 |
#################################################################
|
994 |
# MIDI events
|
995 |
|
996 |
+
if (command == 0x80):
|
997 |
if 'note_off' in exclude:
|
998 |
continue
|
999 |
E = ['note_off', time, channel, parameter[0], parameter[1]]
|
|
|
1004 |
elif (command == 0xA0):
|
1005 |
if 'key_after_touch' in exclude:
|
1006 |
continue
|
1007 |
+
E = ['key_after_touch', time, channel, parameter[0], parameter[1]]
|
1008 |
elif (command == 0xB0):
|
1009 |
if 'control_change' in exclude:
|
1010 |
continue
|
1011 |
+
E = ['control_change', time, channel, parameter[0], parameter[1]]
|
1012 |
elif (command == 0xC0):
|
1013 |
if 'patch_change' in exclude:
|
1014 |
continue
|
|
|
1021 |
if 'pitch_wheel_change' in exclude:
|
1022 |
continue
|
1023 |
E = ['pitch_wheel_change', time, channel,
|
1024 |
+
_read_14_bit(parameter) - 0x2000]
|
1025 |
else:
|
1026 |
+
_warn("Shouldn't get here; command=" + hex(command))
|
1027 |
|
1028 |
elif (first_byte == 0xFF): # It's a Meta-Event! ##################
|
1029 |
+
# [command, length, remainder] =
|
1030 |
# unpack("xCwa*", substr(trackdata, $Pointer, 6));
|
1031 |
+
# Pointer += 6 - len(remainder);
|
1032 |
# # Move past JUST the length-encoded.
|
1033 |
+
command = trackdata[0] & 0xFF
|
1034 |
+
trackdata = trackdata[1:]
|
1035 |
[length, trackdata] = _unshift_ber_int(trackdata)
|
1036 |
+
if (command == 0x00):
|
1037 |
+
if (length == 2):
|
1038 |
+
E = ['set_sequence_number', time, _twobytes2int(trackdata)]
|
1039 |
+
else:
|
1040 |
+
_warn('set_sequence_number: length must be 2, not ' + str(length))
|
1041 |
+
E = ['set_sequence_number', time, 0]
|
1042 |
+
|
1043 |
+
elif command >= 0x01 and command <= 0x0f: # Text events
|
1044 |
# 6.2 take it in bytes; let the user get the right encoding.
|
1045 |
# text_str = trackdata[0:length].decode('ascii','ignore')
|
1046 |
# text_str = trackdata[0:length].decode('ISO-8859-1')
|
1047 |
# 6.4 take it in bytes; let the user get the right encoding.
|
1048 |
+
text_data = bytes(trackdata[0:length]) # 6.4
|
1049 |
# Defined text events
|
1050 |
if (command == 0x01):
|
1051 |
+
E = ['text_event', time, text_data]
|
1052 |
elif (command == 0x02):
|
1053 |
+
E = ['copyright_text_event', time, text_data]
|
1054 |
elif (command == 0x03):
|
1055 |
+
E = ['track_name', time, text_data]
|
1056 |
elif (command == 0x04):
|
1057 |
+
E = ['instrument_name', time, text_data]
|
1058 |
elif (command == 0x05):
|
1059 |
+
E = ['lyric', time, text_data]
|
1060 |
elif (command == 0x06):
|
1061 |
+
E = ['marker', time, text_data]
|
1062 |
elif (command == 0x07):
|
1063 |
+
E = ['cue_point', time, text_data]
|
1064 |
# Reserved but apparently unassigned text events
|
1065 |
elif (command == 0x08):
|
1066 |
+
E = ['text_event_08', time, text_data]
|
1067 |
elif (command == 0x09):
|
1068 |
+
E = ['text_event_09', time, text_data]
|
1069 |
elif (command == 0x0a):
|
1070 |
+
E = ['text_event_0a', time, text_data]
|
1071 |
elif (command == 0x0b):
|
1072 |
+
E = ['text_event_0b', time, text_data]
|
1073 |
elif (command == 0x0c):
|
1074 |
+
E = ['text_event_0c', time, text_data]
|
1075 |
elif (command == 0x0d):
|
1076 |
+
E = ['text_event_0d', time, text_data]
|
1077 |
elif (command == 0x0e):
|
1078 |
+
E = ['text_event_0e', time, text_data]
|
1079 |
elif (command == 0x0f):
|
1080 |
+
E = ['text_event_0f', time, text_data]
|
1081 |
|
1082 |
# Now the sticky events -------------------------------------
|
1083 |
elif (command == 0x2F):
|
1084 |
+
E = ['end_track', time]
|
1085 |
+
# The code for handling this, oddly, comes LATER,
|
1086 |
+
# in the event registrar.
|
1087 |
+
elif (command == 0x51): # DTime, Microseconds/Crochet
|
1088 |
+
if length != 3:
|
1089 |
+
_warn('set_tempo event, but length=' + str(length))
|
1090 |
+
E = ['set_tempo', time,
|
1091 |
+
struct.unpack(">I", b'\x00' + trackdata[0:3])[0]]
|
1092 |
elif (command == 0x54):
|
1093 |
+
if length != 5: # DTime, HR, MN, SE, FR, FF
|
1094 |
+
_warn('smpte_offset event, but length=' + str(length))
|
1095 |
+
E = ['smpte_offset', time] + list(struct.unpack(">BBBBB", trackdata[0:5]))
|
1096 |
elif (command == 0x58):
|
1097 |
+
if length != 4: # DTime, NN, DD, CC, BB
|
1098 |
+
_warn('time_signature event, but length=' + str(length))
|
1099 |
+
E = ['time_signature', time] + list(trackdata[0:4])
|
1100 |
elif (command == 0x59):
|
1101 |
+
if length != 2: # DTime, SF(signed), MI
|
1102 |
+
_warn('key_signature event, but length=' + str(length))
|
1103 |
+
E = ['key_signature', time] + list(struct.unpack(">bB", trackdata[0:2]))
|
1104 |
+
elif (command == 0x7F): # 6.4
|
1105 |
+
E = ['sequencer_specific', time, bytes(trackdata[0:length])]
|
1106 |
else:
|
1107 |
+
E = ['raw_meta_event', time, command,
|
1108 |
+
bytes(trackdata[0:length])] # 6.0
|
1109 |
+
# "[uninterpretable meta-event command of length length]"
|
1110 |
+
# DTime, Command, Binary Data
|
1111 |
+
# It's uninterpretable; record it as raw_data.
|
1112 |
|
1113 |
# Pointer += length; # Now move Pointer
|
1114 |
trackdata = trackdata[length:]
|
|
|
1125 |
# is omitted if this is a non-final block in a multiblock sysex;
|
1126 |
# but the F7 (if there) is counted in the message's declared
|
1127 |
# length, so we don't have to think about it anyway.)
|
1128 |
+
# command = trackdata.pop(0)
|
1129 |
[length, trackdata] = _unshift_ber_int(trackdata)
|
1130 |
if first_byte == 0xF0:
|
1131 |
# 20091008 added ISO-8859-1 to get an 8-bit str
|
|
|
1149 |
# from the MIDI file spec. So, I'm going to assume that
|
1150 |
# they CAN, in practice, occur. I don't know whether it's
|
1151 |
# proper for you to actually emit these into a MIDI file.
|
1152 |
+
|
1153 |
+
elif (first_byte == 0xF2): # DTime, Beats
|
1154 |
# <song position msg> ::= F2 <data pair>
|
1155 |
E = ['song_position', time, _read_14_bit(trackdata[:2])]
|
1156 |
trackdata = trackdata[2:]
|
1157 |
|
1158 |
+
elif (first_byte == 0xF3): # <song select msg> ::= F3 <data singlet>
|
1159 |
# E = ['song_select', time, struct.unpack('>B',trackdata.pop(0))[0]]
|
1160 |
E = ['song_select', time, trackdata[0]]
|
1161 |
trackdata = trackdata[1:]
|
1162 |
# DTime, Thing (what?! song number? whatever ...)
|
1163 |
|
1164 |
+
elif (first_byte == 0xF6): # DTime
|
1165 |
E = ['tune_request', time]
|
1166 |
# What would a tune request be doing in a MIDI /file/?
|
1167 |
|
1168 |
+
#########################################################
|
1169 |
+
# ADD MORE META-EVENTS HERE. TODO:
|
1170 |
+
# f1 -- MTC Quarter Frame Message. One data byte follows
|
1171 |
+
# the Status; it's the time code value, from 0 to 127.
|
1172 |
+
# f8 -- MIDI clock. no data.
|
1173 |
+
# fa -- MIDI start. no data.
|
1174 |
+
# fb -- MIDI continue. no data.
|
1175 |
+
# fc -- MIDI stop. no data.
|
1176 |
+
# fe -- Active sense. no data.
|
1177 |
+
# f4 f5 f9 fd -- unallocated
|
1178 |
|
1179 |
r'''
|
1180 |
elif (first_byte > 0xF0) { # Some unknown kinda F-series event ####
|
|
|
1189 |
elif first_byte > 0xF0: # Some unknown F-series event
|
1190 |
# Here we only produce a one-byte piece of raw data.
|
1191 |
# E = ['raw_data', time, bytest(trackdata[0])] # 6.4
|
1192 |
+
E = ['raw_data', time, trackdata[0]] # 6.4 6.7
|
1193 |
trackdata = trackdata[1:]
|
1194 |
else: # Fallthru.
|
1195 |
+
_warn("Aborting track. Command-byte first_byte=" + hex(first_byte))
|
1196 |
break
|
1197 |
# End of the big if-group
|
1198 |
|
|
|
1199 |
######################################################################
|
1200 |
# THE EVENT REGISTRAR...
|
1201 |
+
if E and (E[0] == 'end_track'):
|
1202 |
# This is the code for exceptional handling of the EOT event.
|
1203 |
eot = True
|
1204 |
if not no_eot_magic:
|
1205 |
if E[1] > 0: # a null text-event to carry the delta-time
|
1206 |
E = ['text_event', E[1], '']
|
1207 |
else:
|
1208 |
+
E = [] # EOT with a delta-time of 0; ignore it.
|
1209 |
+
|
1210 |
if E and not (E[0] in exclude):
|
1211 |
+
# if ( $exclusive_event_callback ):
|
1212 |
# &{ $exclusive_event_callback }( @E );
|
1213 |
+
# else:
|
1214 |
# &{ $event_callback }( @E ) if $event_callback;
|
1215 |
+
events.append(E)
|
1216 |
if eot:
|
1217 |
break
|
1218 |
|
|
|
1832 |
plt.close()
|
1833 |
|
1834 |
if return_plt:
|
1835 |
+
plt.close(fig)
|
1836 |
return fig
|
1837 |
|
1838 |
plt.show()
|
|
|
4609 |
|
4610 |
###################################################################################
|
4611 |
|
4612 |
+
def check_and_fix_tones_chord(tones_chord, use_full_chords=True):
|
4613 |
|
4614 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord), 0, -1) for comb in combinations(tones_chord, i)]
|
4615 |
|
4616 |
+
if use_full_chords:
|
4617 |
+
CHORDS = ALL_CHORDS_FULL
|
4618 |
+
|
4619 |
+
else:
|
4620 |
+
CHORDS = ALL_CHORDS_SORTED
|
4621 |
+
|
4622 |
for c in tones_chord_combs:
|
4623 |
+
if c in CHORDS:
|
4624 |
checked_tones_chord = c
|
4625 |
break
|
4626 |
|
|
|
4633 |
|
4634 |
###################################################################################
|
4635 |
|
4636 |
+
def advanced_check_and_fix_tones_chord(tones_chord, high_pitch=0, use_full_chords=True):
|
4637 |
|
4638 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord), 0, -1) for comb in combinations(tones_chord, i)]
|
4639 |
|
4640 |
+
if use_full_chords:
|
4641 |
+
CHORDS = ALL_CHORDS_FULL
|
4642 |
+
|
4643 |
+
else:
|
4644 |
+
CHORDS = ALL_CHORDS_SORTED
|
4645 |
+
|
4646 |
for c in tones_chord_combs:
|
4647 |
+
if c in CHORDS:
|
4648 |
tchord = c
|
4649 |
|
4650 |
if 0 < high_pitch < 128 and len(tchord) == 1:
|
|
|
4690 |
pitch_shift=0,
|
4691 |
ceil_timings=False,
|
4692 |
round_timings=False,
|
4693 |
+
legacy_timings=True,
|
4694 |
+
sort_drums_last=False
|
4695 |
):
|
4696 |
|
4697 |
esn = copy.deepcopy(enhanced_score_notes)
|
|
|
4741 |
esn.sort(key=lambda x: x[6])
|
4742 |
esn.sort(key=lambda x: x[4], reverse=True)
|
4743 |
esn.sort(key=lambda x: x[1])
|
4744 |
+
|
4745 |
+
if sort_drums_last:
|
4746 |
+
esn.sort(key=lambda x: (x[1], -x[4], x[6]) if x[6] != 128 else (x[1], x[6], -x[4]))
|
4747 |
|
4748 |
return esn
|
4749 |
|
|
|
5183 |
pitches_index=4,
|
5184 |
patches_index=6,
|
5185 |
use_filtered_chords=False,
|
5186 |
+
use_full_chords=False,
|
5187 |
remove_duplicate_pitches=True,
|
5188 |
+
fix_bad_tones_chords=False,
|
5189 |
fix_bad_pitches=False,
|
5190 |
skip_drums=False
|
5191 |
):
|
|
|
5239 |
tones_counts = Counter([p % 12 for p in pitches_chord]).most_common()
|
5240 |
|
5241 |
if tones_counts[0][1] > 1:
|
5242 |
+
good_tone = tones_counts[0][0]
|
5243 |
+
bad_tone = tones_counts[1][0]
|
5244 |
|
5245 |
elif tones_counts[1][1] > 1:
|
5246 |
+
good_tone = tones_counts[1][0]
|
5247 |
+
bad_tone = tones_counts[0][0]
|
5248 |
|
5249 |
else:
|
5250 |
+
good_tone = pitches_chord[0] % 12
|
5251 |
+
bad_tone = [t for t in tones_chord if t != good_tone][0]
|
5252 |
|
5253 |
+
tones_chord = [good_tone]
|
5254 |
+
|
5255 |
+
if fix_bad_tones_chords:
|
5256 |
+
|
5257 |
+
if good_tone > bad_tone:
|
5258 |
+
|
5259 |
+
if sorted([good_tone, (12+(bad_tone+1)) % 12]) in CHORDS:
|
5260 |
+
tones_chord = sorted([good_tone, (12+(bad_tone-1)) % 12])
|
5261 |
+
|
5262 |
+
elif sorted([good_tone, (12+(bad_tone-1)) % 12]) in CHORDS:
|
5263 |
+
tones_chord = sorted([good_tone, (12+(bad_tone+1)) % 12])
|
5264 |
+
|
5265 |
+
else:
|
5266 |
+
|
5267 |
+
if sorted([good_tone, (12+(bad_tone-1)) % 12]) in CHORDS:
|
5268 |
+
tones_chord = sorted([good_tone, (12+(bad_tone-1)) % 12])
|
5269 |
+
|
5270 |
+
elif sorted([good_tone, (12+(bad_tone+1)) % 12]) in CHORDS:
|
5271 |
+
tones_chord = sorted([good_tone, (12+(bad_tone+1)) % 12])
|
5272 |
+
|
5273 |
+
if len(tones_chord) > 2:
|
5274 |
tones_chord_combs = [list(comb) for i in range(len(tones_chord)-1, 0, -1) for comb in combinations(tones_chord, i)]
|
5275 |
|
5276 |
for co in tones_chord_combs:
|
5277 |
if co in CHORDS:
|
|
|
5278 |
break
|
5279 |
|
5280 |
+
if fix_bad_tones_chords:
|
5281 |
+
|
5282 |
+
dt_chord = list(set(co) ^ set(tones_chord))
|
5283 |
+
|
5284 |
+
for t in dt_chord:
|
5285 |
+
tones_chord.append((12+(t+1)) % 12)
|
5286 |
+
tones_chord.append((12+(t-1)) % 12)
|
5287 |
+
|
5288 |
+
ex_tones_chord = sorted(set(tones_chord))
|
5289 |
+
|
5290 |
+
tones_chord_combs = [list(comb) for i in range(4, 0, -2) for comb in combinations(ex_tones_chord, i) if all(t in list(comb) for t in co)]
|
5291 |
+
|
5292 |
+
for eco in tones_chord_combs:
|
5293 |
+
if eco in CHORDS:
|
5294 |
+
tones_chord = eco
|
5295 |
+
break
|
5296 |
+
|
5297 |
+
else:
|
5298 |
+
tones_chord = co
|
5299 |
+
|
5300 |
if len(tones_chord) == 1:
|
5301 |
tones_chord = [pitches_chord[0] % 12]
|
5302 |
|
|
|
7101 |
else:
|
7102 |
durs = [e[durs_index] for e in escore_notes if e[chans_index] != 9]
|
7103 |
|
7104 |
+
if len(times) == 0:
|
7105 |
+
times = [0]
|
7106 |
+
|
7107 |
+
if len(durs) == 0:
|
7108 |
+
durs = [0]
|
7109 |
+
|
7110 |
if return_ptcs_and_vels:
|
7111 |
if average_drums:
|
7112 |
ptcs = [e[ptcs_index] for e in escore_notes]
|
7113 |
vels = [e[vels_index] for e in escore_notes]
|
7114 |
else:
|
7115 |
ptcs = [e[ptcs_index] for e in escore_notes if e[chans_index] != 9]
|
7116 |
+
vels = [e[vels_index] for e in escore_notes if e[chans_index] != 9]
|
7117 |
+
|
7118 |
+
if len(ptcs) == 0:
|
7119 |
+
ptcs = [0]
|
7120 |
+
|
7121 |
+
if len(vels) == 0:
|
7122 |
+
vels = [0]
|
7123 |
|
7124 |
return [sum(times) / len(times), sum(durs) / len(durs), sum(ptcs) / len(ptcs), sum(vels) / len(vels)]
|
7125 |
|
|
|
7899 |
chord = []
|
7900 |
|
7901 |
for cc in c:
|
|
|
7902 |
|
7903 |
+
if cc[channels_index] != 9:
|
7904 |
+
if cc[pitches_index] not in seen:
|
7905 |
+
|
7906 |
cc[channels_index] = 0
|
7907 |
cc[patches_index] = 0
|
7908 |
+
|
7909 |
chord.append(cc)
|
7910 |
seen.append(cc[pitches_index])
|
7911 |
+
|
7912 |
+
else:
|
7913 |
+
if keep_drums:
|
7914 |
+
if cc[pitches_index]+128 not in seen:
|
7915 |
chord.append(cc)
|
7916 |
+
seen.append(cc[pitches_index]+128)
|
7917 |
|
7918 |
sp_escore_notes.append(chord)
|
7919 |
|
|
|
8945 |
def all_consequtive(list_of_values):
|
8946 |
return all(b > a for a, b in zip(list_of_values[:-1], list_of_values[1:]))
|
8947 |
|
8948 |
+
###################################################################################
|
8949 |
+
|
8950 |
+
def escore_notes_patches(escore_notes, patches_index=6):
|
8951 |
+
return sorted(set([e[patches_index] for e in escore_notes]))
|
8952 |
+
|
8953 |
+
###################################################################################
|
8954 |
+
|
8955 |
+
def build_suffix_array(lst):
|
8956 |
+
|
8957 |
+
n = len(lst)
|
8958 |
+
|
8959 |
+
suffixes = [(lst[i:], i) for i in range(n)]
|
8960 |
+
suffixes.sort()
|
8961 |
+
suffix_array = [suffix[1] for suffix in suffixes]
|
8962 |
+
|
8963 |
+
return suffix_array
|
8964 |
+
|
8965 |
+
###################################################################################
|
8966 |
+
|
8967 |
+
def build_lcp_array(lst, suffix_array):
|
8968 |
+
|
8969 |
+
n = len(lst)
|
8970 |
+
rank = [0] * n
|
8971 |
+
lcp = [0] * n
|
8972 |
+
|
8973 |
+
for i, suffix in enumerate(suffix_array):
|
8974 |
+
rank[suffix] = i
|
8975 |
+
|
8976 |
+
h = 0
|
8977 |
+
|
8978 |
+
for i in range(n):
|
8979 |
+
if rank[i] > 0:
|
8980 |
+
|
8981 |
+
j = suffix_array[rank[i] - 1]
|
8982 |
+
|
8983 |
+
while i + h < n and j + h < n and lst[i + h] == lst[j + h]:
|
8984 |
+
h += 1
|
8985 |
+
|
8986 |
+
lcp[rank[i]] = h
|
8987 |
+
|
8988 |
+
if h > 0:
|
8989 |
+
h -= 1
|
8990 |
+
|
8991 |
+
return lcp
|
8992 |
+
|
8993 |
+
###################################################################################
|
8994 |
+
|
8995 |
+
def find_lrno_pattern_fast(lst):
|
8996 |
+
n = len(lst)
|
8997 |
+
if n == 0:
|
8998 |
+
return []
|
8999 |
+
|
9000 |
+
suffix_array = build_suffix_array(lst)
|
9001 |
+
lcp_array = build_lcp_array(lst, suffix_array)
|
9002 |
+
|
9003 |
+
max_len = 0
|
9004 |
+
start_index = 0
|
9005 |
+
|
9006 |
+
for i in range(1, n):
|
9007 |
+
if lcp_array[i] > max_len:
|
9008 |
+
if suffix_array[i] + lcp_array[i] <= suffix_array[i - 1] or suffix_array[i - 1] + lcp_array[i - 1] <= suffix_array[i]:
|
9009 |
+
max_len = lcp_array[i]
|
9010 |
+
start_index = suffix_array[i]
|
9011 |
+
|
9012 |
+
return lst[start_index:start_index + max_len]
|
9013 |
+
|
9014 |
+
###################################################################################
|
9015 |
+
|
9016 |
+
def find_chunk_indexes(original_list, chunk, ignore_index=-1):
|
9017 |
+
|
9018 |
+
chunk_length = len(chunk)
|
9019 |
+
|
9020 |
+
for i in range(len(original_list) - chunk_length + 1):
|
9021 |
+
|
9022 |
+
chunk_index = 0
|
9023 |
+
start_index = ignore_index
|
9024 |
+
|
9025 |
+
for j in range(i, len(original_list)):
|
9026 |
+
if original_list[j] == chunk[chunk_index]:
|
9027 |
+
|
9028 |
+
if start_index == ignore_index:
|
9029 |
+
start_index = j
|
9030 |
+
|
9031 |
+
chunk_index += 1
|
9032 |
+
|
9033 |
+
if chunk_index == chunk_length:
|
9034 |
+
return [start_index, j]
|
9035 |
+
|
9036 |
+
elif original_list[j] != ignore_index:
|
9037 |
+
break
|
9038 |
+
|
9039 |
+
return None
|
9040 |
+
|
9041 |
+
###################################################################################
|
9042 |
+
|
9043 |
+
def escore_notes_lrno_pattern_fast(escore_notes,
|
9044 |
+
channels_index=3,
|
9045 |
+
pitches_index=4,
|
9046 |
+
zero_start_time=True
|
9047 |
+
):
|
9048 |
+
|
9049 |
+
cscore = chordify_score([1000, escore_notes])
|
9050 |
+
|
9051 |
+
score_chords = []
|
9052 |
+
|
9053 |
+
for c in cscore:
|
9054 |
+
|
9055 |
+
tchord = sorted(set([e[pitches_index] % 12 for e in c if e[channels_index] != 9]))
|
9056 |
+
|
9057 |
+
chord_tok = -1
|
9058 |
+
|
9059 |
+
if tchord:
|
9060 |
+
|
9061 |
+
if tchord not in ALL_CHORDS_FULL:
|
9062 |
+
tchord = check_and_fix_tones_chord(tchord)
|
9063 |
+
|
9064 |
+
chord_tok = ALL_CHORDS_FULL.index(tchord)
|
9065 |
+
|
9066 |
+
score_chords.append(chord_tok)
|
9067 |
+
|
9068 |
+
schords = [c for c in score_chords if c != -1]
|
9069 |
+
|
9070 |
+
lrno = find_lrno_pattern_fast(schords)
|
9071 |
+
|
9072 |
+
if lrno:
|
9073 |
+
|
9074 |
+
sidx, eidx = find_chunk_indexes(score_chords, lrno)
|
9075 |
+
|
9076 |
+
escore_notes_lrno_pattern = flatten(cscore[sidx:eidx+1])
|
9077 |
+
|
9078 |
+
if escore_notes_lrno_pattern is not None:
|
9079 |
+
|
9080 |
+
if zero_start_time:
|
9081 |
+
return recalculate_score_timings(escore_notes_lrno_pattern)
|
9082 |
+
|
9083 |
+
else:
|
9084 |
+
return escore_notes_lrno_pattern
|
9085 |
+
|
9086 |
+
else:
|
9087 |
+
return None
|
9088 |
+
|
9089 |
+
else:
|
9090 |
+
return None
|
9091 |
+
|
9092 |
+
###################################################################################
|
9093 |
+
|
9094 |
+
def escore_notes_durations_counter(escore_notes,
|
9095 |
+
min_duration=0,
|
9096 |
+
durations_index=2,
|
9097 |
+
channels_index=3
|
9098 |
+
):
|
9099 |
+
|
9100 |
+
escore = [e for e in escore_notes if e[channels_index] != 9]
|
9101 |
+
durs = [e[durations_index] for e in escore if e[durations_index] >= min_duration]
|
9102 |
+
zero_durs = sum([1 for e in escore if e[durations_index] == 0])
|
9103 |
+
|
9104 |
+
return [len(durs), len(escore), zero_durs, Counter(durs).most_common()]
|
9105 |
+
|
9106 |
+
###################################################################################
|
9107 |
+
|
9108 |
+
def count_bad_chords_in_chordified_score(chordified_score,
|
9109 |
+
pitches_index=4,
|
9110 |
+
patches_index=6,
|
9111 |
+
max_patch=127,
|
9112 |
+
use_full_chords=False
|
9113 |
+
):
|
9114 |
+
|
9115 |
+
if use_full_chords:
|
9116 |
+
CHORDS = ALL_CHORDS_FULL
|
9117 |
+
|
9118 |
+
else:
|
9119 |
+
CHORDS = ALL_CHORDS_SORTED
|
9120 |
+
|
9121 |
+
bad_chords_count = 0
|
9122 |
+
|
9123 |
+
for c in chordified_score:
|
9124 |
+
|
9125 |
+
cpitches = [e[pitches_index] for e in c if e[patches_index] <= max_patch]
|
9126 |
+
tones_chord = sorted(set([p % 12 for p in cpitches]))
|
9127 |
+
|
9128 |
+
if tones_chord:
|
9129 |
+
if tones_chord not in CHORDS:
|
9130 |
+
bad_chords_count += 1
|
9131 |
+
|
9132 |
+
return [bad_chords_count, len(chordified_score)]
|
9133 |
+
|
9134 |
+
###################################################################################
|
9135 |
+
|
9136 |
+
def needleman_wunsch_aligner(seq1,
|
9137 |
+
seq2,
|
9138 |
+
align_idx,
|
9139 |
+
gap_penalty=-1,
|
9140 |
+
match_score=2,
|
9141 |
+
mismatch_penalty=-1
|
9142 |
+
):
|
9143 |
+
|
9144 |
+
n = len(seq1)
|
9145 |
+
m = len(seq2)
|
9146 |
+
|
9147 |
+
score_matrix = [[0] * (m + 1) for _ in range(n + 1)]
|
9148 |
+
|
9149 |
+
for i in range(1, n + 1):
|
9150 |
+
score_matrix[i][0] = gap_penalty * i
|
9151 |
+
for j in range(1, m + 1):
|
9152 |
+
score_matrix[0][j] = gap_penalty * j
|
9153 |
+
|
9154 |
+
for i in range(1, n + 1):
|
9155 |
+
for j in range(1, m + 1):
|
9156 |
+
match = score_matrix[i-1][j-1] + (match_score if seq1[i-1][align_idx] == seq2[j-1][align_idx] else mismatch_penalty)
|
9157 |
+
delete = score_matrix[i-1][j] + gap_penalty
|
9158 |
+
insert = score_matrix[i][j-1] + gap_penalty
|
9159 |
+
score_matrix[i][j] = max(match, delete, insert)
|
9160 |
+
|
9161 |
+
align1, align2 = [], []
|
9162 |
+
i, j = n, m
|
9163 |
+
|
9164 |
+
while i > 0 and j > 0:
|
9165 |
+
|
9166 |
+
score = score_matrix[i][j]
|
9167 |
+
score_diag = score_matrix[i-1][j-1]
|
9168 |
+
score_up = score_matrix[i-1][j]
|
9169 |
+
score_left = score_matrix[i][j-1]
|
9170 |
+
|
9171 |
+
if score == score_diag + (match_score if seq1[i-1][align_idx] == seq2[j-1][align_idx] else mismatch_penalty):
|
9172 |
+
align1.append(seq1[i-1])
|
9173 |
+
align2.append(seq2[j-1])
|
9174 |
+
i -= 1
|
9175 |
+
j -= 1
|
9176 |
+
elif score == score_up + gap_penalty:
|
9177 |
+
align1.append(seq1[i-1])
|
9178 |
+
align2.append([None] * 6)
|
9179 |
+
i -= 1
|
9180 |
+
elif score == score_left + gap_penalty:
|
9181 |
+
align1.append([None] * 6)
|
9182 |
+
align2.append(seq2[j-1])
|
9183 |
+
j -= 1
|
9184 |
+
|
9185 |
+
while i > 0:
|
9186 |
+
align1.append(seq1[i-1])
|
9187 |
+
align2.append([None] * 6)
|
9188 |
+
i -= 1
|
9189 |
+
while j > 0:
|
9190 |
+
align1.append([None] * 6)
|
9191 |
+
align2.append(seq2[j-1])
|
9192 |
+
j -= 1
|
9193 |
+
|
9194 |
+
align1.reverse()
|
9195 |
+
align2.reverse()
|
9196 |
+
|
9197 |
+
return align1, align2
|
9198 |
+
|
9199 |
+
###################################################################################
|
9200 |
+
|
9201 |
+
def align_escore_notes_to_escore_notes(src_escore_notes,
|
9202 |
+
trg_escore_notes,
|
9203 |
+
recalculate_scores_timings=True,
|
9204 |
+
pitches_idx=4
|
9205 |
+
):
|
9206 |
+
|
9207 |
+
if recalculate_scores_timings:
|
9208 |
+
src_escore_notes = recalculate_score_timings(src_escore_notes)
|
9209 |
+
trg_escore_notes = recalculate_score_timings(trg_escore_notes)
|
9210 |
+
|
9211 |
+
src_align1, trg_align2 = needleman_wunsch_aligner(src_escore_notes, trg_escore_notes, pitches_idx)
|
9212 |
+
|
9213 |
+
aligned_scores = [[al[0], al[1]] for al in zip(src_align1, trg_align2) if al[0][0] is not None and al[1][0] is not None]
|
9214 |
+
|
9215 |
+
return aligned_scores
|
9216 |
+
|
9217 |
+
###################################################################################
|
9218 |
+
|
9219 |
+
def t_to_n(arr, si, t):
|
9220 |
+
|
9221 |
+
ct = 0
|
9222 |
+
ci = si
|
9223 |
+
|
9224 |
+
while ct + arr[ci][1] < t and ci < len(arr)-1:
|
9225 |
+
ct += arr[ci][1]
|
9226 |
+
ci += 1
|
9227 |
+
|
9228 |
+
return ci+1
|
9229 |
+
|
9230 |
+
###################################################################################
|
9231 |
+
|
9232 |
+
def max_sum_chunk_idxs(arr, t=255):
|
9233 |
+
|
9234 |
+
n = t_to_n(arr, 0, t)
|
9235 |
+
|
9236 |
+
if n > len(arr):
|
9237 |
+
return [0, n]
|
9238 |
+
|
9239 |
+
max_sum = 0
|
9240 |
+
max_sum_start_index = 0
|
9241 |
+
|
9242 |
+
max_sum_start_idxs = [0, len(arr), sum([a[0] for a in arr])]
|
9243 |
+
|
9244 |
+
for i in range(len(arr)):
|
9245 |
+
|
9246 |
+
n = t_to_n(arr, i, t)
|
9247 |
+
|
9248 |
+
current_sum = sum([a[0] for a in arr[i:n]])
|
9249 |
+
current_time = sum([a[1] for a in arr[i:n]])
|
9250 |
+
|
9251 |
+
if current_sum > max_sum and current_time <= t:
|
9252 |
+
max_sum = current_sum
|
9253 |
+
max_sum_start_idxs = [i, n, max_sum]
|
9254 |
+
|
9255 |
+
return max_sum_start_idxs
|
9256 |
+
|
9257 |
+
###################################################################################
|
9258 |
+
|
9259 |
+
def find_highest_density_escore_notes_chunk(escore_notes, max_chunk_time=512):
|
9260 |
+
|
9261 |
+
dscore = delta_score_notes(escore_notes)
|
9262 |
+
|
9263 |
+
cscore = chordify_score([d[1:] for d in dscore])
|
9264 |
+
|
9265 |
+
notes_counts = [[len(c), c[0][0]] for c in cscore]
|
9266 |
+
|
9267 |
+
msc_idxs = max_sum_chunk_idxs(notes_counts, max_chunk_time)
|
9268 |
+
|
9269 |
+
chunk_dscore = [['note'] + c for c in flatten(cscore[msc_idxs[0]:msc_idxs[1]])]
|
9270 |
+
|
9271 |
+
chunk_escore = recalculate_score_timings(delta_score_to_abs_score(chunk_dscore))
|
9272 |
+
|
9273 |
+
return chunk_escore
|
9274 |
+
|
9275 |
+
###################################################################################
|
9276 |
+
|
9277 |
+
def advanced_add_drums_to_escore_notes(escore_notes,
|
9278 |
+
main_beat_min_dtime=5,
|
9279 |
+
main_beat_dtime_thres=1,
|
9280 |
+
drums_durations_value=2,
|
9281 |
+
drums_pitches_velocities=[(36, 100),
|
9282 |
+
(38, 100),
|
9283 |
+
(41, 125)],
|
9284 |
+
recalculate_score_timings=True,
|
9285 |
+
intro_drums_count=4,
|
9286 |
+
intro_drums_time_k=4,
|
9287 |
+
intro_drums_pitch_velocity=[37, 110]
|
9288 |
+
):
|
9289 |
+
|
9290 |
+
#===========================================================
|
9291 |
+
|
9292 |
+
new_dscore = delta_score_notes(escore_notes)
|
9293 |
+
|
9294 |
+
times = [d[1] for d in new_dscore if d[1] != 0]
|
9295 |
+
|
9296 |
+
time = [c[0] for c in Counter(times).most_common() if c[0] >= main_beat_min_dtime][0]
|
9297 |
+
|
9298 |
+
#===========================================================
|
9299 |
+
|
9300 |
+
if intro_drums_count > 0:
|
9301 |
+
|
9302 |
+
drums_score = []
|
9303 |
+
|
9304 |
+
for i in range(intro_drums_count):
|
9305 |
+
|
9306 |
+
if i == 0:
|
9307 |
+
dtime = 0
|
9308 |
+
|
9309 |
+
else:
|
9310 |
+
dtime = time
|
9311 |
+
|
9312 |
+
drums_score.append(['note',
|
9313 |
+
dtime * intro_drums_time_k,
|
9314 |
+
drums_durations_value,
|
9315 |
+
9,
|
9316 |
+
intro_drums_pitch_velocity[0],
|
9317 |
+
intro_drums_pitch_velocity[1],
|
9318 |
+
128]
|
9319 |
+
)
|
9320 |
+
|
9321 |
+
new_dscore[0][1] = time * intro_drums_time_k
|
9322 |
+
|
9323 |
+
new_dscore = drums_score + new_dscore
|
9324 |
+
|
9325 |
+
#===========================================================
|
9326 |
+
|
9327 |
+
for e in new_dscore:
|
9328 |
+
|
9329 |
+
if abs(e[1] - time) == main_beat_dtime_thres:
|
9330 |
+
e[1] = time
|
9331 |
+
|
9332 |
+
if recalculate_score_timings:
|
9333 |
+
|
9334 |
+
if e[1] % time != 0 and e[1] > time:
|
9335 |
+
if e[1] % time < time // 2:
|
9336 |
+
e[1] -= e[1] % time
|
9337 |
+
|
9338 |
+
else:
|
9339 |
+
e[1] += time - (e[1] % time)
|
9340 |
+
|
9341 |
+
#===========================================================
|
9342 |
+
|
9343 |
+
drums_score = []
|
9344 |
+
|
9345 |
+
dtime = 0
|
9346 |
+
|
9347 |
+
idx = 0
|
9348 |
+
|
9349 |
+
for i, e in enumerate(new_dscore):
|
9350 |
+
|
9351 |
+
drums_score.append(e)
|
9352 |
+
|
9353 |
+
dtime += e[1]
|
9354 |
+
|
9355 |
+
if e[1] != 0:
|
9356 |
+
idx += 1
|
9357 |
+
|
9358 |
+
if i >= intro_drums_count:
|
9359 |
+
|
9360 |
+
if (e[1] % time == 0 and e[1] != 0) or i == 0:
|
9361 |
+
|
9362 |
+
if idx % 2 == 0 and e[1] != 0:
|
9363 |
+
drums_score.append(['note',
|
9364 |
+
0,
|
9365 |
+
drums_durations_value,
|
9366 |
+
9,
|
9367 |
+
drums_pitches_velocities[0][0],
|
9368 |
+
drums_pitches_velocities[0][1],
|
9369 |
+
128]
|
9370 |
+
)
|
9371 |
+
|
9372 |
+
if idx % 2 != 0 and e[1] != 0:
|
9373 |
+
drums_score.append(['note',
|
9374 |
+
0,
|
9375 |
+
drums_durations_value,
|
9376 |
+
9,
|
9377 |
+
drums_pitches_velocities[1][0],
|
9378 |
+
drums_pitches_velocities[1][1],
|
9379 |
+
128]
|
9380 |
+
)
|
9381 |
+
|
9382 |
+
if idx % 4 == 0 and e[1] != 0:
|
9383 |
+
drums_score.append(['note',
|
9384 |
+
0,
|
9385 |
+
drums_durations_value,
|
9386 |
+
9,
|
9387 |
+
drums_pitches_velocities[2][0],
|
9388 |
+
drums_pitches_velocities[2][1],
|
9389 |
+
128]
|
9390 |
+
)
|
9391 |
+
|
9392 |
+
#===========================================================
|
9393 |
+
|
9394 |
+
return delta_score_to_abs_score(drums_score)
|
9395 |
+
|
9396 |
+
###################################################################################
|
9397 |
+
|
9398 |
+
MIDI_TEXT_EVENTS = ['text_event',
|
9399 |
+
'copyright_text_event',
|
9400 |
+
'track_name',
|
9401 |
+
'instrument_name',
|
9402 |
+
'lyric',
|
9403 |
+
'marker',
|
9404 |
+
'cue_point',
|
9405 |
+
'text_event_08',
|
9406 |
+
'text_event_09',
|
9407 |
+
'text_event_0a',
|
9408 |
+
'text_event_0b',
|
9409 |
+
'text_event_0c',
|
9410 |
+
'text_event_0d',
|
9411 |
+
'text_event_0e',
|
9412 |
+
'text_event_0f'
|
9413 |
+
]
|
9414 |
+
|
9415 |
+
###################################################################################
|
9416 |
+
|
9417 |
+
import hashlib
|
9418 |
+
import re
|
9419 |
+
|
9420 |
+
###################################################################################
|
9421 |
+
|
9422 |
+
def get_md5_hash(data):
|
9423 |
+
return hashlib.md5(data).hexdigest()
|
9424 |
+
|
9425 |
+
###################################################################################
|
9426 |
+
|
9427 |
+
def is_valid_md5_hash(string):
|
9428 |
+
return bool(re.match(r'^[a-fA-F0-9]{32}$', string))
|
9429 |
+
|
9430 |
+
###################################################################################
|
9431 |
+
|
9432 |
+
def clean_string(original_string,
|
9433 |
+
regex=r'[^a-zA-Z0-9 ]',
|
9434 |
+
remove_duplicate_spaces=True,
|
9435 |
+
title=False
|
9436 |
+
):
|
9437 |
+
|
9438 |
+
cstr1 = re.sub(regex, '', original_string)
|
9439 |
+
|
9440 |
+
if title:
|
9441 |
+
cstr1 = cstr1.title()
|
9442 |
+
|
9443 |
+
if remove_duplicate_spaces:
|
9444 |
+
return re.sub(r'\s+', ' ', cstr1).strip()
|
9445 |
+
|
9446 |
+
else:
|
9447 |
+
return cstr1
|
9448 |
+
|
9449 |
+
###################################################################################
|
9450 |
+
|
9451 |
+
def encode_to_ord(text, chars_range=[], sub_char='', chars_shift=0):
|
9452 |
+
|
9453 |
+
if not chars_range:
|
9454 |
+
chars_range = [32] + list(range(65, 91)) + list(range(97, 123))
|
9455 |
+
|
9456 |
+
if sub_char:
|
9457 |
+
chars_range.append(ord(sub_char))
|
9458 |
+
|
9459 |
+
chars_range = sorted(set(chars_range))
|
9460 |
+
|
9461 |
+
encoded = []
|
9462 |
+
|
9463 |
+
for char in text:
|
9464 |
+
if ord(char) in chars_range:
|
9465 |
+
encoded.append(chars_range.index(ord(char)) + chars_shift)
|
9466 |
+
|
9467 |
+
else:
|
9468 |
+
if sub_char:
|
9469 |
+
encoded.append(chars_range.index(ord(sub_char)) + chars_shift)
|
9470 |
+
|
9471 |
+
|
9472 |
+
return [encoded, chars_range]
|
9473 |
+
|
9474 |
+
###################################################################################
|
9475 |
+
|
9476 |
+
def decode_from_ord(ord_list, chars_range=[], sub_char='', chars_shift=0):
|
9477 |
+
|
9478 |
+
if not chars_range:
|
9479 |
+
chars_range = [32] + list(range(65, 91)) + list(range(97, 123))
|
9480 |
+
|
9481 |
+
if sub_char:
|
9482 |
+
chars_range.append(ord(sub_char))
|
9483 |
+
|
9484 |
+
chars_range = sorted(set(chars_range))
|
9485 |
+
|
9486 |
+
return ''.join(chr(chars_range[num-chars_shift]) if 0 <= num-chars_shift < len(chars_range) else sub_char for num in ord_list)
|
9487 |
+
|
9488 |
+
###################################################################################
|
9489 |
+
|
9490 |
+
def lists_similarity(list1, list2, by_elements=True, by_sum=True):
|
9491 |
+
|
9492 |
+
if len(list1) != len(list2):
|
9493 |
+
return -1
|
9494 |
+
|
9495 |
+
element_ratios = []
|
9496 |
+
total_counts1 = sum(list1)
|
9497 |
+
total_counts2 = sum(list2)
|
9498 |
+
|
9499 |
+
for a, b in zip(list1, list2):
|
9500 |
+
if a == 0 and b == 0:
|
9501 |
+
element_ratios.append(1)
|
9502 |
+
elif a == 0 or b == 0:
|
9503 |
+
element_ratios.append(0)
|
9504 |
+
else:
|
9505 |
+
element_ratios.append(min(a, b) / max(a, b))
|
9506 |
+
|
9507 |
+
average_element_ratio = sum(element_ratios) / len(element_ratios)
|
9508 |
+
|
9509 |
+
total_counts_ratio = min(total_counts1, total_counts2) / max(total_counts1, total_counts2)
|
9510 |
+
|
9511 |
+
if by_elements and by_sum:
|
9512 |
+
return (average_element_ratio + total_counts_ratio) / 2
|
9513 |
+
|
9514 |
+
elif by_elements and not by_sum:
|
9515 |
+
return average_element_ratio
|
9516 |
+
|
9517 |
+
elif not by_elements and by_sum:
|
9518 |
+
return total_counts_ratio
|
9519 |
+
|
9520 |
+
else:
|
9521 |
+
return -1
|
9522 |
+
|
9523 |
+
###################################################################################
|
9524 |
+
|
9525 |
+
def find_indexes(lst, value, mode='equal', dual_mode=True):
|
9526 |
+
|
9527 |
+
indexes = []
|
9528 |
+
|
9529 |
+
if mode == 'equal' or dual_mode:
|
9530 |
+
indexes.extend([index for index, elem in enumerate(lst) if elem == value])
|
9531 |
+
|
9532 |
+
if mode == 'smaller':
|
9533 |
+
indexes.extend([index for index, elem in enumerate(lst) if elem < value])
|
9534 |
+
|
9535 |
+
if mode == 'larger':
|
9536 |
+
indexes.extend([index for index, elem in enumerate(lst) if elem > value])
|
9537 |
+
|
9538 |
+
return sorted(set(indexes))
|
9539 |
+
|
9540 |
+
###################################################################################
|
9541 |
+
|
9542 |
+
NUMERALS = ["one", "two", "three", "four",
|
9543 |
+
"five", "six", "seven", "eight",
|
9544 |
+
"nine", "ten", "eleven", "twelve",
|
9545 |
+
"thirteen", "fourteen", "fifteen", "sixteen"
|
9546 |
+
]
|
9547 |
+
|
9548 |
+
SEMITONES = ["C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"]
|
9549 |
+
|
9550 |
+
MOOD_SCALES = ['Major', 'Minor', 'Major Minor']
|
9551 |
+
|
9552 |
+
###################################################################################
|
9553 |
+
|
9554 |
+
def alpha_str(string):
|
9555 |
+
return re.sub(r'[^a-zA-Z ()]', '', string).strip()
|
9556 |
+
|
9557 |
+
###################################################################################
|
9558 |
+
|
9559 |
+
def escore_notes_to_text_description(escore_notes, song_name='', artist_name=''):
|
9560 |
+
|
9561 |
+
#==============================================================================
|
9562 |
+
|
9563 |
+
song_time_min = (escore_notes[-1][1] * 16) / 1000 / 60
|
9564 |
+
|
9565 |
+
if song_time_min < 1.5:
|
9566 |
+
song_length = 'short'
|
9567 |
+
|
9568 |
+
elif 1.5 <= song_time_min < 2.5:
|
9569 |
+
song_length = 'average'
|
9570 |
+
|
9571 |
+
elif song_time_min >= 2.5:
|
9572 |
+
song_length = 'long'
|
9573 |
+
|
9574 |
+
#==============================================================================
|
9575 |
+
|
9576 |
+
escore_times = [e[1] for e in escore_notes if e[3] != 9]
|
9577 |
+
|
9578 |
+
comp_type = ''
|
9579 |
+
|
9580 |
+
if len(escore_times) > 0:
|
9581 |
+
if len(escore_times) == len(set(escore_times)):
|
9582 |
+
comp_type = 'monophonic melody'
|
9583 |
+
|
9584 |
+
elif len(escore_times) >= len(set(escore_times)) and 1 in Counter(escore_times).values():
|
9585 |
+
comp_type = 'melody and accompaniment'
|
9586 |
+
|
9587 |
+
elif len(escore_times) >= len(set(escore_times)) and 1 not in Counter(escore_times).values():
|
9588 |
+
comp_type = 'accompaniment'
|
9589 |
+
|
9590 |
+
else:
|
9591 |
+
comp_type = 'drums only'
|
9592 |
+
|
9593 |
+
#==============================================================================
|
9594 |
+
|
9595 |
+
patches = sorted(set([e[6] for e in escore_notes]))
|
9596 |
+
|
9597 |
+
instruments = [alpha_str(Number2patch[p]) for p in patches if p < 128]
|
9598 |
+
|
9599 |
+
if 128 in patches:
|
9600 |
+
drums_present = True
|
9601 |
+
|
9602 |
+
else:
|
9603 |
+
drums_present = False
|
9604 |
+
|
9605 |
+
drums_pitches = [e[4] for e in escore_notes if e[3] == 9]
|
9606 |
+
|
9607 |
+
most_common_drums = [alpha_str(Notenum2percussion[p[0]]) for p in Counter(drums_pitches).most_common(3) if p[0] in Notenum2percussion]
|
9608 |
+
|
9609 |
+
#==============================================================================
|
9610 |
+
|
9611 |
+
pitches = [e[4] for e in escore_notes if e[3] != 9]
|
9612 |
+
|
9613 |
+
key = ''
|
9614 |
+
|
9615 |
+
if pitches:
|
9616 |
+
key = SEMITONES[statistics.mode(pitches) % 12]
|
9617 |
+
|
9618 |
+
#==============================================================================
|
9619 |
+
|
9620 |
+
mood = ''
|
9621 |
+
|
9622 |
+
if pitches:
|
9623 |
+
|
9624 |
+
cscore = chordify_score([1000, escore_notes])
|
9625 |
+
|
9626 |
+
tones_chords = Counter()
|
9627 |
+
|
9628 |
+
for c in cscore:
|
9629 |
+
if len([e for e in c if e[3] != 9]) > 0:
|
9630 |
+
tones_chords[tuple(sorted(set([e[4] % 12 for e in c if e[3] != 9])))] += 1
|
9631 |
+
|
9632 |
+
most_common_tones_chords = [check_and_fix_tones_chord(list(c[0])) for c in tones_chords.most_common(10)]
|
9633 |
+
|
9634 |
+
mood_scale = statistics.mode(tones_chords_to_types(most_common_tones_chords, return_chord_type_index=True)) % 3
|
9635 |
+
|
9636 |
+
mood = MOOD_SCALES[mood_scale]
|
9637 |
+
|
9638 |
+
#==============================================================================
|
9639 |
+
|
9640 |
+
if pitches:
|
9641 |
+
|
9642 |
+
escore_averages = escore_notes_averages(escore_notes, return_ptcs_and_vels=True)
|
9643 |
+
|
9644 |
+
if escore_averages[0] < 8:
|
9645 |
+
rythm = 'fast'
|
9646 |
+
|
9647 |
+
elif 8 <= escore_averages[0] <= 12:
|
9648 |
+
rythm = 'average'
|
9649 |
+
|
9650 |
+
elif escore_averages[0] > 12:
|
9651 |
+
rythm = 'slow'
|
9652 |
+
|
9653 |
+
if escore_averages[1] < 16:
|
9654 |
+
tempo = 'fast'
|
9655 |
+
|
9656 |
+
elif 16 <= escore_averages[1] <= 24:
|
9657 |
+
tempo = 'average'
|
9658 |
+
|
9659 |
+
elif escore_averages[1] > 24:
|
9660 |
+
tempo = 'slow'
|
9661 |
+
|
9662 |
+
if escore_averages[2] < 50:
|
9663 |
+
tone = 'bass'
|
9664 |
+
|
9665 |
+
elif 50 <= escore_averages[2] <= 70:
|
9666 |
+
tone = 'midrange'
|
9667 |
+
|
9668 |
+
elif escore_averages[2] > 70:
|
9669 |
+
tone = 'treble'
|
9670 |
+
|
9671 |
+
if escore_averages[3] < 80:
|
9672 |
+
dynamics = 'quiet'
|
9673 |
+
|
9674 |
+
elif 80 <= escore_averages[3] <= 100:
|
9675 |
+
dynamics = 'average'
|
9676 |
+
|
9677 |
+
elif escore_averages[3] > 100:
|
9678 |
+
dynamics = 'loud'
|
9679 |
+
|
9680 |
+
#==============================================================================
|
9681 |
+
|
9682 |
+
description = ''
|
9683 |
+
|
9684 |
+
if song_name != '':
|
9685 |
+
description = 'Song "' + song_name + '"'
|
9686 |
+
|
9687 |
+
if artist_name != '':
|
9688 |
+
description += ' by ' + artist_name
|
9689 |
+
|
9690 |
+
if song_name != '' or artist_name != '':
|
9691 |
+
description += '.'
|
9692 |
+
description += '\n'
|
9693 |
+
|
9694 |
+
description += 'The song is '
|
9695 |
+
|
9696 |
+
if song_length != 'average':
|
9697 |
+
description += 'a ' + song_length
|
9698 |
+
|
9699 |
+
else:
|
9700 |
+
description += 'an ' + song_length
|
9701 |
+
|
9702 |
+
description += ' duration '
|
9703 |
+
|
9704 |
+
description += comp_type + ' composition'
|
9705 |
+
|
9706 |
+
if comp_type != 'drum track':
|
9707 |
+
|
9708 |
+
if drums_present:
|
9709 |
+
description += ' with drums'
|
9710 |
+
|
9711 |
+
else:
|
9712 |
+
description += ' without drums'
|
9713 |
+
|
9714 |
+
if key and mood:
|
9715 |
+
description += ' in ' + key + ' ' + mood
|
9716 |
+
|
9717 |
+
description += '.'
|
9718 |
+
|
9719 |
+
description += '\n'
|
9720 |
+
|
9721 |
+
if pitches:
|
9722 |
+
|
9723 |
+
description += 'It has '
|
9724 |
+
|
9725 |
+
description += rythm + ' rythm, '
|
9726 |
+
description += tempo + ' tempo, '
|
9727 |
+
description += tone + ' tone and '
|
9728 |
+
description += dynamics + ' dynamics.'
|
9729 |
+
|
9730 |
+
description += '\n'
|
9731 |
+
|
9732 |
+
description += 'The song '
|
9733 |
+
|
9734 |
+
if len(instruments) == 1:
|
9735 |
+
description += 'is played on a solo ' + instruments[0] + '.'
|
9736 |
+
|
9737 |
+
else:
|
9738 |
+
description += 'features ' + NUMERALS[max(0, min(15, len(instruments)-1))] + ' instruments: '
|
9739 |
+
description += ', '.join(instruments[:-1]) + ' and ' + instruments[-1] + '.'
|
9740 |
+
|
9741 |
+
description += '\n'
|
9742 |
+
|
9743 |
+
if drums_present and most_common_drums:
|
9744 |
+
description += 'The drum track has predominant '
|
9745 |
+
description += ', '.join(most_common_drums[:-1]) + ' and ' + most_common_drums[-1] + '.'
|
9746 |
+
|
9747 |
+
#==============================================================================
|
9748 |
+
|
9749 |
+
return description
|
9750 |
+
|
9751 |
+
###################################################################################
|
9752 |
+
|
9753 |
+
#==================================================================================
|
9754 |
+
#
|
9755 |
+
# Below constants code is a courtesy of MidiTok
|
9756 |
+
#
|
9757 |
+
# Retrieved on 12/29/2024
|
9758 |
+
#
|
9759 |
+
# https://github.com/Natooz/MidiTok/blob/main/src/miditok/constants.py
|
9760 |
+
#
|
9761 |
+
#==================================================================================
|
9762 |
+
|
9763 |
+
MIDI_FILES_EXTENSIONS = [".mid", ".midi", ".kar", ".MID", ".MIDI", ".KAR"]
|
9764 |
+
|
9765 |
+
# The recommended pitches for piano in the GM2 specs are from 21 to 108
|
9766 |
+
PIANO_PITCH_RANGE = range(21, 109)
|
9767 |
+
|
9768 |
+
# Chord params
|
9769 |
+
# "chord_unknown" specifies the range of number of notes that can form "unknown" chords
|
9770 |
+
# (that do not fit in "chord_maps") to add in tokens.
|
9771 |
+
# Known chord maps, with 0 as root note
|
9772 |
+
BASIC_CHORDS_MAP = {
|
9773 |
+
"min": (0, 3, 7),
|
9774 |
+
"maj": (0, 4, 7),
|
9775 |
+
"dim": (0, 3, 6),
|
9776 |
+
"aug": (0, 4, 8),
|
9777 |
+
"sus2": (0, 2, 7),
|
9778 |
+
"sus4": (0, 5, 7),
|
9779 |
+
"7dom": (0, 4, 7, 10),
|
9780 |
+
"7min": (0, 3, 7, 10),
|
9781 |
+
"7maj": (0, 4, 7, 11),
|
9782 |
+
"7halfdim": (0, 3, 6, 10),
|
9783 |
+
"7dim": (0, 3, 6, 9),
|
9784 |
+
"7aug": (0, 4, 8, 11),
|
9785 |
+
"9maj": (0, 4, 7, 10, 14),
|
9786 |
+
"9min": (0, 4, 7, 10, 13),
|
9787 |
+
}
|
9788 |
+
|
9789 |
+
# Drums
|
9790 |
+
# Recommended range from the GM2 specs
|
9791 |
+
DRUMS_PITCH_RANGE = range(27, 90)
|
9792 |
+
|
9793 |
+
# Used with chords
|
9794 |
+
PITCH_CLASSES = ["C", "C#", "D", "D#", "E", "F", "F#", "G", "G#", "A", "A#", "B"]
|
9795 |
+
|
9796 |
+
# http://newt.phys.unsw.edu.au/jw/notes.html
|
9797 |
+
# https://www.midi.org/specifications
|
9798 |
+
|
9799 |
+
# index i = program i+1 in the GM2 specs (7. Appendix A)
|
9800 |
+
# index i = program i as retrieved by packages
|
9801 |
+
MIDI_INSTRUMENTS = [
|
9802 |
+
# Piano
|
9803 |
+
{"name": "Acoustic Grand Piano", "pitch_range": range(21, 109)},
|
9804 |
+
{"name": "Bright Acoustic Piano", "pitch_range": range(21, 109)},
|
9805 |
+
{"name": "Electric Grand Piano", "pitch_range": range(21, 109)},
|
9806 |
+
{"name": "Honky-tonk Piano", "pitch_range": range(21, 109)},
|
9807 |
+
{"name": "Electric Piano 1", "pitch_range": range(28, 104)},
|
9808 |
+
{"name": "Electric Piano 2", "pitch_range": range(28, 104)},
|
9809 |
+
{"name": "Harpsichord", "pitch_range": range(41, 90)},
|
9810 |
+
{"name": "Clavi", "pitch_range": range(36, 97)},
|
9811 |
+
# Chromatic Percussion
|
9812 |
+
{"name": "Celesta", "pitch_range": range(60, 109)},
|
9813 |
+
{"name": "Glockenspiel", "pitch_range": range(72, 109)},
|
9814 |
+
{"name": "Music Box", "pitch_range": range(60, 85)},
|
9815 |
+
{"name": "Vibraphone", "pitch_range": range(53, 90)},
|
9816 |
+
{"name": "Marimba", "pitch_range": range(48, 85)},
|
9817 |
+
{"name": "Xylophone", "pitch_range": range(65, 97)},
|
9818 |
+
{"name": "Tubular Bells", "pitch_range": range(60, 78)},
|
9819 |
+
{"name": "Dulcimer", "pitch_range": range(60, 85)},
|
9820 |
+
# Organs
|
9821 |
+
{"name": "Drawbar Organ", "pitch_range": range(36, 97)},
|
9822 |
+
{"name": "Percussive Organ", "pitch_range": range(36, 97)},
|
9823 |
+
{"name": "Rock Organ", "pitch_range": range(36, 97)},
|
9824 |
+
{"name": "Church Organ", "pitch_range": range(21, 109)},
|
9825 |
+
{"name": "Reed Organ", "pitch_range": range(36, 97)},
|
9826 |
+
{"name": "Accordion", "pitch_range": range(53, 90)},
|
9827 |
+
{"name": "Harmonica", "pitch_range": range(60, 85)},
|
9828 |
+
{"name": "Tango Accordion", "pitch_range": range(53, 90)},
|
9829 |
+
# Guitars
|
9830 |
+
{"name": "Acoustic Guitar (nylon)", "pitch_range": range(40, 85)},
|
9831 |
+
{"name": "Acoustic Guitar (steel)", "pitch_range": range(40, 85)},
|
9832 |
+
{"name": "Electric Guitar (jazz)", "pitch_range": range(40, 87)},
|
9833 |
+
{"name": "Electric Guitar (clean)", "pitch_range": range(40, 87)},
|
9834 |
+
{"name": "Electric Guitar (muted)", "pitch_range": range(40, 87)},
|
9835 |
+
{"name": "Overdriven Guitar", "pitch_range": range(40, 87)},
|
9836 |
+
{"name": "Distortion Guitar", "pitch_range": range(40, 87)},
|
9837 |
+
{"name": "Guitar Harmonics", "pitch_range": range(40, 87)},
|
9838 |
+
# Bass
|
9839 |
+
{"name": "Acoustic Bass", "pitch_range": range(28, 56)},
|
9840 |
+
{"name": "Electric Bass (finger)", "pitch_range": range(28, 56)},
|
9841 |
+
{"name": "Electric Bass (pick)", "pitch_range": range(28, 56)},
|
9842 |
+
{"name": "Fretless Bass", "pitch_range": range(28, 56)},
|
9843 |
+
{"name": "Slap Bass 1", "pitch_range": range(28, 56)},
|
9844 |
+
{"name": "Slap Bass 2", "pitch_range": range(28, 56)},
|
9845 |
+
{"name": "Synth Bass 1", "pitch_range": range(28, 56)},
|
9846 |
+
{"name": "Synth Bass 2", "pitch_range": range(28, 56)},
|
9847 |
+
# Strings & Orchestral instruments
|
9848 |
+
{"name": "Violin", "pitch_range": range(55, 94)},
|
9849 |
+
{"name": "Viola", "pitch_range": range(48, 85)},
|
9850 |
+
{"name": "Cello", "pitch_range": range(36, 73)},
|
9851 |
+
{"name": "Contrabass", "pitch_range": range(28, 56)},
|
9852 |
+
{"name": "Tremolo Strings", "pitch_range": range(28, 94)},
|
9853 |
+
{"name": "Pizzicato Strings", "pitch_range": range(28, 94)},
|
9854 |
+
{"name": "Orchestral Harp", "pitch_range": range(23, 104)},
|
9855 |
+
{"name": "Timpani", "pitch_range": range(36, 58)},
|
9856 |
+
# Ensembles
|
9857 |
+
{"name": "String Ensembles 1", "pitch_range": range(28, 97)},
|
9858 |
+
{"name": "String Ensembles 2", "pitch_range": range(28, 97)},
|
9859 |
+
{"name": "SynthStrings 1", "pitch_range": range(36, 97)},
|
9860 |
+
{"name": "SynthStrings 2", "pitch_range": range(36, 97)},
|
9861 |
+
{"name": "Choir Aahs", "pitch_range": range(48, 80)},
|
9862 |
+
{"name": "Voice Oohs", "pitch_range": range(48, 80)},
|
9863 |
+
{"name": "Synth Voice", "pitch_range": range(48, 85)},
|
9864 |
+
{"name": "Orchestra Hit", "pitch_range": range(48, 73)},
|
9865 |
+
# Brass
|
9866 |
+
{"name": "Trumpet", "pitch_range": range(58, 95)},
|
9867 |
+
{"name": "Trombone", "pitch_range": range(34, 76)},
|
9868 |
+
{"name": "Tuba", "pitch_range": range(29, 56)},
|
9869 |
+
{"name": "Muted Trumpet", "pitch_range": range(58, 83)},
|
9870 |
+
{"name": "French Horn", "pitch_range": range(41, 78)},
|
9871 |
+
{"name": "Brass Section", "pitch_range": range(36, 97)},
|
9872 |
+
{"name": "Synth Brass 1", "pitch_range": range(36, 97)},
|
9873 |
+
{"name": "Synth Brass 2", "pitch_range": range(36, 97)},
|
9874 |
+
# Reed
|
9875 |
+
{"name": "Soprano Sax", "pitch_range": range(54, 88)},
|
9876 |
+
{"name": "Alto Sax", "pitch_range": range(49, 81)},
|
9877 |
+
{"name": "Tenor Sax", "pitch_range": range(42, 76)},
|
9878 |
+
{"name": "Baritone Sax", "pitch_range": range(37, 69)},
|
9879 |
+
{"name": "Oboe", "pitch_range": range(58, 92)},
|
9880 |
+
{"name": "English Horn", "pitch_range": range(52, 82)},
|
9881 |
+
{"name": "Bassoon", "pitch_range": range(34, 73)},
|
9882 |
+
{"name": "Clarinet", "pitch_range": range(50, 92)},
|
9883 |
+
# Pipe
|
9884 |
+
{"name": "Piccolo", "pitch_range": range(74, 109)},
|
9885 |
+
{"name": "Flute", "pitch_range": range(60, 97)},
|
9886 |
+
{"name": "Recorder", "pitch_range": range(60, 97)},
|
9887 |
+
{"name": "Pan Flute", "pitch_range": range(60, 97)},
|
9888 |
+
{"name": "Blown Bottle", "pitch_range": range(60, 97)},
|
9889 |
+
{"name": "Shakuhachi", "pitch_range": range(55, 85)},
|
9890 |
+
{"name": "Whistle", "pitch_range": range(60, 97)},
|
9891 |
+
{"name": "Ocarina", "pitch_range": range(60, 85)},
|
9892 |
+
# Synth Lead
|
9893 |
+
{"name": "Lead 1 (square)", "pitch_range": range(21, 109)},
|
9894 |
+
{"name": "Lead 2 (sawtooth)", "pitch_range": range(21, 109)},
|
9895 |
+
{"name": "Lead 3 (calliope)", "pitch_range": range(36, 97)},
|
9896 |
+
{"name": "Lead 4 (chiff)", "pitch_range": range(36, 97)},
|
9897 |
+
{"name": "Lead 5 (charang)", "pitch_range": range(36, 97)},
|
9898 |
+
{"name": "Lead 6 (voice)", "pitch_range": range(36, 97)},
|
9899 |
+
{"name": "Lead 7 (fifths)", "pitch_range": range(36, 97)},
|
9900 |
+
{"name": "Lead 8 (bass + lead)", "pitch_range": range(21, 109)},
|
9901 |
+
# Synth Pad
|
9902 |
+
{"name": "Pad 1 (new age)", "pitch_range": range(36, 97)},
|
9903 |
+
{"name": "Pad 2 (warm)", "pitch_range": range(36, 97)},
|
9904 |
+
{"name": "Pad 3 (polysynth)", "pitch_range": range(36, 97)},
|
9905 |
+
{"name": "Pad 4 (choir)", "pitch_range": range(36, 97)},
|
9906 |
+
{"name": "Pad 5 (bowed)", "pitch_range": range(36, 97)},
|
9907 |
+
{"name": "Pad 6 (metallic)", "pitch_range": range(36, 97)},
|
9908 |
+
{"name": "Pad 7 (halo)", "pitch_range": range(36, 97)},
|
9909 |
+
{"name": "Pad 8 (sweep)", "pitch_range": range(36, 97)},
|
9910 |
+
# Synth SFX
|
9911 |
+
{"name": "FX 1 (rain)", "pitch_range": range(36, 97)},
|
9912 |
+
{"name": "FX 2 (soundtrack)", "pitch_range": range(36, 97)},
|
9913 |
+
{"name": "FX 3 (crystal)", "pitch_range": range(36, 97)},
|
9914 |
+
{"name": "FX 4 (atmosphere)", "pitch_range": range(36, 97)},
|
9915 |
+
{"name": "FX 5 (brightness)", "pitch_range": range(36, 97)},
|
9916 |
+
{"name": "FX 6 (goblins)", "pitch_range": range(36, 97)},
|
9917 |
+
{"name": "FX 7 (echoes)", "pitch_range": range(36, 97)},
|
9918 |
+
{"name": "FX 8 (sci-fi)", "pitch_range": range(36, 97)},
|
9919 |
+
# Ethnic Misc.
|
9920 |
+
{"name": "Sitar", "pitch_range": range(48, 78)},
|
9921 |
+
{"name": "Banjo", "pitch_range": range(48, 85)},
|
9922 |
+
{"name": "Shamisen", "pitch_range": range(50, 80)},
|
9923 |
+
{"name": "Koto", "pitch_range": range(55, 85)},
|
9924 |
+
{"name": "Kalimba", "pitch_range": range(48, 80)},
|
9925 |
+
{"name": "Bag pipe", "pitch_range": range(36, 78)},
|
9926 |
+
{"name": "Fiddle", "pitch_range": range(55, 97)},
|
9927 |
+
{"name": "Shanai", "pitch_range": range(48, 73)},
|
9928 |
+
# Percussive
|
9929 |
+
{"name": "Tinkle Bell", "pitch_range": range(72, 85)},
|
9930 |
+
{"name": "Agogo", "pitch_range": range(60, 73)},
|
9931 |
+
{"name": "Steel Drums", "pitch_range": range(52, 77)},
|
9932 |
+
{"name": "Woodblock", "pitch_range": range(128)},
|
9933 |
+
{"name": "Taiko Drum", "pitch_range": range(128)},
|
9934 |
+
{"name": "Melodic Tom", "pitch_range": range(128)},
|
9935 |
+
{"name": "Synth Drum", "pitch_range": range(128)},
|
9936 |
+
{"name": "Reverse Cymbal", "pitch_range": range(128)},
|
9937 |
+
# SFX
|
9938 |
+
{"name": "Guitar Fret Noise, Guitar Cutting Noise", "pitch_range": range(128)},
|
9939 |
+
{"name": "Breath Noise, Flute Key Click", "pitch_range": range(128)},
|
9940 |
+
{
|
9941 |
+
"name": "Seashore, Rain, Thunder, Wind, Stream, Bubbles",
|
9942 |
+
"pitch_range": range(128),
|
9943 |
+
},
|
9944 |
+
{"name": "Bird Tweet, Dog, Horse Gallop", "pitch_range": range(128)},
|
9945 |
+
{
|
9946 |
+
"name": "Telephone Ring, Door Creaking, Door, Scratch, Wind Chime",
|
9947 |
+
"pitch_range": range(128),
|
9948 |
+
},
|
9949 |
+
{"name": "Helicopter, Car Sounds", "pitch_range": range(128)},
|
9950 |
+
{
|
9951 |
+
"name": "Applause, Laughing, Screaming, Punch, Heart Beat, Footstep",
|
9952 |
+
"pitch_range": range(128),
|
9953 |
+
},
|
9954 |
+
{"name": "Gunshot, Machine Gun, Lasergun, Explosion", "pitch_range": range(128)},
|
9955 |
+
]
|
9956 |
+
|
9957 |
+
INSTRUMENTS_CLASSES = [
|
9958 |
+
{"name": "Piano", "program_range": range(8)}, # 0
|
9959 |
+
{"name": "Chromatic Percussion", "program_range": range(8, 16)},
|
9960 |
+
{"name": "Organ", "program_range": range(16, 24)},
|
9961 |
+
{"name": "Guitar", "program_range": range(24, 32)},
|
9962 |
+
{"name": "Bass", "program_range": range(32, 40)},
|
9963 |
+
{"name": "Strings", "program_range": range(40, 48)}, # 5
|
9964 |
+
{"name": "Ensemble", "program_range": range(48, 56)},
|
9965 |
+
{"name": "Brass", "program_range": range(56, 64)},
|
9966 |
+
{"name": "Reed", "program_range": range(64, 72)},
|
9967 |
+
{"name": "Pipe", "program_range": range(72, 80)},
|
9968 |
+
{"name": "Synth Lead", "program_range": range(80, 88)}, # 10
|
9969 |
+
{"name": "Synth Pad", "program_range": range(88, 96)},
|
9970 |
+
{"name": "Synth Effects", "program_range": range(96, 104)},
|
9971 |
+
{"name": "Ethnic", "program_range": range(104, 112)},
|
9972 |
+
{"name": "Percussive", "program_range": range(112, 120)},
|
9973 |
+
{"name": "Sound Effects", "program_range": range(120, 128)}, # 15
|
9974 |
+
{"name": "Drums", "program_range": range(-1, 0)},
|
9975 |
+
]
|
9976 |
+
|
9977 |
+
# To easily get the class index of any instrument program
|
9978 |
+
CLASS_OF_INST = [
|
9979 |
+
i
|
9980 |
+
for i, inst_class in enumerate(INSTRUMENTS_CLASSES)
|
9981 |
+
for _ in inst_class["program_range"]
|
9982 |
+
]
|
9983 |
+
|
9984 |
+
# index i = program i+1 in the GM2 specs (8. Appendix B)
|
9985 |
+
# index i = program i retrieved by packages
|
9986 |
+
DRUMS_SETS = {
|
9987 |
+
0: "Standard",
|
9988 |
+
8: "Room",
|
9989 |
+
16: "Power",
|
9990 |
+
24: "Electronic",
|
9991 |
+
25: "Analog",
|
9992 |
+
32: "Jazz",
|
9993 |
+
40: "Brush",
|
9994 |
+
48: "Orchestra",
|
9995 |
+
56: "SFX",
|
9996 |
+
}
|
9997 |
+
|
9998 |
+
# Control changes list (without specifications):
|
9999 |
+
# https://www.midi.org/specifications-old/item/table-3-control-change-messages-data-bytes-2
|
10000 |
+
# Undefined and general control changes are not considered here
|
10001 |
+
# All these attributes can take values from 0 to 127, with some of them being on/off
|
10002 |
+
CONTROL_CHANGES = {
|
10003 |
+
# MSB
|
10004 |
+
0: "Bank Select",
|
10005 |
+
1: "Modulation Depth",
|
10006 |
+
2: "Breath Controller",
|
10007 |
+
4: "Foot Controller",
|
10008 |
+
5: "Portamento Time",
|
10009 |
+
6: "Data Entry",
|
10010 |
+
7: "Channel Volume",
|
10011 |
+
8: "Balance",
|
10012 |
+
10: "Pan",
|
10013 |
+
11: "Expression Controller",
|
10014 |
+
# LSB
|
10015 |
+
32: "Bank Select",
|
10016 |
+
33: "Modulation Depth",
|
10017 |
+
34: "Breath Controller",
|
10018 |
+
36: "Foot Controller",
|
10019 |
+
37: "Portamento Time",
|
10020 |
+
38: "Data Entry",
|
10021 |
+
39: "Channel Volume",
|
10022 |
+
40: "Balance",
|
10023 |
+
42: "Pan",
|
10024 |
+
43: "Expression Controller",
|
10025 |
+
# On / Off control changes, ≤63 off, ≥64 on
|
10026 |
+
64: "Damper Pedal",
|
10027 |
+
65: "Portamento",
|
10028 |
+
66: "Sostenuto",
|
10029 |
+
67: "Soft Pedal",
|
10030 |
+
68: "Legato Footswitch",
|
10031 |
+
69: "Hold 2",
|
10032 |
+
# Continuous controls
|
10033 |
+
70: "Sound Variation",
|
10034 |
+
71: "Timbre/Harmonic Intensity",
|
10035 |
+
72: "Release Time",
|
10036 |
+
73: "Attack Time",
|
10037 |
+
74: "Brightness",
|
10038 |
+
75: "Decay Time",
|
10039 |
+
76: "Vibrato Rate",
|
10040 |
+
77: "Vibrato Depth",
|
10041 |
+
78: "Vibrato Delay",
|
10042 |
+
84: "Portamento Control",
|
10043 |
+
88: "High Resolution Velocity Prefix",
|
10044 |
+
# Effects depths
|
10045 |
+
91: "Reverb Depth",
|
10046 |
+
92: "Tremolo Depth",
|
10047 |
+
93: "Chorus Depth",
|
10048 |
+
94: "Celeste Depth",
|
10049 |
+
95: "Phaser Depth",
|
10050 |
+
# Registered parameters numbers
|
10051 |
+
96: "Data Increment",
|
10052 |
+
97: "Data Decrement",
|
10053 |
+
# 98: 'Non-Registered Parameter Number (NRPN) - LSB',
|
10054 |
+
# 99: 'Non-Registered Parameter Number (NRPN) - MSB',
|
10055 |
+
100: "Registered Parameter Number (RPN) - LSB",
|
10056 |
+
101: "Registered Parameter Number (RPN) - MSB",
|
10057 |
+
# Channel mode controls
|
10058 |
+
120: "All Sound Off",
|
10059 |
+
121: "Reset All Controllers",
|
10060 |
+
122: "Local Control On/Off",
|
10061 |
+
123: "All Notes Off",
|
10062 |
+
124: "Omni Mode Off", # + all notes off
|
10063 |
+
125: "Omni Mode On", # + all notes off
|
10064 |
+
126: "Mono Mode On", # + poly off, + all notes off
|
10065 |
+
127: "Poly Mode On", # + mono off, +all notes off
|
10066 |
+
}
|
10067 |
+
|
10068 |
+
###################################################################################
|
10069 |
+
|
10070 |
+
def patches_onset_times(escore_notes, times_idx=1, patches_idx=6):
|
10071 |
+
|
10072 |
+
patches = [e[patches_idx] for e in escore_notes]
|
10073 |
+
|
10074 |
+
patches_oset = ordered_set(patches)
|
10075 |
+
|
10076 |
+
patches_onset_times = []
|
10077 |
+
|
10078 |
+
for p in patches_oset:
|
10079 |
+
for e in escore_notes:
|
10080 |
+
if e[patches_idx] == p:
|
10081 |
+
patches_onset_times.append([p, e[times_idx]])
|
10082 |
+
break
|
10083 |
+
|
10084 |
+
return patches_onset_times
|
10085 |
+
|
10086 |
+
###################################################################################
|
10087 |
+
|
10088 |
+
def count_escore_notes_patches(escore_notes, patches_idx=6):
|
10089 |
+
|
10090 |
+
patches = [e[patches_idx] for e in escore_notes]
|
10091 |
+
|
10092 |
+
return Counter(patches).most_common()
|
10093 |
+
|
10094 |
+
###################################################################################
|
10095 |
+
|
10096 |
+
def escore_notes_monoponic_melodies(escore_notes,
|
10097 |
+
bad_notes_ratio=0.0,
|
10098 |
+
times_idx=1,
|
10099 |
+
patches_idx=6
|
10100 |
+
):
|
10101 |
+
|
10102 |
+
patches = escore_notes_patches(escore_notes, patches_index=patches_idx)
|
10103 |
+
|
10104 |
+
monophonic_melodies = []
|
10105 |
+
|
10106 |
+
for p in patches:
|
10107 |
+
patch_score = [e for e in escore_notes if e[patches_idx] == p]
|
10108 |
+
|
10109 |
+
ps_times = [e[times_idx] for e in patch_score]
|
10110 |
+
|
10111 |
+
if len(ps_times) <= len(set(ps_times)) * (1+bad_notes_ratio):
|
10112 |
+
monophonic_melodies.append([p, len(patch_score)])
|
10113 |
+
|
10114 |
+
return monophonic_melodies
|
10115 |
+
|
10116 |
+
###################################################################################
|
10117 |
+
|
10118 |
+
from itertools import groupby
|
10119 |
+
from operator import itemgetter
|
10120 |
+
|
10121 |
+
def group_by_threshold(data, threshold, groupby_idx):
|
10122 |
+
|
10123 |
+
data.sort(key=itemgetter(groupby_idx))
|
10124 |
+
|
10125 |
+
grouped_data = []
|
10126 |
+
cluster = []
|
10127 |
+
|
10128 |
+
for i, item in enumerate(data):
|
10129 |
+
if not cluster:
|
10130 |
+
cluster.append(item)
|
10131 |
+
elif abs(item[groupby_idx] - cluster[-1][groupby_idx]) <= threshold:
|
10132 |
+
cluster.append(item)
|
10133 |
+
else:
|
10134 |
+
grouped_data.append(cluster)
|
10135 |
+
cluster = [item]
|
10136 |
+
|
10137 |
+
if cluster:
|
10138 |
+
grouped_data.append(cluster)
|
10139 |
+
|
10140 |
+
return grouped_data
|
10141 |
+
|
10142 |
+
###################################################################################
|
10143 |
+
|
10144 |
+
def split_escore_notes_by_time(escore_notes, time_threshold=256):
|
10145 |
+
|
10146 |
+
dscore = delta_score_notes(escore_notes, timings_clip_value=time_threshold-1)
|
10147 |
+
|
10148 |
+
score_chunks = []
|
10149 |
+
|
10150 |
+
ctime = 0
|
10151 |
+
pchunk_idx = 0
|
10152 |
+
|
10153 |
+
for i, e in enumerate(dscore):
|
10154 |
+
|
10155 |
+
ctime += e[1]
|
10156 |
+
|
10157 |
+
if ctime >= time_threshold:
|
10158 |
+
score_chunks.append(escore_notes[pchunk_idx:i])
|
10159 |
+
pchunk_idx = i
|
10160 |
+
ctime = 0
|
10161 |
+
|
10162 |
+
return score_chunks
|
10163 |
+
|
10164 |
+
###################################################################################
|
10165 |
+
|
10166 |
+
def escore_notes_grouped_patches(escore_notes, time_threshold=256):
|
10167 |
+
|
10168 |
+
split_score_chunks = split_escore_notes_by_time(escore_notes,
|
10169 |
+
time_threshold=time_threshold
|
10170 |
+
)
|
10171 |
+
|
10172 |
+
chunks_patches = []
|
10173 |
+
|
10174 |
+
for s in split_score_chunks:
|
10175 |
+
chunks_patches.append(escore_notes_patches(s))
|
10176 |
+
|
10177 |
+
return chunks_patches
|
10178 |
+
|
10179 |
+
###################################################################################
|
10180 |
+
|
10181 |
+
def computeLPSArray(pattern, M, lps):
|
10182 |
+
length = 0
|
10183 |
+
i = 1
|
10184 |
+
|
10185 |
+
lps[0] = 0
|
10186 |
+
|
10187 |
+
while i < M:
|
10188 |
+
if pattern[i] == pattern[length]:
|
10189 |
+
length += 1
|
10190 |
+
lps[i] = length
|
10191 |
+
i += 1
|
10192 |
+
else:
|
10193 |
+
if length != 0:
|
10194 |
+
length = lps[length-1]
|
10195 |
+
else:
|
10196 |
+
lps[i] = 0
|
10197 |
+
i += 1
|
10198 |
+
|
10199 |
+
###################################################################################
|
10200 |
+
|
10201 |
+
def find_pattern_idxs(sub_pattern, pattern):
|
10202 |
+
|
10203 |
+
lst = pattern
|
10204 |
+
pattern = sub_pattern
|
10205 |
+
|
10206 |
+
M = len(pattern)
|
10207 |
+
N = len(lst)
|
10208 |
+
|
10209 |
+
lps = [0] * M
|
10210 |
+
j = 0 # index for pattern[]
|
10211 |
+
|
10212 |
+
computeLPSArray(pattern, M, lps)
|
10213 |
+
|
10214 |
+
i = 0 # index for lst[]
|
10215 |
+
indexes = []
|
10216 |
+
|
10217 |
+
while i < N:
|
10218 |
+
if pattern[j] == lst[i]:
|
10219 |
+
i += 1
|
10220 |
+
j += 1
|
10221 |
+
|
10222 |
+
if j == M:
|
10223 |
+
end_index = i - 1
|
10224 |
+
start_index = end_index - M + 1
|
10225 |
+
indexes.append((start_index, end_index))
|
10226 |
+
j = lps[j-1]
|
10227 |
+
elif i < N and pattern[j] != lst[i]:
|
10228 |
+
if j != 0:
|
10229 |
+
j = lps[j-1]
|
10230 |
+
else:
|
10231 |
+
i += 1
|
10232 |
+
|
10233 |
+
return indexes
|
10234 |
+
|
10235 |
+
###################################################################################
|
10236 |
+
|
10237 |
+
def escore_notes_patch_lrno_patterns(escore_notes,
|
10238 |
+
patch=0,
|
10239 |
+
zero_score_timings=False,
|
10240 |
+
pitches_idx=4,
|
10241 |
+
patches_idx=6
|
10242 |
+
):
|
10243 |
+
|
10244 |
+
patch_escore = [e for e in escore_notes if e[patches_idx] == patch]
|
10245 |
+
|
10246 |
+
if patch_escore:
|
10247 |
+
|
10248 |
+
patch_cscore = chordify_score([1000, patch_escore])
|
10249 |
+
|
10250 |
+
patch_tscore = []
|
10251 |
+
|
10252 |
+
for c in patch_cscore:
|
10253 |
+
|
10254 |
+
tones_chord = sorted(set([p[pitches_idx] % 12 for p in c]))
|
10255 |
+
|
10256 |
+
if tones_chord not in ALL_CHORDS_SORTED:
|
10257 |
+
tnoes_chord = check_and_fix_tones_chord(tones_chord)
|
10258 |
+
|
10259 |
+
patch_tscore.append(ALL_CHORDS_SORTED.index(tones_chord))
|
10260 |
+
|
10261 |
+
pattern = find_lrno_pattern_fast(patch_tscore)
|
10262 |
+
|
10263 |
+
patterns_idxs = find_pattern_idxs(pattern, patch_tscore)
|
10264 |
+
|
10265 |
+
patch_lrno_scores = []
|
10266 |
+
|
10267 |
+
for idxs in patterns_idxs:
|
10268 |
+
|
10269 |
+
score = patch_escore[idxs[0]:idxs[1]]
|
10270 |
+
|
10271 |
+
if zero_score_timings:
|
10272 |
+
score = recalculate_score_timings(score)
|
10273 |
+
|
10274 |
+
patch_lrno_scores.append(score)
|
10275 |
+
|
10276 |
+
return patch_lrno_scores
|
10277 |
+
|
10278 |
+
else:
|
10279 |
+
return []
|
10280 |
+
|
10281 |
+
###################################################################################
|
10282 |
+
|
10283 |
+
ALL_BASE_CHORDS_SORTED = [[0], [0, 2], [0, 2, 4], [0, 2, 4, 6], [0, 2, 4, 6, 8], [0, 2, 4, 6, 8, 10],
|
10284 |
+
[0, 2, 4, 6, 9], [0, 2, 4, 6, 10], [0, 2, 4, 7], [0, 2, 4, 7, 9],
|
10285 |
+
[0, 2, 4, 7, 10], [0, 2, 4, 8], [0, 2, 4, 8, 10], [0, 2, 4, 9], [0, 2, 4, 10],
|
10286 |
+
[0, 2, 5], [0, 2, 5, 7], [0, 2, 5, 7, 9], [0, 2, 5, 7, 10], [0, 2, 5, 8],
|
10287 |
+
[0, 2, 5, 8, 10], [0, 2, 5, 9], [0, 2, 5, 10], [0, 2, 6], [0, 2, 6, 8],
|
10288 |
+
[0, 2, 6, 8, 10], [0, 2, 6, 9], [0, 2, 6, 10], [0, 2, 7], [0, 2, 7, 9],
|
10289 |
+
[0, 2, 7, 10], [0, 2, 8], [0, 2, 8, 10], [0, 2, 9], [0, 2, 10], [0, 3],
|
10290 |
+
[0, 3, 5], [0, 3, 5, 7], [0, 3, 5, 7, 9], [0, 3, 5, 7, 10], [0, 3, 5, 8],
|
10291 |
+
[0, 3, 5, 8, 10], [0, 3, 5, 9], [0, 3, 5, 10], [0, 3, 6], [0, 3, 6, 8],
|
10292 |
+
[0, 3, 6, 8, 10], [0, 3, 6, 9], [0, 3, 6, 10], [0, 3, 7], [0, 3, 7, 9],
|
10293 |
+
[0, 3, 7, 10], [0, 3, 8], [0, 3, 8, 10], [0, 3, 9], [0, 3, 10], [0, 4],
|
10294 |
+
[0, 4, 6], [0, 4, 6, 8], [0, 4, 6, 8, 10], [0, 4, 6, 9], [0, 4, 6, 10],
|
10295 |
+
[0, 4, 7], [0, 4, 7, 9], [0, 4, 7, 10], [0, 4, 8], [0, 4, 8, 10], [0, 4, 9],
|
10296 |
+
[0, 4, 10], [0, 5], [0, 5, 7], [0, 5, 7, 9], [0, 5, 7, 10], [0, 5, 8],
|
10297 |
+
[0, 5, 8, 10], [0, 5, 9], [0, 5, 10], [0, 6], [0, 6, 8], [0, 6, 8, 10],
|
10298 |
+
[0, 6, 9], [0, 6, 10], [0, 7], [0, 7, 9], [0, 7, 10], [0, 8], [0, 8, 10],
|
10299 |
+
[0, 9], [0, 10]]
|
10300 |
+
|
10301 |
+
###################################################################################
|
10302 |
#
|
10303 |
# This is the end of the TMIDI X Python module
|
10304 |
#
|