Tone.js/Tone/source/Source.js

288 lines
7.6 KiB
JavaScript
Raw Normal View History

2017-10-21 23:02:46 +00:00
define(["Tone/core/Tone", "Tone/core/Transport", "Tone/component/Volume", "Tone/core/Master", "Tone/type/Type",
"Tone/core/TimelineState", "Tone/signal/Signal", "Tone/core/AudioNode"], function(Tone){
"use strict";
2014-06-16 01:00:22 +00:00
/**
2015-06-14 02:03:06 +00:00
* @class Base class for sources. Sources have start/stop methods
* and the ability to be synced to the
* start/stop of Tone.Transport.
2014-06-19 05:40:16 +00:00
*
2014-06-16 01:00:22 +00:00
* @constructor
* @extends {Tone.AudioNode}
* @example
* //Multiple state change events can be chained together,
* //but must be set in the correct order and with ascending times
*
* // OK
* state.start().stop("+0.2");
* // AND
* state.start().stop("+0.2").start("+0.4").stop("+0.7")
*
* // BAD
* state.stop("+0.2").start();
* // OR
* state.start("+0.3").stop("+0.2");
*
*/
2015-02-02 01:38:06 +00:00
Tone.Source = function(options){
2016-09-20 03:02:42 +00:00
options = Tone.defaultArg(options, Tone.Source.defaults);
Tone.AudioNode.call(this);
2014-06-19 05:40:16 +00:00
2015-11-03 01:08:53 +00:00
/**
* The output volume node
* @type {Tone.Volume}
* @private
*/
this._volume = this.output = new Tone.Volume(options.volume);
/**
2015-02-27 16:19:45 +00:00
* The volume of the output in decibels.
2015-06-13 23:50:39 +00:00
* @type {Decibels}
* @signal
2015-02-27 16:19:45 +00:00
* @example
* source.volume.value = -6;
*/
2015-11-03 01:08:53 +00:00
this.volume = this._volume.volume;
this._readOnly("volume");
2015-02-02 02:32:07 +00:00
/**
2015-08-18 20:29:39 +00:00
* Keep track of the scheduled state.
* @type {Tone.TimelineState}
2015-02-02 02:32:07 +00:00
* @private
*/
2015-08-18 20:29:39 +00:00
this._state = new Tone.TimelineState(Tone.State.Stopped);
this._state.memory = 100;
/**
* The synced `start` callback function from the transport
* @type {Function}
* @private
*/
this._synced = false;
/**
* Keep track of all of the scheduled event ids
* @type {Array}
* @private
*/
this._scheduled = [];
//make the output explicitly stereo
2015-11-03 01:08:53 +00:00
this._volume.output.output.channelCount = 2;
this._volume.output.output.channelCountMode = "explicit";
//mute initially
this.mute = options.mute;
2014-06-16 01:00:22 +00:00
};
Tone.extend(Tone.Source, Tone.AudioNode);
2014-06-16 01:00:22 +00:00
2015-02-02 01:38:06 +00:00
/**
2015-02-27 16:19:45 +00:00
* The default parameters
2015-02-02 01:38:06 +00:00
* @static
* @const
* @type {Object}
*/
Tone.Source.defaults = {
"volume" : 0,
"mute" : false
2015-02-02 01:38:06 +00:00
};
2015-02-27 16:19:45 +00:00
/**
* Returns the playback state of the source, either "started" or "stopped".
* @type {Tone.State}
* @readOnly
* @memberOf Tone.Source#
* @name state
*/
Object.defineProperty(Tone.Source.prototype, "state", {
get : function(){
if (this._synced){
if (Tone.Transport.state === Tone.State.Started){
2016-12-19 03:14:14 +00:00
return this._state.getValueAtTime(Tone.Transport.seconds);
} else {
return Tone.State.Stopped;
}
} else {
2016-12-19 03:14:14 +00:00
return this._state.getValueAtTime(this.now());
}
}
});
/**
* Mute the output.
* @memberOf Tone.Source#
* @type {boolean}
* @name mute
* @example
* //mute the output
* source.mute = true;
*/
Object.defineProperty(Tone.Source.prototype, "mute", {
get : function(){
2016-05-14 21:34:23 +00:00
return this._volume.mute;
},
set : function(mute){
2016-05-14 21:34:23 +00:00
this._volume.mute = mute;
}
});
//overwrite these functions
Tone.Source.prototype._start = Tone.noOp;
Tone.Source.prototype.restart = Tone.noOp;
Tone.Source.prototype._stop = Tone.noOp;
/**
* Start the source at the specified time. If no time is given,
2015-06-20 19:50:57 +00:00
* start the source now.
* @param {Time} [time=now] When the source should be started.
* @returns {Tone.Source} this
2015-02-27 16:19:45 +00:00
* @example
2015-06-14 02:03:06 +00:00
* source.start("+0.5"); //starts the source 0.5 seconds from now
2014-06-16 01:00:22 +00:00
*/
Tone.Source.prototype.start = function(time, offset, duration){
if (Tone.isUndef(time) && this._synced){
time = Tone.Transport.seconds;
} else {
time = this.toSeconds(time);
}
//if it's started, stop it and restart it
if (this._state.getValueAtTime(time) === Tone.State.Started){
this._state.cancel(time);
this._state.setStateAtTime(Tone.State.Started, time);
this.restart(time, offset, duration);
} else {
this._state.setStateAtTime(Tone.State.Started, time);
if (this._synced){
// add the offset time to the event
var event = this._state.get(time);
event.offset = Tone.defaultArg(offset, 0);
event.duration = duration;
var sched = Tone.Transport.schedule(function(t){
this._start(t, offset, duration);
}.bind(this), time);
this._scheduled.push(sched);
//if it's already started
if (Tone.Transport.state === Tone.State.Started){
this._syncedStart(this.now(), Tone.Transport.seconds);
}
} else {
this._start.apply(this, arguments);
}
2015-02-02 02:32:07 +00:00
}
return this;
};
2014-06-16 01:00:22 +00:00
/**
* Stop the source at the specified time. If no time is given,
2015-06-20 19:50:57 +00:00
* stop the source now.
* @param {Time} [time=now] When the source should be stopped.
* @returns {Tone.Source} this
2015-02-27 16:19:45 +00:00
* @example
2015-06-14 02:03:06 +00:00
* source.stop(); // stops the source immediately
2014-06-16 01:00:22 +00:00
*/
2015-02-02 02:32:07 +00:00
Tone.Source.prototype.stop = function(time){
if (Tone.isUndef(time) && this._synced){
time = Tone.Transport.seconds;
} else {
time = this.toSeconds(time);
}
if (!this._synced){
this._stop.apply(this, arguments);
} else {
var sched = Tone.Transport.schedule(this._stop.bind(this), time);
this._scheduled.push(sched);
}
this._state.cancel(time);
this._state.setStateAtTime(Tone.State.Stopped, time);
2015-02-02 02:32:07 +00:00
return this;
};
2014-06-19 05:40:16 +00:00
/**
* Sync the source to the Transport so that all subsequent
* calls to `start` and `stop` are synced to the TransportTime
* instead of the AudioContext time.
2014-06-25 16:47:47 +00:00
*
* @returns {Tone.Source} this
2015-06-14 02:30:33 +00:00
* @example
* //sync the source so that it plays between 0 and 0.3 on the Transport's timeline
* source.sync().start(0).stop(0.3);
* //start the transport.
2015-06-14 02:30:33 +00:00
* Tone.Transport.start();
*
* @example
* //start the transport with an offset and the sync'ed sources
* //will start in the correct position
* source.sync().start(0.1);
* //the source will be invoked with an offset of 0.4
* Tone.Transport.start("+0.5", 0.5);
2014-06-19 05:40:16 +00:00
*/
Tone.Source.prototype.sync = function(){
this._synced = true;
this._syncedStart = function(time, offset){
if (offset > 0){
// get the playback state at that time
var stateEvent = this._state.get(offset);
// listen for start events which may occur in the middle of the sync'ed time
if (stateEvent && stateEvent.state === Tone.State.Started && stateEvent.time !== offset){
// get the offset
var startOffset = offset - this.toSeconds(stateEvent.time);
var duration;
if (stateEvent.duration){
duration = this.toSeconds(stateEvent.duration) - startOffset;
}
2016-10-01 22:31:24 +00:00
this._start(time, this.toSeconds(stateEvent.offset) + startOffset, duration);
}
}
}.bind(this);
this._syncedStop = function(time){
2016-12-19 03:14:14 +00:00
if (this._state.getValueAtTime(Tone.Transport.seconds) === Tone.State.Started){
this._stop(time);
}
}.bind(this);
Tone.Transport.on("start loopStart", this._syncedStart);
Tone.Transport.on("stop pause loopEnd", this._syncedStop);
2015-02-21 19:06:58 +00:00
return this;
2014-06-19 05:40:16 +00:00
};
/**
2015-06-14 01:54:20 +00:00
* Unsync the source to the Transport. See Tone.Source.sync
* @returns {Tone.Source} this
2014-06-19 05:40:16 +00:00
*/
Tone.Source.prototype.unsync = function(){
if (this._synced){
Tone.Transport.off("stop pause loopEnd", this._syncedStop);
Tone.Transport.off("start loopStart", this._syncedStart);
}
this._synced = false;
// clear all of the scheduled ids
for (var i = 0; i < this._scheduled.length; i++){
var id = this._scheduled[i];
Tone.Transport.clear(id);
}
this._scheduled = [];
this._state.cancel(0);
2015-02-21 19:06:58 +00:00
return this;
2014-06-19 05:40:16 +00:00
};
/**
2015-06-14 02:30:33 +00:00
* Clean up.
* @return {Tone.Source} this
*/
2015-02-02 03:05:24 +00:00
Tone.Source.prototype.dispose = function(){
Tone.AudioNode.prototype.dispose.call(this);
this.unsync();
this._scheduled = null;
this._writable("volume");
2015-11-03 01:08:53 +00:00
this._volume.dispose();
this._volume = null;
this.volume = null;
2015-08-18 20:29:39 +00:00
this._state.dispose();
this._state = null;
};
2014-06-16 01:00:22 +00:00
return Tone.Source;
});