Skip to content

felixroos/idlecycles

Folders and files

NameName
Last commit message
Last commit date

Latest commit

 

History

66 Commits
 
 
 
 
 
 
 
 
 
 
 
 
 
 

Repository files navigation

IdleCycles

A naive take on TidalCycles in JS, guided by learning.

Tidal from Scratch

I've created this repo to understand the core of TidalCycles at a deeper level by re-implementing its core logic. This part of the README will document my progress and could be helpful for people trying to understand tidal. Disclaimer: This is not tidal, so take it with a grain of salt.

Table Of Contents

  1. Patterns as Functions of Time
  2. Adding more Functions
  3. Language Improvements
  4. Mini Notation Parsing
  5. Operating on Values
  6. Joining Patterns Together

Chapter 1: Patterns as Functions of Time

The core primitive of tidal are Patterns, which are essentially functions of time:

let silence = (a, b) => [];
silence(0, 1); // []
silence(42, 79); // []

The above example is the most simple pattern I can think of: silence. It will always return an empty Array. The inputs a and b are the timespan we want to know about.

We love repetition

The above example doesn't really do much, so let's implement another pattern that repeats the same thing over and over:

let repeat = (a, b) => {
  a = Math.floor(a);
  b = Math.ceil(b);
  let bag = [];
  while (a < b) {
    bag.push({ a, b: a + 1, value: "X" });
    a++;
  }
  return bag;
};
repeat(0, 1); // [{a: 0, b: 1, value: 'X'}]
repeat(0, 2); // [{a: 0, b: 1, value: 'X'}, {a: 1, b: 2, value: 'X'}]
repeat(1.5, 2.5); // [{a: 1, b: 2, value: 'X'}, {a: 2, b: 3, value: 'X'}]

This function will give us one object per whole number within the given time span. The object contains the timespan during it is active + some value.

Higher Order Functions

Ideally, we'd want to choose the value to repeat, so let's make it a higher order function:

let repeat = (value) => (a, b) => {
  a = Math.floor(a);
  b = Math.ceil(b);
  let bag = [];
  while (a < b) {
    bag.push({ a, b: a + 1, value });
    a++;
  }
  return bag;
};
let snakes = repeat("snake");
let dogs = repeat("dog");
snakes(0, 2); // [{a: 0, b: 1, value: 'snake'},{a: 1, b: 2, value: 'snake'}]
dogs(1, 3); // [{a: 1, b: 2, value: 'dog'}, {a: 2, b: 3, value: 'dog'}]

Higher order just means we have a function that returns a function. The outer function takes a value and returns the inner function, which is a Pattern.

It's Hap-pening

Our returned object could be expressed as a typescript type like this:

declare type Hap<T> = { a: number; b: number; value: T };

With these Hap's, we can now define a Pattern like this:

declare type Pattern<T> = (a: number, b: number) => Hap<T>[];

Note that these typings are only a simplified version of what strudel or tidal use, but it's enough to get started thinking about Patterns.

Conc-cat-enation

Our repeat function is still a bit boring.. How about switching between different values?

let cat =
  (...values) =>
  (a, b) => {
    a = Math.floor(a);
    b = Math.ceil(b);
    let bag = [];
    while (a < b) {
      const value = a % values.length;
      bag.push({ a, b: a + 1, value });
      a++;
    }
    return bag;
  };
let pat = cat("snake", "dog");
pat(0, 2); // [{a: 0, b: 1, value: 'snake'},{a: 1, b: 2, value: 'dog'}]
pat(1, 3); // [{a: 1, b: 2, value: 'dog'}, {a: 2, b: 3, value: 'snake'}]

The above pattern gives us an infinite timeline of snakes and dogs!

The implementation looks very similar to repeat, maybe we should generalize this loop:

let cycle = (callback) => (a, b) => {
  a = Math.floor(a);
  b = Math.ceil(b);
  let bag = [];
  while (a < b) {
    bag = bag.concat(callback(a, a + 1));
    a++;
  }
  return bag;
};

let repeat = (value) => cycle((a, b) => [{ a, b, value }]);

let cat = (...values) =>
  cycle((a, b) => {
    let value = values[a % values.length];
    return [{ a, b, value }];
  });

With this, we could implement repeat and cat much more concise!

Patterns as Functions of Space

So far, I've talked about a and b as a time span, but that's just a metaphor, because tidal is used for music (typically). But it's actually just a numerical range, so nothing stops us from interpreting Patterns as functions of space:

cat

In this other kind of cat picture, we see each whole number cycle represented as a line, and the value of each Hap is interpreted as a color.

