296 lines
11 KiB
C#
296 lines
11 KiB
C#
|
using System;
|
||
|
using System.IO;
|
||
|
using System.Text;
|
||
|
using System.Collections.Generic;
|
||
|
using NAudio.Utils;
|
||
|
|
||
|
namespace NAudio.Midi
|
||
|
{
|
||
|
/// <summary>
|
||
|
/// Class able to read a MIDI file
|
||
|
/// </summary>
|
||
|
public class MidiFile
|
||
|
{
|
||
|
private readonly MidiEventCollection events;
|
||
|
private readonly ushort fileFormat;
|
||
|
//private ushort tracks;
|
||
|
private readonly ushort deltaTicksPerQuarterNote;
|
||
|
private readonly bool strictChecking;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Opens a MIDI file for reading
|
||
|
/// </summary>
|
||
|
/// <param name="filename">Name of MIDI file</param>
|
||
|
public MidiFile(string filename)
|
||
|
: this(filename,true)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// MIDI File format
|
||
|
/// </summary>
|
||
|
public int FileFormat => fileFormat;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Opens a MIDI file for reading
|
||
|
/// </summary>
|
||
|
/// <param name="filename">Name of MIDI file</param>
|
||
|
/// <param name="strictChecking">If true will error on non-paired note events</param>
|
||
|
public MidiFile(string filename, bool strictChecking) :
|
||
|
this(File.OpenRead(filename), strictChecking, true)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Opens a MIDI file stream for reading
|
||
|
/// </summary>
|
||
|
/// <param name="inputStream">The input stream containing a MIDI file</param>
|
||
|
/// <param name="strictChecking">If true will error on non-paired note events</param>
|
||
|
public MidiFile(Stream inputStream, bool strictChecking) :
|
||
|
this(inputStream, strictChecking, false)
|
||
|
{
|
||
|
}
|
||
|
|
||
|
private MidiFile(Stream inputStream, bool strictChecking, bool ownInputStream)
|
||
|
{
|
||
|
this.strictChecking = strictChecking;
|
||
|
|
||
|
var br = new BinaryReader(inputStream);
|
||
|
try
|
||
|
{
|
||
|
string chunkHeader = Encoding.UTF8.GetString(br.ReadBytes(4));
|
||
|
if(chunkHeader != "MThd")
|
||
|
{
|
||
|
throw new FormatException("Not a MIDI file - header chunk missing");
|
||
|
}
|
||
|
uint chunkSize = SwapUInt32(br.ReadUInt32());
|
||
|
|
||
|
if(chunkSize != 6)
|
||
|
{
|
||
|
throw new FormatException("Unexpected header chunk length");
|
||
|
}
|
||
|
// 0 = single track, 1 = multi-track synchronous, 2 = multi-track asynchronous
|
||
|
fileFormat = SwapUInt16(br.ReadUInt16());
|
||
|
int tracks = SwapUInt16(br.ReadUInt16());
|
||
|
deltaTicksPerQuarterNote = SwapUInt16(br.ReadUInt16());
|
||
|
|
||
|
events = new MidiEventCollection((fileFormat == 0) ? 0 : 1, deltaTicksPerQuarterNote);
|
||
|
for (int n = 0; n < tracks; n++)
|
||
|
{
|
||
|
events.AddTrack();
|
||
|
}
|
||
|
|
||
|
long absoluteTime = 0;
|
||
|
|
||
|
for(int track = 0; track < tracks; track++)
|
||
|
{
|
||
|
if(fileFormat == 1)
|
||
|
{
|
||
|
absoluteTime = 0;
|
||
|
}
|
||
|
chunkHeader = Encoding.UTF8.GetString(br.ReadBytes(4));
|
||
|
if(chunkHeader != "MTrk")
|
||
|
{
|
||
|
throw new FormatException("Invalid chunk header");
|
||
|
}
|
||
|
chunkSize = SwapUInt32(br.ReadUInt32());
|
||
|
|
||
|
long startPos = br.BaseStream.Position;
|
||
|
MidiEvent me = null;
|
||
|
var outstandingNoteOns = new List<NoteOnEvent>();
|
||
|
while(br.BaseStream.Position < startPos + chunkSize)
|
||
|
{
|
||
|
try
|
||
|
{
|
||
|
me = MidiEvent.ReadNextEvent(br, me);
|
||
|
}
|
||
|
catch (InvalidDataException)
|
||
|
{
|
||
|
if (strictChecking) throw;
|
||
|
continue;
|
||
|
}
|
||
|
catch (FormatException)
|
||
|
{
|
||
|
if (strictChecking) throw;
|
||
|
continue;
|
||
|
}
|
||
|
|
||
|
absoluteTime += me.DeltaTime;
|
||
|
me.AbsoluteTime = absoluteTime;
|
||
|
events[track].Add(me);
|
||
|
if (me.CommandCode == MidiCommandCode.NoteOn)
|
||
|
{
|
||
|
var ne = (NoteEvent) me;
|
||
|
if(ne.Velocity > 0)
|
||
|
{
|
||
|
outstandingNoteOns.Add((NoteOnEvent) ne);
|
||
|
}
|
||
|
else
|
||
|
{
|
||
|
// don't remove the note offs, even though
|
||
|
// they are annoying
|
||
|
// events[track].Remove(me);
|
||
|
FindNoteOn(ne,outstandingNoteOns);
|
||
|
}
|
||
|
}
|
||
|
else if(me.CommandCode == MidiCommandCode.NoteOff)
|
||
|
{
|
||
|
FindNoteOn((NoteEvent) me,outstandingNoteOns);
|
||
|
}
|
||
|
else if(me.CommandCode == MidiCommandCode.MetaEvent)
|
||
|
{
|
||
|
MetaEvent metaEvent = (MetaEvent) me;
|
||
|
if(metaEvent.MetaEventType == MetaEventType.EndTrack)
|
||
|
{
|
||
|
//break;
|
||
|
// some dodgy MIDI files have an event after end track
|
||
|
if (strictChecking)
|
||
|
{
|
||
|
if (br.BaseStream.Position < startPos + chunkSize)
|
||
|
{
|
||
|
throw new FormatException(
|
||
|
$"End Track event was not the last MIDI event on track {track}");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
if(outstandingNoteOns.Count > 0)
|
||
|
{
|
||
|
if (strictChecking)
|
||
|
{
|
||
|
throw new FormatException(
|
||
|
$"Note ons without note offs {outstandingNoteOns.Count} (file format {fileFormat})");
|
||
|
}
|
||
|
}
|
||
|
if(br.BaseStream.Position != startPos + chunkSize)
|
||
|
{
|
||
|
throw new FormatException($"Read too far {chunkSize}+{startPos}!={br.BaseStream.Position}");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
finally
|
||
|
{
|
||
|
if (ownInputStream)
|
||
|
{
|
||
|
br.Dispose();
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// The collection of events in this MIDI file
|
||
|
/// </summary>
|
||
|
public MidiEventCollection Events => events;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Number of tracks in this MIDI file
|
||
|
/// </summary>
|
||
|
public int Tracks => events.Tracks;
|
||
|
|
||
|
/// <summary>
|
||
|
/// Delta Ticks Per Quarter Note
|
||
|
/// </summary>
|
||
|
public int DeltaTicksPerQuarterNote => deltaTicksPerQuarterNote;
|
||
|
|
||
|
private void FindNoteOn(NoteEvent offEvent, List<NoteOnEvent> outstandingNoteOns)
|
||
|
{
|
||
|
bool found = false;
|
||
|
foreach(NoteOnEvent noteOnEvent in outstandingNoteOns)
|
||
|
{
|
||
|
if ((noteOnEvent.Channel == offEvent.Channel) && (noteOnEvent.NoteNumber == offEvent.NoteNumber))
|
||
|
{
|
||
|
noteOnEvent.OffEvent = offEvent;
|
||
|
outstandingNoteOns.Remove(noteOnEvent);
|
||
|
found = true;
|
||
|
break;
|
||
|
}
|
||
|
}
|
||
|
if(!found)
|
||
|
{
|
||
|
if (strictChecking)
|
||
|
{
|
||
|
throw new FormatException($"Got an off without an on {offEvent}");
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
|
||
|
private static uint SwapUInt32(uint i)
|
||
|
{
|
||
|
return ((i & 0xFF000000) >> 24) | ((i & 0x00FF0000) >> 8) | ((i & 0x0000FF00) << 8) | ((i & 0x000000FF) << 24);
|
||
|
}
|
||
|
|
||
|
private static ushort SwapUInt16(ushort i)
|
||
|
{
|
||
|
return (ushort) (((i & 0xFF00) >> 8) | ((i & 0x00FF) << 8));
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Describes the MIDI file
|
||
|
/// </summary>
|
||
|
/// <returns>A string describing the MIDI file and its events</returns>
|
||
|
public override string ToString()
|
||
|
{
|
||
|
var sb = new StringBuilder();
|
||
|
sb.AppendFormat("Format {0}, Tracks {1}, Delta Ticks Per Quarter Note {2}\r\n",
|
||
|
fileFormat,Tracks,deltaTicksPerQuarterNote);
|
||
|
for (var n = 0; n < Tracks; n++)
|
||
|
{
|
||
|
foreach (var midiEvent in events[n])
|
||
|
{
|
||
|
sb.AppendFormat("{0}\r\n", midiEvent);
|
||
|
}
|
||
|
}
|
||
|
return sb.ToString();
|
||
|
}
|
||
|
|
||
|
/// <summary>
|
||
|
/// Exports a MIDI file
|
||
|
/// </summary>
|
||
|
/// <param name="filename">Filename to export to</param>
|
||
|
/// <param name="events">Events to export</param>
|
||
|
public static void Export(string filename, MidiEventCollection events)
|
||
|
{
|
||
|
if (events.MidiFileType == 0 && events.Tracks > 1)
|
||
|
{
|
||
|
throw new ArgumentException("Can't export more than one track to a type 0 file");
|
||
|
}
|
||
|
using (var writer = new BinaryWriter(File.Create(filename)))
|
||
|
{
|
||
|
writer.Write(Encoding.UTF8.GetBytes("MThd"));
|
||
|
writer.Write(SwapUInt32(6)); // chunk size
|
||
|
writer.Write(SwapUInt16((ushort)events.MidiFileType));
|
||
|
writer.Write(SwapUInt16((ushort)events.Tracks));
|
||
|
writer.Write(SwapUInt16((ushort)events.DeltaTicksPerQuarterNote));
|
||
|
|
||
|
for (int track = 0; track < events.Tracks; track++ )
|
||
|
{
|
||
|
IList<MidiEvent> eventList = events[track];
|
||
|
|
||
|
writer.Write(Encoding.UTF8.GetBytes("MTrk"));
|
||
|
long trackSizePosition = writer.BaseStream.Position;
|
||
|
writer.Write(SwapUInt32(0));
|
||
|
|
||
|
long absoluteTime = events.StartAbsoluteTime;
|
||
|
|
||
|
// use a stable sort to preserve ordering of MIDI events whose
|
||
|
// absolute times are the same
|
||
|
MergeSort.Sort(eventList, new MidiEventComparer());
|
||
|
if (eventList.Count > 0)
|
||
|
{
|
||
|
System.Diagnostics.Debug.Assert(MidiEvent.IsEndTrack(eventList[eventList.Count - 1]), "Exporting a track with a missing end track");
|
||
|
}
|
||
|
foreach (var midiEvent in eventList)
|
||
|
{
|
||
|
midiEvent.Export(ref absoluteTime, writer);
|
||
|
}
|
||
|
|
||
|
uint trackChunkLength = (uint)(writer.BaseStream.Position - trackSizePosition) - 4;
|
||
|
writer.BaseStream.Position = trackSizePosition;
|
||
|
writer.Write(SwapUInt32(trackChunkLength));
|
||
|
writer.BaseStream.Position += trackChunkLength;
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|
||
|
}
|