2024-07-15 11:46:59 +08:00

201 lines
4.5 KiB
C++

//
// Programmer: Craig Stuart Sapp <craig@ccrma.stanford.edu>
// Creation Date: Tue Oct 16 07:34:30 PDT 2012
// Last Modified: Mon Feb 9 20:40:50 PST 2015 Updated for C++11.
// Filename: tools/midicat.cpp
// URL: https://github.com/craigsapp/midifile/blob/master/tools/midicat.cpp
// Syntax: C++11
// vim: ts=3
//
// Description: Concatenate multiple MIDI files into single type-0 MIDI file.
//
#include "MidiFile.h"
#include "Options.h"
#include <iostream>
#include <vector>
using namespace std;
using namespace smf;
void checkOptions (Options& opts, int argc, char** argv);
void example (void);
void usage (const char* command);
void appendMidi (MidiFile& outfile, const char* filename,
double seconds, int initQ);
// user interface variables
Options options;
double seconds = 2.0; // used with -p option
int binaryQ = 1; // used with -a option
//////////////////////////////////////////////////////////////////////////
int main(int argc, char* argv[]) {
checkOptions(options, argc, argv);
MidiFile outfile;
outfile.joinTracks();
outfile.deltaTicks();
int i;
int initQ = 0;
for (i=1; i<=options.getArgCount(); i++) {
appendMidi(outfile, options.getArg(i).c_str(), seconds, initQ++);
}
// insert an end-of track Meta Event
int tpq = outfile.getTicksPerQuarterNote();
MidiEvent mfevent;
mfevent.tick = tpq;
mfevent.track = 0;
mfevent.resize(3);
mfevent[0] = 0xff;
mfevent[1] = 0x2f;
mfevent[2] = 0;
outfile.addEvent(mfevent);
if (binaryQ) {
outfile.write(cout);
} else {
cout << outfile;
}
return 0;
}
//////////////////////////////////////////////////////////////////////////
//////////////////////////////
//
// appendMidi --
//
void appendMidi(MidiFile& outfile, const char* filename,
double seconds, int initQ) {
MidiFile infile(filename);
infile.joinTracks();
infile.deltaTicks();
int i;
int tpq;
int count;
MidiEvent anevent;
if (initQ == 0) {
outfile.joinTracks();
count = infile.getEventCount(0);
// don't include end-of-track meta event
count--;
tpq = infile.getTicksPerQuarterNote();
outfile.setTicksPerQuarterNote(tpq);
for (i=0; i<count; i++) {
anevent = infile.getEvent(0,i);
// storing as a type-0 file, so remove track information
anevent.track = 0;
outfile.addEvent(anevent);
}
return;
}
// presuming constant tpq for different files.
tpq = outfile.getTicksPerQuarterNote();
if (seconds > 0.0) {
// insert a tempo marking of 120, and then a pause related to how
// long in second to wait until next MIDI file contents.
// micro-seconds per quarter note is last three bytes of meta message
// If quarter note is 120 bpm, that is 0.5 seconds or 500000 usec.
// In hex 500000 is 07 a1 20
// Tempo meta event: ff 51 03 07 a1 20
vector<uchar> bpm120;
bpm120.resize(3);
bpm120[0] = 0x07;
bpm120[1] = 0xa1;
bpm120[2] = 0x20;
outfile.addMetaEvent(0, 0, 0x51, bpm120);
infile.getEvent(0,0).tick = int(seconds * 2 * tpq + 0.5);
}
count = infile.getEventCount(0);
// don't include end-of-track meta event
count--;
for (i=0; i<count; i++) {
anevent = infile.getEvent(0,i);
anevent.track = 0;
outfile.addEvent(anevent);
}
}
//////////////////////////////
//
// checkOptions --
//
void checkOptions(Options& opts, int argc, char* argv[]) {
opts.define("p|pause=d:2.0", "Pause given number of secs after each file");
opts.define("a|ascii=b", "Display MIDI output as ASCII text");
opts.define("author=b", "author of program");
opts.define("version=b", "compilation info");
opts.define("example=b", "example usages");
opts.define("h|help=b", "short description");
opts.process(argc, argv);
// handle basic options:
if (opts.getBoolean("author")) {
cout << "Written by Craig Stuart Sapp, "
<< "craig@ccrma.stanford.edu, 16 October 2012" << endl;
exit(0);
} else if (opts.getBoolean("version")) {
cout << argv[0] << ", version: October 2012" << endl;
cout << "compiled: " << __DATE__ << endl;
exit(0);
} else if (opts.getBoolean("help")) {
usage(opts.getCommand().c_str());
exit(0);
} else if (opts.getBoolean("example")) {
example();
exit(0);
}
if (opts.getArgCount() <= 1) {
usage(opts.getCommand().c_str());
exit(1);
}
seconds = opts.getDouble("pause");
binaryQ = !opts.getBoolean("ascii");
}
//////////////////////////////
//
// example --
//
void example(void) {
// add examples here
}
//////////////////////////////
//
// usage --
//
void usage(const char* command) {
// add usages here
}