define(["Tone/core/Tone", "Tone/instrument/MonoSynth", "Tone/signal/Signal", "Tone/signal/Multiply", "Tone/instrument/Monophonic"], function(Tone){ "use strict"; /** * @class the FMSynth is composed of two MonoSynths where one MonoSynth is the * carrier and the second is the modulator. * * @constructor * @extends {Tone.Monophonic} * @param {Object} options the options available for the synth * see defaults below */ Tone.FMSynth = function(options){ options = this.defaultArg(options, Tone.FMSynth.defaults); Tone.Monophonic.call(this, options); /** * the first voice * @type {Tone.MonoSynth} */ this.carrier = new Tone.MonoSynth(options.carrier); this.carrier.setVolume(-10); /** * the second voice * @type {Tone.MonoSynth} */ this.modulator = new Tone.MonoSynth(options.modulator); this.modulator.setVolume(-10); /** * the frequency control * @type {Tone.Signal} */ this.frequency = new Tone.Signal(440); /** * the ratio between the two voices * @type {Tone.Multiply} * @private */ this._harmonicity = new Tone.Multiply(options.harmonicity); /** * which is in essence the depth or amount of the modulation. In other terms it is the * ratio of the frequency of the modulating signal (mf) to the amplitude of the * modulating signal (ma) -- as in ma/mf. * * @type {Tone.Multiply} * @private */ this._modulationIndex = new Tone.Multiply(options.modulationIndex); /** * the node where the modulation happens * @type {GainNode} * @private */ this._modulationNode = this.context.createGain(); //control the two voices frequency this.frequency.connect(this.carrier.frequency); this.frequency.chain(this._harmonicity, this.modulator.frequency); this.frequency.chain(this._modulationIndex, this._modulationNode); this.modulator.connect(this._modulationNode.gain); this._modulationNode.gain.value = 0; this._modulationNode.connect(this.carrier.frequency); this.carrier.connect(this.output); }; Tone.extend(Tone.FMSynth, Tone.Monophonic); /** * @static * @type {Object} */ Tone.FMSynth.defaults = { "harmonicity" : 3, "modulationIndex" : 10, "carrier" : { "volume" : -10, "portamento" : 0, "oscillator" : { "type" : "sine" }, "envelope" : { "attack" : 0.01, "decay" : 0.0, "sustain" : 1, "release" : 0.5 }, "filterEnvelope" : { "attack" : 0.01, "decay" : 0.0, "sustain" : 1, "release" : 0.5, "min" : 20000, "max" : 20000 } }, "modulator" : { "volume" : -10, "portamento" : 0, "oscillator" : { "type" : "triangle" }, "envelope" : { "attack" : 0.01, "decay" : 0.0, "sustain" : 1, "release" : 0.5 }, "filterEnvelope" : { "attack" : 0.01, "decay" : 0.0, "sustain" : 1, "release" : 0.5, "min" : 20000, "max" : 20000 } } }; /** * trigger the attack portion of the note * * @param {Tone.Time} [time=now] the time the note will occur * @param {number} [velocity=1] the velocity of the note */ Tone.FMSynth.prototype.triggerEnvelopeAttack = function(time, velocity){ //the port glide time = this.toSeconds(time); //the envelopes this.carrier.envelope.triggerAttack(time, velocity); this.modulator.envelope.triggerAttack(time); this.carrier.filterEnvelope.triggerAttack(time); this.modulator.filterEnvelope.triggerAttack(time); }; /** * trigger the release portion of the note * * @param {Tone.Time} [time=now] the time the note will release */ Tone.FMSynth.prototype.triggerEnvelopeRelease = function(time){ this.carrier.triggerRelease(time); this.modulator.triggerRelease(time); }; /** * set the ratio between the two carrier and the modulator * @param {number} ratio */ Tone.FMSynth.prototype.setHarmonicity = function(ratio){ this._harmonicity.setValue(ratio); }; /** * set the modulation index * @param {number} index */ Tone.FMSynth.prototype.setModulationIndex = function(index){ this._modulationIndex.setValue(index); }; /** * bulk setter * @param {Object} param */ Tone.FMSynth.prototype.set = function(params){ if (!this.isUndef(params.harmonicity)) this.setHarmonicity(params.harmonicity); if (!this.isUndef(params.modulationIndex)) this.setModulationIndex(params.modulationIndex); if (!this.isUndef(params.carrier)) this.carrier.set(params.carrier); if (!this.isUndef(params.modulator)) this.modulator.set(params.modulator); Tone.Monophonic.prototype.set.call(this, params); }; /** * clean up */ Tone.FMSynth.prototype.dispose = function(){ Tone.Monophonic.prototype.dispose.call(this); this.carrier.dispose(); this.modulator.dispose(); this.frequency.dispose(); this._modulationIndex.dispose(); this._harmonicity.dispose(); this._modulationNode.disconnect(); this.carrier = null; this.modulator = null; this.frequency = null; this._modulationIndex = null; this._harmonicity = null; this._modulationNode = null; }; return Tone.FMSynth; });