mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-10 02:48:47 +00:00
693d96b1b7
the native onended is too flakey. addresses #48
168 lines
No EOL
4.1 KiB
JavaScript
168 lines
No EOL
4.1 KiB
JavaScript
define(["Tone/core/Tone", "Tone/signal/Signal"], function(Tone){
|
|
|
|
"use strict";
|
|
|
|
/**
|
|
* @class a sample accurate clock built on an oscillator.
|
|
* Invokes the tick method at the set rate
|
|
*
|
|
* @internal
|
|
* @constructor
|
|
* @extends {Tone}
|
|
* @param {number} rate the rate of the callback
|
|
* @param {function} callback the callback to be invoked with the time of the audio event
|
|
*/
|
|
Tone.Clock = function(rate, callback){
|
|
|
|
/**
|
|
* the oscillator
|
|
* @type {OscillatorNode}
|
|
* @private
|
|
*/
|
|
this._oscillator = null;
|
|
|
|
/**
|
|
* the script processor which listens to the oscillator
|
|
* @type {ScriptProcessorNode}
|
|
* @private
|
|
*/
|
|
this._jsNode = this.context.createScriptProcessor(this.bufferSize, 1, 1);
|
|
this._jsNode.onaudioprocess = this._processBuffer.bind(this);
|
|
|
|
/**
|
|
* the rate control signal
|
|
* @type {Tone.Signal}
|
|
* @private
|
|
*/
|
|
this._controlSignal = new Tone.Signal(1);
|
|
|
|
/**
|
|
* whether the tick is on the up or down
|
|
* @type {boolean}
|
|
* @private
|
|
*/
|
|
this._upTick = false;
|
|
|
|
/**
|
|
* the callback which is invoked on every tick
|
|
* with the time of that tick as the argument
|
|
* @type {function(number)}
|
|
*/
|
|
this.tick = this.defaultArg(callback, function(){});
|
|
|
|
//setup
|
|
this._jsNode.noGC();
|
|
this.setRate(rate);
|
|
};
|
|
|
|
Tone.extend(Tone.Clock);
|
|
|
|
/**
|
|
* set the rate of the clock
|
|
* optionally ramp to the rate over the rampTime
|
|
* @param {Tone.Time} rate
|
|
* @param {Tone.Time=} rampTime
|
|
*/
|
|
Tone.Clock.prototype.setRate = function(rate, rampTime){
|
|
//convert the time to a to frequency
|
|
var freqVal = this.secondsToFrequency(this.toSeconds(rate));
|
|
if (!rampTime){
|
|
this._controlSignal.cancelScheduledValues(0);
|
|
this._controlSignal.setValue(freqVal);
|
|
} else {
|
|
this._controlSignal.exponentialRampToValueNow(freqVal, rampTime);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* return the current rate
|
|
*
|
|
* @return {number}
|
|
*/
|
|
Tone.Clock.prototype.getRate = function(){
|
|
return this._controlSignal.getValue();
|
|
};
|
|
|
|
/**
|
|
* start the clock
|
|
* @param {Tone.Time} time the time when the clock should start
|
|
*/
|
|
Tone.Clock.prototype.start = function(time){
|
|
if (!this._oscillator){
|
|
this._oscillator = this.context.createOscillator();
|
|
this._oscillator.type = "square";
|
|
this._oscillator.connect(this._jsNode);
|
|
//connect it up
|
|
this._controlSignal.connect(this._oscillator.frequency);
|
|
this._upTick = false;
|
|
var startTime = this.toSeconds(time);
|
|
this._oscillator.start(startTime);
|
|
}
|
|
};
|
|
|
|
/**
|
|
* stop the clock
|
|
* @param {Tone.Time} time the time when the clock should stop
|
|
* @param {function} onend called when the oscilator stops
|
|
*/
|
|
Tone.Clock.prototype.stop = function(time, onend){
|
|
if (this._oscillator){
|
|
var now = this.now();
|
|
var stopTime = this.toSeconds(time, now);
|
|
this._oscillator.stop(stopTime);
|
|
this._oscillator = null;
|
|
//set a timeout for when it stops
|
|
if (time){
|
|
setTimeout(onend, (stopTime - now) * 1000);
|
|
} else {
|
|
onend();
|
|
}
|
|
}
|
|
};
|
|
|
|
/**
|
|
* @private
|
|
* @param {AudioProcessingEvent} event
|
|
*/
|
|
Tone.Clock.prototype._processBuffer = function(event){
|
|
var now = this.defaultArg(event.playbackTime, this.now());
|
|
var bufferSize = this._jsNode.bufferSize;
|
|
var incomingBuffer = event.inputBuffer.getChannelData(0);
|
|
var upTick = this._upTick;
|
|
var self = this;
|
|
for (var i = 0; i < bufferSize; i++){
|
|
var sample = incomingBuffer[i];
|
|
if (sample > 0 && !upTick){
|
|
upTick = true;
|
|
//get the callback out of audio thread
|
|
setTimeout(function(){
|
|
//to account for the double buffering
|
|
var tickTime = now + self.samplesToSeconds(i + bufferSize * 2);
|
|
return function(){
|
|
self.tick(tickTime);
|
|
};
|
|
}(), 0); // jshint ignore:line
|
|
} else if (sample < 0 && upTick){
|
|
upTick = false;
|
|
}
|
|
}
|
|
this._upTick = upTick;
|
|
};
|
|
|
|
/**
|
|
* clean up
|
|
*/
|
|
Tone.Clock.prototype.dispose = function(){
|
|
this._jsNode.disconnect();
|
|
this._controlSignal.dispose();
|
|
if (this._oscillator){
|
|
this._oscillator.disconnect();
|
|
}
|
|
this._jsNode.onaudioprocess = function(){};
|
|
this._jsNode = null;
|
|
this._controlSignal = null;
|
|
this._oscillator = null;
|
|
};
|
|
|
|
return Tone.Clock;
|
|
}); |