2024-05-03 14:10:40 +00:00
|
|
|
import { EffectOptions } from "./Effect.js";
|
2024-05-03 15:09:28 +00:00
|
|
|
import {
|
|
|
|
connect,
|
|
|
|
connectSeries,
|
|
|
|
OutputNode,
|
|
|
|
ToneAudioNode,
|
|
|
|
} from "../core/context/ToneAudioNode.js";
|
2024-05-03 14:10:40 +00:00
|
|
|
import { CrossFade } from "../component/channel/CrossFade.js";
|
|
|
|
import { Signal } from "../signal/Signal.js";
|
|
|
|
import { Split } from "../component/channel/Split.js";
|
|
|
|
import { Gain } from "../core/context/Gain.js";
|
|
|
|
import { Merge } from "../component/channel/Merge.js";
|
|
|
|
import { readOnly } from "../core/util/Interface.js";
|
2019-10-30 17:13:26 +00:00
|
|
|
|
|
|
|
export type StereoEffectOptions = EffectOptions;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Base class for Stereo effects.
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
export class StereoEffect<
|
|
|
|
Options extends StereoEffectOptions,
|
|
|
|
> extends ToneAudioNode<Options> {
|
2019-10-30 17:13:26 +00:00
|
|
|
readonly name: string = "StereoEffect";
|
|
|
|
|
|
|
|
readonly input: Gain;
|
|
|
|
readonly output: CrossFade;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the drywet knob to control the amount of effect
|
|
|
|
*/
|
|
|
|
private _dryWet: CrossFade;
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-10-30 17:13:26 +00:00
|
|
|
/**
|
|
|
|
* The wet control, i.e. how much of the effected
|
|
|
|
* will pass through to the output.
|
|
|
|
*/
|
|
|
|
readonly wet: Signal<"normalRange">;
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-10-30 17:13:26 +00:00
|
|
|
/**
|
|
|
|
* Split it
|
|
|
|
*/
|
2019-11-04 23:59:32 +00:00
|
|
|
protected _split: Split;
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-10-30 17:13:26 +00:00
|
|
|
/**
|
|
|
|
* the stereo effect merger
|
|
|
|
*/
|
2019-11-04 23:59:32 +00:00
|
|
|
protected _merge: Merge;
|
2019-10-30 17:13:26 +00:00
|
|
|
|
|
|
|
constructor(options: StereoEffectOptions) {
|
|
|
|
super(options);
|
|
|
|
|
|
|
|
this.input = new Gain({ context: this.context });
|
|
|
|
// force mono sources to be stereo
|
|
|
|
this.input.channelCount = 2;
|
|
|
|
this.input.channelCountMode = "explicit";
|
|
|
|
|
|
|
|
this._dryWet = this.output = new CrossFade({
|
|
|
|
context: this.context,
|
2024-05-03 15:09:28 +00:00
|
|
|
fade: options.wet,
|
2019-10-30 17:13:26 +00:00
|
|
|
});
|
|
|
|
this.wet = this._dryWet.fade;
|
|
|
|
this._split = new Split({ context: this.context, channels: 2 });
|
|
|
|
this._merge = new Merge({ context: this.context, channels: 2 });
|
|
|
|
|
|
|
|
// connections
|
|
|
|
this.input.connect(this._split);
|
|
|
|
// dry wet connections
|
|
|
|
this.input.connect(this._dryWet.a);
|
|
|
|
this._merge.connect(this._dryWet.b);
|
|
|
|
readOnly(this, ["wet"]);
|
|
|
|
}
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-10-30 17:13:26 +00:00
|
|
|
/**
|
|
|
|
* Connect the left part of the effect
|
|
|
|
*/
|
2021-10-13 22:38:30 +00:00
|
|
|
protected connectEffectLeft(...nodes: OutputNode[]): void {
|
2019-11-03 23:34:57 +00:00
|
|
|
this._split.connect(nodes[0], 0, 0);
|
|
|
|
connectSeries(...nodes);
|
2024-05-03 15:09:28 +00:00
|
|
|
connect(nodes[nodes.length - 1], this._merge, 0, 0);
|
2019-10-30 17:13:26 +00:00
|
|
|
}
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-10-30 17:13:26 +00:00
|
|
|
/**
|
|
|
|
* Connect the right part of the effect
|
|
|
|
*/
|
2021-10-13 22:38:30 +00:00
|
|
|
protected connectEffectRight(...nodes: OutputNode[]): void {
|
2019-11-03 23:34:57 +00:00
|
|
|
this._split.connect(nodes[0], 1, 0);
|
|
|
|
connectSeries(...nodes);
|
2024-05-03 15:09:28 +00:00
|
|
|
connect(nodes[nodes.length - 1], this._merge, 0, 1);
|
2019-10-30 17:13:26 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): StereoEffectOptions {
|
|
|
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
|
|
|
wet: 1,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this._dryWet.dispose();
|
|
|
|
this._split.dispose();
|
|
|
|
this._merge.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|