From now on, I'll use this type of visualization, as it makes more sense in a written guide.

Chapter 1 REPL

This is the end of the first chapter! I've decided to create a separate HTML file with the code for each chapter including a minimal REPL for you to play around with.

Open chapter1.html to play with cat and repeat. You can evaluate the code with ctrl+enter to see the visualization in action!

Here are some examples:

This might still not be very exciting, but we're on our way!

Chapter 2: adding more functions

In this chapter we will take the previous primitives and create new useful functions.

slow and fast

One interesting property about Patterns as functions of time (or space) is that we can edit the time span from the outside:

let fast = (factor, pat) => (a, b) =>
  pat(
    a * factor, // scale up timespan
    b * factor
  ).map((hap) => ({
    a: hap.a / factor, // scale down hap timespan
    b: hap.b / factor,
    value: hap.value,
  }));

fast(5, cat("cyan", "magenta", "yellow"));

This will have the effect of scaling the time axis of our Haps. This is the visual representation:

fast

The trick is to first scale up our timespan and then scale down the time span of each Hap by the given factor.

We can implement the opposite version of the function like this:

let slow = (factor, pat) => fast(1 / factor, pat);

Example:

slow

The fact that there are gray areas is due to the way the visual is implemented. Each Hap gets one rectangle whose width is determined by its timespan duration. In the above case, each colored rectangle is 2 screens wide, so you won't see half of it. We could theoretically implement a split logic, where rectangles that are longer than one screen would be split and moved to the next line, but i guess this is fine for now..

seq

Another useful function can be derived by combining cat and fast:

let seq = (...values) => fast(values.length, cat(...values));

This function will squeeze its given values into a single cycle:

seq

Here we can already see a cool property of functional programming: If we have a set of base functions, we can combine them declaratively to create new functions. You can think of it like crafting new materials from a set of raw materials.

stack

The stack function will place all values in the same timespan:

let stack = (...values) =>
  cycle((a, b) => values.map((value) => ({ a, b, value })));

Visually, we're in 2D, so stacked rectangles will end up in the same place. This will only have a visual effect when the stacked colors are transparent:

stack

Patterns of Patterns

Our standard library is growing, but we're still missing an important property of Patterns: the ability to nest Patterns inside of each other. Here's an example:

cat("cyan", seq("magenta", "yellow"));

If we do that, we only get cyan, because the second input of cat is not a color (and the canvas API will resort to the last working color when drawing the rect of the second input).

We can fix it like this:

let nest = (value, a, b) => {
  // if we have a function, we'll assume it's a Pattern function
  if (typeof value === "function") {
    return value(a, b);
  }
  // if it's not a function we'll take it as before
  return [{ a, b, value }];
};
let cat = (...values) =>
  cycle((a, b) => {
    let value = values[a % values.length];
    return nest(value, a, b);
  });

let stack = (...values) =>
  cycle((a, b) => values.map((value) => nest(value, a, b)).flat());

Now if a value is a function, we'll call it with the current cycle, assuming it's a Pattern function, which allows us to nest:

catseq

And magically, we can nest Patterns as deep as we like:

nesting

Chapter 2 REPL

This is the end of the second chapter!

Open chapter2.html to play with all the new functions. You can evaluate the code with ctrl+enter to see the visualization in action!

With these new functions, we can already create some non-trivial patterns, but the syntax is still a bit unwieldy, as it's difficult to keep track of all the parens when nesting.. Luckily, there is a way out, which we'll implement in the next chapter!

Examples:

Chapter 3: Language Improvements

Let's improve the language a bit. In strudel you can write:

cat("cyan", "magenta", "yellow").fast(2);
// is equivalent to:
fast(2, cat("cyan", "magenta", "yellow"));

In our little implementation, we can only write the second variant. To get to the first variant, cat needs to return an object, where fast is a property.

Pattern as a class

We can refactor our existing code by introducing the Pattern class.

class Pattern {
  constructor(query) {
    this.query = query;
  }
}
// Pattern creation shortcut:
let P = (q) => new Pattern(q);

Pattern.prototype.fast = function (speed) {
  return fast(speed, this);
};

Pattern.prototype.slow = function (speed) {
  return slow(speed, this);
};

This still won't work, because fast itself needs to return a Pattern instance + handle a Pattern as its input:

let fast = (factor, pat) =>
  // the P(...) wraps our query function in a Pattern
  // + pat is now a Pattern, which has to queried by calling pat.query
  P((a, b) =>
    pat.query(a * factor, b * factor).map((hap) => ({
      a: hap.a / factor,
      b: hap.b / factor,
      value: hap.value,
    }))
  );

