-
Notifications
You must be signed in to change notification settings - Fork 1
/
write_midi.py
123 lines (111 loc) · 6 KB
/
write_midi.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
import numpy as np
import pretty_midi
def set_piano_roll_to_instrument(piano_roll, instrument, velocity=100, tempo=90.0, beat_resolution=16):
# Calculate time per pixel
tpp = 60.0 / tempo / float(beat_resolution)
threshold = 60.0 / tempo / 4
phrase_end_time = 60.0 / tempo * 4 * piano_roll.shape[0]
# Create piano_roll_search that captures note onsets and offsets
piano_roll = piano_roll.reshape((piano_roll.shape[0] * piano_roll.shape[1], piano_roll.shape[2]))
print(piano_roll.shape)
piano_roll_diff = np.concatenate((np.zeros((piano_roll.shape[0], 12), dtype=np.float32), piano_roll, np.zeros((piano_roll.shape[0], 18), dtype=np.float)),axis=1)
piano_roll_search = np.diff(piano_roll_diff.astype(int), axis=0)
# Iterate through all possible(128) pitches
for note_num in range(128):
# Search for notes
start_idx = (piano_roll_search[:, note_num] > 0).nonzero()
start_time = list(tpp * (start_idx[0].astype(float)))
# print('start_time:', start_time)
# print(len(start_time))
end_idx = (piano_roll_search[:, note_num] < 0).nonzero()
end_time = list(tpp * (end_idx[0].astype(float)))
# print('end_time:', end_time)
# print(len(end_time))
duration = [pair[1] - pair[0] for pair in zip(start_time, end_time)]
# print('duration each note:', duration)
# print(len(duration))
temp_start_time = [i for i in start_time]
temp_end_time = [i for i in end_time]
for i in range(len(start_time)):
# print(start_time)
if start_time[i] in temp_start_time and i != len(start_time) - 1:
# print('i and start_time:', i, start_time[i])
t = []
current_idx = temp_start_time.index(start_time[i])
for j in range(current_idx + 1, len(temp_start_time)):
# print(j, temp_start_time[j])
if temp_start_time[j] < start_time[i] + threshold and temp_end_time[j] <= start_time[i] + threshold:
# print('popped start time:', temp_start_time[j])
t.append(j)
# print('popped temp_start_time:', t)
for _ in t:
temp_start_time.pop(t[0])
temp_end_time.pop(t[0])
# print('popped temp_start_time:', temp_start_time)
start_time = temp_start_time
# print('After checking, start_time:', start_time)
# print(len(start_time))
end_time = temp_end_time
# print('After checking, end_time:', end_time)
# print(len(end_time))
duration = [pair[1] - pair[0] for pair in zip(start_time, end_time)]
# print('After checking, duration each note:', duration)
# print(len(duration))
if len(end_time) < len(start_time):
d = len(start_time) - len(end_time)
start_time = start_time[:-d]
# Iterate through all the searched notes
for idx in range(len(start_time)):
if duration[idx] >= threshold:
# Create an Note object with corresponding note number, start time and end time
note = pretty_midi.Note(velocity=velocity, pitch=note_num, start=start_time[idx], end=end_time[idx])
# Add the note to the Instrument object
instrument.notes.append(note)
else:
if start_time[idx] + threshold <= phrase_end_time:
# Create an Note object with corresponding note number, start time and end time
note = pretty_midi.Note(velocity=velocity, pitch=note_num, start=start_time[idx],
end=start_time[idx] + threshold)
else:
# Create an Note object with corresponding note number, start time and end time
note = pretty_midi.Note(velocity=velocity, pitch=note_num, start=start_time[idx],
end=phrase_end_time)
# Add the note to the Instrument object
instrument.notes.append(note)
# Sort the notes by their start time
instrument.notes.sort(key=lambda note: note.start)
# print(max([i.end for i in instrument.notes]))
# print('tpp, threshold, phrases_end_time:', tpp, threshold, phrase_end_time)
def write_piano_roll_to_midi(piano_roll, filename, program_num=0, is_drum=False, velocity=100,
tempo=90.0, beat_resolution=16):
# Create a PrettyMIDI object
midi = pretty_midi.PrettyMIDI(initial_tempo=tempo)
# Create an Instrument object
instrument = pretty_midi.Instrument(program=program_num, is_drum=is_drum)
# Set the piano roll to the Instrument object
set_piano_roll_to_instrument(piano_roll, instrument, velocity, tempo, beat_resolution)
# Add the instrument to the PrettyMIDI object
midi.instruments.append(instrument)
# Write out the MIDI data
midi.write(filename)
def write_piano_rolls_to_midi(piano_rolls, program_nums=None, is_drum=None, filename='test.mid', velocity=100,
tempo=60.0, beat_resolution=24):
if len(piano_rolls) != len(program_nums) or len(piano_rolls) != len(is_drum):
print("Error: piano_rolls and program_nums have different sizes...")
return False
if not program_nums:
program_nums = [0, 0, 0]
if not is_drum:
is_drum = [False, False, False]
# Create a PrettyMIDI object
midi = pretty_midi.PrettyMIDI(initial_tempo=tempo)
# Iterate through all the input instruments
for idx in range(len(piano_rolls)):
# Create an Instrument object
instrument = pretty_midi.Instrument(program=program_nums[idx], is_drum=is_drum[idx])
# Set the piano roll to the Instrument object
set_piano_roll_to_instrument(piano_rolls[idx], instrument, velocity, tempo, beat_resolution)
# Add the instrument to the PrettyMIDI object
midi.instruments.append(instrument)
# Write out the MIDI data
midi.write(filename)