package cas.cs4tb3.mellowd;
import cas.cs4tb3.mellowd.midi.MidiNoteMessageSource;
import javax.sound.midi.InvalidMidiDataException;
import javax.sound.midi.ShortMessage;
import java.util.Collection;
import java.util.Collections;
import java.util.List;
Pitch is a key building block in the Mellow D compiler. It specifies the value of the played note. MIDI restricts the note value to 7 bits, hence we have 0-127 to work with. 0 corresponds to the musical note c with each consecutive number adding a semi-tone to the note value.
public final class Pitch implements MidiNoteMessageSource {
The NOTE_NAMES
array maps a midi num in the lowest octave to a note name. Here
it is clear to see the mapping from number to note for a total of 12 semi-tones in
an octave.
private static String[] NOTE_NAMES = new String[] {
"c", "c#", "d", "d#", "e", "f", "f#", "g", "g#", "a", "a#", "b"
};
A pitch is just a MIDI number (a int) wrapped with a bunch of utility methods. We can represent every possible MIDI pitch with 128 instances. Most songs would easily hit this number of pitch instantiations so we can create them all at once and share them.
private static Pitch[] ALL_PITCHES = new Pitch[128];
static {
for (int i = 0; i < 128; i++) ALL_PITCHES[i] = new Pitch(i);
}
To get started a static reference by name to a whole tone pitch is retrieved. From here the caller would chain calls to the various wrapper methods to obtain the desired pitch.
public static final Pitch C = ALL_PITCHES[0];
public static final Pitch D = ALL_PITCHES[2];
public static final Pitch E = ALL_PITCHES[4];
public static final Pitch F = ALL_PITCHES[5];
public static final Pitch G = ALL_PITCHES[7];
public static final Pitch A = ALL_PITCHES[9];
public static final Pitch B = ALL_PITCHES[11];
The REST
is a special instance that will be check for reference equality to insert
silence rather than sound. It is described in a Mellow D source file with a *
.
public static final Pitch REST = new Pitch(0);
This is the equivalent of a constructor. It retrieves the correct pitch instance for the given MIDI note value. If the given value is negative or exceeds 127 it is pulled into the range on the end that it exceeded.
public static Pitch getPitch(int midiNum) {
return ALL_PITCHES[Math.max(0, Math.min(midiNum, 127))];
}
private int midiNum;
private Pitch(int midiNum) {
this.midiNum = midiNum;
}
Sharp, or #
in a Mellow D source file, increases the note value by 1 semi-tone.
public Pitch sharp() {
return getPitch(midiNum + 1);
}
Flat, or $
in a Mellow D source file, decreases the note value by 1 semi-tone.
public Pitch flat() {
return getPitch(midiNum - 1);
}
Each octave shift is described in a Mellow D source as +
or -
a [0-9]+
.
As each octave contains 12 semi-tones, adding 12 to the MIDI note value increase the octave
by 1.
The shift up and shift down methods shift the octave in the appropriate direction ignoring
the sign of the amt
.
public Pitch shiftOctaveUp(int amt) {
return getPitch(midiNum + (12 * Math.abs(amt)));
}
public Pitch shiftOctaveDown(int amt) {
return getPitch(midiNum - (12 * Math.abs(amt)));
}
The plain shift octave uses the sign of the amt
to determine the direction for the shift.
@Override
public Pitch shiftOctave(int amt) {
return getPitch(midiNum + (12 * amt));
}
Using java’s integer division logic we can divide by 12 (the semi-tones per octave) and round down all at once to get the octave a note is in.
public int getOctave() {
return midiNum / 12;
}
In order to get the note value we can take the remainder of the MIDI note number when divided
by 12 (midiNum % 12
). This gives the note number in octave 0. The we can just preform an
octave shift up to the desired octave (octave * 12
).
public Pitch inOctave(int octave) {
return getPitch((octave * 12) + (midiNum % 12));
}
There are various common intervals that the Mellow D compiler heavily makes use of in constructing chords.
majorThird: 4 semi-tones above the root
public Pitch majorThird() {
return getPitch(midiNum + 4);
}
minorThird: 3 semi-tones above the root
public Pitch minorThird() {
return getPitch(midiNum + 3);
}
perfectFifth: 7 semi-tones above the root
public Pitch perfectFifth() {
return getPitch(midiNum + 7);
}
augmentedFifth: 8 semi-tones above the root
public Pitch augmentedFifth() {
return getPitch(midiNum + 8);
}
diminishedFifth: 6 semi-tones above the root
public Pitch diminishedFifth() {
return getPitch(midiNum + 6);
}
diminishedFifth: 9 semi-tones above the root
public Pitch diminishedSeventh() {
return getPitch(midiNum + 9);
}
minorSeventh: 10 semi-tones above the root
public Pitch minorSeventh() {
return getPitch(midiNum + 10);
}
majorSeventh: 11 semi-tones above the root
public Pitch majorSeventh() {
return getPitch(midiNum + 11);
}
getMidiNum
resolves the pitch to its MIDI note number
public int getMidiNum() {
return midiNum;
}
Pitch’s toString
implementation concatenates the MIDI note value with ‘:’, the
name of the note and the octave for a nice pitch description.
@Override
public String toString() {
return midiNum + ":" + NOTE_NAMES[midiNum%12] + Integer.toString(midiNum/12);
}
Implement the MidiNoteMessageSource methods. They just put a
new short message into a list for compatibility reasons. The only special case is the REST
which
results in no messages being generated.
@Override
public Collection<ShortMessage> noteOn(int channel, int velocity) throws InvalidMidiDataException {
if (this == REST)
return Collections.emptyList();
return Collections.singletonList(new ShortMessage(ShortMessage.NOTE_ON, channel, midiNum, velocity));
}
@Override
public List<ShortMessage> noteOff(int channel, int velocity) throws InvalidMidiDataException {
if (this == REST)
return Collections.emptyList();
return Collections.singletonList(new ShortMessage(ShortMessage.NOTE_OFF, channel, midiNum, velocity));
}
}