We're almost there, now the last refactoring to do is to make sure all the other functions return a Pattern instance, so we can call .fast and .slow on them. Luckily, our cycle function is the only thing we need to touch:

let cycle = (callback) =>
  P((a, b) => {
    // ^ the P(...) is new
    a = Math.floor(a);
    b = Math.ceil(b);
    let bag = [];
    while (a < b) {
      bag = bag.concat(callback(a, a + 1));
      a++;
    }
    return bag;
  });

Finally, we can do:

fastmethod

We need another little fix for nested Patterns:

let nest = (value, a, b) => {
  if (value instanceof Pattern) {
    return value.query(a, b);
  }
  return [{ a, b, value }];
};

Before we were checking for value being a function, now we have to check if it's a Pattern.

More Meta Programming

We have added our fast and slow methods to a Pattern by defining a function on Pattern.prototype. Because we want to add many more functions that transform a Pattern, we can add a helper function to simplify that:

let register = (name, fn) => {
  Pattern.prototype[name] = function (...args) {
    return fn(args, this);
  };
  return fn;
};

let fast = register("fast", (factor, pat) => /* ... */);
let slow = register("slow", (factor, pat) => fast(1 / factor, pat));

This paves the way for a lot more functions to come!

lastOf / firstOf

Last but not least, let's add another set of functions:

let firstOf = register("firstOf", (n, fn, pat) =>
  cycle((a, b) => (a % n === 0 ? fn(pat) : pat).query(a, b))
);
let lastOf = register("lastOf", (n, fn, pat) =>
  cycle((a, b) => (a % n === n - 1 ? fn(pat) : pat).query(a, b))
);

This allows us to conditionally apply a function every n cycles:

lastOf

Chapter 3 REPL

This is the end of the third chapter!

Open chapter3.html to play with the new syntax. You can evaluate the code with ctrl+enter to see the visualization in action!

Examples

Chapter 4: Mini Notation Parsing

One very distinct feature of tidal is its mini notation. Let's build it from scratch! Most of the logic is heavily inspired by A Compiler From Scratch.

Our goal

We want to be able to write stuff like:

mini("<cyan magenta [yellow white]>").fast(5);
// which is equivalent to
cat("cyan", "magenta", seq("yellow", "white")).fast(5);

To be able to do this, we need to parse the mini notation string and somehow construct the appropriate function calls.

The Tokenizer

We start with a list of valid tokens, which are the smallest meaningful chunks our code can have:

