-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathAdditive_Synthesis_GUI_Patch_3.scd
197 lines (162 loc) · 5.66 KB
/
Additive_Synthesis_GUI_Patch_3.scd
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
// ************************************
// Additive Synthesis Demo (GUI)
// Patch 3 - Inharmonic Partials
// Bruno Ruviaro, 2013-07-24
// ************************************
/*
2 button rows to play up to 16 inharmonic partials of a given fundamental frequency.
Horizontal Sliders control ADSR envelope, Global Volume, and Fundamental Frequency.
All partials have equal amplitude.
How to start: select all (ctrl + A), then evaluate (ctrl + enter)
(on a Mac, use command key instead of control key)
*/
s.waitForBoot({
var win, buttonArray, knobArray, notesArray, partialArray, attackSlider, decaySlider, sustainSlider, releaseSlider, volumeSlider, fundamentalSlider, slidersWidth, fundamentalBus, updateButtonDisplay, att = 0.01, dec = 0.3, sus = 0.5, rel = 1.0, masterOut = 0.1;
notesArray = Array.newClear(16);
partialArray = Array.newClear(16);
slidersWidth = 375;
fundamentalBus = Bus.control(s, 1);
fundamentalBus.set(110);
Window.closeAll;
s.meter;
win = Window.new("Additive Synthesis - Inharmonics", Rect(500, 100, 800, 550)).front;
win.background = Color.grey;
win.alpha = 0.95;
win.onClose = {s.freeAll; "Done!".postln};
CmdPeriod.doOnce({win.close});
// Change the gaps and margins to see how they work
win.view.decorator = FlowLayout(win.view.bounds, margin: 10@10, gap: 20@20 );
buttonArray = Array.fill(16, {Button(win.view, 80@80)});
knobArray = Array.fill(16, {Knob(win.view, 80@50)});
// Little function to update numbers displayed on buttons
updateButtonDisplay = {arg button, index;
var current = button.value;
button.states = [
[(partialArray[index]).asString, Color.black],
[(partialArray[index]).asString, Color.black, Color.green]];
button.value = current;
};
knobArray.do({arg item, count;
knobArray[count].centered = true;
knobArray[count].value = rrand(0.4, 0.6).round(0.01); // de-centered positions (center=0.5)
knobArray[count].action = {arg knob;
// update array of partials
partialArray[count] = knob.value.linlin(0, 1, -0.2, 0.2).round(0.01) + count + 1;
// update button display
updateButtonDisplay.value(buttonArray[count], count);
// if this partial is currently playing, update its partial number:
if(notesArray[count].isNil.not,
{notesArray[count].set(\partial, partialArray[count])})}});
// Initialize array of partials with first knob values
partialArray.do({arg item, count;
partialArray[count] = knobArray[count].value.linlin(0, 1, -0.2, 0.2).round(0.01) + count + 1});
// in the line above, "+ 0.5" adjusts for knob center = 0.5 (becomes center = 1);
// this deviation value then is multiplied by the partial number
// Initialize Buttons with proper number display
buttonArray.do({arg item, count; updateButtonDisplay.value(item, count)});
buttonArray.do({arg item, count;
item.action = {arg state;
case
{state.value==1} {notesArray[count] = Synth("addsynth", [
\fundamental, fundamentalBus.asMap,
\partial, partialArray[count],
\amp, 0.1,
\att, att,
\dec, dec,
\sus, sus,
\rel, rel])}
{state.value==0} {notesArray[count].release; notesArray[count] = nil}
}});
attackSlider = EZSlider(
parent: win,
bounds: slidersWidth @ 30,
label: "Attack",
controlSpec: ControlSpec(0.01, 4.0, \lin, 0.01, 0.1, "sec"),
action: {|ez| att = ez.value},
initVal: 0.01,
unitWidth: 30)
.setColors(
stringColor: Color.black,
sliderBackground: Color.grey(0.7),
numNormalColor: Color.black);
decaySlider = EZSlider(
parent: win,
bounds: slidersWidth @ 30,
label: "Decay",
controlSpec: ControlSpec(0.01, 1, \lin, 0.01, 0.1, "sec"),
action: {|ez| dec = ez.value},
initVal: 0.3,
unitWidth: 30)
.setColors(
stringColor: Color.black,
sliderBackground: Color.grey(0.7),
numNormalColor: Color.black);
sustainSlider = EZSlider(
parent: win,
bounds: slidersWidth @ 30,
label: "Sustain",
controlSpec: ControlSpec(1, 100, \lin, 1, 75, "%"),
action: {|ez| sus = ez.value/100.0},
initVal: 75,
unitWidth: 30)
.setColors(
stringColor: Color.black,
sliderBackground: Color.grey(0.7),
numNormalColor: Color.black);
releaseSlider = EZSlider(
parent: win,
bounds: slidersWidth @ 30,
label: "Release",
controlSpec: ControlSpec(0.3, 6, \lin, 0.1, 0.5, "sec"),
action: {|ez| rel = ez.value},
initVal: 0.5,
unitWidth: 30)
.setColors(
stringColor: Color.black,
sliderBackground: Color.grey(0.7),
numNormalColor: Color.black);
volumeSlider = EZSlider(
parent: win,
bounds: slidersWidth*2.05 @ 30,
label: "VOL",
controlSpec: ControlSpec(1, 100, \lin, 1, 10, "%"),
action: {|ez| masterOut.set(\amp, ez.value/100)},
initVal: 10,
unitWidth: 30)
.setColors(
stringColor: Color.white,
sliderBackground: Color.grey(0.9),
numNormalColor: Color.grey);
fundamentalSlider = EZSlider(
parent: win,
bounds: slidersWidth*2.05 @ 30,
label: "FREQ",
controlSpec: ControlSpec(50, 200, \lin, 1, 110, "Hz"),
action: {|ez| fundamentalBus.set(ez.value)},
initVal: 110,
unitWidth: 30)
.setColors(
stringColor: Color.white,
sliderBackground: Color.grey(0.9),
numNormalColor: Color.grey);
// SynthDefs
{
SynthDef("addsynth", {
arg fundamental = 110, partial = 2.1, amp = 0.1, gate = 1, att = 0.01, dec = 0.3, sus = 0.5, rel = 1;
var snd, env, freq;
freq = fundamental * partial;
env = EnvGen.ar(Env.adsr(att, dec, sus, rel), gate, doneAction: 2);
snd = SinOsc.ar(Lag.kr(freq, 1), 0, amp) * env;
Out.ar(0, snd!2);
}).add;
SynthDef(\amp, {arg inbus=0, amp = 0.1;
ReplaceOut.ar(inbus, In.ar(inbus, 2) * amp);
}).add;
// Wait for SynthDefs to be added...
s.sync;
// Now call the Master Out Synth:
masterOut = Synth("amp", addAction: \addToTail);
}.fork;
"Additive Synthesis Demo 1".postln;
"".postln;
});