mirror of
https://github.com/Tonejs/Tone.js
synced 2025-01-21 00:03:54 +00:00
322 lines
7.8 KiB
TypeScript
322 lines
7.8 KiB
TypeScript
|
import { Gain } from "../../core/context/Gain";
|
||
|
import { Param } from "../../core/context/Param";
|
||
|
import { InputNode, OutputNode, ToneAudioNode, ToneAudioNodeOptions } from "../../core/context/ToneAudioNode";
|
||
|
import { AudioRange, Cents, Degrees, Frequency, NormalRange, Time, UnitName } from "../../core/type/Units";
|
||
|
import { optionsFromArguments } from "../../core/util/Defaults";
|
||
|
import { readOnly } from "../../core/util/Interface";
|
||
|
import { BasicPlaybackState } from "../../core/util/StateTimeline";
|
||
|
import { AudioToGain } from "../../signal/AudioToGain";
|
||
|
import { Scale } from "../../signal/Scale";
|
||
|
import { connectSignal, Signal } from "../../signal/Signal";
|
||
|
import { Zero } from "../../signal/Zero";
|
||
|
import { Oscillator, ToneOscillatorType } from "./Oscillator";
|
||
|
import { ToneOscillatorInterface } from "./OscillatorInterface";
|
||
|
|
||
|
export interface LFOOptions extends ToneAudioNodeOptions {
|
||
|
type: ToneOscillatorType;
|
||
|
min: number;
|
||
|
max: number;
|
||
|
phase: Degrees;
|
||
|
frequency: Frequency;
|
||
|
amplitude: NormalRange;
|
||
|
units: UnitName;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* LFO stands for low frequency oscillator. LFO produces an output signal
|
||
|
* which can be attached to an AudioParam or Tone.Signal
|
||
|
* in order to modulate that parameter with an oscillator. The LFO can
|
||
|
* also be synced to the transport to start/stop and change when the tempo changes.
|
||
|
*
|
||
|
* @example
|
||
|
* var lfo = new LFO("4n", 400, 4000);
|
||
|
* lfo.connect(filter.frequency);
|
||
|
*/
|
||
|
export class LFO extends ToneAudioNode<LFOOptions> {
|
||
|
|
||
|
readonly name: string = "LFO";
|
||
|
|
||
|
/**
|
||
|
* The oscillator.
|
||
|
*/
|
||
|
private _oscillator: Oscillator;
|
||
|
|
||
|
/**
|
||
|
* The gain of the output
|
||
|
*/
|
||
|
private _amplitudeGain: Gain<NormalRange>;
|
||
|
|
||
|
/**
|
||
|
* The amplitude of the LFO, which controls the output range between
|
||
|
* the min and max output. For example if the min is -10 and the max
|
||
|
* is 10, setting the amplitude to 0.5 would make the LFO modulate
|
||
|
* between -5 and 5.
|
||
|
*/
|
||
|
readonly amplitude: Param<NormalRange>;
|
||
|
|
||
|
/**
|
||
|
* The signal which is output when the LFO is stopped
|
||
|
*/
|
||
|
private _stoppedSignal: Signal<AudioRange>;
|
||
|
|
||
|
/**
|
||
|
* Just outputs zeros. This is used so that scaled signal is not
|
||
|
* optimized to silence.
|
||
|
*/
|
||
|
private _zeros: Zero;
|
||
|
|
||
|
/**
|
||
|
* The value that the LFO outputs when it's stopped
|
||
|
*/
|
||
|
private _stoppedValue: number = 0;
|
||
|
|
||
|
/**
|
||
|
* Convert the oscillators audio range to an output between 0-1 so it can be scaled
|
||
|
*/
|
||
|
private _a2g: AudioToGain;
|
||
|
|
||
|
/**
|
||
|
* Scales the final output to the min and max value
|
||
|
*/
|
||
|
private _scaler: Scale;
|
||
|
|
||
|
/**
|
||
|
* The output of the LFO
|
||
|
*/
|
||
|
readonly output: OutputNode;
|
||
|
|
||
|
/**
|
||
|
* There is no input node
|
||
|
*/
|
||
|
readonly input: undefined;
|
||
|
|
||
|
/**
|
||
|
* A private placeholder for the units
|
||
|
*/
|
||
|
private _units: UnitName = "number";
|
||
|
|
||
|
/**
|
||
|
* If the input value is converted using the [[units]]
|
||
|
*/
|
||
|
convert: boolean = true;
|
||
|
|
||
|
/**
|
||
|
* The frequency value of the LFO
|
||
|
*/
|
||
|
readonly frequency: Signal<Frequency>;
|
||
|
|
||
|
/**
|
||
|
* @param frequency The frequency of the oscillation.
|
||
|
* Typically, LFOs will be in the frequency range of 0.1 to 10 hertz.
|
||
|
* @param min The minimum output value of the LFO.
|
||
|
* @param max The maximum value of the LFO.
|
||
|
*/
|
||
|
constructor(frequency?: Frequency, min?: number, max?: number);
|
||
|
constructor(options?: Partial<LFOOptions>);
|
||
|
constructor() {
|
||
|
|
||
|
super(optionsFromArguments(LFO.getDefaults(), arguments, ["frequency", "min", "max"]));
|
||
|
const options = optionsFromArguments(LFO.getDefaults(), arguments, ["frequency", "min", "max"]);
|
||
|
|
||
|
// @ts-ignore
|
||
|
this._oscillator = new Oscillator({
|
||
|
context: this.context,
|
||
|
frequency : options.frequency,
|
||
|
type : options.type,
|
||
|
});
|
||
|
this.frequency = this._oscillator.frequency;
|
||
|
|
||
|
this._amplitudeGain = new Gain({
|
||
|
context: this.context,
|
||
|
gain: options.amplitude,
|
||
|
units: "normalRange",
|
||
|
});
|
||
|
this.amplitude = this._amplitudeGain.gain;
|
||
|
this._stoppedSignal = new Signal({
|
||
|
context: this.context,
|
||
|
units: "audioRange",
|
||
|
value: 0,
|
||
|
});
|
||
|
this._zeros = new Zero({ context: this.context });
|
||
|
this._a2g = new AudioToGain({ context : this.context });
|
||
|
this._scaler = this.output = new Scale({
|
||
|
context: this.context,
|
||
|
max: options.max,
|
||
|
min: options.min,
|
||
|
});
|
||
|
|
||
|
this.min = options.min;
|
||
|
this.max = options.max;
|
||
|
this.units = options.units;
|
||
|
|
||
|
// connect it up
|
||
|
this._oscillator.chain(this._a2g, this._amplitudeGain, this._scaler);
|
||
|
this._zeros.connect(this._a2g);
|
||
|
this._stoppedSignal.connect(this._a2g);
|
||
|
readOnly(this, ["amplitude", "frequency"]);
|
||
|
this.phase = options.phase;
|
||
|
}
|
||
|
|
||
|
static getDefaults(): LFOOptions {
|
||
|
return Object.assign(ToneAudioNode.getDefaults(), {
|
||
|
amplitude : 1,
|
||
|
frequency : "4n",
|
||
|
max : 1,
|
||
|
min : 0,
|
||
|
phase : 0,
|
||
|
type : "sine" as ToneOscillatorType,
|
||
|
units : "number" as UnitName,
|
||
|
});
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Start the LFO.
|
||
|
* @param time The time the LFO will start
|
||
|
*/
|
||
|
start(time?: Time): this {
|
||
|
time = this.toSeconds(time);
|
||
|
this._stoppedSignal.setValueAtTime(0, time);
|
||
|
this._oscillator.start(time);
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Stop the LFO.
|
||
|
* @param time The time the LFO will stop
|
||
|
*/
|
||
|
stop(time?: Time): this {
|
||
|
time = this.toSeconds(time);
|
||
|
this._stoppedSignal.setValueAtTime(this._stoppedValue, time);
|
||
|
this._oscillator.stop(time);
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Sync the start/stop/pause to the transport
|
||
|
* and the frequency to the bpm of the transport
|
||
|
* @example
|
||
|
* lfo.frequency.value = "8n";
|
||
|
* lfo.sync().start(0)
|
||
|
* //the rate of the LFO will always be an eighth note,
|
||
|
* //even as the tempo changes
|
||
|
*/
|
||
|
sync(): this {
|
||
|
this._oscillator.sync();
|
||
|
this._oscillator.syncFrequency();
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* unsync the LFO from transport control
|
||
|
*/
|
||
|
unsync(): this {
|
||
|
this._oscillator.unsync();
|
||
|
this._oscillator.unsyncFrequency();
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The minimum output of the LFO.
|
||
|
*/
|
||
|
get min(): number {
|
||
|
return this._toType(this._scaler.min);
|
||
|
}
|
||
|
set min(min) {
|
||
|
min = this._fromType(min);
|
||
|
this._scaler.min = min;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The maximum output of the LFO.
|
||
|
*/
|
||
|
get max(): number {
|
||
|
return this._toType(this._scaler.max);
|
||
|
}
|
||
|
set max(max) {
|
||
|
max = this._fromType(max);
|
||
|
this._scaler.max = max;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The type of the oscillator: sine, square, sawtooth, triangle.
|
||
|
*/
|
||
|
get type(): ToneOscillatorType {
|
||
|
return this._oscillator.type;
|
||
|
}
|
||
|
set type(type) {
|
||
|
this._oscillator.type = type;
|
||
|
this._stoppedValue = this._oscillator.getInitialValue();
|
||
|
this._stoppedSignal.value = this._stoppedValue;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The phase of the LFO.
|
||
|
*/
|
||
|
get phase(): Degrees {
|
||
|
return this._oscillator.phase;
|
||
|
}
|
||
|
set phase(phase) {
|
||
|
this._oscillator.phase = phase;
|
||
|
this._stoppedValue = this._oscillator.getInitialValue();
|
||
|
this._stoppedSignal.value = this._stoppedValue;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* The output units of the LFO.
|
||
|
*/
|
||
|
get units(): UnitName {
|
||
|
return this._units;
|
||
|
}
|
||
|
set units(val) {
|
||
|
const currentMin = this.min;
|
||
|
const currentMax = this.max;
|
||
|
// convert the min and the max
|
||
|
this._units = val;
|
||
|
this.min = currentMin;
|
||
|
this.max = currentMax;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Returns the playback state of the source, either "started" or "stopped".
|
||
|
*/
|
||
|
get state(): BasicPlaybackState {
|
||
|
return this._oscillator.state;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* @param destination the destination to connect to
|
||
|
* @param outputNum the optional output number
|
||
|
* @param inputNum the input number
|
||
|
*/
|
||
|
connect(node: InputNode, outputNum?: number, inputNum?: number): this {
|
||
|
if (node instanceof Param || node instanceof Signal) {
|
||
|
this.convert = node.convert;
|
||
|
this.units = node.units;
|
||
|
}
|
||
|
connectSignal(this, node, outputNum, inputNum);
|
||
|
return this;
|
||
|
}
|
||
|
|
||
|
/**
|
||
|
* Private methods borrowed from Param
|
||
|
*/
|
||
|
// @ts-ignore
|
||
|
private _fromType = Param.prototype._fromType;
|
||
|
// @ts-ignore
|
||
|
private _toType = Param.prototype._toType;
|
||
|
// @ts-ignore
|
||
|
private _is = Param.prototype._is;
|
||
|
|
||
|
dispose(): this {
|
||
|
super.dispose();
|
||
|
this._oscillator.dispose();
|
||
|
this._stoppedSignal.dispose();
|
||
|
this._zeros.dispose();
|
||
|
this._scaler.dispose();
|
||
|
this._a2g.dispose();
|
||
|
this._amplitudeGain.dispose();
|
||
|
this.amplitude.dispose();
|
||
|
return this;
|
||
|
}
|
||
|
}
|