-
Notifications
You must be signed in to change notification settings - Fork 6
/
Copy pathmidi_to_csv.py
96 lines (74 loc) · 2.79 KB
/
midi_to_csv.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
"""
This is the midi to csv module.
"""
# TODO: ADD CHANNEL SUPPORT
import os
import random
import sys
import argparse
import music21
import pandas as pd
def is_dir(dirname):
"""Checks if a path is an actual directory"""
if not os.path.isdir(dirname):
msg = "{0} is not a directory".format(dirname)
raise argparse.ArgumentTypeError(msg)
else:
return dirname
def main():
#us = music21.environment.UserSettings()
#us.getSettingsPath()
# print command line arguments
parser = argparse.ArgumentParser()
parser.add_argument("input_dir_name", help="name of input directory of mid files",
type=str)
parser.add_argument("-o", "--output_dir_name", help="name of output directory of csv files, if argument not specified, will create one with suffix output-csvs",
type=str)
args = parser.parse_args()
is_dir(args.input_dir_name)
input_dir = os.fsencode(args.input_dir_name)
output_dir_name = args.output_dir_name
if not output_dir_name:
output_dir_name = args.input_dir_name + "-output-csvs"
if not os.path.exists(output_dir_name):
os.makedirs(output_dir_name)
print("Outputting csv files in to " + output_dir_name)
for file in os.listdir(input_dir):
filename = os.fsdecode(file)
print("Processing " + filename + " in to " + filename[:-4] + ".csv")
assert filename.endswith(".mid"), "files must be midi files"
mf = music21.midi.MidiFile()
mf.open(args.input_dir_name + "/" + filename)
mf.read()
mf.close()
s = music21.midi.translate.midiFileToStream(mf, quantizePost=False).flat #quantize is what rounds all note durations to real music note types, not needed for our application
# Convert chords in to notes.
# TODO: consider chords as separate objects from notes? Everything's in music21 anyways
df = pd.DataFrame(columns=["note_name", "start_time", "duration", "velocity", "tempo"])
for g in s.recurse().notes:
#print(g)
if g.isChord:
#print("chord")
#print(g.pitches)
for pitch in g.pitches:
x = music21.note.Note(pitch, duration=g.duration)
x.volume.velocity = g.volume.velocity
x.offset = g.offset
#print(x)
s.insert(x)
# ALERT: assumes only one tempo
note_tempo = s.metronomeMarkBoundaries()[0][2].number
for note in s.recurse().notes:
if note.isNote:
# print(note.offset)
# print(note.duration)
new_df = pd.DataFrame([[note.pitch, round(float(note.offset), 3), round(note.duration.quarterLength, 3), note.volume.velocity, note_tempo]], columns=["note_name", "start_time", "duration", "velocity", "tempo"])
df = df.append(new_df, ignore_index=True)
#print(df)
#print(new_df)
#print(note)
#print(df)
df.to_csv(output_dir_name + "/" + filename[:-4] + ".csv")
print("Done creating csvs!")
if __name__ == "__main__":
main()