2019-11-14 17:15:47 +00:00
|
|
|
import { InputNode, ToneAudioNode, ToneAudioNodeOptions } from "../../core/context/ToneAudioNode";
|
2019-11-14 17:13:09 +00:00
|
|
|
import { Compressor, CompressorOptions } from "./Compressor";
|
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
|
|
|
import { readOnly, RecursivePartial } from "../../core/util/Interface";
|
|
|
|
import { Frequency } from "../../core/type/Units";
|
|
|
|
import { MultibandSplit } from "../channel/MultibandSplit";
|
|
|
|
import { Signal } from "../../signal/Signal";
|
|
|
|
import { Gain } from "../../core/context/Gain";
|
|
|
|
|
|
|
|
export interface MultibandCompressorOptions extends ToneAudioNodeOptions {
|
|
|
|
mid: Omit<CompressorOptions, keyof ToneAudioNodeOptions>;
|
|
|
|
low: Omit<CompressorOptions, keyof ToneAudioNodeOptions>;
|
|
|
|
high: Omit<CompressorOptions, keyof ToneAudioNodeOptions>;
|
|
|
|
lowFrequency: Frequency;
|
|
|
|
highFrequency: Frequency;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2024-04-29 16:59:49 +00:00
|
|
|
* A compressor with separate controls over low/mid/high dynamics.
|
|
|
|
* @see {@link Compressor} and {@link MultibandSplit}
|
2019-11-14 17:13:09 +00:00
|
|
|
*
|
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const multiband = new Tone.MultibandCompressor({
|
2019-11-14 17:13:09 +00:00
|
|
|
* lowFrequency: 200,
|
|
|
|
* highFrequency: 1300,
|
|
|
|
* low: {
|
|
|
|
* threshold: -12
|
|
|
|
* }
|
|
|
|
* });
|
2020-09-02 20:53:38 +00:00
|
|
|
* @category Component
|
2019-11-14 17:13:09 +00:00
|
|
|
*/
|
|
|
|
export class MultibandCompressor extends ToneAudioNode<MultibandCompressorOptions> {
|
|
|
|
|
|
|
|
readonly name: string = "MultibandCompressor";
|
|
|
|
|
|
|
|
readonly input: InputNode;
|
|
|
|
readonly output: ToneAudioNode;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Split the incoming signal into high/mid/low
|
|
|
|
*/
|
|
|
|
private _splitter: MultibandSplit;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* low/mid crossover frequency.
|
|
|
|
*/
|
|
|
|
readonly lowFrequency: Signal<"frequency">;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* mid/high crossover frequency.
|
|
|
|
*/
|
2020-04-17 02:24:18 +00:00
|
|
|
readonly highFrequency: Signal<"frequency">;
|
2019-11-14 17:13:09 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The compressor applied to the low frequencies
|
|
|
|
*/
|
|
|
|
readonly low: Compressor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The compressor applied to the mid frequencies
|
|
|
|
*/
|
|
|
|
readonly mid: Compressor;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The compressor applied to the high frequencies
|
|
|
|
*/
|
|
|
|
readonly high: Compressor;
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-14 17:13:09 +00:00
|
|
|
constructor(options?: RecursivePartial<MultibandCompressorOptions>);
|
|
|
|
constructor() {
|
|
|
|
super(Object.assign(optionsFromArguments(MultibandCompressor.getDefaults(), arguments)));
|
|
|
|
const options = optionsFromArguments(MultibandCompressor.getDefaults(), arguments);
|
|
|
|
|
|
|
|
this._splitter = this.input = new MultibandSplit({
|
|
|
|
context: this.context,
|
|
|
|
lowFrequency: options.lowFrequency,
|
|
|
|
highFrequency: options.highFrequency
|
|
|
|
});
|
|
|
|
this.lowFrequency = this._splitter.lowFrequency;
|
|
|
|
this.highFrequency = this._splitter.highFrequency;
|
|
|
|
this.output = new Gain({ context: this.context });
|
|
|
|
this.low = new Compressor(Object.assign(options.low, { context: this.context }));
|
|
|
|
this.mid = new Compressor(Object.assign(options.mid, { context: this.context }));
|
|
|
|
this.high = new Compressor(Object.assign(options.high, { context: this.context }));
|
|
|
|
|
|
|
|
// connect the compressor
|
|
|
|
this._splitter.low.chain(this.low, this.output);
|
|
|
|
this._splitter.mid.chain(this.mid, this.output);
|
|
|
|
this._splitter.high.chain(this.high, this.output);
|
|
|
|
|
|
|
|
readOnly(this, ["high", "mid", "low", "highFrequency", "lowFrequency"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): MultibandCompressorOptions {
|
|
|
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
|
|
|
lowFrequency: 250,
|
|
|
|
highFrequency: 2000,
|
|
|
|
low: {
|
|
|
|
ratio: 6,
|
|
|
|
threshold: -30,
|
|
|
|
release: 0.25,
|
|
|
|
attack: 0.03,
|
|
|
|
knee: 10
|
|
|
|
},
|
|
|
|
mid: {
|
|
|
|
ratio: 3,
|
|
|
|
threshold: -24,
|
|
|
|
release: 0.03,
|
|
|
|
attack: 0.02,
|
|
|
|
knee: 16
|
|
|
|
},
|
|
|
|
high: {
|
|
|
|
ratio: 3,
|
|
|
|
threshold: -24,
|
|
|
|
release: 0.03,
|
|
|
|
attack: 0.02,
|
|
|
|
knee: 16
|
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this._splitter.dispose();
|
|
|
|
this.low.dispose();
|
|
|
|
this.mid.dispose();
|
|
|
|
this.high.dispose();
|
|
|
|
this.output.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|