2017-10-21 23:02:46 +00:00
|
|
|
define(["Tone/core/Tone", "Tone/signal/TickSignal", "Tone/core/TimelineState",
|
2017-02-19 16:45:43 +00:00
|
|
|
"Tone/core/Emitter", "Tone/core/Context"], function (Tone) {
|
2014-09-04 04:41:40 +00:00
|
|
|
|
2015-10-21 16:11:19 +00:00
|
|
|
"use strict";
|
|
|
|
|
2014-07-30 17:54:55 +00:00
|
|
|
/**
|
2017-10-21 23:02:46 +00:00
|
|
|
* @class A sample accurate clock which provides a callback at the given rate.
|
2015-06-20 19:50:57 +00:00
|
|
|
* While the callback is not sample-accurate (it is still susceptible to
|
|
|
|
* loose JS timing), the time passed in as the argument to the callback
|
|
|
|
* is precise. For most applications, it is better to use Tone.Transport
|
2015-08-18 22:14:26 +00:00
|
|
|
* instead of the Clock by itself since you can synchronize multiple callbacks.
|
2014-07-30 17:54:55 +00:00
|
|
|
*
|
|
|
|
* @constructor
|
2016-09-24 15:26:57 +00:00
|
|
|
* @extends {Tone.Emitter}
|
2015-06-20 19:50:57 +00:00
|
|
|
* @param {function} callback The callback to be invoked with the time of the audio event
|
2015-10-09 15:01:03 +00:00
|
|
|
* @param {Frequency} frequency The rate of the callback
|
2015-06-14 00:20:36 +00:00
|
|
|
* @example
|
|
|
|
* //the callback will be invoked approximately once a second
|
|
|
|
* //and will print the time exactly once a second apart.
|
2015-08-18 22:14:26 +00:00
|
|
|
* var clock = new Tone.Clock(function(time){
|
2015-06-14 00:20:36 +00:00
|
|
|
* console.log(time);
|
2015-08-18 22:14:26 +00:00
|
|
|
* }, 1);
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
2015-08-18 22:14:26 +00:00
|
|
|
Tone.Clock = function(){
|
|
|
|
|
2017-04-26 02:23:22 +00:00
|
|
|
var options = Tone.defaults(arguments, ["callback", "frequency"], Tone.Clock);
|
2016-09-24 15:26:57 +00:00
|
|
|
Tone.Emitter.call(this);
|
2015-08-18 22:14:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The callback function to invoke at the scheduled tick.
|
|
|
|
* @type {Function}
|
|
|
|
*/
|
|
|
|
this.callback = options.callback;
|
2014-07-30 17:54:55 +00:00
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* The next time the callback is scheduled.
|
|
|
|
* @type {Number}
|
|
|
|
* @private
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
2016-11-21 08:05:19 +00:00
|
|
|
this._nextTick = 0;
|
2014-07-30 17:54:55 +00:00
|
|
|
|
|
|
|
/**
|
2016-11-21 08:05:19 +00:00
|
|
|
* The last state of the clock.
|
|
|
|
* @type {State}
|
2014-07-30 17:54:55 +00:00
|
|
|
* @private
|
|
|
|
*/
|
2016-11-21 08:05:19 +00:00
|
|
|
this._lastState = Tone.State.Stopped;
|
2015-08-18 22:14:26 +00:00
|
|
|
|
|
|
|
/**
|
2017-10-21 23:02:46 +00:00
|
|
|
* The rate the callback function should be invoked.
|
2015-08-18 22:14:26 +00:00
|
|
|
* @type {BPM}
|
|
|
|
* @signal
|
|
|
|
*/
|
2017-05-28 23:39:18 +00:00
|
|
|
this.frequency = new Tone.TickSignal(options.frequency, Tone.Type.Frequency);
|
2016-11-21 08:05:19 +00:00
|
|
|
this._readOnly("frequency");
|
2015-08-18 22:14:26 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of times the callback was invoked. Starts counting at 0
|
2017-10-21 23:02:46 +00:00
|
|
|
* and increments after the callback was invoked.
|
2015-08-18 22:14:26 +00:00
|
|
|
* @type {Ticks}
|
|
|
|
* @readOnly
|
|
|
|
*/
|
|
|
|
this.ticks = 0;
|
2014-07-30 17:54:55 +00:00
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* The state timeline
|
|
|
|
* @type {Tone.TimelineState}
|
|
|
|
* @private
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
2015-08-18 22:14:26 +00:00
|
|
|
this._state = new Tone.TimelineState(Tone.State.Stopped);
|
2014-07-30 17:54:55 +00:00
|
|
|
|
2015-05-05 19:36:06 +00:00
|
|
|
/**
|
2017-10-21 23:02:46 +00:00
|
|
|
* The loop function bound to its context.
|
2016-02-27 22:14:39 +00:00
|
|
|
* This is necessary to remove the event in the end.
|
|
|
|
* @type {Function}
|
2015-08-18 22:14:26 +00:00
|
|
|
* @private
|
2015-05-05 19:36:06 +00:00
|
|
|
*/
|
2015-08-18 22:14:26 +00:00
|
|
|
this._boundLoop = this._loop.bind(this);
|
2015-05-05 19:36:06 +00:00
|
|
|
|
2016-02-27 22:14:39 +00:00
|
|
|
//bind a callback to the worker thread
|
2017-10-21 23:02:46 +00:00
|
|
|
this.context.on("tick", this._boundLoop);
|
2014-07-30 17:54:55 +00:00
|
|
|
};
|
|
|
|
|
2016-09-24 15:26:57 +00:00
|
|
|
Tone.extend(Tone.Clock, Tone.Emitter);
|
2014-07-30 17:54:55 +00:00
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* The defaults
|
|
|
|
* @const
|
|
|
|
* @type {Object}
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
2015-08-18 22:14:26 +00:00
|
|
|
Tone.Clock.defaults = {
|
|
|
|
"callback" : Tone.noOp,
|
|
|
|
"frequency" : 1,
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Returns the playback state of the source, either "started", "stopped" or "paused".
|
|
|
|
* @type {Tone.State}
|
|
|
|
* @readOnly
|
|
|
|
* @memberOf Tone.Clock#
|
|
|
|
* @name state
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Clock.prototype, "state", {
|
|
|
|
get : function(){
|
2016-12-19 03:14:14 +00:00
|
|
|
return this._state.getValueAtTime(this.now());
|
2015-01-05 03:19:33 +00:00
|
|
|
}
|
2015-08-18 22:14:26 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Start the clock at the given time. Optionally pass in an offset
|
|
|
|
* of where to start the tick counter from.
|
2017-06-21 14:21:38 +00:00
|
|
|
* @param {Time=} time The time the clock should start
|
2015-08-18 22:14:26 +00:00
|
|
|
* @param {Ticks=} offset Where the tick counter starts counting from.
|
|
|
|
* @return {Tone.Clock} this
|
|
|
|
*/
|
|
|
|
Tone.Clock.prototype.start = function(time, offset){
|
2016-12-21 03:49:20 +00:00
|
|
|
time = this.toSeconds(time);
|
2016-12-19 03:14:14 +00:00
|
|
|
if (this._state.getValueAtTime(time) !== Tone.State.Started){
|
2017-05-28 23:39:18 +00:00
|
|
|
this._state.setStateAtTime(Tone.State.Started, time);
|
|
|
|
this._state.get(time).offset = offset;
|
2015-08-18 22:14:26 +00:00
|
|
|
}
|
2017-10-21 23:02:46 +00:00
|
|
|
return this;
|
2014-07-30 17:54:55 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* Stop the clock. Stopping the clock resets the tick counter to 0.
|
2015-06-14 00:20:36 +00:00
|
|
|
* @param {Time} [time=now] The time when the clock should stop.
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Clock} this
|
2015-06-14 00:20:36 +00:00
|
|
|
* @example
|
|
|
|
* clock.stop();
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
2015-05-05 19:36:06 +00:00
|
|
|
Tone.Clock.prototype.stop = function(time){
|
2015-08-18 22:14:26 +00:00
|
|
|
time = this.toSeconds(time);
|
2016-04-18 05:12:35 +00:00
|
|
|
this._state.cancel(time);
|
|
|
|
this._state.setStateAtTime(Tone.State.Stopped, time);
|
2017-10-21 23:02:46 +00:00
|
|
|
return this;
|
2014-07-30 17:54:55 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* Pause the clock. Pausing does not reset the tick counter.
|
|
|
|
* @param {Time} [time=now] The time when the clock should stop.
|
|
|
|
* @returns {Tone.Clock} this
|
|
|
|
*/
|
|
|
|
Tone.Clock.prototype.pause = function(time){
|
|
|
|
time = this.toSeconds(time);
|
2016-12-19 03:14:14 +00:00
|
|
|
if (this._state.getValueAtTime(time) === Tone.State.Started){
|
2015-08-18 22:14:26 +00:00
|
|
|
this._state.setStateAtTime(Tone.State.Paused, time);
|
|
|
|
}
|
2017-10-21 23:02:46 +00:00
|
|
|
return this;
|
2015-08-18 22:14:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The scheduling loop.
|
2014-07-30 17:54:55 +00:00
|
|
|
* @private
|
|
|
|
*/
|
2016-03-03 06:34:48 +00:00
|
|
|
Tone.Clock.prototype._loop = function(){
|
2016-11-21 08:05:19 +00:00
|
|
|
|
2017-05-28 23:39:18 +00:00
|
|
|
//the end of the update interval
|
|
|
|
var endTime = this.now() + this.context.updateInterval;
|
|
|
|
|
|
|
|
//the current event at the time of the loop
|
|
|
|
var event = this._state.get(endTime);
|
|
|
|
|
|
|
|
if (event){
|
|
|
|
//state change events
|
|
|
|
if (event.state !== this._lastState){
|
|
|
|
this._lastState = event.state;
|
|
|
|
switch(event.state){
|
|
|
|
case Tone.State.Started:
|
|
|
|
if (!Tone.isUndef(event.offset)){
|
|
|
|
this.ticks = event.offset;
|
|
|
|
}
|
|
|
|
this._nextTick = event.time;
|
|
|
|
this.emit("start", event.time, this.ticks);
|
|
|
|
break;
|
|
|
|
case Tone.State.Stopped:
|
|
|
|
this.ticks = 0;
|
|
|
|
this.emit("stop", event.time);
|
|
|
|
break;
|
|
|
|
case Tone.State.Paused:
|
|
|
|
this.emit("pause", event.time);
|
|
|
|
break;
|
2015-08-18 22:14:26 +00:00
|
|
|
}
|
|
|
|
}
|
2017-05-28 23:39:18 +00:00
|
|
|
|
|
|
|
//all the tick events
|
|
|
|
while(endTime > this._nextTick && this._state){
|
|
|
|
var tickTime = this._nextTick;
|
|
|
|
if (this.frequency){
|
|
|
|
this._nextTick += this.frequency.getDurationOfTicks(1, this._nextTick);
|
|
|
|
if (event.state === Tone.State.Started){
|
|
|
|
try {
|
|
|
|
this.callback(tickTime);
|
|
|
|
this.ticks++;
|
|
|
|
} catch(e){
|
|
|
|
this.ticks++;
|
2017-06-28 19:22:26 +00:00
|
|
|
throw e;
|
2017-05-28 23:39:18 +00:00
|
|
|
}
|
2017-05-08 15:45:21 +00:00
|
|
|
}
|
2016-11-21 08:05:19 +00:00
|
|
|
}
|
2016-09-24 15:26:57 +00:00
|
|
|
}
|
2014-07-30 17:54:55 +00:00
|
|
|
}
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-08-18 22:14:26 +00:00
|
|
|
* Returns the scheduled state at the given time.
|
|
|
|
* @param {Time} time The time to query.
|
|
|
|
* @return {String} The name of the state input in setStateAtTime.
|
|
|
|
* @example
|
|
|
|
* clock.start("+0.1");
|
2016-12-19 04:39:53 +00:00
|
|
|
* clock.getStateAtTime("+0.1"); //returns "started"
|
2015-08-18 22:14:26 +00:00
|
|
|
*/
|
2016-12-19 04:39:53 +00:00
|
|
|
Tone.Clock.prototype.getStateAtTime = function(time){
|
2016-09-20 23:35:21 +00:00
|
|
|
time = this.toSeconds(time);
|
2016-12-19 03:14:14 +00:00
|
|
|
return this._state.getValueAtTime(time);
|
2015-08-18 22:14:26 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Clean up
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Clock} this
|
2014-07-30 17:54:55 +00:00
|
|
|
*/
|
|
|
|
Tone.Clock.prototype.dispose = function(){
|
2016-09-24 15:26:57 +00:00
|
|
|
Tone.Emitter.prototype.dispose.call(this);
|
2017-02-19 16:45:43 +00:00
|
|
|
this.context.off("tick", this._boundLoop);
|
2015-08-18 22:14:26 +00:00
|
|
|
this._writable("frequency");
|
2015-02-10 21:33:18 +00:00
|
|
|
this.frequency.dispose();
|
|
|
|
this.frequency = null;
|
2016-02-27 22:14:39 +00:00
|
|
|
this._boundLoop = null;
|
2015-08-18 22:14:26 +00:00
|
|
|
this._nextTick = Infinity;
|
|
|
|
this.callback = null;
|
|
|
|
this._state.dispose();
|
|
|
|
this._state = null;
|
2014-07-30 17:54:55 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.Clock;
|
2017-10-21 23:02:46 +00:00
|
|
|
});
|