Mercurial > ~darius > hgwebdir.cgi > musiccutter
changeset 0:0773354c7428
Initial commit, WIP.
author | Daniel O'Connor <darius@dons.net.au> |
---|---|
date | Sat, 05 Mar 2016 21:59:44 +1030 (2016-03-05) |
parents | |
children | 123b42d95ab8 |
files | Makefile README musiccutter.py notes test.mup |
diffstat | 5 files changed, 179 insertions(+), 0 deletions(-) [+] |
line wrap: on
line diff
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/Makefile Sat Mar 05 21:59:44 2016 +1030 @@ -0,0 +1,24 @@ +.SUFFIXES: .mup .ps .midi + +MUP= mup +RM= rm -f +MOPTS= +MVOPTS= +MPOPTS= -DPIANO -DNOVOICE + +all: test.midi test.ps +#all: test.ps test-piano.ps test.midi + +test-piano.ps: test.mup + $(MUP) $(MOPTS) $(MPOPTS) $> > $@ + +clean: + $(RM) test.ps test-piano.ps + +# Generic rules +.mup.midi: + $(MUP) -m $@ $> + +.mup.ps: + $(MUP) $(MOPTS) $(MVOPTS) $> > $@ +
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/README Sat Mar 05 21:59:44 2016 +1030 @@ -0,0 +1,1 @@ +https://github.com/vishnubob/python-midi.git
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/musiccutter.py Sat Mar 05 21:59:44 2016 +1030 @@ -0,0 +1,93 @@ +#!/usr/bin/env python + +import exceptions +import mido +import svgwrite +import sys + +def test(): + # http://www.orgues-de-barbarie.com/wp-content/uploads/2014/09/format-cartons.pdf + conf = { 'notefile' : 'notes', 'pitch' : 5.5 } + midi2note, note2midi = genmidi2note() + note2slot = loadnote2slot(conf['notefile'], note2midi) + midi2svg('test.midi', 'test%d.svg', conf['pitch'], midi2note, note2midi, note2slot) + +# http://www.electronics.dit.ie/staff/tscarff/Music_technology/midi/midi_note_numbers_for_octaves.htm +def genmidi2note(): + '''Create forward & reverse tables for midi number to note name (assuming 69 == A440)''' + names = ['C%d', 'C%d#', 'D%d', 'D%d#', 'E%d', 'F%d', 'F%d#', 'G%d', 'G%d#', 'A%d', 'A%d#', 'B%d'] + midi2note = {} + note2midi = {} + for midi in range(128): + octave = midi / len(names) + index = midi % len(names) + name = names[index] % (octave) + midi2note[midi] = name + note2midi[name] = midi + + return midi2note, note2midi + +def loadnote2slot(fname, note2midi): + svg = svgwrite.Drawing(fname) + + note2slot = {} + index = 0 + + for note in file(fname): + note = note.strip() + if note[0] == '#': + continue + if note not in note2midi: + raise exceptions.ValueError('Note \'%s\' not valid' % note) + note2slot[note] = index + index += 1 + + return note2slot + +def emitnote(svg, slot, start, notelen): + # Let us say 185 ticks for a quaver and that is 5mm long + tickscale = 185.0 / 5.0 + pitch = 5.5 + offset = 6 + x = start / tickscale + y = offset + slot * pitch + w = notelen / tickscale + h = pitch + print x, y, w, h + svg.add(svgwrite.shapes.Rect(insert = ('%.3fcm' % (x / 10, '%.3fcm' % (y / 10)), size = (w, h), stroke = 'red', stroke_width = '1px', fill = 'red')) + +def midi2svg(inf, outpat, pitch, midi2note, note2midi, note2slot): + midi = mido.MidiFile(inf) + svg = svgwrite.Drawing(outpat % (0)) + tnum = 0 + for t in midi.tracks: + print "Track", tnum + tnum += 1 + channels = [] + for i in range(16): + channels.append({}) + ctime = 0 + for ev in t: + ctime += ev.time + if ev.type == 'note_on' and ev.velocity > 0: + if ev.note in channels[ev.channel]: + print "Duplicate channel on message" + else: + channels[ev.channel][ev.note] = ctime + elif ev.type == 'note_off' or (ev.type == 'note_on' and ev.velocity == 0): + if ev.note not in channels[ev.channel]: + print 'Note_off with no corresponding note_on for track %d channel %d note %d' % (tnum, ev.channel, ev.note) + else: + note = midi2note[ev.note] + if note not in note2slot: + print 'Skipping unplayable note %s' % (note) + else: + notelen = ctime - channels[ev.channel][ev.note] + slot = note2slot[note] + print 'Note %s (%d) at %d length %d' % (note, slot, ctime, notelen) + emitnote(svg, slot, ctime, notelen) + del channels[ev.channel][ev.note] + svg.save() + +if __name__ == '__main__': + main()
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/notes Sat Mar 05 21:59:44 2016 +1030 @@ -0,0 +1,30 @@ +# List of notes the organ plays in order +# Any line beginning with a # is a comment +# http://www.orgues-de-barbarie.com/wp-content/uploads/2014/09/gammes.pdf +C3 +D3 +F3 +G3 +C4 +D4 +E4 +F4 +F4# +G4 +A4 +A4# +B4 +C5 +C5# +D5 +D5# +E5 +F5 +F5# +G5 +G5# +A5 +A5# +B5 +C6 +D6
--- /dev/null Thu Jan 01 00:00:00 1970 +0000 +++ b/test.mup Sat Mar 05 21:59:44 2016 +1030 @@ -0,0 +1,31 @@ +header + title(22) "Test" + +score + scale = 0.75 + packfact = 0.5 + units = cm +// pageheight = 29.7 +// pagewidth = 21 + topmargin = 1 + botmargin = 1 + leftmargin = 1 + rightmargin = 1 + measnum = y + staffs = 1 + +music + +// Put each voice on a different channel +// using a different instrument sound. +// The program numbers correspond to the +// General MIDI sounds as noted. +midi 1 1: 0 "channel=1"; 0 "program=1"; 0 "parameter=10,0"; + +// Bar 1 +1 1: 4a; 4b; 2d; +endbar + +// Bar 2 +1 1: 4c+#; 4r; 2e; +endbar \ No newline at end of file