mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-13 04:18:52 +00:00
201 lines
4.8 KiB
TypeScript
201 lines
4.8 KiB
TypeScript
import { StereoEffect, StereoEffectOptions } from "./StereoEffect.js";
|
|
import { Frequency, Hertz, Positive } from "../core/type/Units.js";
|
|
import { optionsFromArguments } from "../core/util/Defaults.js";
|
|
import { LFO } from "../source/oscillator/LFO.js";
|
|
import { Signal } from "../signal/Signal.js";
|
|
import { readOnly } from "../core/util/Interface.js";
|
|
|
|
export interface PhaserOptions extends StereoEffectOptions {
|
|
frequency: Frequency;
|
|
octaves: Positive;
|
|
stages: Positive;
|
|
Q: Positive;
|
|
baseFrequency: Frequency;
|
|
}
|
|
|
|
/**
|
|
* Phaser is a phaser effect. Phasers work by changing the phase
|
|
* of different frequency components of an incoming signal. Read more on
|
|
* [Wikipedia](https://en.wikipedia.org/wiki/Phaser_(effect)).
|
|
* Inspiration for this phaser comes from [Tuna.js](https://github.com/Dinahmoe/tuna/).
|
|
* @example
|
|
* const phaser = new Tone.Phaser({
|
|
* frequency: 15,
|
|
* octaves: 5,
|
|
* baseFrequency: 1000
|
|
* }).toDestination();
|
|
* const synth = new Tone.FMSynth().connect(phaser);
|
|
* synth.triggerAttackRelease("E3", "2n");
|
|
* @category Effect
|
|
*/
|
|
export class Phaser extends StereoEffect<PhaserOptions> {
|
|
readonly name: string = "Phaser";
|
|
|
|
/**
|
|
* the lfo which controls the frequency on the left side
|
|
*/
|
|
private _lfoL: LFO;
|
|
|
|
/**
|
|
* the lfo which controls the frequency on the right side
|
|
*/
|
|
private _lfoR: LFO;
|
|
|
|
/**
|
|
* the base modulation frequency
|
|
*/
|
|
private _baseFrequency: Hertz;
|
|
|
|
/**
|
|
* the octaves of the phasing
|
|
*/
|
|
private _octaves: Positive;
|
|
|
|
/**
|
|
* The quality factor of the filters
|
|
*/
|
|
readonly Q: Signal<"positive">;
|
|
|
|
/**
|
|
* the array of filters for the left side
|
|
*/
|
|
private _filtersL: BiquadFilterNode[];
|
|
|
|
/**
|
|
* the array of filters for the left side
|
|
*/
|
|
private _filtersR: BiquadFilterNode[];
|
|
|
|
/**
|
|
* the frequency of the effect
|
|
*/
|
|
readonly frequency: Signal<"frequency">;
|
|
|
|
/**
|
|
* @param frequency The speed of the phasing.
|
|
* @param octaves The octaves of the effect.
|
|
* @param baseFrequency The base frequency of the filters.
|
|
*/
|
|
constructor(
|
|
frequency?: Frequency,
|
|
octaves?: Positive,
|
|
baseFrequency?: Frequency
|
|
);
|
|
constructor(options?: Partial<PhaserOptions>);
|
|
constructor() {
|
|
super(
|
|
optionsFromArguments(Phaser.getDefaults(), arguments, [
|
|
"frequency",
|
|
"octaves",
|
|
"baseFrequency",
|
|
])
|
|
);
|
|
const options = optionsFromArguments(Phaser.getDefaults(), arguments, [
|
|
"frequency",
|
|
"octaves",
|
|
"baseFrequency",
|
|
]);
|
|
|
|
this._lfoL = new LFO({
|
|
context: this.context,
|
|
frequency: options.frequency,
|
|
min: 0,
|
|
max: 1,
|
|
});
|
|
this._lfoR = new LFO({
|
|
context: this.context,
|
|
frequency: options.frequency,
|
|
min: 0,
|
|
max: 1,
|
|
phase: 180,
|
|
});
|
|
this._baseFrequency = this.toFrequency(options.baseFrequency);
|
|
this._octaves = options.octaves;
|
|
this.Q = new Signal({
|
|
context: this.context,
|
|
value: options.Q,
|
|
units: "positive",
|
|
});
|
|
this._filtersL = this._makeFilters(options.stages, this._lfoL);
|
|
this._filtersR = this._makeFilters(options.stages, this._lfoR);
|
|
|
|
this.frequency = this._lfoL.frequency;
|
|
this.frequency.value = options.frequency;
|
|
|
|
// connect them up
|
|
this.connectEffectLeft(...this._filtersL);
|
|
this.connectEffectRight(...this._filtersR);
|
|
// control the frequency with one LFO
|
|
this._lfoL.frequency.connect(this._lfoR.frequency);
|
|
// set the options
|
|
this.baseFrequency = options.baseFrequency;
|
|
this.octaves = options.octaves;
|
|
// start the lfo
|
|
this._lfoL.start();
|
|
this._lfoR.start();
|
|
readOnly(this, ["frequency", "Q"]);
|
|
}
|
|
|
|
static getDefaults(): PhaserOptions {
|
|
return Object.assign(StereoEffect.getDefaults(), {
|
|
frequency: 0.5,
|
|
octaves: 3,
|
|
stages: 10,
|
|
Q: 10,
|
|
baseFrequency: 350,
|
|
});
|
|
}
|
|
|
|
private _makeFilters(
|
|
stages: number,
|
|
connectToFreq: LFO
|
|
): BiquadFilterNode[] {
|
|
const filters: BiquadFilterNode[] = [];
|
|
// make all the filters
|
|
for (let i = 0; i < stages; i++) {
|
|
const filter = this.context.createBiquadFilter();
|
|
filter.type = "allpass";
|
|
this.Q.connect(filter.Q);
|
|
connectToFreq.connect(filter.frequency);
|
|
filters.push(filter);
|
|
}
|
|
return filters;
|
|
}
|
|
|
|
/**
|
|
* The number of octaves the phase goes above the baseFrequency
|
|
*/
|
|
get octaves() {
|
|
return this._octaves;
|
|
}
|
|
set octaves(octaves) {
|
|
this._octaves = octaves;
|
|
const max = this._baseFrequency * Math.pow(2, octaves);
|
|
this._lfoL.max = max;
|
|
this._lfoR.max = max;
|
|
}
|
|
|
|
/**
|
|
* The the base frequency of the filters.
|
|
*/
|
|
get baseFrequency(): Frequency {
|
|
return this._baseFrequency;
|
|
}
|
|
set baseFrequency(freq) {
|
|
this._baseFrequency = this.toFrequency(freq);
|
|
this._lfoL.min = this._baseFrequency;
|
|
this._lfoR.min = this._baseFrequency;
|
|
this.octaves = this._octaves;
|
|
}
|
|
|
|
dispose(): this {
|
|
super.dispose();
|
|
this.Q.dispose();
|
|
this._lfoL.dispose();
|
|
this._lfoR.dispose();
|
|
this._filtersL.forEach((f) => f.disconnect());
|
|
this._filtersR.forEach((f) => f.disconnect());
|
|
this.frequency.dispose();
|
|
return this;
|
|
}
|
|
}
|