let token_types = {
  open_cat: /^\</, // "<"
  close_cat: /^\>/, // ">"
  open_seq: /^\[/, // "["
  close_seq: /^\]/, // "]"
  plain: /^[a-zA-Z0-9\.\#]+/, // values
};

This is a mapping from a human friendly token type to the regex that parses it.

With it, we can tokenize a string like this:

function next_token(code) {
  for (let type in token_types) {
    const match = code.match(token_types[type]);
    if (match) {
      return { type, value: match[0] };
    }
  }
  throw new Error(`could not match "${code}"`);
}
next_token("<cyan magenta>"); // { type: "open_cat", value: "<" }
next_token("cyan magenta>"); // { type: "plain", value: "cyan" }
// ...

Now we can use that function repeatedly to tokenize a whole string:

function tokenize(code) {
  let tokens = [];
  while (code.length > 0) {
    code = code.trim(); // trim white spaces
    const token = next_token(code);
    code = code.slice(token.value.length);
    tokens.push(token);
  }
  return tokens;
}
tokenize("<cyan magenta>");
/* [
{ type: "open_cat", value: "<" },
{ type: "plain", value: "cyan" },
{ type: "plain", value: "magenta" },
 { type: "close_cat", value: ">" },
] */

Here, we are taking token by token out of the string, until it is empty. That's already a fine tokenizer, let's move on to the parser!

The Parser

After the code has been tokenized, we can construct a tree. Mini notation supports arbitrary levels of nesting, so we want to support stuff like <cyan [magenta [white black] yellow]>. For this example we'd expect something like:

{
  "type": "cat",
  "args": [
    { "type": "plain", "value": "cyan" },
    {
      "type": "seq",
      "args": [
        { "type": "plain", "value": "magenta" },
        {
          "type": "seq",
          "args": [
            { "type": "plain", "value": "white" },
            { "type": "plain", "value": "black" }
          ]
        },
        { "type": "plain", "value": "yellow" }
      ]
    }
  ]
}

This might look impossible at first, but it's very doable with some sprinkles of recursion.

Let's start by defining a Parser that is really dumb:

class Parser {
  // take code, return tree
  parse(code) {
    this.tokens = tokenize(code);
    return this.parse_cat();
  }
  // parse_cat (wip)
  parse_cat() {
    this.consume("open_cat");
    const a = this.consume("plain");
    const b = this.consume("plain");
    this.consume("close_cat");
    return { type: "cat", args: [a, b] };
  }
  consume(type) {
    // shift removes first element and returns it
    const token = this.tokens.shift();
    if (token.type !== type) {
      throw new Error(`expected token type ${type}, got ${token.type}`);
    }
    return token;
  }
}
new Parser().parse("<cyan magenta>");

The last line will return

{
  "type": "cat",
  "args": [
    {
      "type": "plain",
      "value": "cyan"
    },
    {
      "type": "plain",
      "value": "magenta"
    }
  ]
}

... which is exactly what we want! The problem is that the parser can only parse cat with exactly 2 plain values... When we try:

new Parser().parse("<cyan magenta white>");

.. we get expected token type close_cat, got plain, because we have 3 plain values. To match an arbitrary number of values, we can do this:

class Parser {
  /* .. */
  parse_cat() {
    this.consume("open_cat");
    let args = [];
    while (this.tokens[0]?.type !== "close_cat") {
      args.push(this.consume("plain"));
    }
    this.consume("close_cat");
    return { type: "cat", args };
  }
}
// all of these work now:
new Parser().parse("<cyan magenta white>");
new Parser().parse("<cyan magenta>");
new Parser().parse("<cyan magenta brown red>");

A similar function can be implemented for seq:

class Parser {
  /* .. */
  parse(code) {
    this.tokens = tokenize(code);
    return this.parse_seq();
  }
  parse_seq() {
    this.consume("open_seq");
    let args = [];
    while (this.tokens[0]?.type !== "close_seq") {
      args.push(this.consume("plain"));
    }
    this.consume("close_seq");
    return { type: "seq", args };
  }
}
new Parser().parse("[cyan magenta white]");

This smells like we could use an abstraction:

class Parser {
  /* .. */
  parse_args(close_type) {
    const args = [];
    while (this.tokens[0]?.type !== close_type) {
      args.push(this.consume("plain"));
    }
    return args;
  }
  parse_seq() {
    this.consume("open_seq");
    const args = this.parse_args("close_seq");
    this.consume("close_seq");
    return { type: "seq", args };
  }
  parse_cat() {
    this.consume("open_cat");
    const args = this.parse_args("close_cat");
    this.consume("close_cat");
    return { type: "cat", args };
  }
}

We can now tie both functions together with a function that can parse any expression:

class Parser {
  /* .. */
  parse(code) {
    this.tokens = tokenize(code);
    return this.parse_expr();
  }
  parse_expr() {
    let next = this.tokens[0]?.type;
    if (next === "open_cat") {
      return this.parse_cat();
    }
    if (next === "open_seq") {
      return this.parse_seq();
    }
    if (next === "plain") {
      return this.consume("plain");
    }
    throw new Error(`unexpected token "${this.tokens[0].value}"`);
  }
  parse_args(close_type) {
    const args = [];
    while (this.tokens[0]?.type !== close_type) {
      args.push(parse_expr()); // <-- use it here as well!
    }
    return args;
  }
}
// this now actually works:
new Parser().parse("<cyan [magenta [white black] yellow]>");

That's already a basic version of our parser!

From a Tree to a Pattern

Now having a tree that represents the layers of nesting in our syntax, the last step is to convert it into pattern function calls:

function patternifyTree(tree) {
  if (tree.type === "cat") {
    const args = tree.args.map((arg) => patternifyTree(arg));
    return cat(...args);
  }
  if (tree.type === "seq") {
    const args = tree.args.map((arg) => patternifyTree(arg));
    return seq(...args);
  }
  if (tree.type === "plain") {
    return tree.value;
  }
}

This is all we need! Now we can tie the whole thing together and call it a day:

const parser = new Parser();
let mini = (code) => {
  const tree = parser.parse(code);
  const pat = patternifyTree(tree);
  if (pat instanceof Pattern) {
    return pat;
  }
  return repeat(pat); // repeat plain values
};
mini("<cyan [magenta white]>");
// = cat("cyan", seq("magenta", "white"))

And that's a basic implementation of the mini notation! It certainly doesn't cover all features, but it should give you the basic idea.

Chapter 4 REPL

This is the end of the fourth chapter!

Open chapter4.html to play with the new syntax. You can evaluate the code with ctrl+enter to see the visualization in action!

Examples:

Chapter 5: Operating on Values

Up until now, our functions to modify a pattern were very much concerned with the timespan. Let's look at ways to modify the value of a Hap.

withValue

let withHap = register("withHap", (fn, pat) =>
  P((a, b) => pat.query(a, b).map(fn))
);
let withValue = register("withValue", (fn, pat) =>
  pat.withHap((hap) => ({ ...hap, value: fn(hap.value) }))
);

The function withHap lets us edit a hap as we like. withValue uses it to run a function on the Hap value specifically, returning a new Hap.

With that, we could put together different shades of blue:

withValue

Color from Numbers

Operating on values gets more exciting when we have numbers to work with. For that matter, let's update the draw logic that sets the rectangle color:

if (isNaN(Number(value))) {
  ctx.fillStyle = value; // <- like before
} else {
  value = Number(value);
  ctx.fillStyle = `hsl(${value}turn 50% 50%)`;
}

Now, if we receive a valid number, we'll use it as the hue value in an hsl color! Here's a rainbow:

rainbow

Arithmetic

Now that we have numbers, we can do some arithmetic!

let add = register("add", (n, pat) => pat.withValue((v) => v + n));
let sub = register("sub", (n, pat) => pat.withValue((v) => v - n));
let mul = register("mul", (n, pat) => pat.withValue((v) => v * n));
let div = register("div", (n, pat) => pat.withValue((v) => v / n));
let mod = register("mod", (n, pat) => pat.withValue((v) => v % n));

Using add and mod, we could rotate the color palette:

addA

..changing the value of add:

addB

Object as Values

Right now, we are only able to pattern the hue. Wouldn't it be nice if we could pattern other properties separately, like hue, saturation, lightness, transparency, rect size, ...? We can do this by adjusting our draw code again:

const haps = pat.query(0, cycles);
haps.forEach(({ a, b, value }) => {
  let { color, h: hue, s: saturation, l: lightness, a: alpha } = value;
  let fillStyle;
  if (color) {
    fillStyle = color;
  }
  if (hue !== undefined) {
    hue = hue * 360;
    saturation = (saturation ?? 0.5) * 100;
    lightness = (lightness ?? 0.5) * 100;
    alpha = alpha ?? 1;
    fillStyle = `hsla(${hue},${saturation}%,${lightness}%,${alpha})`;
    console.log("fillStyle", fillStyle);
  }
  ctx.fillStyle = fillStyle;
});

With that in place, our patterns now need to contain objects:

cat(
  { color: "yellow" },
  { h: 0.5, s: 0 },
  { h: 0.5, s: 0.25 },
  { h: 0.5, s: 0.5 },
  { h: 0.5, s: 0.75 },
  { h: 0.5, s: 1 }
).fast(7);

hapObject

This notation is rather clunky, we need control functions to the rescue!

Control Functions

A control is basically a shortcut to get a pattern of objects, which means each Hap has an object as value.

let control = (name) =>
  register(name, (value, pat) => {
    if (pat) {
      // e.g. s(.5).h(.1) === h(.1, s(.5))
      return pat.withValue((v) => Object.assign(v, { [name]: value }));
    }
    // e.g. h(cat(.1, .2, .3))
    if (value instanceof Pattern) {
      return value.withValue((v) => ({ [name]: v }));
    }
    // e.g. h(.3)
    return repeat({ [name]: value });
  });

let color = control("color");
let h = control("h");
let s = control("s");
let l = control("l");
let a = control("a");

Controls are special in the sense that they can be used to create a pattern but also modify a pattern. This allows us to completely avoid curly braces {} and keep using functions with method chaining for object values.

controls

Implicit Mini Notation

In Chapter 4, we've implemented mini notation. We can modify the register function to treat any string as mini notation like this:

let register = (name, fn) => {
  let q = (...args) => {
    args = args.map((arg) => {
      if (typeof arg === "string") {
        return mini(arg);
      }
      return arg;
    });
    return fn(...args);
  };
  Pattern.prototype[name] = function (...args) {
    return q(...args, this);
  };
  return q;
};

Now we can set our controls with mini notation:

s("<0 .25 .5 .75>").h(0.1).fast(7);

implicitmini

End of Chapter 5

This is starting to look very usable! In the next chapter we will see how we can join multiple Patterns into one, making our language much more expressive.

Chapter 5 REPL

Open chapter5.html to play with the new syntax. You can evaluate the code with ctrl+enter to see the visualization in action!

Examples:

Chapter 6: Joining Patterns Together

In Strudel, we can use multiple Patterns in one expression:

s("[0 .5]").h("<.1 .3>");

This is not possible with our current implementation... If we look at the haps from this pattern, we see the following:

[
  // cycle 1
  { a: 0, b: 0.5, value: { s: 0, h: Pattern } },
  { a: 0.5, b: 1, value: { s: 0.5, h: Pattern } },
  // cycle 2
  { a: 1, b: 1.5, value: { s: 0, h: Pattern } },
  { a: 1.5, b: 2, value: { s: 0.5, h: Pattern } },
  // ....
];

The property h ends up as a Pattern, which will confuse our drawing function, because it expects a number. Our desired result would be:

[
  // cycle 1
  { a: 0, b: 0.5, value: { s: 0, h: 0.1 } },
  { a: 0.5, b: 1, value: { s: 0.5, h: 0.1 } },
  // cycle 2
  { a: 1, b: 1.5, value: { s: 0, h: 0.3 } },
  { a: 1.5, b: 2, value: { s: 0.5, h: 0.3 } },
  // ....
];

To get this, we need to somehow combine the 2 Patterns into a single Pattern.

Structure from the Left

When joining patterns, the structure comes "from the left" by default. This means that the leftmost pattern dictates the temporal structure:

s("[0 .5]").h("[.1 .2 .3 .4]");
// [ {s:0, h:.1}, {s:0.5, h:.3} ]

In the first line, the first cycle has 2 Haps, because the leftmost pattern "[0 .5]" has 2 Haps. The value of h is aligned to that pattern. which is why the first Hap gets .1 (starting at 0) and the second .3 (starting at .5).

This is how we could implement this one specific case:

let left = s("[0 .5]");
let right = h("[.1 .2 .3 .4]");
let joined = left
  .query(0, 1)
  /* 
  [ 
    { a: 0, b: 0.5, value: { s: 0 } }, 
    { a: .5, b: 1, value: { s: .5 } } 
  ]
  */
  .map((l) => {
    let r = right.query(l.a, l.b)[0];
    let value = { ...r.value, ...l.value };
    return { ...l, value };
  });
/* 
[ 
  { a: 0, b: 0.5, value: { s: 0, h: .1 } }, 
  { a: .5, b: 1, value: { s: .5, h: .3 } } 
]
*/

The idea is:

  1. query the left pattern
  2. query the right pattern for each hap of the left pattern
  3. merge the values

Patternify Args

To make this work for any function, we can implement a more generalized version of the above in the register function:

let register = (name, fn) => {
  let q = (...args) => patternifyArgs(fn, args);
  Pattern.prototype[name] = function (...args) {
    return q(...args, this);
  };
  return q;
};

function patternifyArgs(fn, args) {
  // interpret strings as mini notation
  args = args.map((arg) => {
    if (typeof arg === "string") {
      return mini(arg);
    }
    return arg;
  });
  const pat = args[args.length - 1];
  const rest = args.slice(0, -1);
  // short circuit if no extra args or extra args have no pattern
  if (!rest.length || !rest.find((arg) => arg instanceof Pattern)) {
    return fn(...args);
  }
  return joinLeft(pat, rest, fn); // <- TBD
}

Here, patternifyArgs preprocesses the arguments to the function fn, allowing us to join any Patterns if needed. To understand what happens here, let's look at how we could call register for h and s:

let h = register("h", (value, pat) =>
  pat.withValue((v) => ({ ...v, h: value }))
);
let s = register("s", (value, pat) =>
  pat.withValue((v) => ({ ...v, s: value }))
);

Our example case s("[0 .5]").h("[.1 .2 .3 .4]") could also be written as:

h("[.1 .2 .3 .4]", s("[0 .5]"));
// ^ value         ^ pat

Here, our fn is h and args are ["[.1 .2 .3 .4]", s("[0 .5]")]. The first arg is a string, so it will be parsed as mini notation, turning it into a Pattern. This means we now have 2 Patterns that need to be joined..

joinLeft

joinLeft is called with pat, args and fn. In our example, these are:

  • pat: s("[0 .5]")
  • args: [ mini("[.1 .2 .3 .4]") ]
  • fn: (value, pat) => pat.withValue((v) => ({ ...v, h: value })))

