Tone.js/Tone/component/channel/MultibandSplit.ts

172 lines
4 KiB
TypeScript
Raw Normal View History

import { Gain } from "../../core/context/Gain.js";
import {
ToneAudioNode,
ToneAudioNodeOptions,
} from "../../core/context/ToneAudioNode.js";
import { Frequency, Positive } from "../../core/type/Units.js";
import { optionsFromArguments } from "../../core/util/Defaults.js";
import { readOnly, writable } from "../../core/util/Interface.js";
import { Signal } from "../../signal/Signal.js";
import { Filter } from "../filter/Filter.js";
interface MultibandSplitOptions extends ToneAudioNodeOptions {
Q: Positive;
lowFrequency: Frequency;
highFrequency: Frequency;
}
/**
2019-09-14 20:39:18 +00:00
* Split the incoming signal into three bands (low, mid, high)
* with two crossover frequency controls.
* ```
* +----------------------+
* +-> input < lowFrequency +------------------> low
* | +----------------------+
* |
* | +--------------------------------------+
* input ---+-> lowFrequency < input < highFrequency +--> mid
* | +--------------------------------------+
* |
* | +-----------------------+
* +-> highFrequency < input +-----------------> high
* +-----------------------+
* ```
2019-09-16 14:15:23 +00:00
* @category Component
*/
export class MultibandSplit extends ToneAudioNode<MultibandSplitOptions> {
2019-09-04 23:18:44 +00:00
readonly name: string = "MultibandSplit";
/**
2019-09-14 20:39:18 +00:00
* the input
*/
readonly input = new Gain({ context: this.context });
/**
* no output node, use either low, mid or high outputs
*/
readonly output = undefined;
/**
2019-09-14 20:39:18 +00:00
* The low band.
*/
readonly low = new Filter({
context: this.context,
frequency: 0,
type: "lowpass",
});
/**
2019-09-14 20:39:18 +00:00
* the lower filter of the mid band
*/
private _lowMidFilter = new Filter({
context: this.context,
frequency: 0,
type: "highpass",
});
/**
2019-09-14 20:39:18 +00:00
* The mid band output.
*/
readonly mid = new Filter({
context: this.context,
frequency: 0,
type: "lowpass",
});
/**
2019-09-14 20:39:18 +00:00
* The high band output.
*/
readonly high = new Filter({
context: this.context,
frequency: 0,
type: "highpass",
});
/**
2019-09-14 20:39:18 +00:00
* The low/mid crossover frequency.
*/
readonly lowFrequency: Signal<"frequency">;
/**
2019-09-14 20:39:18 +00:00
* The mid/high crossover frequency.
*/
readonly highFrequency: Signal<"frequency">;
protected _internalChannels = [this.low, this.mid, this.high];
/**
2019-09-14 20:39:18 +00:00
* The Q or Quality of the filter
*/
readonly Q: Signal<"positive">;
2019-08-27 17:02:31 +00:00
/**
* @param lowFrequency the low/mid crossover frequency
* @param highFrequency the mid/high crossover frequency
2019-08-27 17:02:31 +00:00
*/
constructor(lowFrequency?: Frequency, highFrequency?: Frequency);
constructor(options?: Partial<MultibandSplitOptions>);
constructor() {
const options = optionsFromArguments(
MultibandSplit.getDefaults(),
arguments,
["lowFrequency", "highFrequency"]
);
super(options);
this.lowFrequency = new Signal({
context: this.context,
units: "frequency",
value: options.lowFrequency,
});
this.highFrequency = new Signal({
context: this.context,
units: "frequency",
value: options.highFrequency,
});
this.Q = new Signal({
context: this.context,
units: "positive",
value: options.Q,
});
this.input.fan(this.low, this.high);
this.input.chain(this._lowMidFilter, this.mid);
// the frequency control signal
this.lowFrequency.fan(this.low.frequency, this._lowMidFilter.frequency);
this.highFrequency.fan(this.mid.frequency, this.high.frequency);
// the Q value
this.Q.connect(this.low.Q);
this.Q.connect(this._lowMidFilter.Q);
this.Q.connect(this.mid.Q);
this.Q.connect(this.high.Q);
readOnly(this, ["high", "mid", "low", "highFrequency", "lowFrequency"]);
}
static getDefaults(): MultibandSplitOptions {
return Object.assign(ToneAudioNode.getDefaults(), {
Q: 1,
highFrequency: 2500,
lowFrequency: 400,
});
}
/**
2019-09-14 20:39:18 +00:00
* Clean up.
*/
dispose(): this {
super.dispose();
writable(this, ["high", "mid", "low", "highFrequency", "lowFrequency"]);
this.low.dispose();
this._lowMidFilter.dispose();
this.mid.dispose();
this.high.dispose();
this.lowFrequency.dispose();
this.highFrequency.dispose();
this.Q.dispose();
return this;
}
}