SuperCollider
Audio File (Soundcloud): "Glass Bead" made in SuperCollider
Attached below is an example of a section of the SuperCollider code used to make "Glass Bead" (2015). The use of Tdefs in the code allows for real-time substitution of complete tasks (Just in Time programming). This is a kind of coding that allows for changes to happen while the work is running. This is very useful in terms of "improvising", changing the inputs as the work progresses in real time. Also important to the piece is the stacking of various Tdefs (isee code example below): up to 4 of these voices at one time. This was in addition to the amplitude scaled "partials" within the Tdefs. Note that the "partials" here are sometimes complex waveforms and not always a basic sine wave. The aesthetic goal here was a to create "thickenedf"or "doubled" melodic lines (unison doubling or harmonic reinforcement through multiple voices).
(
~notes= [40, 44, 63,80, 47, 57,63,68,49,54,59,42,56,65,59,64,47,54,57,45,48,54,57,45,66,76,68,76,49,67,71,76];
~i=0;
//create an array of midi notes inside an environmental variable and then create various Task definitions that make use of the array.
Tdef(\i,{10.do{|x|(instrument:\zsine_1,freq:((~notes[~i-1]).midicps*(x+1)),amp: [1/(x+1)]*0.02,pan: 1.0.rand2;).play;0.001.wait}});
Tdef(\f,{1.do{|x|(instrument:\square2a,freq:((~notes[~i-1]).midicps),amp:[1/(x+1)]*0.1,pan: 1.0.rand2;).play;}});
Tdef(\a,{3.do{|x|(instrument:\zsine_1,freq: (~notes[~i-1]).midicps*(x+1),amp: [1/(x+1)]*0.02,pan: 1.0.rand2;).play;0.3.wait}});
Tdef(\j,{10.do{|x|(instrument:\zsine_1,freq:((~notes[~i-1]).midicps*(x+1)),amp: [1/(x+1)]*0.02,pan: 1.0.rand2;).play;0.001.wait}});
Tdef(\k,{1.do{|x|(instrument:\square2a,freq:((~notes[~i-1]).midicps*(x+1)),amp: [1/(x+1)]*0.004,pan: 1.0.rand2;).play}});
Tdef(\l,{1.do{|x|(instrument:\sine_double,freq:((~notes[~i-1]).midicps),amp: [1/(x+1)]*0.009,pan: 1.0.rand2;).play}});
Tdef(\c,{65.do{|x|(instrument:\zsine_10,freq:((~notes[~i-1]).midicps),amp: [1/(x+1)]*0.009,pan: 1.0.rand2;).play;0.02.wait}});
Tdef(\h,{4.do{|x|(instrument:\zsine_10,freq:((~notes[~i-1]+7).midicps*(x+1)),amp: [1/(x+1)]*0.02,pan: 1.0.rand2;).play;0.1.wait}});
Tdef(\x,{
loop{
while ( { ~i < ~notes.size }, {
~i = ~i + 1;
Tdef(\j).fork;
Tdef(\a).fork;
Tdef(\i).fork;
Tdef(\f).fork;
"1...".postln;
1.wait;
});
1.do{
~notes = [40, 44, 63,80, 47, 57,63,68,49,54,59,42,56,65,59,64,47,54,57,45,48,54,57,45,66,76,68,76,49,67,71,76];
~i=0;
while ( { ~i < ~notes.size }, {
~i = ~i + 1;
Tdef(\f).fork;
Tdef(\k).fork;
Tdef(\h).fork;
Tdef(\c).fork;
"2...".postln;
1.wait;
});};
20.do{
~notes= [40, 44, 63,80, 47, 57,63,68,49,54,59,42,56,65,59,64,47,54,57,45,48,54,57,45,66,76,68,76,49,67,71,76].scramble;
"3...".postln;
~notes[0].postln;
//Tdef(\a).fork;
Tdef(\f).fork;
Tdef(\k).fork;
Tdef(\i).fork;
1.wait;};
20.do{
~notes= [40, 44, 63,80, 47, 57,63,68,49,54,59,42,56,65,59,64,47,54,57,45,48,54,57,45,66,76,68,76,49,67,71,76].scramble;
"4...".postln;
~notes[0].postln;
//use sawtooth below....
Tdef(\i).fork;
Tdef(\a).fork;
Tdef(\l).fork;
0.5.wait;};
}}).play)
Note: the above code will not make any sound until the Synth Defs that the tasks make use of are added. Interested parties can insert their own definitions in all of the slots that require them.
The task definitions in the code are looped inside a further TDef. The different definitions are forked in order that they play concurrently. The environmental variable that holds the array of midi notes (~notes) is run inside a while loop. The variable "~i" is incremented at every pass and provides the fundamental of each of the "voices" (\i and \a are two such voices). The variable ~i has one subtracted from its starting point in the individual voices (to account for the increment within the while loop). The first set of frequencies for the voice "\i" are:
82.406889228217
164.81377845643
247.22066768465
329.62755691287
412.03444614109
494.4413353693
576.84822459752
659.25511382574
741.66200305396
824.06889228217
10 notes produced in rapid succession (basically concurrently when set at very low "wait" levels). The notes produced are scaled in amplitude to create a series of "partials". Taking into account that each of the notes is itself complex (more than a simple sine tone), and that these voices are then stacked, the result is a "reinforced" musical line.
Tdefs can be embedded or forked in another task - in the above snippet they are forked. The looping Tdef (\x) runs through the task of incrementing ~i and triggering the forked Tdefs every one second (thus the "1.wait" command). The different sections of the looping structure are controlled via a "while" loop set to the size of the ~notes array. The triggered Tdefs within the main loop run through their processes much more quickly, yet they are working in the same basic manner as the looping \x. Each of the Tdefs (\j, \k, \l etc.) sound out like a single tone - the "wait" command is set to very small values (e.g. 0.001 ms).
The "fundamental" that is set for each of the Synth Defs in the upper section (i.e those contained in \j, \i etc.) is called from within \x and set via the value of ~i. The layers of sound are built by using the argument of the Tdef within the frequency and amplitude variables of the Synth.
// a simplified version with a SynthDef included. Set the wait period in Tdef \a to different amounts.
SynthDef(\zsine_1,{arg freq = 440, amp = 1, pan = 0, env;
var sig;
env = EnvGen.ar(Env.perc, doneAction: 2);
sig = SinOsc.ar(freq);
Out.ar(0, sig*env*amp);
}).add;
//evaluate this section next and then the GUI at bottom of page.....
(
~amp = 1;
~notes = [40,44,63,80,47,57,63,68,49,54,59,42,56,65,59,64,47,54,57,45,48,54,57,45,66,76,68,76,49,67,71,76];
~i = 0;
//create an array of midi notes inside an environmental variable and then create various Task definitions that make use of the array.
Tdef(\a,{4.do{|x|(instrument:\zsine_1,freq:( ~notes[~i].midicps*(x+1)).postln,
amp:([1/(x+1)]*0.1)*~amp).play; 0.2.wait}});
//the wait amount in Tdef \a can be randomized (using .rand).
~tasking = Tdef(\x,{
loop{
while ( { ~i < ~notes.size }, {
~i.postln;
Tdef(\a).fork;
"playing...".postln;
1.wait;
~i = ~i + 1;
});
1.do{
~i=0;
"fin".postln;
1.wait;
};
}}))
//primitive GUI for testing purposes....:)
(
w = Window.new("Caverns",Rect(100,100,500,500));
g = Slider(w,Rect(100,100,150,10))
.minHeight_(20)
.thumbSize_(50)
.action_({ arg slider2;
~amp = slider2.value.postln;
});
b = Button(w, Rect(60, 20, 40, 30))
.states_([
["on", Color.black, Color.red],
["off", Color.white, Color.black]
])
.action_({ arg zz;
zz.value.postln;
if(zz.value == 1){~tasking.play;};
if(zz.value == 0){CmdPeriod.run;};
});
w.front;
)