mirror of
https://github.com/Tonejs/Tone.js
synced 2024-12-31 22:18:44 +00:00
102 lines
2.6 KiB
TypeScript
102 lines
2.6 KiB
TypeScript
|
import { InputNode, OutputNode, ToneAudioNode, ToneAudioNodeOptions } from "../../core/context/ToneAudioNode";
|
||
|
import { gainToDb } from "../../core/type/Conversions";
|
||
|
import { Decibels, NormalRange } from "../../core/type/Units";
|
||
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
||
|
import { Analyser } from "./Analyser";
|
||
|
|
||
|
export interface MeterOptions extends ToneAudioNodeOptions {
|
||
|
smoothing: NormalRange;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Meter gets the [RMS](https://en.wikipedia.org/wiki/Root_mean_square)
|
||
|
* of an input signal. It can also get the raw value of the input signal.
|
||
|
*
|
||
|
* @example
|
||
|
* var meter = new Meter();
|
||
|
* var mic = new Tone.UserMedia().open();
|
||
|
* //connect mic to the meter
|
||
|
* mic.connect(meter);
|
||
|
* //the current level of the mic input in decibels
|
||
|
* var level = meter.getLevel();
|
||
|
*/
|
||
|
export class Meter extends ToneAudioNode<MeterOptions> {
|
||
|
|
||
|
readonly name: string = "Meter";
|
||
|
|
||
|
/**
|
||
|
* A value from between 0 and 1 where 0 represents no time averaging with the last analysis frame.
|
||
|
*/
|
||
|
smoothing: number;
|
||
|
|
||
|
/**
|
||
|
* The previous frame's value
|
||
|
*/
|
||
|
private _rms = 0;
|
||
|
|
||
|
/**
|
||
|
* The signal to be analysed
|
||
|
*/
|
||
|
input: InputNode;
|
||
|
|
||
|
/**
|
||
|
* The output is just a pass through of the input
|
||
|
*/
|
||
|
output: OutputNode;
|
||
|
|
||
|
/**
|
||
|
* The analyser node for the incoming signal
|
||
|
*/
|
||
|
private _analyser: Analyser;
|
||
|
|
||
|
/**
|
||
|
* @param smoothing The amount of smoothing applied between frames.
|
||
|
*/
|
||
|
constructor(smoothing: NormalRange);
|
||
|
constructor(options?: Partial<MeterOptions>);
|
||
|
constructor() {
|
||
|
super(optionsFromArguments(Meter.getDefaults(), arguments, ["smoothing"]));
|
||
|
const options = optionsFromArguments(Meter.getDefaults(), arguments, ["smoothing"]);
|
||
|
|
||
|
this.smoothing = options.smoothing;
|
||
|
this.input = this.output = this._analyser = new Analyser({
|
||
|
context: this.context,
|
||
|
size: 256,
|
||
|
type: "waveform",
|
||
|
});
|
||
|
}
|
||
|
|
||
|
static getDefaults(): MeterOptions {
|
||
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
||
|
smoothing: 0.8,
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the current decibel value of the incoming signal
|
||
|
*/
|
||
|
getLevel(): Decibels {
|
||
|
const values = this._analyser.getValue();
|
||
|
const totalSquared = values.reduce((total, current) => total + current * current, 0);
|
||
|
const rms = Math.sqrt(totalSquared / values.length);
|
||
|
// the rms can only fall at the rate of the smoothing
|
||
|
// but can jump up instantly
|
||
|
this._rms = Math.max(rms, this._rms * this.smoothing);
|
||
|
return gainToDb(this._rms);
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Get the signal value of the incoming signal
|
||
|
*/
|
||
|
getValue(): number {
|
||
|
const value = this._analyser.getValue();
|
||
|
return value[0];
|
||
|
}
|
||
|
|
||
|
dispose(): this {
|
||
|
super.dispose();
|
||
|
this._analyser.dispose();
|
||
|
return this;
|
||
|
}
|
||
|
}
|