With that in mind, let's look at the implementation for joinLeft:

function joinLeft(pat, args, fn) {
  return P((a, b) => {
    let haps = [];
    // pat = structure source
    const src = reify(pat).query(a, b);
    for (let j in src) {
      const hap = src[j]; // source hap
      const subargs = args.map((arg) => {
        if (arg instanceof Pattern) {
          // query arg within source hap
          const arghaps = arg
            .query(hap.a, hap.b)
            .filter((h) => intersects(h, hap));
          // take first intersection
          return arghaps[0].value;
        }
        return arg;
      });
      // construct function call where subargs are not patterns anymore
      const subhaps = fn(...subargs, pat)
        .query(hap.a, hap.b)
        .filter((h) => intersects(h, hap));
      // concat haps
      haps = haps.concat(subhaps);
    }
    return haps;
  });
}
function intersects(h1, h2) {
  return h1.a < h2.b && h2.a < h1.b;
}

Like before in our narrow example, we will query each Pattern arg once for each source hap to get the value for that point in time. This allows us to call fn without Pattern args, with the correct value at each point in time. Filtering with intersects makes sure we remove haps out of range (query might return haps of the same cycle, outside of the given range).

We can now finally use multiple patterns at once:

leftJoin

leftJoin2

Bonus: Extra Controls

