2024-05-03 18:31:14 +00:00
|
|
|
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";
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
interface MultibandSplitOptions extends ToneAudioNodeOptions {
|
2019-06-23 10:53:57 +00:00
|
|
|
Q: Positive;
|
|
|
|
lowFrequency: Frequency;
|
|
|
|
highFrequency: Frequency;
|
2019-06-23 09:36:42 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Split the incoming signal into three bands (low, mid, high)
|
|
|
|
* with two crossover frequency controls.
|
2019-07-25 14:43:47 +00:00
|
|
|
* ```
|
|
|
|
* +----------------------+
|
2019-08-03 16:00:14 +00:00
|
|
|
* +-> input < lowFrequency +------------------> low
|
2019-07-25 14:43:47 +00:00
|
|
|
* | +----------------------+
|
|
|
|
* |
|
|
|
|
* | +--------------------------------------+
|
2019-08-03 16:00:14 +00:00
|
|
|
* input ---+-> lowFrequency < input < highFrequency +--> mid
|
2019-07-25 14:43:47 +00:00
|
|
|
* | +--------------------------------------+
|
|
|
|
* |
|
|
|
|
* | +-----------------------+
|
2019-08-03 16:00:14 +00:00
|
|
|
* +-> highFrequency < input +-----------------> high
|
2019-07-25 14:43:47 +00:00
|
|
|
* +-----------------------+
|
|
|
|
* ```
|
2019-09-16 14:15:23 +00:00
|
|
|
* @category Component
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-24 18:34:17 +00:00
|
|
|
export class MultibandSplit extends ToneAudioNode<MultibandSplitOptions> {
|
2019-09-04 23:18:44 +00:00
|
|
|
readonly name: string = "MultibandSplit";
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* the input
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
|
|
|
readonly input = new Gain({ context: this.context });
|
|
|
|
|
2019-08-03 16:00:14 +00:00
|
|
|
/**
|
|
|
|
* no output node, use either low, mid or high outputs
|
|
|
|
*/
|
|
|
|
readonly output = undefined;
|
|
|
|
|
2019-06-23 09:36:42 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The low band.
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-24 18:34:17 +00:00
|
|
|
readonly low = new Filter({
|
|
|
|
context: this.context,
|
|
|
|
frequency: 0,
|
|
|
|
type: "lowpass",
|
|
|
|
});
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* the lower filter of the mid band
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-24 18:34:17 +00:00
|
|
|
private _lowMidFilter = new Filter({
|
|
|
|
context: this.context,
|
|
|
|
frequency: 0,
|
|
|
|
type: "highpass",
|
|
|
|
});
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The mid band output.
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-24 18:34:17 +00:00
|
|
|
readonly mid = new Filter({
|
|
|
|
context: this.context,
|
|
|
|
frequency: 0,
|
|
|
|
type: "lowpass",
|
|
|
|
});
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The high band output.
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-24 18:34:17 +00:00
|
|
|
readonly high = new Filter({
|
|
|
|
context: this.context,
|
|
|
|
frequency: 0,
|
|
|
|
type: "highpass",
|
|
|
|
});
|
2019-06-23 09:36:42 +00:00
|
|
|
|
2019-06-24 18:34:17 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The low/mid crossover frequency.
|
2019-06-24 18:34:17 +00:00
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
readonly lowFrequency: Signal<"frequency">;
|
2019-06-24 18:34:17 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The mid/high crossover frequency.
|
2019-06-24 18:34:17 +00:00
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
readonly highFrequency: Signal<"frequency">;
|
2019-06-23 09:36:42 +00:00
|
|
|
|
2019-08-03 16:00:14 +00:00
|
|
|
protected _internalChannels = [this.low, this.mid, this.high];
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* The Q or Quality of the filter
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-10-28 15:37:53 +00:00
|
|
|
readonly Q: Signal<"positive">;
|
2019-06-23 09:36:42 +00:00
|
|
|
|
2019-08-27 17:02:31 +00:00
|
|
|
/**
|
2019-08-30 16:06:38 +00:00
|
|
|
* @param lowFrequency the low/mid crossover frequency
|
|
|
|
* @param highFrequency the mid/high crossover frequency
|
2019-08-27 17:02:31 +00:00
|
|
|
*/
|
2019-06-23 10:53:57 +00:00
|
|
|
constructor(lowFrequency?: Frequency, highFrequency?: Frequency);
|
|
|
|
constructor(options?: Partial<MultibandSplitOptions>);
|
|
|
|
constructor() {
|
2024-05-03 18:31:14 +00:00
|
|
|
const options = optionsFromArguments(
|
|
|
|
MultibandSplit.getDefaults(),
|
|
|
|
arguments,
|
|
|
|
["lowFrequency", "highFrequency"]
|
|
|
|
);
|
2024-05-06 14:55:55 +00:00
|
|
|
super(options);
|
2019-06-23 09:36:42 +00:00
|
|
|
|
|
|
|
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);
|
2019-06-23 10:53:57 +00:00
|
|
|
// the frequency control signal
|
2019-06-24 18:34:17 +00:00
|
|
|
this.lowFrequency.fan(this.low.frequency, this._lowMidFilter.frequency);
|
|
|
|
this.highFrequency.fan(this.mid.frequency, this.high.frequency);
|
2019-06-23 10:53:57 +00:00
|
|
|
// the Q value
|
2019-06-23 09:36:42 +00:00
|
|
|
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"]);
|
2019-06-23 10:53:57 +00:00
|
|
|
}
|
2019-06-23 09:36:42 +00:00
|
|
|
|
2019-06-23 10:53:57 +00:00
|
|
|
static getDefaults(): MultibandSplitOptions {
|
|
|
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
2019-06-23 09:36:42 +00:00
|
|
|
Q: 1,
|
2019-06-23 10:53:57 +00:00
|
|
|
highFrequency: 2500,
|
|
|
|
lowFrequency: 400,
|
|
|
|
});
|
2019-06-23 09:36:42 +00:00
|
|
|
}
|
2019-06-23 10:53:57 +00:00
|
|
|
|
2019-06-23 09:36:42 +00:00
|
|
|
/**
|
2019-09-14 20:39:18 +00:00
|
|
|
* Clean up.
|
2019-06-23 09:36:42 +00:00
|
|
|
*/
|
2019-06-23 10:53:57 +00:00
|
|
|
dispose(): this {
|
2019-06-23 09:36:42 +00:00
|
|
|
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;
|
2019-06-23 10:53:57 +00:00
|
|
|
}
|
|
|
|
}
|