2014-12-01 04:26:06 +00:00
|
|
|
define(["Tone/core/Tone", "Tone/signal/SignalBase"], function(Tone){
|
2014-11-30 18:16:20 +00:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @class Wraps the WaveShaperNode
|
|
|
|
*
|
|
|
|
* @example
|
|
|
|
* var timesTwo = new Tone.WaveShaper(function(val, position){
|
|
|
|
* return val * 2;
|
|
|
|
* }, 2048);
|
|
|
|
*
|
2014-12-01 04:26:06 +00:00
|
|
|
* @extends {Tone.SignalBase}
|
2014-11-30 18:16:20 +00:00
|
|
|
* @constructor
|
|
|
|
* @param {function(number, number)|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
|
|
|
|
* set as the wave shapping function
|
|
|
|
* @param {number} [bufferLen=1024] the length of the WaveShaperNode buffer.
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper = function(mapping, bufferLen){
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.setCurve(mapping);
|
|
|
|
} else if (isFinite(mapping) || this.isUndef(mapping)){
|
|
|
|
this._curve = new Float32Array(this.defaultArg(mapping, 1024));
|
|
|
|
} else if (typeof mapping === "function"){
|
|
|
|
this._curve = new Float32Array(this.defaultArg(bufferLen, 1024));
|
|
|
|
this.setMap(mapping);
|
|
|
|
}
|
|
|
|
};
|
|
|
|
|
2014-12-01 04:26:06 +00:00
|
|
|
Tone.extend(Tone.WaveShaper, Tone.SignalBase);
|
2014-11-30 18:16:20 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* uses a mapping function to set the value of the curve
|
|
|
|
* @param {function(number, 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
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper.prototype.setMap = function(mapping){
|
|
|
|
for (var i = 0, len = this._curve.length; i < len; i++){
|
|
|
|
var normalized = (i / (len)) * 2 - 1;
|
|
|
|
var normOffOne = (i / (len - 1)) * 2 - 1;
|
|
|
|
this._curve[i] = mapping(normalized, i, normOffOne);
|
|
|
|
}
|
|
|
|
this._shaper.curve = this._curve;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* use an array to set the waveshaper curve
|
|
|
|
* @param {Array} mapping the array to use as the waveshaper
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper.prototype.setCurve = function(mapping){
|
|
|
|
//fixes safari WaveShaperNode bug
|
|
|
|
if (this._isSafari()){
|
|
|
|
var first = mapping[0];
|
|
|
|
mapping.unshift(first);
|
|
|
|
}
|
|
|
|
this._curve = new Float32Array(mapping);
|
|
|
|
this._shaper.curve = this._curve;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* set the oversampling
|
|
|
|
* @param {string} oversampling can either be "none", "2x" or "4x"
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper.prototype.setOversample = function(oversampling) {
|
|
|
|
this._shaper.oversample = oversampling;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* returns true if the browser is safari
|
|
|
|
* @return {boolean}
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper.prototype._isSafari = function(){
|
|
|
|
var ua = navigator.userAgent.toLowerCase();
|
|
|
|
return ua.indexOf("safari") !== -1 && ua.indexOf("chrome") === -1;
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* clean up
|
|
|
|
*/
|
|
|
|
Tone.WaveShaper.prototype.dispose = function(){
|
|
|
|
Tone.prototype.dispose.call(this);
|
|
|
|
this._shaper.disconnect();
|
|
|
|
this._shaper = null;
|
|
|
|
this._curve = null;
|
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.WaveShaper;
|
|
|
|
});
|