InstrGateSpawner
superclass: InstrSpawner, Patch
see InstrSpawner
instr - as per Patch and InstrSpawner
the Synth is expected to end itself via EnvGen doneAction or similar.
args - as per InstrSpawner
each arg that is a stream (its rate returns \stream) is passed the InstrGateSpawner as an
arg. Two useful things can be done with this: asking the .beat and the current .delta
till the next event.
delta - a float, function or stream in beats
it stops when any arg or the delta returns nil.
(
Tempo.bpm = 140.0;
InstrGateSpawner({ arg freq=440,rq=0.1,width=0.5,fenv,fenvmod=1000,envadsr,dur=0.1,tempo;
var gate;
gate = Trig1.kr(1.0,dur / tempo);
RLPF.ar(
VarSaw.ar(
freq,
0.0,
width
),
EnvGen.kr(fenv,gate,levelScale: fenvmod),
rq)
* EnvGen.kr(envadsr,gate,0.2, doneAction: 2)
},[
Pseq((70 + [ 0, 1, 5, 6, 10 ]).midicps ,inf) *
Prand(2 ** [0,-1,2,-2,-3],inf), // octave shifts
0.2,
Patch({ LFTri.kr(0.1,[0.0,0.5],0.5,0.5) }),
Env.adsr(0.3,sustainLevel: 0.4),
Pbrown(2000,10000,100),
Env.adsr(releaseTime: 0.5),
// dur uses a Pfunc to ask the delta till the next event
Pfunc({ arg igs; (igs.delta * 0.9) }),
TempoPlayer.new
],
0.25
).play
)
(
InstrGateSpawner({ arg freq=440,rq=0.1,width=0.5,fenv,fenvmod=1000,envadsr,dur=0.1,tempo;
var gate;
gate = Trig1.kr(1.0,dur / tempo);
RLPF.ar(
VarSaw.ar(
freq,
0.0,
width
),
EnvGen.kr(fenv,gate,levelScale: fenvmod),
rq)
* EnvGen.kr(envadsr,gate,0.15, doneAction: 2)
},[
PdurStutter(
Pseq(#[1,1,1,1,1,2,2,2,2,2,3,3,3,3,4,4,0,4,4],inf),
Pseq((70 + [ 0, 1, 4, 8, 10, 15 ]).midicps ,inf)
),
0.2,
Patch({ LFTri.kr(FSinOsc.kr(1.0).range(0.1,3.0),[0.0,0.5],0.5,0.5) }),
Env.adsr(0.3,sustainLevel: 0.1),
Pbrown(6000,10000,100),
Env.adsr(releaseTime: 0.5),
// dur uses a Pfunc to ask the delta till the next event
Pfunc({ arg igs; (igs.delta * rrand(0.9,2.0)) }),
TempoPlayer.new
],
Prand([
Pn(0.125,16),
Pn(0.25,8),
Prand([0.5,0.25,0.125,0.3],4),
0.5,0.75,2.0],inf) // in beats
).play
)
Chord changes
however crazy the durations get, the current chord and pitch is determined by indexing into
a chord progression using the current beat: igs.beat
(
InstrGateSpawner({ arg freq,rq,width,fenv,fenvmod,envperc,dur,tempo;
var gate;
gate = Trig1.kr(1.0,dur / tempo);
RLPF.ar(
Pulse.ar(
freq,
width
),
EnvGen.kr(fenv,gate,levelScale: fenvmod),
rq)
* EnvGen.kr(envperc,gate,0.3, doneAction: 2)
},[
// using the beat to index into chord changes
Pfunc({ arg igs;
(
// choose degree convert it via a scale to a note number
( [1,3,5].choose.degreeToKey([ 0, 1, 4, 8, 12 ]) )
// choose an octave
+ 30 + [0,11,23,35].choose
// 1 4 5 1 chord progression
+ [1,4,5,1].wrapAt(igs.beat div: 16)
).midicps
}),
// rq input is a mono kr rate patch
Patch({ LFTri.kr(FSinOsc.kr(0.05).range(0.01,1.0),0.0).range(0.01,0.4) }),
//note that the patch has expanded to stereo because the width input is a stereo kr rate Patch :
Patch({
LFTri.kr(FSinOsc.kr(0.05).range(0.01,1.0),[0.0,0.5],0.5,0.5)
}),
Env.adsr(0.2,0.01,0.1),
12000,
Env.adsr(0.01,0.1,0.2, 0.05),
Pfunc({ arg igs; (igs.delta * rrand(0.1,5.0)) }),
TempoPlayer.new
],
PdurStutter(
Pseq([1,1,1,1,2,2,2,2,2,2,2,2,4,4,3,3,0,2,1,1,1,1],inf),
0.25
)
).play
)
anything that returns a rate of \stream (a Pattern or Function) will cause an .ir rate input to be created for the synth.
// if you click on the sample's filename, you can select a new sample while playing
// and it will load it.
// note that the old tempo settings persist until the end of the bar
(
InstrGateSpawner({ arg sample,start=0,dur,env,tempo;
var gate;
gate = Trig1.kr(1.0,dur / tempo);
PlayBuf.ar(
sample.numChannels,
sample.bufnumIr,
sample.pchRatioKr,
1,
start * sample.beatsizeIr,
1
) * EnvGen.kr(env,gate, doneAction: 2)
},[
Sample("a11wlk01.wav"),
0,//Pseq([0,1,2,3],inf),
4.0,
Env.adsr(0.02,releaseTime:0.01)
],4.0).gui
)
// basic time stretching
// change the (global) tempo using the tempo control at the top
// you can also change the beats setting on the sample
(
var sample;
InstrGateSpawner({ arg sample,start=0,dur,pchRatio,env,tempo;
var gate;
gate = Trig1.kr(1.0,dur / tempo);
PlayBuf.ar(
sample.numChannels,
sample.bufnumIr,
sample.bufRateScaleIr * pchRatio,
1,
start,
1
) * EnvGen.kr(env,gate, doneAction: 2)
},[
sample = Sample("a11wlk01.wav"),
Pfunc({ arg igs;
(igs.beat * sample.beatsize).wrap(0,sample.end) // in samples
}),
0.125,
KrNumberEditor(1.0,[-4.0,4.0]),
Env.adsr(0.02,releaseTime:0.1)
],0.125).gui
)
There is a large client side performance hit to build the sample's waveform display
during which the grain commands stop getting sent. (someone will make a better
waveform display some day...)
There is also a problem (currently) on the server whereby it can't accept the OSC commands
while it is dealing with loading the sample. You can use ScurryableInstrGateSpawner-scurry,
a subclass of InstrGateSpawner that can scurry ahead of time, sending OSC bundles
to keep the server's synthesis thread occupied while the thread that does the soundfile
loading is busy. This issue will probably be resolved on the server.
See [Interface] so you can design guis with no sample display or interfaces with no gui at all.
when using your own guis :
to use a slider:
Pfunc({ numberEditor.value })
or
AsStream( NumberEditor.new )
or make your own gui and poll the slider:
Pfunc({ slider.value })
A metro-gnome
(
var nome,layout;
nome = InstrGateSpawner({ arg freq,amp;
Decay2.ar(
Impulse.ar(0.0), 0.01,0.11,
SinOsc.ar( freq, 0, amp )
) * EnvGen.kr(Env.perc,doneAction: 2)
},[
Pseq([ 750, 500, 300, 500, 750, 500, 400, 500, 750, 500, 400, 500, 750, 500, 400, 500 ],inf),
Pseq([1,0.25,0.5,0.25,0.75,0.25,0.5,0.25,0.75,0.25,0.5,0.25,0.75,0.25,0.5,0.25] * 0.1,inf)
],1.0);
layout = FlowView.new;
ToggleButton(layout,"Gnome",{
if(nome.isPlaying.not,{ nome.play(atTime: 1) })
},{
if(nome.isPlaying,{ nome.stop })
},minWidth: 250);
)
Compared to Pbind
Its a different approach and has plusses and minuses.
Pbind streams each item, whether it is going to change or not (a float will not change, an Env will not change).
so InstrGateSpawner has much more simple/efficient event handling,
not using an environment
streamed items are placed directly into the OSC message to be sent
only using the features that are needed for the ugenFunc
no lag, offsetChannel etc.
But as computers and SC get faster, this speed advantage is less significant.
allows easy patching of players into the inputs
allows easy patching of Samples and allocates them seamlessly.
beat is passed into each stream item
this solves a common 'sync' problem with event stream's note-by-note streaming.
but you could use the same solution in a Pfunc and just query the clock to find out the current beat.
the usual flexibility of players with regards to saving
cons:
can't do Pseq([ Pbind, Pbind ] ) etc.
it isn't a pattern, doesn't end with a nil, so you can't sequence sections with it.
but it is possible to use CropPlayer and Pseq or PlayerSeqTrack.
can't use pattern filters
can't switch ugenFuncs dynamically
compare with the first example from Pbind help
(
SynthDef(\cfstring1.postln, { arg i_out, freq = 360, gate = 1, pan, amp=0.1;
var out, eg, fc, osc, a, b, w;
fc = LinExp.kr(LFNoise1.kr(Rand(0.25,0.4)), -1,1,500,2000);
osc = Mix.fill(8, { LFSaw.ar(freq * [Rand(0.99,1.01),Rand(0.99,1.01)], 0, amp) }).distort * 0.2;
eg = EnvGen.kr(Env.asr(1,1,1), gate, doneAction:2);
out = eg * RLPF.ar(osc, fc, 0.1);
#a, b = out;
Out.ar(i_out, Mix.ar(PanAz.ar(4, [a, b], [pan, pan+0.3])));
}).load(s);
SynthDescLib.global.read;
)
e = Pbind(\degree, Pwhite(0,12), \dur, 0.2, \instrument, \cfstring1).play; // returns an EventStream
(
InstrGateSpawner({ arg freq = 360, pan=0, amp=0.1,dur=0.2,tempo;
var out, eg, fc, osc, a, b, w,gate;
gate = Trig.kr(1.0,dur / tempo);
fc = LinExp.kr(LFNoise1.kr(Rand(0.25,0.4)), -1,1,500,2000);
osc = Mix.fill(8, { LFSaw.ar(freq * [Rand(0.99,1.01),Rand(0.99,1.01)], 0, amp) }).distort * 0.2;
eg = EnvGen.kr(Env.asr(1,1,1), gate, doneAction:2);
out = eg * RLPF.ar(osc, fc, 0.1);
#a, b = out;
Mix.ar(PanAz.ar(4, [a, b], [pan, pan+0.3]))
},[
(Pwhite(0,12).degreeToKey([0, 2, 4, 5, 7, 9, 11] + (5*12)) ).midicps,
0,
0.1,
0.2,
TempoPlayer.new
],
0.2
).play
)
the server usage is identical as should be expected, but the client side (using top) is about half the cpu.
on another day, the test shows equal usage !
and then a minute later Pbind does show more usage.
top is not really a good measure.
simply because there is a lot less code running, InstrGateSpawner must be faster.
they both have advantages in their way of working, and can both be used side by side.
a compile-string-saveable way to express the Pfunc
ValueAndAsStream(
Patch([\ampPatterns,\accent], [ accentArray ])
)
Instr([\ampPatterns,\accent],
{ arg accentArray;
Pfunc({ arg beat; accentArray.at((beat % 4).asInteger) })
});
the ValueAndAsStream values the Patch which returns the Pfunc.