2024-05-03 18:31:14 +00:00
|
|
|
import {
|
|
|
|
InputNode,
|
|
|
|
ToneAudioNode,
|
|
|
|
ToneAudioNodeOptions,
|
|
|
|
} from "../../core/context/ToneAudioNode.js";
|
|
|
|
import { Compressor, CompressorOptions } from "./Compressor.js";
|
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults.js";
|
|
|
|
import { readOnly, RecursivePartial } from "../../core/util/Interface.js";
|
|
|
|
import { Frequency } from "../../core/type/Units.js";
|
|
|
|
import { MultibandSplit } from "../channel/MultibandSplit.js";
|
|
|
|
import { Signal } from "../../signal/Signal.js";
|
|
|
|
import { Gain } from "../../core/context/Gain.js";
|
2019-11-14 17:13:09 +00:00
|
|
|
|
|
|
|
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-05-03 18:31:14 +00:00
|
|
|
* A compressor with separate controls over low/mid/high dynamics.
|
2024-04-29 16:59:49 +00:00
|
|
|
* @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() {
|
2024-05-03 18:31:14 +00:00
|
|
|
const options = optionsFromArguments(
|
|
|
|
MultibandCompressor.getDefaults(),
|
|
|
|
arguments
|
|
|
|
);
|
2024-05-06 14:55:55 +00:00
|
|
|
super(options);
|
2019-11-14 17:13:09 +00:00
|
|
|
|
|
|
|
this._splitter = this.input = new MultibandSplit({
|
|
|
|
context: this.context,
|
|
|
|
lowFrequency: options.lowFrequency,
|
2024-05-03 18:31:14 +00:00
|
|
|
highFrequency: options.highFrequency,
|
2019-11-14 17:13:09 +00:00
|
|
|
});
|
|
|
|
this.lowFrequency = this._splitter.lowFrequency;
|
|
|
|
this.highFrequency = this._splitter.highFrequency;
|
|
|
|
this.output = new Gain({ context: this.context });
|
2024-05-03 18:31:14 +00:00
|
|
|
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 })
|
|
|
|
);
|
2019-11-14 17:13:09 +00:00
|
|
|
|
|
|
|
// 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,
|
2024-05-03 18:31:14 +00:00
|
|
|
knee: 10,
|
2019-11-14 17:13:09 +00:00
|
|
|
},
|
|
|
|
mid: {
|
|
|
|
ratio: 3,
|
|
|
|
threshold: -24,
|
|
|
|
release: 0.03,
|
|
|
|
attack: 0.02,
|
2024-05-03 18:31:14 +00:00
|
|
|
knee: 16,
|
2019-11-14 17:13:09 +00:00
|
|
|
},
|
|
|
|
high: {
|
|
|
|
ratio: 3,
|
|
|
|
threshold: -24,
|
|
|
|
release: 0.03,
|
|
|
|
attack: 0.02,
|
2024-05-03 18:31:14 +00:00
|
|
|
knee: 16,
|
2019-11-14 17:13:09 +00:00
|
|
|
},
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this._splitter.dispose();
|
|
|
|
this.low.dispose();
|
|
|
|
this.mid.dispose();
|
|
|
|
this.high.dispose();
|
|
|
|
this.output.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|