mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-21 00:03:54 +00:00
124 lines
No EOL
2.8 KiB
JavaScript
124 lines
No EOL
2.8 KiB
JavaScript
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);
|
|
|
|
/**
|
|
* 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;
|
|
}); |