2024-05-03 14:10:40 +00:00
|
|
|
import { StereoEffect, StereoEffectOptions } from "./StereoEffect.js";
|
|
|
|
import { NormalRange } from "../core/type/Units.js";
|
|
|
|
import { Signal } from "../signal/Signal.js";
|
|
|
|
import { Gain } from "../core/context/Gain.js";
|
|
|
|
import { readOnly } from "../core/util/Interface.js";
|
|
|
|
import { Split } from "../component/channel/Split.js";
|
|
|
|
import { Merge } from "../component/channel/Merge.js";
|
2019-11-05 00:09:13 +00:00
|
|
|
|
|
|
|
export interface StereoFeedbackEffectOptions extends StereoEffectOptions {
|
|
|
|
feedback: NormalRange;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
2020-09-02 20:53:38 +00:00
|
|
|
* Base class for stereo feedback effects where the effectReturn is fed back into the same channel.
|
2019-11-05 00:09:13 +00:00
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
export class StereoFeedbackEffect<
|
|
|
|
Options extends StereoFeedbackEffectOptions,
|
|
|
|
> extends StereoEffect<Options> {
|
2019-11-05 00:09:13 +00:00
|
|
|
/**
|
|
|
|
* The amount of feedback from the output
|
|
|
|
* back into the input of the effect (routed
|
|
|
|
* across left and right channels).
|
|
|
|
*/
|
|
|
|
readonly feedback: Signal<"normalRange">;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the left side feedback
|
|
|
|
*/
|
|
|
|
protected _feedbackL: Gain;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the right side feedback
|
|
|
|
*/
|
|
|
|
protected _feedbackR: Gain;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Split the channels for feedback
|
|
|
|
*/
|
|
|
|
protected _feedbackSplit: Split;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Merge the channels for feedback
|
|
|
|
*/
|
|
|
|
protected _feedbackMerge: Merge;
|
|
|
|
|
|
|
|
constructor(options: StereoFeedbackEffectOptions) {
|
|
|
|
super(options);
|
|
|
|
|
|
|
|
this.feedback = new Signal({
|
|
|
|
context: this.context,
|
2024-05-03 15:09:28 +00:00
|
|
|
value: options.feedback,
|
|
|
|
units: "normalRange",
|
2019-11-05 00:09:13 +00:00
|
|
|
});
|
|
|
|
this._feedbackL = new Gain({ context: this.context });
|
|
|
|
this._feedbackR = new Gain({ context: this.context });
|
|
|
|
|
|
|
|
this._feedbackSplit = new Split({ context: this.context, channels: 2 });
|
|
|
|
this._feedbackMerge = new Merge({ context: this.context, channels: 2 });
|
|
|
|
|
|
|
|
this._merge.connect(this._feedbackSplit);
|
|
|
|
this._feedbackMerge.connect(this._split);
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-11-13 17:58:37 +00:00
|
|
|
// the left output connected to the left input
|
2019-11-05 00:09:13 +00:00
|
|
|
this._feedbackSplit.connect(this._feedbackL, 0, 0);
|
|
|
|
this._feedbackL.connect(this._feedbackMerge, 0, 0);
|
|
|
|
|
2019-11-13 17:58:37 +00:00
|
|
|
// the right output connected to the right input
|
2019-11-05 00:09:13 +00:00
|
|
|
this._feedbackSplit.connect(this._feedbackR, 1, 0);
|
|
|
|
this._feedbackR.connect(this._feedbackMerge, 0, 1);
|
2024-05-03 15:09:28 +00:00
|
|
|
|
2019-11-05 00:09:13 +00:00
|
|
|
// the feedback control
|
|
|
|
this.feedback.fan(this._feedbackL.gain, this._feedbackR.gain);
|
|
|
|
readOnly(this, ["feedback"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): StereoFeedbackEffectOptions {
|
|
|
|
return Object.assign(StereoEffect.getDefaults(), {
|
|
|
|
feedback: 0.5,
|
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.feedback.dispose();
|
|
|
|
this._feedbackL.dispose();
|
|
|
|
this._feedbackR.dispose();
|
|
|
|
this._feedbackSplit.dispose();
|
|
|
|
this._feedbackMerge.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|