To flex a little bit with the join feature, let's add more controls:

let size = control("size");
let rotate = control("rotate");
let stroke = control("stroke");
let strokeStyle = control("strokeStyle");

Check out the draw function for how these controls are drawn.. Now we can create some more interesting patterns:

rotate

Chapter 6 REPL

Open chapter6.html to play with the new joins. You can evaluate the code with ctrl+enter to see the visualization in action!

Examples:

Chapter 7: Let's make sound

Let's switch things up a bit and use our patterns to make sound.

Loading and Playing Samples

A lot of fun can be had with samples, let's build a mini sampler using the Web Audio API. This is how we load and play a sound:

const ac = new AudioContext();
// fetches the sound file from the given url and creates an AudioBuffer
function loadSound(url) {
  return fetch(url)
    .then((res) => res.arrayBuffer())
    .then((buf) => ac.decodeAudioData(buf));
}
// plays the given AudioBuffer
function playSound(buffer, time = ac.currentTime, latency = 0.001) {
  // create source
  const src = new AudioBufferSourceNode(ac);
  src.buffer = buffer;
  // connect to output
  src.connect(ac.destination);
  // play sound
  const begin = time + latency;
  const end = begin + buffer.duration;
  src.start(begin);
  src.stop(end);
}
loadSound(
  "https://raw.githubusercontent.com/tidalcycles/Dirt-Samples/master/808bd/BD0000.WAV"
).then((buffer) => {
  console.log("sound loaded");
  // attach click listener when sound is loaded
  document.body.addEventListener("click", () => playSound(buffer));
});

