Skip to content

Commit f14890c

Browse files
authored
Merge pull request #292 from therewasaguy/synth-default-params
synth play method has a default sustain
2 parents 09f822d + ccc07a9 commit f14890c

File tree

5 files changed

+56
-56
lines changed

5 files changed

+56
-56
lines changed

examples/_monosynth_basic/sketch.js

+1-1
Original file line numberDiff line numberDiff line change
@@ -14,7 +14,7 @@ function setup() {
1414

1515
function mousePressed() {
1616
// pick a random midi note
17-
var midiVal = round( random(50,72) );
17+
var midiVal = midiToFreq(round( random(50,72) ));
1818
monoSynth.triggerAttack(midiVal, random() );
1919
}
2020

src/audioVoice.js

-17
Original file line numberDiff line numberDiff line change
@@ -17,23 +17,6 @@ define(function() {
1717
p5sound.soundArray.push(this);
1818
};
1919

20-
/**
21-
* This method converts midi notes specified as a string "C4", "Eb3"...etc
22-
* to frequency
23-
* @private
24-
* @method _setNote
25-
* @param {String} note
26-
*/
27-
p5.AudioVoice.prototype._setNote = function(note) {
28-
var wholeNotes = {A:21, B:23, C:24, D:26, E:28, F:29, G:31};
29-
var value = wholeNotes[ note[0] ];
30-
var octave = typeof Number(note.slice(-1)) === 'number'? note.slice(-1) : 0;
31-
value += 12 * octave;
32-
value = note[1] === '#' ? value+1 : note[1] ==='b' ? value - 1 : value;
33-
//return midi value converted to frequency
34-
return p5.prototype.midiToFreq(value);
35-
};
36-
3720
p5.AudioVoice.prototype.play = function (note, velocity, secondsFromNow, sustime) {
3821
};
3922

src/helpers.js

+26-2
Original file line numberDiff line numberDiff line change
@@ -65,10 +65,33 @@ define(function (require) {
6565
* }
6666
* </code></div>
6767
*/
68-
p5.prototype.midiToFreq = function(m) {
68+
var midiToFreq = p5.prototype.midiToFreq = function(m) {
6969
return 440 * Math.pow(2, (m-69)/12.0);
7070
};
7171

72+
// This method converts ANSI notes specified as a string "C4", "Eb3" to a frequency
73+
noteToFreq = function(note) {
74+
if (typeof note !== 'string') {
75+
return note;
76+
}
77+
var wholeNotes = {A:21, B:23, C:24, D:26, E:28, F:29, G:31};
78+
var value = wholeNotes[ note[0].toUpperCase() ];
79+
var octave = ~~note.slice(-1);
80+
value += 12 * (octave -1);
81+
82+
switch(note[1]) {
83+
case '#':
84+
value += 1;
85+
break;
86+
case 'b':
87+
value -= 1;
88+
break;
89+
default:
90+
break;
91+
}
92+
return midiToFreq(value);
93+
}
94+
7295
/**
7396
* List the SoundFile formats that you will include. LoadSound
7497
* will search your directory for these extensions, and will pick
@@ -202,7 +225,8 @@ define(function (require) {
202225
};
203226

204227
return {
205-
midiToFreq: p5.prototype.midiToFreq
228+
midiToFreq: midiToFreq,
229+
noteToFreq: noteToFreq
206230
};
207231

208232
});

src/monosynth.js

+8-11
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,9 @@ define(function (require) {
33

44
var p5sound = require('master');
55
var AudioVoice = require('audioVoice');
6+
var noteToFreq = require('helpers').noteToFreq;
7+
8+
var DEFAULT_SUSTAIN = 0.15;
69

710
/**
811
* A MonoSynth is used as a single voice for sound synthesis.
@@ -125,11 +128,8 @@ define(function (require) {
125128
*
126129
*/
127130
p5.MonoSynth.prototype.play = function (note, velocity, secondsFromNow, susTime) {
128-
// set range of env (TO DO: allow this to be scheduled in advance)
129-
var susTime = susTime || this.sustain;
130-
this.susTime = susTime;
131-
this.triggerAttack(note, velocity, secondsFromNow);
132-
this.triggerRelease(secondsFromNow + susTime);
131+
this.triggerAttack(note, velocity, ~~secondsFromNow);
132+
this.triggerRelease(~~secondsFromNow + (susTime || DEFAULT_SUSTAIN));
133133
};
134134

135135
/**
@@ -159,12 +159,9 @@ define(function (require) {
159159
* </code></div>
160160
*/
161161
p5.MonoSynth.prototype.triggerAttack = function (note, velocity, secondsFromNow) {
162-
var secondsFromNow = secondsFromNow || 0;
163-
164-
//triggerAttack uses ._setNote to convert a midi string to a frequency if necessary
165-
var freq = typeof note === 'string' ? this._setNote(note)
166-
: typeof note === 'number' ? note : 440;
167-
var vel = velocity || 1;
162+
var secondsFromNow = ~~secondsFromNow;
163+
var freq = noteToFreq(note);
164+
var vel = velocity || 0.1;
168165
this._isOn = true;
169166
this.oscillator.freq(freq, 0, secondsFromNow);
170167
this.env.ramp(this.output, secondsFromNow, vel);

src/polysynth.js

+21-25
Original file line numberDiff line numberDiff line change
@@ -1,11 +1,9 @@
11
'use strict';
22
define(function (require) {
33

4-
54
var p5sound = require('master');
65
var TimelineSignal = require('Tone/signal/TimelineSignal');
7-
require('sndcore');
8-
6+
var noteToFreq = require('helpers').noteToFreq;
97

108
/**
119
* An AudioVoice is used as a single voice for sound synthesis.
@@ -247,30 +245,27 @@ define(function (require) {
247245
* </code></div>
248246
*/
249247
p5.PolySynth.prototype.noteAttack = function (_note, _velocity, secondsFromNow) {
250-
var now = p5sound.audiocontext.currentTime;
251-
252248
//this value goes to the audiovoices which handle their own scheduling
253-
var tFromNow = secondsFromNow || 0;
249+
var secondsFromNow = ~~secondsFromNow;
254250

255251
//this value is used by this._voicesInUse
256-
var t = now + tFromNow;
252+
var acTime = p5sound.audiocontext.currentTime + secondsFromNow;
257253

258254
//Convert note to frequency if necessary. This is because entries into this.notes
259255
//should be based on frequency for the sake of consistency.
260-
var note = typeof _note === 'string' ? this.AudioVoice.prototype._setNote(_note)
261-
: typeof _note === 'number' ? _note : 440;
262-
var velocity = _velocity === undefined ? 1 : _velocity;
256+
var note = noteToFreq(_note);
257+
var velocity = _velocity || 0.1;
263258

264259
var currentVoice;
265260

266261
//Release the note if it is already playing
267-
if ( this.notes[note] !== undefined && this.notes[note].getValueAtTime(t) !== null) {
268-
this.noteRelease(note,0);
262+
if (this.notes[note] && this.notes[note].getValueAtTime(acTime) !== null) {
263+
this.noteRelease(note, 0);
269264
}
270265

271266
//Check to see how many voices are in use at the time the note will start
272-
if(this._voicesInUse.getValueAtTime(t) < this.maxVoices) {
273-
currentVoice = Math.max(~~this._voicesInUse.getValueAtTime(t), 0);
267+
if (this._voicesInUse.getValueAtTime(acTime) < this.maxVoices) {
268+
currentVoice = Math.max(~~this._voicesInUse.getValueAtTime(acTime), 0);
274269
}
275270
//If we are exceeding the polyvalue, bump off the oldest notes and replace
276271
//with a new note
@@ -285,23 +280,23 @@ define(function (require) {
285280
//Overrite the entry in the notes object. A note (frequency value)
286281
//corresponds to the index of the audiovoice that is playing it
287282
this.notes[note] = new TimelineSignal();
288-
this.notes[note].setValueAtTime(currentVoice,t);
283+
this.notes[note].setValueAtTime(currentVoice, acTime);
289284

290285
//Find the scheduled change in this._voicesInUse that will be previous to this new note
291286
//Add 1 and schedule this value at time 't', when this note will start playing
292-
var previousVal = this._voicesInUse._searchBefore(t) === null ? 0 : this._voicesInUse._searchBefore(t).value;
293-
this._voicesInUse.setValueAtTime(previousVal + 1, t);
287+
var previousVal = this._voicesInUse._searchBefore(acTime) === null ? 0 : this._voicesInUse._searchBefore(acTime).value;
288+
this._voicesInUse.setValueAtTime(previousVal + 1, acTime);
294289

295290
//Then update all scheduled values that follow to increase by 1
296-
this._updateAfter(t, 1);
291+
this._updateAfter(acTime, 1);
297292

298293
this._newest = currentVoice;
299294
//The audiovoice handles the actual scheduling of the note
300295
if (typeof velocity === 'number') {
301-
var maxRange = 1 / this._voicesInUse.getValueAtTime(t) * 2;
296+
var maxRange = 1 / this._voicesInUse.getValueAtTime(acTime) * 2;
302297
velocity = velocity > maxRange ? maxRange : velocity;
303298
}
304-
this.audiovoices[currentVoice].triggerAttack(note, velocity, tFromNow);
299+
this.audiovoices[currentVoice].triggerAttack(note, velocity, secondsFromNow);
305300
};
306301

307302
/**
@@ -368,16 +363,16 @@ define(function (require) {
368363
});
369364
this._voicesInUse.setValueAtTime(0, t);
370365
for (var n in this.notes) {
371-
this.notes[n].setValueAtTime(null, t)
366+
this.notes[n].dispose();
367+
delete this.notes[n];
372368
}
373369
return;
374370
}
375371

376372
//Make sure note is in frequency inorder to query the this.notes object
377-
var note = typeof _note === 'string' ? this.AudioVoice.prototype._setNote(_note)
378-
: typeof _note === 'number' ? _note : this.audiovoices[this._newest].oscillator.freq().value;
373+
var note = noteToFreq(_note);
379374

380-
if (this.notes[note].getValueAtTime(t) === null) {
375+
if (!this.notes[note] || this.notes[note].getValueAtTime(t) === null) {
381376
console.warn('Cannot release a note that is not already playing');
382377
} else {
383378
//Find the scheduled change in this._voicesInUse that will be previous to this new note
@@ -390,7 +385,8 @@ define(function (require) {
390385
}
391386

392387
this.audiovoices[ this.notes[note].getValueAtTime(t) ].triggerRelease(tFromNow);
393-
this.notes[note].setValueAtTime( null, t);
388+
this.notes[note].dispose();
389+
delete this.notes[note];
394390

395391
this._newest = this._newest === 0 ? 0 : (this._newest - 1) % (this.maxVoices - 1);
396392
}

0 commit comments

Comments
 (0)