Generating MIDI Clips
This is a follow up to "The Live API" tutorial. In this tutorial, we'll use the Live API to generate MIDI clips from scratch with JavaScript code.
The plan is to generate MIDI clips for controlling drum racks using prime numbers to make interesting rhythmic patterns. The idea was inspired by "The Rhythm of The Primes" Youtube video.
Once again we'll start with the basic setup from "Getting Started" with a Max MIDI
Effect device containing a v8
or v8.codebox
object. I
won't explicitly use our log()
function here, but please use it if you need to inspect
anything or do some debugging.
Creating a Clip
Although we could create a clip by hand and use the Live API to fill it with notes, let's figure out how to create clips with the API too. Then we'll be equipped to generate a lot of different clips across multiple tracks automatically.
Live's tracks and clips come in two types: MIDI and audio. MIDI tracks can only contain MIDI clips, and audio tracks can only contain audio clips. The Live API handles the two types of clips slightly differently. We will be working with MIDI clips in this tutorial and generating them into the first track, so make sure the first track in your Live Set is a MIDI track, otherwise the code below will not work.
In the Live API, a ClipSlot is the object you use to manage clips in Session view. They're the individual cells in the Session view's grid of clips. We can access any clip slot by its track index and clip slot index, where clip slot indexes correspond to the scene index: the top row of Session view are the clip slots with index 0, the second row is index 1, and so on.
We can get a LiveAPI object for the first clip slot of the first track like this:
const clipSlot = new LiveAPI("live_set tracks 0 clip_slots 0");
And we can create a clip in a clip slot like this:
clipSlot.call("create_clip", 4);
The create_clip
function takes a parameter for the clip length in beats. We used 4
here to create a clip that's four
beats in length (one measure in 4/4 time).
You can try running those lines together:
const clipSlot = new LiveAPI("live_set tracks 0 clip_slots 0");
clipSlot.call("create_clip", 4);
If the clip slot is empty, it creates a clip. If there's already a clip, it prints an error to the Max console:
This clip slot already has a clip: 'create_clip 4'
Let's avoid the error by only creating a clip if the clip slot is empty. We can check if the clip slot is empty like this:
const clipSlot = new LiveAPI("live_set tracks 0 clip_slots 0");
if(clipSlot.get("has_clip") == false) {
clipSlot.call("create_clip", 4);
}
You might want to write if(!clipSlot.get("has_clip")) { ... }
, but an idiosyncrasy/bug of Max for Live causes
!clipSlot.get("has_clip")
to always be false, so don't check its value that way! Don't use === false
to do a strict comparison either.
Clearing a Clip
For the purpose of this tutorial, we want to start with a "clean slate" blank clip and re-generate all the notes every time we run our script. So, if the clip already exists let's delete all the notes inside of it. Alternately, we could delete and recreate the clip, but if we are looping the clip while coding and trying out different ideas, deleting the clip will cause playback of that clip slot to stop. By reusing the clip, we get something akin to live coding and can automatically hear the results in a looping clip as we save changes to our script. It's a nice workflow!
To clear all the notes from a clip, first we need to create a
Clip object with the Live API. A Clip
is a child of
ClipSlot
so, as we learned in "The Live API", we access the clip slot's clip by appending
"clip"
to the ClipSlot
path, so for example, here's the Clip
in the first clip slot
of the first track:
const clip = new LiveAPI("live_set tracks 0 clip_slots 0 clip");
In our script, we already have the ClipSlot
assigned to our clipSlot
variable, so we
can do this:
const clip = new LiveAPI(`${clipSlot.unquotedpath} clip`);
clipSlot.path
evaluates to the string "\"live_set tracks 0 clip_slots 0\""
with quote characters at the beginning and end of the string. You need to use clipSlot.unquotedpath
to make valid Live API paths.
To delete all the notes, we call the
remove_notes_extended
function on the
clip. It takes four parameters: from_pitch
, pitch_span
, from_time
, and time_span
. The from_*
parameters are
the starting pitch and starting time beats. We want to start from 0
in both cases. The *_span
parameters specify the
relative end values by adding to the start. So for example from_pitch=60, pitch_span=12
would cover pitches 60-71 (the
end value of 72 in not inclusive in the range).
MIDI pitches range from 0, 127, so from_pitch=0, pitch_span=128
will delete all pitches. For the end time, we can ask
the clip for its length with clip.get("length")
:
const clipSlot = new LiveAPI("live_set tracks 0 clip_slots 0");
if(clipSlot.get("has_clip") == false) {
clipSlot.call("create_clip", 4);
}
const clip = new LiveAPI(`${clipSlot.unquotedpath} clip`);
clip.call("remove_notes_extended", 0, 128, 0, clip.get("length"));
If you manually add some notes to your MIDI clip, re-running this script should now delete them all. If you are trying this code in a real music project (which is not recommended when experimenting with the Live API), be careful you don't clear a clip you care about!
Why is the function called "remove_notes_extended" and not "remove_notes"? In 2018, the MIDI standard received a major upgrade called MPE (MIDI polyphonic expression) that supports more expressive MIDI controller hardware where every note can be varied simultaneously and independently. MPE adds a lot of complexity to the MIDI standard, and Ableton needed to break backward compatibility to support it. They handled this by keeping the existing pre-MPE function named "remove_notes" for older Max devices running in Live version 10 and earlier where MPE is not supported.
Starting in Live 11 where MPE is supported, all Live API calls to remove notes from clips need to use "remove_notes_extended" to work properly . A similar situation exists with other clip functions such as "get_notes_extended". The older pre-MPE functions are no longer documented, and if you somehow accidentally use them (maybe by copy and pasting old code off the Internet), Live will warn you and tell you to use the newer functions. You generally don't need to think about it, but do pay attention to those warnings if you ever see them. You can try changing the clip.call(...)
to use "remove_notes" to see what the warning looks like.
Before we move on, let's convert what we've done so far into a reusable class that works on any clip slot.
class ClipSlot {
#api;
constructor(trackIndex, sceneIndex) {
this.#api = new LiveAPI(
`live_set tracks ${trackIndex} clip_slots ${sceneIndex}`);
}
makeClip() {
const clipSlot = this.#api;
if (clipSlot.get("has_clip") == false) {
clipSlot.call("create_clip", 4);
}
const clip = new LiveAPI(`${clipSlot.unquotedpath} clip`);
clip.call("remove_notes_extended", 0, 128, 0, clip.get("length"));
return clip;
}
}
const clipSlot = new ClipSlot(0, 0);
const clip = clipSlot.makeClip();
Let's also parameterize it on the clip length. To set an existing clip's length, we need to set both its normal start/end markers and its loop points:
class ClipSlot {
#api;
constructor(trackIndex, sceneIndex) {
this.#api = new LiveAPI(
`live_set tracks ${trackIndex} clip_slots ${sceneIndex}`);
}
makeClip(lengthInBeats) {
const clipSlot = this.#api;
if (clipSlot.get("has_clip") == false) {
clipSlot.call("create_clip", 4);
}
const clip = new LiveAPI(`${clipSlot.unquotedpath} clip`);
clip.call("remove_notes_extended", 0, 128, 0, clip.get("length"));
clip.set("start_marker", 0);
clip.set("end_marker", lengthInBeats);
clip.set("loop_start", 0);
clip.set("loop_end", lengthInBeats);
return clip;
}
}
const clipSlot = new ClipSlot(0, 0);
const clip = clipSlot.makeClip(4);
Now we can create new clips or clear out existing clips and set their clip length in any clip slot. Try changing the
code to a different length like clipSlot.makeClip(16);
and observe the results. If you
increase an existing clip's length, you may need to zoom out in the clip view to see all of it.
Adding Notes to the Clip
We use a clip's "add_new_notes" function to add notes to it. It takes an object with a notes array.
The documentation for "add_new_notes" explains
the supported note properties. There are three required properties for each note: pitch
(MIDI pitch number),
start_time
(in beats, relative to clip start), and duration
(in beats). We'll start with those required properties
and explore some of the optional properties soon.
Let's add a C3 note starting on the second beat and playing for three beats. Pitch C3 is MIDI pitch number 60. If you're ever unsure about the MIDI pitch number, draw a note into a MIDI clip and hover over it. The Live status bar shows a bunch of info about that note including the MIDI pitch number.
Other music software and hardware may define MIDI pitch number 60 as C4. There's unfortunately no standard for this, only conventions.
Here's how we call the Live API to create our note. Note I have omitted the
class ClipSlot {...}
that we built above, because it will not change again in
this tutorial. You'll need to keep it in your script. I'll include it again in the final version at the end so you can
see all the code together.
const clipSlot = new ClipSlot(0, 0);
const clip = clipSlot.makeClip(4);
clip.call("add_new_notes", {notes: [
{pitch: 60, duration: 3, start_time: 1},
]});
This generates a four-beat clip with our note starting on the second beat:
Let's add more notes by building up an array of note objects and adding them all in one Live API call. I also made the clip length a variable for clarity:
const clipLengthInBeats = 4;
const clipSlot = new ClipSlot(0, 0);
const clip = clipSlot.makeClip(clipLengthInBeats);
const notes = [];
notes.push({ pitch: 60, duration: 3, start_time: 1 });
notes.push({ pitch: 64, duration: 2, start_time: 2 });
clip.call("add_new_notes", {notes});
Now we have two notes:
Sweet! This is a good foundation for generating an arbitrary number of notes.
The Rhythm of the Primes
As mentioned in the intro, I discovered a Youtube video called "The Rhythm of the Primes" that intrigued me and inspired me to explore using prime numbers to generate rhythms. JavaScript is a powerful tool for this.
First, we need to know a bunch of prime numbers. Finding prime numbers can be an interesting programming exercise, but coding that up from scratch is a distraction to our main goal, so let's grab a list from Wikipedia and convert it to JavaScript:
const primes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41,
43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97];
Many of Live's drum racks have 16 different drum sounds at MIDI pitches 36-51. Let's write code to generate notes for those pitches.
Each of those 16 pitches will have a rhythm based on one of the first 16 primes. If the prime is 2, we'll play every 2 beats. If the prime is 3, we'll play every 3 beats, and so on.
For a given pitch
and prime
, we can implement that
note generation logic like this with modular arithmetic:
for (let start = 0; start < clipLengthInBeats; start++) {
if (start % prime == 0) {
notes.push({ pitch, duration, start_time: start });
}
}
Since we are triggering drum sounds, the duration doesn't really matter, so I picked 1
.
It needs to be short enough that notes don't overlap.
The note generation logic loops over the list of primes while incrementing a pitch
starting from the basePitch
of 36, since that's what Ableton uses for the lowest note
of their built-in drum racks. We also increase the clipLengthInBeats
significantly because
it takes a long time for primes-based patterns to repeat.
const primes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41,
43, 47, 53, 59, 61, 67, 71, 73, 79, 83, 89, 97];
const basePitch = 36;
const duration = 1;
const clipLengthInBeats = 256;
const clipSlot = new ClipSlot(0, 0);
const clip = clipSlot.makeClip(clipLengthInBeats);
const notes = [];
let pitch = basePitch;
for (const prime of primes) {
for (let start=0; start < clipLengthInBeats; start++) {
if(start % prime == 0) {
notes.push({ pitch, duration, start_time: start });
}
}
pitch++;
}
clip.call("add_new_notes", {notes});
This should have generated a clip 256 beats in length (64 measures in 4/4) with a lot of notes in it. You might need to zoom out in the clip to see it. Time to test it out.
On the very first beat, 16 drum samples play at the same time. It might be a bit loud, so keep your volume levels moderate. We'll address this momentarily.
Add a drum rack from the Live library to the track. You can find these under "Drums" and filtering by "Drum Kit". Most drum racks should have at least 16 samples and will work well with our generated clip. The nice thing about drum racks is it shows you the name of each drum in the MIDI clip view.
Only basePitch
on the First Beat
First improvement: Let's make it so only the basePitch
plays on the first beat. If you
zoom in on the clip's start, here's how things look now:
Was that loud? Sorry about that. Always be mindful of your levels and don't turn things up too loud when experimenting with new things.
Let's fix this. When we decide to add a note, let's add more conditions: either the
pitch
must be the basePitch
, or
start
must be greater than zero (the first beat). We can do it all in the same
if
condition:
let pitch = basePitch;
for (const prime of primes) {
for (let start=0; start < clipLengthInBeats; start++) {
if(start % prime == 0 && (pitch == basePitch || start > 0)) {
notes.push({ pitch, duration, start_time: start });
}
}
pitch++;
}
Here's the result:
Perfect.
Speeding Things Up
My next issue with the generated clip is it's boring because it's too slow. Let's double the speed. The core logic of
the loop to increment start++
and check
start % prime == 0
is nice and simple, so I don't want to mess with
that.
Here's one way to think about it: If we're doubling the speed, we're going to squeeze twice as many notes into the same amount of time. So what if we calculate the notes for a clip twice as long using the same algorithm we already built, and squish those notes into a space half the size by dividing time in half when we construct the note? Here's what I mean:
let pitch = basePitch;
for (const prime of primes) {
for (let start=0; start < 2*clipLengthInBeats; start++) {
if(start % prime == 0 && (pitch == basePitch || start > 0)) {
notes.push({ pitch, duration, start_time: start/2 });
}
}
pitch++;
}
Give it a try! Now the speed seems reasonable in the 120+ BPM range. Bump the tempo up as much as you want depending on your caffeine level.
Now might be a good time to play around, try different drum racks, and swap samples around. Maybe you don't want the
kick drum to be the most frequently triggered sample. Hi-hats often work better as the fast drum. I like making hi-hats
the fast drum (the 2
prime), a contrasting hat-like sound such as maracas the next fastest (the 3
prime) and the
kick drum the one after that (the 5
prime). You do you.
But When Does it Repeat?
We can keep increasing the clipLengthInBeats
to try to let the pattern play out:
const clipLengthInBeats = 1024;
It takes a while for this baby to repeat exactly. It's not even close to repeating yet!
In case you're wondering, that is 3434 notes. I logged notes.length
to find out. Have
you ever put that many notes in a single clip before? Me neither. This is the power of Max for Live. We could keep
going, but be warned it can make your computer hang. I got scared after 30,000 notes. I bet I crashed someone's old
computer. If you're feeling adventurous, go nuts.
How many beats will it take to repeat? Because all the numbers are prime, they have no common factors and we multiply
them all together to find out how long it takes to repeat exactly. For the first sixteen primes, we're talking about
2×3×5×7×11×13×17×19×23×29×31×37×41×43×47×53
. I was going to walk you through calculating it in JavaScript, but guess
what, it's bigger than Number.MAX_SAFE_INTEGER
and we can't even calculate it properly
in JavaScript. Let's trust Wolfram Alpha
with this question.
Right. Ok. It will take 32,589,158,477,190,044,730 beats before this pattern repeats. 32 quintillion beats. I don't completely trust my math at this point but I'm getting this: if we play it at 120 BPM, it will take 37 times the age of the universe before the pattern repeats. Even if I'm off by a few orders of magnitude, you get the idea.
TL;DR - We cannot possibly make a Live clip long enough for this pattern to repeat exactly. Let's move on.
Varying Velocity
With some well-chosen drum sounds, I feel like there's some decent musical potential in this drum pattern. But it's too robotic: every drum hit sounds the same.
The main properties of a MIDI note are its pitch, its velocity, and when it occurs. So far we've only worked with the pitch and timing. By controlling the velocity, we can make variations between the individual notes for a given pitch, which gives the pattern more character and vibe.
I want to tie the velocity values to the pattern of primes. Modular arithmetic is good for creating cyclical patterns,
like the pattern of notes and silences created by (start % prime
) in the loop. We can
do something similar for cycling through repeating patterns of velocities.
Here's the idea: the velocity pattern length will be the same as the current pitch's prime number. If the prime is 2, we'll cycling between loud and quiet notes. If the prime is 3, we'll cycling between loud, medium, and quiet notes. For 4, we'll cycling between loud, medium-loud, medium-quiet, quiet. And so on.
In MIDI, velocity is an integer ranging from 0-127 (like pitch). 0 is a special case that represents "off". 1 represents
the smallest intensity for a played note, and 127 represents the highest intensity. We'll start each pitch off with it's
highest velocity of 127 for the loud notes, and reduce it by 100 for it's quietest note. Every time we set a new
pitch
and prime
, we'll start a new
noteCounter
variable at 0. Every time we play a note, we increment the counter. Then we
can create the pattern of velocities I described with:
127 - 100 * (noteCounter % prime) / (prime - 1);
To make sense of this, let's consider some specific cases. When prime
is 2,
noteCounter % prime
will alternate between 0 and 1, so the overall expression
alternates between 127 and (127 - 100).
When prime
is 3, noteCounter % prime
will cycle
between 0, 1, and 2. In this case, the value of (prime - 1)
is 2 and the overall
expression cycles between 127, (127 - 100 * 1/2) and (127 - 100). In all cases, the number multiplied by 100 goes from
0.0 to 1.0, and the overall velocity goes from 127 to 27.
let pitch = basePitch;
for (const prime of primes) {
let noteCounter = 0;
for (let start=0; start < 2*clipLengthInBeats; start++) {
if(start % prime == 0 && (pitch == basePitch || start > 0)) {
const velocity = 127 - (100 * (noteCounter % prime)) / (prime - 1);
notes.push({ pitch, velocity, duration, start_time: start/2 });
noteCounter++;
}
}
pitch++;
}
We can check this works as intended in the clip view by selecting individual drums/pitches and examining the velocities in the velocity lane. Here's the second pitch with a cycle of 3:
And the third pitch with a cycle of 5:
Looks good.
You may not be able to hear this very well though. Not all drum racks respond to velocity equally. If all the notes still sound too similar and repetitive, try changing drum racks again. You're looking for drum racks where the drum samples have velocity affecting volume, which will often be via this control:
If you want to exaggerate the affect of velocity, you can increase that "Vol < Vel" control to closer to 100%, and then right click and select "Copy value to siblings" to apply this setting to all drums in the drum rack.
Other Note Properties
Out of the remaining note properties, probability
and velocity_deviation
are the most interesting for this script.
You can use probability
to randomly skip notes, and velocity_deviation
to randomly change the velocity within a
specified range.
If you've been following along, you should have all the tools you need to explore these features if desired. Here's the basic usage:
notes.push({
pitch,
velocity,
duration,
start_time: start / 2,
probability: 0.5, // randomly skip about half the notes
velocity_deviation: 50, // randomly add up to 50 to the velocity
// Note: velocity_deviation can be negative
});
I'm not including this in the final version of the script, but I encourage you to explore these features.
We're Doing it "Wrong" Again
In case you didn't see this in the previous tutorial or forgot, I need to remind you we are doing an "exploratory coding
session", and writing new LiveAPI()
at the top-level of the JavaScript code won't work
in a real device. It doesn't matter that it's inside the ClipSlot
class, it's still
executed immediately from the top-level code.
The general simple solution is to wrap the code (everything outside the ClipSlot
class
definition) with function bang() {...}
and trigger it from the Max patch. I covered
this in detail in "Safely Constructing a LiveAPI Object", so give
it a (re-)read if needed, especially if you want to expand the ideas here into a real device for day-to-day usage in
Live.
For a note generator devices like this, putting it in a Max MIDI Tool is probably the best solution. Alas, I don't have time to get into that here, so check out the official docs.
Final Version
For reference, here's the complete script with all the code:
class ClipSlot {
#api;
constructor(trackIndex, sceneIndex) {
this.#api = new LiveAPI(
`live_set tracks ${trackIndex} clip_slots ${sceneIndex}`);
}
initClip(lengthInBeats) {
const clipSlot = this.#api;
if (clipSlot.get("has_clip") == false) {
clipSlot.call("create_clip", lengthInBeats);
}
const clip = new LiveAPI(`${clipSlot.unquotedpath} clip`);
clip.call("remove_notes_extended", 0, 128, 0, clip.get("length"));
clip.set("start_marker", 0);
clip.set("end_marker", lengthInBeats);
clip.set("loop_start", 0);
clip.set("loop_end", lengthInBeats);
return clip;
}
}
const primes = [2, 3, 5, 7, 11, 13, 17, 19, 23, 29, 31, 37, 41, 43, 47, 53]; // , 59, 61, 67, 71, 73, 79, 83, 89, 97];
const basePitch = 36;
const duration = 1;
const clipLengthInBeats = 256;
const clipSlot = new ClipSlot(0,0);
const clip = clipSlot.initClip(clipLengthInBeats);
const notes = [];
let pitch = basePitch;
for (const prime of primes) {
let noteCounter = 0;
for (let start=0; start < 2*clipLengthInBeats; start++) {
if (start % prime == 0 && (pitch == basePitch || start > 0)) {
const velocity = 127 - (100 * (noteCounter % prime)) / (prime - 1);
notes.push({ pitch, velocity, duration, start_time: start/2 });
noteCounter++;
}
}
pitch++;
}
clip.call("add_new_notes", { notes });
Next Steps
Let's revisit and improve our log()
function in
"Max Console Part 2: Better Logging for Objects".