CoreMIDI отслеживает доступ к информации о времени подписи
Я разрабатываю ритмическую игру, основанную на MIDI-треке. MIDI-сообщения запускают высвобождение экранных элементов. Я загружаю данные MIDI из файла, а затем воспроизводю их с помощью MusicSequence и MusicPlayer.
Я понимаю, что MIDI-файлы содержат информацию о времени и ключевой подписи в виде мета-сообщений в начале файла. Однако я не нашел способа получить эту информацию ни из MusicPlayer, ни из MusicSequence.
Необходимая мне информация - это количество секунд, необходимое для воспроизведения квавера, вязания крючком и т. Д. Я ожидаю, что на это повлияют сигнатура времени и значение MusicPlayerPlayRateScalar.
Похоже, эту информацию можно найти в часах CoreAudio, но я не смог понять, как к ней обращаться для определенной музыкальной последовательности.
Есть ли эксперты CoreAudio, которые знают, как это сделать?
1 ответ
Вам нужно получить дорожку темпа миди-файла и затем пройти по нему, чтобы получить темп (ы).
Чтобы получить длину последовательности, вам нужно найти самую длинную дорожку:
(MusicTimeStamp)getSequenceLength:(MusicSequence)aSequence {
UInt32 tracks;
MusicTimeStamp len = 0.0f;
if (MusicSequenceGetTrackCount(sequence, &tracks) != noErr)
return len;
for (UInt32 i = 0; i < tracks; i++) {
MusicTrack track = NULL;
MusicTimeStamp trackLen = 0;
UInt32 trackLenLen = sizeof(trackLen);
MusicSequenceGetIndTrack(sequence, i, &track);
MusicTrackGetProperty(track, kSequenceTrackProperty_TrackLength, &trackLen, &trackLenLen);
if (len < trackLen)
len = trackLen;
}
return len;
}
// - получить дорожку темпа:
OSStatus result = noErr;
MusicTrack tempoTrack;
result = MusicSequenceGetTempoTrack(sequence, &tempoTrack);
if (noErr != result) {[self printErrorMessage: @"MusicSequenceGetTempoTrack" withStatus: result];}
MusicEventIterator iterator = NULL;
NewMusicEventIterator(tempoTrack, &iterator);
MusicTimeStamp timestamp = 0;
MusicEventType eventType = 0;
const void *eventData = NULL;
UInt32 eventDataSize = 0;
MusicEventIteratorGetEventInfo(iterator, ×tamp, &eventType, &eventData, &eventDataSize);