music21:读取MIDI文件的BPM和乐器信息并写回文件

8
我正在尝试实现一个读取MIDI文件并将其写回的代码。
我有以下代码来解析持续时间、音高和位置。
import music21
from music21 import *
piece=converter.parse('input.mid')

all_parts=[]
for part in piece.parts:
    part_tuples=[]
    try:
        track_name = part[0].bestName()
    except AttributeError:
        track_name = 'None'
    part_tuples.append(track_name)
    for event in part:
        for y in event.contextSites():
            if y[0] is part:
                offset=y[1]
        if getattr(event,'isNote',None) and event.isNote:
            part_tuples.append([event.quarterLength,event.pitch.midi,offset])
        if getattr(event,'isRest',None) and event.isRest:
            part_tuples.append([event.quarterLength,'Rest',offset])

    all_parts.append(part_tuples)

然后我进行一些转换,并将其写回文件,如果音高为-1(“isRest”),则扩展t(所有音符的速度设置为90):

mt = midi.MidiTrack(1)

t=0
tLast=0
for d,p,v in converted_notes:
    if p!=-1:
        dt = midi.DeltaTime(mt)
        dt.time = t-tLast
        #add to track events
        mt.events.append(dt)

        me=midi.MidiEvent(mt)
        me.type="NOTE_ON"
        me.channel=1
        me.time= None #d
        me.pitch = p
        me.velocity = v
        mt.events.append(me)

    # add note off / velocity zero message
        dt = midi.DeltaTime(mt)
        dt.time = d
    # add to track events
        mt.events.append(dt)

        me=midi.MidiEvent(mt)
        me.type="NOTE_ON"
        me.channel=1
        me.time= None #d
        me.pitch = p
        me.velocity = 0
        mt.events.append(me)

        tLast = t+d
    #    t +=2*d
        t+=d

    else:
        t+=d

dt=midi.DeltaTime(mt)
dt.time = 0
mt.events.append(dt)
me = midi.MidiEvent(mt)
me.type = "END_OF_TRACK"
me.channel = 1
me.data =''  # must set data to empty string
mt.events.append(me)


mf = midi.MidiFile()
mf.ticksPerQuarterNote = 1024 # cannot use: 10080
mf.tracks.append(mt)

#mf.tracks.append(mt2)


mf.open('writeback.mid', 'wb')
mf.write()
mf.close()

然而,读取部分没有包含MIDI文件的总体速度/BPM或特定乐器来源('bestName'似乎只是猜测),因此写入部分不会强制执行任何BPM或乐器来源信息。
是否有一种方法可以读取/解析并为新的MIDI文件写入/强制执行相同的速度和乐器?
我查看了文档中的MidiFile和MidiTrack部分(http://web.mit.edu/music21/doc/moduleReference/moduleMidi.html#midifile),但只能找到有关通道或每四分音符刻度的信息,这些信息并不完全符合我的要求。
**************编辑**********
我发现了一种获取曲目BPM的方法,尽管这是一种非常笨拙的方法。
for i in range(0,20):
    bpm =str(part[i])
    if 'MetronomeMark' in bpm:
        eq_ind=bpm.index('=')
        bpm=bpm[eq_ind+1:]
        bpm=bpm.replace('>','')
        break
bpm=float(bpm)

除了最初的问题,我还需要确定每个音轨的通道号,以便区分打击乐和非打击乐音轨。


在6.7.1版本中仍然存在问题吗?我在6.5中为此添加了一个功能。还有一个已关闭的PR可用于保存MIDI通道信息,您可以挑选或支持:https://github.com/cuthbertLab/music21/pull/712 - Jacob Walls
1个回答

0

使用music21读取文件并将其写回的最简单代码是:

s = converter.parse('filein.mid')
s.write('midi', fp='fileout.mid')

在这两行之间,您可以对中间步骤进行任何操作:

for n in s[note.Note]:  # or in < v7: s.recurse().getElementsByClass('Note')
    n.midi += 2
s = s.augmentOrDiminish(2)  # twice as slow

这将按照您在问题中提出的要求完成所有操作,但是您可能会有一个后续问题:“在将MIDI加载到music21中并将其写出时,一些MIDI事件消失了。” 您是正确的。 Music21不是旨在成为完整功能的MIDI编辑器,可以将所有内容放回原处。 有一些这样的程序,但它们没有music21的音乐理论/符号知识。 您需要决定您愿意接受哪些权衡,或者子类化music21中的MIDI转换模块。


网页内容由stack overflow 提供, 点击上面的
可以查看英文原文,
原文链接