-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathFeatureExtractor.scd
94 lines (73 loc) · 3.25 KB
/
FeatureExtractor.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
s.boot // Boot the server first
(
x = Task.new({
~vowelA = [1620, 780]; // A
~vowelE = [2150, 440]; // E
~vowelI = [2300, 300]; // I
~vowelO = [900, 580]; // O
~vowelU = [780, 290]; // U
"fatto".postln;
// OSC OUTPUT
~outputIP = "localhost";
~outputPort = 8339;
~outputNet = NetAddr.new(~outputIP, ~outputPort);
~arrayOSC = [0.0, 0.0, 0.0, 0.0, 0.0];
~arrayOSCounter = 0;
~oscFunc = OSCFunc({ arg msg, time;
var value;
value = msg[3]; // Prende il valore del sig
~arrayOSCounter = ~arrayOSCounter + 1 % 5;
case
{~arrayOSCounter==1} {~arrayOSC[0] = value}
{~arrayOSCounter==2} {~arrayOSC[1] = value}
{~arrayOSCounter==3} {~arrayOSC[2] = value}
{~arrayOSCounter==4} {~arrayOSC[3] = value}
{~arrayOSCounter==0} {~arrayOSC[4] = value; ~outputNet.sendMsg("/audio", ~arrayOSC[0], ~arrayOSC[1], ~arrayOSC[2], ~arrayOSC[3], ~arrayOSC[4]); ~arrayOSC.postln};
},'/tr', s.addr);
SynthDef(\vowels, {|filterReson = 0.05, f1 = 1620, f2 = 780, sourceFreq = 70, amp = 1.0|
var aFilter, eFilter, iFilter, oFilter, uFilter,
output, input,
rmsA, rmsE, rmsI, rmsO, rmsU,
chain1, chain2, magnitudeAbove, localMax, loudnessTracker, triggerInterval, detect, gate;
//input = BPF.ar(Saw.ar(sourceFreq), f1, 1.0) + BPF.ar(Saw.ar(sourceFreq), f2, 1.0);
//input = PinkNoise.ar(0.1);
input = SoundIn.ar(0);
// GATE PART
chain1 = FFT(LocalBuf(2048), input);
loudnessTracker = Loudness.kr(chain1); // If loudness > 20 gate aperto
loudnessTracker = (loudnessTracker - 15); // -15 Threshold
gate = Select.kr(loudnessTracker > DC.kr(0.0), [DC.ar(0.0), DC.ar(1.0)]);
input = input * gate;
// AMPLITUDE NORMALISATION AND SPECTRAL LOCAL MAX FILTERING
input = Normalizer.ar(input); // Amplitude normalization
input = HPF.ar(input, 100); // Don't consider vowels under 100
input = LPF.ar(input, 3000); // Don't consider vowels over 3000
chain2 = FFT(LocalBuf(2048), input);
localMax = PV_LocalMax(chain2, 30); // Try 0-50.0
input = IFFT(localMax);
aFilter = [BBandPass.ar(input, ~vowelA[0], filterReson), BBandPass.ar(input, ~vowelA[1], filterReson)];
aFilter = Mix.ar(aFilter);
eFilter = [BBandPass.ar(input, ~vowelE[0], filterReson), BBandPass.ar(input, ~vowelE[1], filterReson)];
eFilter = Mix.ar(eFilter);
iFilter = [BBandPass.ar(input, ~vowelI[0], filterReson), BBandPass.ar(input, ~vowelI[1], filterReson)];
iFilter = Mix.ar(iFilter);
oFilter = [BBandPass.ar(input, ~vowelO[0], filterReson), BBandPass.ar(input, ~vowelO[1], filterReson)];
oFilter = Mix.ar(oFilter);
uFilter = [BBandPass.ar(input, ~vowelU[0], filterReson), BBandPass.ar(input, ~vowelU[1], filterReson)];
uFilter = Mix.ar(uFilter);
rmsA = RunningSum.rms(aFilter, 1200) * 1000;
rmsE = RunningSum.rms(eFilter, 1200) * 1000;
rmsI = RunningSum.rms(iFilter, 1200) * 1000;
rmsO = RunningSum.rms(oFilter, 1200) * 1000;
rmsU = RunningSum.rms(uFilter, 1200) * 1000;
triggerInterval = s.options.blockSize; // Interval to send OSC messages
//triggerInterval = 1; // Interval to send OSC messages
SendTrig.ar(Impulse.ar(triggerInterval), 0, rmsA);
SendTrig.ar(Impulse.ar(triggerInterval), 1, rmsE);
SendTrig.ar(Impulse.ar(triggerInterval), 2, rmsI);
SendTrig.ar(Impulse.ar(triggerInterval), 3, rmsO);
SendTrig.ar(Impulse.ar(triggerInterval), 4, rmsU);
}
).play;
}).play
)