1 title:: PG_04_Words_to_Phrases
2 summary:: Nesting patterns, arranging music in terms of phrases
3 related:: Tutorials/A-Practical-Guide/PG_03_What_Is_Pbind, Tutorials/A-Practical-Guide/PG_05_Math_on_Patterns
4 categories:: Streams-Patterns-Events>A-Practical-Guide
6 section::From words to phrases
8 A single pattern by itself is not so exciting. But patterns can be used together to get more complex results.
10 subsection::Patterns within list patterns
12 We saw list patterns ( link::Classes/Pseq::, link::Classes/Prand::, etc.) that returned numbers from a preset list, either in the order given or rearranged randomly. The list may also include other patterns. When a list pattern encounters another pattern in its list, the inner pattern is emphasis::embedded:: into the stream. That is, the inner pattern takes over until it runs out of values to return. Then, control returns to the outer list pattern. This is like calling a function in the middle of another function.
14 There is no preset limit to the number of levels of embedding.
16 If a single pattern is like a word, a list pattern that uses other patterns could be more like a sentence or phrase. You can alternate between different behaviors, either in a predictable order as in the example below, or randomly by using one of the random-order list patterns.
19 // Scale segments, in the sequence: up, up, down (repeat)
21 TempoClock.default.tempo = 1;
24 Pseries({ rrand(0, 7) }, 1, { rrand(4, 8) }), // up (step = 1)
25 Pseries({ rrand(0, 7) }, 1, { rrand(4, 8) }), // up (step = 1)
26 Pseries({ rrand(7, 14) }, -1, { rrand(4, 8) }) // down (step = -1)
35 But it gets even more fun -- list patterns don't care whether they're enclosing value patterns (as in the previous example) or event patterns. That means you can write a set of Pbind-style patterns, each one representing a phrase, and string them together. This next example is longer, but that's only because of a larger number of phrase patterns. The structure is very simple, though: code::Pxrand([Pbind(), Pmono(), Pmono()...], inf):: . Some of the phrases are written with Pmono to slide between notes.
39 SynthDef(\bass, { |freq = 440, gate = 1, amp = 0.5, slideTime = 0.17, ffreq = 1100, width = 0.15,
40 detune = 1.005, preamp = 4|
42 env = Env.adsr(0.01, 0.3, 0.4, 0.1);
43 freq = Lag.kr(freq, slideTime);
44 sig = Mix(VarSaw.ar([freq, freq * detune], 0, width, preamp)).distort * amp
45 * EnvGen.kr(env, gate, doneAction: 2);
46 sig = LPF.ar(sig, ffreq);
52 TempoClock.default.tempo = 132/60;
54 Pbind( // repeated notes
57 \dur, Pseq([0.75, 0.25, 0.25, 0.25, 0.5], 1),
58 \legato, Pseq([0.9, 0.3, 0.3, 0.3, 0.3], 1),
59 \amp, 0.5, \detune, 1.005
61 Pmono(\bass, // octave jump
62 \midinote, Pseq([36, 48, 36], 1),
63 \dur, Pseq([0.25, 0.25, 0.5], 1),
64 \amp, 0.5, \detune, 1.005
66 Pmono(\bass, // tritone jump
67 \midinote, Pseq([36, 42, 41, 33], 1),
68 \dur, Pseq([0.25, 0.25, 0.25, 0.75], 1),
69 \amp, 0.5, \detune, 1.005
71 Pmono(\bass, // diminished triad
72 \midinote, Pseq([36, 39, 36, 42], 1),
73 \dur, Pseq([0.25, 0.5, 0.25, 0.5], 1),
74 \amp, 0.5, \detune, 1.005
76 ], inf).play(quant: 1);
82 strong::Shortcut notation:: : Just like you can concatenate arrays with ++, you can also concatenate patterns the same way. Writing code::pattern1 ++ pattern2:: is the same as writing code::Pseq([pattern1, pattern2], 1):: .
84 subsection::Some ways to string together patterns
88 ## Sequentially || Each sub-pattern follows the next in the same order every time. Use link::Classes/Pseq:: or link::Classes/Pser::.
90 ## Randomized order || Sub-patterns in completely random order ( link::Classes/Prand:: ), random order with no repeats ( link::Classes/Pxrand:: ), or random order according to a set of probabilities ( link::Classes/Pwrand:: ). link::Classes/Pshuf:: creates one random ordering and uses it repeatedly.
92 ## Direct array indexing || Patterns can be chosen in arbitrary order by index. This gives you more control than Pwrand. Both link::Classes/Pindex:: and link::Classes/Pswitch:: can be used for this.
95 // scale degree segments, every fifth choice is odd-numbered only (descending)
98 scaleSegments = Array.fill(n, { |i|
100 Pseries(11, -1, rrand(5, 10))
102 Pseries(rrand(-4, 4), 1, i+2)
106 TempoClock.default.tempo = 1;
108 \degree, Pswitch(scaleSegments, Pseq([Pwhite(0, n-1, 4), Pwhite(0, n-1, 1).select(_.odd)], inf)),
116 ## Finite state machine (Pfsm, Pdfsm) || A finite state machine is a way of associating an item with its possible successors. It is closer to a "grammar" than purely random selection. link::Classes/Pfsm:: defines a finite state machine as a set of possible "entry points," followed by a list of the possible "states" of the machine and, for each state, a list of the possible states that may follow the current state. States may be single values or patterns, meaning that phrases can be linked to other phrases that "make sense" in succession (and unwanted transitions can be prevented).
118 If this sounds a bit like a Markov chain, that's because the Pfsm implementation is a special case of a Markov chain where there is an equal probability of choosing the next state from the valid successors. In a Markov chain, the probabilities are weighted according to analysis of a real-world data stream.
120 The Pfsm help file includes very good examples of organizing single values and pattern phrases. Also see link::Tutorials/A-Practical-Guide/PG_Cookbook06_Phrase_Network:: for an application of Pfsm to generate a corny jazz solo.
122 The name link::Classes/Pdfsm:: stands for "deterministic finite state machine," where there is no random selection.
124 - strong::Third-party extension alert:: : A good Markov chain implementation for SuperCollider exists in the MathLib quark.
127 subsection::Library of named sub-patterns
129 One very effective way to manage phrases is to make a library, or more precisely link::Classes/Dictionary::, of sub-patterns, and then call them up one at a time. link::Classes/Psym:: is the pattern to do this. The advantage here is that you can store the phrases in a separate place, while the pattern that you actually play is much simpler and describes the musical intent at a much higher level.
132 // Uses the bass SynthDef above
138 \dur, Pseq([0.75, 0.25, 0.25, 0.25, 0.5], 1),
139 \legato, Pseq([0.9, 0.3, 0.3, 0.3, 0.3], 1),
140 \amp, 0.5, \detune, 1.005
143 \midinote, Pseq([36, 48, 36], 1),
144 \dur, Pseq([0.25, 0.25, 0.5], 1),
145 \amp, 0.5, \detune, 1.005
147 tritone: Pmono(\bass,
148 \midinote, Pseq([36, 42, 41, 33], 1),
149 \dur, Pseq([0.25, 0.25, 0.25, 0.75], 1),
150 \amp, 0.5, \detune, 1.005
153 \midinote, Pseq([36, 39, 36, 42], 1),
154 \dur, Pseq([0.25, 0.5, 0.25, 0.5], 1),
155 \amp, 0.5, \detune, 1.005
159 TempoClock.default.tempo = 128/60;
161 // the higher level control pattern is really simple now
162 p = Psym(Pxrand(#[repeated, octave, tritone, dim], inf), ~phrases).play;
168 A complicated pattern with lots of embedding can be hard to read because it's more work to separate note-level details from the larger structure. The pattern choosing the phrases -- code::Pxrand(#[repeated, octave, tritone, dim], inf) :: -- is self-explanatory, however, and Psym fills in the details transparently.
171 Because of some special handling needed for event patterns, there are two versions of Psym. link::Classes/Psym:: handles event patterns, while link::Classes/Pnsym:: is for value patterns. Think of it this way: Pbind can be contained within Psym, but it contains Pnsym.
174 ( Psym ( Pbind ( Pnsym ) ) )
180 ## code::Psym(**, (pattern1: Pbind(**)) ::
181 ## code::Pbind(\someValue, Pnsym(**, (pattern1: Pwhite(**))) ::
185 ## code::Pbind(\someValue, Psym(**, (pattern1: Pwhite(**))) ::
190 subsection::Switching between patterns for individual values
192 In the examples above, if a list pattern encounters another pattern in its input values, the subpattern is embedded in its entirety before the list pattern is allowed to continue. Sometimes you might want to get just one value out of the subpattern, and then choose a different subpattern on the next event. Pswitch, Psym and Pnsym have cousins that do exactly this: link::Classes/Pswitch1::, Psym1 and Pnsym1.
195 // random pitches in two distinct ranges; use a coin toss to decide which for this event
198 TempoClock.default.tempo = 1;
200 \degree, Pswitch1([Pwhite(7, 14, inf), Pwhite(-7, 0, inf)], Pfunc { 0.7.coin.binaryValue }),
208 Compare to the following:
213 \degree, Pswitch([Pwhite(7, 14, inf), Pwhite(-7, 0, inf)], Pfunc { 0.7.coin.binaryValue }),
221 With Pswitch, one of the items is chosen from the list and keeps playing until it's finished. But the length of both Pwhite patterns is infinite, so whichever one is chosen first retains control. Pswitch1 does the coin toss on every event and embeds just one item.
223 Psym1 and Pnsym1 behave similarly, choosing the name to look up the pattern for each event.
225 subsection::Related: Conditional patterns
227 link::Classes/Pif:: supports this kind of structure: If the next value from a Boolean pattern is true, return the next item from pattern A, otherwise take it from pattern B. Another way to write the Pswitch1 example is to use a Boolean test directly on Pwhite, instead of writing a Pfunc for the coin toss. This might be clearer to read. However, this works only when there are two alternatives. Pswitch1 and Psym1 allow any number of choices.
231 TempoClock.default.tempo = 1;
233 // translation: if(0.7.coin) { rrand(-7, 0) } { rrand(7, 14 }
234 \degree, Pif(Pwhite(0.0, 1.0, inf) < 0.7, Pwhite(-7, 0, inf), Pwhite(7, 14, inf)),
242 We will see in link::Tutorials/A-Practical-Guide/PG_06e_Language_Control:: that Pif can be used on values that were previously calculated in the Pbind. It adds considerably to the intelligence Pbind can manage, when its value streams are aware of other values in the event.
245 Previous: link::Tutorials/A-Practical-Guide/PG_03_What_Is_Pbind::
247 Next: link::Tutorials/A-Practical-Guide/PG_05_Math_on_Patterns::