The code above is the basic way to load and play a sound. Note that we only have to load the buffer once, but we can play it as often as we like by creating a new AudioBufferSourceNode for each playback. An AudioBufferSourceNode can not be reused!

You can try this code out by copy pasting it to the browser console, then clicking into the page!

Loading a Sample Map

We could create a sample map by mapping keys to urls like this:

// soundMap example: { bd: "<audio-file-url>", hh: "<audio-file-url>", ... }
async function loadSoundMap(soundMap) {
  const entries = Object.entries(soundMap);
  const buffers = await Promise.all(
    entries.map(([key, url]) => loadSound(url))
  );
  // return same format as input soundMap, but with AudioBuffer's instead of urls
  return Object.fromEntries(
    buffers.map((buffer, i) => [entries[i][0], buffer])
  );
}
let loadSounds = loadSoundMap({
  bd: "https://raw.githubusercontent.com/tidalcycles/Dirt-Samples/master/808bd/BD0000.WAV",
  hh: "https://raw.githubusercontent.com/tidalcycles/Dirt-Samples/master/808hc/HC00.WAV",
  cp: "https://raw.githubusercontent.com/tidalcycles/Dirt-Samples/master/808/CP.WAV",
});
document.body.addEventListener("click", async () => {
  const sounds = await loadSounds;
  // play random sound on click
  const keys = Object.keys(sounds);
  const pick = keys[Math.floor(Math.random() * keys.length)];
  const buffer = sounds[pick];
  playSound(buffer);
});

A Very Simple Scheduler

To play a Pattern, we can create a query loop with some Web Audio trickery:

// helper to get a callback for a given audio time
function webAudioTimeout(audioContext, onComplete, startTime, stopTime) {
  const constantNode = audioContext.createConstantSource();
  constantNode.start(startTime);
  constantNode.stop(stopTime);
  constantNode.onended = () => onComplete();
}

