2019-09-09 21:53:39 +00:00
|
|
|
import { gainToDb } from "../../core/type/Conversions";
|
2019-12-15 21:43:41 +00:00
|
|
|
import { NormalRange } from "../../core/type/Units";
|
2019-09-09 21:53:39 +00:00
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
2019-09-09 23:27:45 +00:00
|
|
|
import { MeterBase, MeterBaseOptions } from "./MeterBase";
|
2019-09-20 04:16:03 +00:00
|
|
|
import { warn } from "../../core/util/Debug";
|
2019-12-15 21:43:41 +00:00
|
|
|
import { Analyser } from "./Analyser";
|
2019-09-09 21:53:39 +00:00
|
|
|
|
2019-09-09 23:27:45 +00:00
|
|
|
export interface MeterOptions extends MeterBaseOptions {
|
2019-09-09 21:53:39 +00:00
|
|
|
smoothing: NormalRange;
|
2019-12-15 21:01:19 +00:00
|
|
|
normalRange: boolean;
|
2021-10-13 17:09:09 +00:00
|
|
|
channelCount: number;
|
2019-09-09 21:53:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* 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.
|
2020-10-23 17:42:01 +00:00
|
|
|
* Setting `normalRange` to `true` will covert the output to a range of
|
|
|
|
* 0-1. See an example using a graphical display
|
2020-10-23 17:51:08 +00:00
|
|
|
* [here](https://tonejs.github.io/examples/meter). See also {@link DCMeter}.
|
2019-09-09 21:53:39 +00:00
|
|
|
*
|
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const meter = new Tone.Meter();
|
|
|
|
* const mic = new Tone.UserMedia();
|
2019-10-23 03:39:35 +00:00
|
|
|
* mic.open();
|
2019-10-23 03:04:52 +00:00
|
|
|
* // connect mic to the meter
|
2019-09-09 21:53:39 +00:00
|
|
|
* mic.connect(meter);
|
2019-10-23 03:39:35 +00:00
|
|
|
* // the current level of the mic
|
2020-07-26 21:06:00 +00:00
|
|
|
* setInterval(() => console.log(meter.getValue()), 100);
|
2019-09-16 14:15:23 +00:00
|
|
|
* @category Component
|
2019-09-09 21:53:39 +00:00
|
|
|
*/
|
2019-09-09 23:27:45 +00:00
|
|
|
export class Meter extends MeterBase<MeterOptions> {
|
2019-09-09 21:53:39 +00:00
|
|
|
|
|
|
|
readonly name: string = "Meter";
|
|
|
|
|
2019-12-15 21:01:19 +00:00
|
|
|
/**
|
|
|
|
* If the output should be in decibels or normal range between 0-1. If `normalRange` is false,
|
|
|
|
* the output range will be the measured decibel value, otherwise the decibel value will be converted to
|
|
|
|
* the range of 0-1
|
|
|
|
*/
|
|
|
|
normalRange: boolean;
|
|
|
|
|
2019-09-09 21:53:39 +00:00
|
|
|
/**
|
|
|
|
* A value from between 0 and 1 where 0 represents no time averaging with the last analysis frame.
|
|
|
|
*/
|
|
|
|
smoothing: number;
|
|
|
|
|
|
|
|
/**
|
2022-01-06 05:45:52 +00:00
|
|
|
* The previous frame's value for each channel.
|
2019-09-09 21:53:39 +00:00
|
|
|
*/
|
2022-01-06 05:45:52 +00:00
|
|
|
private _rms: number[];
|
2019-09-09 21:53:39 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param smoothing The amount of smoothing applied between frames.
|
|
|
|
*/
|
2019-09-09 22:23:48 +00:00
|
|
|
constructor(smoothing?: NormalRange);
|
2019-09-09 21:53:39 +00:00
|
|
|
constructor(options?: Partial<MeterOptions>);
|
|
|
|
constructor() {
|
|
|
|
super(optionsFromArguments(Meter.getDefaults(), arguments, ["smoothing"]));
|
|
|
|
const options = optionsFromArguments(Meter.getDefaults(), arguments, ["smoothing"]);
|
|
|
|
|
2019-12-15 21:43:41 +00:00
|
|
|
this.input = this.output = this._analyser = new Analyser({
|
|
|
|
context: this.context,
|
|
|
|
size: 256,
|
|
|
|
type: "waveform",
|
2021-10-13 17:09:09 +00:00
|
|
|
channels: options.channelCount,
|
2019-12-15 21:43:41 +00:00
|
|
|
});
|
|
|
|
|
|
|
|
this.smoothing = options.smoothing,
|
2020-06-17 03:14:19 +00:00
|
|
|
this.normalRange = options.normalRange;
|
2022-01-06 05:45:52 +00:00
|
|
|
this._rms = new Array(options.channelCount);
|
|
|
|
this._rms.fill(0);
|
2019-09-09 21:53:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): MeterOptions {
|
2019-09-09 23:27:45 +00:00
|
|
|
return Object.assign(MeterBase.getDefaults(), {
|
2019-09-09 21:53:39 +00:00
|
|
|
smoothing: 0.8,
|
2019-12-15 21:01:19 +00:00
|
|
|
normalRange: false,
|
2021-10-13 17:09:09 +00:00
|
|
|
channelCount: 1,
|
2019-09-09 21:53:39 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-09 23:27:45 +00:00
|
|
|
* Use [[getValue]] instead. For the previous getValue behavior, use DCMeter.
|
|
|
|
* @deprecated
|
2019-09-09 21:53:39 +00:00
|
|
|
*/
|
2019-12-15 21:43:41 +00:00
|
|
|
getLevel(): number | number[] {
|
2019-09-20 04:16:03 +00:00
|
|
|
warn("'getLevel' has been changed to 'getValue'");
|
2019-09-09 23:27:45 +00:00
|
|
|
return this.getValue();
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-12-15 21:02:19 +00:00
|
|
|
* Get the current value of the incoming signal.
|
|
|
|
* Output is in decibels when [[normalRange]] is `false`.
|
2019-12-15 21:43:41 +00:00
|
|
|
* If [[channels]] = 1, then the output is a single number
|
|
|
|
* representing the value of the input signal. When [[channels]] > 1,
|
|
|
|
* then each channel is returned as a value in a number array.
|
|
|
|
*/
|
|
|
|
getValue(): number | number[] {
|
|
|
|
const aValues = this._analyser.getValue();
|
|
|
|
const channelValues = this.channels === 1 ? [aValues as Float32Array] : aValues as Float32Array[];
|
2022-01-06 05:45:52 +00:00
|
|
|
const vals = channelValues.map((values, index) => {
|
2019-12-15 21:43:41 +00:00
|
|
|
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
|
2023-01-26 18:31:38 +00:00
|
|
|
this._rms[index] = Math.max(rms, this._rms[index] * this.smoothing);
|
2022-01-06 05:45:52 +00:00
|
|
|
return this.normalRange ? this._rms[index] : gainToDb(this._rms[index]);
|
2019-12-15 21:43:41 +00:00
|
|
|
});
|
|
|
|
if (this.channels === 1) {
|
|
|
|
return vals[0];
|
|
|
|
} else {
|
|
|
|
return vals;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The number of channels of analysis.
|
2019-09-09 23:27:45 +00:00
|
|
|
*/
|
2019-12-15 21:43:41 +00:00
|
|
|
get channels(): number {
|
|
|
|
return this._analyser.channels;
|
2019-09-09 21:53:39 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this._analyser.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|