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; });