let pattern;
let running = false;
let offset = 0; // keeps track of cycles passed
async function play(pat) {
  pattern = pat;
  const sounds = await loadSounds;
  const cycles = 1; // how many cycles we're querying per loop
  const latency = 0.1; // playback latency
  if (running) {
    // clock is already running
    return;
  }
  const firstCycle = ac.currentTime;
  const run = () => {
    running = true;
    const haps = pattern
      .query(offset, offset + cycles)
      // make sure onset happens after offset
      .filter((hap) => hap.a >= offset);
    haps.forEach((hap) => {
      const buffer = sounds[hap.value.s];
      const time = firstCycle + hap.a + latency;
      playSound(buffer, time);
    });
    const start = firstCycle + offset;
    const end = firstCycle + offset + cycles;
    // schedule next loop
    webAudioTimeout(ac, run, start, end);
    offset += cycles;
  };
  run();
}

Chapter 8: Failure

My previous attempt at joining Patterns did not work for some cases. I've now copy pasted a tiny bit of Strudel core code.. Let's break it down.. Let's take an example of a function with multiple arguments:

let euclid = register("euclid", (accents, steps, pat) => /**/);

The implementation is not important here. Let's look at what happens in register:

// name = 'euclid', fn = (accents, steps, pat) => /**/
let register = (name, fn) => {
  let q = (...args) => patternifyArgs(fn, args);
  Pattern.prototype[name] = function (...args) {
    return q(...args, this);
  };
  return q;
};

This is how it could be called:

s("bd").euclid("<3 5>", "<8 16>");
// which is equivalent to:
euclid("<3 5>", "<8 16>", s("bd"));

In this case, euclid calls q with these args: ["<3 5>", "<8 16>", s("bd")], which will be passed to patternifyArgs along with the euclid implementation:

// fn = (accents, steps, pat) => /**/
// args = ["<3 5>", "<8 16>", s("bd")]
function patternifyArgs(fn, args) {
  // pat = s("bd")
  const pat = args[args.length - 1];
  // rest = ["<3 5>", "<8 16>"]
  const rest = args.slice(0, -1);
  // short circuit if no extra args or extra args have no pattern
  if (!rest.length || !rest.find((arg) => arg instanceof Pattern)) {
    // this would happen if euclid would be called as s("bd").euclid(3, 8)
    return fn(...args);
  }

  const [left, ...right] = rest;
  // left = "<3 5>"
  // right = ["<8 16>"]

  let mapFn = (...args) => fn(...args, pat);
  mapFn = curry(mapFn, rest.length);
  // rest.length = 2
  // mapFn = (a) => (b) => fn(a, b, s("bd"))

  let acc = left.withValue(mapFn);
  // "<3 5>".withValue((a) => (b) => fn(a, b, s("bd")))
  /*
  [
   {a:0, b:1, value: (b) => fn(3, b, s("bd")) },
   {a:1, b:2, value: (b) => fn(5, b, s("bd")) },
  ]
  */
  for (let i in right) {
    // right[i] = "<8 16>"
    acc = appLeft(right[i], acc);
    /*
    cat(
      euclid(3, 8, s("bd")),
      euclid(5, 16, s("bd"))
    )
    */
  }
  return P((a, b) =>
    acc
      .query(a, b)
      .map((hap_pat) => hap_pat.value.query(a, b))
      .flat()
  );
}

Here's appLeft:

// pat_val = "<8 16>"
// pat_func = "<3 5>".withValue((a) => (b) => fn(a, b, s("bd")))
function appLeft(pat_val, pat_func) {
  pat_val = reify(pat_val);
  pat_func = reify(pat_func);
  function query(a, b) {
    const haps = [];
    for (const hap_func of pat_func.query(a, b)) {
      /* 
        // hap_func is one of
        {a:0, b:1, value: (b) => euclid(3, b, s("bd")) }
        {a:1, b:2, value: (b) => euclid(5, b, s("bd")) }
      */
      /*
        "<8 16>".query(0, 1)
        "<8 16>".query(1, 2)
      */
      const hap_vals = pat_val.query(hap_func.a, hap_func.b);
      // [{a:0, b:1, value:8}, {a:1, b:2, value:16}]
      for (const hap_val of hap_vals) {
        if (intersects(hap_func, hap_val)) {
          const value = hap_func.value(hap_val.value);
          // value = euclid(3, 8, s("bd")))
          // value = euclid(3, 16, s("bd")))
          haps.push({ a: hap_func.a, b: hap_func.b, value });
        }
      }
    }
    /* 
      // hap_func is one of
      {a:0, b:1, value: euclid(3, 8, s("bd")) }
      {a:1, b:2, value: euclid(5, 16, s("bd")) }
    */
    return haps;
  }
  return new Pattern(query);
}

The outcome of appLeft is equivalent to:

cat(s("bd").euclid(3, 8), s("bd").euclid(5, 16));

... which is exactly what we want! How magical

To be continued

Old Examples

These are kind of outdated

Releases

No releases published

Packages

No packages published

Languages