Sequencing with Patterns The previous section demonstrated how - TopicsExpress



          

Sequencing with Patterns The previous section demonstrated how to use data routines to generate sequences of synthesis parameters. However, writing a routine with explicit yields is not a very convenient syntax. Since this is an essential part of creating computer music, we really need an easier way. Patterns greatly simplify the use of data streams. A pattern is essentially a factory for a stream. The pattern objects includes the data you want to come out of the stream, and the type of pattern determines how the data will be streamed. For example, we used this routine to output MIDI note numbers to play a couple of phrases from Over the Rainbow. r = Routine({ [60, 72, 71, 67, 69, 71, 72, 60, 69, 67].do({ |midi| midi.yield }); }); while { (m = r.next).notNil } { m.postln }; With patterns, we can express the idea of a stream returning the same values, but more clearly and concisely. Because we dont have to write the yield explicitly, there is nothing in the pattern to distract attention from the data (which are the real concern in composition). Pseq (Pattern-sequence) means simply to spit out the values in the array one by one, in order, as many times as the second argument (here, only once). p = Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1); r = p.asStream; while { (m = r.next).notNil } { m.postln }; Note that the Pseq is not streamable by itself, but it creates a stream (Routine) when you call asStream on it. This routine can then be used exactly like to any other routine -- the while loop used to read out the stream values is exactly the same for both, even though they are written differently. Thus the Over the Rainbow example could be rewritten, with less clutter: ( var midi, dur; midi = Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1).asStream; dur = Pseq([2, 2, 1, 0.5, 0.5, 1, 1, 2, 2, 3], 1).asStream; SynthDef(\smooth, { |freq = 440, sustain = 1, amp = 0.5| var sig; sig = SinOsc.ar(freq, 0, amp) * EnvGen.kr(Env.linen(0.05, sustain, 0.1), doneAction: 2); Out.ar(0, sig ! 2) }).send(s); r = Task({ var delta; while { delta = dur.next; delta.notNil } { Synth(\smooth, [freq: midi.next.midicps, sustain: delta]); delta.yield; } }).play(quant: TempoClock.default.beats + 1.0); ) What else can patterns do? The SuperCollider pattern library is large (over 120 classes, not including extension libraries), obviously beyond the scope of a tutorial to cover in depth. But some patterns youll come back to again and again. Many patterns take lists of values and return them in some order. Pseq(list, repeats, offset) -- return the lists values in order Pshuf(list, repeats) -- scramble the list into random order Prand(list, repeats) -- choose from the lists values randomly Pxrand(list, repeats) -- choose randomly, but never return the same list item twice in a row Pwrand(list, weights, repeats) -- like Prand, but chooses values according to a list of probabilities/weights Other patterns generate values according to various parameters. In addition to these basic patterns, there is a whole set of random number generators that produce specific distributions, and also chaotic functions. Pseries(start, step, length) -- arithmetic series, e.g., 1, 2, 3, 4, 5 Pgeom(start, grow, length) -- geometric series, e.g., 1, 2, 4, 8, 16 Pwhite(lo, hi, length) -- random number generator, uses rrand(lo, hi) -- equal distribution Pexprand(lo, hi, length) -- random number generator, uses exprand(lo, hi) -- exponential distribution Other patterns modify the output of value patterns. These are called FilterPatterns. Pn(pattern, repeats) -- repeat the pattern as many times as repeats indicates Pstutter(n, pattern) -- repeat individual values from a pattern n times. n may be a numeric pattern itself. You can use patterns inside of other patterns. Here, we generate random numbers over a gradually increasing range. The upper bound on the random number generator is a stream that starts at 0.01, then proceeds to 0.02, 0.03 and so on, as the plot shows clearly. p = Pwhite(0.0, Pseries(0.01, 0.01, inf), 100).asStream; // .all pulls from the stream until it returns nil // obviously you dont want to do this for an inf length stream! p.all.plot; Or, for another example, if you want to order a set of numbers randomly so that all numbers come out before a new order is chosen, use Pn to repeat a Pshuf. p = Pn(Pshuf([1, 2, 3, 4, 5], 1), inf).asStream; p.nextN(15); // get 15 values from the patterns stream This is just a taste, meant to illustrate the kinds of flexibility you can get with patterns. As with any rich and adaptable structure, the best way is to start with simple cases and gradually extend into more complicated setups. Playing notes with a pattern: Pbind Not only can patterns produce data for notes, but they can also play the notes themselves. Over the Rainbow again. ( SynthDef(\smooth, { |freq = 440, sustain = 1, amp = 0.5| var sig; sig = SinOsc.ar(freq, 0, amp) * EnvGen.kr(Env.linen(0.05, sustain, 0.1), doneAction: 2); Out.ar(0, sig ! 2) }).memStore; ) ( p = Pbind( // the name of the SynthDef to use for each note \instrument, \smooth, // MIDI note numbers -- converted automatically to Hz \midinote, Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1), // rhythmic values \dur, Pseq([2, 2, 1, 0.5, 0.5, 1, 1, 2, 2, 3], 1) ).play; ) The first thing to notice is how short, concise and clean the syntax is. Nothing is extra; it focuses all your attention on what is supposed to play and minimizes distractions from program logic. The Streams documentation explains how all of this works in detail. The high-level overview goes like this: - The Pbind pattern generates Event objects, which contain names and values describing how the note is supposed to sound. - It does this by reading through the name, pattern pairs, getting values from each pattern stream in turn and adding the values to the result Event. - Then the event is played. It interprets the values according to a set of defaults and rules encoded within the event prototype and performs an action in response. The default action is to play a new synth on the server. You can choose from several other actions defined in the default event prototype, which are documented in the Streams series of help files. - To play the synth, the event needs to know which values to pass as arguments to the server. SuperCollider can store information about a synthdef into a library of synthdef descriptions using the store and memStore methods. .store corresponds to .load(s), saving the synthdef file on disk and reading argument information from the disk file. .memStore is like .send(s); the synthdef is sent directly to the server with no disk file, and the description library extracts the required information from the synthdef in memory. - The delta value in the event tells SuperCollider how long to wait until playing the next event. An introductory tutorial cannot cover all the possibilities. Learning a set of core pattern classes is important; the Practical Guide to Patterns help file series is a more comprehensive introduction. Pattern manipulations, and ways to combine or nest patterns, open up the field to nearly every compositional need. For example, we can generate a rhythmic (but not necessarily metric) bassline by choosing randomly from a set of Pbind sequences. (Some of these will use Pmono, which is a variant of Pbind designed to play monophonic synth lines.) While this is a bigger block of code, its structure is fairly simple and it brings together several concepts introduced in the sequencing tutorials. Note that the quant argument to play is used to keep a couple of distinct sequences together on the beat. Dont be intimidated by the bassline pattern. At a higher level, it reduces to Pxrand([a, b, c, d], inf), which simply chooses items randomly without repeating any of them twice in a row. It happens that each item is an event pattern that plays a series of notes, but this doesnt matter to Pxrand. It just chooses an item, plays it through to the end, and then chooses the next, and so forth. Viewed this way, the pattern is an elegant expression of the idea of selecting phrases. The code representation is straightforward to relate to a musical conception. ( SynthDef(\bass, { |freq = 440, gate = 1, amp = 0.5, slideTime = 0.17, ffreq = 1100, width = 0.15, detune = 1.005, preamp = 4| var sig, env = Env.adsr(0.01, 0.3, 0.4, 0.1); freq = Lag.kr(freq, slideTime); sig = Mix(VarSaw.ar([freq, freq * detune], 0, width, preamp)).distort * amp * EnvGen.kr(env, gate, doneAction: 2); sig = LPF.ar(sig, ffreq); Out.ar(0, sig ! 2) }).memStore; TempoClock.default.tempo = 132/60; p = Pxrand([ Pbind( \instrument, \bass, \midinote, 36, \dur, Pseq([0.75, 0.25, 0.25, 0.25, 0.5], 1), \legato, Pseq([0.9, 0.3, 0.3, 0.3, 0.3], 1), \amp, 0.5, \detune, 1.005 ), Pmono(\bass, \midinote, Pseq([36, 48, 36], 1), \dur, Pseq([0.25, 0.25, 0.5], 1), \amp, 0.5, \detune, 1.005 ), Pmono(\bass, \midinote, Pseq([36, 42, 41, 33], 1), \dur, Pseq([0.25, 0.25, 0.25, 0.75], 1), \amp, 0.5, \detune, 1.005 ), Pmono(\bass, \midinote, Pseq([36, 39, 36, 42], 1), \dur, Pseq([0.25, 0.5, 0.25, 0.5], 1), \amp, 0.5, \detune, 1.005 ) ], inf).play(quant: 1); ) // totally cheesy, but who could resist? ( SynthDef(\kik, { |preamp = 1, amp = 1| var freq = EnvGen.kr(Env([400, 66], [0.08], -3)), sig = SinOsc.ar(freq, 0.5pi, preamp).distort * amp * EnvGen.kr(Env([0, 1, 0.8, 0], [0.01, 0.1, 0.2]), doneAction: 2); Out.ar(0, sig ! 2); }).memStore; // before you play: // what do you anticipate \delta, 1 will do? k = Pbind(\instrument, \kik, \delta, 1, \preamp, 4.5, \amp, 0.32).play(quant: 1); ) p.stop; k.stop; Further reading: Streams Streams-Patterns-Events Practical Guide to Patterns Suggested exercises: - Choose a familiar tune and write a Pbind for it, using any synthdef you like. - Add as many phrases as you wish to the bassline sequence in the previous example. ____________________ This document is part of the tutorial Getting Started With SuperCollider. Click here to return to the table of Contents: Getting Started With SC
Posted on: Sun, 23 Mar 2014 23:02:52 +0000

Trending Topics




Welp G.morning fb fam friendz an fakes, woke up with smiles on my
Bankruptcy Attorney Asheville Nc
Cyber Monday Deal E-Z Ink (TM) Remanufactured Black Ink Cartridge
Hi ! Guys, Just Get In Line And Get Paid. Yes, Stiforp Is Doing
Win 1 of 5 double passes to our movie of the week: Michael
You can now download the Membership Form Send all your
Compared to other foods, grapefruits are relatively new and
( Im looking for a bully) I need the help f some individuals here
ਹੁਕਮਨਾਮਾ ਸ਼੍ਰੀ ਦਰਬਾਰ

Recently Viewed Topics




© 2015