1 title:: 16. Sequencing with Patterns
2 summary:: Getting Started With SuperCollider
3 categories:: Tutorials>Getting-Started
4 related:: Tutorials/Getting-Started/00-Getting-Started-With-SC
6 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.
8 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.
10 For example, we used this routine to output MIDI note numbers to play a couple of phrases from 'Over the Rainbow.'
14 [60, 72, 71, 67, 69, 71, 72, 60, 69, 67].do({ |midi| midi.yield });
17 while { (m = r.next).notNil } { m.postln };
20 With patterns, we can express the idea of a stream returning the same values, but more clearly and concisely. Because we don't 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).
22 link::Classes/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).
25 p = Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1);
27 while { (m = r.next).notNil } { m.postln };
30 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.
32 Thus the 'Over the Rainbow' example could be rewritten, with less clutter:
37 midi = Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1).asStream;
38 dur = Pseq([2, 2, 1, 0.5, 0.5, 1, 1, 2, 2, 3], 1).asStream;
40 SynthDef(\smooth, { |freq = 440, sustain = 1, amp = 0.5|
42 sig = SinOsc.ar(freq, 0, amp) * EnvGen.kr(Env.linen(0.05, sustain, 0.1), doneAction: 2);
52 Synth(\smooth, [freq: midi.next.midicps, sustain: delta]);
55 }).play(quant: TempoClock.default.beats + 1.0);
59 section::What else can patterns do?
61 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 you'll come back to again and again.
63 Many patterns take lists of values and return them in some order.
66 ## Pseq(list, repeats, offset) || return the list's values in order
67 ## Pshuf(list, repeats) || scramble the list into random order
68 ## Prand(list, repeats) || choose from the list's values randomly
69 ## Pxrand(list, repeats) || choose randomly, but never return the same list item twice in a row
70 ## Pwrand(list, weights, repeats) || like Prand, but chooses values according to a list of probabilities/weights
73 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.
76 ## Pseries(start, step, length) || arithmetic series, e.g., 1, 2, 3, 4, 5
77 ## Pgeom(start, grow, length) || geometric series, e.g., 1, 2, 4, 8, 16
78 ## Pwhite(lo, hi, length) || random number generator, uses rrand(lo, hi) -- equal distribution
79 ## Pexprand(lo, hi, length) || random number generator, uses exprand(lo, hi) -- exponential distribution
82 Other patterns modify the output of value patterns. These are called FilterPatterns.
85 ## Pn(pattern, repeats) || repeat the pattern as many times as repeats indicates
86 ## Pstutter(n, pattern) || repeat individual values from a pattern emphasis::n:: times. emphasis::n:: may be a numeric pattern itself.
89 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.
92 p = Pwhite(0.0, Pseries(0.01, 0.01, inf), 100).asStream;
93 // .all pulls from the stream until it returns nil
94 // obviously you don't want to do this for an 'inf' length stream!
98 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.
101 p = Pn(Pshuf([1, 2, 3, 4, 5], 1), inf).asStream;
102 p.nextN(15); // get 15 values from the pattern's stream
105 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.
107 section::Playing notes with a pattern: Pbind
109 Not only can patterns produce data for notes, but they can also play the notes themselves. 'Over the Rainbow' again.
113 SynthDef(\smooth, { |freq = 440, sustain = 1, amp = 0.5|
115 sig = SinOsc.ar(freq, 0, amp) * EnvGen.kr(Env.linen(0.05, sustain, 0.1), doneAction: 2);
122 // the name of the SynthDef to use for each note
123 \instrument, \smooth,
124 // MIDI note numbers -- converted automatically to Hz
125 \midinote, Pseq([60, 72, 71, 67, 69, 71, 72, 60, 69, 67], 1),
127 \dur, Pseq([2, 2, 1, 0.5, 0.5, 1, 1, 2, 2, 3], 1)
132 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.
134 The link::Overviews/Streams:: documentation explains how all of this works in detail. The high-level overview goes like this:
136 - The Pbind pattern generates Event objects, which contain names and values describing how the note is supposed to sound.
138 - 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.
140 - 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.
142 - 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 strong::add:: method.
144 - The delta value in the event tells SuperCollider how long to wait until playing the next event.
146 An introductory tutorial cannot cover all the possibilities. Learning a set of core pattern classes is important; the link::Tutorials/A-Practical-Guide/PG_01_Introduction##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.
148 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.
150 Don't be intimidated by the bassline pattern. At a higher level, it reduces to strong::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 doesn't 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.
154 SynthDef(\bass, { |freq = 440, gate = 1, amp = 0.5, slideTime = 0.17, ffreq = 1100, width = 0.15,
155 detune = 1.005, preamp = 4|
157 env = Env.adsr(0.01, 0.3, 0.4, 0.1);
158 freq = Lag.kr(freq, slideTime);
159 sig = Mix(VarSaw.ar([freq, freq * detune], 0, width, preamp)).distort * amp
160 * EnvGen.kr(env, gate, doneAction: 2);
161 sig = LPF.ar(sig, ffreq);
165 TempoClock.default.tempo = 132/60;
171 \dur, Pseq([0.75, 0.25, 0.25, 0.25, 0.5], 1),
172 \legato, Pseq([0.9, 0.3, 0.3, 0.3, 0.3], 1),
173 \amp, 0.5, \detune, 1.005
176 \midinote, Pseq([36, 48, 36], 1),
177 \dur, Pseq([0.25, 0.25, 0.5], 1),
178 \amp, 0.5, \detune, 1.005
181 \midinote, Pseq([36, 42, 41, 33], 1),
182 \dur, Pseq([0.25, 0.25, 0.25, 0.75], 1),
183 \amp, 0.5, \detune, 1.005
186 \midinote, Pseq([36, 39, 36, 42], 1),
187 \dur, Pseq([0.25, 0.5, 0.25, 0.5], 1),
188 \amp, 0.5, \detune, 1.005
190 ], inf).play(quant: 1);
193 // totally cheesy, but who could resist?
195 SynthDef(\kik, { |preamp = 1, amp = 1|
196 var freq = EnvGen.kr(Env([400, 66], [0.08], -3)),
197 sig = SinOsc.ar(freq, 0.5pi, preamp).distort * amp
198 * EnvGen.kr(Env([0, 1, 0.8, 0], [0.01, 0.1, 0.2]), doneAction: 2);
203 // what do you anticipate '\delta, 1' will do?
204 k = Pbind(\instrument, \kik, \delta, 1, \preamp, 4.5, \amp, 0.32).play(quant: 1);
211 section::Further reading
213 link::Overviews/Streams::, link::Tutorials/Streams-Patterns-Events1##Streams-Patterns-Events::, link::Tutorials/A-Practical-Guide/PG_01_Introduction##Practical Guide to Patterns::
215 section::Suggested exercises
218 ## Choose a familiar tune and write a Pbind for it, using any synthdef you like.
219 ## Add as many phrases as you wish to the bassline sequence in the previous example.
224 This document is part of the tutorial strong::Getting Started With SuperCollider::.
226 Click here to return to the table of Contents: link::Tutorials/Getting-Started/00-Getting-Started-With-SC::