forked from monome/grid-studies-pd
-
Notifications
You must be signed in to change notification settings - Fork 0
/
index.html
725 lines (542 loc) · 30 KB
/
index.html
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
331
332
333
334
335
336
337
338
339
340
341
342
343
344
345
346
347
348
349
350
351
352
353
354
355
356
357
358
359
360
361
362
363
364
365
366
367
368
369
370
371
372
373
374
375
376
377
378
379
380
381
382
383
384
385
386
387
388
389
390
391
392
393
394
395
396
397
398
399
400
401
402
403
404
405
406
407
408
409
410
411
412
413
414
415
416
417
418
419
420
421
422
423
424
425
426
427
428
429
430
431
432
433
434
435
436
437
438
439
440
441
442
443
444
445
446
447
448
449
450
451
452
453
454
455
456
457
458
459
460
461
462
463
464
465
466
467
468
469
470
471
472
473
474
475
476
477
478
479
480
481
482
483
484
485
486
487
488
489
490
491
492
493
494
495
496
497
498
499
500
501
502
503
504
505
506
507
508
509
510
511
512
513
514
515
516
517
518
519
520
521
522
523
524
525
526
527
528
529
530
531
532
533
534
535
536
537
538
539
540
541
542
543
544
545
546
547
548
549
550
551
552
553
554
555
556
557
558
559
560
561
562
563
564
565
566
567
568
569
570
571
572
573
574
575
576
577
578
579
580
581
582
583
584
585
586
587
588
589
590
591
592
593
594
595
596
597
598
599
600
601
602
603
604
605
606
607
608
609
610
611
612
613
614
615
616
617
618
619
620
621
622
623
624
625
626
627
628
629
630
631
632
633
634
635
636
637
638
639
640
641
642
643
644
645
646
647
648
649
650
651
652
653
654
655
656
657
658
659
660
661
662
663
664
665
666
667
668
669
670
671
672
673
674
675
676
677
678
679
680
681
682
683
684
685
686
687
688
689
690
691
692
693
694
695
696
697
698
699
700
701
702
703
704
705
706
707
708
709
710
711
712
713
714
715
716
717
718
719
720
721
722
723
724
725
<!DOCTYPE html><html>
<head>
<meta charset="utf-8">
<title>index</title>
<style type="text/css">
body {
font-family: Helvetica, arial, sans-serif;
font-size: 14px;
line-height: 1.6;
padding-top: 10px;
padding-bottom: 10px;
background-color: white;
padding: 30px; }
body > *:first-child {
margin-top: 0 !important; }
body > *:last-child {
margin-bottom: 0 !important; }
a {
color: #4183C4; }
a.absent {
color: #cc0000; }
a.anchor {
display: block;
padding-left: 30px;
margin-left: -30px;
cursor: pointer;
position: absolute;
top: 0;
left: 0;
bottom: 0; }
h1, h2, h3, h4, h5, h6 {
margin: 20px 0 10px;
padding: 0;
font-weight: bold;
-webkit-font-smoothing: antialiased;
cursor: text;
position: relative; }
h1:hover a.anchor, h2:hover a.anchor, h3:hover a.anchor, h4:hover a.anchor, h5:hover a.anchor, h6:hover a.anchor {
background: url() no-repeat 10px center;
text-decoration: none; }
h1 tt, h1 code {
font-size: inherit; }
h2 tt, h2 code {
font-size: inherit; }
h3 tt, h3 code {
font-size: inherit; }
h4 tt, h4 code {
font-size: inherit; }
h5 tt, h5 code {
font-size: inherit; }
h6 tt, h6 code {
font-size: inherit; }
h1 {
font-size: 28px;
color: black; }
h2 {
font-size: 24px;
border-bottom: 1px solid #cccccc;
color: black; }
h3 {
font-size: 18px; }
h4 {
font-size: 16px; }
h5 {
font-size: 14px; }
h6 {
color: #777777;
font-size: 14px; }
p, blockquote, ul, ol, dl, li, table, pre {
margin: 15px 0; }
hr {
background: transparent url() repeat-x 0 0;
border: 0 none;
color: #cccccc;
height: 4px;
padding: 0;
}
body > h2:first-child {
margin-top: 0;
padding-top: 0; }
body > h1:first-child {
margin-top: 0;
padding-top: 0; }
body > h1:first-child + h2 {
margin-top: 0;
padding-top: 0; }
body > h3:first-child, body > h4:first-child, body > h5:first-child, body > h6:first-child {
margin-top: 0;
padding-top: 0; }
a:first-child h1, a:first-child h2, a:first-child h3, a:first-child h4, a:first-child h5, a:first-child h6 {
margin-top: 0;
padding-top: 0; }
h1 p, h2 p, h3 p, h4 p, h5 p, h6 p {
margin-top: 0; }
li p.first {
display: inline-block; }
li {
margin: 0; }
ul, ol {
padding-left: 30px; }
ul :first-child, ol :first-child {
margin-top: 0; }
dl {
padding: 0; }
dl dt {
font-size: 14px;
font-weight: bold;
font-style: italic;
padding: 0;
margin: 15px 0 5px; }
dl dt:first-child {
padding: 0; }
dl dt > :first-child {
margin-top: 0; }
dl dt > :last-child {
margin-bottom: 0; }
dl dd {
margin: 0 0 15px;
padding: 0 15px; }
dl dd > :first-child {
margin-top: 0; }
dl dd > :last-child {
margin-bottom: 0; }
blockquote {
border-left: 4px solid #dddddd;
padding: 0 15px;
color: #777777; }
blockquote > :first-child {
margin-top: 0; }
blockquote > :last-child {
margin-bottom: 0; }
table {
padding: 0;border-collapse: collapse; }
table tr {
border-top: 1px solid #cccccc;
background-color: white;
margin: 0;
padding: 0; }
table tr:nth-child(2n) {
background-color: #f8f8f8; }
table tr th {
font-weight: bold;
border: 1px solid #cccccc;
margin: 0;
padding: 6px 13px; }
table tr td {
border: 1px solid #cccccc;
margin: 0;
padding: 6px 13px; }
table tr th :first-child, table tr td :first-child {
margin-top: 0; }
table tr th :last-child, table tr td :last-child {
margin-bottom: 0; }
img {
max-width: 100%; }
span.frame {
display: block;
overflow: hidden; }
span.frame > span {
border: 1px solid #dddddd;
display: block;
float: left;
overflow: hidden;
margin: 13px 0 0;
padding: 7px;
width: auto; }
span.frame span img {
display: block;
float: left; }
span.frame span span {
clear: both;
color: #333333;
display: block;
padding: 5px 0 0; }
span.align-center {
display: block;
overflow: hidden;
clear: both; }
span.align-center > span {
display: block;
overflow: hidden;
margin: 13px auto 0;
text-align: center; }
span.align-center span img {
margin: 0 auto;
text-align: center; }
span.align-right {
display: block;
overflow: hidden;
clear: both; }
span.align-right > span {
display: block;
overflow: hidden;
margin: 13px 0 0;
text-align: right; }
span.align-right span img {
margin: 0;
text-align: right; }
span.float-left {
display: block;
margin-right: 13px;
overflow: hidden;
float: left; }
span.float-left span {
margin: 13px 0 0; }
span.float-right {
display: block;
margin-left: 13px;
overflow: hidden;
float: right; }
span.float-right > span {
display: block;
overflow: hidden;
margin: 13px auto 0;
text-align: right; }
code, tt {
margin: 0 2px;
padding: 0 5px;
white-space: nowrap;
border: 1px solid #eaeaea;
background-color: #f8f8f8;
border-radius: 3px; }
pre code {
margin: 0;
padding: 0;
white-space: pre;
border: none;
background: transparent; }
.highlight pre {
background-color: #f8f8f8;
border: 1px solid #cccccc;
font-size: 13px;
line-height: 19px;
overflow: auto;
padding: 6px 10px;
border-radius: 3px; }
pre {
background-color: #f8f8f8;
border: 1px solid #cccccc;
font-size: 13px;
line-height: 19px;
overflow: auto;
padding: 6px 10px;
border-radius: 3px; }
pre code, pre tt {
background-color: transparent;
border: none; }
sup {
font-size: 0.83em;
vertical-align: super;
line-height: 0;
}
* {
-webkit-print-color-adjust: exact;
}
@media screen and (min-width: 914px) {
body {
width: 854px;
margin:0 auto;
}
}
@media print {
table, pre {
page-break-inside: avoid;
}
pre {
word-wrap: break-word;
}
}
</style>
<style type="text/css">
/**
* prism.js default theme for JavaScript, CSS and HTML
* Based on dabblet (http://dabblet.com)
* @author Lea Verou
*/
code[class*="language-"],
pre[class*="language-"] {
color: black;
text-shadow: 0 1px white;
font-family: Consolas, Monaco, 'Andale Mono', monospace;
direction: ltr;
text-align: left;
white-space: pre;
word-spacing: normal;
word-break: normal;
line-height: 1.5;
-moz-tab-size: 4;
-o-tab-size: 4;
tab-size: 4;
-webkit-hyphens: none;
-moz-hyphens: none;
-ms-hyphens: none;
hyphens: none;
}
pre[class*="language-"]::-moz-selection, pre[class*="language-"] ::-moz-selection,
code[class*="language-"]::-moz-selection, code[class*="language-"] ::-moz-selection {
text-shadow: none;
background: #b3d4fc;
}
pre[class*="language-"]::selection, pre[class*="language-"] ::selection,
code[class*="language-"]::selection, code[class*="language-"] ::selection {
text-shadow: none;
background: #b3d4fc;
}
@media print {
code[class*="language-"],
pre[class*="language-"] {
text-shadow: none;
}
}
/* Code blocks */
pre[class*="language-"] {
padding: 1em;
margin: .5em 0;
overflow: auto;
}
:not(pre) > code[class*="language-"],
pre[class*="language-"] {
background: #f5f2f0;
}
/* Inline code */
:not(pre) > code[class*="language-"] {
padding: .1em;
border-radius: .3em;
}
.token.comment,
.token.prolog,
.token.doctype,
.token.cdata {
color: slategray;
}
.token.punctuation {
color: #999;
}
.namespace {
opacity: .7;
}
.token.property,
.token.tag,
.token.boolean,
.token.number,
.token.constant,
.token.symbol,
.token.deleted {
color: #905;
}
.token.selector,
.token.attr-name,
.token.string,
.token.char,
.token.builtin,
.token.inserted {
color: #690;
}
.token.operator,
.token.entity,
.token.url,
.language-css .token.string,
.style .token.string {
color: #a67f59;
background: hsla(0, 0%, 100%, .5);
}
.token.atrule,
.token.attr-value,
.token.keyword {
color: #07a;
}
.token.function {
color: #DD4A68;
}
.token.regex,
.token.important,
.token.variable {
color: #e90;
}
.token.important,
.token.bold {
font-weight: bold;
}
.token.italic {
font-style: italic;
}
.token.entity {
cursor: help;
}
</style>
</head>
<body>
<h1 id="toc_0">Grid Studies: Pure Data</h1>
<p>By design the monome grid does nothing on it's own. You the user assign it purpose and meaning: instrument, experiment, tool, toy... choose your own adventure. This grid is <em>intended</em> to be reimagined. Here we set forth to impart some introductory knowledge: potential energy for radical creative freedom.</p>
<p>Pure Data (Pd) is a visual programming language developed by Miller Puckette in the 1990s for creating interactive computer music and multimedia works. While Puckette is the main author of the program, Pd is an open source project with a large developer base working on new extensions. (from <a href="http://en.wikipedia.org/wiki/Pure_Data">Wikipedia</a>).</p>
<h2 id="toc_1">Prerequisites</h2>
<p>If you're new to pd, here are some tutorials (...)</p>
<p>Download Pd Extended: <a href="http://puredata.info/downloads/pd-extended">puredata.info/</a></p>
<p>Download the monome installer: <a href="http://monome.org/docs/begin">monome.org/docs/begin</a></p>
<p>Download the code examples here: <a href="https://github.com/monome/grid-studies-pd/releases/latest">github.com/monome/grid-studies-pd/releases/latest</a></p>
<h2 id="toc_2">1. Connect</h2>
<p><em>See grid-studies-1-1.pd for this section.</em></p>
<p>To communicate with grids we trade OSC messages with serialosc. serialosc translates OSC messages to streams of numbers over USB.</p>
<p>First we will show how to talk to serialosc.</p>
<p>Open Pd and start a new patch.</p>
<p>Save the blank patch to the <code>/files</code> folder as <code>study.pd</code>.</p>
<p>Create a new object (win/linux: control+1, mac: command+1) and type <code>serialosc</code> then place it in your patch with a mouse click.</p>
<p><img src="images/grid-studies-1-1-1.png" alt=""></p>
<p>Attach your grid and you can now communicate with it through this object.</p>
<p>Note: this object you've imbedded isn't serialosc itself, which is an invisible daemon on your computer. The object is a helper patcher, or 'abstraction' to simplify using serialosc. We'll refer to this helper as serialosc, hopefully without much confusion.</p>
<p>The serialosc abstraction is actually the patch <code>serialosc.pd</code> inside an object. In order for your patch to use serialosc, the abstraction needs to be in the same folder as your saved patch, hence why we saved our patch to <code>/files</code> above.</p>
<h2 id="toc_3">2. Basics</h2>
<p>Messages are sent to serialosc through the top left inlet, and received out the bottom left outlet.</p>
<p><em>See grid-studies-2-1.pd for this section.</em></p>
<h3 id="toc_4">2.1 Key input</h3>
<p>To see what is coming from the grid, create a <code>print</code> object and connect the left outlet of serialosc to it. Open the Pd window (Window -> Pd Window) and press some keys on the grid. OSC data will be displayed in the console.</p>
<p>Examining the output, key data fits this form:</p>
<pre><code>/monome/grid/key x y state</code></pre>
<p>Where x,y is the position and z indicates key down (1) or key up (0).</p>
<p>Other messages (such as connect and disconnect) come from this same outlet, so we want to filter for the key messages.</p>
<p>Change the <code>print</code> object to <code>routeOSC /monome/grid/key</code> and then print the output from the routeOSC.</p>
<p>We now have a list of 3 numbers according to each key action. Use an unpack to break this down further into individual numbers.</p>
<p><img src="images/grid-studies-2-1-1.png" alt=""></p>
<h3 id="toc_5">2.2 LED output</h3>
<p>Above the serialosc box create a message (control/command + 2) and type:</p>
<pre><code>/monome/grid/led/set 2 0 1</code></pre>
<p>Connect this to the left inlet of serialosc.</p>
<p>Clicking this box will light up LED 2 in row 0. The message format is:</p>
<pre><code>/monome/grid/led/set x y z</code></pre>
<p>This is similar to the key input message, where z is on (1) or off (0).</p>
<p>Using Pd's list methods, use <code>$1 $2 $3</code> to change LEDs more dynamically. With a single message box as a sort of funnel, we can change various positions with message boxes and toggles.</p>
<p>To clear the entire grid, use the following message:</p>
<pre><code>/monome/grid/led/all 0</code></pre>
<p><img src="images/grid-studies-2-2-1.png" alt=""></p>
<h3 id="toc_6">2.3 Coupled interaction</h3>
<p>Connect the output of</p>
<pre><code>routeOSC /monome/grid/key</code></pre>
<p>to the <code>/monome/grid/led/set $1 $2 $3</code> message box above serialosc which changes LEDs.</p>
<p>You now have a coupled interface, where the key state is reflected by the the LEDs.</p>
<p><img src="images/grid-studies-2-3-1.png" alt=""></p>
<h3 id="toc_7">2.4 Decoupled interaction</h3>
<p>The most fundamental decoupled interface is an array of toggles. We can accomplish this by ignoring the key up state, switching the LED state only on key down.</p>
<p>We'll start with a single toggle for the upper left key position (0, 0).</p>
<p>Remove the connection between routeOSC and the LED-driving message box.</p>
<p>To respond to the 0,0 position, we can use two <code>route</code> objects to filter the x, then y location of the press. Then we can respond only to key down by looking for values of 1 with a <code>sel</code> object.</p>
<pre><code>route 0
route 0
sel 1</code></pre>
<p>Connect the output from the sel into a toggle box (control/command + shift + T). You can now see the toggle reflect the toggle state by pressing the upper-left key on the grid.</p>
<p>To complete the cycle we can then drive the corresponding LED with a message box <code>/monome/grid/led/set 0 0 $1</code> connected to serialosc's left inlet.</p>
<p><img src="images/grid-studies-2-4-1.png" alt=""></p>
<h2 id="toc_8">3.0 Further</h2>
<p>Now we'll show how basic grid applications are developed by creating a step sequencer. We will add features incrementally:</p>
<ul>
<li>Use the top six rows as toggles.</li>
<li>Accept a clock pulse to advance the playhead from left to right, one column at a time. Wrap back to 0 at the end.</li>
<li>Display the play head on "position" (bottom) row.</li>
<li>Indicate the "activity" row (second to last) with a low brightness.</li>
<li>Trigger an event when the playhead reads an "on" toggle. Our "event" will be to turn on the corresponding LED in the "activity" row.</li>
<li>Jump to playback position when key pressed in the position row.</li>
<li>Adjust playback loop with two-key gesture in position row.</li>
</ul>
<h3 id="toc_9">3.1 Toggles</h3>
<p><em>See grid-studies-3-1.pd for this step.</em></p>
<p>Before we can make our bank of toggles, we need a way to look at the top six rows only, as the last two rows are not part of our toggle bank. First we'll use a message box to switch around our key input to place the row number first, then route off the last two rows for use later.</p>
<pre><code>$2 $1 $3
route 6 7</code></pre>
<p>By switching the first and second elements and then putting them into the route object, rows 0-5 are passed to the right outlet.</p>
<p>In our previous example we only created a single toggle, and the chosen approach is not appropriate for dealing with a large bank of toggles. Instead we'll create a <code>table</code> which functions as a large array of data for remembering the toggle states of the top six rows. Our table is inside the subpatcher <code>pd toggles</code>.</p>
<p>Note how after formatting the LED messages from <code>pd toggles</code> we use a <code>s osc-out</code> to send osc messages to serialosc without cluttering our patch with long patch connections.</p>
<p><img src="images/grid-studies-3-1-2.png" alt=""></p>
<p>The toggles sub-patch can be broken down into a number of small steps to understand it, though don't fret if it doesn't make total sense immediately. This sub-patch can simply be copied to your own projects without modification and it will work for the full 16x8 grid.</p>
<p><img src="images/grid-studies-3-1-1.png" alt=""></p>
<p>In order to save the state of the bank of toggles we create a table called 'grid' with 128 values.</p>
<pre><code>table grid 128</code></pre>
<p>Rearranging the input key press information we can ignore key-ups as before. Inline comments track the current order of our input message.</p>
<pre><code>$3 $2 $1
route 1</code></pre>
<p>In order to put our 2-dimensional grid into a table, we have to 'flatten' it into a long list. We do so by multiplying the y value by 16, and adding the result to our x value. The grid is thus read like a book, from left to right, starting at the top and working downward.</p>
<pre><code>unpack
| |
| * 16
| /
+</code></pre>
<p>This index value is then used to set two values before toggling that state. The index is converted back to the x,y coordinate for LED feedback, plus we save the index to update the table with our new toggle state.</p>
<p>Finally we use the index to find the current state of the table, then invert that state with '== 0'. This works because our state is always a 0 or 1. This value is then sent to update the grid LED and is stored back in the table.</p>
<h3 id="toc_10">3.2 Play</h3>
<p><em>See grid-studies-3-2.pd for this step.</em></p>
<p>We can create a "play head" with a simple counter.</p>
<p><img src="images/grid-studies-3-2-1.png" alt=""></p>
<p>To see the play position on the bottom row, we will turn on the corresponding LED position after first clearing the entire row. We can clear a row by using a new OSC message:</p>
<pre><code>/monome/grid/led/row 0 7 0 0</code></pre>
<p>where the format of the message is:</p>
<pre><code>/monome/grid/led/row x_offset y d[...]</code></pre>
<p>Here y is 7, the last row. Check out the full OSC spec for more information on this message.</p>
<p>After we clear the row, we turn on the corresponding LED with a normal single-LED message:</p>
<pre><code>/monome/grid/led/set $1 0 1</code></pre>
<p>Now when you turn on the clock, you'll see the playhead moving along the bottom row.</p>
<h3 id="toc_11">3.3 Triggers</h3>
<p><em>See grid-studies-3-3.pd for this step.</em></p>
<p>As the playhead moves we will read the contents of the corresponding column and trigger events based on which toggles are turned on.</p>
<p>We do this by scanning through our table, looking at every 16th value, driven by the play position. The <code>pd triggers</code> sub patch shows how to read data out of the table we created in <code>pd toggles</code>.</p>
<p><img src="images/grid-studies-3-3-1.png" alt=""></p>
<p>Using the uzi object we read the state of the toggle for each subsequent row, by multiplying the uzi output by 16 (to jump through the array by rows), and adding the current play position to point at the correct column of data. Note that we subtract one from uzi's right outlet as the grid is indexed from zero, whereas uzi starts counting from one. If the discovered value is 1 we send out the row value to indicate an "event" has occured.</p>
<p>To indicate these "events" we will light up the corresponding x position in the 6th row:</p>
<pre><code>/monome/grid/led/set $1 6 $2</code></pre>
<p>Similarly to the play position display, we need to clear the row between refreshes.</p>
<p>Furthermore, to give the interface some delineation (not just a field of random LEDs) we will "clear" this row to a low-brightness level rather than completely off:</p>
<pre><code>/monome/grid/led/level/row 0 6 5 5 5 5 5 5 5 5 5 5 5 5 5 5 5 5</code></pre>
<p><img src="images/grid-studies-3-3-2.png" alt=""></p>
<p>This /level/ message is in the format:</p>
<pre><code>/monome/grid/led/level/row x_off y d[...]</code></pre>
<p>The format is similar to the monochromatic <code>row</code> message, but here <code>d[...]</code> is discrete LED values of 0-15. The message we're using has 16 of the number 5, which sets the entire 6th row to a dim level.</p>
<p>The "triggered" LEDs will be full brightness on top of this dim row.</p>
<p>Lastly, there's a tiny sound engine so you can actually hear something. Turn on the DAC and turn up the gain slider.</p>
<h3 id="toc_12">3.4 Cutting and Looping</h3>
<p><em>See grid-studies-3-4.pd for this step.</em></p>
<p>To liven up the sequencer, we will have key presses on the play row jump to the pressed position. But we also want a two-key gesture (holding down a first while pressing a second) to set the start-end loop boundaries. This requires keeping track of how many keys are being held down in the last row.</p>
<p>Inside <code>pd cutting</code> we process all presses from the last row, and send our commands to the counter object.</p>
<p><img src="images/grid-studies-3-4-2.png" alt=""></p>
<p>We unpack the incoming message and keep track of the accumulation of key ups and downs. This is accomplished by adding one for each key up and subtracting one for each key down. This looks weird as a pd patch, but tracing through it will reveal the logic.</p>
<p>The number of keys held will gate the output of the x position of the key. When a single key is pressed the x position goes out the left outlet of the gate, setting the position of the counter. This first position is also stored for potential use later.</p>
<p>If a second key is pressed (in this same row) while a first is held, the current x position pressed is set as the loop max (with a <code>max $1</code> message) and the previously pressed x position is recalled and set as the loop minimum (with a <code>min $1</code> message).</p>
<p><img src="images/grid-studies-3-4-1.png" alt=""></p>
<h2 id="toc_13">Closing</h2>
<p>We've created a minimal yet intuitive interface for rapidly exploring sequences. We can intuitively change event triggers, loop points, and jump around the data performatively. Many more features could be added, and there are numerous other ways to think about interaction between key press and light feedback in completely different contexts.</p>
<h3 id="toc_14">Suggested exercises</h3>
<ul>
<li>Display the loop range with dim LED levels.</li>
<li>"Record" keypresses in the "trigger" row to the toggle matrix.</li>
<li>Display the play head position as a dim column behind the toggle data.</li>
<li>Use the rightmost key in the "trigger" row as an "alt" key.
<ul>
<li>If "alt" is held while pressing a toggle, clear the entire row.</li>
<li>If "alt" is held while pressing the play row, reverse the direction of play.</li>
</ul></li>
</ul>
<p><em>Max</em> was designed by Miller Puckette and is actively developed as an open source project. <a href="http://puredata.info">puredata.info</a>.</p>
<p>This tutorial was created by <a href="http://whimsicalraps.com">Trent Gill</a> for <a href="monome.org">monome.org</a>.</p>
<p>Contributions welcome. Submit a pull request to <a href="https://github.com/monome/grid-studies-pd">github.com/monome/grid-studies-pd</a> or e-mail <a href="mailto:[email protected]">[email protected]</a>.</p>
<script type="text/javascript">
self="undefined"!=typeof window?window:"undefined"!=typeof WorkerGlobalScope&&self instanceof WorkerGlobalScope?self:{};var Prism=function(){var e=/\blang(?:uage)?-(?!\*)(\w+)\b/i,t=self.Prism={util:{encode:function(e){return e instanceof n?new n(e.type,t.util.encode(e.content),e.alias):"Array"===t.util.type(e)?e.map(t.util.encode):e.replace(/&/g,"&").replace(/</g,"<").replace(/\u00a0/g," ")},type:function(e){return Object.prototype.toString.call(e).match(/\[object (\w+)\]/)[1]},clone:function(e){var n=t.util.type(e);switch(n){case"Object":var a={};for(var r in e)e.hasOwnProperty(r)&&(a[r]=t.util.clone(e[r]));return a;case"Array":return e.map(function(e){return t.util.clone(e)})}return e}},languages:{extend:function(e,n){var a=t.util.clone(t.languages[e]);for(var r in n)a[r]=n[r];return a},insertBefore:function(e,n,a,r){r=r||t.languages;var i=r[e];if(2==arguments.length){a=arguments[1];for(var l in a)a.hasOwnProperty(l)&&(i[l]=a[l]);return i}var o={};for(var s in i)if(i.hasOwnProperty(s)){if(s==n)for(var l in a)a.hasOwnProperty(l)&&(o[l]=a[l]);o[s]=i[s]}return t.languages.DFS(t.languages,function(t,n){n===r[e]&&t!=e&&(this[t]=o)}),r[e]=o},DFS:function(e,n,a){for(var r in e)e.hasOwnProperty(r)&&(n.call(e,r,e[r],a||r),"Object"===t.util.type(e[r])?t.languages.DFS(e[r],n):"Array"===t.util.type(e[r])&&t.languages.DFS(e[r],n,r))}},highlightAll:function(e,n){for(var a,r=document.querySelectorAll('code[class*="language-"], [class*="language-"] code, code[class*="lang-"], [class*="lang-"] code'),i=0;a=r[i++];)t.highlightElement(a,e===!0,n)},highlightElement:function(a,r,i){for(var l,o,s=a;s&&!e.test(s.className);)s=s.parentNode;if(s&&(l=(s.className.match(e)||[,""])[1],o=t.languages[l]),o){a.className=a.className.replace(e,"").replace(/\s+/g," ")+" language-"+l,s=a.parentNode,/pre/i.test(s.nodeName)&&(s.className=s.className.replace(e,"").replace(/\s+/g," ")+" language-"+l);var g=a.textContent;if(g){g=g.replace(/^(?:\r?\n|\r)/,"");var u={element:a,language:l,grammar:o,code:g};if(t.hooks.run("before-highlight",u),r&&self.Worker){var c=new Worker(t.filename);c.onmessage=function(e){u.highlightedCode=n.stringify(JSON.parse(e.data),l),t.hooks.run("before-insert",u),u.element.innerHTML=u.highlightedCode,i&&i.call(u.element),t.hooks.run("after-highlight",u)},c.postMessage(JSON.stringify({language:u.language,code:u.code}))}else u.highlightedCode=t.highlight(u.code,u.grammar,u.language),t.hooks.run("before-insert",u),u.element.innerHTML=u.highlightedCode,i&&i.call(a),t.hooks.run("after-highlight",u)}}},highlight:function(e,a,r){var i=t.tokenize(e,a);return n.stringify(t.util.encode(i),r)},tokenize:function(e,n){var a=t.Token,r=[e],i=n.rest;if(i){for(var l in i)n[l]=i[l];delete n.rest}e:for(var l in n)if(n.hasOwnProperty(l)&&n[l]){var o=n[l];o="Array"===t.util.type(o)?o:[o];for(var s=0;s<o.length;++s){var g=o[s],u=g.inside,c=!!g.lookbehind,f=0,h=g.alias;g=g.pattern||g;for(var p=0;p<r.length;p++){var d=r[p];if(r.length>e.length)break e;if(!(d instanceof a)){g.lastIndex=0;var m=g.exec(d);if(m){c&&(f=m[1].length);var y=m.index-1+f,m=m[0].slice(f),v=m.length,k=y+v,b=d.slice(0,y+1),w=d.slice(k+1),O=[p,1];b&&O.push(b);var N=new a(l,u?t.tokenize(m,u):m,h);O.push(N),w&&O.push(w),Array.prototype.splice.apply(r,O)}}}}}return r},hooks:{all:{},add:function(e,n){var a=t.hooks.all;a[e]=a[e]||[],a[e].push(n)},run:function(e,n){var a=t.hooks.all[e];if(a&&a.length)for(var r,i=0;r=a[i++];)r(n)}}},n=t.Token=function(e,t,n){this.type=e,this.content=t,this.alias=n};if(n.stringify=function(e,a,r){if("string"==typeof e)return e;if("[object Array]"==Object.prototype.toString.call(e))return e.map(function(t){return n.stringify(t,a,e)}).join("");var i={type:e.type,content:n.stringify(e.content,a,r),tag:"span",classes:["token",e.type],attributes:{},language:a,parent:r};if("comment"==i.type&&(i.attributes.spellcheck="true"),e.alias){var l="Array"===t.util.type(e.alias)?e.alias:[e.alias];Array.prototype.push.apply(i.classes,l)}t.hooks.run("wrap",i);var o="";for(var s in i.attributes)o+=s+'="'+(i.attributes[s]||"")+'"';return"<"+i.tag+' class="'+i.classes.join(" ")+'" '+o+">"+i.content+"</"+i.tag+">"},!self.document)return self.addEventListener?(self.addEventListener("message",function(e){var n=JSON.parse(e.data),a=n.language,r=n.code;self.postMessage(JSON.stringify(t.util.encode(t.tokenize(r,t.languages[a])))),self.close()},!1),self.Prism):self.Prism;var a=document.getElementsByTagName("script");return a=a[a.length-1],a&&(t.filename=a.src,document.addEventListener&&!a.hasAttribute("data-manual")&&document.addEventListener("DOMContentLoaded",t.highlightAll)),self.Prism}();"undefined"!=typeof module&&module.exports&&(module.exports=Prism);
</script>
</body>
</html>