2019-11-04 01:33:46 +00:00
|
|
|
import { NormalRange } from "../core/type/Units";
|
|
|
|
import { StereoEffect, StereoEffectOptions } from "./StereoEffect";
|
|
|
|
import { optionsFromArguments } from "../core/util/Defaults";
|
|
|
|
import { Scale } from "../signal/Scale";
|
|
|
|
import { Signal } from "../signal/Signal";
|
|
|
|
import { FeedbackCombFilter } from "../component/filter/FeedbackCombFilter";
|
|
|
|
import { readOnly } from "../core/util/Interface";
|
|
|
|
|
|
|
|
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.
|
2024-04-29 14:48:37 +00:00
|
|
|
* 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}.
|
2019-11-04 01:33:46 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const reverb = new Tone.JCReverb(0.4).toDestination();
|
|
|
|
* const delay = new Tone.FeedbackDelay(0.5);
|
2019-11-04 01:33:46 +00:00
|
|
|
* // connecting the synth to reverb through delay
|
2020-04-17 02:24:18 +00:00
|
|
|
* const synth = new Tone.DuoSynth().chain(delay, reverb);
|
2019-11-04 01:33:46 +00:00
|
|
|
* synth.triggerAttackRelease("A4", "8n");
|
2019-11-05 03:29:14 +00:00
|
|
|
*
|
|
|
|
* @category Effect
|
2019-11-04 01:33:46 +00:00
|
|
|
*/
|
|
|
|
export class JCReverb extends StereoEffect<JCReverbOptions> {
|
2019-11-04 02:43:01 +00:00
|
|
|
|
|
|
|
readonly name: string = "JCReverb";
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-04 01:33:46 +00:00
|
|
|
/**
|
|
|
|
* 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,
|
2020-04-17 02:24:18 +00:00
|
|
|
min: -0.733,
|
2019-11-04 01:33:46 +00:00
|
|
|
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;
|
|
|
|
});
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-04 01:33:46 +00:00
|
|
|
// and the comb filters
|
|
|
|
this._feedbackCombFilters = combFilterDelayTimes.map((delayTime, index) => {
|
|
|
|
const fbcf = new FeedbackCombFilter({
|
|
|
|
context: this.context,
|
2020-04-17 02:24:18 +00:00
|
|
|
delayTime,
|
2019-11-04 01:33:46 +00:00
|
|
|
});
|
|
|
|
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;
|
|
|
|
}
|
|
|
|
}
|