Tone.js/Tone/signal/WaveShaper.js

135 lines
4.2 KiB
JavaScript
Raw Normal View History

define(["../core/Tone", "../signal/SignalBase", "../shim/WaveShaperNode"], function(Tone){
"use strict";
/**
2017-12-28 23:49:47 +00:00
* @class Wraps the native Web Audio API
2015-07-04 19:25:37 +00:00
* [WaveShaperNode](http://webaudio.github.io/web-audio-api/#the-waveshapernode-interface).
*
* @extends {Tone.SignalBase}
* @constructor
2017-12-28 23:49:47 +00:00
* @param {function|Array|Number} mapping The function used to define the values.
* The mapping function should take two arguments:
* the first is the value at the current position
* and the second is the array position.
* If the argument is an array, that array will be
2015-06-19 04:52:04 +00:00
* set as the wave shaping function. The input
* signal is an AudioRange [-1, 1] value and the output
2017-12-28 23:49:47 +00:00
* signal can take on any numerical values.
*
2015-06-20 19:50:57 +00:00
* @param {Number} [bufferLen=1024] The length of the WaveShaperNode buffer.
2015-02-27 18:40:35 +00:00
* @example
2015-06-14 05:17:09 +00:00
* var timesTwo = new Tone.WaveShaper(function(val){
* return val * 2;
* }, 2048);
2015-06-20 19:50:57 +00:00
* @example
* //a waveshaper can also be constructed with an array of values
* var invert = new Tone.WaveShaper([1, -1]);
*/
Tone.WaveShaper = function(mapping, bufferLen){
Tone.SignalBase.call(this);
/**
* the waveshaper
* @type {WaveShaperNode}
* @private
*/
this._shaper = this.input = this.output = this.context.createWaveShaper();
/**
* the waveshapers curve
* @type {Float32Array}
* @private
*/
this._curve = null;
if (Array.isArray(mapping)){
this.curve = mapping;
} else if (isFinite(mapping) || Tone.isUndef(mapping)){
2017-04-30 19:03:49 +00:00
this._curve = new Float32Array(Tone.defaultArg(mapping, 1024));
} else if (Tone.isFunction(mapping)){
2017-04-30 19:03:49 +00:00
this._curve = new Float32Array(Tone.defaultArg(bufferLen, 1024));
this.setMap(mapping);
2017-12-28 23:49:47 +00:00
}
};
Tone.extend(Tone.WaveShaper, Tone.SignalBase);
/**
2017-12-28 23:49:47 +00:00
* Uses a mapping function to set the value of the curve.
* @param {function} mapping The function used to define the values.
* The mapping function take two arguments:
* the first is the value at the current position
2015-06-19 04:52:04 +00:00
* which goes from -1 to 1 over the number of elements
2017-12-28 23:49:47 +00:00
* in the curve array. The second argument is the array position.
* @returns {Tone.WaveShaper} this
2015-06-19 04:52:04 +00:00
* @example
* //map the input signal from [-1, 1] to [0, 10]
* shaper.setMap(function(val, index){
* return (val + 1) * 5;
* })
*/
Tone.WaveShaper.prototype.setMap = function(mapping){
2017-12-28 23:49:47 +00:00
var array = new Array(this._curve.length);
for (var i = 0, len = this._curve.length; i < len; i++){
var normalized = (i / (len - 1)) * 2 - 1;
2017-12-28 23:49:47 +00:00
array[i] = mapping(normalized, i);
}
2017-12-28 23:49:47 +00:00
this.curve = array;
2015-02-02 03:56:33 +00:00
return this;
};
/**
2015-06-19 04:52:04 +00:00
* The array to set as the waveshaper curve. For linear curves
* array length does not make much difference, but for complex curves
2017-12-28 23:49:47 +00:00
* longer arrays will provide smoother interpolation.
* @memberOf Tone.WaveShaper#
* @type {Array}
* @name curve
*/
Object.defineProperty(Tone.WaveShaper.prototype, "curve", {
get : function(){
return this._shaper.curve;
},
set : function(mapping){
this._curve = new Float32Array(mapping);
this._shaper.curve = this._curve;
}
});
/**
2017-12-28 23:49:47 +00:00
* Specifies what type of oversampling (if any) should be used when
* applying the shaping curve. Can either be "none", "2x" or "4x".
* @memberOf Tone.WaveShaper#
* @type {string}
* @name oversample
*/
Object.defineProperty(Tone.WaveShaper.prototype, "oversample", {
get : function(){
return this._shaper.oversample;
},
set : function(oversampling){
2017-12-28 23:49:47 +00:00
if (["none", "2x", "4x"].includes(oversampling)){
this._shaper.oversample = oversampling;
} else {
2016-05-26 00:50:46 +00:00
throw new RangeError("Tone.WaveShaper: oversampling must be either 'none', '2x', or '4x'");
}
}
});
/**
2015-06-19 04:52:04 +00:00
* Clean up.
* @returns {Tone.WaveShaper} this
*/
Tone.WaveShaper.prototype.dispose = function(){
Tone.SignalBase.prototype.dispose.call(this);
this._shaper.disconnect();
this._shaper = null;
this._curve = null;
2015-02-02 03:56:33 +00:00
return this;
};
return Tone.WaveShaper;
2017-10-26 20:02:01 +00:00
});