2014-11-30 18:20:35 +00:00
|
|
|
define(["Tone/core/Tone", "Tone/effect/Effect", "Tone/signal/WaveShaper"], function(Tone){
|
2014-10-19 23:39:50 +00:00
|
|
|
|
|
|
|
"use strict";
|
|
|
|
|
|
|
|
/**
|
2015-07-04 16:40:33 +00:00
|
|
|
* @class Tone.ChebyShev is a Chebyshev waveshaper, an effect which is good
|
|
|
|
* for making different types of distortion sounds.
|
|
|
|
* Note that odd orders sound very different from even ones,
|
|
|
|
* and order = 1 is no change.
|
2015-07-04 19:25:37 +00:00
|
|
|
* Read more at [music.columbia.edu](http://music.columbia.edu/cmc/musicandcomputers/chapter4/04_06.php).
|
2014-10-19 23:39:50 +00:00
|
|
|
*
|
|
|
|
* @extends {Tone.Effect}
|
|
|
|
* @constructor
|
2015-06-22 05:20:57 +00:00
|
|
|
* @param {Positive|Object} [order] The order of the chebyshev polynomial. Normal range between 1-100.
|
2015-02-27 21:53:10 +00:00
|
|
|
* @example
|
2015-07-04 16:40:33 +00:00
|
|
|
* //create a new cheby
|
|
|
|
* var cheby = new Tone.Chebyshev(50);
|
|
|
|
* //create a monosynth connected to our cheby
|
|
|
|
* synth = new Tone.MonoSynth().connect(cheby);
|
2014-10-19 23:39:50 +00:00
|
|
|
*/
|
|
|
|
Tone.Chebyshev = function(){
|
|
|
|
|
|
|
|
var options = this.optionsObject(arguments, ["order"], Tone.Chebyshev.defaults);
|
|
|
|
Tone.Effect.call(this);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @type {WaveShaperNode}
|
|
|
|
* @private
|
|
|
|
*/
|
2014-11-30 18:20:35 +00:00
|
|
|
this._shaper = new Tone.WaveShaper(4096);
|
2014-11-04 00:22:17 +00:00
|
|
|
|
2015-01-20 13:55:28 +00:00
|
|
|
/**
|
2015-02-10 16:40:27 +00:00
|
|
|
* holds onto the order of the filter
|
|
|
|
* @type {number}
|
|
|
|
* @private
|
2015-01-20 13:55:28 +00:00
|
|
|
*/
|
|
|
|
this._order = options.order;
|
|
|
|
|
2014-10-19 23:39:50 +00:00
|
|
|
this.connectEffect(this._shaper);
|
2015-02-10 16:40:27 +00:00
|
|
|
this.order = options.order;
|
|
|
|
this.oversample = options.oversample;
|
2014-10-19 23:39:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
Tone.extend(Tone.Chebyshev, Tone.Effect);
|
|
|
|
|
|
|
|
/**
|
|
|
|
* @static
|
|
|
|
* @const
|
|
|
|
* @type {Object}
|
|
|
|
*/
|
|
|
|
Tone.Chebyshev.defaults = {
|
|
|
|
"order" : 1,
|
|
|
|
"oversample" : "none"
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get the coefficient for that degree
|
|
|
|
* @param {number} x the x value
|
|
|
|
* @param {number} degree
|
|
|
|
* @param {Object} memo memoize the computed value.
|
|
|
|
* this speeds up computation greatly.
|
|
|
|
* @return {number} the coefficient
|
|
|
|
* @private
|
|
|
|
*/
|
|
|
|
Tone.Chebyshev.prototype._getCoefficient = function(x, degree, memo){
|
|
|
|
if (memo.hasOwnProperty(degree)){
|
|
|
|
return memo[degree];
|
|
|
|
} else if (degree === 0){
|
|
|
|
memo[degree] = 0;
|
|
|
|
} else if (degree === 1){
|
|
|
|
memo[degree] = x;
|
|
|
|
} else {
|
|
|
|
memo[degree] = 2 * x * this._getCoefficient(x, degree - 1, memo) - this._getCoefficient(x, degree - 2, memo);
|
|
|
|
}
|
|
|
|
return memo[degree];
|
|
|
|
};
|
|
|
|
|
|
|
|
/**
|
2015-06-22 05:20:57 +00:00
|
|
|
* The order of the Chebyshev polynomial which creates
|
|
|
|
* the equation which is applied to the incoming
|
|
|
|
* signal through a Tone.WaveShaper. The equations
|
|
|
|
* are in the form:<br>
|
|
|
|
* order 2: 2x^2 + 1<br>
|
|
|
|
* order 3: 4x^3 + 3x <br>
|
2015-02-10 16:40:27 +00:00
|
|
|
* @memberOf Tone.Chebyshev#
|
2015-06-22 05:20:57 +00:00
|
|
|
* @type {Positive}
|
2015-02-10 16:40:27 +00:00
|
|
|
* @name order
|
2014-10-19 23:39:50 +00:00
|
|
|
*/
|
2015-02-10 16:40:27 +00:00
|
|
|
Object.defineProperty(Tone.Chebyshev.prototype, "order", {
|
|
|
|
get : function(){
|
|
|
|
return this._order;
|
|
|
|
},
|
|
|
|
set : function(order){
|
|
|
|
this._order = order;
|
|
|
|
var curve = new Array(4096);
|
|
|
|
var len = curve.length;
|
|
|
|
for (var i = 0; i < len; ++i) {
|
|
|
|
var x = i * 2 / len - 1;
|
|
|
|
if (x === 0){
|
|
|
|
//should output 0 when input is 0
|
|
|
|
curve[i] = 0;
|
|
|
|
} else {
|
|
|
|
curve[i] = this._getCoefficient(x, order, {});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
this._shaper.curve = curve;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The oversampling of the effect. Can either be "none", "2x" or "4x".
|
|
|
|
* @memberOf Tone.Chebyshev#
|
|
|
|
* @type {string}
|
|
|
|
* @name oversample
|
|
|
|
*/
|
|
|
|
Object.defineProperty(Tone.Chebyshev.prototype, "oversample", {
|
|
|
|
get : function(){
|
|
|
|
return this._shaper.oversample;
|
|
|
|
},
|
|
|
|
set : function(oversampling){
|
|
|
|
this._shaper.oversample = oversampling;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
2014-10-19 23:39:50 +00:00
|
|
|
/**
|
2015-06-22 05:20:57 +00:00
|
|
|
* Clean up.
|
2015-06-14 00:54:29 +00:00
|
|
|
* @returns {Tone.Chebyshev} this
|
2014-10-19 23:39:50 +00:00
|
|
|
*/
|
|
|
|
Tone.Chebyshev.prototype.dispose = function(){
|
|
|
|
Tone.Effect.prototype.dispose.call(this);
|
2014-11-30 18:20:35 +00:00
|
|
|
this._shaper.dispose();
|
2014-10-19 23:39:50 +00:00
|
|
|
this._shaper = null;
|
2015-02-02 18:22:16 +00:00
|
|
|
return this;
|
2014-10-19 23:39:50 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
return Tone.Chebyshev;
|
|
|
|
});
|