2024-05-03 18:31:14 +00:00
|
|
|
import {
|
|
|
|
ToneAudioNode,
|
|
|
|
ToneAudioNodeOptions,
|
|
|
|
} from "../../core/context/ToneAudioNode.js";
|
|
|
|
import { Merge } from "./Merge.js";
|
|
|
|
import { Add } from "../../signal/Add.js";
|
|
|
|
import { Multiply } from "../../signal/Multiply.js";
|
|
|
|
import { Subtract } from "../../signal/Subtract.js";
|
|
|
|
import { Gain } from "../../core/context/Gain.js";
|
|
|
|
import { optionsFromArguments } from "../../core/util/Defaults.js";
|
2019-10-29 00:42:43 +00:00
|
|
|
|
|
|
|
export type MidSideMergeOptions = ToneAudioNodeOptions;
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* MidSideMerge merges the mid and side signal after they've been separated by {@link MidSideSplit}
|
2019-10-29 00:42:43 +00:00
|
|
|
* ```
|
|
|
|
* Mid = (Left+Right)/sqrt(2); // obtain mid-signal from left and right
|
2020-09-02 20:53:38 +00:00
|
|
|
* Side = (Left-Right)/sqrt(2); // obtain side-signal from left and right
|
2019-10-29 00:42:43 +00:00
|
|
|
* ```
|
2020-09-02 20:53:38 +00:00
|
|
|
* @category Component
|
2019-10-29 00:42:43 +00:00
|
|
|
*/
|
|
|
|
export class MidSideMerge extends ToneAudioNode<MidSideMergeOptions> {
|
|
|
|
readonly name: string = "MidSideMerge";
|
|
|
|
|
|
|
|
/**
|
2024-04-29 14:48:37 +00:00
|
|
|
* There is no input, connect sources to either {@link mid} or {@link side} inputs.
|
2019-10-29 00:42:43 +00:00
|
|
|
*/
|
|
|
|
readonly input: undefined;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The merged signal
|
|
|
|
*/
|
|
|
|
readonly output: Merge;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Merge the incoming signal into left and right channels
|
|
|
|
*/
|
|
|
|
private _merge: Merge;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The "mid" input.
|
|
|
|
*/
|
2019-11-04 02:35:26 +00:00
|
|
|
readonly mid: ToneAudioNode;
|
2019-10-29 00:42:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* The "side" input.
|
|
|
|
*/
|
2019-11-04 02:35:26 +00:00
|
|
|
readonly side: ToneAudioNode;
|
2019-10-29 00:42:43 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* Recombine the mid/side into Left
|
|
|
|
*/
|
|
|
|
private _left: Add;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Recombine the mid/side into Right
|
|
|
|
*/
|
|
|
|
private _right: Subtract;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Multiply the right by sqrt(1/2)
|
|
|
|
*/
|
|
|
|
private _leftMult: Multiply;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Multiply the left by sqrt(1/2)
|
|
|
|
*/
|
|
|
|
private _rightMult: Multiply;
|
2024-05-03 18:31:14 +00:00
|
|
|
|
2019-10-29 00:42:43 +00:00
|
|
|
constructor(options?: Partial<MidSideMergeOptions>);
|
|
|
|
constructor() {
|
|
|
|
super(optionsFromArguments(MidSideMerge.getDefaults(), arguments));
|
|
|
|
this.mid = new Gain({ context: this.context });
|
|
|
|
this.side = new Gain({ context: this.context });
|
|
|
|
this._left = new Add({ context: this.context });
|
|
|
|
this._leftMult = new Multiply({
|
2024-05-03 18:31:14 +00:00
|
|
|
context: this.context,
|
|
|
|
value: Math.SQRT1_2,
|
2019-10-29 00:42:43 +00:00
|
|
|
});
|
|
|
|
this._right = new Subtract({ context: this.context });
|
|
|
|
this._rightMult = new Multiply({
|
2024-05-03 18:31:14 +00:00
|
|
|
context: this.context,
|
|
|
|
value: Math.SQRT1_2,
|
2019-10-29 00:42:43 +00:00
|
|
|
});
|
|
|
|
this._merge = this.output = new Merge({ context: this.context });
|
|
|
|
|
|
|
|
this.mid.fan(this._left);
|
|
|
|
this.side.connect(this._left.addend);
|
|
|
|
this.mid.connect(this._right);
|
|
|
|
this.side.connect(this._right.subtrahend);
|
|
|
|
this._left.connect(this._leftMult);
|
|
|
|
this._right.connect(this._rightMult);
|
|
|
|
this._leftMult.connect(this._merge, 0, 0);
|
|
|
|
this._rightMult.connect(this._merge, 0, 1);
|
|
|
|
}
|
2024-05-03 18:31:14 +00:00
|
|
|
|
2019-10-29 00:42:43 +00:00
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
|
|
|
this.mid.dispose();
|
|
|
|
this.side.dispose();
|
|
|
|
this._leftMult.dispose();
|
|
|
|
this._rightMult.dispose();
|
|
|
|
this._left.dispose();
|
|
|
|
this._right.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|