如何从iOS中的MusicPlayer(对于Midi文件)获取持续时间?

时间:2019-07-17 06:56:37

标签: ios swift midi coremidi

我需要确定midi文件的持续时间和结束事件。我正在使用下面的代码播放midi文件。我尝试过,但没有发现任何东西。预先感谢

var s: MusicSequence?
NewMusicSequence(&s)

let midiFilePath = Bundle.main.path(forResource: "CCL-20180308-A-04", ofType: "mid")
let midiFileURL = URL(fileURLWithPath: midiFilePath ?? "")

MusicSequenceFileLoad(s!, midiFileURL as CFURL, MusicSequenceFileTypeID(rawValue: 0)!, [])

var p: MusicPlayer?
NewMusicPlayer(&p)

MusicPlayerSetSequence(p!, s)
MusicPlayerPreroll(p!)
MusicPlayerStart(p!)

usleep(3 * 100 * 100)
var now: MusicTimeStamp = 0
MusicPlayerGetTime(p!, &now)

C11 7.21.5.3

1 个答案:

答案 0 :(得分:1)

这将起作用:

var s: MusicSequence!
NewMusicSequence(&s)

let midiFileURL = Bundle.main.url(forResource: "CCL-20180308-A-04", withExtension: "mid")!

MusicSequenceFileLoad(s!, midiFileURL as CFURL, .midiType, [])

var p: MusicPlayer!
NewMusicPlayer(&p)

MusicPlayerSetSequence(p, s)
MusicPlayerPreroll(p)
MusicPlayerStart(p)

var numTracks: UInt32 = 0
MusicSequenceGetTrackCount(s, &numTracks)
let length = (0..<numTracks).map { (index: UInt32) -> (MusicTimeStamp) in
    var track: MusicTrack?
    MusicSequenceGetIndTrack(s, index, &track)
    var size = UInt32(MemoryLayout<MusicTimeStamp>.size)
    var scratchLength = MusicTimeStamp(0)
    MusicTrackGetProperty(track!, kSequenceTrackProperty_TrackLength, &scratchLength, &size)
    return scratchLength
}.max() ?? 0
var lengthInSeconds = Float64(0)
MusicSequenceGetSecondsForBeats(s, length, &lengthInSeconds)

self.timer = Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true, block: { (t) in
    var now: MusicTimeStamp = 0
    MusicPlayerGetTime(p, &now)
    var nowInSeconds = Float64(0)
    MusicSequenceGetSecondsForBeats(s, now, &nowInSeconds)
    print("\(nowInSeconds) / \(lengthInSeconds)")
})

您缺少的重要部分是通过找到最长轨道的长度来获得总序列长度。您可以使用MusicTrackGetProperty()属性的kSequenceTrackProperty_TrackLength获得曲目的长度。

就其价值而言,CoreMIDI够粗糙的,尤其是在Swift中,我认为使用更高级别的API是值得的。签出AVMIDIPlayer,它是AVFoundation的一部分。如果您需要更高级的东西,可以查看MIKMIDI,这是一个开放源MIDI库,它建立在Core MIDI上,但增加了许多附加功能,并且非常易于使用。 (免责声明:我是MIKMIDI的原始作者和维护者。)使用MIKMIDI,您可以这样做:

let midiFileURL = Bundle.main.url(forResource: "CCL-20180308-A-04", withExtension: "mid")!
let sequence = try! MIKMIDISequence(fileAt: midiFileURL)
let sequencer = MIKMIDISequencer(sequence: sequence)
sequencer.startPlayback()

self.timer = Timer.scheduledTimer(withTimeInterval: 0.1, repeats: true, block: { (t) in
    let now = sequencer.timeInSeconds(forMusicTimeStamp: sequencer.currentTimeStamp, options: [])
    let length = sequence.durationInSeconds
    print("\(now) / \(length)")
})

简单一点!如果您想进行录音,更复杂的合成,将MIDI路由到/从外部设备路由等,事情将变得更加有趣。