import {
MidSideEffect,
MidSideEffectOptions,
} from "../effect/MidSideEffect.js";
import { Signal } from "../signal/Signal.js";
import { Multiply } from "../signal/Multiply.js";
import { Subtract } from "../signal/Subtract.js";
import { NormalRange } from "../core/type/Units.js";
import { optionsFromArguments } from "../core/util/Defaults.js";
import { readOnly } from "../core/util/Interface.js";
import { connect } from "../core/context/ToneAudioNode.js";
export interface StereoWidenerOptions extends MidSideEffectOptions {
width: NormalRange;
}
/**
* Applies a width factor to the mid/side seperation.
* 0 is all mid and 1 is all side.
* Algorithm found in [kvraudio forums](http://www.kvraudio.com/forum/viewtopic.php?t=212587).
* ```
* Mid *= 2*(1-width)
* Side *= 2*width
* ```
* @category Effect
*/
export class StereoWidener extends MidSideEffect {
readonly name: string = "StereoWidener";
/**
* The width control. 0 = 100% mid. 1 = 100% side. 0.5 = no change.
*/
readonly width: Signal<"normalRange">;
/**
* Two times the (1-width) for the mid channel
*/
private _twoTimesWidthMid: Multiply;
/**
* Two times the width for the side channel
*/
private _twoTimesWidthSide: Multiply;
/**
* Mid multiplier
*/
private _midMult: Multiply;
/**
* 1 - width
*/
private _oneMinusWidth: Subtract;
/**
* Side multiplier
*/
private _sideMult: Multiply;
/**
* @param width The stereo width. A width of 0 is mono and 1 is stereo. 0.5 is no change.
*/
constructor(width?: NormalRange);
constructor(options?: Partial);
constructor() {
super(
optionsFromArguments(StereoWidener.getDefaults(), arguments, [
"width",
])
);
const options = optionsFromArguments(
StereoWidener.getDefaults(),
arguments,
["width"]
);
this.width = new Signal({
context: this.context,
value: options.width,
units: "normalRange",
});
readOnly(this, ["width"]);
this._twoTimesWidthMid = new Multiply({
context: this.context,
value: 2,
});
this._twoTimesWidthSide = new Multiply({
context: this.context,
value: 2,
});
this._midMult = new Multiply({ context: this.context });
this._twoTimesWidthMid.connect(this._midMult.factor);
this.connectEffectMid(this._midMult);
this._oneMinusWidth = new Subtract({ context: this.context });
this._oneMinusWidth.connect(this._twoTimesWidthMid);
connect(this.context.getConstant(1), this._oneMinusWidth);
this.width.connect(this._oneMinusWidth.subtrahend);
this._sideMult = new Multiply({ context: this.context });
this.width.connect(this._twoTimesWidthSide);
this._twoTimesWidthSide.connect(this._sideMult.factor);
this.connectEffectSide(this._sideMult);
}
static getDefaults(): StereoWidenerOptions {
return Object.assign(MidSideEffect.getDefaults(), {
width: 0.5,
});
}
dispose(): this {
super.dispose();
this.width.dispose();
this._midMult.dispose();
this._sideMult.dispose();
this._twoTimesWidthMid.dispose();
this._twoTimesWidthSide.dispose();
this._oneMinusWidth.dispose();
return this;
}
}