Server.killAll // Execute this before booting the server Server.default.options.inDevice_("Scarlett 2i2 USB"); ( ~to = TouchOsc("192.168.0.209", 9000); ~patchdir = "~/Music/SuperCollider/Patches/granulator/"; ~to.button('/record', 1, { | v | ~bufrecorder.set("record", v) }); ~to.button('/clear', 0, { | v | if( v > 0, { var bl = ~to.v('/buflength'), ~to.v('/grains/speed')[0]; ~newbuffer = Buffer.alloc(s, s.sampleRate * bl, 1); ~granulator.set(\buffer, ~newbuffer); ~bufrecorder.set(\buffer, ~newbuffer); ~frippbuffer.free; ~frippbuffer = ~newbuffer; ~currentpos.set(\speed, sp / bl); }); }) ~to.slider('/mix', 0.25, TouchOSCScale(0, 1), { |v| ~bufrecorder.set(\mix, v) } ); ~to.slider('/grainamp', 0.5, TouchOSCScale(0, 1), { |v| ~granulator.set(\amp, v) } ); ~to.slider('/passthrough', 0.5, TouchOSCScale(0, 1), { |v| ~mixer.set(\passthrough, v) } ); // page 1: grains ~modes = [ [ ~grainsaw, ~grainsawb ], [ ~grainreverse, ~grainreverseb ], [ ~grainsin, ~grainsinb ], [ ~graintri, ~graintrib ], [ ~grainrand, ~grainrandb ] ] ~to.button('/grains/mode', 0, { |v| var mode = ~modes[v]; ~granulator.set(\posb, mode[1]); ~currentpos = mode[0]; ~currentpos.set(\speed, ~to.v('/grains/speed')[0]); }); ~to.slider('/grains/length', 4.0, TouchOSCScale(0.1, 10.0), {}); ~to.xy('/grains/speed', [ 0, 4 ], TouchOSCScale(-4, 4), TouchOSCScale(0, 10), { | v | var trate, qspeed; qspeed = 2.pow(v[0].floor); ~currentpos.set(\speed, qspeed / ~buflen); trate = 2.pow(v[1].floor) / ~buflen; ~granulator.set(\trate, trate); }); ~to.slider('/grains/size', 12, TouchOSCScale(0, 20),{ |v| ~granulator.set(\size, v) }); // Page 2: grainfx ~to.slider('/grainfx/blur', 0, TouchOSCScale(0, 1), { |v| ~granulator.set(\blur, v) }); ~to.slider('/grainfx/back', 1, { |v| ~granulator.set(\rate, if( v > 0, { 1 }, { -1}))}); ~to.slider('/grainfx/chorus', 0, { |v| ~granulator.set(\chorus, v) }); ~to.slider('/grainfx/dust', 0, { |v| ~granulator.set(\dust, v) }); // pitch gets quantised to octaves from 3 below to 3 above. // NOTE: the pitch TouchOSC control is -1 to 1, not 0 to 1 // min/max gets ignored because I'm overloading the ctrlset/get // TODO: fixme, // ~to.slider('/grainfx/pitch', -1, 1, 1, // { |self| ~granulator.set("rate", self.v) }, // { |self, ctrlv | self.v = 2.pow((ctrlv * 3).floor) }, // { |self| self.v.log2.floor / 3; } // ); ~to.slider('/fx/feedback', 0, TouchOSCScale(0, 0.25), { |v| ~bufrecorder.set(\feedback, v) } ); ~to.slider('/fx/freq', 10000, TouchOSCScale(200, 10000), { |v| ~granulator.set(\freq, v) } ); ~to.slider('/fx/rq', 0.3, TouchOSCScale(0.1, 1), { |v| ~granulator.set(\res, v) } ); ~to.slider('/fx/lfofreq', 0.5,TouchOSCScale(0.001, 4), { |v| ~lfo.set(\freq, v) } ); ~to.slider('/fx/lfoamp', 0, TouchOSCScale(0, 1), { |v| ~lfo.set(\amp, v) }); ~to.slider('/fx/fuzz', 0, TouchOSCScale(0, 1), { |v| ~fuzzbox.set(\distort, v) }); // Now the actual sound synthesis part // audio buses // recordb = input to bufrecorder // granulatorb = output from granulator ~usbinput = 2; ~recordb = Bus.audio(s, 1); ~granulatorb = Bus.audio(s, 2); // LFO bus and synth used to modulate the filter // TODO - have a couple of LFOs and an interface to patch them to // different settings ~lfob = Bus.control(s, 1); ~lfo = SynthDef( \lfo, { arg out=5, freq=1, amp=0; Out.kr(out, SinOsc.kr(freq, 0, amp)); } ).play(s, [\out, ~lfob, \freq, 1, \amp, 0]); // input filter chain // ~infilter = SynthDef( // \input_null, // {x // arg in = 2, out = 4; // Out.ar(out, In.ar(in)); // } // ).play(s, [\in, ~usbinput, \out, ~recordb]); ~fuzzbox = SynthDef( \fuzzbox, { arg in=2, out=4, distort=0.1, decay=0.999; var raw, cross, pf; raw = In.ar(in, 1).softclip; cross = CrossoverDistortion.ar(raw, 0.5, 0.5); pf = PeakFollower.ar(raw, decay); Out.ar(out, ((1 - distort) * raw) + (distort * pf * cross)); } ).play(s, [\in, ~usbinput, \out, ~recordb, \distort, 0 ]); // ~decimator = SynthDef( // \decimator, // { // arg in=2, out=4, modb, rate=10000, smooth=0.5; // var raw, mod, decimated; // raw = In.ar(in, 1); // mod = In.kr (modb, 1); // decimated = SmoothDecimator.ar(raw, rate + (0.2 * rate * mod), smooth); // Out.ar(out, decimated); // } // ).play(s, [\in, ~usbinput, \out, ~recordb, \modb, ~lfob, \rate, 10000 ]); // // ~localmax = SynthDef( // \localmax, // { // arg in=2, out=4, threshold=25; // var chain; // chain = FFT(LocalBuf(2048), In.ar(in, 1).distort); // chain = PV_LocalMax(chain, threshold); // Out.ar(out, IFFT.ar(chain)); // } // ).play(s, [\in, ~usbinput, \out, ~recordb, \threshold, 25 ]); // // ~scramble = SynthDef( // \scramble, // { // arg in=2, out=4, shift=1; // var chain; // chain = FFT(LocalBuf(2048), In.ar(in, 1).distort); // chain = PV_BinScramble(chain, 0.5, 0.2, Impulse.kr(shift)); // Out.ar(out, IFFT.ar(chain)); // } // ).play(s, [\in, ~usbinput, \out, ~recordb, \shift, 1 ]); // // buffer recorder ~frippbuffer = Buffer.alloc(s, s.sampleRate * ~sets.at(\buflength).v, 1); ~bufrecorder = SynthDef( \fripp_record, { arg in = 2, fb = 4, buffer = 0, mix = 0.25, record = 0.0, feedback = 0.0; var insig, fbsig; insig = record * In.ar(in, 1); fbsig = feedback * Mix.ar(In.ar(fb, 2)); RecordBuf.ar(insig + fbsig, buffer, 0, mix, 1 - mix, loop: 1) } ).play(s, [\in, ~recordb, \record, 1.0, \fb, ~granulatorb, \out, 0, \buffer, ~frippbuffer], \addToTail); // granulator playback modes // each of these is a control bus with a synth that drives the pattern // the granulator mode control switches between them // more ideas for modules: scramble - do a permutation of ABCDEFGH slots // todo - encapsulate these in a class ~grainsinb = Bus.control(s, 1); ~grainsin = SynthDef( \grainsin, { arg out=5, speed=1; Out.kr(out, 0.5 + SinOsc.kr(speed, 0, 0.5)); } ).play(s, [\out, ~grainsinb, \speed, 1]); ~grainsawb = Bus.control(s, 1); ~grainsaw = SynthDef( \grainsaw, { arg out=5, speed=1; Out.kr(out, 0.5 + LFSaw.kr(speed, 0, 0.5)); } ).play(s, [\out, ~grainsawb, \speed, 1]); ~grainreverseb = Bus.control(s, 1); ~grainreverse = SynthDef( \grainreverse, { arg out=5, speed=1; Out.kr(out, 0.5 - LFSaw.kr(speed, 0, 0.5)); } ).play(s, [\out, ~grainreverseb, \speed, 1]); ~graintrib = Bus.control(s, 1); ~graintri = SynthDef( \graintri, { arg out=5, speed=1; Out.kr(out, 0.5 + LFTri.kr(speed, 0, 0.5)); } ).play(s, [\out, ~graintrib, \speed, 1]); ~grainrandb = Bus.control(s, 1); ~grainrand = SynthDef( \grainsin, { arg out=5, speed=1; Out.kr(out, 0.5 + WhiteNoise.kr(0.5)); } ).play(s, [\out, ~grainrandb, \speed, 1]); // the main granulator synth // todo - different styles of trigger ~granulator = SynthDef( \grainsynth, { arg out=0, modb, trate=120, size=12, rate=1, posb=5, amp=1.0, freq=10000, rq=0.3, sweep=0.25, chorus=0.0, blur=0.0, dust = 0, buffer; var dur, blen, clk, chor, pos, pan, grains, filtfreq; dur = size / trate; clk = (Impulse.kr(trate) * (1 - dust)) + (Dust.kr(trate) * dust); chor = chorus * 2.pow((LFNoise0.kr(trate) + 0.5).floor) + (1 - chorus); blen = BufDur.kr(buffer); pos = Wrap.kr(In.kr(posb, 1) + WhiteNoise.kr(blur), 0, 1); pan = WhiteNoise.kr(1 - sweep) + (2 * sweep * (pos - 1)); filtfreq = (In.kr(modb, 1) * freq * 0.5) + freq; grains = TGrains.ar(2, clk, buffer, chor * rate, pos * blen, dur, pan, amp); Out.ar(out, RLPF.ar(grains, freq, rq)); // note that I've turned off freq lfo mod here } ).play(s, [\out, ~granulatorb, \buffer, ~frippbuffer, \posb, ~grainsawb, \modb, ~lfob]); ~mixerb = Bus.audio(s, 2); // this is what we will record from ~mixer = SynthDef( \mixer_synth, { arg in = 2, gbus = 4, out = 0, amp = 1.0, passthrough = 0.0; //Out.ar(out, In.ar(gbus, 2)); Out.ar(out, (amp * In.ar(gbus, 2)) + (passthrough * In.ar(~recordb, 1) ! 2)); } ).play(s, [\in, 2, \out, ~mixerb, \gbus, ~granulatorb, \amp, 1.0, \passthrough, 0.0], \addToTail); // ~monitor = SynthDef( \monitor_synth, { arg in=2, out=0; Out.ar(out, In.ar(in, 2)) } ).play(s, [\in, ~mixerb, \out, 0 ], \addToTail); )