2024-05-03 14:10:40 +00:00
|
|
|
import { StereoEffect, StereoEffectOptions } from "./StereoEffect.js";
|
|
|
|
import { Frequency, NormalRange } from "../core/type/Units.js";
|
|
|
|
import { optionsFromArguments } from "../core/util/Defaults.js";
|
|
|
|
import { readOnly } from "../core/util/Interface.js";
|
|
|
|
import { Signal } from "../signal/Signal.js";
|
|
|
|
import { LowpassCombFilter } from "../component/filter/LowpassCombFilter.js";
|
2019-11-03 23:35:17 +00:00
|
|
|
|
|
|
|
export interface FreeverbOptions extends StereoEffectOptions {
|
|
|
|
dampening: Frequency;
|
|
|
|
roomSize: NormalRange;
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* An array of comb filter delay values from Freeverb implementation
|
|
|
|
*/
|
2024-05-03 15:09:28 +00:00
|
|
|
const combFilterTunings = [
|
|
|
|
1557 / 44100,
|
|
|
|
1617 / 44100,
|
|
|
|
1491 / 44100,
|
|
|
|
1422 / 44100,
|
|
|
|
1277 / 44100,
|
|
|
|
1356 / 44100,
|
|
|
|
1188 / 44100,
|
|
|
|
1116 / 44100,
|
|
|
|
];
|
2019-11-03 23:35:17 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* An array of allpass filter frequency values from Freeverb implementation
|
|
|
|
*/
|
|
|
|
const allpassFilterFrequencies = [225, 556, 441, 341];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* Freeverb is a reverb based on [Freeverb](https://ccrma.stanford.edu/~jos/pasp/Freeverb.html).
|
|
|
|
* Read more on reverb on [Sound On Sound](https://web.archive.org/web/20160404083902/http://www.soundonsound.com:80/sos/feb01/articles/synthsecrets.asp).
|
2024-04-29 14:48:37 +00:00
|
|
|
* Freeverb is now implemented with an AudioWorkletNode which may result on performance degradation on some platforms. Consider using {@link Reverb}.
|
2019-11-03 23:36:51 +00:00
|
|
|
* @example
|
2020-04-17 02:24:18 +00:00
|
|
|
* const freeverb = new Tone.Freeverb().toDestination();
|
2019-11-03 23:36:51 +00:00
|
|
|
* freeverb.dampening = 1000;
|
|
|
|
* // routing synth through the reverb
|
2020-04-17 02:24:18 +00:00
|
|
|
* const synth = new Tone.NoiseSynth().connect(freeverb);
|
2019-11-03 23:36:51 +00:00
|
|
|
* synth.triggerAttackRelease(0.05);
|
2019-11-05 03:29:14 +00:00
|
|
|
* @category Effect
|
2019-11-03 23:35:17 +00:00
|
|
|
*/
|
|
|
|
export class Freeverb extends StereoEffect<FreeverbOptions> {
|
|
|
|
readonly name: string = "Freeverb";
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The roomSize value between 0 and 1. A larger roomSize will result in a longer decay.
|
|
|
|
*/
|
|
|
|
readonly roomSize: Signal<"normalRange">;
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the comb filters
|
|
|
|
*/
|
|
|
|
private _combFilters: LowpassCombFilter[] = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the allpass filters on the left
|
|
|
|
*/
|
|
|
|
private _allpassFiltersL: BiquadFilterNode[] = [];
|
|
|
|
|
|
|
|
/**
|
|
|
|
* the allpass filters on the right
|
|
|
|
*/
|
2020-04-17 02:24:18 +00:00
|
|
|
private _allpassFiltersR: BiquadFilterNode[] = [];
|
2019-11-03 23:35:17 +00:00
|
|
|
|
|
|
|
/**
|
|
|
|
* @param roomSize Correlated to the decay time.
|
|
|
|
* @param dampening The cutoff frequency of a lowpass filter as part of the reverb.
|
|
|
|
*/
|
|
|
|
constructor(roomSize?: NormalRange, dampening?: Frequency);
|
|
|
|
constructor(options?: Partial<FreeverbOptions>);
|
|
|
|
constructor() {
|
2024-05-03 15:09:28 +00:00
|
|
|
super(
|
|
|
|
optionsFromArguments(Freeverb.getDefaults(), arguments, [
|
|
|
|
"roomSize",
|
|
|
|
"dampening",
|
|
|
|
])
|
|
|
|
);
|
|
|
|
const options = optionsFromArguments(
|
|
|
|
Freeverb.getDefaults(),
|
|
|
|
arguments,
|
|
|
|
["roomSize", "dampening"]
|
|
|
|
);
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-03 23:35:17 +00:00
|
|
|
this.roomSize = new Signal({
|
|
|
|
context: this.context,
|
|
|
|
value: options.roomSize,
|
|
|
|
units: "normalRange",
|
|
|
|
});
|
|
|
|
|
|
|
|
// make the allpass filters on the right
|
2024-05-03 15:09:28 +00:00
|
|
|
this._allpassFiltersL = allpassFilterFrequencies.map((freq) => {
|
2019-11-04 01:34:01 +00:00
|
|
|
const allpassL = this.context.createBiquadFilter();
|
2019-11-03 23:35:17 +00:00
|
|
|
allpassL.type = "allpass";
|
2019-11-04 01:34:01 +00:00
|
|
|
allpassL.frequency.value = freq;
|
|
|
|
return allpassL;
|
|
|
|
});
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-03 23:35:17 +00:00
|
|
|
// make the allpass filters on the left
|
2024-05-03 15:09:28 +00:00
|
|
|
this._allpassFiltersR = allpassFilterFrequencies.map((freq) => {
|
2019-11-04 01:34:01 +00:00
|
|
|
const allpassR = this.context.createBiquadFilter();
|
2019-11-03 23:35:17 +00:00
|
|
|
allpassR.type = "allpass";
|
2019-11-04 01:34:01 +00:00
|
|
|
allpassR.frequency.value = freq;
|
|
|
|
return allpassR;
|
|
|
|
});
|
2019-11-03 23:35:17 +00:00
|
|
|
|
|
|
|
// make the comb filters
|
2019-11-04 01:34:01 +00:00
|
|
|
this._combFilters = combFilterTunings.map((delayTime, index) => {
|
2019-11-03 23:35:17 +00:00
|
|
|
const lfpf = new LowpassCombFilter({
|
|
|
|
context: this.context,
|
|
|
|
dampening: options.dampening,
|
2019-11-04 01:34:01 +00:00
|
|
|
delayTime,
|
2019-11-03 23:35:17 +00:00
|
|
|
});
|
2019-11-04 01:34:01 +00:00
|
|
|
if (index < combFilterTunings.length / 2) {
|
2019-11-03 23:35:17 +00:00
|
|
|
this.connectEffectLeft(lfpf, ...this._allpassFiltersL);
|
|
|
|
} else {
|
|
|
|
this.connectEffectRight(lfpf, ...this._allpassFiltersR);
|
|
|
|
}
|
|
|
|
this.roomSize.connect(lfpf.resonance);
|
2019-11-04 01:34:01 +00:00
|
|
|
return lfpf;
|
|
|
|
});
|
2019-11-03 23:35:17 +00:00
|
|
|
|
|
|
|
readOnly(this, ["roomSize"]);
|
|
|
|
}
|
|
|
|
|
|
|
|
static getDefaults(): FreeverbOptions {
|
|
|
|
return Object.assign(StereoEffect.getDefaults(), {
|
|
|
|
roomSize: 0.7,
|
2024-05-03 15:09:28 +00:00
|
|
|
dampening: 3000,
|
2019-11-03 23:35:17 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* The amount of dampening of the reverberant signal.
|
|
|
|
*/
|
2020-04-17 02:24:18 +00:00
|
|
|
|
2019-11-03 23:35:17 +00:00
|
|
|
get dampening(): Frequency {
|
|
|
|
return this._combFilters[0].dampening;
|
|
|
|
}
|
|
|
|
set dampening(d) {
|
2024-05-03 15:09:28 +00:00
|
|
|
this._combFilters.forEach((c) => (c.dampening = d));
|
2019-11-03 23:35:17 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
dispose(): this {
|
|
|
|
super.dispose();
|
2024-05-03 15:09:28 +00:00
|
|
|
this._allpassFiltersL.forEach((al) => al.disconnect());
|
|
|
|
this._allpassFiltersR.forEach((ar) => ar.disconnect());
|
|
|
|
this._combFilters.forEach((cf) => cf.dispose());
|
2019-11-03 23:35:17 +00:00
|
|
|
this.roomSize.dispose();
|
|
|
|
return this;
|
|
|
|
}
|
|
|
|
}
|