define(["Tone/core/Tone", "Tone/component/Analyser"], function(Tone){ "use strict"; /** * @class Tone.Meter gets the [RMS](https://en.wikipedia.org/wiki/Root_mean_square) * of an input signal with some averaging applied. It can also get the raw * value of the input signal. * * @constructor * @extends {Tone} * @param {String} type Either "level" or "signal". * @param {Number} smoothing The amount of smoothing applied between frames. * @example * var meter = new Tone.Meter(); * var mic = new Tone.UserMedia().open(); * //connect mic to the meter * mic.connect(meter); * //the current level of the mic input * var level = meter.value; */ Tone.Meter = function(){ var options = Tone.defaults(arguments, ["type", "smoothing"], Tone.Meter); Tone.call(this); /** * The type of the meter, either "level" or "signal". * A "level" meter will return the volume level (rms) of the * input signal and a "signal" meter will return * the signal value of the input. * @type {String} */ this.type = options.type; /** * The analyser node which computes the levels. * @private * @type {Tone.Analyser} */ this.input = this.output = this._analyser = new Tone.Analyser("waveform", 512); this._analyser.returnType = "float"; /** * The amount of carryover between the current and last frame. * Only applied meter for "level" type. * @type {Number} */ this.smoothing = options.smoothing; /** * The last computed value * @type {Number} * @private */ this._lastValue = 0; }; Tone.extend(Tone.Meter); /** * @private * @enum {String} */ Tone.Meter.Type = { Level : "level", Signal : "signal" }; /** * The defaults * @type {Object} * @static * @const */ Tone.Meter.defaults = { "smoothing" : 0.8, "type" : Tone.Meter.Type.Level }; /** * The current value of the meter. A value of 1 is * "unity". * @memberOf Tone.Meter# * @type {Number} * @name value * @readOnly */ Object.defineProperty(Tone.Meter.prototype, "value", { get : function(){ var signal = this._analyser.analyse(); if (this.type === Tone.Meter.Type.Level){ //rms var sum = 0; for (var i = 0; i < signal.length; i++){ sum += Math.pow(signal[i], 2); } var rms = Math.sqrt(sum / signal.length); //smooth it rms = Math.max(rms, this._lastValue * this.smoothing); this._lastValue = rms; //scale it var unity = 0.35; var val = rms / unity; //scale the output curve return Math.sqrt(val); } else { return signal[0]; } }, }); /** * Clean up. * @returns {Tone.Meter} this */ Tone.Meter.prototype.dispose = function(){ Tone.prototype.dispose.call(this); this._analyser.dispose(); this._analyser = null; return this; }; return Tone.Meter; });