package org.herac.tuxguitar.io.midi;
import java.io.DataOutputStream;
import java.io.IOException;
import java.io.OutputStream;
import org.herac.tuxguitar.io.midi.base.MidiEvent;
import org.herac.tuxguitar.io.midi.base.MidiMessage;
import org.herac.tuxguitar.io.midi.base.MidiSequence;
import org.herac.tuxguitar.io.midi.base.MidiTrack;
public class MidiFileWriter implements MidiFileHeader{
public void write(MidiSequence sequence,int type,OutputStream stream)throws IOException{
DataOutputStream out = new DataOutputStream(stream);
out.writeInt(HEADER_MAGIC);
out.writeInt(HEADER_LENGTH);
out.writeShort(type);
out.writeShort(sequence.countTracks());
out.writeShort( (sequence.getDivisionType() == MidiSequence.PPQ)?(sequence.getResolution() & 0x7fff):0 );
for (int i = 0; i < sequence.countTracks(); i++){
writeTrack(sequence.getTrack(i),out);
}
out.flush();
out.close();
}
private static int writeTrack(MidiTrack track,DataOutputStream out)throws IOException{
int length = 0;
if (out != null){
out.writeInt(TRACK_MAGIC);
}
if (out != null){
out.writeInt( writeTrack(track, null) );
}
MidiEvent previous = null;
for (int i = 0; i < track.size(); i++){
MidiEvent event = track.get(i);
length += writeEvent(event,previous,out);
previous = event;
}
return length;
}
private static int writeEvent(MidiEvent event,MidiEvent previous,OutputStream out)throws IOException{
int length = writeVariableLengthQuantity(((previous != null)?(event.getTick() - previous.getTick()):0), out);
MidiMessage message = event.getMessage();
if(message.getType() == MidiMessage.TYPE_SHORT){
length += writeShortMessage(message,out);
}
else if(message.getType() == MidiMessage.TYPE_META){
length += writeMetaMessage(message,out);
}
return length;
}
private static int writeShortMessage(MidiMessage message,OutputStream out)throws IOException{
byte[] data = message.getData();
int length = data.length;
if (out != null){
out.write(message.getData(),0,length);
}
return length;
}
private static int writeMetaMessage(MidiMessage message,OutputStream out)throws IOException{
int length = 0;
byte[] data = message.getData();
if (out != null){
out.write(0xFF);
out.write(message.getCommand());
}
length += 2;
length += writeVariableLengthQuantity(data.length,out);
if (out != null){
out.write(data);
}
length += data.length;
return length;
}
private static int writeVariableLengthQuantity(long value, OutputStream out)throws IOException{
boolean started = false;
int length = 0;
int data = (int) ((value >> 21) & 0x7f);
if (data != 0){
if (out != null){
out.write(data | 0x80);
}
length++;
started = true;
}
data = (int) ((value >> 14) & 0x7f);
if (data != 0 || started){
if (out != null){
out.write(data | 0x80);
}
length++;
started = true;
}
data = (int) ((value >> 7) & 0x7f);
if (data != 0 || started){
if (out != null){
out.write(data | 0x80);
}
length++;
}
data = (int) (value & 0x7f);
if (out != null){
out.write(data);
}
length++;
return length;
}
}