Tone.js/Tone/source/OscillatorNode.js
2019-03-08 12:17:27 -05:00

240 lines
6 KiB
JavaScript

import Tone from "../core/Tone";
import "../core/Buffer";
import "../source/Source";
import "../core/Gain";
import "../core/AudioNode";
import "../shim/OscillatorNode";
/**
* @class Wrapper around the native fire-and-forget OscillatorNode. Adds the
* ability to reschedule the stop method. ***[Tone.Oscillator](Oscillator) is better
* for most use-cases***
* @extends {Tone.AudioNode}
* @param {AudioBuffer|Tone.Buffer} buffer The buffer to play
* @param {Function} onload The callback to invoke when the
* buffer is done playing.
*/
Tone.OscillatorNode = function(){
var options = Tone.defaults(arguments, ["frequency", "type"], Tone.OscillatorNode);
Tone.AudioNode.call(this, options);
/**
* The callback to invoke after the
* buffer source is done playing.
* @type {Function}
*/
this.onended = options.onended;
/**
* The oscillator start time
* @type {Number}
* @private
*/
this._startTime = -1;
/**
* The oscillator stop time
* @type {Number}
* @private
*/
this._stopTime = -1;
/**
* The gain node which envelopes the OscillatorNode
* @type {Tone.Gain}
* @private
*/
this._gainNode = this.output = new Tone.Gain(0);
/**
* The oscillator
* @type {OscillatorNode}
* @private
*/
this._oscillator = this.context.createOscillator();
Tone.connect(this._oscillator, this._gainNode);
this.type = options.type;
/**
* The frequency of the oscillator
* @type {Frequency}
* @signal
*/
this.frequency = new Tone.Param({
param : this._oscillator.frequency,
units : Tone.Type.Frequency,
value : options.frequency
});
/**
* The detune of the oscillator
* @type {Frequency}
* @signal
*/
this.detune = new Tone.Param({
param : this._oscillator.detune,
units : Tone.Type.Cents,
value : options.detune
});
/**
* The value that the buffer ramps to
* @type {Gain}
* @private
*/
this._gain = 1;
};
Tone.extend(Tone.OscillatorNode, Tone.AudioNode);
/**
* The defaults
* @const
* @type {Object}
*/
Tone.OscillatorNode.defaults = {
"frequency" : 440,
"detune" : 0,
"type" : "sine",
"onended" : Tone.noOp
};
/**
* Returns the playback state of the oscillator, either "started" or "stopped".
* @type {Tone.State}
* @readOnly
* @memberOf Tone.OscillatorNode#
* @name state
*/
Object.defineProperty(Tone.OscillatorNode.prototype, "state", {
get : function(){
return this.getStateAtTime(this.now());
}
});
/**
* Get the playback state at the given time
* @param {Time} time The time to test the state at
* @return {Tone.State} The playback state.
*/
Tone.OscillatorNode.prototype.getStateAtTime = function(time){
time = this.toSeconds(time);
if (this._startTime !== -1 && time >= this._startTime && (this._stopTime === -1 || time <= this._stopTime)){
return Tone.State.Started;
} else {
return Tone.State.Stopped;
}
};
/**
* Start the oscillator node at the given time
* @param {Time=} time When to start the oscillator
* @return {OscillatorNode} this
*/
Tone.OscillatorNode.prototype.start = function(time){
this.log("start", time);
if (this._startTime === -1){
this._startTime = this.toSeconds(time);
this._startTime = Math.max(this._startTime, this.context.currentTime);
this._oscillator.start(this._startTime);
this._gainNode.gain.setValueAtTime(1, this._startTime);
} else {
throw new Error("cannot call OscillatorNode.start more than once");
}
return this;
};
/**
* Sets an arbitrary custom periodic waveform given a PeriodicWave.
* @param {PeriodicWave} periodicWave PeriodicWave should be created with context.createPeriodicWave
* @return {OscillatorNode} this
*/
Tone.OscillatorNode.prototype.setPeriodicWave = function(periodicWave){
this._oscillator.setPeriodicWave(periodicWave);
return this;
};
/**
* Stop the oscillator node at the given time
* @param {Time=} time When to stop the oscillator
* @return {OscillatorNode} this
*/
Tone.OscillatorNode.prototype.stop = function(time){
this.log("stop", time);
this.assert(this._startTime !== -1, "'start' must be called before 'stop'");
//cancel the previous stop
this.cancelStop();
//reschedule it
this._stopTime = this.toSeconds(time);
this._stopTime = Math.max(this._stopTime, this.context.currentTime);
if (this._stopTime > this._startTime){
this._gainNode.gain.setValueAtTime(0, this._stopTime);
this.context.clearTimeout(this._timeout);
this._timeout = this.context.setTimeout(function(){
this._oscillator.stop(this.now());
this.onended();
//dispose the object when it's ended
setTimeout(function(){
if (this._oscillator){
this.dispose();
}
}.bind(this), 100);
}.bind(this), this._stopTime - this.context.currentTime);
} else {
//cancel the stop envelope
this._gainNode.gain.cancelScheduledValues(this._startTime);
}
return this;
};
/**
* Cancel a scheduled stop event
* @return {Tone.OscillatorNode} this
*/
Tone.OscillatorNode.prototype.cancelStop = function(){
if (this._startTime !== -1){
//cancel the stop envelope
this._gainNode.gain.cancelScheduledValues(this._startTime+this.sampleTime);
this.context.clearTimeout(this._timeout);
this._stopTime = -1;
}
return this;
};
/**
* The oscillator type. Either 'sine', 'sawtooth', 'square', or 'triangle'
* @memberOf Tone.OscillatorNode#
* @type {Time}
* @name type
*/
Object.defineProperty(Tone.OscillatorNode.prototype, "type", {
get : function(){
return this._oscillator.type;
},
set : function(type){
this._oscillator.type = type;
}
});
/**
* Clean up.
* @return {Tone.OscillatorNode} this
*/
Tone.OscillatorNode.prototype.dispose = function(){
this.context.clearTimeout(this._timeout);
Tone.AudioNode.prototype.dispose.call(this);
this.onended = null;
this._oscillator.disconnect();
this._oscillator = null;
this._gainNode.dispose();
this._gainNode = null;
this.frequency.dispose();
this.frequency = null;
this.detune.dispose();
this.detune = null;
return this;
};
export default Tone.OscillatorNode;