mirror of
https://github.com/scratchfoundation/scratch-audio.git
synced 2025-01-10 06:41:56 -05:00
make objects for each effect
This commit is contained in:
parent
f797ec80f3
commit
9120c3d430
9 changed files with 369 additions and 264 deletions
52
src/effects/EchoEffect.js
Normal file
52
src/effects/EchoEffect.js
Normal file
|
@ -0,0 +1,52 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
An echo effect
|
||||||
|
|
||||||
|
0 mutes the effect
|
||||||
|
Values up to 100 set the echo feedback amount,
|
||||||
|
increasing the time it takes the echo to fade away
|
||||||
|
|
||||||
|
Clamped 0-100
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function EchoEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.delay = new Tone.FeedbackDelay(0.25, 0.5);
|
||||||
|
|
||||||
|
this.effectSend.chain(this.delay, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(EchoEffect, Tone.Effect);
|
||||||
|
|
||||||
|
EchoEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
this.value = this.clamp(this.value, 0, 100);
|
||||||
|
|
||||||
|
// mute the effect if value is 0
|
||||||
|
if (this.value == 0) {
|
||||||
|
this.wet.value = 0;
|
||||||
|
} else {
|
||||||
|
this.wet.value = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
var feedback = (this.value / 100) * 0.75;
|
||||||
|
this.delay.feedback.rampTo(feedback, 1/60);
|
||||||
|
};
|
||||||
|
|
||||||
|
EchoEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
EchoEffect.prototype.clamp = function (input, min, max) {
|
||||||
|
return Math.min(Math.max(input, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = EchoEffect;
|
||||||
|
|
44
src/effects/FuzzEffect.js
Normal file
44
src/effects/FuzzEffect.js
Normal file
|
@ -0,0 +1,44 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A fuzz effect
|
||||||
|
|
||||||
|
Distortion
|
||||||
|
|
||||||
|
the value controls the wet/dry amount
|
||||||
|
|
||||||
|
Clamped 0-100
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function FuzzEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.distortion = new Tone.Distortion(1);
|
||||||
|
|
||||||
|
this.effectSend.chain(this.distortion, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(FuzzEffect, Tone.Effect);
|
||||||
|
|
||||||
|
FuzzEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
this.value = this.clamp(this.value, 0, 100);
|
||||||
|
|
||||||
|
this.distortion.wet.value = this.value / 100;
|
||||||
|
};
|
||||||
|
|
||||||
|
FuzzEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
FuzzEffect.prototype.clamp = function (input, min, max) {
|
||||||
|
return Math.min(Math.max(input, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = FuzzEffect;
|
||||||
|
|
42
src/effects/PanEffect.js
Normal file
42
src/effects/PanEffect.js
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A Pan effect
|
||||||
|
|
||||||
|
-100 puts the audio on the left channel, 0 centers it, 100 makes puts it on the right.
|
||||||
|
|
||||||
|
Clamped -100 to 100
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function PanEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.panner = new Tone.Panner();
|
||||||
|
|
||||||
|
this.effectSend.chain(this.panner, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(PanEffect, Tone.Effect);
|
||||||
|
|
||||||
|
PanEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
this.value = this.clamp(this.value, -100, 100);
|
||||||
|
|
||||||
|
this.panner.pan.value = this.value / 100;
|
||||||
|
};
|
||||||
|
|
||||||
|
PanEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
PanEffect.prototype.clamp = function (input, min, max) {
|
||||||
|
return Math.min(Math.max(input, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = PanEffect;
|
||||||
|
|
39
src/effects/PitchEffect.js
Normal file
39
src/effects/PitchEffect.js
Normal file
|
@ -0,0 +1,39 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A Pitch effect
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function PitchEffect () {
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.tone = new Tone();
|
||||||
|
}
|
||||||
|
|
||||||
|
PitchEffect.prototype.set = function (val, players) {
|
||||||
|
this.value = val;
|
||||||
|
this.updatePlayers(players);
|
||||||
|
};
|
||||||
|
|
||||||
|
PitchEffect.prototype.changeBy = function (val, players) {
|
||||||
|
this.set(this.value + val, players);
|
||||||
|
};
|
||||||
|
|
||||||
|
PitchEffect.prototype.getRatio = function () {
|
||||||
|
return this.tone.intervalToFrequencyRatio(this.value / 10);
|
||||||
|
};
|
||||||
|
|
||||||
|
PitchEffect.prototype.updatePlayers = function (players) {
|
||||||
|
if (!players) return;
|
||||||
|
|
||||||
|
var ratio = this.getRatio();
|
||||||
|
for (var i=0; i<players.length; i++) {
|
||||||
|
players[i].setPlaybackRate(ratio);
|
||||||
|
}
|
||||||
|
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = PitchEffect;
|
||||||
|
|
42
src/effects/ReverbEffect.js
Normal file
42
src/effects/ReverbEffect.js
Normal file
|
@ -0,0 +1,42 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A Reverb effect
|
||||||
|
|
||||||
|
The value controls the wet/dry amount of the effect
|
||||||
|
|
||||||
|
Clamped 0 to 100
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function ReverbEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.reverb = new Tone.Freeverb();
|
||||||
|
|
||||||
|
this.effectSend.chain(this.reverb, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(ReverbEffect, Tone.Effect);
|
||||||
|
|
||||||
|
ReverbEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
this.value = this.clamp(this.value, 0, 100);
|
||||||
|
|
||||||
|
this.reverb.wet.value = this.value / 100;
|
||||||
|
};
|
||||||
|
|
||||||
|
ReverbEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
ReverbEffect.prototype.clamp = function (input, min, max) {
|
||||||
|
return Math.min(Math.max(input, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = ReverbEffect;
|
||||||
|
|
59
src/effects/RoboticEffect.js
Normal file
59
src/effects/RoboticEffect.js
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A robot-voice effect
|
||||||
|
|
||||||
|
A feedback comb filter with a short delay time creates a low-pitched buzzing
|
||||||
|
The effect value controls the length of this delay time, changing the pitch
|
||||||
|
|
||||||
|
0 mutes the effect
|
||||||
|
|
||||||
|
Other values changes the pitch of the effect, in units of 10 steps per semitone
|
||||||
|
|
||||||
|
Not clamped
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function RoboticEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
var time = this._delayTimeForValue(100);
|
||||||
|
this.feedbackCombFilter = new Tone.FeedbackCombFilter(time, 0.9);
|
||||||
|
|
||||||
|
this.effectSend.chain(this.feedbackCombFilter, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(RoboticEffect, Tone.Effect);
|
||||||
|
|
||||||
|
RoboticEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
// mute the effect if value is 0
|
||||||
|
if (this.value == 0) {
|
||||||
|
this.wet.value = 0;
|
||||||
|
} else {
|
||||||
|
this.wet.value = 1;
|
||||||
|
}
|
||||||
|
|
||||||
|
// set delay time using the value
|
||||||
|
var time = this._delayTimeForValue(this.value);
|
||||||
|
this.feedbackCombFilter.delayTime.rampTo(time, 1/60);
|
||||||
|
};
|
||||||
|
|
||||||
|
RoboticEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
RoboticEffect.prototype._delayTimeForValue = function (val) {
|
||||||
|
// convert effect setting range, typically 0-100 but can be outside that,
|
||||||
|
// to a musical note, and return the period of the frequency of that note
|
||||||
|
var midiNote = ((val - 100) / 10) + 36;
|
||||||
|
var freq = Tone.Frequency(midiNote, 'midi').eval();
|
||||||
|
return 1 / freq;
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = RoboticEffect;
|
||||||
|
|
47
src/effects/WobbleEffect.js
Normal file
47
src/effects/WobbleEffect.js
Normal file
|
@ -0,0 +1,47 @@
|
||||||
|
/*
|
||||||
|
|
||||||
|
A wobble effect
|
||||||
|
|
||||||
|
A low frequency oscillator (LFO) controls a gain node
|
||||||
|
This creates an effect like tremolo
|
||||||
|
|
||||||
|
Clamped 0 to 100
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
function WobbleEffect () {
|
||||||
|
Tone.Effect.call(this);
|
||||||
|
|
||||||
|
this.value = 0;
|
||||||
|
|
||||||
|
this.wobbleLFO = new Tone.LFO(10, 0, 1).start();
|
||||||
|
this.wobbleGain = new Tone.Gain();
|
||||||
|
this.wobbleLFO.connect(this.wobbleGain.gain);
|
||||||
|
|
||||||
|
this.effectSend.chain(this.wobbleGain, this.effectReturn);
|
||||||
|
}
|
||||||
|
|
||||||
|
Tone.extend(WobbleEffect, Tone.Effect);
|
||||||
|
|
||||||
|
WobbleEffect.prototype.set = function (val) {
|
||||||
|
this.value = val;
|
||||||
|
|
||||||
|
this.value = this.clamp(this.value, 0, 100);
|
||||||
|
|
||||||
|
this.wet.value = this.value / 100;
|
||||||
|
|
||||||
|
this.wobbleLFO.frequency.rampTo(this.value / 10, 1/60);
|
||||||
|
};
|
||||||
|
|
||||||
|
WobbleEffect.prototype.changeBy = function (val) {
|
||||||
|
this.set(this.value + val);
|
||||||
|
};
|
||||||
|
|
||||||
|
WobbleEffect.prototype.clamp = function (input, min, max) {
|
||||||
|
return Math.min(Math.max(input, min), max);
|
||||||
|
};
|
||||||
|
|
||||||
|
module.exports = WobbleEffect;
|
||||||
|
|
169
src/index.js
169
src/index.js
|
@ -1,5 +1,14 @@
|
||||||
var log = require('./log');
|
var log = require('./log');
|
||||||
var Tone = require('tone');
|
var Tone = require('tone');
|
||||||
|
|
||||||
|
var PitchEffect = require('./effects/PitchEffect');
|
||||||
|
var EchoEffect = require('./effects/EchoEffect');
|
||||||
|
var PanEffect = require('./effects/PanEffect');
|
||||||
|
var RoboticEffect = require('./effects/RoboticEffect');
|
||||||
|
var ReverbEffect = require('./effects/ReverbEffect');
|
||||||
|
var FuzzEffect = require('./effects/FuzzEffect');
|
||||||
|
var WobbleEffect = require('./effects/WobbleEffect');
|
||||||
|
|
||||||
var SoundPlayer = require('./SoundPlayer');
|
var SoundPlayer = require('./SoundPlayer');
|
||||||
var Soundfont = require('soundfont-player');
|
var Soundfont = require('soundfont-player');
|
||||||
var ADPCMSoundLoader = require('./ADPCMSoundLoader');
|
var ADPCMSoundLoader = require('./ADPCMSoundLoader');
|
||||||
|
@ -11,52 +20,31 @@ function AudioEngine (sounds) {
|
||||||
this.tone = new Tone();
|
this.tone = new Tone();
|
||||||
|
|
||||||
// effects setup
|
// effects setup
|
||||||
// each effect has a single parameter controlled by the effects block
|
this.pitchEffect = new PitchEffect();
|
||||||
|
this.echoEffect = new EchoEffect();
|
||||||
this.delay = new Tone.FeedbackDelay(0.25, 0.5);
|
this.panEffect = new PanEffect();
|
||||||
this.panner = new Tone.Panner();
|
this.reverbEffect = new ReverbEffect();
|
||||||
this.reverb = new Tone.Freeverb();
|
this.fuzzEffect = new FuzzEffect();
|
||||||
this.distortion = new Tone.Distortion(1);
|
this.wobbleEffect = new WobbleEffect();
|
||||||
this.pitchEffectValue;
|
this.roboticEffect = new RoboticEffect();
|
||||||
|
|
||||||
this.robotic = new Tone.Effect();
|
|
||||||
// use the period of a musical note to set the feedback filter delay time
|
|
||||||
var delayTime = 1 / Tone.Frequency('Gb3').eval();
|
|
||||||
var robotFilter = new Tone.FeedbackCombFilter(delayTime, 0.9);
|
|
||||||
this.robotic.effectSend.chain(robotFilter, this.robotic.effectReturn);
|
|
||||||
|
|
||||||
this.wobble = new Tone.Effect();
|
|
||||||
var wobbleLFO = new Tone.LFO(10, 0, 1).start();
|
|
||||||
var wobbleGain = new Tone.Gain();
|
|
||||||
wobbleLFO.connect(wobbleGain.gain);
|
|
||||||
this.wobble.effectSend.chain(wobbleGain, this.wobble.effectReturn);
|
|
||||||
|
|
||||||
// telephone effect - simulating the 'tinny' sound coming over a phone line
|
|
||||||
// using a lowpass filter and a highpass filter
|
|
||||||
this.telephone = new Tone.Effect();
|
|
||||||
var telephoneLP = new Tone.Filter(1200, 'lowpass', -24);
|
|
||||||
var telephoneHP = new Tone.Filter(800, 'highpass', -24);
|
|
||||||
this.telephone.effectSend.chain(telephoneLP, telephoneHP, this.telephone.effectReturn);
|
|
||||||
|
|
||||||
// the effects are chained to an effects node for this clone, then to the master output
|
// the effects are chained to an effects node for this clone, then to the master output
|
||||||
// so audio is sent from each player or instrument, through the effects in order, then out
|
// so audio is sent from each player or instrument, through the effects in order, then out
|
||||||
// note that the pitch effect works differently - it sets the playback rate for each player
|
// note that the pitch effect works differently - it sets the playback rate for each player
|
||||||
this.effectsNode = new Tone.Gain();
|
this.effectsNode = new Tone.Gain();
|
||||||
this.effectsNode.chain(
|
this.effectsNode.chain(
|
||||||
this.robotic, this.distortion, this.delay, this.telephone,
|
this.roboticEffect, this.fuzzEffect, this.echoEffect,
|
||||||
this.wobble, this.panner, this.reverb, Tone.Master);
|
this.wobbleEffect, this.panEffect, this.reverbEffect, Tone.Master);
|
||||||
|
|
||||||
// reset effects to their default parameters
|
// reset effects to their default parameters
|
||||||
this.clearEffects();
|
this.clearEffects();
|
||||||
|
|
||||||
this.effectNames = ['PITCH', 'PAN', 'ECHO', 'REVERB', 'FUZZ', 'TELEPHONE', 'WOBBLE', 'ROBOTIC'];
|
this.effectNames = ['PITCH', 'PAN', 'ECHO', 'REVERB', 'FUZZ', 'TELEPHONE', 'WOBBLE', 'ROBOTIC'];
|
||||||
|
|
||||||
|
|
||||||
// load sounds
|
// load sounds
|
||||||
|
|
||||||
this.soundPlayers = [];
|
this.soundPlayers = [];
|
||||||
this.loadSounds(sounds);
|
this.loadSounds(sounds);
|
||||||
// Tone.Buffer.on('load', this._soundsLoaded.bind(this));
|
|
||||||
|
|
||||||
// soundfont setup
|
// soundfont setup
|
||||||
|
|
||||||
|
@ -69,21 +57,12 @@ function AudioEngine (sounds) {
|
||||||
'music_box', 'steel_drums', 'marimba', 'lead_1_square', 'fx_4_atmosphere'];
|
'music_box', 'steel_drums', 'marimba', 'lead_1_square', 'fx_4_atmosphere'];
|
||||||
|
|
||||||
this.instrumentNum;
|
this.instrumentNum;
|
||||||
this.setInstrument(0);
|
this.setInstrument(1);
|
||||||
|
|
||||||
// tempo in bpm (beats per minute)
|
// tempo in bpm (beats per minute)
|
||||||
// default is 60bpm
|
// default is 60bpm
|
||||||
|
|
||||||
this.currentTempo = 60;
|
this.currentTempo = 60;
|
||||||
|
|
||||||
// theremin setup
|
|
||||||
|
|
||||||
this.theremin = new Tone.Synth();
|
|
||||||
this.portamentoTime = 0.25;
|
|
||||||
this.thereminVibrato = new Tone.Vibrato(4, 0.5);
|
|
||||||
this.theremin.chain(this.thereminVibrato, this.effectsNode);
|
|
||||||
this.thereminTimeout;
|
|
||||||
this.thereminIsPlaying = false;
|
|
||||||
}
|
}
|
||||||
|
|
||||||
AudioEngine.prototype.loadSounds = function (sounds) {
|
AudioEngine.prototype.loadSounds = function (sounds) {
|
||||||
|
@ -151,30 +130,6 @@ AudioEngine.prototype.playNoteForBeats = function (note, beats) {
|
||||||
*/
|
*/
|
||||||
};
|
};
|
||||||
|
|
||||||
AudioEngine.prototype.playThereminForBeats = function (note, beats) {
|
|
||||||
// if the theremin is playing
|
|
||||||
// ramp to new frequency
|
|
||||||
// else
|
|
||||||
// trigger attack
|
|
||||||
// create a timeout for slightly longer than the duration of the block
|
|
||||||
// that releases the theremin - so we can slide continuously between
|
|
||||||
// successive notes without releasing and re-attacking
|
|
||||||
|
|
||||||
var freq = this._midiToFreq(note);
|
|
||||||
|
|
||||||
if (this.thereminIsPlaying) {
|
|
||||||
this.theremin.frequency.rampTo(freq, this.portamentoTime);
|
|
||||||
} else {
|
|
||||||
this.theremin.triggerAttack(freq);
|
|
||||||
this.thereminIsPlaying = true;
|
|
||||||
}
|
|
||||||
clearTimeout(this.thereminTimeout);
|
|
||||||
this.thereminTimeout = setTimeout(function () {
|
|
||||||
this.theremin.triggerRelease();
|
|
||||||
this.thereminIsPlaying = false;
|
|
||||||
}.bind(this), (1000 * beats) + 100);
|
|
||||||
};
|
|
||||||
|
|
||||||
AudioEngine.prototype._midiToFreq = function (midiNote) {
|
AudioEngine.prototype._midiToFreq = function (midiNote) {
|
||||||
var freq = this.tone.intervalToFrequencyRatio(midiNote - 60) * 261.63; // 60 is C4
|
var freq = this.tone.intervalToFrequencyRatio(midiNote - 60) * 261.63; // 60 is C4
|
||||||
return freq;
|
return freq;
|
||||||
|
@ -204,28 +159,25 @@ AudioEngine.prototype.stopAllSounds = function () {
|
||||||
AudioEngine.prototype.setEffect = function (effect, value) {
|
AudioEngine.prototype.setEffect = function (effect, value) {
|
||||||
switch (effect) {
|
switch (effect) {
|
||||||
case 'PITCH':
|
case 'PITCH':
|
||||||
this._setPitchShift(value);
|
this.pitchEffect.set(value, this.soundPlayers);
|
||||||
break;
|
break;
|
||||||
case 'PAN':
|
case 'PAN':
|
||||||
this.panner.pan.value = value / 100;
|
this.panEffect.set(value);
|
||||||
break;
|
break;
|
||||||
case 'ECHO':
|
case 'ECHO':
|
||||||
this.delay.wet.value = (value / 100) / 2; // max 50% wet
|
this.echoEffect.set(value);
|
||||||
break;
|
break;
|
||||||
case 'REVERB':
|
case 'REVERB':
|
||||||
this.reverb.wet.value = value / 100;
|
this.reverbEffect.set(value);
|
||||||
break;
|
break;
|
||||||
case 'FUZZ' :
|
case 'FUZZ' :
|
||||||
this.distortion.wet.value = value / 100;
|
this.fuzzEffect.set(value);
|
||||||
break;
|
|
||||||
case 'TELEPHONE' :
|
|
||||||
this.telephone.wet.value = value / 100;
|
|
||||||
break;
|
break;
|
||||||
case 'WOBBLE' :
|
case 'WOBBLE' :
|
||||||
this.wobble.wet.value = value / 100;
|
this.wobbleEffect.set(value);
|
||||||
break;
|
break;
|
||||||
case 'ROBOTIC' :
|
case 'ROBOTIC' :
|
||||||
this.robotic.wet.value = value / 100;
|
this.roboticEffect.set(value);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
@ -233,66 +185,46 @@ AudioEngine.prototype.setEffect = function (effect, value) {
|
||||||
AudioEngine.prototype.changeEffect = function (effect, value) {
|
AudioEngine.prototype.changeEffect = function (effect, value) {
|
||||||
switch (effect) {
|
switch (effect) {
|
||||||
case 'PITCH':
|
case 'PITCH':
|
||||||
this._setPitchShift(this.pitchEffectValue + Number(value));
|
this.pitchEffect.changeBy(value, this.soundPlayers);
|
||||||
break;
|
break;
|
||||||
case 'PAN':
|
case 'PAN':
|
||||||
this.panner.pan.value += value / 100;
|
this.panEffect.changeBy(value);
|
||||||
this.panner.pan.value = this._clamp(this.panner.pan.value, -1, 1);
|
|
||||||
break;
|
break;
|
||||||
case 'ECHO':
|
case 'ECHO':
|
||||||
this.delay.wet.value += (value / 100) / 2; // max 50% wet
|
this.echoEffect.changeBy(value);
|
||||||
this.delay.wet.value = this._clamp(this.delay.wet.value, 0, 0.5);
|
|
||||||
break;
|
break;
|
||||||
case 'REVERB':
|
case 'REVERB':
|
||||||
this.reverb.wet.value += value / 100;
|
this.reverbEffect.changeBy(value);
|
||||||
this.reverb.wet.value = this._clamp(this.reverb.wet.value, 0, 1);
|
|
||||||
break;
|
break;
|
||||||
case 'FUZZ' :
|
case 'FUZZ' :
|
||||||
this.distortion.wet.value += value / 100;
|
this.fuzzEffect.changeBy(value);
|
||||||
this.distortion.wet.value = this._clamp(this.distortion.wet.value, 0, 1);
|
|
||||||
break;
|
|
||||||
case 'TELEPHONE' :
|
|
||||||
this.telephone.wet.value += value / 100;
|
|
||||||
this.telephone.wet.value = this._clamp(this.telephone.wet.value, 0, 1);
|
|
||||||
break;
|
break;
|
||||||
case 'WOBBLE' :
|
case 'WOBBLE' :
|
||||||
this.wobble.wet.value += value / 100;
|
this.wobbleEffect.changeBy(value);
|
||||||
this.wobble.wet.value = this._clamp(this.wobble.wet.value, 0, 1);
|
|
||||||
break;
|
break;
|
||||||
case 'ROBOTIC' :
|
case 'ROBOTIC' :
|
||||||
this.robotic.wet.value += value / 100;
|
this.roboticEffect.changeBy(value);
|
||||||
this.robotic.wet.value = this._clamp(this.robotic.wet.value, 0, 1);
|
|
||||||
break;
|
break;
|
||||||
|
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
AudioEngine.prototype._setPitchShift = function (value) {
|
AudioEngine.prototype.clearEffects = function () {
|
||||||
this.pitchEffectValue = value;
|
this.echoEffect.set(0);
|
||||||
|
this.panEffect.set(0);
|
||||||
|
this.reverbEffect.set(0);
|
||||||
|
this.fuzzEffect.set(0);
|
||||||
|
this.roboticEffect.set(0);
|
||||||
|
this.wobbleEffect.set(0);
|
||||||
|
this.pitchEffect.set(0, this.soundPlayers);
|
||||||
|
|
||||||
if (!this.soundPlayers) {
|
this.effectsNode.gain.value = 1;
|
||||||
return;
|
|
||||||
}
|
|
||||||
|
|
||||||
var ratio = this._getPitchRatio();
|
|
||||||
this._setPlaybackRateForAllSoundPlayers(ratio);
|
|
||||||
|
|
||||||
};
|
|
||||||
|
|
||||||
AudioEngine.prototype._setPlaybackRateForAllSoundPlayers = function (rate) {
|
|
||||||
for (var i=0; i<this.soundPlayers.length; i++) {
|
|
||||||
this.soundPlayers[i].setPlaybackRate(rate);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
AudioEngine.prototype._getPitchRatio = function () {
|
|
||||||
return this.tone.intervalToFrequencyRatio(this.pitchEffectValue / 10);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
AudioEngine.prototype.setInstrument = function (instrumentNum) {
|
AudioEngine.prototype.setInstrument = function (instrumentNum) {
|
||||||
this.instrumentNum = instrumentNum;
|
this.instrumentNum = instrumentNum - 1;
|
||||||
|
|
||||||
return Soundfont.instrument(Tone.context, this.instrumentNames[instrumentNum]).then(
|
return Soundfont.instrument(Tone.context, this.instrumentNames[this.instrumentNum]).then(
|
||||||
function (inst) {
|
function (inst) {
|
||||||
this.instrument = inst;
|
this.instrument = inst;
|
||||||
this.instrument.connect(this.effectsNode);
|
this.instrument.connect(this.effectsNode);
|
||||||
|
@ -300,19 +232,6 @@ AudioEngine.prototype.setInstrument = function (instrumentNum) {
|
||||||
);
|
);
|
||||||
};
|
};
|
||||||
|
|
||||||
AudioEngine.prototype.clearEffects = function () {
|
|
||||||
this.delay.wet.value = 0;
|
|
||||||
this.panner.pan.value = 0;
|
|
||||||
this.reverb.wet.value = 0;
|
|
||||||
this.distortion.wet.value = 0;
|
|
||||||
this.robotic.wet.value = 0;
|
|
||||||
this.wobble.wet.value = 0;
|
|
||||||
this.telephone.wet.value = 0;
|
|
||||||
this._setPitchShift(0);
|
|
||||||
|
|
||||||
this.effectsNode.gain.value = 1;
|
|
||||||
};
|
|
||||||
|
|
||||||
AudioEngine.prototype.setVolume = function (value) {
|
AudioEngine.prototype.setVolume = function (value) {
|
||||||
var vol = this._clamp(value, 0, 100);
|
var vol = this._clamp(value, 0, 100);
|
||||||
vol /= 100;
|
vol /= 100;
|
||||||
|
|
139
src/vocoder.js
139
src/vocoder.js
|
@ -1,139 +0,0 @@
|
||||||
/*
|
|
||||||
|
|
||||||
This is a port of Jamison Dance's port of Chris Wilson's WebAudio vocoder:
|
|
||||||
https://github.com/jergason/Vocoder
|
|
||||||
https://github.com/cwilso/Vocoder
|
|
||||||
|
|
||||||
I have adapted it to use the ToneJs library, making it a Tone Effect that
|
|
||||||
can be added to an audio effects chain.
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
/*
|
|
||||||
|
|
||||||
The MIT License (MIT)
|
|
||||||
|
|
||||||
Copyright (c) 2014 Chris Wilson
|
|
||||||
|
|
||||||
Permission is hereby granted, free of charge, to any person obtaining a copy
|
|
||||||
of this software and associated documentation files (the "Software"), to deal
|
|
||||||
in the Software without restriction, including without limitation the rights
|
|
||||||
to use, copy, modify, merge, publish, distribute, sublicense, and/or sell
|
|
||||||
copies of the Software, and to permit persons to whom the Software is
|
|
||||||
furnished to do so, subject to the following conditions:
|
|
||||||
|
|
||||||
The above copyright notice and this permission notice shall be included in all
|
|
||||||
copies or substantial portions of the Software.
|
|
||||||
|
|
||||||
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
|
|
||||||
IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY,
|
|
||||||
FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE
|
|
||||||
AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER
|
|
||||||
LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM,
|
|
||||||
OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE
|
|
||||||
SOFTWARE.
|
|
||||||
|
|
||||||
*/
|
|
||||||
|
|
||||||
var Tone = require('tone');
|
|
||||||
|
|
||||||
function Vocoder () {
|
|
||||||
Tone.Effect.call(this);
|
|
||||||
|
|
||||||
this.modulatorInput = new Tone.Gain();
|
|
||||||
this.carrierInput = new Tone.Gain();
|
|
||||||
this.outputGain = new Tone.Gain();
|
|
||||||
|
|
||||||
this.oscillatorNode;
|
|
||||||
|
|
||||||
this.vocoderBands = this.generateVocoderBands(55, 7040, 8);
|
|
||||||
this.initBandpassFilters();
|
|
||||||
this.createCarrier();
|
|
||||||
|
|
||||||
this.effectSend.connect(this.modulatorInput);
|
|
||||||
this.outputGain.connect(this.effectReturn);
|
|
||||||
}
|
|
||||||
|
|
||||||
Tone.extend(Vocoder, Tone.Effect);
|
|
||||||
|
|
||||||
Vocoder.prototype.setCarrierOscFrequency = function (freq) {
|
|
||||||
this.oscillatorNode.frequency.rampTo(freq, 0.05);
|
|
||||||
};
|
|
||||||
|
|
||||||
// this function algorithmically calculates vocoder bands, distributing evenly
|
|
||||||
// from startFreq to endFreq, splitting evenly (logarhythmically) into a given numBands.
|
|
||||||
Vocoder.prototype.generateVocoderBands = function (startFreq, endFreq, numBands) {
|
|
||||||
// Remember: 1200 cents in octave, 100 cents per semitone
|
|
||||||
|
|
||||||
var totalRangeInCents = 1200 * Math.log( endFreq / startFreq ) / Math.LN2;
|
|
||||||
var centsPerBand = totalRangeInCents / numBands;
|
|
||||||
var scale = Math.pow( 2, centsPerBand / 1200 ); // This is the scaling for successive bands
|
|
||||||
|
|
||||||
var vocoderBands = [];
|
|
||||||
var currentFreq = startFreq;
|
|
||||||
|
|
||||||
for (var i=0; i<numBands; i++) {
|
|
||||||
vocoderBands[i] = new Object();
|
|
||||||
vocoderBands[i].frequency = currentFreq;
|
|
||||||
currentFreq = currentFreq * scale;
|
|
||||||
}
|
|
||||||
|
|
||||||
return vocoderBands;
|
|
||||||
};
|
|
||||||
|
|
||||||
Vocoder.prototype.initBandpassFilters = function () {
|
|
||||||
|
|
||||||
var FILTER_QUALITY = 6; // The Q value for the carrier and modulator filters
|
|
||||||
|
|
||||||
// Set up a high-pass filter to add back in the fricatives, etc.
|
|
||||||
var hpFilter = new Tone.Filter(8000, 'highpass');
|
|
||||||
this.modulatorInput.connect(hpFilter);
|
|
||||||
hpFilter.connect(this.outputGain);
|
|
||||||
|
|
||||||
for (var i=0; i<this.vocoderBands.length; i++) {
|
|
||||||
|
|
||||||
// CREATE THE MODULATOR CHAIN
|
|
||||||
// create the bandpass filter in the modulator chain
|
|
||||||
var modulatorFilter = new Tone.Filter(this.vocoderBands[i].frequency, 'bandpass', -24);
|
|
||||||
modulatorFilter.Q.value = FILTER_QUALITY;
|
|
||||||
this.modulatorInput.connect(modulatorFilter);
|
|
||||||
|
|
||||||
// create a post-filtering gain to bump the levels up.
|
|
||||||
var modulatorFilterPostGain = new Tone.Gain(6);
|
|
||||||
modulatorFilter.connect(modulatorFilterPostGain);
|
|
||||||
|
|
||||||
// add a rectifier with a lowpass filter to turn the bandpass filtered signal
|
|
||||||
// into a smoothed control signal for the carrier filter
|
|
||||||
var rectifier = new Tone.WaveShaper([1,0,1]);
|
|
||||||
modulatorFilterPostGain.connect(rectifier);
|
|
||||||
var rectifierLowPass = new Tone.Filter(50, 'lowpass');
|
|
||||||
rectifier.connect(rectifierLowPass);
|
|
||||||
|
|
||||||
// Create the bandpass filter in the carrier chain
|
|
||||||
var carrierFilter = new Tone.Filter(this.vocoderBands[i].frequency, 'bandpass', -24);
|
|
||||||
carrierFilter.Q.value = FILTER_QUALITY;
|
|
||||||
this.carrierInput.connect(carrierFilter);
|
|
||||||
var carrierFilterPostGain = new Tone.Gain(10);
|
|
||||||
carrierFilter.connect(carrierFilterPostGain);
|
|
||||||
|
|
||||||
// Create the carrier band gain node
|
|
||||||
var bandGain = new Tone.Gain(0);
|
|
||||||
carrierFilterPostGain.connect(bandGain);
|
|
||||||
|
|
||||||
// the modulator band filter output controls the gain of the carrier band
|
|
||||||
rectifierLowPass.connect(bandGain.gain);
|
|
||||||
|
|
||||||
bandGain.connect(this.outputGain);
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
Vocoder.prototype.createCarrier = function () {
|
|
||||||
this.oscillatorNode = new Tone.Oscillator('C3', 'sawtooth').start();
|
|
||||||
this.oscillatorNode.connect(this.carrierInput);
|
|
||||||
|
|
||||||
var noiseNode = new Tone.Noise('white').start();
|
|
||||||
noiseNode.volume.value = -12;
|
|
||||||
noiseNode.connect(this.carrierInput);
|
|
||||||
};
|
|
||||||
|
|
||||||
module.exports = Vocoder;
|
|
Loading…
Reference in a new issue