Real Time MIDI Processing
This tutorial builds on the setup from "Getting Started". You should be comfortable creating Max
for Live devices with v8
objects in them. Before we get into the details of debugging
with the Max Console and using the Live API, let's jump into building something.
We will alter MIDI in real time to change the pitch being played based on a simple set of rules. This means we can play a MIDI clip in Live or physically play some notes on MIDI controller hardware and JavaScript will change the pitches being played as they are occurring.
Here's the idea: The first time any pitch repeats between consecutive notes, play it one octave up. The second time a pitch repeats (in a row), play it one octave down. The third time the pitch repeats, repeat the pattern by playing the original octave again, and so on: normal, up, down, normal, up, down, etc. And whenever the pitch changes, the octave resets to normal. For example, if we started with this MIDI clip:
It will look like this after our JavaScript processes it:
Look closely and notice the notes are slightly delayed (check the left edge of the first note in the two screenshots). That's because real time processing adds delay, referred to as latency in this context. As discussed in limitations, JavaScript is relatively slow and will inherently add some latency. I also had the device open in the Max editor when I recorded the processed MIDI in the screenshot above. Actively editing a Live device with the Max editor adds a lot more latency than using the Max device directly inside Live. So keep that in mind: if you have too much latency and you are editing your Max patch, try saving and closing the Max editor, and test again directly in Live.
Real Time Note Events
MIDI note events have a pitch and velocity values (velocity represents how intensely the note is played), and they come
in two general flavors: "note on" events representing the start of a note, and "note off" events representing the end of
a note. If you've ever handled keyboard events in a web app, this is analogous to
keyDown
and keyUp
events. In a Live
MIDI clip, the left edge of a note in the piano roll is the "note on" event and the right edge is the "note off" event.
Our JavaScript code will see a separate "note on" and "note off" event for every note in the piano roll.
It's important to understand this because later we'll be working with notes in non-realtime and we can work with note objects that have a start time and duration. The real time situation is completely different: there is no start time because everything is happening now (from the code's perspective), and we can't know a duration because the "note off" event that determines duration hasn't happened yet (when we're processing "note on" events).
How do we know the difference between a "note on" and a "note off" event? By convention, when
velocity == 0
it's a "note off", and when
velocity > 0
, it's a "note on" event. Negative pitches and velocities are
invalid and unlikely to be encountered, so you can ignore them.
Intercepting MIDI Notes Events
Ok, enough theory. Let's figure out how to pass all MIDI data through our Live device while intercepting note events. A
fresh Max MIDI Effect device starts with a simple MIDI passthrough that connects a
midiin
object directly to a midiout
object:
To intercept the note events while continuing to pass through everything else, we need to split apart the raw MIDI data
stream into the different types of MIDI events. We do that by inserting a midiparse
object after the midiin
. We then immediate recombine all the separated MIDI events into
a single MIDI data stream with a midiformat
object that connects all the outlets from
midiparse
(with the exception of the rightmost) to its corresponding inlets, and
continue to use midiout
for the final output, like this:
Now is a good time to check that MIDI is in fact passing through the device. Do this:
- Save the device if you haven't already, for example:
octave-switcher.amxd
- Add a MIDI clip to the track containing your device
- Add some notes to the clip
- Add an instrument to the track if you haven't already
- Instrument selection tips: We will encounter bugs with stuck notes, so it's useful to choose an instrument that plays long sustained notes so the problem can be easily heard. To test things, we'll play clips with lots of short notes, so a fast attack works better than a slow drone/pad sound. A preset for Ableton's built-in Drift instrument called "Synthetic Xylophone" works well.
- Play the clip
You should hear the instrument playing the notes:
All good? If not, debug it before we go any further.
Back to the patch: The leftmost outlet of midiparse
outputs the MIDI note events we are
interested in. If you hover over it, you should see a tooltip:
This is telling us that outlet sends two-item lists containing pitch and velocity that represents "note on" and "note
off" events. Let's take a look: Add a message object (by dragging from the top toolbar or pressing 'm') and connect the
midiparse
leftmost outlet to the right inlet of the message object. Play the MIDI clip
again and you can see the note data, which is two numbers:
That's the pitch and velocity. This screenshot shows "60 100", which is a "note on" event with a pitch of 60 and a velocity of 100.
We intercept this data by routing it through the v8
object, like this:
That's the complete Max patch structure, but the data in not passing through the v8
object because we haven't written any code yet. So how do we pass that data from the v8
inlet to its outlet? Remember, the input is a two-number list containing the pitch and velocity. In
v8
, we define a handler for list input by defining a
list()
function, and access the arguments with the
arguments
array. If you open up the v8
help
file, the example script in there shows examples of handling various types of input. For our case, we want to do this:
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
outlet(0, pitch, velocity);
}
After we've read the pitch
and
velocity
, we can output them as a two item list with
outlet(0, pitch, velocity);
, where 0
is the outlet index (we can configure v8
with multiple outlets, but that's a topic for
later). Note: this is equivalent to outlet(0, [pitch, velocity]);
but it's
unnecessary and inefficient to create an array like that.
Now with the Max patch editor still open, you can open the Max Console and watch the JavaScript code log the pitch and velocity data when you play the MIDI clip again:
pitch=60, velocity=100 pitch=60, velocity=0 pitch=60, velocity=100 pitch=60, velocity=0 pitch=65, velocity=100 pitch=65, velocity=0 pitch=65, velocity=100
Detecting Note On Events and Repeated Pitches
As discussed above, "note on" events have a velocity higher than zero, which is easily checked in code:
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
if (velocity > 0) {
post("- note on\n");
} else {
post("- note off\n");
}
outlet(0, pitch, velocity);
}
To detect repeat pitches, we can define a top-level variable outside of the function to remember state between different
function calls. At the end of each call to our list()
handler, we'll store the
current pitch
in the previousPitch
top-level variable. Then, our logic can check the incoming current pitch
is the
same as previousPitch
to detect repeated pitches.
For the purpose of checking the pitch of consecutive notes, it doesn't make sense to check "note off" events, because they represent the end of an existing note instead of a new note. Therefore, we put the logic to check for repeating pitches inside the conditional block for "note on" events:
let previousPitch = null;
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
if (velocity > 0) {
post("- note on\n");
if (pitch == previousPitch) {
post(`- repeat pitch\n`);
}
previousPitch = pitch;
} else {
post("- note off\n");
}
outlet(0, pitch, velocity);
}
We're adding all the post()
calls so we can confirm things are working as
intended in the Max console logs. Does it look like what you expect?
pitch=60, velocity=100 - note on pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch pitch=60, velocity=0 - note off pitch=65, velocity=100 - note on pitch=65, velocity=0
Remembering State in an Object
Since we're using v8
, we can use all the powerful features of modern JavaScript. Before
we go any further, let's stop using a top-level variable and follow good practices for
encapsulation by storing our persistent state in
member variables of a class. We'll call the class OctaveSwitcher
.
We make class members private in JavaScript by prefixing them with a #
. We can
set the initial value for member variables by assigning them directly in the class body, so
#previousPitch = null;
will make every instance of
OctaveSwitcher
start out with a value of
null
for its #previousPitch
:
class OctaveSwitcher {
#previousPitch = null;
process(pitch, velocity) {
if (velocity > 0) {
post("- note on\n");
if (pitch == this.#previousPitch) {
post(`- repeat pitch\n`);
}
this.#previousPitch = pitch;
} else {
post("- note off\n");
}
}
}
const processor = new OctaveSwitcher();
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
processor.process(pitch, velocity);
outlet(0, pitch, velocity);
}
Cycling Octave States
Let's define some top-level constants to represent the normal pass-through, octave up, and octave down states that we'll cycle through on repeat pitches:
const STATE = Object.freeze({
PASSTHROUGH: 0,
UP: 1,
DOWN: 2,
});
And add another private member variable #state
to the
OctaveSwitcher
class to store the current state, which defaults to the normal
pass-through:
#state = STATE.PASSTHROUGH;
When a repeat pitch is detected, we want to cycle through the states. We can define a private getter function (a
function call that acts like you are accessing a member variable) in
OctaveSwitcher
to tell us the next state in the cycle:
get #nextState() {
if (this.#state == STATE.PASSTHROUGH) return STATE.UP;
if (this.#state == STATE.UP) return STATE.DOWN;
else return STATE.PASSTHROUGH;
}
This cyclical pattern can be expressed much more succinctly as:
get #nextState() {
return (this.#state + 1) % 3
}
but I think the more verbose version is much clearer about the intent (and it can recover if
#state
somehow gets an invalid value), so I'll keep using the verbose version
in this tutorial.
We also need to reset this.#state = STATE.PASSTHROUGH;
whenever the pitch does
not repeat.
All those changes together look like:
const STATE = Object.freeze({
PASSTHROUGH: 0,
UP: 1,
DOWN: 2,
});
class OctaveSwitcher {
#state = STATE.PASSTHROUGH;
#previousPitch = null;
process(pitch, velocity) {
if (velocity > 0) {
post("- note on\n");
if (pitch == this.#previousPitch) {
post(`- repeat pitch\n`);
this.#state = this.#nextState;
} else {
this.#state = STATE.PASSTHROUGH;
}
post(`- state: ${this.#state}\n`);
this.#previousPitch = pitch;
} else {
post("- note off\n");
}
}
get #nextState() {
if (this.#state == STATE.PASSTHROUGH) return STATE.UP;
if (this.#state == STATE.UP) return STATE.DOWN;
else return STATE.PASSTHROUGH;
}
}
const processor = new OctaveSwitcher();
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
processor.process(pitch, velocity);
outlet(0, pitch, velocity);
}
Continuing to test with the logs (remember: state 0 is passthrough, 1 is up, 2 is down, and states cycle on repeat pitches):
pitch=60, velocity=100 - note on - state: 0 pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch - state: 1 pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch - state: 2 pitch=60, velocity=0 - note off pitch=60, velocity=100 - note on - repeat pitch - state: 0 pitch=60, velocity=0 - note off pitch=65, velocity=100 - note on - state: 0 pitch=65, velocity=0
That looks promising. Still with me? Let's actually change the octave based on the
state
:
const STATE = Object.freeze({
PASSTHROUGH: 0,
UP: 1,
DOWN: 2,
});
class OctaveSwitcher {
#state = STATE.PASSTHROUGH;
#previousPitch = null;
process(pitch, velocity) {
if (velocity > 0) {
post("- note on\n");
if (pitch == this.#previousPitch) {
post(`- repeat pitch\n`);
this.#state = this.#nextState;
} else {
this.#state = STATE.PASSTHROUGH;
}
post(`- state: ${this.#state}\n`);
this.#previousPitch = pitch;
if (this.#state == STATE.UP) pitch += 12;
if (this.#state == STATE.DOWN) pitch -= 12;
} else {
post("- note off\n");
}
return pitch;
}
get #nextState() {
if (this.#state == STATE.PASSTHROUGH) return STATE.UP;
if (this.#state == STATE.UP) return STATE.DOWN;
else return STATE.PASSTHROUGH;
}
}
const processor = new OctaveSwitcher();
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
const p = processor.process(pitch, velocity);
post(`- output pitch: ${p == pitch ? "unchanged" : p}\n`);
outlet(0, p, velocity);
}
To help test the new logic, I also added a log line to show when a pitch is being changed before we send the value to the outlet:
post(`- output pitch: ${p == pitch ? "unchanged" : p}\n`);
Handling Note Off Events
Most of the repeat pitch detection and octave-switching logic is in place, but there is a major flaw. Earlier we were discussing how every note comes in a "note on" + "note off" event pair. A key aspect of "note off" events is they always have the same pitch as their corresponding "note on" event. Whenever a "note off" occurs, it ends whatever "note on" was playing at that pitch.
See the problem? We are only changing the pitch for "note on" events, so the notes can no longer end properly. You might have noticed something seems wrong in the logs with the latest code:
pitch=60, velocity=100 - note on - repeat pitch - state: 1 - output pitch: 72 pitch=60, velocity=0 - note off - output pitch: unchanged
The second note that was changed from pitch 60 to 72 should have had the corresponding note off also changed to 72, but as seen in the last log line here, it is unchanged.
We can investigate exactly what is happening by creating another MIDI track, setting its input to the track with our Max for Live device, and recording the MIDI output from our device. When you setup your track for recording, be sure to select "Post FX" in the input / "MIDI From" settings. If you select "Pre FX", it will bypass the Max for Live device. Don't forget to arm the track for recording:
The recorded notes should be the same duration as the original MIDI clip. In my case all original notes were short 16th notes. Some of the notes are playing for way too long. The long notes are the ones where we changed the pitch. You may even have stuck notes. If that happens, hit the stop button in Live's transport to send an "all notes off" message to everything.
To fix this, we need to keep track of how we are mapping "note on" pitches from their original pitch to the modified pitch, and apply the same mapping for "note off" events. A basic JavaScript object works as a map data structure:
#noteMap = new Map();
To keep track of whether the pitch is changing, it's convenient to introduce another variable for the incoming pitch that will not change:
process(pitch, velocity) {
const originalPitch = pitch;
Then, after we might have changed the pitch, we check if we actually changed the pitch for this note. If so, we then
keep track of how we mapped the original pitch to the modified pitch in our
#noteMap
data structure:
if (state == OCTAVE_UP) pitch += 12;
else if (state == OCTAVE_DOWN) pitch -= 12;
if (pitch != originalPitch) {
this.#noteMap.set(originalPitch, pitch);
}
Now we are ready to implement the "note off" behavior. If the pitch was modified for a "note on", make the same modification to the corresponding "note off". We don't want to keep applying the modification, so we also remove this entry from the map:
} else {
post("Note off\n");
if (this.#noteMap.has(pitch)) {
pitch = this.#noteMap.get(pitch);
this.#noteMap.delete(originalPitch);
}
}
Note that when we delete an entry from the #noteMap
, we need to delete the key for
originalPitch
because we changed the pitch
by
assigning it to the value in the note map.
Putting it all together:
const STATE = Object.freeze({
PASSTHROUGH: 0,
UP: 1,
DOWN: 2,
});
class OctaveSwitcher {
#state = STATE.PASSTHROUGH;
#previousPitch = null;
#noteMap = new Map();
process(pitch, velocity) {
const originalPitch = pitch;
if (velocity > 0) {
post("- note on\n");
if (pitch == this.#previousPitch) {
post(`- repeat pitch\n`);
this.#state = this.#nextState;
} else {
this.#state = STATE.PASSTHROUGH;
}
post(`- state: ${this.#state}\n`);
this.#previousPitch = pitch;
if (this.#state == STATE.UP) pitch += 12;
if (this.#state == STATE.DOWN) pitch -= 12;
if (pitch != originalPitch) {
this.#noteMap.set(originalPitch, pitch);
}
} else {
post("- note off\n");
if (this.#noteMap.has(pitch)) {
pitch = this.#noteMap.get(pitch);
this.#noteMap.delete(originalPitch);
}
}
return pitch;
}
get #nextState() {
if (this.#state == STATE.PASSTHROUGH) return STATE.UP;
if (this.#state == STATE.UP) return STATE.DOWN;
else return STATE.PASSTHROUGH;
}
}
const processor = new OctaveSwitcher();
function list(pitch, velocity) {
post(`pitch=${pitch}, velocity=${velocity}\n`);
const p = processor.process(pitch, velocity);
post(`- output pitch: ${p == pitch ? "unchanged" : p}\n`);
outlet(0, p, velocity);
}
Now the logs and MIDI output should be as intended:
pitch=60, velocity=100 - note on - repeat pitch - state: 1 - output pitch: 72 pitch=60, velocity=0 - note off - output pitch: 72
Wrapping Up
Before moving on, let's clean up by splitting out private functions for handling note on and note off. We'll need to
keep track of originalPitch
separately in the two new functions. Once we've
confirmed everything still works, we can remove most of the logging. Here's the final version:
const STATE = Object.freeze({
PASSTHROUGH: 0,
UP: 1,
DOWN: 2,
});
class OctaveSwitcher {
#state = STATE.PASSTHROUGH;
#previousPitch = null;
#noteMap = new Map();
process(pitch, velocity) {
if (velocity > 0) {
return this.#processNoteOn(pitch);
} else {
return this.#processNoteOff(pitch);
}
}
#processNoteOn(pitch) {
const originalPitch = pitch;
if (pitch == this.#previousPitch) {
this.#state = this.#nextState;
} else {
this.#state = STATE.PASSTHROUGH;
}
this.#previousPitch = pitch;
if (this.#state == STATE.UP) pitch += 12;
if (this.#state == STATE.DOWN) pitch -= 12;
if (pitch != originalPitch) {
this.#noteMap.set(originalPitch, pitch);
}
return pitch;
}
#processNoteOff(pitch) {
const originalPitch = pitch;
if (this.#noteMap.has(pitch)) {
pitch = this.#noteMap.get(pitch);
this.#noteMap.delete(originalPitch);
}
return pitch;
}
get #nextState() {
if (this.#state == STATE.PASSTHROUGH) return STATE.UP;
if (this.#state == STATE.UP) return STATE.DOWN;
else return STATE.PASSTHROUGH;
}
}
const processor = new OctaveSwitcher();
function list(pitch, velocity) {
const p = processor.process(pitch, velocity);
post(`pitch=${p == pitch ? p : `${pitch} → ${p}`}, velocity=${velocity}\n`);
outlet(0, p, velocity);
}
Here is the entire Max for Live device (including the JavaScript embedded inside), which you can copy and paste directly into an empty Max MIDI Device, in case you had any trouble following along:
----------begin_max5_patcher----------
1018.3ocwWs0ahaDE9Y3Wwod6CfJKx1vlaHZUp5ptuzknRh5CkUIClgjIxLi
0LiIY2H+Z+Azeh8WRmqfMABtJD0Gv1y4124lOGySMaDLk8HVD.mA+IznwSMa
zvPRSng6bifEnGSRQBiXAT7Cro2GzwxRheTZHu7DOoLjL4NB81q43Do0xQG0
MrCbbe803Xy0vtgvWbZPlYLgxruum2JyYTof7MrlSjR3RjonEFxAmyInT3mY
oy7ZQyWPnoXowUiVSjkK2jpkj7qYXqOFDrxelSRwq.gxn3xQqloK83xOkxBI
LpPBiu77K+HLDFM8dUJn6bNF+MbqmlPA3hyGO9xO86it5W+zYPXGMoqtP6T5
m9kQ+wmOChUOWzdvD5DpIqCiRjnk3wOPT4ULGL14cBIRhUXXvpaIyNvvNiiW
RX4hKzJoDilmlZ4PYR7ugxzzvO.pmZYwBfLNKAKDsxz5zAVhSYID4WaaQD.x
bnkmH7iP3JF.vwxbNEj2QDcemyNeVAzHp0ZsGXkr.voBbMza97MTTeqv5maE
.uMsU.FmbKghR8guQFWXZCjLKmgqvtT9pTfY45S11STU0drlxNBpp5r8BjKh
fsA+V82JF0a0qtnsUT3GFBQwC1i359KuBu2ovlYjuaX0j2yxEt9mtBrrUEI6
.OuhspB6CnsWDWUsqWUrZP5cn6PtV21k7YWYdCe+VkuWwY2L3lgUyFvUiuZF
XJaCqaQZU4smsVUJ0Zz1aUe4cPMZ.pnhtD6TxzTVg4lMgJO13z1rsqfv3tQC
UG63lRLOmlHILJjRDxcLovYNcAyaxt6Z1hwOxXJacig0vu+oLc3Yqa+jxJmA
2nnoOV.+ye82f5PwMEqMgRC+iESlPuwZR6z8VpMMYUQSEvto46dNuhSJ5Vy1
hPOE7ho3Yk1ezH.kKYOn20YnZHVzzsnPnxbytlYl+eMRJ4jo4R6d1x6MxPbE
7RL+ZkeL0tWIzYol9qJi95VMufLinPRfegMzwlMz80KZgS68RanO8MeC8I6b
Ccm0+HTYoakeD9xgM0MmwWfj6M2EEdh91Iu3m2DEd.ydGu0rW7txduU4GEP6
O4zOxzec7KlbheyasBOrgNgt2Huz6U6Lt+e7id0MEUZHLNXPJgt4+JvDgZ5U
SUBVNOwaM27AXcXNCKjpMn5UFkjomUlsVL9ufSzdvQ+5lVnWMPw0An3C.P8p
CP8N..0uN.0+..zGpCPe3..zQ0Anidk.0qFc2Ffdss2Q0AoSOD.EVmPJ9YHY
GTfxxVh4BmzFPTSOumw0GOsi4HgZOZFmEn+mNd46Ynf3pQmR0bybtcD2iG0O
vpJaFlSyItuvx74UAlIy5wghLjMRLCvaVz7eAbGjZn.
-----------end_max5_patcher-----------
Next steps
The next tutorial, "The Max Console", shows how to better use
post()
and the Max console to inspect and debug JavaScript programs running in
Max for Live. Tired of ending every post()
call with
"\n"
? Read on!.