Tone.js/Tone/effect/JCReverb.ts
Yotam Mann aaf880c925
Using web-test-runner for tests, updating import paths (#1242)
* WIP moving tests to web-test-runner

* updating thresholds

* Adding file extensions

* Testing integrations

* linting

* fixing dep

* moving back to root dir

* prettier all of the files

* updating eslint rules to use with prettier

* remove import package

* moving tsignore around

* removing unneeded ignores

* all tests run on puppeteer, no need for testing guards

* linting

* import type syntax

* cleaning up

* Update package.json
2024-05-03 14:31:14 -04:00

143 lines
3.7 KiB
TypeScript

import { NormalRange } from "../core/type/Units.js";
import { StereoEffect, StereoEffectOptions } from "./StereoEffect.js";
import { optionsFromArguments } from "../core/util/Defaults.js";
import { Scale } from "../signal/Scale.js";
import { Signal } from "../signal/Signal.js";
import { FeedbackCombFilter } from "../component/filter/FeedbackCombFilter.js";
import { readOnly } from "../core/util/Interface.js";
export interface JCReverbOptions extends StereoEffectOptions {
roomSize: NormalRange;
}
/**
* an array of the comb filter delay time values
*/
const combFilterDelayTimes = [
1687 / 25000,
1601 / 25000,
2053 / 25000,
2251 / 25000,
];
/**
* the resonances of each of the comb filters
*/
const combFilterResonances = [0.773, 0.802, 0.753, 0.733];
/**
* the allpass filter frequencies
*/
const allpassFilterFreqs = [347, 113, 37];
/**
* JCReverb is a simple [Schroeder Reverberator](https://ccrma.stanford.edu/~jos/pasp/Schroeder_Reverberators.html)
* tuned by John Chowning in 1970.
* It is made up of three allpass filters and four {@link FeedbackCombFilter}.
* JCReverb is now implemented with an AudioWorkletNode which may result on performance degradation on some platforms. Consider using {@link Reverb}.
* @example
* const reverb = new Tone.JCReverb(0.4).toDestination();
* const delay = new Tone.FeedbackDelay(0.5);
* // connecting the synth to reverb through delay
* const synth = new Tone.DuoSynth().chain(delay, reverb);
* synth.triggerAttackRelease("A4", "8n");
*
* @category Effect
*/
export class JCReverb extends StereoEffect<JCReverbOptions> {
readonly name: string = "JCReverb";
/**
* Room size control values.
*/
readonly roomSize: Signal<"normalRange">;
/**
* Scale the room size
*/
private _scaleRoomSize: Scale;
/**
* a series of allpass filters
*/
private _allpassFilters: BiquadFilterNode[] = [];
/**
* parallel feedback comb filters
*/
private _feedbackCombFilters: FeedbackCombFilter[] = [];
/**
* @param roomSize Correlated to the decay time.
*/
constructor(roomSize?: NormalRange);
constructor(options?: Partial<JCReverbOptions>);
constructor() {
super(
optionsFromArguments(JCReverb.getDefaults(), arguments, [
"roomSize",
])
);
const options = optionsFromArguments(
JCReverb.getDefaults(),
arguments,
["roomSize"]
);
this.roomSize = new Signal({
context: this.context,
value: options.roomSize,
units: "normalRange",
});
this._scaleRoomSize = new Scale({
context: this.context,
min: -0.733,
max: 0.197,
});
// make the allpass filters
this._allpassFilters = allpassFilterFreqs.map((freq) => {
const allpass = this.context.createBiquadFilter();
allpass.type = "allpass";
allpass.frequency.value = freq;
return allpass;
});
// and the comb filters
this._feedbackCombFilters = combFilterDelayTimes.map(
(delayTime, index) => {
const fbcf = new FeedbackCombFilter({
context: this.context,
delayTime,
});
this._scaleRoomSize.connect(fbcf.resonance);
fbcf.resonance.value = combFilterResonances[index];
if (index < combFilterDelayTimes.length / 2) {
this.connectEffectLeft(...this._allpassFilters, fbcf);
} else {
this.connectEffectRight(...this._allpassFilters, fbcf);
}
return fbcf;
}
);
// chain the allpass filters together
this.roomSize.connect(this._scaleRoomSize);
readOnly(this, ["roomSize"]);
}
static getDefaults(): JCReverbOptions {
return Object.assign(StereoEffect.getDefaults(), {
roomSize: 0.5,
});
}
dispose(): this {
super.dispose();
this._allpassFilters.forEach((apf) => apf.disconnect());
this._feedbackCombFilters.forEach((fbcf) => fbcf.dispose());
this.roomSize.dispose();
this._scaleRoomSize.dispose();
return this;
}
}